gcdevproxy 0.3.0

GoodCam Device Proxy library
Documentation
use std::{
    ffi::c_void,
    os::raw::{c_char, c_int},
};

use log::{Level, LevelFilter, Log, Metadata, Record};

/// Foreign log handler.
type RawLogHandlerFn = unsafe extern "C" fn(context: *mut c_void, request: *const Record);

/// Logger using a foreign log handler.
struct Logger {
    context: *mut c_void,
    handler: RawLogHandlerFn,
}

impl Log for Logger {
    fn enabled(&self, _: &Metadata) -> bool {
        true
    }

    fn flush(&self) {}

    fn log(&self, record: &Record) {
        unsafe { (self.handler)(self.context, record) }
    }
}

unsafe impl Send for Logger {}
unsafe impl Sync for Logger {}

/// Set log handler.
///
/// The log handler can be set only once. The handler is expected to be a
/// function that accepts two arguments (context and log record) and does not
/// return anything (see the log_handler_t type for more info).
///
/// The handler and the context MUST be thread-safe. It must be safe to share
/// the context between multiple threads and it must be safe to call the
/// handler from multiple threads.
///
/// The context is optional and it may be NULL.
#[no_mangle]
extern "C" fn gcdp_set_logger(handler: RawLogHandlerFn, context: *mut c_void) -> c_int {
    let logger = Logger { context, handler };

    try_result!(libc::EALREADY, log::set_boxed_logger(Box::new(logger)));

    log::set_max_level(LevelFilter::Debug);

    0
}

/// Get log level from a given log record.
#[no_mangle]
extern "C" fn gcdp__log_record__get_level(record: *const Record) -> c_int {
    let record = unsafe { &*record };

    match record.level() {
        Level::Trace => 0,
        Level::Debug => 1,
        Level::Info => 2,
        Level::Warn => 3,
        Level::Error => 4,
    }
}

/// Get log message from a given log record.
///
/// The log message will be copied into a given buffer (unless it is NULL). The
/// size parameter is expected to contain the buffer capacity and after the
/// function returns, it will contain the original length of the log message.
///
/// The size cannot be NULL.
///
/// The string copied into the buffer may be truncated but it will be always
/// null-terminated.
#[no_mangle]
unsafe extern "C" fn gcdp__log_record__get_message(
    record: *const Record,
    buffer: *mut c_char,
    size: *mut usize,
) {
    let record = &*record;

    let args = record.args();

    if let Some(msg) = args.as_str() {
        *size = super::str_to_cstr(msg, buffer, *size);
    } else {
        *size = super::str_to_cstr(&args.to_string(), buffer, *size);
    }
}