adc.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. /* Copyright Statement:
  2. *
  3. * This software/firmware and related documentation ("AutoChips Software") are
  4. * protected under relevant copyright laws. The information contained herein is
  5. * confidential and proprietary to AutoChips Inc. and/or its licensors. Without
  6. * the prior written permission of AutoChips inc. and/or its licensors, any
  7. * reproduction, modification, use or disclosure of AutoChips Software, and
  8. * information contained herein, in whole or in part, shall be strictly
  9. * prohibited.
  10. *
  11. * AutoChips Inc. (C) 2016. All rights reserved.
  12. *
  13. * BY OPENING THIS FILE, RECEIVER HEREBY UNEQUIVOCALLY ACKNOWLEDGES AND AGREES
  14. * THAT THE SOFTWARE/FIRMWARE AND ITS DOCUMENTATIONS ("AUTOCHIPS SOFTWARE")
  15. * RECEIVED FROM AUTOCHIPS AND/OR ITS REPRESENTATIVES ARE PROVIDED TO RECEIVER
  16. * ON AN "AS-IS" BASIS ONLY. AUTOCHIPS EXPRESSLY DISCLAIMS ANY AND ALL
  17. * WARRANTIES, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE IMPLIED
  18. * WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE OR
  19. * NONINFRINGEMENT. NEITHER DOES AUTOCHIPS PROVIDE ANY WARRANTY WHATSOEVER WITH
  20. * RESPECT TO THE SOFTWARE OF ANY THIRD PARTY WHICH MAY BE USED BY,
  21. * INCORPORATED IN, OR SUPPLIED WITH THE AUTOCHIPS SOFTWARE, AND RECEIVER AGREES
  22. * TO LOOK ONLY TO SUCH THIRD PARTY FOR ANY WARRANTY CLAIM RELATING THERETO.
  23. * RECEIVER EXPRESSLY ACKNOWLEDGES THAT IT IS RECEIVER'S SOLE RESPONSIBILITY TO
  24. * OBTAIN FROM ANY THIRD PARTY ALL PROPER LICENSES CONTAINED IN AUTOCHIPS
  25. * SOFTWARE. AUTOCHIPS SHALL ALSO NOT BE RESPONSIBLE FOR ANY AUTOCHIPS SOFTWARE
  26. * RELEASES MADE TO RECEIVER'S SPECIFICATION OR TO CONFORM TO A PARTICULAR
  27. * STANDARD OR OPEN FORUM. RECEIVER'S SOLE AND EXCLUSIVE REMEDY AND AUTOCHIPS'S
  28. * ENTIRE AND CUMULATIVE LIABILITY WITH RESPECT TO THE AUTOCHIPS SOFTWARE
  29. * RELEASED HEREUNDER WILL BE, AT AUTOCHIPS'S OPTION, TO REVISE OR REPLACE THE
  30. * AUTOCHIPS SOFTWARE AT ISSUE, OR REFUND ANY SOFTWARE LICENSE FEES OR SERVICE
  31. * CHARGE PAID BY RECEIVER TO AUTOCHIPS FOR SUCH AUTOCHIPS SOFTWARE AT ISSUE.
  32. */
  33. /**********<Incldue>**********/
  34. #include "adc.h"
  35. #include "timer.h"
  36. #include "math.h"
  37. /*********<Variable>********/
  38. uint8_t g_dmaFinish = 0;//DMA传输完成
  39. uint8_t g_halfDmaFinish = 0; //DMA传输半完成
  40. uint8_t g_dmaTransError = 0; //DMA传输错误
  41. uint16_t __align(2) g_ADCValueBuffer[ADC_SAMPLE_CHANNEL*ADC_FILTER_NUM] = {0};
  42. uint32_t g_timer0Cnt = 0;
  43. //uint32_t g_averageSampleValue = 0;
  44. /**
  45. * ADC_DMACallback
  46. *
  47. * @param[in] wpara:dmaChannelStatus
  48. * lpara:0
  49. * @return 0
  50. *
  51. * @brief ADC DMA中断回调函数。
  52. */
  53. void ADC_DMACallback(void *device, uint32_t wpara, uint32_t lpara)
  54. {
  55. /*
  56. wparam为DMA通道状态,状态含义可参考CHANNELx_STATUS寄存器,
  57. CHANNELx_STATUS[2] 传输错误
  58. CHANNELx_STATUS[1] 半传输完成(相对设置的transferNum,如果半传输中断有使能,transferNum设为6,则DATA_TRANS_NUM为3时产生中断,进入回调)
  59. CHANNELx_STATUS[0] 传输完成
  60. */
  61. if ((wpara & 0x01) == 0x1)
  62. {
  63. g_dmaFinish = 1;
  64. }
  65. if ((wpara & 0x02) == 0x2)
  66. {
  67. g_halfDmaFinish = 1;
  68. }
  69. if ((wpara & 0x04) == 0x4)
  70. {
  71. g_dmaTransError = 1;
  72. }
  73. }
  74. /**
  75. * ADC_DMAInit
  76. *
  77. * @param[in] void
  78. * @return void
  79. *
  80. * @brief ADC DMA初始化,配置DMA相关参数。
  81. */
  82. void ADC_DMAInit(void)
  83. {
  84. DMA_ConfigType tmpDMAConfig;
  85. memset(&tmpDMAConfig, 0x00, sizeof(DMA_ConfigType));
  86. tmpDMAConfig.channelEn = ENABLE; ///<使能DMAx通道
  87. tmpDMAConfig.finishInterruptEn = ENABLE; ///<使能DMA传输完成中断
  88. tmpDMAConfig.halfFinishInterruptEn = DISABLE; ///<去能DMA半传输完成中断
  89. tmpDMAConfig.errorInterruptEn = ENABLE; ///<使能DMA传输错误中断
  90. tmpDMAConfig.channelPriority = DMA_PRIORITY_VERY_HIGH;///<设置DMA通道优先级,0~3 :优先级由低到高
  91. tmpDMAConfig.circular = ENABLE; ///<使能循环模式,如果只想工作一次,设为0即可。
  92. tmpDMAConfig.direction = DMA_READ_FROM_PERIPH; ///<0: 从外设读取,1:从存储器读取
  93. tmpDMAConfig.MEM2MEM = DISABLE; ///<0:在非存储器与存储器之间传输,1:在存储器与存储器之间传输
  94. tmpDMAConfig.memByteMode = DMA_MEM_BYTE_MODE_1TIME; ///<MEM字分割传输数,0:32-bit,1:16-bit[15:0]; 2:16-bit[23:16][7:0];3:8-bit。详情可参考AC781X芯片手册 表20-2 可编程数据宽度&数据对齐
  95. tmpDMAConfig.memIncrement = ENABLE; ///<1:MEM地址增加
  96. tmpDMAConfig.periphIncrement = DISABLE; ///<0:外设地址固定
  97. tmpDMAConfig.memSize = DMA_MEM_SIZE_16BIT; ///<0:8-bit,1:16-bit,2:32-bit
  98. tmpDMAConfig.periphSize = DMA_PERIPH_SIZE_16BIT; ///<0:8-bit,1:16-bit,2:32-bit
  99. tmpDMAConfig.transferNum = ADC_SAMPLE_CHANNEL*ADC_FILTER_NUM; ///<DMA通道传输长度
  100. tmpDMAConfig.periphSelect = DMA_PEPIRH_ADC0; //外设选择
  101. tmpDMAConfig.periphStartAddr = (uint32_t)(&(ADC0->RDR)); ///<Move ADC DR to memory
  102. tmpDMAConfig.memStartAddr = (uint32_t)g_ADCValueBuffer; //设置DMA开始地址
  103. tmpDMAConfig.memEndAddr = (uint32_t)g_ADCValueBuffer+sizeof(g_ADCValueBuffer);//设置DMA结束地址
  104. tmpDMAConfig.callBack = ADC_DMACallback; ///<设置DMA中断回调
  105. DMA_Init(DMA0_CHANNEL2, &tmpDMAConfig); ///<ADC 使用DMA1通道,每个模块对应的DMA通道,可参考 AC781X芯片手册 表20-1 DMA请求列表
  106. NVIC_SetPriority(DMA0_CHANNEL2_IRQn, 3);
  107. NVIC_ClearPendingIRQ(DMA0_CHANNEL2_IRQn);
  108. NVIC_EnableIRQ(DMA0_CHANNEL2_IRQn); ///<使能DMA1中断请求
  109. }
  110. /**
  111. * CTU_Config
  112. *
  113. * @param[in] void
  114. * @return void
  115. *
  116. * @brief 配置CTU模块,Timer0触发ADC规则组采样。
  117. */
  118. void CTU_Config(void)
  119. {
  120. CTU_ConfigType ctuConfig;
  121. memset(&ctuConfig, 0x00, sizeof(ctuConfig));
  122. ctuConfig.uart0RxFilterEn = DISABLE; //去能UART0_RX滤波
  123. ctuConfig.rtcCaptureEn = DISABLE; //去能RTC捕获
  124. ctuConfig.acmpCaptureEn = DISABLE; //去能ACMP捕获
  125. ctuConfig.uart0RxCaptureEn = DISABLE; //去能UART0_RX捕获
  126. ctuConfig.uartTxModulateEn = DISABLE; //去能UART0_TX调制
  127. ctuConfig.clkPsc = CTU_CLK_PRESCALER_1; //分频
  128. ctuConfig.adcRegularTriggerSource = CTU_TRIGGER_ADC_TIMER_CH0_OVERFLOW; //Timer0触发ADC规则组采样。
  129. ctuConfig.delay0Time = 0; //触发延迟
  130. // ctuConfig.adcInjectTriggerSource = CTU_TRIGGER_ADC_PWM0_INIT; //
  131. // ctuConfig.delay1Time = 0;
  132. // ctuConfig.pwdt0In3Source = CTU_PWDT_IN3_SOURCE_UART0_RX;
  133. // ctuConfig.pwdt1In3Source = CTU_PWDT_IN3_SOURCE_UART0_RX;
  134. CTU_Init(&ctuConfig);
  135. }
  136. /**
  137. * ADC_init
  138. *
  139. * @param[in] void
  140. * @return void
  141. *
  142. * @brief 初始化ADC,配置ADC参数。
  143. */
  144. void ADC_init()
  145. {
  146. ADC_ConfigType tempAdcConfig;
  147. ADC_ConfigType* adcConfig;
  148. adcConfig = &tempAdcConfig;
  149. //配置PINMUX
  150. //GPIO_SetFunc(GPIOA, GPIO_PIN2, GPIO_FUN2);///<ADC_IN8 Analog function enable
  151. adcConfig->scanModeEn = ENABLE; //扫描模式
  152. adcConfig->continousModeEn = DISABLE; //连续模式
  153. adcConfig->regularDiscontinousModeEn = DISABLE; //1:打开规则组间断转换模式
  154. adcConfig->injectDiscontinousModeEn = DISABLE; //1:打开注入组间断转换模式
  155. adcConfig->injectAutoModeEn = DISABLE; //1:自动注入模式
  156. adcConfig->intervalModeEn = DISABLE; //1:注入组为间隔转换模式
  157. adcConfig->regularDiscontinousNum = 0; //
  158. adcConfig->EOCInterruptEn = DISABLE; //EOC中断去能
  159. adcConfig->IEOCInterruptEn = DISABLE; //IEOC中断去能
  160. adcConfig->interruptEn = DISABLE; //去能中断
  161. adcConfig->regularDMAEn = ENABLE; //使能ADC DMA
  162. adcConfig->regularTriggerMode = ADC_TRIGGER_EXTERNAL;//ADC触发源,外部触发
  163. //adcConfig->injectTriggerMode = ADC_TRIGGER_INTERNAL; //ADC触发源,内部触发
  164. adcConfig->regularSequenceLength = ADC_SAMPLE_CHANNEL; //规则组长度设为1
  165. //adcConfig->injectSequenceLength = 0; //注入组长度设为0
  166. adcConfig->dataAlign = ADC_DATA_ALIGN_RIGHT; //右对齐
  167. adcConfig->powerMode = ADC_POWER_ON; //上电
  168. adcConfig->clkPsc = ADC_CLK_PRESCALER_1; ///<Set ADC Clk = 24M/2/(0+1)
  169. ADC_Init(ADC0, adcConfig); ///<ADC works Mode Config
  170. /*
  171. ADC转换率计算公式:
  172. 转换时间= 采样时间+转换时间+同步时间
  173. 转换时间= (SPT+12)/ADC模块时钟频率+5/APB时钟频率
  174. 备注:
  175. 1.同步时间为5个APB CLK。
  176. 2.ADC时钟频率 = APB时钟频率 /(分频系数+1)
  177. */
  178. ADC_SetRegularGroupChannel(ADC0, ADC_CH_0, ADC_SPT_CLK_7, 0); //set ADC_CH_7 为第1个采样序列 NTC 温度采集
  179. ADC_SetRegularGroupChannel(ADC0, ADC_CH_7, ADC_SPT_CLK_7, 1); //set ADC_CH_7 为第1个采样序列 线性hall 1
  180. ADC_SetRegularGroupChannel(ADC0, ADC_CH_8, ADC_SPT_CLK_7, 2); ///set ADC_CH_8 为第2个采样序列 线性hall 2
  181. //ADC_SetRegularGroupChannel(ADC0, ADC_CH_BANDGAP, ADC_SPT_CLK_7, 2); ///set ADC_CH_BANDGAP 为第3个采样序列
  182. //ADC_SetRegularGroupChannel(ADC0, ADC_CH_TSENSOR, ADC_SPT_CLK_7, 3); ///set ADC_CH_TSENSOR 为第4个采样序列
  183. ADC_DMAInit(); //ADC DMA初始化
  184. }
  185. float getTemperature(void)
  186. {
  187. uint16_t NtcAdc = 0;
  188. uint8_t i;
  189. float Rt = 0; //NTC 电阻
  190. float R = 50000; //50K 固定阻值电阻
  191. float T0 = 273.15+25; //转换为开尔文温度
  192. float B=3950; //B值
  193. float Ka=273.15; //K值
  194. float Vr=0; //电压值
  195. float temp = 0;
  196. for(i=0; i<ADC_FILTER_NUM; i++){
  197. NtcAdc += g_ADCValueBuffer[3*i];
  198. }
  199. NtcAdc = NtcAdc/ADC_FILTER_NUM;
  200. Vr = (float)(3.3*NtcAdc/4096);
  201. Rt = (3.3-Vr)*50000/Vr;
  202. temp=1/(1/T0+log(Rt/R)/B)-Ka+0.5; //计算温度
  203. return temp;
  204. }
  205. void getHallValue(uint16_t* pHall_1, uint16_t* pHall_2)
  206. {
  207. int16_t hall1_ADC = 0;
  208. int16_t hall2_ADC = 0;
  209. int8_t i;
  210. for(i=0; i<ADC_FILTER_NUM; i++){
  211. hall1_ADC += g_ADCValueBuffer[3*i+1];
  212. hall2_ADC += g_ADCValueBuffer[3*i+2];
  213. }
  214. *pHall_1 = hall1_ADC/ADC_FILTER_NUM;
  215. *pHall_2 = hall2_ADC/ADC_FILTER_NUM;
  216. }
  217. int16_t getHalldiff(void)
  218. {
  219. int16_t hall1_ADC = 0;
  220. int16_t hall2_ADC = 0;
  221. int8_t i;
  222. for(i=0; i<ADC_FILTER_NUM; i++){
  223. hall1_ADC += g_ADCValueBuffer[3*i+1];
  224. hall2_ADC += g_ADCValueBuffer[3*i+2];
  225. }
  226. hall1_ADC = hall1_ADC/ADC_FILTER_NUM;
  227. hall2_ADC = hall2_ADC/ADC_FILTER_NUM;
  228. return (hall1_ADC - hall2_ADC);
  229. }
  230. void printADCValue(void)
  231. {
  232. uint8_t index;
  233. printf("\r\n adc sample data is:");
  234. for (index = 0; index<ADC_SAMPLE_CHANNEL*ADC_FILTER_NUM; index++)
  235. {
  236. printf(" 0x%x",g_ADCValueBuffer[index]);
  237. }
  238. printf(", g_timer0Cnt:%d \r\n", g_timer0Cnt);
  239. }
  240. float getBatteryVoltage(void)
  241. {
  242. uint16_t sum = 0;
  243. uint8_t i;
  244. for(i=0; i<ADC_FILTER_NUM; i++){
  245. sum += g_ADCValueBuffer[2*i];
  246. }
  247. sum = sum/ADC_FILTER_NUM;
  248. return ((6.6*sum)/4096);
  249. }
  250. float getMotorCurrent(void)
  251. {
  252. uint16_t sum = 0;
  253. uint8_t i;
  254. for(i=0; i<ADC_FILTER_NUM; i++){
  255. sum += g_ADCValueBuffer[2*i+1];
  256. }
  257. sum = sum/ADC_FILTER_NUM;
  258. return ((3.3*sum)/4096);
  259. }
  260. /**
  261. * ADC_SampleTimerTrigerRegular
  262. *
  263. * @param[in] void
  264. * @return void
  265. *
  266. * @brief Timer定时触发规则组ADC_CHANNEL0单次采样。
  267. */
  268. void ADCSample_Init(void)
  269. {
  270. CTU_Config();
  271. ADC_init();
  272. Timer0_Init();
  273. }
  274. /**********<End>*********/