use core::sync::atomic::{Ordering, fence};
use ax_errno::{AxError, AxResult};
use ax_task::current;
use linux_raw_sys::general::membarrier_cmd;
use crate::task::AsThread;
const MEMBARRIER_CMD_QUERY: i32 = membarrier_cmd::MEMBARRIER_CMD_QUERY as i32;
const MEMBARRIER_CMD_GLOBAL: i32 = membarrier_cmd::MEMBARRIER_CMD_GLOBAL as i32;
const MEMBARRIER_CMD_GLOBAL_EXPEDITED: i32 = membarrier_cmd::MEMBARRIER_CMD_GLOBAL_EXPEDITED as i32;
const MEMBARRIER_CMD_REGISTER_GLOBAL_EXPEDITED: i32 =
membarrier_cmd::MEMBARRIER_CMD_REGISTER_GLOBAL_EXPEDITED as i32;
const MEMBARRIER_CMD_PRIVATE_EXPEDITED: i32 =
membarrier_cmd::MEMBARRIER_CMD_PRIVATE_EXPEDITED as i32;
const MEMBARRIER_CMD_REGISTER_PRIVATE_EXPEDITED: i32 =
membarrier_cmd::MEMBARRIER_CMD_REGISTER_PRIVATE_EXPEDITED as i32;
const MEMBARRIER_STATE_PRIVATE_EXPEDITED: u32 = MEMBARRIER_CMD_PRIVATE_EXPEDITED as u32;
const MEMBARRIER_STATE_GLOBAL_EXPEDITED: u32 = MEMBARRIER_CMD_GLOBAL_EXPEDITED as u32;
const SUPPORTED_COMMANDS: i32 = MEMBARRIER_CMD_GLOBAL
| MEMBARRIER_CMD_GLOBAL_EXPEDITED
| MEMBARRIER_CMD_REGISTER_GLOBAL_EXPEDITED
| MEMBARRIER_CMD_PRIVATE_EXPEDITED
| MEMBARRIER_CMD_REGISTER_PRIVATE_EXPEDITED;
fn smp_mb() {
fence(Ordering::SeqCst);
}
pub fn sys_membarrier(cmd: i32, flags: u32, _cpu_id: i32) -> AxResult<isize> {
if flags != 0 {
return Err(AxError::InvalidInput);
}
match cmd {
MEMBARRIER_CMD_QUERY => Ok(SUPPORTED_COMMANDS as isize),
MEMBARRIER_CMD_GLOBAL => {
smp_mb();
Ok(0)
}
MEMBARRIER_CMD_REGISTER_GLOBAL_EXPEDITED => {
current()
.as_thread()
.proc_data
.register_membarrier_state(MEMBARRIER_STATE_GLOBAL_EXPEDITED);
Ok(0)
}
MEMBARRIER_CMD_GLOBAL_EXPEDITED => {
let proc_data = current().as_thread().proc_data.clone();
if proc_data.membarrier_state() & MEMBARRIER_STATE_GLOBAL_EXPEDITED == 0 {
return Err(AxError::OperationNotPermitted);
}
smp_mb();
Ok(0)
}
MEMBARRIER_CMD_REGISTER_PRIVATE_EXPEDITED => {
current()
.as_thread()
.proc_data
.register_membarrier_state(MEMBARRIER_STATE_PRIVATE_EXPEDITED);
Ok(0)
}
MEMBARRIER_CMD_PRIVATE_EXPEDITED => {
let proc_data = current().as_thread().proc_data.clone();
if proc_data.membarrier_state() & MEMBARRIER_STATE_PRIVATE_EXPEDITED == 0 {
return Err(AxError::OperationNotPermitted);
}
smp_mb();
Ok(0)
}
_ => Err(AxError::InvalidInput),
}
}