futex.c 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. /*
  2. * Copyright (c) 2019 Intel corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <kernel.h>
  7. #include <kernel_structs.h>
  8. #include <spinlock.h>
  9. #include <kswap.h>
  10. #include <syscall_handler.h>
  11. #include <init.h>
  12. #include <ksched.h>
  13. static struct z_futex_data *k_futex_find_data(struct k_futex *futex)
  14. {
  15. struct z_object *obj;
  16. obj = z_object_find(futex);
  17. if (obj == NULL || obj->type != K_OBJ_FUTEX) {
  18. return NULL;
  19. }
  20. return obj->data.futex_data;
  21. }
  22. int z_impl_k_futex_wake(struct k_futex *futex, bool wake_all)
  23. {
  24. k_spinlock_key_t key;
  25. unsigned int woken = 0U;
  26. struct k_thread *thread;
  27. struct z_futex_data *futex_data;
  28. futex_data = k_futex_find_data(futex);
  29. if (futex_data == NULL) {
  30. return -EINVAL;
  31. }
  32. key = k_spin_lock(&futex_data->lock);
  33. do {
  34. thread = z_unpend_first_thread(&futex_data->wait_q);
  35. if (thread != NULL) {
  36. woken++;
  37. arch_thread_return_value_set(thread, 0);
  38. z_ready_thread(thread);
  39. }
  40. } while (thread && wake_all);
  41. z_reschedule(&futex_data->lock, key);
  42. return woken;
  43. }
  44. static inline int z_vrfy_k_futex_wake(struct k_futex *futex, bool wake_all)
  45. {
  46. if (Z_SYSCALL_MEMORY_WRITE(futex, sizeof(struct k_futex)) != 0) {
  47. return -EACCES;
  48. }
  49. return z_impl_k_futex_wake(futex, wake_all);
  50. }
  51. #include <syscalls/k_futex_wake_mrsh.c>
  52. int z_impl_k_futex_wait(struct k_futex *futex, int expected,
  53. k_timeout_t timeout)
  54. {
  55. int ret;
  56. k_spinlock_key_t key;
  57. struct z_futex_data *futex_data;
  58. futex_data = k_futex_find_data(futex);
  59. if (futex_data == NULL) {
  60. return -EINVAL;
  61. }
  62. if (atomic_get(&futex->val) != (atomic_val_t)expected) {
  63. return -EAGAIN;
  64. }
  65. key = k_spin_lock(&futex_data->lock);
  66. ret = z_pend_curr(&futex_data->lock,
  67. key, &futex_data->wait_q, timeout);
  68. if (ret == -EAGAIN) {
  69. ret = -ETIMEDOUT;
  70. }
  71. return ret;
  72. }
  73. static inline int z_vrfy_k_futex_wait(struct k_futex *futex, int expected,
  74. k_timeout_t timeout)
  75. {
  76. if (Z_SYSCALL_MEMORY_WRITE(futex, sizeof(struct k_futex)) != 0) {
  77. return -EACCES;
  78. }
  79. return z_impl_k_futex_wait(futex, expected, timeout);
  80. }
  81. #include <syscalls/k_futex_wait_mrsh.c>