Quickstart: Collisional Ionization Equilibrium

Example of using jaco to solve for collisional ionization equilibrium (CIE) for a hydrogen-helium mixture and plot the ionization states as a function of temperature.

%matplotlib inline
%config InlineBackend.figure_format='retina'
import numpy as np
from matplotlib import pyplot as plt
import sympy as sp

Simple processes

A simple process is defined by a single reaction, with a specified rate.

Let’s inspect the structure of a single process, the gas-phase recombination of H+: H+ + e- -> H +

from jaco.processes import CollisionalIonization, GasPhaseRecombination

process = GasPhaseRecombination("H+")
print(f"Name: {process.name}")
print(f"Heating rate coefficient: {process.heat_rate_coefficient}")
print(f"Heating rate per cm^-3: {process.heat}"),
print(f"Rate coefficient: {process.rate_coefficient}")
print(f"Recombination rate per cm^-3: {process.rate}")
print(f"RHS of e- number density equation: {process.network['e-']}")
Name: Gas-phase recombination of H+
Heating rate coefficient: -1.46719838641439e-26*sqrt(T)/((0.00119216696847702*sqrt(T) + 1.0)**1.748*(0.563615123664978*sqrt(T) + 1.0)**0.252)
Heating rate per cm^-3: -1.46719838641439e-26*C_2*sqrt(T)*n_H+*n_e-/((0.00119216696847702*sqrt(T) + 1.0)**1.748*(0.563615123664978*sqrt(T) + 1.0)**0.252)
Rate coefficient: 1.41621465870114e-10/(sqrt(T)*(0.00119216696847702*sqrt(T) + 1.0)**1.748*(0.563615123664978*sqrt(T) + 1.0)**0.252)
Recombination rate per cm^-3: 1.41621465870114e-10*C_2*n_H+*n_e-/(sqrt(T)*(0.00119216696847702*sqrt(T) + 1.0)**1.748*(0.563615123664978*sqrt(T) + 1.0)**0.252)
RHS of e- number density equation: Eq(Derivative(n_e-(t), t), -1.41621465870114e-10*C_2*n_H+*n_e-/(sqrt(T)*(0.00119216696847702*sqrt(T) + 1.0)**1.748*(0.563615123664978*sqrt(T) + 1.0)**0.252))

Note that all symbolic representations assume CGS units as is standard in ISM physics.

Composing processes

Now let’s define our full network as a sum of simple processes

processes = [CollisionalIonization(s) for s in ("H", "He", "He+")] + [GasPhaseRecombination(i) for i in ("H+", "He+", "He++")]
system = sum(processes)

system.subprocesses
[Collisional Ionization of H,
 Collisional Ionization of He,
 Collisional Ionization of He+,
 Gas-phase recombination of H+,
 Gas-phase recombination of He+,
 Gas-phase recombination of He++]

Summed processes keep track of all subprocesses, e.g. the total net heating rate is:

system.heat
\[\displaystyle - \frac{1.55 \cdot 10^{-26} C_{2} n_{He+} n_{e-}}{T^{0.3647}} - \frac{1.46719838641439 \cdot 10^{-26} C_{2} \sqrt{T} n_{H+} n_{e-}}{\left(0.00119216696847702 \sqrt{T} + 1.0\right)^{1.748} \left(0.563615123664978 \sqrt{T} + 1.0\right)^{0.252}} - \frac{5.86879354565754 \cdot 10^{-26} C_{2} \sqrt{T} n_{He++} n_{e-}}{\left(0.00119216696847702 \sqrt{T} + 1.0\right)^{1.748} \left(0.563615123664978 \sqrt{T} + 1.0\right)^{0.252}} - \frac{1.2746917300104 \cdot 10^{-21} \sqrt{T} n_{H} n_{e-} e^{- \frac{157809.1}{T}}}{\frac{\sqrt{10} \sqrt{T}}{1000} + 1} - \frac{9.37661057635428 \cdot 10^{-22} \sqrt{T} n_{He} n_{e-} e^{- \frac{285335.4}{T}}}{\frac{\sqrt{10} \sqrt{T}}{1000} + 1} - \frac{4.9524176975855 \cdot 10^{-22} \sqrt{T} n_{He+} n_{e-} e^{- \frac{631515}{T}}}{\frac{\sqrt{10} \sqrt{T}}{1000} + 1}\]

