ad7616.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769
  1. /******************************************************************************
  2. 版权所有:
  3. 文件名称: adc.c
  4. 文件版本: 01.01
  5. 创建作者: sunxi
  6. 创建日期: 2025-06-09
  7. 功能说明: adc驱动程序。
  8. 其它说明:
  9. 修改记录:
  10. */
  11. /*------------------------------ 头文件 ---------------------------------------
  12. */
  13. #include "bsp.h"
  14. #include "rt.h"
  15. #include "head.h"
  16. #include <malloc.h>
  17. #include "rt_printf.h"
  18. #include "shm_comm_packet.h"
  19. #include <mb.h>
  20. /*------------------------------ 宏定义 ---------------------------------------
  21. */
  22. #if CFG_BSP_DEBUG
  23. #define _ADC_DEBUG
  24. #endif
  25. #define AC_HZ_NORMAL 50.0 // 交流标准频率
  26. #define ADC_DOT_BUF_SIZE ((CFG_ADC_CHANNEL)*(CFG_ADC_DOTS_PER_CHANNEL*2)*2)
  27. /*------------------------------ 全局变量 -------------------------------------
  28. */
  29. // AD片选地址,8、9、10、11槽对应CS2、CS1、CS3、CS4。
  30. static int g_adc_slot2cs[EQU_SLOT_AC_NUM] = {0};
  31. static u8 g_adc_chip;
  32. static u32 g_adc_range[EQU_SLOT_AC_NUM];
  33. const u8 g_cs_channel_table[CFG_ADC_CHANNEL]={1,9,2,10,3,11,4,12,5,13,6,14,7,15,8,16};
  34. //每个通道占用的空间:CFG_ADC_DOTS_PER_CHANNEL*2*2 = 128*2*2=512
  35. //通道数:CFG_ADC_CHANNEL+2= 64+2=66
  36. //总空间:512*18=9216=0x2400
  37. //保证首地址是buf长度两倍的倍数。以利用MAC计算的MASK寄存器来达到
  38. //循环buf的功能。注意每个通道buf的后半部分目前没有利用,以后可以用来干需要干的事情.
  39. //通道加2,用作开入开出等状态保存。
  40. // 不能使用分配的DMA内存,因为冷火实时linux平台不能使用动态内存映射。
  41. //short (*g_adc_dots)[CFG_ADC_DOTS_PER_CHANNEL*2];
  42. //s16 (*g_adc_dots)[CFG_ADC_DOTS_PER_CHANNEL*2] = (short (*)[CFG_ADC_DOTS_PER_CHANNEL*2])0x8ff00000;
  43. s16 (*g_adc_dots)[CFG_ADC_DOTS_PER_CHANNEL*2];
  44. struct timespec ts_adc_dots[CFG_ADC_DOTS_PER_CHANNEL];//时间戳
  45. // 128点录波,32点计算
  46. // 注意此buf由于是DMA类型,不能在实时中断中使用,只能在线程中使用。
  47. //s16 g_adc_dots_rec[CFG_ADC_DOTS_PER_CHANNEL*4][CFG_ADC_CHANNEL];
  48. s16 (*g_adc_dots_rec)[CFG_ADC_CHANNEL];
  49. u32 g_adc_dots_rec_dma;
  50. // 直流量测量采样点BUF
  51. u16 g_adc_dots_dc[CFG_ADC_CHANNEL_DC][CFG_ADC_DOTS_PER_CHANNEL];
  52. //ADC同步标志,表示ADC采样点BUFFER中的数据是否有效,AD7616转换的第一次数据也无效。
  53. int g_adc_sync = -(CFG_ADC_DOTS_PER_CHANNEL+1);
  54. // ADC点BUF的索引,当前索引下无值。
  55. volatile unsigned long g_adc_dots_count;
  56. int g_adc_channel;
  57. unsigned long g_adZero=14790;
  58. char g_adc_name[CFG_ADC_CHANNEL_ALL][20];
  59. struct rt_stat g_stat_adc[CFG_ADC_CHANNEL_ALL];
  60. // 跟踪频率
  61. float g_freq = 50.0 ;
  62. // 直流量测量采样点BUF
  63. u16 g_adc_dots_dc[CFG_ADC_CHANNEL_DC][CFG_ADC_DOTS_PER_CHANNEL];
  64. uint32_t g_yx_buf[2] = {0xffffffff,0xffffffff}; //预留64个遥信
  65. int g_adc_stat_flag=0;
  66. volatile unsigned int adc_wp;
  67. volatile unsigned int adc_rp;
  68. bool prt_flag;
  69. volatile unsigned long g_adc_dots_data_count;
  70. SHM_VER_T LuoHe_version;
  71. /*------------------------------ 函数声明 -------------------------------------
  72. */
  73. void dido_di_poll(SHM_SAMPLE_T *sample_data);
  74. /*------------------------------ 外部函数 -------------------------------------
  75. */
  76. //设置各通道的参考电压
  77. static int write_v_default(uint8_t no, uint8_t *data)
  78. {
  79. int ret = 0;
  80. int cnt = 0;
  81. if(data == NULL)
  82. return -1;
  83. while(1)
  84. {
  85. ret = shm_comm_packet_write(SHM_ADDR_D_RFV_1+(SHM_ADDR_D_RFV_2-SHM_ADDR_D_RFV_1)*no, data, sizeof(SHM_RFV_T));
  86. if(ret > 0)
  87. {
  88. break;
  89. }
  90. if(++cnt > 3)
  91. break;
  92. usleep(30);
  93. }
  94. return ret;
  95. }
  96. int adc_set_v_default(int no, u32 value)
  97. {
  98. int ret = 0;
  99. SHM_RFV_T v;
  100. v.rc = value;
  101. ret = write_v_default(no,(uint8_t *)&v);
  102. return ret;
  103. }
  104. void adc_get_range_config(int no,int slot)
  105. {
  106. int i,scale,channel,shift;
  107. u32 rc;
  108. rc = 0;
  109. for(i=0;i<EQU_SLOT_AC_CHN;i++)
  110. {
  111. channel = equ_get_ac_channel(slot,i)&0x0f;
  112. // 确定需要移位的数量
  113. // 双数是通道A,在低16位
  114. // 单数是通道B,在高16位
  115. shift = (channel/2)*2;
  116. if((channel&0x01))
  117. {
  118. shift += 16;
  119. }
  120. if(channel == AD7616_DC0_CHANNEL || channel == AD7616_DC1_CHANNEL)
  121. {
  122. rc |= 2<<shift;// 5V
  123. continue;
  124. }
  125. scale = equ_get_ac_scale(slot,i);
  126. switch(scale)
  127. {
  128. case EQU_SCALE_PT_DEFAULT:
  129. case EQU_SCALE_PT_120K:
  130. case EQU_SCALE_PT_172K:
  131. case EQU_SCALE_PT_204K:
  132. case EQU_SCALE_CT_10A_51:
  133. case EQU_SCALE_CT_10A_120:
  134. rc |= 1<<shift; // 2.5V
  135. break;
  136. case EQU_SCALE_PT_102K:
  137. case EQU_SCALE_PT_120K_604:
  138. case EQU_SCALE_PT_264V_3V53:
  139. case EQU_SCALE_CT_100A_3V53:
  140. case EQU_SCALE_CT_DEFAULT:
  141. case EQU_SCALE_CT_10A_249:
  142. case EQU_SCALE_CT_10A_604:
  143. case EQU_SCALE_EVT_3V25_100V:
  144. case EQU_SCALE_PT_400V_3V53:
  145. case EQU_SCALE_CT_20A_3V53:
  146. rc |= 2<<shift;// 5V
  147. break;
  148. case EQU_SCALE_EVT_6V50_100V:
  149. case EQU_SCALE_ECT_1V_1A:
  150. case EQU_SCALE_ECT_0V2_1A: // 零序电子CT采用2.5V的范围
  151. rc |= 3<<shift; // 10V
  152. break;
  153. }
  154. }
  155. adc_set_v_default(no,rc);
  156. mb_notice_RfV_update(0);
  157. }
  158. static int get_adc_wp(void)
  159. {
  160. WP_T wp;
  161. int ret = 0;
  162. int cnt = 0;
  163. while(1)
  164. {
  165. ret = shm_comm_packet_read(SHM_ADDR_U_WP, sizeof(wp), (uint8_t *)&wp, sizeof(wp));
  166. if(ret > 0)
  167. {
  168. return wp.write_flag_A;
  169. }
  170. if(++cnt > 3)
  171. break;
  172. usleep(30);
  173. }
  174. return ret;
  175. }
  176. static int get_adc_sample(int index, uint8_t *out_data)
  177. {
  178. SHM_SAMPLE_T sample;
  179. int ret = 0;
  180. int cnt = 0;
  181. if(out_data == NULL)
  182. return -1;
  183. while(1)
  184. {
  185. ret = shm_comm_packet_read_noABcrc(SHM_ADDR_U_ADC_1+index*(sizeof(sample)), sizeof(sample), out_data, sizeof(sample));
  186. if(ret > 0)
  187. {
  188. break;
  189. }
  190. if(++cnt > 3)
  191. break;
  192. usleep(30);
  193. }
  194. return ret;
  195. }
  196. static int get_e907_version(SHM_VER_T* par)
  197. {
  198. SHM_VER_T ver;
  199. int ret = 0;
  200. int cnt = 0;
  201. if(par == NULL)
  202. return -1;
  203. while(1)
  204. {
  205. ret = shm_comm_packet_read(SHM_ADDR_U_VER, sizeof(ver), (uint8_t *)&ver, sizeof(ver));
  206. if(ret > 0)
  207. {
  208. par->ver = ver.ver;
  209. par->v_crc = ver.v_crc;
  210. strcpy(par->time,ver.time);
  211. return 0;
  212. }
  213. if(++cnt > 3)
  214. break;
  215. usleep(30);
  216. }
  217. return -1;
  218. }
  219. int adc_init(void)
  220. {
  221. int ret = 0;
  222. int i;
  223. /*
  224. ret = get_adc_wp();
  225. if(ret < 0)
  226. {
  227. rt_printf("get_adc_wp 返回错误值,ret=%d!\r\n",adc_wp);
  228. return -1;
  229. }
  230. adc_wp = adc_rp = ret;
  231. */
  232. memset(&LuoHe_version,0,sizeof(LuoHe_version));
  233. g_adc_dots_count=0;
  234. prt_flag=false;
  235. // 采样计算点BUF初始化
  236. // 32;周波数
  237. g_adc_dots = malloc(ADC_DOT_BUF_SIZE);
  238. if(g_adc_dots == NULL)
  239. {
  240. rt_printf("g_adc_dots 分配失败!\r\n");
  241. return -1;
  242. }
  243. memset(g_adc_dots, 0x55, ADC_DOT_BUF_SIZE);
  244. g_adc_dots_data_count=0;
  245. // 录波点BUF初始化
  246. // 32;周波数
  247. g_adc_dots_rec = malloc(CFG_ADC_CHANNEL*2*ADC_REC_DOTS_CHANNEL);
  248. if(g_adc_dots_rec == NULL)
  249. {
  250. rt_printf("g_adc_dots_rec 分配失败!\r\n");
  251. rt_free(g_adc_dots);
  252. g_adc_dots = NULL;
  253. return -1;
  254. }
  255. // 初始化统计变量
  256. for(i=0; i<CFG_ADC_CHANNEL_ALL; i++)
  257. {
  258. sprintf(g_adc_name[i],"adc%02d[%08lx]",i,(u32)g_adc_dots + i*sizeof(g_adc_dots[0]));
  259. rt_stat_init(&g_stat_adc[i],g_adc_name[i]);
  260. }
  261. for(i=0; i<EQU_SLOT_AC_NUM; i++) //配置通道的采样参数
  262. {
  263. adc_get_range_config(i,g_ac_slot_begin+g_adc_slot2cs[i]);
  264. }
  265. #if 0
  266. // 检查和槽对应的ADC芯片是否安装或损坏
  267. // printk("g_adc_chip=%02x.\r\n",g_adc_chip);
  268. for(i=0; i< EQU_SLOT_AC_NUM; i++)
  269. {
  270. if(equ_get_ac_num(g_ac_slot_begin+i))
  271. {
  272. if(adc_slot_is_ok(i) == 0)
  273. {
  274. rt_err_set(ERR_CODE_INIT_ADC,i);
  275. rt_printf("槽%02ld:ADC 芯片未安装或错误!\r\n",g_ac_slot_begin+i);
  276. }
  277. }
  278. }
  279. #endif
  280. return ret;
  281. }
  282. int adc_exit(void)
  283. {
  284. int ret=0;
  285. if(g_adc_dots_rec)
  286. {
  287. free(g_adc_dots_rec);
  288. g_adc_dots_rec=0;
  289. }
  290. if(g_adc_dots)
  291. {
  292. free(g_adc_dots);
  293. g_adc_dots=0;
  294. }
  295. return ret;
  296. }
  297. int adc_isr(int mod)
  298. {
  299. int opt,i;
  300. SHM_SAMPLE_T sample_data;
  301. int ret = 0;
  302. int cnt = 0;
  303. int index = 0;
  304. static int start_flag = 0;
  305. static int recv_init = 0;
  306. static int get_version_flag = 0;
  307. ret = get_adc_wp();
  308. if(ret < 0)
  309. {
  310. // rt_printf("get_adc_wp 返回错误值,ret=%d!\r\n",ret);
  311. return -1;
  312. }
  313. if(get_version_flag == 0)
  314. {
  315. ret = get_e907_version(&LuoHe_version);
  316. if(ret == 0)
  317. {
  318. get_version_flag = 1;//noted by sunxi:这个应该要读成功后,才置为1.
  319. rt_printf("裸核版本号: ver=v%d.%02d,%s,crc=%X\r\n",LuoHe_version.ver/100,LuoHe_version.ver%100,LuoHe_version.time,LuoHe_version.v_crc);
  320. }
  321. }
  322. if(start_flag == 0)
  323. {
  324. //第1次运行,初始化 adc_rp
  325. start_flag++;
  326. adc_wp = ret;
  327. adc_rp = ret;
  328. }
  329. else
  330. {
  331. adc_wp = ret;
  332. }
  333. if(adc_wp>=SHM_ADC_WR_NUM)
  334. {
  335. adc_wp=0;
  336. }
  337. if(adc_rp>=SHM_ADC_WR_NUM)
  338. {
  339. adc_rp=0;
  340. }
  341. while(adc_wp!=adc_rp)
  342. {
  343. opt=((g_adc_dots_data_count&(ADC_REC_SAMPLE_RATE-1))==0)?TRUE:FALSE;
  344. ret = get_adc_sample(adc_rp,(uint8_t *)&sample_data);
  345. if(ret < 0)
  346. {
  347. if(++cnt > 3)
  348. {
  349. rt_printf("get_adc_sample 返回错误值, ret=%d!\r\n",ret);
  350. break;
  351. }
  352. continue;
  353. }
  354. //保存yx值
  355. g_yx_buf[0] = sample_data.yx_buf[0];
  356. g_yx_buf[1] = sample_data.yx_buf[1];
  357. if(!recv_init)
  358. {
  359. recv_init = 1;
  360. //开机时裸核还没有把遥信送上下,这里重新初始化plc_init
  361. dido_init_di();
  362. plc_init();
  363. }
  364. dido_di_poll(&sample_data);
  365. for(i=0; i<CFG_ADC_CHANNEL; i++)
  366. {
  367. g_adc_dots_rec[g_adc_dots_index_rec][i] = sample_data.sample_buf[i];
  368. if(opt==TRUE)
  369. {
  370. g_adc_dots[i][g_adc_dots_index] = sample_data.sample_buf[i];
  371. ts_adc_dots[g_adc_dots_index] = sample_data.ts;
  372. }
  373. }
  374. g_adc_dots_dc[0][g_adc_dots_data_count&CFG_ADC_DOTS_MASK] = sample_data.sample_buf[AD7616_DC0_CHANNEL];
  375. // g_adc_dots_dc[1][g_adc_dots_data_count&CFG_ADC_DOTS_MASK] = sample_data.sample_buf[AD7616_DC0_CHANNEL];
  376. if(opt==TRUE)
  377. {
  378. index = g_adc_dots_index;
  379. // 统计数据
  380. if(g_adc_stat_flag)
  381. {
  382. for(i=0; i<CFG_ADC_CHANNEL; i++)
  383. {
  384. rt_stat_in(&g_stat_adc[i],(unsigned short)(g_adc_dots[i][index]+32768));
  385. }
  386. }
  387. // 通道取反
  388. for(i=0; i<g_equ_adc_inv_num; i++)
  389. {
  390. g_adc_dots[g_equ_adc_inv[i]][index] = -(g_adc_dots[g_equ_adc_inv[i]][index]);
  391. }
  392. // 全部BUFFER都是新数据,才能算同步好。
  393. if(g_adc_sync < 1)
  394. {
  395. g_adc_sync ++;
  396. }
  397. // 索引值加一
  398. g_adc_dots_count++;
  399. //保护计数+突变量
  400. {
  401. dTCounter++;
  402. if(dTCounter%8==0)
  403. {
  404. prt_flag=true;
  405. }
  406. //突变量
  407. if(g_adc_sync == 1 && g_sw_init == 1)
  408. {
  409. protect_tbl_qd();
  410. }
  411. }
  412. }
  413. g_adc_dots_data_count++;
  414. if(++adc_rp>=SHM_ADC_WR_NUM)
  415. adc_rp=0;
  416. }
  417. return 0;
  418. }
  419. /******************************************************************************
  420. 函数名称: adc_slot_is_ok
  421. 函数版本: 01.01
  422. 创建作者: sunxi
  423. 创建日期: 2016-02-01
  424. 函数说明: 检查对应槽的ADC芯片是否安装或损坏。
  425. 参数说明:
  426. slot: 0~3;从g_ac_slot_begin开始
  427. 返回值: 成功返回0.
  428. 修改记录:
  429. */
  430. int adc_slot_is_ok(int slot)
  431. {
  432. if((u32)slot >= EQU_SLOT_AC_NUM)
  433. {
  434. return 0;
  435. }
  436. if(g_adc_chip & (1<<g_adc_slot2cs[slot]))
  437. {
  438. return 1;
  439. }
  440. return 0;
  441. }
  442. /******************************************************************************
  443. 函数名称: adc_fourier_address
  444. 函数版本: 01.01
  445. 创建作者: sunxi
  446. 创建日期: 2008-06-26
  447. 函数说明: 返回指定的通道进行傅立叶计算的首地址。首地址保证4bytes对齐。
  448. 参数说明:
  449. channel:ADC的通道,取值0~7。
  450. index: 要进行傅立叶计算的32点中的最后一个点的索引。这么设置是为了和
  451. g_adc_dots_index匹配。保证g_adc_dots_index可以直接作为参数传进来。
  452. 返回值: 成功返回0.
  453. 修改记录:
  454. */
  455. #if 0
  456. short * adc_fourier_address(int channel,int index)
  457. {
  458. short *p;
  459. //{index + CFG_ADC_DOTS_PER_CHANNEL}是为了确保索引在相应的范围之内.
  460. index = index + CFG_ADC_DOTS_PER_CHANNEL - CFG_ADC_DOTS_PER_PERIOD;
  461. //使用偶数索引,保证4bytes对齐
  462. index &= (~0x1);
  463. //得到首地址,注意地址必须和CFG_ADC_BUF_MASK相与。
  464. p = (short *)&g_adc_dots[channel][index];
  465. p = (short *)((unsigned int)p&CFG_ADC_BUF_MASK);
  466. return p;
  467. }
  468. #else
  469. short sam_buf[CFG_ADC_CHANNEL][CFG_ADC_DOTS_PER_PERIOD];
  470. short * adc_fourier_address(int channel,int index)
  471. {
  472. short *p;
  473. int i;
  474. if(index >= CFG_ADC_DOTS_PER_PERIOD)
  475. {
  476. index = index - CFG_ADC_DOTS_PER_PERIOD;
  477. index &= (~0x1);
  478. p = (short *)&g_adc_dots[channel][index];
  479. }
  480. else
  481. {
  482. index = CFG_ADC_DOTS_PER_CHANNEL-CFG_ADC_DOTS_PER_PERIOD+index;
  483. index &= (~0x1);
  484. for(i = 0;i < CFG_ADC_DOTS_PER_CHANNEL - index;i++)
  485. {
  486. sam_buf[channel][i] = g_adc_dots[channel][index+i];
  487. }
  488. for(i = 0;i <CFG_ADC_DOTS_PER_PERIOD +index-CFG_ADC_DOTS_PER_CHANNEL;i++)
  489. {
  490. sam_buf[channel][i+CFG_ADC_DOTS_PER_CHANNEL - index] = g_adc_dots[channel][i];
  491. }
  492. p = sam_buf[channel];
  493. }
  494. return p;
  495. }
  496. #endif
  497. /******************************************************************************
  498. 函数名称: adc_fourier_address_half
  499. 函数版本: 01.01
  500. 创建作者: sunxi
  501. 创建日期: 2008-06-26
  502. 函数说明: 返回指定的通道进行傅立叶计算的首地址。首地址保证4bytes对齐。
  503. 参数说明:
  504. channel:ADC的通道,取值0~7。
  505. index: 要进行傅立叶计算的32点中的最后一个点的索引。这么设置是为了和
  506. g_adc_dots_index匹配。保证g_adc_dots_index可以直接作为参数传进来。
  507. 返回值: 成功返回0.
  508. 修改记录:
  509. */
  510. short * adc_fourier_address_half(int channel,int index)
  511. {
  512. short *p;
  513. //{index - (CFG_ADC_DOTS_PER_PERIOD - 1)}是因为索引下有实际的采样值,所以不是
  514. // index - CFG_ADC_DOTS_PER_PERIOD.
  515. //{index + CFG_ADC_DOTS_PER_CHANNEL}是为了确保索引在相应的范围之内.
  516. index = index + CFG_ADC_DOTS_PER_CHANNEL - (CFG_ADC_DOTS_PER_PERIOD/2 - 1);
  517. //使用偶数索引,保证4bytes对齐
  518. index &= (~0x1);
  519. //得到首地址,注意地址必须和CFG_ADC_BUF_MASK相与。
  520. p = (short *)&g_adc_dots[channel][index];
  521. p = (short *)((unsigned int)p&CFG_ADC_BUF_MASK);
  522. return p;
  523. }
  524. int adc_stat_printf(void)
  525. {
  526. int i;
  527. unsigned long avg = 0;
  528. struct rt_stat *stat;
  529. if(g_adc_stat_flag == 0)
  530. {
  531. g_adc_stat_flag = 1;
  532. rt_printf("\r\n!!!ADC采样统计启动, 执行reset命令关闭。!!!\r\n");
  533. }
  534. rt_printf("\r\n频率跟踪:%fHZ.\r\n",g_freq);
  535. rt_printf("g_adc_chip:0x%02x.\r\n",g_adc_chip);
  536. for(i=0;i<EQU_SLOT_AC_NUM;i++)
  537. {
  538. rt_printf("槽[%02lu]: 是否有对应芯片:%d,输入范围:0x%08lx.\r\n",g_ac_slot_begin+i,adc_slot_is_ok(i),g_adc_range[g_adc_slot2cs[i]]);
  539. }
  540. rt_printf("\r\n[ADC中断周期统计]\r\n");
  541. rt_printf("name\t\t\tmin\tmax\tavg\tsum\t\tcnt\n");
  542. // rt_stat_printf(&g_stat_adc_time);
  543. rt_printf("\r\n[ADC抖动统计]\r\n");
  544. rt_printf("name\t\t\tmin\tmax\tdif\tavg\tsum\t\tcnt\n");
  545. rt_printf("\r\n");
  546. for(i=0; i<CFG_ADC_CHANNEL_ALL; i++)
  547. {
  548. stat = &g_stat_adc[i];
  549. if(stat->cnt)
  550. {
  551. avg = stat->sum/stat->cnt;
  552. }
  553. rt_printf("%-24s%ld\t%ld\t%ld\t%ld\t%-16ld%ld\r\n",stat->name,stat->min,stat->max,stat->max - stat->min,avg,stat->sum,stat->cnt);
  554. }
  555. return 0;
  556. }
  557. int adc_stat_reset(void)
  558. {
  559. int i;
  560. g_adc_stat_flag = 0;
  561. for(i=0; i<CFG_ADC_CHANNEL_ALL; i++)
  562. {
  563. rt_stat_init(&g_stat_adc[i],g_stat_adc[i].name);
  564. }
  565. // rt_stat_init(&g_stat_adc_time,g_stat_adc_time.name);
  566. return 0;
  567. }
  568. #if 0
  569. static int get_adc_dc(uint8_t *out_data)
  570. {
  571. int ret = 0;
  572. int cnt = 0;
  573. if(out_data == NULL)
  574. return -1;
  575. while(1)
  576. {
  577. ret = shm_comm_packet_read(SHM_ADDR_U_DC, sizeof(SHM_DC_T), out_data, sizeof(SHM_DC_T));
  578. if(ret > 0)
  579. {
  580. break;
  581. }
  582. if(++cnt > 3)
  583. break;
  584. usleep(30);
  585. }
  586. return ret;
  587. }
  588. #endif
  589. float dc_get(unsigned int index)
  590. {
  591. u32 i,v;
  592. float f;
  593. v = 0;
  594. for(i=0;i<CFG_ADC_DOTS_PER_CHANNEL;i++)
  595. {
  596. v += (u16)g_adc_dots_dc[index][i];
  597. }
  598. f = (float)v/CFG_ADC_DOTS_PER_CHANNEL;
  599. // 乘理论系数
  600. f /= 129.791f;
  601. return f;
  602. }
  603. /*------------------------------ 测试函数 -------------------------------------
  604. */
  605. #ifdef _ADC_DEBUG
  606. int adc_test(void)
  607. {
  608. return 0;
  609. }
  610. #endif
  611. /*------------------------------ 文件结束 -------------------------------------
  612. */