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-15 16:00:54 +08:00
|
|
|
use num_enum::TryFromPrimitive;
|
2026-05-10 23:16:56 +08:00
|
|
|
use object_pool::ObjectPool;
|
2026-05-11 13:33:01 +08:00
|
|
|
use std::sync::{LazyLock, 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-11 13:33:01 +08:00
|
|
|
pub use cstr_ffi::CStyleString;
|
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),
|
|
|
|
|
|
2026-05-15 16:00:54 +08:00
|
|
|
/// Error occurs when checking enum value
|
|
|
|
|
#[error("")]
|
|
|
|
|
EnumOutOfRange,
|
2026-05-10 23:16:56 +08:00
|
|
|
/// 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
|
|
|
|
|
|
2026-05-11 13:33:01 +08:00
|
|
|
macro_rules! in_param_ty {
|
|
|
|
|
($t:ty) => {
|
|
|
|
|
$t
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
macro_rules! out_param_ty {
|
|
|
|
|
($t:ty) => {
|
|
|
|
|
*mut $t
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
macro_rules! set_out_param {
|
|
|
|
|
($lhs:expr, $rhs:expr) => {
|
|
|
|
|
unsafe { *$lhs = $rhs };
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-15 16:00:54 +08:00
|
|
|
macro_rules! resolve_enum {
|
|
|
|
|
($t:ty, $v:expr) => {
|
|
|
|
|
<$t>::try_from($v).map_err(|_| Error::EnumOutOfRange)
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-11 13:33:01 +08:00
|
|
|
macro_rules! pull_reader {
|
|
|
|
|
($pool:expr) => {
|
|
|
|
|
$pool.read().map_err(|_| Error::PoisonRwLock)
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
macro_rules! pull_writer {
|
|
|
|
|
($pool:expr) => {
|
|
|
|
|
$pool.write().map_err(|_| Error::PoisonRwLock)
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-11 13:54:47 +08:00
|
|
|
/// Macro to wrap inner function execution with standard error handling pattern.
|
|
|
|
|
///
|
|
|
|
|
/// For functions with no output parameter and no input parameters:
|
|
|
|
|
/// ```ignore
|
|
|
|
|
/// cffi_wrapper!(|| {
|
|
|
|
|
/// // inner function body returning Result<()>
|
|
|
|
|
/// });
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
|
|
|
|
/// For functions with no output parameter and with input parameters:
|
|
|
|
|
/// ```ignore
|
|
|
|
|
/// cffi_wrapper!(|param1: Type1, param2: Type2| {
|
|
|
|
|
/// // inner function body using param1, param2 returning Result<()>
|
|
|
|
|
/// });
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
|
|
|
|
/// For functions with one output parameter and no input parameters:
|
|
|
|
|
/// ```ignore
|
|
|
|
|
/// cffi_wrapper!(|| -> (out_param, OutType) {
|
|
|
|
|
/// // inner function body returning Result<T>
|
|
|
|
|
/// });
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
|
|
|
|
/// For functions with one output parameter and with input parameters:
|
|
|
|
|
/// ```ignore
|
|
|
|
|
/// cffi_wrapper!(|param1: Type1| -> (out_param, OutType) {
|
|
|
|
|
/// // inner function body using param1 returning Result<T>
|
|
|
|
|
/// });
|
|
|
|
|
/// ```
|
|
|
|
|
macro_rules! cffi_wrapper {
|
|
|
|
|
// Case with output parameter and input parameters
|
|
|
|
|
(|$($param:ident: $param_ty:ty),*| -> ($out_param:ident: $out_param_ty:ty) $inner_body:block) => {{
|
|
|
|
|
fn inner($($param: $param_ty),*) -> Result<$out_param_ty> $inner_body
|
|
|
|
|
|
|
|
|
|
match inner($($param),*) {
|
|
|
|
|
Ok(rv) => {
|
|
|
|
|
set_out_param!($out_param, rv);
|
|
|
|
|
last_error::clear_last_error();
|
|
|
|
|
true
|
|
|
|
|
}
|
|
|
|
|
Err(e) => {
|
|
|
|
|
last_error::set_last_error(e.to_string().as_str());
|
|
|
|
|
false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}};
|
|
|
|
|
|
|
|
|
|
// Case with output parameter and no input parameters
|
|
|
|
|
(|| -> ($out_param:ident: $out_param_ty:ty) $inner_body:block) => {{
|
|
|
|
|
fn inner() -> Result<$out_param_ty> $inner_body
|
|
|
|
|
|
|
|
|
|
match inner() {
|
|
|
|
|
Ok(rv) => {
|
|
|
|
|
set_out_param!($out_param, rv);
|
|
|
|
|
last_error::clear_last_error();
|
|
|
|
|
true
|
|
|
|
|
}
|
|
|
|
|
Err(e) => {
|
|
|
|
|
last_error::set_last_error(e.to_string().as_str());
|
|
|
|
|
false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}};
|
|
|
|
|
|
|
|
|
|
// Case without output parameter but with input parameters
|
|
|
|
|
(|$($param:ident: $param_ty:ty),*| $inner_body:block) => {{
|
|
|
|
|
fn inner($($param: $param_ty),*) -> Result<()> $inner_body
|
|
|
|
|
|
|
|
|
|
match inner($($param),*) {
|
|
|
|
|
Ok(_) => {
|
|
|
|
|
last_error::clear_last_error();
|
|
|
|
|
true
|
|
|
|
|
}
|
|
|
|
|
Err(e) => {
|
|
|
|
|
last_error::set_last_error(e.to_string().as_str());
|
|
|
|
|
false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}};
|
|
|
|
|
|
|
|
|
|
// Case without output parameter and no input parameters
|
2026-05-11 14:24:28 +08:00
|
|
|
(|| $inner_body:block) => {{
|
2026-05-11 13:54:47 +08:00
|
|
|
fn inner() -> Result<()> $inner_body
|
|
|
|
|
|
|
|
|
|
match inner() {
|
|
|
|
|
Ok(_) => {
|
|
|
|
|
last_error::clear_last_error();
|
|
|
|
|
true
|
|
|
|
|
}
|
|
|
|
|
Err(e) => {
|
|
|
|
|
last_error::set_last_error(e.to_string().as_str());
|
|
|
|
|
false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}};
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-18 18:17:05 +08:00
|
|
|
// endregion
|
2025-10-04 22:04:30 +08:00
|
|
|
|
2026-05-15 16:00:54 +08:00
|
|
|
// region: Object Pools
|
|
|
|
|
|
|
|
|
|
static SCHEMA_POOL: LazyLock<RwLock<ObjectPool<Schema>>> =
|
|
|
|
|
LazyLock::new(|| RwLock::new(ObjectPool::new()));
|
|
|
|
|
|
|
|
|
|
static PROGRAM_POOL: LazyLock<RwLock<ObjectPool<Program>>> =
|
|
|
|
|
LazyLock::new(|| RwLock::new(ObjectPool::new()));
|
|
|
|
|
|
|
|
|
|
// endregion
|
|
|
|
|
|
|
|
|
|
// region: Exposed Types
|
|
|
|
|
|
|
|
|
|
#[repr(u32)]
|
|
|
|
|
#[derive(Debug, Copy, Clone, TryFromPrimitive)]
|
|
|
|
|
pub enum Scope {
|
|
|
|
|
User = 0,
|
|
|
|
|
System = 1,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl From<Scope> for wfassoc::Scope {
|
|
|
|
|
fn from(value: Scope) -> Self {
|
|
|
|
|
match value {
|
|
|
|
|
Scope::User => wfassoc::Scope::User,
|
|
|
|
|
Scope::System => wfassoc::Scope::System,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[repr(u32)]
|
|
|
|
|
#[derive(Debug, Copy, Clone, TryFromPrimitive)]
|
|
|
|
|
pub enum View {
|
2026-05-15 17:02:29 +08:00
|
|
|
User = 0,
|
|
|
|
|
System = 1,
|
|
|
|
|
Hybrid = 2,
|
2026-05-15 16:00:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl From<View> for wfassoc::View {
|
|
|
|
|
fn from(value: View) -> Self {
|
|
|
|
|
match value {
|
|
|
|
|
View::User => wfassoc::View::User,
|
|
|
|
|
View::System => wfassoc::View::System,
|
|
|
|
|
View::Hybrid => wfassoc::View::Hybrid,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// endregion
|
|
|
|
|
|
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-11 14:24:28 +08:00
|
|
|
// Initialize all pool by fetching writer from them
|
|
|
|
|
cffi_wrapper!(|| {
|
|
|
|
|
let _pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let _pool = pull_writer!(PROGRAM_POOL)?;
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
2025-10-18 23:11:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFShutdown() -> bool {
|
2026-05-11 14:24:28 +08:00
|
|
|
// Free all pool stored objects
|
|
|
|
|
cffi_wrapper!(|| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
pool.clear();
|
|
|
|
|
let mut pool = pull_writer!(PROGRAM_POOL)?;
|
|
|
|
|
pool.clear();
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
2025-10-18 23:11:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
2026-05-11 13:33:01 +08:00
|
|
|
pub extern "C" fn WFGetLastError() -> CStyleString {
|
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
|
|
|
}
|
|
|
|
|
|
2026-05-10 23:16:56 +08:00
|
|
|
// endregion
|
|
|
|
|
|
|
|
|
|
// region: Schema
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
2026-05-11 13:33:01 +08:00
|
|
|
pub extern "C" fn WFSchemaCreate(out_schema: out_param_ty!(Token)) -> bool {
|
2026-05-11 13:54:47 +08:00
|
|
|
cffi_wrapper!(|| -> (out_schema: Token) {
|
2026-05-11 13:33:01 +08:00
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
2026-05-10 23:16:56 +08:00
|
|
|
Ok(pool.allocate(Schema::new())?)
|
2026-05-11 13:54:47 +08:00
|
|
|
})
|
2026-05-10 23:16:56 +08:00
|
|
|
}
|
|
|
|
|
|
2026-05-15 16:00:54 +08:00
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaDestroy(in_schema: in_param_ty!(Token)) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_schema: Token| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
Ok(pool.free(in_schema)?)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-11 13:33:01 +08:00
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaSetIdentifier(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
in_value: in_param_ty!(CStyleString),
|
|
|
|
|
) -> bool {
|
2026-05-11 13:54:47 +08:00
|
|
|
cffi_wrapper!(|in_schema: Token, in_value: CStyleString| {
|
2026-05-11 13:33:01 +08:00
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
schema.set_identifier(cstr_ffi::parse_ffi_string(in_value)?);
|
|
|
|
|
Ok(())
|
2026-05-11 13:54:47 +08:00
|
|
|
})
|
2026-05-11 13:33:01 +08:00
|
|
|
}
|
2026-05-10 23:16:56 +08:00
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
2026-05-11 14:24:28 +08:00
|
|
|
pub extern "C" fn WFSchemaSetPath(
|
2026-05-11 13:33:01 +08:00
|
|
|
in_schema: in_param_ty!(Token),
|
2026-05-11 14:24:28 +08:00
|
|
|
in_value: in_param_ty!(CStyleString),
|
2026-05-11 13:33:01 +08:00
|
|
|
) -> bool {
|
2026-05-11 14:24:28 +08:00
|
|
|
cffi_wrapper!(|in_schema: Token, in_value: CStyleString| {
|
2026-05-11 13:33:01 +08:00
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
2026-05-11 14:24:28 +08:00
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
schema.set_path(cstr_ffi::parse_ffi_string(in_value)?);
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
|
|
|
|
}
|
2026-05-10 23:16:56 +08:00
|
|
|
|
2026-05-11 14:24:28 +08:00
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaSetClsid(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
in_value: in_param_ty!(CStyleString),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_schema: Token, in_value: CStyleString| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
schema.set_clsid(cstr_ffi::parse_ffi_string(in_value)?);
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaSetName(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
in_value: in_param_ty!(CStyleString),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_schema: Token, in_value: CStyleString| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
|
|
|
|
|
let name = if in_value.is_null() {
|
|
|
|
|
None
|
|
|
|
|
} else {
|
|
|
|
|
Some(cstr_ffi::parse_ffi_string(in_value)?)
|
|
|
|
|
};
|
|
|
|
|
schema.set_name(name);
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaSetIcon(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
in_value: in_param_ty!(CStyleString),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_schema: Token, in_value: CStyleString| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
|
|
|
|
|
let icon = if in_value.is_null() {
|
|
|
|
|
None
|
|
|
|
|
} else {
|
|
|
|
|
Some(cstr_ffi::parse_ffi_string(in_value)?)
|
|
|
|
|
};
|
|
|
|
|
schema.set_icon(icon);
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaSetBehavior(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
in_value: in_param_ty!(CStyleString),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_schema: Token, in_value: CStyleString| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
|
|
|
|
|
let behavior = if in_value.is_null() {
|
|
|
|
|
None
|
|
|
|
|
} else {
|
|
|
|
|
Some(cstr_ffi::parse_ffi_string(in_value)?)
|
|
|
|
|
};
|
|
|
|
|
schema.set_behavior(behavior);
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaAddStr(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
in_name: in_param_ty!(CStyleString),
|
|
|
|
|
in_value: in_param_ty!(CStyleString),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(
|
|
|
|
|
|in_schema: Token, in_name: CStyleString, in_value: CStyleString| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
schema.add_str(
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_name)?,
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_value)?,
|
|
|
|
|
)?;
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaAddIcon(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
in_name: in_param_ty!(CStyleString),
|
|
|
|
|
in_value: in_param_ty!(CStyleString),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(
|
|
|
|
|
|in_schema: Token, in_name: CStyleString, in_value: CStyleString| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
schema.add_icon(
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_name)?,
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_value)?,
|
|
|
|
|
)?;
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaAddBehavior(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
in_name: in_param_ty!(CStyleString),
|
|
|
|
|
in_value: in_param_ty!(CStyleString),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(
|
|
|
|
|
|in_schema: Token, in_name: CStyleString, in_value: CStyleString| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
schema.add_behavior(
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_name)?,
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_value)?,
|
|
|
|
|
)?;
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFSchemaAddExt(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
in_ext: in_param_ty!(CStyleString),
|
|
|
|
|
in_ext_name: in_param_ty!(CStyleString),
|
|
|
|
|
in_ext_icon: in_param_ty!(CStyleString),
|
|
|
|
|
in_ext_behavior: in_param_ty!(CStyleString),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_schema: Token,
|
|
|
|
|
in_ext: CStyleString,
|
|
|
|
|
in_ext_name: CStyleString,
|
|
|
|
|
in_ext_icon: CStyleString,
|
|
|
|
|
in_ext_behavior: CStyleString| {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.get_mut(in_schema)?;
|
|
|
|
|
schema.add_ext(
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_ext)?,
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_ext_name)?,
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_ext_icon)?,
|
|
|
|
|
cstr_ffi::parse_ffi_string(in_ext_behavior)?,
|
|
|
|
|
)?;
|
|
|
|
|
Ok(())
|
2026-05-11 13:54:47 +08:00
|
|
|
})
|
2026-05-10 23:16:56 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// endregion
|
|
|
|
|
|
|
|
|
|
// region: Program
|
|
|
|
|
|
2026-05-11 14:24:28 +08:00
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFProgramCreate(
|
|
|
|
|
in_schema: in_param_ty!(Token),
|
|
|
|
|
out_program: out_param_ty!(Token),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_schema: Token| -> (out_program: Token) {
|
|
|
|
|
let mut pool = pull_writer!(SCHEMA_POOL)?;
|
|
|
|
|
let schema = pool.pop(in_schema)?;
|
|
|
|
|
|
|
|
|
|
let mut pool = pull_writer!(PROGRAM_POOL)?;
|
|
|
|
|
let program = Program::new(schema)?;
|
|
|
|
|
Ok(pool.allocate(program)?)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFProgramDestroy(in_program: in_param_ty!(Token)) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_program: Token| {
|
|
|
|
|
let mut pool = pull_writer!(PROGRAM_POOL)?;
|
|
|
|
|
Ok(pool.free(in_program)?)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-15 16:00:54 +08:00
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFProgramRegister(
|
|
|
|
|
in_program: in_param_ty!(Token),
|
|
|
|
|
in_scope: in_param_ty!(u32),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_program: Token, in_scope: u32| {
|
|
|
|
|
let mut pool = pull_writer!(PROGRAM_POOL)?;
|
|
|
|
|
let program = pool.get_mut(in_program)?;
|
|
|
|
|
let scope = resolve_enum!(Scope, in_scope)?;
|
|
|
|
|
program.register(scope.into())?;
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFProgramUnregister(
|
|
|
|
|
in_program: in_param_ty!(Token),
|
|
|
|
|
in_scope: in_param_ty!(u32),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_program: Token, in_scope: u32| {
|
|
|
|
|
let mut pool = pull_writer!(PROGRAM_POOL)?;
|
|
|
|
|
let program = pool.get_mut(in_program)?;
|
|
|
|
|
let scope = resolve_enum!(Scope, in_scope)?;
|
|
|
|
|
program.unregister(scope.into())?;
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFProgramIsRegistered(
|
|
|
|
|
in_program: in_param_ty!(Token),
|
|
|
|
|
in_scope: in_param_ty!(u32),
|
|
|
|
|
out_is_registered: out_param_ty!(bool),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_program: Token, in_scope: u32| -> (out_is_registered: bool) {
|
|
|
|
|
let mut pool = pull_writer!(PROGRAM_POOL)?;
|
|
|
|
|
let program = pool.get_mut(in_program)?;
|
|
|
|
|
let scope = resolve_enum!(Scope, in_scope)?;
|
|
|
|
|
Ok(program.is_registered(scope.into())?)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFProgramLinkExt(
|
|
|
|
|
in_program: in_param_ty!(Token),
|
|
|
|
|
in_scope: in_param_ty!(u32),
|
|
|
|
|
in_index: in_param_ty!(usize),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_program: Token, in_scope: u32, in_index: usize| {
|
|
|
|
|
let mut pool = pull_writer!(PROGRAM_POOL)?;
|
|
|
|
|
let program = pool.get_mut(in_program)?;
|
|
|
|
|
let scope = resolve_enum!(Scope, in_scope)?;
|
|
|
|
|
program.link_ext(scope.into(), in_index)?;
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[unsafe(no_mangle)]
|
|
|
|
|
pub extern "C" fn WFProgramUnlinkExt(
|
|
|
|
|
in_program: in_param_ty!(Token),
|
|
|
|
|
in_scope: in_param_ty!(u32),
|
|
|
|
|
in_index: in_param_ty!(usize),
|
|
|
|
|
) -> bool {
|
|
|
|
|
cffi_wrapper!(|in_program: Token, in_scope: u32, in_index: usize| {
|
|
|
|
|
let mut pool = pull_writer!(PROGRAM_POOL)?;
|
|
|
|
|
let program = pool.get_mut(in_program)?;
|
|
|
|
|
let scope = resolve_enum!(Scope, in_scope)?;
|
|
|
|
|
program.unlink_ext(scope.into(), in_index)?;
|
|
|
|
|
Ok(())
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-10 23:16:56 +08:00
|
|
|
// endregion
|
|
|
|
|
|
|
|
|
|
// endregion
|