wait_queue_head.c 2.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182
  1. #include <common/wait_queue_head.h>
  2. #include <process/process.h>
  3. #include <sched/sched.h>
  4. /**
  5. * @brief 初始化等待队列
  6. *
  7. * @param wait_queue 等待队列
  8. * @param pcb pcb
  9. */
  10. void wait_queue_head_init(wait_queue_head_t *wait_queue)
  11. {
  12. list_init(&wait_queue->wait_list);
  13. spin_init(&wait_queue->lock);
  14. }
  15. /**
  16. * @brief 在等待队列上进行等待, 但是你需要确保wait已经被init, 同时wakeup只能使用wake_up_on_stack函数。
  17. *
  18. * @param wait_queue_head 队列头指针
  19. */
  20. void wait_queue_sleep_with_node(wait_queue_head_t *q, wait_queue_node_t *wait)
  21. {
  22. BUG_ON(wait->pcb == NULL);
  23. wait->pcb->state = PROC_UNINTERRUPTIBLE;
  24. list_append(&q->wait_list, &wait->wait_list);
  25. sched();
  26. }
  27. /**
  28. * @brief 在等待队列上进行等待,同时释放自旋锁, 但是你需要确保wait已经被init, 同时wakeup只能使用wake_up_on_stack函数。
  29. *
  30. * @param wait_queue_head 队列头指针
  31. */
  32. void wait_queue_sleep_with_node_unlock(wait_queue_head_t *q, wait_queue_node_t *wait, void *lock)
  33. {
  34. BUG_ON(wait->pcb == NULL);
  35. wait->pcb->state = PROC_UNINTERRUPTIBLE;
  36. list_append(&q->wait_list, &wait->wait_list);
  37. spin_unlock((spinlock_t *)lock);
  38. sched();
  39. }
  40. /**
  41. * @brief 在等待队列上进行等待(允许中断), 但是你需要确保wait已经被init, 同时wakeup只能使用wake_up_on_stack函数。
  42. *
  43. * @param wait_queue_head 队列头指针
  44. */
  45. void wait_queue_sleep_with_node_interriptible(wait_queue_head_t *q, wait_queue_node_t *wait)
  46. {
  47. BUG_ON(wait->pcb == NULL);
  48. wait->pcb->state = PROC_INTERRUPTIBLE;
  49. list_append(&q->wait_list, &wait->wait_list);
  50. sched();
  51. }
  52. /**
  53. * @brief 唤醒在等待队列的头部的进程, 但是不会free掉这个节点的空间(默认这个节点在栈上创建)
  54. *
  55. * @param wait_queue_head
  56. * @param state
  57. */
  58. void wait_queue_wakeup_on_stack(wait_queue_head_t *q, int64_t state)
  59. {
  60. if (list_empty(&q->wait_list))
  61. return;
  62. wait_queue_node_t *wait = container_of(list_next(&q->wait_list), wait_queue_node_t, wait_list);
  63. // 符合唤醒条件
  64. if (wait->pcb->state & state)
  65. {
  66. list_del_init(&wait->wait_list);
  67. process_wakeup(wait->pcb);
  68. }
  69. }