servicepoint/crates/servicepoint_binding_c/src/connection.rs

124 lines
3.2 KiB
Rust
Raw Normal View History

2024-10-14 22:01:45 +02:00
//! C functions for interacting with [SPConnection]s
//!
//! prefix `sp_connection_`
use std::ffi::{c_char, CStr};
use std::ptr::null_mut;
use crate::{SPCommand, SPPacket};
2024-09-05 21:15:53 +02:00
/// A connection to the display.
///
/// # Examples
///
/// ```C
/// CConnection connection = sp_connection_open("172.23.42.29:2342");
/// if (connection != NULL)
/// sp_connection_send_command(connection, sp_command_clear());
2024-09-05 21:15:53 +02:00
/// ```
pub struct SPConnection(pub(crate) servicepoint::Connection);
2024-10-13 18:56:29 +02:00
/// Creates a new instance of [SPConnection].
///
2024-05-28 21:25:59 +02:00
/// returns: NULL if connection fails, or connected instance
///
2024-05-28 21:25:59 +02:00
/// # Panics
///
2024-10-14 22:01:45 +02:00
/// - when `host` is null or an invalid host
2024-05-28 21:25:59 +02:00
///
/// # Safety
///
/// The caller has to make sure that:
///
/// - the returned instance is freed in some way, either by using a consuming function or
2024-09-07 14:11:15 +02:00
/// by explicitly calling `sp_connection_free`.
#[no_mangle]
pub unsafe extern "C" fn sp_connection_open(
host: *const c_char,
2024-09-05 21:15:53 +02:00
) -> *mut SPConnection {
2024-10-14 22:01:45 +02:00
assert!(!host.is_null());
let host = CStr::from_ptr(host).to_str().expect("Bad encoding");
2024-09-05 21:15:53 +02:00
let connection = match servicepoint::Connection::open(host) {
Err(_) => return null_mut(),
Ok(value) => value,
};
2024-09-05 21:15:53 +02:00
Box::into_raw(Box::new(SPConnection(connection)))
}
2024-10-13 18:56:29 +02:00
/// Sends a [SPPacket] to the display using the [SPConnection].
///
/// The passed `packet` gets consumed.
2024-05-28 21:25:59 +02:00
///
/// returns: true in case of success
///
2024-10-14 22:01:45 +02:00
/// # Panics
///
/// - when `connection` is NULL
/// - when `packet` is NULL
///
2024-05-28 21:25:59 +02:00
/// # Safety
///
/// The caller has to make sure that:
///
2024-10-13 18:56:29 +02:00
/// - `connection` points to a valid instance of [SPConnection]
/// - `packet` points to a valid instance of [SPPacket]
/// - `packet` is not used concurrently or after this call
#[no_mangle]
pub unsafe extern "C" fn sp_connection_send_packet(
2024-09-05 21:15:53 +02:00
connection: *const SPConnection,
packet: *mut SPPacket,
) -> bool {
2024-10-14 22:01:45 +02:00
assert!(!connection.is_null());
assert!(!packet.is_null());
2024-05-28 21:25:59 +02:00
let packet = Box::from_raw(packet);
2024-09-05 21:15:53 +02:00
(*connection).0.send((*packet).0).is_ok()
}
2024-10-14 22:01:45 +02:00
/// Sends a [SPCommand] to the display using the [SPConnection].
///
/// The passed `command` gets consumed.
///
/// returns: true in case of success
///
2024-10-14 22:01:45 +02:00
/// # Panics
///
/// - when `connection` is NULL
/// - when `command` is NULL
///
/// # Safety
///
/// The caller has to make sure that:
///
2024-10-13 18:56:29 +02:00
/// - `connection` points to a valid instance of [SPConnection]
/// - `command` points to a valid instance of [SPPacket]
/// - `command` is not used concurrently or after this call
#[no_mangle]
pub unsafe extern "C" fn sp_connection_send_command(
connection: *const SPConnection,
command: *mut SPCommand,
) -> bool {
2024-10-14 22:01:45 +02:00
assert!(!connection.is_null());
assert!(!command.is_null());
let command = (*Box::from_raw(command)).0;
(*connection).0.send(command).is_ok()
}
2024-10-13 18:56:29 +02:00
/// Closes and deallocates a [SPConnection].
///
/// # Panics
///
/// - when `connection` is NULL
2024-05-28 21:25:59 +02:00
///
/// # Safety
///
/// The caller has to make sure that:
///
2024-10-13 18:56:29 +02:00
/// - `connection` points to a valid [SPConnection]
/// - `connection` is not used concurrently or after this call
#[no_mangle]
pub unsafe extern "C" fn sp_connection_free(connection: *mut SPConnection) {
2024-10-14 22:01:45 +02:00
assert!(!connection.is_null());
_ = Box::from_raw(connection);
}