Orbital Advection - PrincetonUniversity/athena-public-version GitHub Wiki
Orbital advection is an algorithm to reduce the computational cost of hydrodynamic/MHD simulations in a shearing box or of astronomical disks (Masset 2000). This is also known as the FARGO (Fast Advection in Rotating Gaseous Objects) algorithm. By decomposing the velocity field v into the steady background orbital motion vK and the residual v', the MHD equations have two parts: the typical MHD part and the linear orbital advection part. The orbital advection solves the equations separately using the operator splitting method. Since vK is steady and known, the time step dt
is determined regardless of the background orbital motion. As a result, large dt
reduces the computational cost. Please see e.g., Stone & Gardinar 2010 for more details.
Nothing to do at the configuration for orbital advection.
Parameter OAorder
in the <orbital_advection>
block is a flag for the orbital advection. By default, OAorder=0
is set and the orbital advection is turned off. Giving OAorder=1
activates the orbital advection using the first-order operator splitting method. Likewise, OAorder=2
specifies the second-order operator splitting method in the orbital advection. Using default orbital velocity profiles, there are other necessary parameters. In the Cartesian coordinates, those are qshear
, the shearing rate (1.5 for Keplerian rotation), and local orbital frequency Omega0
as well as the shearing box. When the orbital advection and the shearing box are used together, shboxcoord = 1
is also required. Then, a typical <orbital_advection>
block should look like:
<orbital_advection>
OAorder = 2 # 0: w/o OA, 1: w/ OA(1st), 2: w/ OA(2nd)
Omega0 = 1.0 # orbital frequency
qshear = 1.5 # shearing rate
shboxcoord = 1 # 1: default x-y shear; 2: 2D x-z shear
In the cylindrical and spherical polar coordinates, a positive value of the point source gravity GM
in the <problem>
block is needed. As an optional parameter, Omega0
in the <orbital_advection>
block leads the rotating frame at the angular velocity Omega0
with the orbital advection. Then, the <orbital_advection>
and <problem>
blocks should be as follows:
<orbital_advection>
OAorder = 2 # 0: w/o OA, 1: w/ OA(1st), 2: w/ OA(2nd)
Omega0 = 1.0 # angular velocity of the rotating system (optional)
<problem>
GM = 1.0 # point source gravity
In order to use the orbital advection, the orbital direction has to be periodic with uniform grid spacing. In Cartesian or cylindrical coordinates, the orbital direction is the x2
direction and one has to change the <mesh>
block as follows:
<mesh>
ix2_bc = periodic # inner-X2 boundary flag
ox2_bc = periodic # outer-X2 boundary flag
x2rat = 1.0 # X2 cell aspect ratio
In spherical polar coordinates, the orbital direction is the x3
direction and one has to change the <mesh>
block as follows:
<mesh>
ix3_bc = periodic # inner-X3 boundary flag
ox3_bc = periodic # outer-X3 boundary flag
x3rat = 1.0 # X3 cell aspect ratio
By default, the extra parameter orbital_system
is set to false
in the <output#>
blocks, and the hydrodynamic variables in the outputs are based on v. To obtain the hydrodynamic variables based on v' rather than v, please set orbital_system = true
.
<output1>
file_type = vtk # VTK data dump
orbital_system = false # false: default
Without a user-defined orbital velocity profile, the default orbital velocity profile is automatically set in OrbitalVelocityFunc OrbitalVelocity
in the OrbitalAdvection class. Then, its derivatives in the directions perpendicular to the orbital direction are also set in OrbitalVelocityFunc OrbitalVelocityDerivative[i]
(i = 0
or i = 1
). One can access those using a pointer OrbitalAdvection *porb
in the MeshBlock class. These functions are defined even if the orbital advection is not activated.
In Cartesian coordinates, vK (x, y, z) = -q Ω0 x. The value and the derivatives at x = (x, y, z) are obtained as follows:
void MeshBlock::ProblemGenerator(ParameterInput *pin) {
Real x, y, z;
< set values in x, y, z >
Real vel = porb->OrbitalVelocity(porb, x, y, z); // vK(x, y, z)
Real dv1 = porb->OrbitalVelocityDerivative[0](porb, x, y, z); // dvK/dx(x, y, z)
Real dv2 = porb->OrbitalVelocityDerivative[1](porb, x, y, z); // dvK/dz(x, y, z)
}
In cylindrical coordinates, vK (R, φ, z) = sqrt(GM/R) - Ω0 R for 2D and vK (R, φ, z) = sqrt(GM/d) (R/d) - Ω0 R for 3D, where d = sqrt(R2 + z2). The value and the derivatives at x = (R, φ, z) are obtained as follows:
void MeshBlock::ProblemGenerator(ParameterInput *pin) {
Real R, phi, z;
< set values in R, phi, z >
Real vel = porb->OrbitalVelocity(porb, R, phi, z); // vK(R, phi, z)
Real dv1 = porb->OrbitalVelocityDerivative[0](porb, R, phi, z); // dvK/dR(R, phi, z)
Real dv2 = porb->OrbitalVelocityDerivative[1](porb, R, phi, z); // dvK/dz(R, phi, z)
}
In spherical polar coordinates, vK (r, θ, φ) = sqrt(GM/(r sin(θ))) - Ω0 r sin(θ). The value and the derivatives at x = (r, θ, φ) are obtained as follows:
void MeshBlock::ProblemGenerator(ParameterInput *pin) {
Real r, theta, phi;
< set values in r, theta, phi >
Real vel = porb->OrbitalVelocity(porb, r, theta, phi); // vK(R, r, theta, phi)
Real dv1 = porb->OrbitalVelocityDerivative[0](porb, r, theta, phi); // dvK/d(theta)(R, r, theta, phi)
Real dv2 = porb->OrbitalVelocityDerivative[1](porb, r, theta, phi); // dvK/d(phi)(R, r, theta, phi)
}
In order to use a user-defined orbital velocity profile, they must be defined and then enrolled in Mesh::InitUserMeshData()
in the problem generator.
First, the orbital velocity function must be defined and match the following function signature:
Real UserOrbitalVelocity(OrbitalAdvection * porb, Real x1, Real x2, Real x3);
The custom function can then be enrolled by calling:
EnrollOrbitalVelocity(UserOrbitalVelocity);
If using the user-defined functions for the derivatives of the orbital velocity, the functions must be defined an match the following function signature:
// x1 direction
Real UserOrbitalVelocityDerivative1(OrbitalAdvection * porb, Real x1, Real x2, Real x3);
// x3 direction in Cartesian and cylindrical, x2 direction in spherical polar
Real UserOrbitalVelocityDerivative2(OrbitalAdvection * porb, Real x1, Real x2, Real x3);
The custom functions can then be enrolled by calling:
// x1 direction
EnrollOrbitalVelocityDerivative(0, UserOrbitalVelocityDerivative1);
// x3 direction in Cartesian and cylindrical, x2 direction in spherical polar
EnrollOrbitalVelocityDerivative(1, UserOrbitalVelocityDerivative2);
Otherwise, the derivatives of the orbital velocity are calculated from the user-defined orbital velocity profile numerically.
The 2D hydro shearing sheet test:
> python configure.py --prob=ssheet --flux=hlle --eos=isothermal
and use input/hydro/athinput.ssheet
.
The MHD shwave test:
> python configure.py -b --prob=jgg --flux=hlld --eos=isothermal
and use input/mhd/athinput.jgg
The 3D MRI test:
> python configure.py -b --prob=hgb --flux=hlld --eos=isothermal
and use input/mhd/athinput.hgb
The disk test in the cylindrical coordinates:
> python configure.py --prob=disk --coord=cylindrical --flux=hlld
and use input/hydro/athinput.disk_cyl
.
The disk test in the spherical polar coordinates:
> python configure.py --prob=disk --coord=spherical_polar --flux=hlld
and use input/hydro/athinput.disk_sph
.
Runs with the orbital advection require additional computational cost for each cycle in comparison with runs without the orbital advection. The cost comes from extra MPI communications and solving the linear advection. We find that the increase of the computational cost for one cycle is from 10% to 20% with the first-order splitting method and from 20% to 40% with the second-order splitting method. Therefore, it is expected that the orbital advection reduces the computational cost when the orbital speed is faster by 20% for OAorder=1
and by 40% OAorder=2
than the sound speed.
- There are two restrictions on the time step
dt
from the orbital advection. One requires that the shear distance between cells used in the reconstruction at each cell is smaller than the half cell size in the orbital direction for the first-order scheme and than the cell size in the orbital direction for the second-order scheme. The other prohibits the orbital motion for one cycle from exceeding the meshblock size in the orbital direction. - The orbital velocity profile must not change in time.
- Orbital advection works with mesh refinement (AMR and SMR), passive scalars, shearing box, diffusion processes (hydro and field), general EOS, self-gravity (FFT and multigrid), and super time stepping.
- The orbital advection works with neither special relativity nor general relativity.
- Both the PLM reconstructions (
time/xorder=2
ortime/xorder=2c
) and the PPM reconstructions (time/xorder=3
ortime/xorder=3c
) are available with the orbital advection. When givingtime/xorder=4
, however, the orbital remapping will be performed with only the 3rd-order spatial accuracy like the shear-periodic boundaries. - Neither
time/integrator=rk4
nortime/integrator=ssprk5_4
are allowed with both the orbital advection and the mesh refinement. - When using orbital advection with STS,
orbital_advection/OAorder=2
andtime/sts_integrator=rkl2
are recommended.