This notebook steps you through setting up fluid simulations and using TensorFlow's differentiation to optimize them.
Execute the cell below to install the ΦFlow Python package from GitHub.
# !pip install --quiet phiflow
from phi.flow import *
ΦFlow is vectorized but object-oriented, i.e. data are represented by Python objects that internally use tensors.
First, we create grids for the quantities we want to simulate. For this example, we require a velocity field and a smoke density field. We sample the smoke field at the cell centers and the velocity in staggered form.
smoke = CenteredGrid(0, extrapolation.BOUNDARY, x=32, y=40, bounds=Box(x=32, y=40)) # sampled at cell centers
velocity = StaggeredGrid(0, extrapolation.ZERO, x=32, y=40, bounds=Box(x=32, y=40)) # sampled in staggered form at face centers
Additionally, we want to add more smoke every time step.
We create the INFLOW
field from a circle (2D Sphere
) which defines where hot smoke is emitted.
Furthermore, we are interested in running the simulation for different inflow locations.
ΦFlow supports data-parallell execution via batch dimensions.
When a quantity has a batch dimension with size n, operations involving that quantity will be performed n times simultaneously and the result will also have that batch dimension. Here we add the batch dimension inflow_loc
.
For an overview of the dimension types, see the documentation or watch the introductory tutorial video.
INFLOW_LOCATION = tensor([(4, 5), (8, 5), (12, 5), (16, 5)], batch('inflow_loc'), channel(vector='x,y'))
INFLOW = 0.6 * CenteredGrid(Sphere(center=INFLOW_LOCATION, radius=3), extrapolation.BOUNDARY, x=32, y=40, bounds=Box(x=32, y=40))
The created grids are instances of the class Grid
.
Like tensors, grids also have the shape
attribute which lists all batch, spatial and channel dimensions.
Shapes in ΦFlow store not only the sizes of the dimensions but also their names and types.
print(f"Smoke: {smoke.shape}")
print(f"Velocity: {velocity.shape}")
print(f"Inflow: {INFLOW.shape}")
print(f"Inflow, spatial only: {INFLOW.shape.spatial}")
Smoke: (xˢ=32, yˢ=40) Velocity: (xˢ=32, yˢ=40, vectorᶜ=x,y) Inflow: (inflow_locᵇ=4, xˢ=32, yˢ=40) Inflow, spatial only: (xˢ=32, yˢ=40)
The grid values can be accessed using the values
property.
print(smoke.values)
print(velocity.values)
print(INFLOW.values)
(xˢ=32, yˢ=40) const 0.0 (~vectorᵈ=x,y, xˢ=~(x=31, y=32) int64, yˢ=~(x=40, y=39) int64) const 0.0 (inflow_locᵇ=4, xˢ=32, yˢ=40) 0.015 ± 0.094 (0e+00...6e-01)
Grids have many more properties which are documented here. Also note that the staggered grid has a non-uniform shape because the number of faces is not equal to the number of cells.
Next, let's do some physics!
Since the initial velocity is zero, we just add the inflow and the corresponding buoyancy force.
For the buoyancy force we use the factor (0, 0.5)
to specify strength and direction.
Finally, we project the velocity field to make it incompressible.
Note that the @
operator is a shorthand for resampling a field at different points. Since smoke
is sampled at cell centers and velocity
at face centers, this conversion is necessary.
smoke += INFLOW
buoyancy_force = smoke * (0, 0.5) @ velocity
velocity += buoyancy_force
velocity, _ = fluid.make_incompressible(velocity)
vis.plot(smoke)
/opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phi/vis/_matplotlib/_matplotlib_plots.py:167: UserWarning: This figure includes Axes that are not compatible with tight_layout, so results might be incorrect. plt.tight_layout() # because subplot titles can be added after figure creation
Let's run a longer simulation! Now we add the transport or advection operations to the simulation. ΦFlow provides multiple algorithms for advection. Here we use semi-Lagrangian advection for the velocity and MacCormack advection for the smoke distribution.
trajectory = [smoke]
for i in range(20):
print(i, end=' ')
smoke = advect.mac_cormack(smoke, velocity, dt=1) + INFLOW
buoyancy_force = smoke * (0, 0.5) @ velocity
velocity = advect.semi_lagrangian(velocity, velocity, dt=1) + buoyancy_force
velocity, _ = fluid.make_incompressible(velocity)
trajectory.append(smoke)
trajectory = field.stack(trajectory, batch('time'))
vis.plot(trajectory, animate='time')
0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
<Figure size 640x480 with 0 Axes>
The simulation we just computed was using pure NumPy so all operations were non-differentiable.
To enable differentiability, we need to use either PyTorch, TensorFlow or Jax.
This can be achieved by changing the import statement to phi.tf.flow
, phi.torch.flow
or phi.jax.flow
, respectively.
Tensors created after this import will be allocated using PyTorch / TensorFlow / Jax and operations on these will be executed with the corresponding backend.
These operations can make use of a GPU through CUDA if your configuration supports it.
# from phi.jax.flow import *
from phi.torch.flow import *
# from phi.tf.flow import *
We set up the simulation as before.
INFLOW_LOCATION = tensor([(4, 5), (8, 5), (12, 5), (16, 5)], batch('inflow_loc'), channel(vector='x,y'))
INFLOW = 0.6 * CenteredGrid(Sphere(center=INFLOW_LOCATION, radius=3), extrapolation.BOUNDARY, x=32, y=40, bounds=Box(x=32, y=40))
We can verify that tensors are now backed by TensorFlow / PyTorch / Jax.
type(INFLOW.values.native(INFLOW.shape))
torch.Tensor
Note that tensors created with NumPy will keep using NumPy/SciPy operations unless a TensorFlow tensor is also passed to the same operation.
Let's look at how to get gradients from our simulation.
Say we want to optimize the initial velocities so that all simulations arrive at a final state that is similar to the right simulation where the inflow is located at (16, 5)
.
To achieve this, we define the loss function as $L = | D(s - s_r) |^2$ where $s$ denotes the smoke density and the function $D$ diffuses the difference to smoothen the gradients.
def simulate(smoke: CenteredGrid, velocity: StaggeredGrid):
for _ in range(20):
smoke = advect.mac_cormack(smoke, velocity, dt=1) + INFLOW
buoyancy_force = smoke * (0, 0.5) @ velocity
velocity = advect.semi_lagrangian(velocity, velocity, dt=1) + buoyancy_force
velocity, _ = fluid.make_incompressible(velocity)
loss = math.sum(field.l2_loss(diffuse.explicit(smoke - field.stop_gradient(smoke.inflow_loc[-1]), 1, 1, 10)))
return loss, smoke, velocity
Now it is important that the initial velocity has the inflow_loc
dimension before we record the gradients.
initial_smoke = CenteredGrid(0, extrapolation.BOUNDARY, x=32, y=40, bounds=Box(x=32, y=40))
initial_velocity = StaggeredGrid(math.zeros(batch(inflow_loc=4)), extrapolation.ZERO, x=32, y=40, bounds=Box(x=32, y=40))
Finally, we use gradient_function()
to obtain the gradient with respect to the initial velocity. Since the velocity is the second argument to the simulate()
function, we pass wrt=[1]
.
sim_grad = field.gradient(simulate, wrt='velocity', get_output=False)
The argument get_output=False
specifies that we are not interested in the actual output of the function. By setting it to True
, we would also get the loss value and the final simulation state.
To evaluate the gradient, we simply call the gradient function with the same arguments as we would call the simulation.
velocity_grad = sim_grad(initial_smoke, initial_velocity)
vis.plot(velocity_grad)
/opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/backend/torch/_torch_backend.py:805: UserWarning: Sparse CSR tensor support is in beta state. If you miss a functionality in the sparse tensor support, please submit a feature request to https://github.com/pytorch/pytorch/issues. (Triggered internally at ../aten/src/ATen/SparseCsrTensorImpl.cpp:53.) return torch.sparse_csr_tensor(row_pointers, column_indices, values, shape, device=values.device)
--------------------------------------------------------------------------- AssertionError Traceback (most recent call last) Cell In[14], line 1 ----> 1 velocity_grad = sim_grad(initial_smoke, initial_velocity) 3 vis.plot(velocity_grad) File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:637, in GradientFunction.__call__(self, *args, **kwargs) 635 if key not in self.traces: 636 self.traces[key] = self._trace_grad(key, wrt_natives) --> 637 native_result = self.traces[key](*natives) 638 output_key = match_output_signature(key, self.recorded_mappings, self) 639 jac_shape = output_key.shapes[0].non_batch # ToDo prepend this to all wrt shapes File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/backend/torch/_torch_backend.py:952, in TorchBackend.jacobian.<locals>.eval_grad(*args) 950 if np.prod(self.staticshape(loss)) == 1: 951 assert loss.requires_grad, f"Failed to compute gradient because function output does not depend on any input. Inputs: {args}" --> 952 grads = torch.autograd.grad(loss, wrt_args) # grad() cannot be called during jit trace 953 else: 954 raise NotImplementedError(f"Loss must be reduced to a scalar") File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/torch/autograd/__init__.py:496, in grad(outputs, inputs, grad_outputs, retain_graph, create_graph, only_inputs, allow_unused, is_grads_batched, materialize_grads) 492 result = _vmap_internals._vmap(vjp, 0, 0, allow_none_pass_through=True)( 493 grad_outputs_ 494 ) 495 else: --> 496 result = _engine_run_backward( 497 outputs, 498 grad_outputs_, 499 retain_graph, 500 create_graph, 501 inputs, 502 allow_unused, 503 accumulate_grad=False, 504 ) 505 if materialize_grads: 506 if any( 507 result[i] is None and not is_tensor_like(inputs[i]) 508 for i in range(len(inputs)) 509 ): File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/torch/autograd/graph.py:825, in _engine_run_backward(t_outputs, *args, **kwargs) 823 unregister_hooks = _register_logging_hooks_on_whole_graph(t_outputs) 824 try: --> 825 return Variable._execution_engine.run_backward( # Calls into the C++ engine to run the backward pass 826 t_outputs, *args, **kwargs 827 ) # Calls into the C++ engine to run the backward pass 828 finally: 829 if attach_logging_hooks: File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/torch/autograd/function.py:307, in BackwardCFunction.apply(self, *args) 301 raise RuntimeError( 302 "Implementing both 'backward' and 'vjp' for a custom " 303 "Function is not allowed. You should only implement one " 304 "of them." 305 ) 306 user_fn = vjp_fn if vjp_fn is not Function.vjp else backward_fn --> 307 return user_fn(self, *args) File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/backend/torch/_torch_backend.py:1220, in construct_torch_custom_function.<locals>.TorchCustomFunction.backward(ctx, *grad_args) 1218 output = jit_g[0](x, y, grad_args) 1219 else: -> 1220 output = g(x, y, grad_args) 1221 result = output[0] if len(output) == 1 else (*output,) 1222 return result File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:938, in CustomGradientFunction._trace.<locals>.backward_native(x_natives, y_natives, dy_natives) 936 assert isinstance(result, dict) and all(key in kwargs for key in result.keys()), f"gradient function must return a dict containing only parameter names of the forward function. Forward '{f_name(self.f)}' has arguments {kwargs}." 937 full_result = tuple(result.get(name, None) for name in in_key.tree.keys()) --> 938 result_natives = self.incomplete_tree_to_natives(full_result, tuple(in_key.tree.values()), list(in_key.shapes)) 939 ML_LOGGER.debug(f"Backward pass of custom op {backward_native.__name__} returned gradients for {tuple(result.keys())} out of {tuple(in_key.tree.keys())} containing {len(result_natives)} native tensors") 940 return result_natives File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:988, in CustomGradientFunction.incomplete_tree_to_natives(incomplete, tree, complete_shapes) 986 else: 987 assert type(tree) == type(incomplete) and len(tree) == len(incomplete) --> 988 return sum([CustomGradientFunction.incomplete_tree_to_natives(i_item, c_item, complete_shapes) for i_item, c_item in zip(incomplete, tree)], []) 989 elif isinstance(tree, dict): 990 if incomplete is None: File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:1001, in CustomGradientFunction.incomplete_tree_to_natives(incomplete, tree, complete_shapes) 999 n_val = getattr(tree, attr) 1000 i_val = getattr(incomplete, attr) if incomplete is not None else None -> 1001 natives_item = CustomGradientFunction.incomplete_tree_to_natives(i_val, n_val, complete_shapes) 1002 natives.extend(natives_item) 1003 return natives File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:993, in CustomGradientFunction.incomplete_tree_to_natives(incomplete, tree, complete_shapes) 991 return sum([CustomGradientFunction.incomplete_tree_to_natives(None, item, complete_shapes) for item in tree.values()], []) 992 else: --> 993 assert type(tree) == type(incomplete) and len(tree) == len(incomplete) and set(tree.keys()) == set(incomplete.keys()) 994 return sum([CustomGradientFunction.incomplete_tree_to_natives(incomplete[key], c_item, complete_shapes) for key, c_item in tree.items()], []) 995 elif isinstance(tree, PhiTreeNode): AssertionError:
With the gradient, we can easily perform basic gradient descent optimization. For more advanced optimization techniques and neural network training, see the optimization documentation.
print(f"Initial loss: {simulate(initial_smoke, initial_velocity)[0]}")
initial_velocity -= 0.01 * velocity_grad
print(f"Next loss: {simulate(initial_smoke, initial_velocity)[0]}")
Initial loss: (268.389, 214.372, 136.681, 0.000) along inflow_locᵇ
--------------------------------------------------------------------------- NameError Traceback (most recent call last) Cell In[15], line 2 1 print(f"Initial loss: {simulate(initial_smoke, initial_velocity)[0]}") ----> 2 initial_velocity -= 0.01 * velocity_grad 3 print(f"Next loss: {simulate(initial_smoke, initial_velocity)[0]}") NameError: name 'velocity_grad' is not defined
sim_grad = field.gradient(simulate, wrt='velocity', get_output=True)
for opt_step in range(4):
(loss, final_smoke, _v), velocity_grad = sim_grad(initial_smoke, initial_velocity)
print(f"Step {opt_step}, loss: {loss}")
initial_velocity -= 0.01 * velocity_grad
--------------------------------------------------------------------------- AssertionError Traceback (most recent call last) Cell In[16], line 4 1 sim_grad = field.gradient(simulate, wrt='velocity', get_output=True) 3 for opt_step in range(4): ----> 4 (loss, final_smoke, _v), velocity_grad = sim_grad(initial_smoke, initial_velocity) 5 print(f"Step {opt_step}, loss: {loss}") 6 initial_velocity -= 0.01 * velocity_grad File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:637, in GradientFunction.__call__(self, *args, **kwargs) 635 if key not in self.traces: 636 self.traces[key] = self._trace_grad(key, wrt_natives) --> 637 native_result = self.traces[key](*natives) 638 output_key = match_output_signature(key, self.recorded_mappings, self) 639 jac_shape = output_key.shapes[0].non_batch # ToDo prepend this to all wrt shapes File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/backend/torch/_torch_backend.py:952, in TorchBackend.jacobian.<locals>.eval_grad(*args) 950 if np.prod(self.staticshape(loss)) == 1: 951 assert loss.requires_grad, f"Failed to compute gradient because function output does not depend on any input. Inputs: {args}" --> 952 grads = torch.autograd.grad(loss, wrt_args) # grad() cannot be called during jit trace 953 else: 954 raise NotImplementedError(f"Loss must be reduced to a scalar") File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/torch/autograd/__init__.py:496, in grad(outputs, inputs, grad_outputs, retain_graph, create_graph, only_inputs, allow_unused, is_grads_batched, materialize_grads) 492 result = _vmap_internals._vmap(vjp, 0, 0, allow_none_pass_through=True)( 493 grad_outputs_ 494 ) 495 else: --> 496 result = _engine_run_backward( 497 outputs, 498 grad_outputs_, 499 retain_graph, 500 create_graph, 501 inputs, 502 allow_unused, 503 accumulate_grad=False, 504 ) 505 if materialize_grads: 506 if any( 507 result[i] is None and not is_tensor_like(inputs[i]) 508 for i in range(len(inputs)) 509 ): File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/torch/autograd/graph.py:825, in _engine_run_backward(t_outputs, *args, **kwargs) 823 unregister_hooks = _register_logging_hooks_on_whole_graph(t_outputs) 824 try: --> 825 return Variable._execution_engine.run_backward( # Calls into the C++ engine to run the backward pass 826 t_outputs, *args, **kwargs 827 ) # Calls into the C++ engine to run the backward pass 828 finally: 829 if attach_logging_hooks: File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/torch/autograd/function.py:307, in BackwardCFunction.apply(self, *args) 301 raise RuntimeError( 302 "Implementing both 'backward' and 'vjp' for a custom " 303 "Function is not allowed. You should only implement one " 304 "of them." 305 ) 306 user_fn = vjp_fn if vjp_fn is not Function.vjp else backward_fn --> 307 return user_fn(self, *args) File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/backend/torch/_torch_backend.py:1220, in construct_torch_custom_function.<locals>.TorchCustomFunction.backward(ctx, *grad_args) 1218 output = jit_g[0](x, y, grad_args) 1219 else: -> 1220 output = g(x, y, grad_args) 1221 result = output[0] if len(output) == 1 else (*output,) 1222 return result File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:938, in CustomGradientFunction._trace.<locals>.backward_native(x_natives, y_natives, dy_natives) 936 assert isinstance(result, dict) and all(key in kwargs for key in result.keys()), f"gradient function must return a dict containing only parameter names of the forward function. Forward '{f_name(self.f)}' has arguments {kwargs}." 937 full_result = tuple(result.get(name, None) for name in in_key.tree.keys()) --> 938 result_natives = self.incomplete_tree_to_natives(full_result, tuple(in_key.tree.values()), list(in_key.shapes)) 939 ML_LOGGER.debug(f"Backward pass of custom op {backward_native.__name__} returned gradients for {tuple(result.keys())} out of {tuple(in_key.tree.keys())} containing {len(result_natives)} native tensors") 940 return result_natives File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:988, in CustomGradientFunction.incomplete_tree_to_natives(incomplete, tree, complete_shapes) 986 else: 987 assert type(tree) == type(incomplete) and len(tree) == len(incomplete) --> 988 return sum([CustomGradientFunction.incomplete_tree_to_natives(i_item, c_item, complete_shapes) for i_item, c_item in zip(incomplete, tree)], []) 989 elif isinstance(tree, dict): 990 if incomplete is None: File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:1001, in CustomGradientFunction.incomplete_tree_to_natives(incomplete, tree, complete_shapes) 999 n_val = getattr(tree, attr) 1000 i_val = getattr(incomplete, attr) if incomplete is not None else None -> 1001 natives_item = CustomGradientFunction.incomplete_tree_to_natives(i_val, n_val, complete_shapes) 1002 natives.extend(natives_item) 1003 return natives File /opt/hostedtoolcache/Python/3.12.8/x64/lib/python3.12/site-packages/phiml/math/_functional.py:993, in CustomGradientFunction.incomplete_tree_to_natives(incomplete, tree, complete_shapes) 991 return sum([CustomGradientFunction.incomplete_tree_to_natives(None, item, complete_shapes) for item in tree.values()], []) 992 else: --> 993 assert type(tree) == type(incomplete) and len(tree) == len(incomplete) and set(tree.keys()) == set(incomplete.keys()) 994 return sum([CustomGradientFunction.incomplete_tree_to_natives(incomplete[key], c_item, complete_shapes) for key, c_item in tree.items()], []) 995 elif isinstance(tree, PhiTreeNode): AssertionError:
This notebook provided an introduction to running fluid simulations in NumPy and TensorFlow. It demonstrated how to obtain simulation gradients which can be used to optimize physical variables or train neural networks.
The full ΦFlow documentation is available at https://tum-pbs.github.io/PhiFlow/.
Visit the playground to run ΦFlow code in an empty notebook.