drv_adc.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. /*
  2. * Copyright (c) 2006-2024, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2022-05-16 shelton first version
  9. * 2024-07-21 liujianhua added mcxa153
  10. *
  11. */
  12. #include <rtconfig.h>
  13. #include <rtdevice.h>
  14. #include "fsl_adc12.h"
  15. #include "fsl_pmc.h"
  16. #define DBG_TAG "drv.adc"
  17. #define DBG_LVL DBG_INFO
  18. #include <rtdbg.h>
  19. #ifdef RT_USING_ADC
  20. #define ADC_VBG_CH 27
  21. #define ADC_VBG_VOLT 1000
  22. struct mcx_adc
  23. {
  24. struct rt_adc_device adc_device;
  25. ADC_Type *adc_base;
  26. clock_ip_name_t clock_ip_name;
  27. clock_ip_src_t clock_ip_src;
  28. uint8_t resolution_bits;
  29. uint8_t max_channels;
  30. uint32_t vref;
  31. char *name;
  32. };
  33. static struct mcx_adc mcx_adc_obj[] =
  34. {
  35. #ifdef BSP_USING_ADC0
  36. {
  37. .adc_base = ADC0,
  38. .clock_ip_name = kCLOCK_Adc0,
  39. .clock_ip_src = kCLOCK_IpSrcSysOscAsync,
  40. .resolution_bits = 12,
  41. .max_channels = 32,
  42. .name = "adc0",
  43. },
  44. #endif
  45. #ifdef BSP_USING_ADC1
  46. {
  47. .adc_base = ADC1,
  48. .clock_ip_name = kCLOCK_Adc1,
  49. .clock_ip_src = kCLOCK_IpSrcSysOscAsync,
  50. .resolution_bits = 12,
  51. .max_channels = 32,
  52. .name = "adc1",
  53. },
  54. #endif
  55. };
  56. static uint16_t mcx_adc_get_raw(struct mcx_adc *adc, uint32_t channel)
  57. {
  58. adc12_channel_config_t chnl_cfg = {0};
  59. chnl_cfg.channelNumber = channel;
  60. chnl_cfg.enableInterruptOnConversionCompleted = false;
  61. ADC12_SetChannelConfig(adc->adc_base, 0, &chnl_cfg);
  62. while ((ADC12_GetChannelStatusFlags(adc->adc_base, 0) & kADC12_ChannelConversionCompletedFlag) == 0U)
  63. {
  64. }
  65. return ADC12_GetChannelConversionValue(adc->adc_base, 0);
  66. }
  67. static rt_err_t mcx_adc_set_enabled(struct rt_adc_device *device, rt_int8_t channel, rt_bool_t enabled)
  68. {
  69. RT_ASSERT(device != RT_NULL);
  70. struct mcx_adc *adc = device->parent.user_data;
  71. /* ADC is enabled by global probe. */
  72. RT_UNUSED(adc);
  73. return RT_EOK;
  74. }
  75. static rt_int16_t mcx_get_vref(struct rt_adc_device *device)
  76. {
  77. RT_ASSERT(device != RT_NULL);
  78. struct mcx_adc *adc = device->parent.user_data;
  79. return (int16_t)adc->vref;
  80. }
  81. static rt_err_t mcx_adc_get_value(struct rt_adc_device *device, rt_int8_t channel, rt_uint32_t *value)
  82. {
  83. RT_ASSERT(device != RT_NULL);
  84. struct mcx_adc *adc = device->parent.user_data;
  85. *value = mcx_adc_get_raw(adc, channel);
  86. return RT_EOK;
  87. }
  88. static rt_uint8_t mcx_adc_get_resolution(struct rt_adc_device *device)
  89. {
  90. RT_ASSERT(device != RT_NULL);
  91. struct mcx_adc *adc = device->parent.user_data;
  92. return adc->resolution_bits;
  93. }
  94. static const struct rt_adc_ops mcx_adc_ops =
  95. {
  96. .get_resolution = mcx_adc_get_resolution,
  97. .enabled = mcx_adc_set_enabled,
  98. .convert = mcx_adc_get_value,
  99. .get_vref = mcx_get_vref,
  100. };
  101. static int rt_hw_adc_init(void)
  102. {
  103. int result = RT_EOK;
  104. int i = 0;
  105. for (i = 0; i < sizeof(mcx_adc_obj) / sizeof(mcx_adc_obj[0]); i++)
  106. {
  107. struct mcx_adc *adc = &mcx_adc_obj[i];
  108. CLOCK_SetIpSrc(adc->clock_ip_name, adc->clock_ip_src);
  109. adc12_config_t cfg;
  110. ADC12_GetDefaultConfig(&cfg);
  111. cfg.clockSource = kADC12_ClockSourceAlt0; /* Only available selection. */
  112. cfg.resolution = kADC12_Resolution12Bit;
  113. cfg.sampleClockCount = 64;
  114. ADC12_Init(adc->adc_base, &cfg);
  115. ADC12_EnableHardwareTrigger(adc->adc_base, false);
  116. if (ADC12_DoAutoCalibration(adc->adc_base) != kStatus_Success)
  117. {
  118. return -RT_ERROR;
  119. }
  120. uint32_t vref_raw = mcx_adc_get_raw(adc, ADC_VBG_CH);
  121. adc->vref = (1U << adc->resolution_bits) * ADC_VBG_VOLT / vref_raw;
  122. if (rt_hw_adc_register(&mcx_adc_obj[i].adc_device, mcx_adc_obj[i].name, &mcx_adc_ops, &mcx_adc_obj[i]) != RT_EOK)
  123. {
  124. return -RT_ERROR;
  125. }
  126. }
  127. return result;
  128. }
  129. INIT_BOARD_EXPORT(rt_hw_adc_init);
  130. #endif /* BSP_USING_ADC */