10. Calling external functions

It is sometimes desirable to call an external handwritten function from a code-generated one, and use the result as part of further symbolic expressions. This can enable a few useful behaviors:

  • Evaluate complex logic that cannot easily be expressed in a functional expression tree. For example, solving a small numerical optimization iteratively within a larger expression.

  • Insert custom error-checking or logging logic into generated functions.

  • Interface with user-provided types that are not adequately expressed as a dataclass. For example, you might wish to pass a dynamically sized buffer of values, and perform bilinear interpolation from within the generated code.

These use cases can be achieved by declaring an external function. There are a few caveats to keep in mind:

  • By necessity, wrenfold must assume all calls to external functions are pure (without any side effects). Any two identical calls are assumed to be interchangeable and will be de-duplicated during code-generation.

  • Because external functions are effectively a black box, we cannot propagate the derivative through them.

10.1. Declaring an external function

In this example, we will pass a lookup table to our generated function. To begin with, we need a type to represent the table. We do this by inheriting from wrenfold.type_annotations.Opaque:

class LookupTable(type_annotations.Opaque):
    """
    A placeholder we will map to our actual type during the code-generation step.
    """

The type itself requires no further additions, it is merely a placeholder that we will later map to a real type in the target language.

Next, we declare a function to represent our lookup operation:

interpolate_table = external_functions.declare_external_function(
    name="interpolate_table",
    arguments=[("table", LookupTable), ("arg", type_annotations.FloatScalar)],
    return_type=type_annotations.FloatScalar)  # [interpolate_table_end]

interpolate_table is an instance of wrenfold.external_functions.ExternalFunction. We can call it with symbolic expressions, provided they match the expected types we specified in the arguments list.

Now we can define a symbolic function that uses interpolate_table. We will write a function that computes the bearing vector between two points \(\mathbf{v} = \mathbf{p}_1 - \mathbf{p}_0\), and uses the direction angle of vector \(\theta = \text{atan2}\left(\mathbf{v}_y, \mathbf{v}_x\right)\) as an argument to the lookup table:

def lookup_angle(table: LookupTable, p_0: type_annotations.Vector2, p_1: type_annotations.Vector2):
    """
    Compute bearing angle between two points, and use it as an argument to our lookup table.
    """
    v = p_1 - p_0
    angle = sym.atan2(v[1], v[0])

    # Normalize between [0, 1] (where 0 corresponds to -pi, and 1 corresponds to pi).
    angle_normalized = (angle + sym.pi) / (2 * sym.pi)

    # Perform the lookup.
    table_value = interpolate_table(table=table, arg=angle_normalized)

    # Do some more symbolic operations with the result:
    result = table_value * (p_1 - p_0).squared_norm()
    return [
        code_generation.ReturnValue(result),
    ]

To emit actual code for our LookupTable type and interpolate_table function, we customize the code generator:

class CustomCppGenerator(code_generation.CppGenerator):

    def format_call_external_function(self, element: ast.CallExternalFunction) -> str:
        """
        Place our external function in the ``utilities`` namespace.
        """
        if element.function == interpolate_table:
            args = ', '.join(self.format(x) for x in element.args)
            return f'utilities::{element.function.name}({args})'
        return self.super_format(element)

    def format_custom_type(self, element: type_info.CustomType) -> str:
        """
        Assume the lookup table is implemented as a std::vector<double>.
        """
        if element.python_type == LookupTable:
            return 'std::vector<double>'
        return self.super_format(element)


code = code_generation.generate_function(func=lookup_angle, generator=CustomCppGenerator())
print(code)

Which produces the following C++:

// Our generated method correctly accepts a `std::vector<double>`, and invokes
// the appropriately-namespaced `interpolate_table`.
template <typename Scalar, typename T1, typename T2>
Scalar lookup_angle(const std::vector<double> &table, const T1 &p_0, const T2 &p_1)
{
    auto _p_0 = wf::make_input_span<2, 1>(p_0);
    auto _p_1 = wf::make_input_span<2, 1>(p_1);

    // ...

    const Scalar v002 = _p_0(0, 0);
    const Scalar v008 = _p_0(1, 0);
    const Scalar v000 = _p_1(0, 0);
    const Scalar v007 = _p_1(1, 0);
    const Scalar v005 = v000 + -v002;
    const Scalar v010 = v007 + -v008;
    return (v005 * v005 + v010 * v010) *
            utilities::interpolate_table(
                table, static_cast<Scalar>(0.5) *
                            (static_cast<Scalar>(M_PI) + std::atan2(v010, v005)) *
                            (static_cast<Scalar>(1) / static_cast<Scalar>(M_PI)));
}

Note that nowhere did we explicitly tell wrenfold anything about the nature of std::vector<double>. For the most part we would prefer not to, since it would induce a great deal more complexity in the code-generator. Instead we treat it as an opaque type that can be passed through the generated function to a handwritten one.