The introduction of temperature control in the previous lecture allows us to simulate the system to maintain a constant temperature through heat exchange with the reservoir. While this canonical ensemble enables more realistic modeling of the real world, it still has some limitations. Just imagine that you want to model a periodic system under different temperatures; it is natural to think about the volume of the system being subject to change due to thermal expansion effects.
Within the NVT ensemble, you need to manually adjust the volume of the system when initializing the positions for each different temperature. This can be very tedious and requires trial and error iterations. Thus, we would like to seek a better solution to let the system adjust the volume by itself during the MD simulation.
Similar to how a thermostat maintains constant temperature, a barostat adjusts the simulation box dimensions and/or particle positions to ensure that the system stays at the desired pressure. This is crucial for simulating systems in ensembles like NPT, where pressure fluctuations need to be controlled. Below, I will introduce two barostat techniques.
This is a simple barostat that rescales the simulation box gradually toward the target pressure. To implement a barostat, the key idea is to adjust the simulation box size in response to the difference between the current pressure and the target pressure. This is done by scaling the box dimensions and particle positions, and updating the system’s volume accordingly.
-
Compute the instantaneous pressure. The system pressure
$P$ in an MD simulation can be calculated using the virial equation. It includes contributions from the kinetic energy (related to ideal gas) and the virial of the system (related to particle interactions):
Where:
-
$N$ is the number of particles. -
$k_B$ is the Boltzmann constant. -
$T$ is the temperature. -
$V$ is the volume of the simulation box. -
$\mathbf{r}_{ij}$ is the position vector between particles$i$ and$j$ . -
$\mathbf{F}_{ij}$ is the force acting on particle$i$ due to particle$j$ .
-
Compute pressure difference. At each time step, calculate the difference between the current and target pressures.
-
Adjust the simulation box volume. For isotropic pressure control (same scaling in all directions), the new volume is updated by:
Where:
-
$\tau_P$ is a time constant controlling the pressure coupling strength. -
$dt$ is the time step. -
$V_{old}$ is the current volume.
-
Rescale the positions and velocities. The positions of all particles are scaled accordingly to maintain their relative distances within the simulation box. For isotropic scaling, each position
$\mathbf{r}$ is rescaled:
The pseudo code should look like the following
import numpy as np
P_target = 1.0 # Target pressure (arbitrary units)
tau_P = 0.5 # Pressure coupling constant (barostat relaxation time)
kb = 1.38e-23 # Boltzmann constant in J/K
def compute_temperature(V):
"""
Compute the temperature of the system from particle velocities.
Parameters:
V (np.array): N x 3 array of particle velocities.
Returns:
float: Temperature of the system.
"""
KE = 0.5 * np.sum(MASS * velocities**2) # Sum of kinetic energies
T = (2 * KE) / (3 * len(V)) # Ideal gas temperature relation
return T
def compute_virial_pressure(R, V, volume):
"""
Compute the pressure using the virial equation in the Berendsen barostat.
Parameters:
R (np.array): N x 3 array of particle positions.
V (np.array): N x 3 array of particle velocities.
volume (float): Volume of the simulation box.
Returns:
float: Computed pressure of the system.
"""
N = len(R) # Number of particles
# Step 1: Compute the temperature from velocities
T = compute_temperature(V)
# Step 2: Compute the kinetic contribution to the pressure
P_KE = (N * kB * T) / V
# Step 3: Compute the virial contribution to the pressure
P_virial = 0.0
for i in range(N):
for j in range(i + 1, N):
r_ij = R[i] - R[j] # Displacement vector between particles i and j
F_ij = forces[i] # Force on particle i due to particle j (be careful)
P_virial += np.dot(r_ij, F_ij) # Dot product r_ij · F_ij
P_virial /= (3 * volume) # Virial term divided by 3V
# Total pressure is the sum of kinetic and virial contributions
P = P_kinetic + P_virial
return P
# Barostat function (Berendsen type)
def Berendsen_barostat(R, V, F, volume, P_target, tau_P):
"""Adjust volume and rescale positions to maintain constant pressure."""
# Calculate current pressure
P = compute_virial_pressure(R, V, volume)
# Calculate the scaling factor
dP = P - P_target
scale_factor = 1.0 + (dP / tau_P) * TIMESTEP
volume *= scale_factor
rescale_factor = scale_factor ** (1.0 / 3.0)
# rescale positions and velocities
R *= rescale_factor
V *= rescale_factor
return R, V, volume
This is a relatively simple method, where the system’s volume is gradually rescaled to match the target pressure. It does not rigorously conserve the ensemble, but it is computationally efficient and often used for equilibration runs for the simulation of isotropic systems like liquid.
The Parrinello-Rahman barostat is a more advanced method for controlling pressure in MD simulations, particularly useful when the system undergoes anisotropic volume changes. Unlike Berendsen barostat that scales the simulation box isotropically, the Parrinello-Rahman barostat allows both the shape and size of the simulation box to change. This is especially important in simulations of materials under strain, phase transitions, or when dealing with anisotropic systems like crystals.
-
Matrix Representation of box. To enable this barostat, we first represent simulation box by a matrix
$\mathbf{h}$ that defines the three lattice vectors of the simulation box. This matrix allows for changes in both the box size and shape.
Here,
-
Pressure Tensor: The barostat works with the full pressure tensor
$\mathbf{P}$ , which describes how pressure acts differently along different directions. The pressure tensor$\mathbf{P}$ can be computed from the system’s kinetic energy and virial:
Where:
-
$V$ is the volume of the simulation box. -
$\mathbf{v}_i$ and$\mathbf{F}_i$ are the velocity and force on particle$i$ , respectively. -
$m_i$ is the mass of particle$i$ .
-
Strain Rate Tensor \mathbf{W}. The time evolution of the box matrix
$\mathbf{h}$ is governed by the equation:
Where
Here,
-
Update Particle Positions and Box. Once
$\mathbf{h}$ is updated, the particle positions need to be rescaled by the new box matrix. The rescaled positions are calculated as:
In short, this approach introduces a few additional variables:
-
$\mathbf{h}$ : The simulation box matrix, which evolves over time and controls both the size and shape of the box. -
$Q$ : The fictitious mass associated with the barostat, controlling the rate of volume and shape changes. -
$\mathbf{W}$ : The strain rate tensor, which governs how the box matrix changes over time.
These variables allow the Parrinello-Rahman barostat to apply pressure anisotropically, enabling the box to deform naturally while maintaining the target pressure in the system.
Below is a pseudocode to achieve the Parrinello-Rahman barostat.
import numpy as np
# Constants
dt = 0.001 # Time step
P_target = np.eye(3) * 1.0 # Target pressure tensor (isotropic)
Q = 10.0 # Fictitious mass for the barostat, analogical to the Q in Nose-Hoover thermostat
N = 100 # Number of particles
kb = 1.38e-23 # Boltzmann constant
T = 300 # Temperature
V = 1.0 # Initial volume
positions = np.random.randn(N, 3)
velocities = np.random.randn(N, 3)
forces = np.zeros_like(positions) # Placeholder for forces
h = np.eye(3) * V ** (1/3) # Initial box matrix (cubic box)
def compute_pressure_tensor(R, V, volume):
"""
Compute the internal pressure tensor using the virial equation.
Parameters:
R (np.array): N x 3 array of particle positions.
V (np.array): N x 3 array of particle velocities.
volume (float): Volume of the simulation box.
Returns:
np.array: 3 x 3 pressure tensor.
"""
# Number of particles
N = len(positions)
# Compute the kinetic energy contribution to the pressure
KE = np.sum(0.5 * MASS * V**2)
T = (2 * KE) / (3 * N * kB)
P_kE = N * kB * T / volume
# Compute the virial contribution to the pressure tensor
P_virial = np.zeros((3, 3))
for i in range(N):
for j in range(i + 1, N):
r_ij = R[i] - R[j] # Displacement vector
F_ij = forces[i] # Force on particle i (be careful)
P_virial += np.outer(r_ij, F_ij)
# Average pressure tensor by dividing by the volume
P_virial /= volume
# Total pressure tensor is the sum of kinetic and virial contributions
P_total = P_KE* np.eye(3) + P_virial
return P_total
# Parrinello-Rahman barostat step
def parrinello_rahman_barostat(H, R, V, P_target, Q):
"""Update the box and rescale positions using Parrinello-Rahman barostat."""
# Compute current pressure tensor
volume = np.linalg.det(H) # Current volume
P = calculate_pressure_tensor(R, V, volume)
# Compute strain rate tensor (dW/dt)
W_dot = (P - P_target) / Q
# Update the box matrix h
h_new = h + h @ W_dot * dt
# Rescale positions
R = R @ np.linalg.inv(h) @ h_new # Transform positions to fractional coordinates
return R, h_new
A more complete discussion can be found here.
- Run the lec_03_npt_liquid.py to understand the brute-force implementation of Berendsen Barostat for liquid simulation.
- Run the lec_03_npt_liquid_fast.py.
- Understand the differences of NVT and NPT simulations as shown from the following figure (obtained from running
lec_03_npt_liquid_fast.py
).