diffsol-c 0.4.1

A diffsol wrapper featuring runtime scalar/matrix/solver types and a C API
Documentation
use std::cell::RefCell;
use std::ffi::CString;
use std::os::raw::c_char;
use std::ptr;

struct LastError {
    message: CString,
    file: CString,
    line: u32,
}

thread_local! {
    static LAST_ERROR: RefCell<Option<LastError>> = const { RefCell::new(None) };
}

fn cstring_from_str(value: &str) -> CString {
    if value.as_bytes().contains(&0) {
        let mut bytes = value.as_bytes().to_vec();
        for byte in &mut bytes {
            if *byte == 0 {
                *byte = b'?';
            }
        }
        CString::new(bytes).unwrap_or_else(|_| CString::new("error").unwrap())
    } else {
        CString::new(value).unwrap_or_else(|_| CString::new("error").unwrap())
    }
}

pub(crate) fn set_last_error(message: &str, file: &'static str, line: u32) {
    let message = cstring_from_str(message);
    let file = cstring_from_str(file);
    LAST_ERROR.with(|slot| {
        *slot.borrow_mut() = Some(LastError {
            message,
            file,
            line,
        });
    });
}

pub(crate) fn clear_last_error() {
    LAST_ERROR.with(|slot| {
        *slot.borrow_mut() = None;
    });
}

/// Return whether thread-local error state is currently set.
///
/// # Safety
/// This function is safe to call from C. It relies on thread-local state managed
/// by this library and does not dereference any caller-provided pointers.
#[unsafe(no_mangle)]
pub unsafe extern "C" fn diffsol_error_code() -> i32 {
    LAST_ERROR.with(|slot| if slot.borrow().is_some() { 1 } else { 0 })
}

/// Return the last error message for the current thread, if any.
///
/// # Safety
/// The returned pointer is borrowed from thread-local storage owned by this
/// library and must not be freed or mutated by the caller.
#[unsafe(no_mangle)]
pub unsafe extern "C" fn diffsol_error() -> *const c_char {
    LAST_ERROR.with(|slot| {
        slot.borrow()
            .as_ref()
            .map(|err| err.message.as_ptr())
            .unwrap_or(ptr::null())
    })
}

/// Return the last error message for the current thread, if any.
///
/// # Safety
/// The returned pointer is borrowed from thread-local storage owned by this
/// library and must not be freed or mutated by the caller.
#[unsafe(no_mangle)]
pub unsafe extern "C" fn diffsol_last_error_message() -> *const c_char {
    LAST_ERROR.with(|slot| {
        slot.borrow()
            .as_ref()
            .map(|err| err.message.as_ptr())
            .unwrap_or(ptr::null())
    })
}

/// Return the source file associated with the last error for the current thread.
///
/// # Safety
/// The returned pointer is borrowed from thread-local storage owned by this
/// library and must not be freed or mutated by the caller.
#[unsafe(no_mangle)]
pub unsafe extern "C" fn diffsol_last_error_file() -> *const c_char {
    LAST_ERROR.with(|slot| {
        slot.borrow()
            .as_ref()
            .map(|err| err.file.as_ptr())
            .unwrap_or(ptr::null())
    })
}

/// Return the source line associated with the last error for the current thread.
///
/// # Safety
/// This function is safe to call from C. It does not dereference any
/// caller-provided pointers.
#[unsafe(no_mangle)]
pub unsafe extern "C" fn diffsol_last_error_line() -> u32 {
    LAST_ERROR.with(|slot| slot.borrow().as_ref().map(|err| err.line).unwrap_or(0))
}

/// Clear the last error for the current thread.
///
/// # Safety
/// This function is safe to call from C. It only mutates thread-local state
/// owned by this library.
#[unsafe(no_mangle)]
pub unsafe extern "C" fn diffsol_clear_last_error() {
    clear_last_error();
}