mutex.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. #include <common/mutex.h>
  2. #include <mm/slab.h>
  3. #include <sched/sched.h>
  4. /**
  5. * @brief 初始化互斥量
  6. *
  7. * @param lock mutex结构体
  8. */
  9. void mutex_init(mutex_t *lock)
  10. {
  11. atomic_set(&lock->count, 1);
  12. spin_init(&lock->wait_lock);
  13. list_init(&lock->wait_list);
  14. }
  15. static void __mutex_sleep()
  16. {
  17. current_pcb->state = PROC_UNINTERRUPTIBLE;
  18. sched();
  19. }
  20. static void __mutex_acquire(mutex_t *lock)
  21. {
  22. }
  23. /**
  24. * @brief 对互斥量加锁
  25. *
  26. * @param lock mutex结构体
  27. */
  28. void mutex_lock(mutex_t *lock)
  29. {
  30. bool lock_ok = 0;
  31. while (lock_ok == false)
  32. {
  33. spin_lock(&lock->wait_lock);
  34. if (likely(mutex_is_locked(lock)))
  35. {
  36. struct mutex_waiter_t *waiter = (struct mutex_waiter_t *)kzalloc(sizeof(struct mutex_waiter_t), 0);
  37. if (waiter == NULL)
  38. {
  39. kerror("In mutex_lock: no memory to alloc waiter. Program's behaviour might be indetermined!");
  40. spin_unlock(&lock->wait_lock);
  41. return;
  42. }
  43. // memset(waiter, 0, sizeof(struct mutex_waiter_t));
  44. waiter->pcb = current_pcb;
  45. list_init(&waiter->list);
  46. list_append(&lock->wait_list, &waiter->list);
  47. spin_unlock(&lock->wait_lock);
  48. __mutex_sleep();
  49. }
  50. else
  51. {
  52. atomic_dec(&lock->count);
  53. spin_unlock(&lock->wait_lock);
  54. lock_ok = true;
  55. }
  56. }
  57. }
  58. /**
  59. * @brief 对互斥量解锁
  60. *
  61. * @param lock mutex结构体
  62. */
  63. void mutex_unlock(mutex_t *lock)
  64. {
  65. if (unlikely(!mutex_is_locked(lock)))
  66. return;
  67. spin_lock(&lock->wait_lock);
  68. struct mutex_waiter_t *wt = NULL;
  69. if (mutex_is_locked(lock))
  70. {
  71. if (!list_empty(&lock->wait_list))
  72. wt = container_of(list_next(&lock->wait_list), struct mutex_waiter_t, list);
  73. atomic_inc(&lock->count);
  74. if (wt != NULL)
  75. list_del(&wt->list);
  76. }
  77. spin_unlock(&lock->wait_lock);
  78. if (wt != NULL)
  79. {
  80. process_wakeup(wt->pcb);
  81. kfree(wt);
  82. }
  83. }
  84. /**
  85. * @brief 尝试对互斥量加锁
  86. *
  87. * @param lock mutex结构体
  88. *
  89. * @return 成功加锁->1, 加锁失败->0
  90. */
  91. int mutex_trylock(mutex_t *lock)
  92. {
  93. if (mutex_is_locked(lock))
  94. return 0;
  95. spin_lock(&lock->wait_lock);
  96. if (mutex_is_locked(lock))
  97. {
  98. spin_unlock(&lock->wait_lock);
  99. return 0;
  100. }
  101. else
  102. {
  103. atomic_dec(&lock->count);
  104. spin_unlock(&lock->wait_lock);
  105. return 1;
  106. }
  107. }