2026-05-10 23:16:56 +08:00
|
|
|
mod cstr_ffi;
|
|
|
|
|
mod last_error;
|
|
|
|
|
mod object_pool;
|
2026-05-10 16:43:25 +08:00
|
|
|
|
2026-05-10 23:16:56 +08:00
|
|
|
use object_pool::ObjectPool;
|
2025-10-18 18:17:05 +08:00
|
|
|
use std::ffi::{CString, c_char};
|
2026-05-10 23:16:56 +08:00
|
|
|
use std::sync::{LazyLock, PoisonError, RwLock};
|
2025-10-18 18:17:05 +08:00
|
|
|
use thiserror::Error as TeError;
|
2026-05-10 23:16:56 +08:00
|
|
|
use wfassoc::highlevel::{Program, Schema};
|
2025-10-18 18:17:05 +08:00
|
|
|
|
2026-05-10 23:16:56 +08:00
|
|
|
pub use object_pool::Token;
|
2025-11-26 22:40:17 +08:00
|
|
|
|
2025-10-18 18:17:05 +08:00
|
|
|
// region: Error
|
|
|
|
|
|
|
|
|
|
/// Error occurs in this crate.
|
|
|
|
|
#[derive(Debug, TeError)]
|
|
|
|
|
enum Error {
|
2026-05-10 16:55:09 +08:00
|
|
|
/// Error when operating Schema.
|
|
|
|
|
#[error("{0}")]
|
|
|
|
|
Schema(#[from] wfassoc::highlevel::SchemaError),
|
2026-05-09 16:27:04 +08:00
|
|
|
/// Error when parsing Schema into Program.
|
2025-10-18 18:17:05 +08:00
|
|
|
#[error("{0}")]
|
2026-05-09 16:27:04 +08:00
|
|
|
ParseProgram(#[from] wfassoc::highlevel::ParseProgramError),
|
|
|
|
|
/// Error when operating Program.
|
|
|
|
|
#[error("{0}")]
|
|
|
|
|
Program(#[from] wfassoc::highlevel::ProgramError),
|
2025-10-18 18:17:05 +08:00
|
|
|
|
2026-05-10 16:55:09 +08:00
|
|
|
/// Error when manipulating with C-style string.
|
|
|
|
|
#[error("{0}")]
|
|
|
|
|
CStrFfi(#[from] cstr_ffi::Error),
|
2026-05-10 23:16:56 +08:00
|
|
|
/// Error when manipulating with object pool.
|
|
|
|
|
#[error("{0}")]
|
|
|
|
|
ObjectPool(#[from] object_pool::Error),
|
|
|
|
|
|
|
|
|
|
/// Error when manipulating with poison RwLock
|
|
|
|
|
#[error("RwLock is poisoning")]
|
|
|
|
|
PoisonRwLock,
|
2025-10-18 18:17:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Result type used in this crate.
|
|
|
|
|
type Result<T> = std::result::Result<T, Error>;
|
|
|
|
|
|
|
|
|
|
// endregion
|
|
|
|
|
|
|
|
|
|
// region: Macros
|
|
|
|
|
|
|
|
|
|
// endregion
|
2025-10-04 22:04:30 +08:00
|
|
|
|
2026-05-10 23:16:56 +08:00
|
|
|
// region: Exposed Functions
|
|
|
|
|
|
|
|
|
|
// region: Facilities
|
|
|
|
|
|
2025-10-18 09:55:08 +08:00
|
|
|
#[unsafe(no_mangle)]
|
2025-10-18 23:11:33 +08:00
|
|
|
pub extern "C" fn WFStartup() -> bool {
|
2026-05-10 16:55:09 +08:00
|
|
|
true
|
2025-10-18 23:11:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFShutdown() -> bool {
|
2026-05-10 16:55:09 +08:00
|
|
|
true
|
2025-10-18 23:11:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFGetLastError() -> *const c_char {
|
2025-11-26 22:40:17 +08:00
|
|
|
last_error::get_last_error()
|
2025-10-18 23:11:33 +08:00
|
|
|
}
|
|
|
|
|
|
2025-10-19 14:10:21 +08:00
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFHasPrivilege() -> bool {
|
2026-05-09 16:27:04 +08:00
|
|
|
wfassoc::win32::utilities::has_privilege()
|
2025-10-19 14:10:21 +08:00
|
|
|
}
|
|
|
|
|
|
2025-10-18 23:11:33 +08:00
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFAdd(left: u32, right: u32, rv: *mut u32) -> bool {
|
2026-05-10 23:16:56 +08:00
|
|
|
unsafe {
|
|
|
|
|
*rv = left + right;
|
|
|
|
|
}
|
2025-10-18 23:11:33 +08:00
|
|
|
return true;
|
2025-10-04 22:04:30 +08:00
|
|
|
}
|
2026-05-10 23:16:56 +08:00
|
|
|
|
|
|
|
|
// 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
|