feat: add example for cdylib
This commit is contained in:
@@ -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 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
|
// region: Error
|
||||||
|
|
||||||
@@ -25,6 +28,13 @@ enum Error {
|
|||||||
/// Error when manipulating with C-style string.
|
/// Error when manipulating with C-style string.
|
||||||
#[error("{0}")]
|
#[error("{0}")]
|
||||||
CStrFfi(#[from] cstr_ffi::Error),
|
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.
|
/// Result type used in this crate.
|
||||||
@@ -34,9 +44,12 @@ type Result<T> = std::result::Result<T, Error>;
|
|||||||
|
|
||||||
// region: Macros
|
// region: Macros
|
||||||
|
|
||||||
|
|
||||||
// endregion
|
// endregion
|
||||||
|
|
||||||
|
// region: Exposed Functions
|
||||||
|
|
||||||
|
// region: Facilities
|
||||||
|
|
||||||
#[unsafe(no_mangle)]
|
#[unsafe(no_mangle)]
|
||||||
pub extern "C" fn WFStartup() -> bool {
|
pub extern "C" fn WFStartup() -> bool {
|
||||||
true
|
true
|
||||||
@@ -59,6 +72,89 @@ pub extern "C" fn WFHasPrivilege() -> bool {
|
|||||||
|
|
||||||
#[unsafe(no_mangle)]
|
#[unsafe(no_mangle)]
|
||||||
pub extern "C" fn WFAdd(left: u32, right: u32, rv: *mut u32) -> bool {
|
pub extern "C" fn WFAdd(left: u32, right: u32, rv: *mut u32) -> bool {
|
||||||
unsafe { *rv = left + right; }
|
unsafe {
|
||||||
|
*rv = left + right;
|
||||||
|
}
|
||||||
return true;
|
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
|
||||||
|
|||||||
@@ -44,9 +44,14 @@ impl<T> ObjectPool<T> {
|
|||||||
Ok(Self::key_to_token(&key))
|
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)) {
|
match self.objs.remove(Self::token_to_key(token)) {
|
||||||
Some(_) => Ok(()),
|
Some(obj) => Ok(obj),
|
||||||
None => Err(Error::NoSuchToken),
|
None => Err(Error::NoSuchToken),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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> {}
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user