Summing processes also sums all chemical and gas/dust cooling/heating rates.

Solving ionization equilibrium

We would like to solve for ionization equilibrium given a temperature \(T\), overall H number density \(n_{\rm H,tot}\). We define a dictionary of those input quantities and also one for the initial guesses of the number densities of the species in the reduced network.

Tgrid = np.logspace(3,6,10**6)
ngrid = np.ones_like(Tgrid) * 100

knowns = {"T": Tgrid, "n_Htot": ngrid}

guesses = {
    "H+": 0.9*np.ones_like(Tgrid),
    "He+": 1e-2*np.ones_like(Tgrid),
    "He++": 1e-2*np.ones_like(Tgrid)
}

Note that by default, the solver only directly solves for \(n_{\rm H+}\), \(n_{\rm He+}\) and \(n_{\rm He++}\) because and \(n_{\rm e-}\) and all atomic species are eliminated by conservation equations. So we only need initial guesses for those 3 quantities. By default the solver takes abundances \(x_i = n_i / n_{\rm H,tot}\) as inputs and outputs.

The solve method calls the JAX solver and computes the solution:

sol = system.solve(knowns, guesses,tol=1e-3, verbose=True,careful_steps=30)
print(sol)
Undetermined symbols: {x_He+, y, n_Htot, x_H+, T, C_2, x_He++}
y not specified; assuming y=0.09254634923706946.
C_2 not specified; assuming C_2=1.0.
Free symbols: {x_He+, y, n_Htot, x_H+, T, C_2, x_He++}
Known values: ['T', 'n_Htot']
Assumed values: ['y', 'C_2']
Equations solved: ['He++', 'H+', 'He+']
It's solvin time. Solving for {'He++', 'He+', 'H+'} based on input {'T', 'n_Htot'} and assumptions about {'y', 'C_2'}
num_iter average=35.86250305175781 min=18 max=68
{'He+': Array([8.259601e-17, 8.259580e-17, 8.259559e-17, ..., 7.692539e-06,
       7.692447e-06, 7.692356e-06], dtype=float32), 'H+': Array([4.1381906e-16, 4.1381869e-16, 4.1381975e-16, ..., 9.9999940e-01,
       9.9999940e-01, 9.9999940e-01], dtype=float32), 'He++': Array([1.4600168e-18, 1.4600135e-18, 1.4600135e-18, ..., 9.2538655e-02,
       9.2538655e-02, 9.2538655e-02], dtype=float32), 'He': Array([9.254635e-02, 9.254635e-02, 9.254635e-02, ..., 7.450581e-09,
       7.450581e-09, 7.450581e-09], dtype=float32), 'H': Array([1.0000000e+00, 1.0000000e+00, 1.0000000e+00, ..., 5.9604645e-07,
       5.9604645e-07, 5.9604645e-07], dtype=float32), 'e-': Array([4.9933508e-16, 4.9933450e-16, 4.9933535e-16, ..., 1.1850843e+00,
       1.1850843e+00, 1.1850843e+00], dtype=float32)}
for i, xi in sorted(sol.items()):
    plt.loglog(Tgrid, xi, label=i)
plt.legend(labelspacing=0)
plt.ylabel("$x_i$")
plt.xlabel("T (K)")
plt.ylim(1e-4,3)
(0.0001, 3)
../_images/CIE_15_1.png

Generating code

Suppose you just want the RHS of the system you’re solving, or its Jacobian, because you have a better solver and/or want to embed these equations in some old C or Fortran code without any dependencies. You can do that too with generate_code.

print(system.generate_code(('H+','He+','He++'),language='c'))
/* Computes the RHS function and Jacobian to solve for [x_He+, x_H+, x_He++]

This code was auto-generated by jaco v0.1.1 and is not intended to be modified or maintained by human beings.

INDEX CONVENTION: (0: x_He+) (1: x_H+) (2: x_He++)
*/

