opendp 0.14.2-dev.20260401.2

A library of differential privacy algorithms for the statistical analysis of sensitive private data.
use std::ffi::c_void;

use dashu::integer::IBig;

use crate::{
    core::{FfiResult, IntoAnyTransformationFfiResultExt},
    domains::AtomDomain,
    error::Fallible,
    ffi::any::{AnyDomain, AnyMetric, AnyObject, AnyTransformation, Downcast},
    metrics::AbsoluteDistance,
    traits::{Float, SaturatingMul},
    transformations::make_lipschitz_float_mul,
};

#[unsafe(no_mangle)]
pub extern "C" fn opendp_transformations__make_lipschitz_float_mul(
    input_domain: *const AnyDomain,
    input_metric: *const AnyMetric,
    constant: *const c_void,
    bounds: *const AnyObject,
) -> FfiResult<*mut AnyTransformation> {
    fn monomorphize<T>(
        input_domain: &AnyDomain,
        input_metric: &AnyMetric,
        constant: *const c_void,
        bounds: *const AnyObject,
    ) -> Fallible<AnyTransformation>
    where
        T: 'static + Float + SaturatingMul,
        IBig: From<T::Bits>,
    {
        let input_domain = input_domain.downcast_ref::<AtomDomain<T>>()?.clone();
        let input_metric = input_metric.downcast_ref::<AbsoluteDistance<T>>()?.clone();
        let constant = *try_as_ref!(constant as *const T);
        let bounds = try_as_ref!(bounds).downcast_ref::<(T, T)>()?.clone();
        make_lipschitz_float_mul::<T>(input_domain, input_metric, constant, bounds).into_any()
    }
    let input_domain = try_as_ref!(input_domain);
    let input_metric = try_as_ref!(input_metric);
    let T = try_!(input_domain.type_.get_atom());
    dispatch!(monomorphize, [
        (T, @floats)
    ], (input_domain, input_metric, constant, bounds))
    .into()
}