adc_context.h 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. * Copyright (c) 2017 Intel Corporation
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. #ifndef ZEPHYR_DRIVERS_ADC_ADC_CONTEXT_H_
  8. #define ZEPHYR_DRIVERS_ADC_ADC_CONTEXT_H_
  9. #include <drivers/adc.h>
  10. #include <sys/atomic.h>
  11. #ifdef __cplusplus
  12. extern "C" {
  13. #endif
  14. struct adc_context;
  15. /*
  16. * Each driver should provide implementations of the following two functions:
  17. * - adc_context_start_sampling() that will be called when a sampling (of one
  18. * or more channels, depending on the realized sequence) is to be started
  19. * - adc_context_update_buffer_pointer() that will be called when the sample
  20. * buffer pointer should be prepared for writing of next sampling results,
  21. * the "repeat_sampling" parameter indicates if the results should be written
  22. * in the same place as before (when true) or as consecutive ones (otherwise).
  23. */
  24. static void adc_context_start_sampling(struct adc_context *ctx);
  25. static void adc_context_update_buffer_pointer(struct adc_context *ctx,
  26. bool repeat_sampling);
  27. /*
  28. * If a given driver uses some dedicated hardware timer to trigger consecutive
  29. * samplings, it should implement also the following two functions. Otherwise,
  30. * it should define the ADC_CONTEXT_USES_KERNEL_TIMER macro to enable parts of
  31. * this module that utilize a standard kernel timer.
  32. */
  33. static void adc_context_enable_timer(struct adc_context *ctx);
  34. static void adc_context_disable_timer(struct adc_context *ctx);
  35. struct adc_context {
  36. atomic_t sampling_requested;
  37. #ifdef ADC_CONTEXT_USES_KERNEL_TIMER
  38. struct k_timer timer;
  39. #endif /* ADC_CONTEXT_USES_KERNEL_TIMER */
  40. struct k_sem lock;
  41. struct k_sem sync;
  42. int status;
  43. #ifdef CONFIG_ADC_ASYNC
  44. struct k_poll_signal *signal;
  45. bool asynchronous;
  46. #endif /* CONFIG_ADC_ASYNC */
  47. struct adc_sequence sequence;
  48. struct adc_sequence_options options;
  49. uint16_t sampling_index;
  50. };
  51. #ifdef ADC_CONTEXT_USES_KERNEL_TIMER
  52. #define ADC_CONTEXT_INIT_TIMER(_data, _ctx_name) \
  53. ._ctx_name.timer = Z_TIMER_INITIALIZER(_data._ctx_name.timer, \
  54. adc_context_on_timer_expired, \
  55. NULL)
  56. #endif /* ADC_CONTEXT_USES_KERNEL_TIMER */
  57. #define ADC_CONTEXT_INIT_LOCK(_data, _ctx_name) \
  58. ._ctx_name.lock = Z_SEM_INITIALIZER(_data._ctx_name.lock, 0, 1)
  59. #define ADC_CONTEXT_INIT_SYNC(_data, _ctx_name) \
  60. ._ctx_name.sync = Z_SEM_INITIALIZER(_data._ctx_name.sync, 0, 1)
  61. static inline void adc_context_request_next_sampling(struct adc_context *ctx)
  62. {
  63. if (atomic_inc(&ctx->sampling_requested) == 0) {
  64. adc_context_start_sampling(ctx);
  65. } else {
  66. /*
  67. * If a sampling was already requested and was not finished yet,
  68. * do not start another one from here, this will be done from
  69. * adc_context_on_sampling_done() after the current sampling is
  70. * complete. Instead, note this fact, and inform the user about
  71. * it after the sequence is done.
  72. */
  73. ctx->status = -EBUSY;
  74. }
  75. }
  76. #ifdef ADC_CONTEXT_USES_KERNEL_TIMER
  77. static inline void adc_context_enable_timer(struct adc_context *ctx)
  78. {
  79. k_timer_start(&ctx->timer, K_NO_WAIT, K_USEC(ctx->options.interval_us));
  80. }
  81. static inline void adc_context_disable_timer(struct adc_context *ctx)
  82. {
  83. k_timer_stop(&ctx->timer);
  84. }
  85. static void adc_context_on_timer_expired(struct k_timer *timer_id)
  86. {
  87. struct adc_context *ctx =
  88. CONTAINER_OF(timer_id, struct adc_context, timer);
  89. adc_context_request_next_sampling(ctx);
  90. }
  91. #endif /* ADC_CONTEXT_USES_KERNEL_TIMER */
  92. static inline void adc_context_lock(struct adc_context *ctx,
  93. bool asynchronous,
  94. struct k_poll_signal *signal)
  95. {
  96. k_sem_take(&ctx->lock, K_FOREVER);
  97. #ifdef CONFIG_ADC_ASYNC
  98. ctx->asynchronous = asynchronous;
  99. ctx->signal = signal;
  100. #endif /* CONFIG_ADC_ASYNC */
  101. }
  102. static inline void adc_context_release(struct adc_context *ctx, int status)
  103. {
  104. #ifdef CONFIG_ADC_ASYNC
  105. if (ctx->asynchronous && (status == 0)) {
  106. return;
  107. }
  108. #endif /* CONFIG_ADC_ASYNC */
  109. k_sem_give(&ctx->lock);
  110. }
  111. static inline void adc_context_unlock_unconditionally(struct adc_context *ctx)
  112. {
  113. if (!k_sem_count_get(&ctx->lock)) {
  114. k_sem_give(&ctx->lock);
  115. }
  116. }
  117. static inline int adc_context_wait_for_completion(struct adc_context *ctx)
  118. {
  119. #ifdef CONFIG_ADC_ASYNC
  120. if (ctx->asynchronous) {
  121. return 0;
  122. }
  123. #endif /* CONFIG_ADC_ASYNC */
  124. k_sem_take(&ctx->sync, K_FOREVER);
  125. return ctx->status;
  126. }
  127. static inline void adc_context_complete(struct adc_context *ctx, int status)
  128. {
  129. #ifdef CONFIG_ADC_ASYNC
  130. if (ctx->asynchronous) {
  131. if (ctx->signal) {
  132. k_poll_signal_raise(ctx->signal, status);
  133. }
  134. k_sem_give(&ctx->lock);
  135. return;
  136. }
  137. #endif /* CONFIG_ADC_ASYNC */
  138. /*
  139. * Override the status only when an error is signaled to this function.
  140. * Please note that adc_context_request_next_sampling() might have set
  141. * this field.
  142. */
  143. if (status != 0) {
  144. ctx->status = status;
  145. }
  146. k_sem_give(&ctx->sync);
  147. }
  148. static inline void adc_context_start_read(struct adc_context *ctx,
  149. const struct adc_sequence *sequence)
  150. {
  151. ctx->sequence = *sequence;
  152. ctx->status = 0;
  153. if (sequence->options) {
  154. ctx->options = *sequence->options;
  155. ctx->sequence.options = &ctx->options;
  156. ctx->sampling_index = 0U;
  157. if (ctx->options.interval_us != 0U) {
  158. atomic_set(&ctx->sampling_requested, 0);
  159. adc_context_enable_timer(ctx);
  160. return;
  161. }
  162. }
  163. adc_context_start_sampling(ctx);
  164. }
  165. /*
  166. * This function should be called after a sampling (of one or more channels,
  167. * depending on the realized sequence) is done. It calls the defined callback
  168. * function if required and takes further actions accordingly.
  169. */
  170. static inline void adc_context_on_sampling_done(struct adc_context *ctx,
  171. const struct device *dev)
  172. {
  173. if (ctx->sequence.options) {
  174. adc_sequence_callback callback = ctx->options.callback;
  175. enum adc_action action;
  176. bool finish = false;
  177. bool repeat = false;
  178. if (callback) {
  179. action = callback(dev,
  180. &ctx->sequence,
  181. ctx->sampling_index);
  182. } else {
  183. action = ADC_ACTION_CONTINUE;
  184. }
  185. switch (action) {
  186. case ADC_ACTION_REPEAT:
  187. repeat = true;
  188. break;
  189. case ADC_ACTION_FINISH:
  190. finish = true;
  191. break;
  192. default: /* ADC_ACTION_CONTINUE */
  193. if (ctx->sampling_index <
  194. ctx->options.extra_samplings) {
  195. ++ctx->sampling_index;
  196. } else {
  197. finish = true;
  198. }
  199. }
  200. if (!finish) {
  201. adc_context_update_buffer_pointer(ctx, repeat);
  202. /*
  203. * Immediately start the next sampling if working with
  204. * a zero interval or if the timer expired again while
  205. * the current sampling was in progress.
  206. */
  207. if (ctx->options.interval_us == 0U) {
  208. adc_context_start_sampling(ctx);
  209. } else if (atomic_dec(&ctx->sampling_requested) > 1) {
  210. adc_context_start_sampling(ctx);
  211. }
  212. return;
  213. }
  214. if (ctx->options.interval_us != 0U) {
  215. adc_context_disable_timer(ctx);
  216. }
  217. }
  218. adc_context_complete(ctx, 0);
  219. }
  220. #ifdef __cplusplus
  221. }
  222. #endif
  223. #endif /* ZEPHYR_DRIVERS_ADC_ADC_CONTEXT_H_ */