adc.c 6.4 KB

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