xwrust/xwos/cthd.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290
//! XWOS RUST:当前线程
//! ========
//!
//! 此模块中的方法,只会对代码所运行的线程起作用。命名中的 **c** 是 **current** 的意思。
//!
//!
//! # 获取自身的线程描述符
//!
//! 线程可通过方法 [`i()`] 获取自身的线程描述符。
//!
//! # 通知调度器重新调度线程
//!
//! 线程可通过方法 [`yield_now()`] 主动让出CPU的使用权。
//!
//! # 判断当前线程是否可被冻结
//!
//! 线程可通过方法 [`shld_frz()`] 判断是否需要冻结,如果是,可调用 [`freeze()`] 冻结自身。
//!
//! 线程的冻结只会发生在一些特殊的条件下:
//!
//! + 系统开始低功耗流程;
//! + 线程在CPU之间迁移。
//!
//! # 判断当前线程是否可退出
//!
//! 线程可通过方法 [`shld_stop()`] 判断是否被设置了 **退出状态** ,**退出状态** 是其他线程通过 [`DThdHandle::stop()`] 方法设置的。
//!
//! # 线程睡眠
//!
//! ## 睡眠
//!
//! 线程可以通过方法 [`sleep()`] 指定需要睡眠多长时间,线程唤醒的时间可能会略微超过指定的时长。
//!
//! ## 睡眠到指定的时间点
//!
//! 当需要到某个时间点唤醒时,可以通过 [`sleep_to()`] 方法睡眠。此方法不关心睡眠多长时间,只关心到哪个时间唤醒。
//!
//! ## 从一个时间起点睡眠到另一个时间点
//!
//! 方法 [`sleep_from()`] 可以指定从某个时间点开始睡眠,持续多长时间。
//!
//! # 冻结线程
//!
//! 当线程通过方法 [`shld_frz()`] 判断出需要冻结时,可通过方法 [`freeze()`] 冻结自身。冻结之前需要释放掉所有资源。
//!
//!
//! [`DThdHandle::stop()`]: crate::xwos::thd::DThdHandle::stop
extern crate core;
use core::ffi::*;
use core::ptr;
use crate::types::*;
use super::thd::ThdD;
extern "C" {
fn xwrustffi_cthd_self(thd: *mut *mut c_void, tik: *mut XwSq);
fn xwrustffi_cthd_yield();
fn xwrustffi_cthd_shld_frz() -> bool;
fn xwrustffi_cthd_shld_stop() -> bool;
fn xwrustffi_cthd_sleep(time: XwTm) -> XwEr;
fn xwrustffi_cthd_sleep_to(to: XwTm) -> XwEr;
fn xwrustffi_cthd_sleep_from(origin: *mut XwTm,
inc: XwTm) -> XwEr;
fn xwrustffi_cthd_freeze() -> XwEr;
}
/// 获取当前线程的对象描述符
///
/// # 上下文
///
/// + 线程
///
/// # 示例
///
/// ```rust
/// use use xwrust::xwos::cthd;
///
/// let thdd = cthd::i();
/// ```
pub fn i() -> ThdD {
let mut thd: *mut c_void = ptr::null_mut();
let mut tik: XwSq = 0;
unsafe {
xwrustffi_cthd_self(&mut thd, &mut tik);
}
ThdD {thd, tik}
}
/// 通知调度器重新调度线程
///
/// # 上下文
///
/// + 线程
///
/// # 示例
///
/// ```rust
/// use use xwrust::xwos::cthd;
///
/// cthd::yield_now();
/// ```
pub fn yield_now() {
unsafe { xwrustffi_cthd_yield(); }
}
/// 判断当前线程是否可被冻结
///
/// # 上下文
///
/// + 线程
///
/// # 示例
///
/// ```rust
/// use use xwrust::xwos::cthd;
///
/// if cthd::shld_frz() {
/// cthd::freeze();
/// }
/// ```
pub fn shld_frz() -> bool {
unsafe { xwrustffi_cthd_shld_frz() }
}
/// 判断当前线程是否可退出
///
/// # 上下文
///
/// + 线程
///
/// # 示例
///
/// ```rust
/// use use xwrust::xwos::cthd;
///
/// if cthd::shld_stop() {
/// return result;
/// }
/// ```
pub fn shld_stop() -> bool {
unsafe { xwrustffi_cthd_shld_stop() }
}
/// 线程睡眠一段时间
///
/// 调用此方法的线程会睡眠 **dur** 后被唤醒。
///
/// 此方法等价于 `cthd::sleep_to(xwtm::ft(xwtm::s(dur)))` 。
///
/// # 参数说明
///
/// + dur: 期望睡眠的时间
///
/// # 返回值
///
/// 0: 睡眠成功
/// -EINTR: 睡眠被中断
///
/// # 上下文
///
/// + 线程
///
/// # 示例
///
/// ```rust
/// use use xwrust::xwos::cthd;
/// use use xwrust::xwtx;
///
/// let rc = cthd::sleep(xwtm::s(1)); // 睡眠1s
/// ```
pub fn sleep(dur: XwTm) -> XwEr {
unsafe { xwrustffi_cthd_sleep(dur) }
}
/// 线程睡眠到一个时间点
///
/// 调用此方法的线程会睡眠到时间点 **to** 后被唤醒。
/// 如果 **to** 是过去的时间点,将直接返回 `-ETIMEDOUT` 。
///
/// # 参数说明
///
/// + to: 期望唤醒的时间点
///
/// # 返回值
///
/// + 0: 睡眠成功
/// + -EINTR: 睡眠被中断
/// + -ETIMEDOUT: 时间点是过去
///
/// # 上下文
///
/// + 线程
///
/// # 示例
///
/// ```rust
/// use use xwrust::xwos::cthd;
/// use use xwrust::xwtx;
///
/// let rc = cthd::sleep_to(xwtm::ft(xwtm::s(1))); // 睡眠1s
/// ```
pub fn sleep_to(to: XwTm) -> XwEr {
unsafe { xwrustffi_cthd_sleep_to(to) }
}
/// 线程从一个时间起点睡眠到另一个时间点
///
/// 调用此方法前,需要先确定一个时间起点 `from` ,可以通过 `xwtm::now()` 获取当前的系统时间作为起点。
/// 唤醒时间为 `from` + `dur` 。
/// 此方法会将线程唤醒时的系统时间返回,可作为下一次调用的时间起点。
/// 由此,循环调用,可以形成周期为 `dur` ,更为精确的周期性睡眠唤醒。
///
/// # 参数说明
///
/// + from: 时间起点
/// + dur: 期望被唤醒的时间增量
///
/// # 返回值 (rc, from)
///
/// + rc
/// + 0: 睡眠成功
/// + -EINTR: 睡眠被中断
/// + -ETIMEDOUT: 时间点是过去
/// + from: 线程唤醒时的系统时间
///
/// # 上下文
///
/// + 线程
///
/// # 示例
///
/// ```rust
/// use use xwrust::xwos::cthd;
/// use use xwrust::xwtx;
///
/// let mut from = xwtm::now();
/// loop {
/// let mut (rc, from) = cthd::sleep_from(from, xwtm::s(1)); // 从from开始睡眠1s
/// }
/// ```
pub fn sleep_from(from: XwTm, dur: XwTm) -> (XwEr, XwTm) {
let mut origin: XwTm = from;
unsafe {
let rc = xwrustffi_cthd_sleep_from(&mut origin, dur);
(rc, origin)
}
}
/// 冻结当前线程
///
/// 线程并不能随时冻结,必须满足下列条件之一:
///
/// + 系统已经开始准备进入低功耗;
/// + 线程正准备开始迁移。
///
/// 线程可通过 [`shld_frz()`] 判断是否满足冻结条件。
///
/// 通常冻结操作之前,还需要对资源进行释放,以防止线程阻碍系统进入低功耗;
/// 通常解冻操作之后,还需要重新获取资源。
///
/// # 返回值
///
/// + 0: 睡眠成功
/// + -EPERM: 当前不需要冻结线程
///
/// # 上下文
///
/// + 线程
///
/// # 示例
///
/// ```rust
/// use use xwrust::xwos::cthd;
///
/// if cthd::shld_frz() {
/// let rc = cthd::freeze();
/// }
/// ```
pub fn freeze() -> XwEr {
unsafe { xwrustffi_cthd_freeze() }
}