timer.rs 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. use core::sync::atomic::{compiler_fence, AtomicBool, Ordering};
  2. use alloc::{
  3. boxed::Box,
  4. collections::LinkedList,
  5. sync::{Arc, Weak},
  6. };
  7. use crate::{
  8. arch::{
  9. asm::current::current_pcb,
  10. interrupt::{cli, sti},
  11. sched::sched,
  12. },
  13. exception::softirq::{softirq_vectors, SoftirqNumber, SoftirqVec},
  14. include::bindings::bindings::{process_control_block, process_wakeup, pt_regs, PROC_RUNNING},
  15. kdebug, kerror,
  16. libs::spinlock::SpinLock,
  17. syscall::SystemError,
  18. };
  19. const MAX_TIMEOUT: i64 = i64::MAX;
  20. const TIMER_RUN_CYCLE_THRESHOLD: usize = 20;
  21. static mut TIMER_JIFFIES: u64 = 0;
  22. lazy_static! {
  23. pub static ref TIMER_LIST: SpinLock<LinkedList<Arc<Timer>>> = SpinLock::new(LinkedList::new());
  24. }
  25. /// 定时器要执行的函数的特征
  26. pub trait TimerFunction: Send + Sync {
  27. fn run(&mut self);
  28. }
  29. /// WakeUpHelper函数对应的结构体
  30. pub struct WakeUpHelper {
  31. pcb: &'static mut process_control_block,
  32. }
  33. impl WakeUpHelper {
  34. pub fn new(pcb: &'static mut process_control_block) -> Box<WakeUpHelper> {
  35. return Box::new(WakeUpHelper { pcb });
  36. }
  37. }
  38. impl TimerFunction for WakeUpHelper {
  39. fn run(&mut self) {
  40. unsafe {
  41. process_wakeup(self.pcb);
  42. }
  43. }
  44. }
  45. pub struct Timer(SpinLock<InnerTimer>);
  46. impl Timer {
  47. /// @brief 创建一个定时器(单位:ms)
  48. ///
  49. /// @param timer_func 定时器需要执行的函数对应的结构体
  50. ///
  51. /// @param expire_jiffies 定时器结束时刻
  52. ///
  53. /// @return 定时器结构体
  54. pub fn new(timer_func: Box<dyn TimerFunction>, expire_jiffies: u64) -> Arc<Self> {
  55. let result: Arc<Timer> = Arc::new(Timer(SpinLock::new(InnerTimer {
  56. expire_jiffies,
  57. timer_func,
  58. self_ref: Weak::default(),
  59. })));
  60. result.0.lock().self_ref = Arc::downgrade(&result);
  61. return result;
  62. }
  63. /// @brief 将定时器插入到定时器链表中
  64. pub fn activate(&self) {
  65. let inner_guard = self.0.lock();
  66. let timer_list = &mut TIMER_LIST.lock();
  67. // 链表为空,则直接插入
  68. if timer_list.is_empty() {
  69. // FIXME push_timer
  70. timer_list.push_back(inner_guard.self_ref.upgrade().unwrap());
  71. drop(inner_guard);
  72. drop(timer_list);
  73. compiler_fence(Ordering::SeqCst);
  74. return;
  75. }
  76. let mut split_pos: usize = 0;
  77. for (pos, elt) in timer_list.iter().enumerate() {
  78. if elt.0.lock().expire_jiffies > inner_guard.expire_jiffies {
  79. split_pos = pos;
  80. break;
  81. }
  82. }
  83. let mut temp_list: LinkedList<Arc<Timer>> = timer_list.split_off(split_pos);
  84. timer_list.push_back(inner_guard.self_ref.upgrade().unwrap());
  85. timer_list.append(&mut temp_list);
  86. drop(inner_guard);
  87. drop(timer_list);
  88. }
  89. #[inline]
  90. fn run(&self) {
  91. self.0.lock().timer_func.run();
  92. }
  93. }
  94. /// 定时器类型
  95. pub struct InnerTimer {
  96. /// 定时器结束时刻
  97. pub expire_jiffies: u64,
  98. /// 定时器需要执行的函数结构体
  99. pub timer_func: Box<dyn TimerFunction>,
  100. /// self_ref
  101. self_ref: Weak<Timer>,
  102. }
  103. #[derive(Debug)]
  104. pub struct DoTimerSoftirq {
  105. running: AtomicBool,
  106. }
  107. impl DoTimerSoftirq {
  108. pub fn new() -> Self {
  109. return DoTimerSoftirq {
  110. running: AtomicBool::new(false),
  111. };
  112. }
  113. fn set_run(&self) -> bool {
  114. let x = self
  115. .running
  116. .compare_exchange(false, true, Ordering::Acquire, Ordering::Relaxed);
  117. if x.is_ok() {
  118. return true;
  119. } else {
  120. return false;
  121. }
  122. }
  123. fn clear_run(&self) {
  124. self.running.store(false, Ordering::Release);
  125. }
  126. }
  127. impl SoftirqVec for DoTimerSoftirq {
  128. fn run(&self) {
  129. if self.set_run() == false {
  130. return;
  131. }
  132. // 最多只处理TIMER_RUN_CYCLE_THRESHOLD个计时器
  133. for _ in 0..TIMER_RUN_CYCLE_THRESHOLD {
  134. // kdebug!("DoTimerSoftirq run");
  135. let timer_list = TIMER_LIST.try_lock();
  136. if timer_list.is_err() {
  137. continue;
  138. }
  139. let mut timer_list = timer_list.unwrap();
  140. if timer_list.is_empty() {
  141. break;
  142. }
  143. let timer_list_front = timer_list.pop_front().unwrap();
  144. // kdebug!("to lock timer_list_front");
  145. let mut timer_list_front_guard = None;
  146. for _ in 0..10 {
  147. let x = timer_list_front.0.try_lock();
  148. if x.is_err() {
  149. continue;
  150. }
  151. timer_list_front_guard = Some(x.unwrap());
  152. }
  153. if timer_list_front_guard.is_none() {
  154. continue;
  155. }
  156. let timer_list_front_guard = timer_list_front_guard.unwrap();
  157. if timer_list_front_guard.expire_jiffies > unsafe { TIMER_JIFFIES as u64 } {
  158. drop(timer_list_front_guard);
  159. timer_list.push_front(timer_list_front);
  160. break;
  161. }
  162. drop(timer_list_front_guard);
  163. drop(timer_list);
  164. timer_list_front.run();
  165. }
  166. self.clear_run();
  167. }
  168. }
  169. /// @brief 初始化timer模块
  170. pub fn timer_init() {
  171. // FIXME 调用register_trap
  172. let do_timer_softirq = Arc::new(DoTimerSoftirq::new());
  173. softirq_vectors()
  174. .register_softirq(SoftirqNumber::TIMER, do_timer_softirq)
  175. .expect("Failed to register timer softirq");
  176. kdebug!("timer initiated successfully");
  177. }
  178. /// 计算接下来n毫秒对应的定时器时间片
  179. pub fn next_n_ms_timer_jiffies(expire_ms: u64) -> u64 {
  180. return unsafe { TIMER_JIFFIES as u64 } + 1000 * (expire_ms);
  181. }
  182. /// 计算接下来n微秒对应的定时器时间片
  183. pub fn next_n_us_timer_jiffies(expire_us: u64) -> u64 {
  184. return unsafe { TIMER_JIFFIES as u64 } + (expire_us);
  185. }
  186. /// @brief 让pcb休眠timeout个jiffies
  187. ///
  188. /// @param timeout 需要休眠的时间(单位:jiffies)
  189. ///
  190. /// @return Ok(i64) 剩余需要休眠的时间(单位:jiffies)
  191. ///
  192. /// @return Err(SystemError) 错误码
  193. pub fn schedule_timeout(mut timeout: i64) -> Result<i64, SystemError> {
  194. // kdebug!("schedule_timeout");
  195. if timeout == MAX_TIMEOUT {
  196. sched();
  197. return Ok(MAX_TIMEOUT);
  198. } else if timeout < 0 {
  199. kerror!("timeout can't less than 0");
  200. return Err(SystemError::EINVAL);
  201. } else {
  202. // 禁用中断,防止在这段期间发生调度,造成死锁
  203. cli();
  204. timeout += unsafe { TIMER_JIFFIES } as i64;
  205. let timer = Timer::new(WakeUpHelper::new(current_pcb()), timeout as u64);
  206. timer.activate();
  207. current_pcb().state &= (!PROC_RUNNING) as u64;
  208. sti();
  209. sched();
  210. let time_remaining: i64 = timeout - unsafe { TIMER_JIFFIES } as i64;
  211. if time_remaining >= 0 {
  212. // 被提前唤醒,返回剩余时间
  213. return Ok(time_remaining);
  214. } else {
  215. return Ok(0);
  216. }
  217. }
  218. }
  219. pub fn timer_get_first_expire() -> Result<u64, SystemError> {
  220. // FIXME
  221. // kdebug!("rs_timer_get_first_expire,timer_jif = {:?}", TIMER_JIFFIES);
  222. for _ in 0..10 {
  223. match TIMER_LIST.try_lock() {
  224. Ok(timer_list) => {
  225. // kdebug!("rs_timer_get_first_expire TIMER_LIST lock successfully");
  226. if timer_list.is_empty() {
  227. // kdebug!("timer_list is empty");
  228. return Ok(0);
  229. } else {
  230. // kdebug!("timer_list not empty");
  231. return Ok(timer_list.front().unwrap().0.lock().expire_jiffies);
  232. }
  233. }
  234. // 加锁失败返回啥??
  235. Err(_) => continue,
  236. }
  237. }
  238. return Err(SystemError::EAGAIN_OR_EWOULDBLOCK);
  239. }
  240. pub fn update_timer_jiffies(add_jiffies: u64) -> u64 {
  241. unsafe { TIMER_JIFFIES += add_jiffies };
  242. return unsafe { TIMER_JIFFIES };
  243. }
  244. pub fn clock() -> u64 {
  245. return unsafe { TIMER_JIFFIES };
  246. }
  247. // ====== 重构完成后请删掉extern C ======
  248. #[no_mangle]
  249. pub extern "C" fn rs_clock() -> u64 {
  250. clock()
  251. }
  252. #[no_mangle]
  253. pub extern "C" fn sys_clock(_regs: *const pt_regs) -> u64 {
  254. clock()
  255. }
  256. // ====== 以下为给C提供的接口 ======
  257. #[no_mangle]
  258. pub extern "C" fn rs_schedule_timeout(timeout: i64) -> i64 {
  259. match schedule_timeout(timeout) {
  260. Ok(v) => {
  261. return v;
  262. }
  263. Err(e) => {
  264. kdebug!("rs_schedule_timeout run failed");
  265. return e.to_posix_errno() as i64;
  266. }
  267. }
  268. }
  269. #[no_mangle]
  270. pub extern "C" fn rs_timer_init() {
  271. timer_init();
  272. }
  273. #[no_mangle]
  274. pub extern "C" fn rs_timer_next_n_ms_jiffies(expire_ms: u64) -> u64 {
  275. return next_n_ms_timer_jiffies(expire_ms);
  276. }
  277. #[no_mangle]
  278. pub extern "C" fn rs_timer_next_n_us_jiffies(expire_us: u64) -> u64 {
  279. return next_n_us_timer_jiffies(expire_us);
  280. }
  281. #[no_mangle]
  282. pub extern "C" fn rs_timer_get_first_expire() -> i64 {
  283. match timer_get_first_expire() {
  284. Ok(v) => return v as i64,
  285. Err(_) => return 0,
  286. }
  287. }
  288. #[no_mangle]
  289. pub extern "C" fn rs_update_timer_jiffies(add_jiffies: u64) -> u64 {
  290. return update_timer_jiffies(add_jiffies);
  291. }