AutoPDEx is a free open source partial differential equation (PDE) solver based on the automatic code transformation capabilities of JAX.
The idea of the project is to develop a modular and easily extendable environment for the solution of boundary and initial boundary value problems, which provides automatic sensitivity analysis, allows for good integration with machine learning algorithms and can be executed on accelerators such as GPUs.
The documentation with more examples is available here.
To install AutoPDEx, you can use the following command. Note, that it requires python>=3.10.
pip install --upgrade pip
pip install autopdex
This is a short example for solving Poisson's problem with homogeneous Dirichlet conditions on the domain
where
First, we import the necessary packages and enable double precision.
import jax
import jax.numpy as jnp
import flax
import meshio
from autopdex import seeder, geometry, solver, utility, models, spaces, mesher
jax.config.update("jax_enable_x64", True)
Generate the mesh (or import the node coordinates and connectivity)
pts = [[0., 0.], [1., 0.], [1., 1.], [0., 1.]]
coords, elems = mesher.structured_mesh((200, 200), pts, 'quad')
node_coordinates = {'phi': coords,}
connectivity = {'phi': elems,}
Selection of nodes and degrees of freedom for Dirichlet conditions
sdf = lambda x: geometry.psdf_polygon(x, pts)
dirichlet_nodes = geometry.in_sdfs(node_coordinates['phi'], sdf)
dirichlet_dofs = {'phi': dirichlet_nodes,}
dirichlet_conditions = utility.dict_zeros_like(dirichlet_dofs, dtype=jnp.float64)
The variational problem is defined in terms of the potential
This leads to the following Euler-Lagrange equation (Poisson's equation):
def integrand_fun(x_int, ansatz_fun, settings, static_settings, elem_number, set):
# Definition of custom functional
x = ansatz_fun['physical coor'](x_int)
phi_fun = ansatz_fun['phi']
phi = phi_fun(x_int)
dphi_dx = jax.jacrev(phi_fun)(x_int)
x_2 = x - jnp.array([1., 0.5])
b = 20 * (jnp.sin(10 * x @ x) - jnp.cos(10 * x_2 @ x_2))
return (1/2) * dphi_dx @ dphi_dx - b * phi
Set up the finite element, here Q1 elements for the field 'phi'
user_potential = models.mixed_reference_domain_potential(
integrand_fun,
{'phi': spaces.fem_iso_line_quad_brick,},
*seeder.gauss_legendre_nd(dimension = 2, order = 2),
'phi')
Prepare the settings for autopdex
static_settings = flax.core.FrozenDict({
'assembling mode': ('user potential',),
'solution structure': ('nodal imposition',),
'model': (user_potential, ),
'solver type': 'newton',
'solver backend': 'scipy',
'solver': 'lapack',
'verbose': 1,
})
settings = {
'connectivity': (connectivity,),
'dirichlet dofs': dirichlet_dofs,
'node coordinates': node_coordinates,
'dirichlet conditions': dirichlet_conditions,
}
Compile, assemble and solve linear system
initial_guess = utility.dict_zeros_like(dirichlet_dofs, dtype=jnp.float64)
dofs = solver.solver(initial_guess, settings, static_settings)[0]
Write vtk file for visualization with Paraview
meshio.Mesh(
coords,
{'quad': elems},
point_data={
"phi": dofs['phi'],
},
).write("./short_example.vtk")
If you found this library useful in academic research, please cite the JOSS paper:
@article{Bode_AutoPDEx_An_Automized_2025,
author = {Bode, Tobias},
doi = {10.21105/joss.07300},
journal = {Journal of Open Source Software},
month = apr,
number = {108},
pages = {7300},
title = {{AutoPDEx: An Automized Partial Differential Equation solver based on JAX}},
url = {https://joss.theoj.org/papers/10.21105/joss.07300},
volume = {10},
year = {2025}
}
You are warmly invited to contribute to the project. For larger developments, please get in touch beforehand in order to circumvent double work.
For detailed information on how to contribute, please see our Contribution Guidelines
AutoPDEx is licensed under the GNU Affero General Public License, Version 3.