adc.c 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. #include "bsp/adc.h"
  2. #include "libs/utils.h"
  3. #include "os/co_task.h"
  4. static void _gpio_init(void);
  5. static void _adc0_init(void);
  6. static void _adc0_insert_chan_init(void);
  7. static void _adc0_regular_chan_init(void);
  8. static void _adc1_init(void);
  9. static void _adc1_insert_chan_init(void);
  10. static void _adc1_regular_chan_init(void);
  11. /*
  12. ADC0 inserted 采集母线电流
  13. ADC1 inserted 采集三相电流
  14. 每次同时发送母线电流和响应的相电流,通过母线电流给相电流采集的MOS内阻校准
  15. 可以理解为FOC工作在三电阻采样模式下,只是需要不停的校准MOS的内阻
  16. */
  17. void adc_init(void){
  18. /* init adc input gpio */
  19. _gpio_init();
  20. /* config ADC clock */
  21. rcu_adc_clock_config(RCU_CKADC_CKAPB2_DIV4); //APB2 clk 120M, adc clk 30M
  22. _adc0_init();
  23. _adc1_init();
  24. adc_config_trigger(ADC_TRIGGER_PHASE);
  25. nvic_irq_enable(ADC0_1_IRQn, ADC_IRQ_PRIORITY, 0);
  26. adc_disable_ext_trigger();
  27. }
  28. static void _gpio_init(void) {
  29. rcu_periph_clock_enable(RCU_GPIOA);
  30. rcu_periph_clock_enable(RCU_GPIOB);
  31. gpio_init(GPIOA, GPIO_MODE_AIN, GPIO_OSPEED_50MHZ, GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_2|GPIO_PIN_3|GPIO_PIN_4|GPIO_PIN_5|GPIO_PIN_6|GPIO_PIN_7);
  32. gpio_init(GPIOB, GPIO_MODE_AIN, GPIO_OSPEED_50MHZ, GPIO_PIN_0|GPIO_PIN_1);
  33. }
  34. static void _adc0_init(void) {
  35. u32 adc_dev = ADC0;
  36. /* enable ADC1 clock */
  37. rcu_periph_clock_enable(RCU_ADC0);
  38. /* ADC mode config,adc0 master, adc1 slave */
  39. adc_mode_config(ADC_DAUL_INSERTED_PARALLEL);
  40. /* ADC special function config */
  41. adc_special_function_config(adc_dev, ADC_SCAN_MODE, ENABLE);
  42. adc_special_function_config(adc_dev, ADC_CONTINUOUS_MODE, DISABLE);
  43. /* ADC data alignment config */
  44. adc_data_alignment_config(adc_dev, ADC_DATAALIGN_RIGHT);
  45. /* init insert chans*/
  46. _adc0_insert_chan_init();
  47. /* init regular chans*/
  48. _adc0_regular_chan_init();
  49. adc_interrupt_disable(adc_dev, ADC_INT_EOIC);
  50. adc_interrupt_disable(adc_dev, ADC_INT_EOC);
  51. adc_interrupt_disable(adc_dev, ADC_INT_WDE);
  52. /* enable ADC interface */
  53. adc_enable(adc_dev);
  54. delay_ms(1);
  55. /* ADC calibration and reset calibration */
  56. adc_calibration_enable(adc_dev);
  57. }
  58. static void _adc1_init(void) {
  59. u32 adc_dev = ADC1;
  60. /* enable ADC1 clock */
  61. rcu_periph_clock_enable(RCU_ADC1);
  62. /* ADC mode config,adc0 master, adc1 slave */
  63. adc_mode_config(ADC_DAUL_INSERTED_PARALLEL);
  64. /* ADC special function config */
  65. adc_special_function_config(adc_dev, ADC_SCAN_MODE, ENABLE);
  66. adc_special_function_config(adc_dev, ADC_CONTINUOUS_MODE, DISABLE);
  67. /* ADC data alignment config */
  68. adc_data_alignment_config(adc_dev, ADC_DATAALIGN_RIGHT);
  69. /* init insert chans*/
  70. _adc1_insert_chan_init();
  71. /* init regular chans*/
  72. _adc1_regular_chan_init();
  73. adc_interrupt_disable(adc_dev, ADC_INT_EOIC);
  74. adc_interrupt_disable(adc_dev, ADC_INT_EOC);
  75. adc_interrupt_disable(adc_dev, ADC_INT_WDE);
  76. /* enable ADC interface */
  77. adc_enable(adc_dev);
  78. delay_ms(1);
  79. /* ADC calibration and reset calibration */
  80. adc_calibration_enable(adc_dev);
  81. }
  82. /* ADC0 insert chan sample phase I(use two chan, selected by foc) */
  83. static void _adc0_insert_chan_init(void) {
  84. u32 adc_dev = ADC0;
  85. adc_discontinuous_mode_config(adc_dev, ADC_CHANNEL_DISCON_DISABLE, 0);
  86. /* ADC channel length config */
  87. adc_channel_length_config(adc_dev, ADC_INSERTED_CHANNEL, 1);
  88. /* ADC inserted channel ran config, use ISQ2,ISQ3 */
  89. adc_update_insert_sample_rank(adc_dev, U_PHASE_I_CHAN);
  90. /* config inserted channel sample time */
  91. adc_update_insert_sample_time(adc_dev, U_PHASE_I_CHAN, ADC_SAMPLE_TIME);
  92. adc_update_insert_sample_time(adc_dev, V_PHASE_I_CHAN, ADC_SAMPLE_TIME);
  93. adc_update_insert_sample_time(adc_dev, W_PHASE_I_CHAN, ADC_SAMPLE_TIME);
  94. /* ADC trigger config */
  95. adc_external_trigger_source_config(adc_dev, ADC_INSERTED_CHANNEL, ADC0_1_EXTTRIG_INSERTED_T0_CH3);
  96. /* ADC external trigger enable */
  97. adc_external_trigger_config(adc_dev, ADC_INSERTED_CHANNEL, ENABLE);
  98. }
  99. /* ADC1 insert chan sample vbus I */
  100. static void _adc1_insert_chan_init(void) {
  101. u32 adc_dev = ADC1;
  102. //adc_discontinuous_mode_config(adc_dev, ADC_CHANNEL_DISCON_DISABLE, 0);
  103. /* ADC channel length config */
  104. adc_channel_length_config(adc_dev, ADC_INSERTED_CHANNEL, 1);
  105. /* ADC inserted channel config */
  106. adc_update_insert_sample_rank(adc_dev, V_PHASE_I_CHAN);
  107. /* config inserted channel sample time */
  108. adc_update_insert_sample_time(adc_dev, U_PHASE_I_CHAN, ADC_SAMPLE_TIME);
  109. adc_update_insert_sample_time(adc_dev, V_PHASE_I_CHAN, ADC_SAMPLE_TIME);
  110. adc_update_insert_sample_time(adc_dev, W_PHASE_I_CHAN, ADC_SAMPLE_TIME);
  111. /* ADC trigger config, slave must config to software trigger */
  112. adc_external_trigger_source_config(adc_dev, ADC_INSERTED_CHANNEL, ADC0_1_2_EXTTRIG_REGULAR_NONE);
  113. /* ADC external trigger enable */
  114. adc_external_trigger_config(adc_dev, ADC_INSERTED_CHANNEL, ENABLE);
  115. }
  116. static void _adc0_regular_chan_init(void) {
  117. //adc_discontinuous_mode_config(ADC0, ADC_REGULAR_CHANNEL, 1); //每次转化一个
  118. /* ADC channel length config */
  119. adc_channel_length_config(ADC0, ADC_REGULAR_CHANNEL, 1);
  120. adc_external_trigger_config(ADC0, ADC_REGULAR_CHANNEL, ENABLE);
  121. adc_external_trigger_source_config(ADC0, ADC_REGULAR_CHANNEL, ADC0_1_2_EXTTRIG_REGULAR_NONE);
  122. adc_regular_channel_config(ADC0, 0, MOTOR_TEMP_CHAN, ADC_SAMPLETIME_55POINT5);
  123. //adc_regular_channel_config(ADC0, 1, HANDLERBAR_CHAN, ADC_SAMPLETIME_55POINT5);
  124. //adc_regular_channel_config(ADC0, 2, VBUS_V_CHAN, ADC_SAMPLETIME_55POINT5);
  125. }
  126. static void _adc1_regular_chan_init(void) {
  127. adc_discontinuous_mode_config(ADC1, ADC_REGULAR_CHANNEL, 1); //每次转化一个
  128. /* ADC channel length config */
  129. adc_channel_length_config(ADC1, ADC_REGULAR_CHANNEL, 1);
  130. adc_external_trigger_config(ADC1, ADC_REGULAR_CHANNEL, ENABLE);
  131. adc_external_trigger_source_config(ADC1, ADC_REGULAR_CHANNEL, ADC0_1_2_EXTTRIG_REGULAR_NONE);
  132. adc_regular_channel_config(ADC1, 0, W_PHASE_V_CHAN, ADC_SAMPLETIME_55POINT5);
  133. //adc_regular_channel_config(ADC1, 1, V_PHASE_V_CHAN, ADC_SAMPLETIME_55POINT5);
  134. //adc_regular_channel_config(ADC1, 2, U_PHASE_V_CHAN, ADC_SAMPLETIME_55POINT5);
  135. }
  136. void adc_start_insert_convert(void) {
  137. /* clear the ADC flag */
  138. adc_interrupt_flag_clear(ADC0, ADC_INT_FLAG_EOC);
  139. adc_interrupt_flag_clear(ADC0, ADC_INT_FLAG_EOIC);
  140. adc_interrupt_flag_clear(ADC1, ADC_INT_FLAG_EOC);
  141. adc_interrupt_flag_clear(ADC1, ADC_INT_FLAG_EOIC);
  142. /* enable ADC interrupt */
  143. adc_interrupt_enable(ADC0, ADC_INT_EOIC);
  144. adc_interrupt_enable(ADC1, ADC_INT_EOIC);
  145. }
  146. s32 adc_sample_regular_channel(int channel, int times) {
  147. u32 adc_device = ADC0;
  148. if (channel >= W_PHASE_V_CHAN && channel <= U_PHASE_V_CHAN) {
  149. adc_device = ADC1;
  150. }
  151. int value = 0;
  152. int count = 0;
  153. int min = 0xFFFFF;
  154. int max = -0xFFFFF;
  155. u64 start_time;
  156. adc_regular_channel_config(adc_device, 0, channel, ADC_SAMPLETIME_55POINT5);
  157. while(count < times){
  158. restart:
  159. start_time = co_task_sys64_ticks();
  160. adc_software_trigger_enable(adc_device, ADC_REGULAR_CHANNEL);
  161. while(SET != adc_flag_get(adc_device, ADC_FLAG_EOC)){
  162. if (co_task_sys64_ticks() - start_time >= 2){
  163. goto restart;
  164. }
  165. };
  166. int one = adc_regular_data_read(adc_device);
  167. adc_flag_clear(adc_device, ADC_FLAG_EOC);
  168. value += (one & 0xFFF);
  169. count ++;
  170. if (one > max){
  171. max = one;
  172. }
  173. if (one < min) {
  174. min = one;
  175. }
  176. }
  177. if (times <= 2) {
  178. return value/times;
  179. }
  180. return (value - min - max)/(times-2);
  181. }