ad7616.c 15 KB

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