adc_shell.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. /*
  2. * Copyright (c) 2018 Prevas A/S
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <shell/shell.h>
  7. #include <stdlib.h>
  8. #include <drivers/adc.h>
  9. #include <ctype.h>
  10. #include <sys/util.h>
  11. #include <devicetree.h>
  12. #if DT_HAS_COMPAT_STATUS_OKAY(atmel_sam_afec)
  13. #define DT_DRV_COMPAT atmel_sam_afec
  14. #elif DT_HAS_COMPAT_STATUS_OKAY(atmel_sam0_adc)
  15. #define DT_DRV_COMPAT atmel_sam0_adc
  16. #elif DT_HAS_COMPAT_STATUS_OKAY(ite_it8xxx2_adc)
  17. #define DT_DRV_COMPAT ite_it8xxx2_adc
  18. #elif DT_HAS_COMPAT_STATUS_OKAY(microchip_xec_adc)
  19. #define DT_DRV_COMPAT microchip_xec_adc
  20. #elif DT_HAS_COMPAT_STATUS_OKAY(microchip_xec_adc_v2)
  21. #define DT_DRV_COMPAT microchip_xec_adc_v2
  22. #elif DT_HAS_COMPAT_STATUS_OKAY(nordic_nrf_adc)
  23. #define DT_DRV_COMPAT nordic_nrf_adc
  24. #elif DT_HAS_COMPAT_STATUS_OKAY(nordic_nrf_saadc)
  25. #define DT_DRV_COMPAT nordic_nrf_saadc
  26. #elif DT_HAS_COMPAT_STATUS_OKAY(nxp_kinetis_adc12)
  27. #define DT_DRV_COMPAT nxp_kinetis_adc12
  28. #elif DT_HAS_COMPAT_STATUS_OKAY(nxp_kinetis_adc16)
  29. #define DT_DRV_COMPAT nxp_kinetis_adc16
  30. #elif DT_HAS_COMPAT_STATUS_OKAY(st_stm32_adc)
  31. #define DT_DRV_COMPAT st_stm32_adc
  32. #elif DT_HAS_COMPAT_STATUS_OKAY(nuvoton_npcx_adc)
  33. #define DT_DRV_COMPAT nuvoton_npcx_adc
  34. #elif DT_HAS_COMPAT_STATUS_OKAY(ti_cc32xx_adc)
  35. #define DT_DRV_COMPAT ti_cc32xx_adc
  36. #elif DT_HAS_COMPAT_STATUS_OKAY(zephyr_adc_emul)
  37. #define DT_DRV_COMPAT zephyr_adc_emul
  38. #else
  39. #error No known devicetree compatible match for ADC shell
  40. #endif
  41. #define LOG_LEVEL CONFIG_LOG_DEFAULT_LEVEL
  42. #include <logging/log.h>
  43. LOG_MODULE_REGISTER(adc_shell);
  44. #define CMD_HELP_ACQ_TIME \
  45. "Configure acquisition time." \
  46. "\nUsage: acq_time <time> <unit>" \
  47. "\nunits: us, ns, ticks\n"
  48. #define CMD_HELP_CHANNEL \
  49. "Configure ADC channel\n" \
  50. #define CMD_HELP_CH_ID \
  51. "Configure channel id\n" \
  52. "Usage: id <channel_id>\n"
  53. #define CMD_HELP_CH_NEG \
  54. "Configure channel negative input\n" \
  55. "Usage: negative <negative_input_id>\n"
  56. #define CMD_HELP_CH_POS \
  57. "Configure channel positive input\n" \
  58. "Usage: positive <positive_input_id>\n"
  59. #define CMD_HELP_READ \
  60. "Read adc value\n" \
  61. "Usage: read <channel>\n"
  62. #define CMD_HELP_RES \
  63. "Configure resolution\n" \
  64. "Usage: resolution <resolution>\n"
  65. #define CMD_HELP_REF "Configure reference\n"
  66. #define CMD_HELP_GAIN "Configure gain.\n"
  67. #define CMD_HELP_PRINT "Print current configuration"
  68. #define NODE_LABELS(n) DT_INST_LABEL(n),
  69. #define ADC_HDL_LIST_ENTRY(label) \
  70. { \
  71. .device_label = label, \
  72. .channel_config = { \
  73. .gain = ADC_GAIN_1, \
  74. .reference = ADC_REF_INTERNAL, \
  75. .acquisition_time = ADC_ACQ_TIME_DEFAULT, \
  76. .channel_id = 0, \
  77. }, \
  78. .resolution = 0, \
  79. }
  80. #define INIT_MACRO() DT_INST_FOREACH_STATUS_OKAY(NODE_LABELS) "NA"
  81. #define CHOSEN_STR_LEN 20
  82. static char chosen_reference[CHOSEN_STR_LEN + 1] = "INTERNAL";
  83. static char chosen_gain[CHOSEN_STR_LEN + 1] = "1";
  84. /* This table size is = ADC devices count + 1 (NA). */
  85. static struct adc_hdl {
  86. char *device_label;
  87. struct adc_channel_cfg channel_config;
  88. uint8_t resolution;
  89. } adc_list[] = {
  90. FOR_EACH(ADC_HDL_LIST_ENTRY, (,), INIT_MACRO())
  91. };
  92. static struct adc_hdl *get_adc(const char *device_label)
  93. {
  94. for (int i = 0; i < ARRAY_SIZE(adc_list); i++) {
  95. if (!strcmp(device_label, adc_list[i].device_label)) {
  96. return &adc_list[i];
  97. }
  98. }
  99. /* This will never happen because ADC was prompted by shell */
  100. __ASSERT_NO_MSG(false);
  101. return NULL;
  102. }
  103. static int cmd_adc_ch_id(const struct shell *shell, size_t argc, char **argv)
  104. {
  105. /* -2: index of ADC label name */
  106. struct adc_hdl *adc = get_adc(argv[-2]);
  107. const struct device *adc_dev;
  108. int retval = 0;
  109. adc_dev = device_get_binding(adc->device_label);
  110. if (adc_dev == NULL) {
  111. shell_error(shell, "ADC device not found");
  112. return -ENODEV;
  113. }
  114. if (!isdigit((unsigned char)argv[1][0])) {
  115. shell_error(shell, "<channel> must be digits");
  116. return -EINVAL;
  117. }
  118. adc->channel_config.channel_id = (uint8_t)strtol(argv[1], NULL, 10);
  119. retval = adc_channel_setup(adc_dev, &adc->channel_config);
  120. LOG_DBG("Channel setup returned %i\n", retval);
  121. return retval;
  122. }
  123. static int cmd_adc_ch_neg(const struct shell *shell, size_t argc, char **argv)
  124. {
  125. #if CONFIG_ADC_CONFIGURABLE_INPUTS
  126. /* -2: index of ADC label name */
  127. struct adc_hdl *adc = get_adc(argv[-2]);
  128. const struct device *adc_dev;
  129. int retval = 0;
  130. adc_dev = device_get_binding(adc->device_label);
  131. if (adc_dev == NULL) {
  132. shell_error(shell, "ADC device not found");
  133. return -ENODEV;
  134. }
  135. if (!isdigit((unsigned char)argv[1][0])) {
  136. shell_error(shell, "<negative input> must be digits");
  137. return -EINVAL;
  138. }
  139. adc->channel_config.input_negative = (uint8_t)strtol(argv[1], NULL, 10);
  140. retval = adc_channel_setup(adc_dev, &adc->channel_config);
  141. LOG_DBG("Channel setup returned %i\n", retval);
  142. return retval;
  143. #else
  144. return -EINVAL;
  145. #endif
  146. }
  147. static int cmd_adc_ch_pos(const struct shell *shell, size_t argc, char **argv)
  148. {
  149. #if CONFIG_ADC_CONFIGURABLE_INPUTS
  150. /* -2: index of ADC label name */
  151. struct adc_hdl *adc = get_adc(argv[-2]);
  152. const struct device *adc_dev;
  153. int retval = 0;
  154. adc_dev = device_get_binding(adc->device_label);
  155. if (adc_dev == NULL) {
  156. shell_error(shell, "ADC device not found");
  157. return -ENODEV;
  158. }
  159. if (!isdigit((unsigned char)argv[1][0])) {
  160. shell_error(shell, "<positive input> must be digits");
  161. return -EINVAL;
  162. }
  163. adc->channel_config.input_positive = (uint8_t)strtol(argv[1], NULL, 10);
  164. retval = adc_channel_setup(adc_dev, &adc->channel_config);
  165. LOG_DBG("Channel setup returned %i\n", retval);
  166. return retval;
  167. #else
  168. return -EINVAL;
  169. #endif
  170. }
  171. static int cmd_adc_gain(const struct shell *shell, size_t argc, char **argv,
  172. void *data)
  173. {
  174. /* -2: index of ADC label name */
  175. struct adc_hdl *adc = get_adc(argv[-2]);
  176. enum adc_gain gain = (enum adc_gain)data;
  177. const struct device *adc_dev;
  178. int retval = -EINVAL;
  179. adc_dev = device_get_binding(adc->device_label);
  180. if (adc_dev == NULL) {
  181. shell_error(shell, "ADC device not found");
  182. return -ENODEV;
  183. }
  184. adc->channel_config.gain = gain;
  185. int len = strlen(argv[0]) > CHOSEN_STR_LEN ? CHOSEN_STR_LEN
  186. : strlen(argv[0]);
  187. memcpy(chosen_gain, argv[0], len);
  188. chosen_gain[len] = '\0';
  189. retval = adc_channel_setup(adc_dev, &adc->channel_config);
  190. LOG_DBG("Channel setup returned %i\n", retval);
  191. return retval;
  192. }
  193. static int cmd_adc_acq(const struct shell *shell, size_t argc, char **argv)
  194. {
  195. /* -1 index of ADC label name */
  196. struct adc_hdl *adc = get_adc(argv[-1]);
  197. const struct device *adc_dev;
  198. uint16_t acq_time;
  199. int retval;
  200. adc_dev = device_get_binding(adc->device_label);
  201. if (adc_dev == NULL) {
  202. shell_error(shell, "ADC device not found");
  203. return -ENODEV;
  204. }
  205. if (!isdigit((unsigned char)argv[1][0])) {
  206. shell_error(shell, "<time> must be digits");
  207. return -EINVAL;
  208. }
  209. acq_time = (uint16_t)strtol(argv[1], NULL, 10);
  210. if (!strcmp(argv[2], "us")) {
  211. adc->channel_config.acquisition_time =
  212. ADC_ACQ_TIME(ADC_ACQ_TIME_MICROSECONDS, acq_time);
  213. } else if (!strcmp(argv[2], "ns")) {
  214. adc->channel_config.acquisition_time =
  215. ADC_ACQ_TIME(ADC_ACQ_TIME_NANOSECONDS, acq_time);
  216. } else if (!strcmp(argv[2], "ticks")) {
  217. adc->channel_config.acquisition_time =
  218. ADC_ACQ_TIME(ADC_ACQ_TIME_TICKS, acq_time);
  219. } else {
  220. adc->channel_config.acquisition_time =
  221. ADC_ACQ_TIME_DEFAULT;
  222. }
  223. retval = adc_channel_setup(adc_dev, &adc->channel_config);
  224. LOG_DBG("Channel setup returned %i\n", retval);
  225. return retval;
  226. }
  227. static int cmd_adc_reso(const struct shell *shell, size_t argc, char **argv)
  228. {
  229. /* -1 index of ADC label name */
  230. struct adc_hdl *adc = get_adc(argv[-1]);
  231. const struct device *adc_dev;
  232. int retval;
  233. adc_dev = device_get_binding(adc->device_label);
  234. if (adc_dev == NULL) {
  235. shell_error(shell, "ADC device not found");
  236. return -ENODEV;
  237. }
  238. if (!isdigit((unsigned char)argv[1][0])) {
  239. shell_error(shell, "<resolution> must be digits");
  240. return -EINVAL;
  241. }
  242. adc->resolution = (uint8_t)strtol(argv[1], NULL, 10);
  243. retval = adc_channel_setup(adc_dev, &adc->channel_config);
  244. return retval;
  245. }
  246. static int cmd_adc_ref(const struct shell *shell, size_t argc, char **argv,
  247. void *data)
  248. {
  249. /* -2 index of ADC label name */
  250. struct adc_hdl *adc = get_adc(argv[-2]);
  251. enum adc_reference reference = (enum adc_reference)data;
  252. const struct device *adc_dev;
  253. int retval = -EINVAL;
  254. adc_dev = device_get_binding(adc->device_label);
  255. if (adc_dev == NULL) {
  256. shell_error(shell, "ADC device not found");
  257. return -ENODEV;
  258. }
  259. int len = strlen(argv[0]) > CHOSEN_STR_LEN ? CHOSEN_STR_LEN
  260. : strlen(argv[0]);
  261. memcpy(chosen_reference, argv[0], len);
  262. chosen_reference[len] = '\0';
  263. adc->channel_config.reference = reference;
  264. retval = adc_channel_setup(adc_dev, &adc->channel_config);
  265. LOG_DBG("Channel setup returned %i\n", retval);
  266. return retval;
  267. }
  268. #define BUFFER_SIZE 1
  269. static int cmd_adc_read(const struct shell *shell, size_t argc, char **argv)
  270. {
  271. uint8_t adc_channel_id = strtol(argv[1], NULL, 10);
  272. /* -1 index of adc label name */
  273. struct adc_hdl *adc = get_adc(argv[-1]);
  274. uint16_t m_sample_buffer[BUFFER_SIZE];
  275. const struct device *adc_dev;
  276. int retval;
  277. adc_dev = device_get_binding(adc->device_label);
  278. if (adc_dev == NULL) {
  279. shell_error(shell, "adc device not found");
  280. return -ENODEV;
  281. }
  282. adc->channel_config.channel_id = adc_channel_id;
  283. const struct adc_sequence sequence = {
  284. .channels = BIT(adc->channel_config.channel_id),
  285. .buffer = m_sample_buffer,
  286. .buffer_size = sizeof(m_sample_buffer),
  287. .resolution = adc->resolution,
  288. };
  289. retval = adc_read(adc_dev, &sequence);
  290. if (retval >= 0) {
  291. shell_print(shell, "read: %i", m_sample_buffer[0]);
  292. }
  293. return retval;
  294. }
  295. static int cmd_adc_print(const struct shell *shell, size_t argc, char **argv)
  296. {
  297. /* -1 index of ADC label name */
  298. struct adc_hdl *adc = get_adc(argv[-1]);
  299. shell_print(shell, "%s:\n"
  300. "Gain: %s\n"
  301. "Reference: %s\n"
  302. "Acquisition Time: %u\n"
  303. "Channel ID: %u\n"
  304. "Resolution: %u",
  305. adc->device_label,
  306. chosen_gain,
  307. chosen_reference,
  308. adc->channel_config.acquisition_time,
  309. adc->channel_config.channel_id,
  310. adc->resolution);
  311. return 0;
  312. }
  313. SHELL_SUBCMD_DICT_SET_CREATE(sub_ref_cmds, cmd_adc_ref,
  314. (VDD_1, ADC_REF_VDD_1),
  315. (VDD_1_2, ADC_REF_VDD_1_2),
  316. (VDD_1_3, ADC_REF_VDD_1_3),
  317. (VDD_1_4, ADC_REF_VDD_1_4),
  318. (INTERNAL, ADC_REF_INTERNAL),
  319. (EXTERNAL_0, ADC_REF_EXTERNAL0),
  320. (EXTERNAL_1, ADC_REF_EXTERNAL1)
  321. );
  322. SHELL_SUBCMD_DICT_SET_CREATE(sub_gain_cmds, cmd_adc_gain,
  323. (GAIN_1_6, ADC_GAIN_1_6),
  324. (GAIN_1_5, ADC_GAIN_1_5),
  325. (GAIN_1_4, ADC_GAIN_1_4),
  326. (GAIN_1_3, ADC_GAIN_1_3),
  327. (GAIN_1_2, ADC_GAIN_1_2),
  328. (GAIN_2_3, ADC_GAIN_2_3),
  329. (GAIN_1, ADC_GAIN_1),
  330. (GAIN_2, ADC_GAIN_2),
  331. (GAIN_3, ADC_GAIN_3),
  332. (GAIN_4, ADC_GAIN_4),
  333. (GAIN_8, ADC_GAIN_8),
  334. (GAIN_16, ADC_GAIN_16),
  335. (GAIN_32, ADC_GAIN_32),
  336. (GAIN_64, ADC_GAIN_64)
  337. );
  338. SHELL_STATIC_SUBCMD_SET_CREATE(sub_channel_cmds,
  339. SHELL_CMD_ARG(id, NULL, CMD_HELP_CH_ID, cmd_adc_ch_id, 2, 0),
  340. SHELL_COND_CMD_ARG(CONFIG_ADC_CONFIGURABLE_INPUTS,
  341. negative, NULL, CMD_HELP_CH_NEG, cmd_adc_ch_neg, 2, 0),
  342. SHELL_COND_CMD_ARG(CONFIG_ADC_CONFIGURABLE_INPUTS,
  343. positive, NULL, CMD_HELP_CH_POS, cmd_adc_ch_pos, 2, 0),
  344. SHELL_SUBCMD_SET_END
  345. );
  346. SHELL_STATIC_SUBCMD_SET_CREATE(sub_adc_cmds,
  347. /* Alphabetically sorted. */
  348. SHELL_CMD_ARG(acq_time, NULL, CMD_HELP_ACQ_TIME, cmd_adc_acq, 3, 0),
  349. SHELL_CMD_ARG(channel, &sub_channel_cmds, CMD_HELP_CHANNEL, NULL, 3, 0),
  350. SHELL_CMD(gain, &sub_gain_cmds, CMD_HELP_GAIN, NULL),
  351. SHELL_CMD_ARG(print, NULL, CMD_HELP_PRINT, cmd_adc_print, 1, 0),
  352. SHELL_CMD_ARG(read, NULL, CMD_HELP_READ, cmd_adc_read, 2, 0),
  353. SHELL_CMD(reference, &sub_ref_cmds, CMD_HELP_REF, NULL),
  354. SHELL_CMD_ARG(resolution, NULL, CMD_HELP_RES, cmd_adc_reso, 2, 0),
  355. SHELL_SUBCMD_SET_END /* Array terminated. */
  356. );
  357. static void cmd_adc_dev_get(size_t idx, struct shell_static_entry *entry)
  358. {
  359. /* -1 because the last element in the list is a "list terminator" */
  360. if (idx < ARRAY_SIZE(adc_list) - 1) {
  361. entry->syntax = adc_list[idx].device_label;
  362. entry->handler = NULL;
  363. entry->subcmd = &sub_adc_cmds;
  364. entry->help = "Select subcommand for ADC property label.\n";
  365. } else {
  366. entry->syntax = NULL;
  367. }
  368. }
  369. SHELL_DYNAMIC_CMD_CREATE(sub_adc_dev, cmd_adc_dev_get);
  370. SHELL_CMD_REGISTER(adc, &sub_adc_dev, "ADC commands", NULL);