1
0

feat: add example for cdylib

This commit is contained in:
2026-05-10 23:16:56 +08:00
parent 7d92f9a4a0
commit f0e610f8c8
3 changed files with 112 additions and 94 deletions

View File

@@ -1,11 +1,14 @@
use std::ffi::{CString, c_char};
use thiserror::Error as TeError;
mod object_pool;
mod last_error;
mod cstr_ffi;
mod wrapper;
mod last_error;
mod object_pool;
use object_pool::ObjectPool;
use std::ffi::{CString, c_char};
use std::sync::{LazyLock, PoisonError, RwLock};
use thiserror::Error as TeError;
use wfassoc::highlevel::{Program, Schema};
pub use object_pool::Token;
// region: Error
@@ -25,6 +28,13 @@ enum Error {
/// Error when manipulating with C-style string.
#[error("{0}")]
CStrFfi(#[from] cstr_ffi::Error),
/// Error when manipulating with object pool.
#[error("{0}")]
ObjectPool(#[from] object_pool::Error),
/// Error when manipulating with poison RwLock
#[error("RwLock is poisoning")]
PoisonRwLock,
}
/// Result type used in this crate.
@@ -34,9 +44,12 @@ type Result<T> = std::result::Result<T, Error>;
// region: Macros
// endregion
// region: Exposed Functions
// region: Facilities
#[unsafe(no_mangle)]
pub extern "C" fn WFStartup() -> bool {
true
@@ -59,6 +72,89 @@ pub extern "C" fn WFHasPrivilege() -> bool {
#[unsafe(no_mangle)]
pub extern "C" fn WFAdd(left: u32, right: u32, rv: *mut u32) -> bool {
unsafe { *rv = left + right; }
unsafe {
*rv = left + right;
}
return true;
}
// endregion
// region: Schema
const SCHEMA_POOL: LazyLock<RwLock<ObjectPool<Schema>>> =
LazyLock::new(|| RwLock::new(ObjectPool::new()));
#[unsafe(no_mangle)]
pub extern "C" fn WFSchemaCreate(out_token: *mut Token) -> bool {
fn inner() -> Result<Token> {
let binding = &SCHEMA_POOL;
let mut pool = binding.write().map_err(|_| Error::PoisonRwLock)?;
Ok(pool.allocate(Schema::new())?)
}
match inner() {
Ok(rv) => {
unsafe { *out_token = rv };
true
}
Err(e) => {
last_error::set_last_error(e.to_string().as_str());
false
}
}
}
#[unsafe(no_mangle)]
pub extern "C" fn WFSchemaIntoProgram(in_token: Token, out_token: *mut Token) -> bool {
fn inner(in_token: Token) -> Result<Token> {
let binding = &SCHEMA_POOL;
let mut pool = binding.write().map_err(|_| Error::PoisonRwLock)?;
let schema = pool.pop(in_token)?;
let binding = &PROGRAM_POOL;
let mut pool = binding.write().map_err(|_| Error::PoisonRwLock)?;
let program = schema.into_program()?;
Ok(pool.allocate(program)?)
}
match inner(in_token) {
Ok(rv) => {
unsafe { *out_token = rv };
true
}
Err(e) => {
last_error::set_last_error(e.to_string().as_str());
false
}
}
}
#[unsafe(no_mangle)]
pub extern "C" fn WFSchemaDestroy(in_token: Token) -> bool {
fn inner(in_token: Token) -> Result<()> {
let binding = &SCHEMA_POOL;
let mut pool = binding.write().map_err(|_| Error::PoisonRwLock)?;
Ok(pool.free(in_token)?)
}
match inner(in_token) {
Ok(_) => true,
Err(e) => {
last_error::set_last_error(e.to_string().as_str());
false
}
}
}
// endregion
// region: Program
const PROGRAM_POOL: LazyLock<RwLock<ObjectPool<Program>>> =
LazyLock::new(|| RwLock::new(ObjectPool::new()));
// endregion
// endregion

View File

@@ -44,9 +44,14 @@ impl<T> ObjectPool<T> {
Ok(Self::key_to_token(&key))
}
pub fn delete(&mut self, token: Token) -> Result<(), Error> {
pub fn free(&mut self, token: Token) -> Result<(), Error> {
let _ = self.pop(token)?;
Ok(())
}
pub fn pop(&mut self, token: Token) -> Result<T, Error> {
match self.objs.remove(Self::token_to_key(token)) {
Some(_) => Ok(()),
Some(obj) => Ok(obj),
None => Err(Error::NoSuchToken),
}
}

View File

@@ -1,83 +0,0 @@
use std::collections::HashMap;
use thiserror::Error as TeError;
/// Error occurs in this module.
#[derive(Debug, TeError)]
pub enum Error {}
/// Result type used in this module.
type Result<T> = std::result::Result<T, Error>;
/// Schema is the sketchpad of complete Program.
///
/// We will create a Schema first, fill some properties, add file extensions,
/// then convert it into immutable Program for following using.
#[derive(Debug)]
pub struct Schema {
identifier: String,
path: String,
clsid: String,
icons: HashMap<String, String>,
behaviors: HashMap<String, String>,
exts: HashMap<String, SchemaExt>,
}
/// Internal used struct as the Schema file extensions hashmap value type.
#[derive(Debug)]
struct SchemaExt {
name: String,
icon: String,
behavior: String,
}
impl Schema {
pub fn new() -> Self {
Self {
identifier: String::new(),
path: String::new(),
clsid: String::new(),
icons: HashMap::new(),
behaviors: HashMap::new(),
exts: HashMap::new(),
}
}
pub fn set_identifier(&mut self, identifier: &str) -> Result<()> {}
pub fn set_path(&mut self, exe_path: &str) -> Result<()> {}
pub fn set_clsid(&mut self, clsid: &str) -> Result<()> {}
pub fn add_icon(&mut self, name: &str, value: &str) -> Result<()> {}
pub fn add_behavior(&mut self, name: &str, value: &str) -> Result<()> {}
pub fn add_ext(
&mut self,
ext: &str,
ext_name: &str,
ext_icon: &str,
ext_behavior: &str,
) -> Result<()> {
}
pub fn into_program(self) -> Result<Program> {
Program::new(self)
}
}
/// Program is a complete and immutable program representer
pub struct Program {}
impl TryFrom<Schema> for Program {
type Error = Error;
fn try_from(value: Schema) -> std::result::Result<Self, Self::Error> {
Self::new(value)
}
}
impl Program {
pub fn new(schema: Schema) -> Result<Self> {}
}