User functions and derivatives
Cubie lets you call your own functions inside dx/dt equations. There are two main cases:
Pure Python callables: These are treated like ordinary functions. When possible, Cubie inlines them symbolically.
CUDA device functions: These are detected and treated as opaque calls in generated code. For differentiation (Jacobian/JVP/VJP), provide a corresponding derivative function.
Providing functions
Pass a mapping of names to callables via user_functions to parse_input. If your function name collides with a SymPy built-in (e.g., exp), the user function takes precedence.
Example (Python function):
Define a simple function:
def ex_squared(x): return x**2
Use it in equations:
index_map, symbols, funcs, eqs, fn_hash = parse_input( dxdt=["dx = ex_squared(a)", "y = x"], user_functions={"ex_squared": ex_squared} )
print_cuda_multiple(eqs, symbols) will emit a*a in code if it can inline the function (i.e. it’s simple and uses Sympy-compatible logic, otherwise it will call ex_squared(a) in code. If it calls ex_squared(a), it will also call ex_squared_grad(a, 0) in Jacobian terms, so you need to provide that function in user_function_derivatives.
Device functions and derivatives
CUDA device functions are detected automatically if they are created with numba.cuda.jit(…, device=True). For differentiation, also provide a derivative function in user_function_derivatives with the same key as the original function name.
The derivative callable signature must be: d_userfunc(funcargs…, argindex) where argindex is 0-based index of the argument with respect to which the derivative is taken.
The derivative callable’s __name__ is used in generated code, so choose a descriptive name (e.g., myfunc_grad).
Example:
Define device function and its derivative name:
from numba import cuda @cuda.jit(device=True) def myfunc(a, b): return a * b # This can be device or pure Python; codegen only needs the name def myfunc_grad(a, b, index): if index == 0: return b elif index == 1: return a return 0
Parse equations with both maps:
index_map, symbols, funcs, eqs, fn_hash = parse_input( dxdt=["dx = myfunc(x, y)", "dy = x"], states=["x", "y"], parameters=[], constants=[], observables=[], user_functions={"myfunc": myfunc}, user_function_derivatives={"myfunc": myfunc_grad} )
Generate JVP code:
code = generate_jvp_code(eqs, index_map) # The code will contain calls to myfunc_grad(..., argindex) in the Jacobian # terms.
Name collisions with SymPy
If your user function has the same name as a SymPy function, Cubie ensures your function wins. Internally it renames your function to a safe symbolic token during parsing and maps it back to your original name when printing code.
Tips
If you don’t provide a derivative for a device function, auto-generated jacobians will not work.
Pure Python user functions that can be evaluated on SymPy symbols may be inlined symbolically; otherwise they are called by name in code, and you’ll need to provide a derivative function for differentiation.