x0 = sqrt(T);
x1 = pow(0.2818075618324889*x0 + 1.0, -0.252);
x2 = pow(0.00059608348423850961*x0 + 1.0, -1.748);
x3 = 5.664858634804579e-10*x1*x2;
x4 = pow(n_Htot, 2);
x5 = x_Heplus + 2*x_Heplusplus + x_Hplus;
x6 = x4*x5;
x7 = 1.0/x0;
x8 = C_2*x7;
x9 = x6*x8;
x10 = x3*x9;
x11 = x10*x_Heplusplus;
x12 = x4*x_Heplus;
x13 = 1.0/T;
x14 = exp(-631515*x13);
x15 = 1.0/((1.0/1000.0)*sqrt(10)*x0 + 1);
x16 = x0*x15;
x17 = 5.68e-12*x14*x16;
x18 = x12*x17;
x19 = x18*x5;
x20 = exp(-285335.40000000002*x13);
x21 = -x_Heplus - x_Heplusplus + y;
x22 = C_2*(0.0019*pow(T, -1.5)*(1 + 0.29999999999999999*exp(-94000.0*x13))*exp(-470000.0*x13) + 1.9324160622805846e-10*x7*pow(0.00016493478118851054*x0 + 1.0, -1.7891999999999999)*pow(4.8416074481177231*x0 + 1.0, -0.21079999999999999));
x23 = x12*x22;
x24 = pow(0.56361512366497779*x0 + 1.0, -0.252);
x25 = pow(0.0011921669684770192*x0 + 1.0, -1.748);
x26 = 1.4162146587011448e-10*x24*x25;
x27 = x26*x9;
x28 = 1 - x_Hplus;
x29 = exp(-157809.10000000001*x13);
x30 = x16*x6;
x31 = 2.3800000000000001e-11*x20*x30;
x32 = x4*x8;
x33 = x32*x_Heplusplus;
x34 = x3*x33;
x35 = x18 - x34;
x36 = x17*x6 + x35;
x37 = -2.3800000000000001e-11*x0*x15*x20*x21*x4 + x23;
x38 = 1.1329717269609158e-9*x1*x2*x33;
x39 = 1.136e-11*x12*x14*x16;
x40 = x32*x_Hplus;
x41 = -5.8500000000000005e-11*x0*x15*x28*x29*x4 + x26*x40;

rhs_result[0] = 2.3800000000000001e-11*x0*x15*x20*x21*x4*x5 + x11 - x19 - x23*x5;
rhs_result[1] = 5.8500000000000005e-11*x0*x15*x28*x29*x4*x5 - x27*x_Hplus;
rhs_result[2] = -x11 + x19;

jac_result[0] = -x22*x6 - x31 - x36 - x37;
jac_result[1] = -x35 - x37;
jac_result[2] = 4.7600000000000002e-11*x0*x15*x20*x21*x4 + x10 - 2*x23 - x31 + x38 - x39;
jac_result[3] = -x41;
jac_result[4] = -x27 - 5.8500000000000005e-11*x29*x30 - x41;
jac_result[5] = 1.1700000000000001e-10*x0*x15*x28*x29*x4 - 2.8324293174022895e-10*x24*x25*x40;
jac_result[6] = x36;
jac_result[7] = x18 - x34;
jac_result[8] = -x10 - x38 + x39;

Let’s break down what happened there. First, jaco is generating the symbolic functions needed to solve the system, as it needs to do before it solves the system with its own solver:

func, jac, _ = system.network.solver_functions(('H','He','He+'),return_jac=True)

Here func represents the set of functions \(f_i\) such that \(f_i = 0\) solves the system. jac encodes the Jacbian of f \(J_{ij} = \frac{\partial f_i}{\partial x_j}\) of derivatives with respect to the solved variables. Note that the two have many common expressions - before being implemented, one should employ common expression elimination to simplify the code and evaluate the functions more efficiently:

cse, (cse_func, cse_jac) = sp.cse((sp.Matrix(func),sp.Matrix(jac)))

