notify.c 1.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. /*
  2. * Copyright (c) 2019 Peter Bigot Consulting, LLC
  3. * Copyright (c) 2020 Nordic Semiconductor ASA
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. #include <kernel.h>
  8. #include <sys/notify.h>
  9. int sys_notify_validate(struct sys_notify *notify)
  10. {
  11. int rv = 0;
  12. if (notify == NULL) {
  13. return -EINVAL;
  14. }
  15. /* Validate configuration based on mode */
  16. switch (sys_notify_get_method(notify)) {
  17. case SYS_NOTIFY_METHOD_SPINWAIT:
  18. break;
  19. case SYS_NOTIFY_METHOD_CALLBACK:
  20. if (notify->method.callback == NULL) {
  21. rv = -EINVAL;
  22. }
  23. break;
  24. #ifdef CONFIG_POLL
  25. case SYS_NOTIFY_METHOD_SIGNAL:
  26. if (notify->method.signal == NULL) {
  27. rv = -EINVAL;
  28. }
  29. break;
  30. #endif /* CONFIG_POLL */
  31. default:
  32. rv = -EINVAL;
  33. break;
  34. }
  35. /* Clear the result here instead of in all callers. */
  36. if (rv == 0) {
  37. notify->result = 0;
  38. }
  39. return rv;
  40. }
  41. sys_notify_generic_callback sys_notify_finalize(struct sys_notify *notify,
  42. int res)
  43. {
  44. struct k_poll_signal *sig = NULL;
  45. sys_notify_generic_callback rv = NULL;
  46. uint32_t method = sys_notify_get_method(notify);
  47. /* Store the result and capture secondary notification
  48. * information.
  49. */
  50. notify->result = res;
  51. switch (method) {
  52. case SYS_NOTIFY_METHOD_SPINWAIT:
  53. break;
  54. case SYS_NOTIFY_METHOD_CALLBACK:
  55. rv = notify->method.callback;
  56. break;
  57. case SYS_NOTIFY_METHOD_SIGNAL:
  58. sig = notify->method.signal;
  59. break;
  60. default:
  61. __ASSERT_NO_MSG(false);
  62. }
  63. /* Mark completion by clearing the flags field to the
  64. * completed state, releasing any spin-waiters, then complete
  65. * secondary notification.
  66. */
  67. compiler_barrier();
  68. notify->flags = SYS_NOTIFY_METHOD_COMPLETED;
  69. if (IS_ENABLED(CONFIG_POLL) && (sig != NULL)) {
  70. k_poll_signal_raise(sig, res);
  71. }
  72. return rv;
  73. }