diff --git a/Cargo.lock b/Cargo.lock index 3c1e086..1c2de8f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -370,6 +370,14 @@ dependencies = [ "local-ip-address", ] +[[package]] +name = "curseofrust-ffi" +version = "0.1.0" +dependencies = [ + "curseofrust", + "curseofrust-cli-parser", +] + [[package]] name = "curseofrust-gui-cocoa" version = "0.1.0" diff --git a/Cargo.toml b/Cargo.toml index 9c7a308..ff4cb5b 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -10,7 +10,7 @@ fastrand = "2.1.0" [workspace] resolver = "2" -members = ["gui-cocoa", "console", "cli", "msg", "server", "net-foundation"] +members = ["gui-cocoa", "console", "cli", "msg", "ffi", "server", "net-foundation"] [profile.release] panic = "abort" diff --git a/ffi/Cargo.toml b/ffi/Cargo.toml new file mode 100644 index 0000000..797b788 --- /dev/null +++ b/ffi/Cargo.toml @@ -0,0 +1,12 @@ +[package] +name = "curseofrust-ffi" +version = "0.1.0" +edition = "2021" +authors = ["C191239 "] + +[lib] +crate-type = ["staticlib"] + +[dependencies] +curseofrust = { path = ".." } +cli-parser = { path = "../cli", package = "curseofrust-cli-parser" } diff --git a/ffi/include/curseofrust.h b/ffi/include/curseofrust.h new file mode 100644 index 0000000..a8dc75b --- /dev/null +++ b/ffi/include/curseofrust.h @@ -0,0 +1,35 @@ +#ifndef CURSEOFRUST_H +#define CURSEOFRUST_H + +#include +#include + +typedef struct { + void *first; + void *second; +} CORFunctionReturn; +typedef struct { + int32_t x; + int32_t y; +} CORPosition; +typedef struct { + CORPosition cursor; + uint16_t xskip; + uint16_t xlen; +} CORInterface; + +typedef void *CORBasicOptsRef; +typedef void *CORMultiplayerOptsRef; +typedef void *CORStateRef; + +CORFunctionReturn CORParseOptions(char *_Nonnull optStringPtr); +void CORReleaseErrorString(char *_Nonnull errorStringPtr); +CORFunctionReturn CORMakeState(CORBasicOptsRef _Nonnull basicOptsPtr); +CORInterface CORMakeInterface(CORStateRef _Nonnull statePtr); +uint64_t CORGetSeed(CORStateRef _Nonnull statePtr); +uint32_t CORGetGridHeight(CORStateRef _Nonnull statePtr); +uint32_t CORGetGridWidth(CORStateRef _Nonnull statePtr); +void CORKingsMove(CORStateRef _Nonnull statePtr); +void CORSimulate(CORStateRef _Nonnull statePtr); + +#endif \ No newline at end of file diff --git a/ffi/src/lib.rs b/ffi/src/lib.rs new file mode 100644 index 0000000..5a849ad --- /dev/null +++ b/ffi/src/lib.rs @@ -0,0 +1,190 @@ +#![allow(non_snake_case)] + +use std::{ + ffi::{c_char, c_void, CStr, CString}, + ptr::null, +}; + +use curseofrust::{ + state::{BasicOpts, State, UI}, + Pos, +}; + +/// Helper struct for functions that: +/// - has two return values. +/// - involves error handling. +#[repr(C)] +pub struct CORFunctionReturn { + first: *const c_void, + second: *const c_void, +} + +/// Parse Curse of Rust options from C string. +/// # Params +/// - optStringPtr: A pointer to options string. +/// +/// # Returns +/// Always return a `CORParseOptionsReturnRef`. +/// ## On success +/// - first: A `CORBasicOptsRef`. +/// - second: A `CORMultiplayerOptsRef`. +/// ## On error +/// - first: A `nil` pointer. +/// - second: Error msg in `char*` form. +#[no_mangle] +pub extern "C" fn CORParseOptions(optStringPtr: *const c_char) -> CORFunctionReturn { + let opt_string = + String::from_utf8_lossy(unsafe { CStr::from_ptr(optStringPtr) }.to_bytes()).to_string(); + cli_parser::parse(opt_string.split_whitespace()).map_or_else( + |e| CORFunctionReturn { + first: null(), + second: CString::new(e.to_string()) + .expect("CString::new failure in CORParseOptions()") + .into_raw() + .cast(), + }, + |(basic_opts, multiplayer_opts)| CORFunctionReturn { + first: Box::into_raw(Box::new(basic_opts)).cast(), + second: Box::into_raw(Box::new(multiplayer_opts)).cast(), + }, + ) +} + +/// Release the error string returned by [`CORParseOptions`], [`CORMakeState`]. +#[no_mangle] +pub extern "C" fn CORReleaseErrorString(errorStringPtr: *const c_char) { + unsafe { drop(CString::from_raw(errorStringPtr as *mut _)) } +} + +/// Generate initial [`State`] from [`BasicOpts`]. +/// # Returns +/// ## On success +/// first: A `CORStateRef`. +/// second: A `nil` pointer. +/// ## On error +/// first: A `nil` pointer. +/// second: Error msg in `char*` form. +/// +/// # Extra info +/// The param `basicOptsPtr` will be consumed. +#[no_mangle] +pub extern "C" fn CORMakeState(basicOptsPtr: *const c_void) -> CORFunctionReturn { + let basic_opts = unsafe { Box::from_raw(basicOptsPtr.cast::() as *mut _) }; + State::new(*basic_opts).map_or_else( + |e| CORFunctionReturn { + first: CString::new(e.to_string()) + .expect("CString::new failure in CORMakeState()") + .into_raw() + .cast(), + second: null(), + }, + |v| CORFunctionReturn { + first: null(), + second: Box::into_raw(Box::new(v)).cast(), + }, + ) +} + +/// C struct of [`Pos`]. +#[repr(C)] +pub struct CORPosition { + /// Horizontal axis. + pub x: i32, + /// Vertical axis. + pub y: i32, +} + +impl From for CORPosition { + fn from(value: Pos) -> Self { + Self { + x: value.0, + y: value.1, + } + } +} + +impl Into for CORPosition { + fn into(self) -> Pos { + Pos(self.x, self.y) + } +} + +/// C struct of [`UI`]. +#[repr(C)] +pub struct CORInterface { + pub cursor: CORPosition, + /// Number of tiles to skip in the beginning of + /// every line. + pub xskip: u16, + /// Total max number of tiles in horizontal direction. + pub xlen: u16, +} + +impl From for CORInterface { + fn from(value: UI) -> Self { + Self { + cursor: value.cursor.into(), + xskip: value.xskip, + xlen: value.xlen, + } + } +} + +/// Generate [`UI`] from [`State`]. +#[no_mangle] +pub extern "C" fn CORMakeInterface(statePtr: *const c_void) -> CORInterface { + let state = unsafe { Box::from_raw(statePtr.cast::() as *mut _) }; + let ui = UI::new(&state).into(); + Box::leak(state); + ui +} + +/// Extract game seed from [`State`]. +#[no_mangle] +pub extern "C" fn CORGetSeed(statePtr: *const c_void) -> u64 { + let state: Box = unsafe { Box::from_raw(statePtr.cast::() as *mut _) }; + let seed = state.seed; + Box::leak(state); + seed +} + +/// Extract grid height form [`State`]. +#[no_mangle] +pub extern "C" fn CORGetGridHeight(statePtr: *const c_void) -> u32 { + let state: Box = unsafe { Box::from_raw(statePtr.cast::() as *mut _) }; + let height = state.grid.height(); + Box::leak(state); + height +} + +/// Extract grid width form [`State`]. +#[no_mangle] +pub extern "C" fn CORGetGridWidth(statePtr: *const c_void) -> u32 { + let state: Box = unsafe { Box::from_raw(statePtr.cast::() as *mut _) }; + let width = state.grid.width(); + Box::leak(state); + width +} + +#[no_mangle] +pub extern "C" fn CORKingsMove(statePtr: *const c_void) { + let mut state: Box = unsafe { Box::from_raw(statePtr.cast::() as *mut _) }; + state.kings_move(); + Box::leak(state); +} + +#[no_mangle] +pub extern "C" fn CORSimulate(statePtr: *const c_void) { + let mut state: Box = unsafe { Box::from_raw(statePtr.cast::() as *mut _) }; + state.simulate(); + Box::leak(state); +} + +#[no_mangle] +pub extern "C" fn CORGetTile(statePtr: *const c_void, pos: CORPosition) { + let state: Box = unsafe { Box::from_raw(statePtr.cast::() as *mut _) }; + let tile = state.grid.tile(pos.into()); + todo!() + // What should it return? + // C191239 2024-06-25 11:44:32 +0800 +}