cse
[(x0, sqrt(T)),
 (x1, 1/x0),
 (x2, n_Htot**2),
 (x3, x_H+ + x_He+ + 2*x_He++),
 (x4, (0.00059608348423851*x0 + 1.0)**(-1.748)),
 (x5, (0.281807561832489*x0 + 1.0)**(-0.252)),
 (x6, x2*x_He+),
 (x7, 1/T),
 (x8, 1/(sqrt(10)*x0/1000 + 1)),
 (x9, x0*x8),
 (x10, 5.68e-12*x9*exp(-631515*x7)),
 (x11, x10*x6),
 (x12, exp(-285335.4*x7)),
 (x13, -x_He+ - x_He++ + y),
 (x14,
  C_2*(0.0019*(1 + 0.3*exp(-94000.0*x7))*exp(-470000.0*x7)/T**1.5 + 1.93241606228058e-10*x1/((0.000164934781188511*x0 + 1.0)**1.7892*(4.84160744811772*x0 + 1.0)**0.2108))),
 (x15, x14*x6),
 (x16, x2*x3)]
cse_func
\[\displaystyle \left[\begin{matrix}5.66485863480458 \cdot 10^{-10} C_{2} x_{1} x_{2} x_{3} x_{4} x_{5} x_{He++} + 2.38 \cdot 10^{-11} x_{0} x_{12} x_{13} x_{2} x_{3} x_{8} - x_{11} x_{3} - x_{15} x_{3}\end{matrix}\right]\]
cse_jac
\[\displaystyle \left[\begin{matrix}5.66485863480458 \cdot 10^{-10} C_{2} x_{1} x_{2} x_{4} x_{5} x_{He++} + 2.38 \cdot 10^{-11} x_{0} x_{12} x_{13} x_{2} x_{8} - x_{10} x_{16} - x_{11} - 2.38 \cdot 10^{-11} x_{12} x_{16} x_{9} - x_{14} x_{16} - x_{15}\end{matrix}\right]\]

One can then take these expressions and convert them to the syntax of the code you wish to embed them in:

from sympy.codegen.ast import Assignment
for expr in cse:
    print(sp.ccode(Assignment(*expr),standard='c99'))

rhs_result = sp.MatrixSymbol('rhs_result', len(func), 1)
jac_result = sp.MatrixSymbol('jac_result', len(func),len(func))
print()
print(sp.ccode(Assignment(rhs_result, cse_func),standard='c99'))
print()
print(sp.ccode(Assignment(jac_result, cse_jac),standard='c99'))
x0 = sqrt(T);
x1 = 1.0/x0;
x2 = pow(n_Htot, 2);
x3 = x_H+ + x_He+ + 2*x_He++;
x4 = pow(0.00059608348423850961*x0 + 1.0, -1.748);
x5 = pow(0.2818075618324889*x0 + 1.0, -0.252);
x6 = x2*x_He+;
x7 = 1.0/T;
x8 = 1.0/((1.0/1000.0)*sqrt(10)*x0 + 1);
x9 = x0*x8;
x10 = 5.68e-12*x9*exp(-631515*x7);
x11 = x10*x6;
x12 = exp(-285335.40000000002*x7);
x13 = -x_He+ - x_He++ + y;
x14 = C_2*(0.0019*pow(T, -1.5)*(1 + 0.29999999999999999*exp(-94000.0*x7))*exp(-470000.0*x7) + 1.9324160622805846e-10*x1*pow(0.00016493478118851054*x0 + 1.0, -1.7891999999999999)*pow(4.8416074481177231*x0 + 1.0, -0.21079999999999999));
x15 = x14*x6;
x16 = x2*x3;

rhs_result[0] = 5.664858634804579e-10*C_2*x1*x2*x3*x4*x5*x_He++ + 2.3800000000000001e-11*x0*x12*x13*x2*x3*x8 - x11*x3 - x15*x3;

jac_result[0] = 5.664858634804579e-10*C_2*x1*x2*x4*x5*x_He++ + 2.3800000000000001e-11*x0*x12*x13*x2*x8 - x10*x16 - x11 - 2.3800000000000001e-11*x12*x16*x9 - x14*x16 - x15;