dido.c 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493
  1. /******************************************************************************
  2. 版权所有:
  3. 文件名称: dido.c
  4. 文件版本: 01.01
  5. 创建作者: sunxi
  6. 创建日期: 2013-02-28
  7. 功能说明: 开入开出
  8. 其它说明:
  9. // 启动继电器
  10. 1、启动继电器可以重复多次启动,每次启动都将重新计算超时时间。
  11. 2、启动继电器有2种关闭方式:
  12. 1) 超时自动关闭,超时时间为装置参数的遥控超时时间。
  13. 2) 通过函数调用关闭。
  14. 3、有保护启动标志时,选择继电器不允许关闭。
  15. 4、需提供启动继电器的超时设置函数.
  16. 5、只所以采用超时自动关闭,是因为以前的程序关闭方式存在问题,可能由于资源竞争的关系,
  17. 导致不该关闭时却关闭了,虽然概率很小,但理论上存在。
  18. 修改记录:
  19. */
  20. /*------------------------------- 头文件 --------------------------------------
  21. */
  22. #include "head.h"
  23. #include "gpio.h"
  24. #include "ad7616.h"
  25. /*------------------------------- 宏定义 --------------------------------------
  26. */
  27. #define DO_PWM_SW_HZ 0
  28. #define DO_PWM_SW_QD 1
  29. #define DO_PWM_SW_NULL 2
  30. #define DO_PWM_SW_FZ 3
  31. #define DO_QD_KC DO_OUT4
  32. /*------------------------------ 类型结构 -------------------------------------
  33. */
  34. /*------------------------------ 全局变量 -------------------------------------
  35. */
  36. extern u32 g_brd_type_kz;
  37. u16 g_do_status; // 开出状态,因为仅主板有开出
  38. u32 g_do_start_time;
  39. int g_do_flag;
  40. u8 g_di_slot = 0;// 开入板卡配置槽位
  41. struct di g_di[EQU_SLOT_NUM_MAX];
  42. struct _do g_do[EQU_SLOT_NUM_MAX];
  43. struct di_struct g_di_st[EQU_SLOT_NUM_MAX][DIDO_MAX_DI_PER_SLOT];
  44. struct do_struct g_do_st[DO_NUM];
  45. struct rt_stat g_stat_di_delay;
  46. struct rt_stat g_stat_do_delay;
  47. static u32 di_error_count = 0;
  48. static u32 error_count = 0;
  49. struct do_time g_do_time[EQU_SLOT_NUM_MAX];
  50. /*------------------------------ 内部函数 -------------------------------------
  51. 内部函数以下划线‘_’开头,不需要检查参数的合法性.
  52. */
  53. /**
  54. * @brief 开出状态更新
  55. * @param[in/out] {int} on
  56. * @param[in/out] {unsigned int} i
  57. * @return * void
  58. * @retval none
  59. *
  60. * @warning none
  61. * @note none
  62. */
  63. static inline void _io_do_status_update(int on, unsigned int i)
  64. {
  65. if (on)
  66. {
  67. g_do_status |= (1 << i);
  68. g_do_st[i].us_on = ustimer_get_origin();
  69. }
  70. else
  71. {
  72. g_do_status &= ~(1 << i);
  73. }
  74. // 更新返校检查时间
  75. g_do_start_time = ustimer_get_origin();
  76. }
  77. /**
  78. * @brief 控制开出动作
  79. * @param[in/out] {u32} slot 板卡索引
  80. * @param[in/out] {u16} v 控制码
  81. * @param[in/out] {int} is_on 合分
  82. * @return * int
  83. * @retval none
  84. *
  85. * @warning none
  86. * @note none
  87. */
  88. static int _dido_do(u32 slot, u16 v, int is_on)
  89. {
  90. int i;
  91. // 循环检查开出位
  92. for (i = 0; i < g_board_info[slot].do_num; i++)
  93. {
  94. // 如果对应开出位有效,开出。
  95. if (((v >> i) & 0x1))
  96. {
  97. dido_do_kz(is_on, equ_get_do_channel(i));
  98. }
  99. }
  100. if (((v >> g_board_info[slot].do_num) & 0x1)) // 启动继电器
  101. {
  102. dido_do_kz(is_on, DO_QD_KC);
  103. }
  104. return 0;
  105. }
  106. #if !defined CPU_FUXI
  107. /**
  108. * @brief 开入防抖处理
  109. * @param[in/out] {type} slot
  110. * @param[in/out] {type} di_num
  111. * @param[in/out] {type} di
  112. * @param[in/out] {type} ts
  113. * @return * int
  114. * @retval none
  115. *
  116. * @warning none
  117. * @note none
  118. */
  119. static int _get_di(int slot, u16 di_num, u32 di, struct timespec ts)
  120. {
  121. int i, b_on;
  122. struct rtc_time_t ct;
  123. uint32 diff_nsec;
  124. // 获取遥信值,并记录时间戳。
  125. // 用秒的低8位和纳秒的高24位合成32位时间戳
  126. // clk_time_get(&ts);
  127. // 循环处理每一个遥信
  128. for (i = 0; i < di_num; i++)
  129. {
  130. struct di_struct *pt = &g_di_st[slot][i];
  131. // 处理当前变位遥信。
  132. b_on = (di & (1 << i)) ? 1 : 0;
  133. if (b_on ^ pt->b_on)
  134. {
  135. if (!pt->b_first_change)
  136. {
  137. pt->b_first_change = true;
  138. pt->tm_back = 0;
  139. pt->tm_keep = 0;
  140. // 记录变位的起始时刻点
  141. pt->ts_di = ts;
  142. pt->dt = g_adc_dots_data_count;
  143. #ifdef DIDO_TS_MODE
  144. pt->ts_tmp = ts;
  145. #endif
  146. }
  147. else
  148. {
  149. // 累计保持时间
  150. #ifdef DIDO_TS_MODE
  151. if (ts.tv_nsec >= pt->ts_tmp.tv_nsec)
  152. {
  153. diff_nsec = ts.tv_nsec - pt->ts_tmp.tv_nsec;
  154. }
  155. else
  156. {
  157. diff_nsec = 1000000000ul - pt->ts_tmp.tv_nsec + ts.tv_nsec;
  158. }
  159. if (diff_nsec > 312500ul)
  160. {
  161. diff_nsec = 156250ul;
  162. }
  163. pt->acc_nsec += diff_nsec;
  164. while (pt->acc_nsec >= 156250ul)
  165. {
  166. pt->tm_keep++;
  167. pt->acc_nsec -= 156250ul;
  168. }
  169. pt->ts_tmp = ts;
  170. // rt_printf("diff_nsec:%d, pt->tm_keep:%d\r\n", diff_nsec, pt->tm_keep);
  171. #else
  172. pt->tm_keep++;
  173. #endif
  174. pt->tm_back = 0; // 现在返回门限是0.5us,所有必须在此清零。
  175. // 保持时间到,处理变位
  176. if (pt->tm_keep > pt->tm_filter)
  177. {
  178. // 记录遥信状态
  179. #if defined YX_DI_ERROR // 广西遥信异常后,遥信屏蔽不保存,不上报
  180. if (g_run_stu.yxEnable == 0 && b_on == 0)
  181. {
  182. pt->type = SOE_TYPE_TST;
  183. pt->b_first_change = false;
  184. pt->tm_back = 0;
  185. pt->tm_keep = 0;
  186. return 0;
  187. }
  188. #endif
  189. pt->b_on = b_on;
  190. if (g_di[slot].inv[0] & (1 << i))
  191. {
  192. b_on = !b_on;
  193. }
  194. if (b_on)
  195. {
  196. g_di[slot].value[0] |= (1 << i);
  197. }
  198. else
  199. {
  200. g_di[slot].value[0] &= ~(1 << i);
  201. }
  202. // 记录SOE
  203. soe_record_yx((WORD)(i + (slot << 8)), b_on, &pt->ts_di);
  204. timespec_to_rtc(ts, &ct, 1);
  205. rt_printf("di=%x value[0]=%x\r\n", di, g_di[slot].value[0]);
  206. // 更新逻辑模型中的值
  207. sw_di_set(g_di_st[slot][i].owner, g_di_st[slot][i].type, b_on ? SW_DI_TYPE_ON : SW_DI_TYPE_OFF);
  208. // 更新记录
  209. g_di[slot].ts_t[i][g_di[slot].ts_i[i].n] = pt->dt;
  210. g_di[slot].ts_v[i][g_di[slot].ts_i[i].n] = b_on;
  211. g_di[slot].ts_i[i].n++;
  212. // 滤波处理数据复位
  213. pt->b_first_change = false;
  214. pt->tm_back = 0;
  215. pt->tm_keep = 0;
  216. }
  217. }
  218. }
  219. else if (pt->b_first_change)
  220. {
  221. // 变位返回处理
  222. pt->tm_back++;
  223. // 屏蔽下一行,
  224. // 最新逻辑,有效电平不连续可计数累加;
  225. // 无效电平连续超0.5ms复归,且不连续累加。
  226. // pt->tm_keep=0;
  227. // if(pt->tm_back > pt->tm_filter)
  228. if (pt->tm_back > 4) // 针对高频振荡波测试优化,返回门限设置为500us附件(4*156=624us)。
  229. {
  230. pt->tm_back = 0;
  231. pt->tm_keep = 0;
  232. pt->b_first_change = false;
  233. }
  234. }
  235. }
  236. return 0;
  237. }
  238. #endif
  239. /*------------------------------ 外部函数 -------------------------------------
  240. 外部函数供其它实体文件引用,必须仔细检查传入参数的合法性.
  241. */
  242. /**
  243. * @brief 开入开出初始化
  244. * @return * int
  245. * @retval none
  246. *
  247. * @warning none
  248. * @note none
  249. */
  250. int dido_init(void)
  251. {
  252. u32 i;
  253. // 检查装置配置
  254. if (equ_config_null())
  255. {
  256. return -1;
  257. }
  258. // 初始化取反标志
  259. for (i = 0; i < g_equ_config->di_num; i++)
  260. {
  261. if (g_equ_config_di[i].is_Inverse == 0)
  262. {
  263. continue;
  264. }
  265. if (g_equ_config_di[i].slot < EQU_SLOT_NUM_MAX && g_equ_config_di[i].index < DIDO_MAX_DI_PER_SLOT)
  266. {
  267. if (g_equ_config_di[i].index < 32)
  268. {
  269. g_di[g_equ_config_di[i].slot].inv[0] |= 1 << g_equ_config_di[i].index;
  270. }
  271. else
  272. {
  273. g_di[g_equ_config_di[i].slot].inv[1] |= 1 << (g_equ_config_di[i].index - 32);
  274. }
  275. }
  276. }
  277. dido_stat_reset();
  278. return 0;
  279. }
  280. /**
  281. * @brief 判断是否需要动作预置继电器
  282. * @param[in/out] {u32} slot
  283. * @param[in/out] {u32} index
  284. * @return * int
  285. * @retval none
  286. *
  287. * @warning none
  288. * @note none
  289. */
  290. int dido_do_have_select(u32 slot, u32 index)
  291. {
  292. int i = 0;
  293. i = equ_get_do_channel(index);
  294. if (i == DO_OUT0 || i == DO_OUT1 || i == DO_OUT2 || i == DO_OUT3)
  295. {
  296. return 1;
  297. }
  298. return 0;
  299. }
  300. /******************************************************************************
  301. 函数名称: dido_do
  302. 函数版本: 01.01
  303. 创建作者:
  304. 创建日期: 2013-03-13
  305. 函数说明: 板卡开出动作
  306. 参数说明: 无
  307. 返回值:
  308. 修改记录:
  309. */
  310. int dido_do(u32 slot, u16 v, int is_on)
  311. {
  312. // unsigned long flags;
  313. // 检查DO资源
  314. if (equ_get_do_num(slot) == 0)
  315. {
  316. return -2;
  317. }
  318. // 保存输出值,在5ms中断中输出
  319. rt_irq_save(flags);
  320. g_do_flag = 1;
  321. if (is_on)
  322. {
  323. g_do[slot].on |= v;
  324. }
  325. else
  326. {
  327. g_do[slot].off |= v;
  328. }
  329. rt_irq_restore(flags);
  330. return 0;
  331. }
  332. /******************************************************************************
  333. 函数名称: do_kout_check
  334. 函数版本:
  335. 创建作者:
  336. 创建日期: 2013-03-13
  337. 函数说明: 开出反校。
  338. 参数说明: 无
  339. 返回值: 结果
  340. 修改记录:
  341. */
  342. void io_do_check(void)
  343. {
  344. u16 now, fj;
  345. u32 us;
  346. #ifdef DO_KOUT_CHECK
  347. u8 index;
  348. // 取返校值,必须在下面时间点检查之
  349. index = g_equ_config_di[g_sw[0].di_cfg_index[SW_DI_KOCHK]].index;
  350. fj = gpio_di_fj(index) & 0x01; // 0x02;
  351. // 返回值只判断启动、分闸、合闸
  352. now = 0;
  353. if (g_do_status & (1 << DO_QD_KC)) // 预置反校DO_PWM_SW_QD
  354. {
  355. now |= 0x01;
  356. }
  357. #else
  358. // 取返校值,必须在下面时间点检查之
  359. fj = gpio_di_fj() & 0x02;
  360. // 返回值只判断启动、分闸、合闸
  361. now = 0;
  362. if (g_do_status & (1 << DO_PWM_SW_QD))
  363. {
  364. // now |= 0x01;
  365. }
  366. if (g_do_status & ((1 << DO_PWM_SW_FZ) | (1 << DO_PWM_SW_HZ)))
  367. {
  368. now |= 0x02;
  369. }
  370. #endif
  371. // 开出动作20ms内不进行开出自检,
  372. // 经实测启动继电器因为是继电器触点返校,其返回时间最长,但也小于10ms。
  373. // 分合闸继电器是光耦返校,本身需要的时间可以忽略不计,但受启动继电器的影响
  374. us = ustimer_get_duration(g_do_start_time);
  375. if (us < 20 * 1000)
  376. {
  377. return;
  378. }
  379. if (fj != now)
  380. {
  381. // rt_printf("返校失败:fj=%04x,now=%04x, g_do_status=%04x(us=%d)\r\n",fj,now, g_do_status,us);
  382. rt_err_set(ERR_CODE_DO_CHECK, 0);
  383. return;
  384. }
  385. }
  386. /******************************************************************************
  387. 函数名称: io_do_return
  388. 函数版本: 01.01
  389. 创建作者:
  390. 创建日期: 2013-03-13
  391. 函数说明: 开出保持时间
  392. 参数说明: 无
  393. 返回值: 无
  394. 修改记录:
  395. */
  396. void io_do_return(void)
  397. {
  398. int i;
  399. u32 dly;
  400. for (i = 0; i < DO_NUM; i++) // 开出保持
  401. {
  402. if (g_do_status & (1 << i)) // 正在开出
  403. {
  404. dly = ustimer_get_duration(g_do_st[i].us_on);
  405. // +2500是为了提高脉宽精度,因为此函数是在5ms(5000)中断中调用的.
  406. // 如果us_keep为0,表示此开出没有自动返回的需求,应忽略。
  407. if (((dly + 2500) > g_do_st[i].us_keep) && (g_do_st[i].us_keep > 0))
  408. {
  409. dido_do_kz(0, i);
  410. }
  411. }
  412. }
  413. }
  414. /******************************************************************************
  415. 函数名称: dido_do_poll
  416. 函数版本: 01.01
  417. 创建作者:
  418. 创建日期: 2013-03-13
  419. 函数说明: 是否开出巡检
  420. 参数说明: 无
  421. 返回值: 无
  422. 修改记录:
  423. */
  424. void dido_do_poll(void)
  425. {
  426. int i;
  427. // 真正开出
  428. if (g_do_flag)
  429. {
  430. g_do_flag = 0;
  431. for (i = 0; i < EQU_SLOT_NUM_MAX; i++)
  432. {
  433. if (g_do[i].on)
  434. {
  435. _dido_do(i, g_do[i].on, 1);
  436. g_do[i].on = 0;
  437. }
  438. if (g_do[i].off)
  439. {
  440. _dido_do(i, g_do[i].off, 0);
  441. g_do[i].off = 0;
  442. }
  443. }
  444. }
  445. // 开出返回
  446. io_do_return();
  447. #if !defined CPU_FUXI
  448. // 开出返校
  449. if (g_board_info[EQU_SLOT_KZ].do_num != 0)
  450. {
  451. #ifdef DO_KOUT_CHECK
  452. if ((short)g_sw[0].di_cfg_index[SW_DI_KOCHK] != INDEX_INVALLID)
  453. {
  454. io_do_check();
  455. }
  456. #else
  457. // io_do_check();
  458. #endif
  459. }
  460. #endif
  461. return;
  462. }
  463. /******************************************************************************
  464. 函数名称: dido_qd_set_keeptime
  465. 函数版本: 01.01
  466. 创建作者: sunxi
  467. 创建日期: 2013-06-10
  468. 函数说明: 设置启动继电器的保持时间
  469. 参数说明: 无
  470. 返回值: 无
  471. 修改记录:
  472. */
  473. void dido_qd_set_keeptime(u32 us)
  474. {
  475. g_do_st[DO_QD_KC].us_keep = us;
  476. }
  477. #if !defined CPU_FUXI
  478. /******************************************************************************
  479. 函数名称: get_di
  480. 函数版本: 01.01
  481. 创建作者:
  482. 创建日期: 2013-03-13
  483. 函数说明: 开入防抖处理
  484. 参数说明: 无
  485. 返回值: 无
  486. 修改记录:
  487. */
  488. // unsigned int di_val;
  489. void dido_di_poll(SHM_SAMPLE_T *sample_data)
  490. {
  491. u32 di, di_num, slot;
  492. u64 us0;
  493. // 如果交流延时失电信号有,闭锁遥信
  494. if (pRunSet->dSDYX_T && g_run_stu.dcjlsd_t)
  495. {
  496. return;
  497. }
  498. us0 = ustimer_get_origin();
  499. // 主板,19个遥信
  500. slot = EQU_SLOT_KZ;
  501. di_num = equ_get_di_num(slot);
  502. // di = gpio_get_di();
  503. di = change_di_ch(~sample_data->yx_buf[0]);
  504. rt_printf("gpio_get_di:yx_buf[0]=%x di=%x\r\n", sample_data->yx_buf[0], di);
  505. _get_di(slot, di_num, di, sample_data->ts);
  506. rt_stat_other_in(6, ustimer_get_duration(us0));
  507. }
  508. #endif
  509. /******************************************************************************
  510. 函数名称: dido_di_is_on
  511. 函数版本: 01.01
  512. 创建作者:
  513. 创建日期: 2013-03-13
  514. 函数说明: 开入状态
  515. 参数说明: 无
  516. 返回值: 1:有开入,0:无开入
  517. 修改记录:
  518. */
  519. int dido_di_is_on(u8 slot, u8 index)
  520. {
  521. if (slot >= EQU_SLOT_NUM_MAX || index >= 64)
  522. {
  523. return 0;
  524. }
  525. if (index < 32)
  526. {
  527. if (g_di[slot].value[0] & (1 << index))
  528. {
  529. return 1;
  530. }
  531. }
  532. else
  533. {
  534. index -= 32;
  535. if (g_di[slot].value[1] & (1 << index))
  536. {
  537. return 1;
  538. }
  539. }
  540. return 0;
  541. }
  542. // 查看某个时间点下的开入值
  543. int dido_di_is_on_ts(u8 slot, u8 index, u32 ts)
  544. {
  545. int i, r;
  546. struct ts_index ts_i;
  547. if (slot >= EQU_SLOT_NUM_MAX || index >= 32)
  548. {
  549. return 0;
  550. }
  551. #if 0
  552. {
  553. static int cnt = 0;
  554. if(cnt++ < 16)
  555. rt_printf("slot=%d,index=%d,ts=%d.\r\n",slot,index,ts);
  556. }
  557. #endif
  558. ts_i.n = g_di[slot].ts_i[index].n - 1; // ts_i下是无值的,所以应该减1
  559. for (i = 0; i < DIDO_TS_NUM; i++)
  560. {
  561. r = (int)(ts - g_di[slot].ts_t[index][ts_i.n]);
  562. if (r >= 0)
  563. {
  564. return g_di[slot].ts_v[index][ts_i.n];
  565. }
  566. ts_i.n--;
  567. }
  568. if (g_di[slot].value[0] & (1 << index))
  569. {
  570. return 1;
  571. }
  572. return 0;
  573. }
  574. int dido_do_kz(int on, unsigned int i)
  575. {
  576. if (on) // 开出,需启动启动继电器打开电源
  577. {
  578. if (i == DO_OUT0 || i == DO_OUT1 || i == DO_OUT2 || i == DO_OUT3)
  579. {
  580. gpio_kout_do(on, DO_QD_KC - DO_OUT0); // 对应X4端子的DO0~DO7
  581. _io_do_status_update(1, DO_QD_KC);
  582. }
  583. }
  584. if (i == DO_QD_KC) // 启动继电器出口
  585. {
  586. if (!on)
  587. {
  588. if (g_tRelay[0].zqd) // 如果保护启动,禁止关闭启动继电器
  589. {
  590. return -2;
  591. }
  592. if (g_do_status & (~(1 << DO_QD_KC))) // 有其他开出,不收启动继电器
  593. {
  594. return -3;
  595. }
  596. }
  597. gpio_kout_do(on, i - DO_OUT0); // 对应X4端子的DO0~DO7
  598. _io_do_status_update(on, i);
  599. }
  600. else if ((i >= DO_OUT0) && (i <= DO_OUT7))
  601. {
  602. gpio_kout_do(on, i - DO_OUT0); // 对应X4端子的DO0~DO7
  603. _io_do_status_update(on, i);
  604. }
  605. else
  606. {
  607. rt_printf("dido_do_kz err(on=%d,i=%d).\r\n", on, i);
  608. }
  609. return 0;
  610. }
  611. #if !defined CPU_FUXI
  612. /**
  613. * @brief 开入量更新
  614. * @param[in/out] {uint32_t} _value_new 值
  615. * @param[in/out] {uint8_t} _index 下标
  616. * @param[in/out] {struct timespec *ts} ts 时间
  617. * @return * int
  618. * @retval none
  619. *
  620. * @warning none
  621. * @note none
  622. */
  623. int dido_di_update(uint32_t _value_new, uint8_t _index, struct timespec *ts)
  624. {
  625. }
  626. #endif
  627. /**
  628. * @brief Get the shm didata object
  629. * @return * int
  630. * @retval none
  631. *
  632. * @warning none
  633. * @note none
  634. */
  635. int get_shm_didata(void)
  636. {
  637. uint8_t b_on = 0;
  638. u32 ul_di_state = 0, dt = 0;
  639. int l_read_ret = 0, l_cnt = 0;
  640. struct t_shmdata_di shm_data_di = {0};
  641. static struct t_sd_di_status smt_di[DI_MAX] = {0}, smt_di_bk[DI_MAX] = {0}; /* 数据 */
  642. struct timespec ts;
  643. struct di_struct *pt_di = NULL;
  644. l_read_ret = shm_packet_read_v2(SHM_ADDR_W_DI, sizeof(shm_data_di), (uint8_t *)&shm_data_di, sizeof(shm_data_di));
  645. if (l_read_ret < 0)
  646. {
  647. return -1;
  648. }
  649. if (l_read_ret > 0)
  650. {
  651. memcpy((void *)&smt_di, (void *)&shm_data_di.di, sizeof(struct t_sd_di_status) * DI_MAX);
  652. /* 第一次 数据初始化 */
  653. if (false == g_di[g_di_slot].bInited)
  654. {
  655. for (uint8_t i = 0; i < DI_MAX; i++)
  656. {
  657. pt_di = &g_di_st[g_di_slot][i];
  658. b_on = smt_di[i].uc_state;
  659. if (g_di[g_di_slot].inv[0] & (1 << i))
  660. {
  661. b_on = !b_on;
  662. }
  663. pt_di->b_on = b_on;
  664. ul_di_state |= b_on << i;
  665. sw_di_set(pt_di->owner,
  666. pt_di->type,
  667. b_on ? SW_DI_TYPE_ON : SW_DI_TYPE_OFF);
  668. // 双点的值也需要重新初始化
  669. plc_db_init_value(g_di_slot, i, b_on);
  670. // 遥信时标
  671. g_di[g_di_slot].ts_t[i][g_di[g_di_slot].ts_i[i].n] = 0;
  672. g_di[g_di_slot].ts_v[i][g_di[g_di_slot].ts_i[i].n] = b_on;
  673. g_di[g_di_slot].ts_i[i].n++;
  674. }
  675. memcpy((void *)&smt_di_bk, (void *)&shm_data_di.di, sizeof(struct t_sd_di_status) * DI_MAX);
  676. g_di[g_di_slot].bInited = TRUE;
  677. }
  678. for (uint8_t i = 0; i < DI_MAX; i++)
  679. {
  680. if (smt_di_bk[i].uc_state != smt_di[i].uc_state)
  681. {
  682. pt_di = &g_di_st[g_di_slot][i];
  683. b_on = smt_di[i].uc_state;
  684. if (g_di[g_di_slot].inv[0] & (1 << i))
  685. {
  686. b_on = !b_on;
  687. }
  688. pt_di->b_on = b_on;
  689. ul_di_state |= b_on << i;
  690. transform_msts_to_tts(smt_di[i].ull_timestamp, &ts);
  691. soe_record_yx(i + (g_di_slot << 8), b_on, &ts);
  692. // 更新逻辑模型中的值
  693. sw_di_set(pt_di->owner,
  694. pt_di->type,
  695. (b_on) ? SW_DI_TYPE_ON : SW_DI_TYPE_OFF);
  696. // 更新记录
  697. dt = g_adc_dots_data_count;
  698. dt = ((dt + ADC_REC_DOTS_MASK) - (pt_di->tm_filter * ADC_REC_SAMPLE)) % ADC_REC_DOTS_MASK;
  699. g_di[g_di_slot].ts_t[i][g_di[g_di_slot].ts_i[i].n] = dt;
  700. g_di[g_di_slot].ts_v[i][g_di[g_di_slot].ts_i[i].n] = b_on;
  701. g_di[g_di_slot].ts_i[i].n++;
  702. smt_di_bk[i].uc_state = smt_di[i].uc_state;
  703. // dp_err_n_c_rt("change smt_di[%02d].uc_state = %d", i, b_on);
  704. }
  705. }
  706. g_di[g_di_slot].value[0] = ul_di_state ^ g_di[g_di_slot].inv[0];
  707. // dp_err_n_c_rt("g_di[g_di_slot].inv[0] = %d, g_di[%02d].value[0] = 0x%04x", g_di[g_di_slot].inv[0], g_di_slot, g_di[g_di_slot].value[0]);
  708. // rt_printf("\r\n");
  709. }
  710. return l_read_ret;
  711. }
  712. /*------------------------------ 测试函数 -------------------------------------
  713. 一个实体文件必须带一个本模块的测试函数来进行单元测试,如果的确不方便在本模块中
  714. 进行单元测试,必须在此注明实际的测试位置(例如在哪个实体文件中使用哪个测试函数).
  715. */
  716. // SOE信号发生器控制全局变量
  717. extern int g_soe_gen_state; // 状态,0停止,1初始化,2运行
  718. extern int g_soe_gen_on_us; // 导通时间
  719. extern int g_soe_gen_off_us; // 断开时间
  720. extern int g_soe_gen_seq_us; // 通道序列间隔时间
  721. #define SOE_GEN_NUM 8
  722. u32 g_soe_gen_count;
  723. u32 g_soe_gen_us0[SOE_GEN_NUM];
  724. u8 g_soe_gen_on[SOE_GEN_NUM];
  725. void dido_soe_gen(void)
  726. {
  727. int i;
  728. if (g_soe_gen_state == 0)
  729. {
  730. return;
  731. }
  732. else if (g_soe_gen_state == 1)
  733. {
  734. pit_156us_period_set(78.125);
  735. g_soe_gen_count = 0;
  736. for (i = 0; i < SOE_GEN_NUM; i++)
  737. {
  738. g_soe_gen_us0[i] = g_soe_gen_seq_us * (i);
  739. g_soe_gen_on[i] = 0;
  740. g_soe_gen_state = 2;
  741. }
  742. }
  743. else if (dido_di_is_on(1, 0))
  744. {
  745. for (i = 0; i < SOE_GEN_NUM; i++)
  746. {
  747. if (g_soe_gen_on[i] == 0)
  748. {
  749. if ((int)(g_soe_gen_count - g_soe_gen_us0[i]) >= g_soe_gen_off_us)
  750. {
  751. g_soe_gen_on[i] = 1;
  752. gpio_kout_do(1, i);
  753. g_soe_gen_us0[i] = g_soe_gen_count;
  754. }
  755. }
  756. else
  757. {
  758. if ((int)(g_soe_gen_count - g_soe_gen_us0[i]) >= g_soe_gen_on_us)
  759. {
  760. g_soe_gen_on[i] = 0;
  761. gpio_kout_do(0, i);
  762. g_soe_gen_us0[i] = g_soe_gen_count;
  763. }
  764. }
  765. }
  766. }
  767. else
  768. {
  769. g_soe_gen_count = 0;
  770. for (i = 0; i < SOE_GEN_NUM; i++)
  771. {
  772. g_soe_gen_us0[i] = g_soe_gen_seq_us * (i);
  773. g_soe_gen_on[i] = 0;
  774. g_soe_gen_state = 2;
  775. }
  776. }
  777. g_soe_gen_count++;
  778. }
  779. int dido_printf(void)
  780. {
  781. int i, slot, index;
  782. struct ts_index ts_i;
  783. rt_printf("槽号\t值\r\n");
  784. for (i = 0; i < EQU_SLOT_NUM_MAX; i++) // xj 2015-5-4
  785. {
  786. rt_printf("%02d:\tvalue=0x%08lx_%08lx,inv=0x%08lx_%08lx\r\n", i, g_di[i].value[1], g_di[i].value[0], g_di[i].inv[1], g_di[i].inv[0]);
  787. }
  788. rt_printf("\r\n");
  789. slot = g_di_slot;
  790. index = 1;
  791. ts_i.n = g_di[slot].ts_i[index].n;
  792. rt_printf("开入时间戳记录:slot=%d,index=%d,ts_i=%d.\r\n", slot, index, ts_i.n);
  793. rt_printf("序号\t值\t时间戳\r\n");
  794. ts_i.n = 0;
  795. for (i = 0; i < DIDO_TS_NUM; i++)
  796. {
  797. rt_printf("%02d\t%d\t%08lu\r\n", ts_i.n, g_di[slot].ts_v[index][ts_i.n], g_di[slot].ts_t[index][ts_i.n]);
  798. ts_i.n++;
  799. }
  800. rt_printf("开出状态:0x%04x\r\n\r\n", g_do_status);
  801. rt_stat_printf(&g_stat_di_delay);
  802. rt_stat_printf(&g_stat_do_delay);
  803. return 0;
  804. }
  805. int dido_stat_reset(void)
  806. {
  807. rt_stat_init(&g_stat_di_delay, "di_delay");
  808. rt_stat_init(&g_stat_do_delay, "do_delay");
  809. return 0;
  810. }
  811. int dido_single_test(u8 slot, u8 point)
  812. {
  813. dido_do(slot, (1 << point), 1);
  814. ustimer_delay(250 * USTIMER_MS);
  815. dido_do(slot, (1 << point), 0);
  816. return 0;
  817. }
  818. int dido_test(void)
  819. {
  820. static unsigned long us0 = 0;
  821. static char dido_test_pair[3][2] =
  822. {
  823. {1, 3},
  824. {2, 4},
  825. {5, 5},
  826. };
  827. static char dido_test_di[32] =
  828. {
  829. 0, 1, 2, 3,
  830. 0, 1, 2, 3,
  831. 4, 5, 6, 7,
  832. 4, 5, 6, 7,
  833. 8, 9, 10, 11,
  834. 8, 9, 10, 11,
  835. 12, 13, 10, 11,
  836. 12, 13, 12, 13};
  837. int i, j, k, err_once, di_num;
  838. u32 slot_di, slot_do;
  839. if (g_test_on == 0)
  840. {
  841. return 0;
  842. }
  843. // 5秒处理一次
  844. if (ustimer_delay_origin2(&us0, 10 * USTIMER_SEC) != 1)
  845. {
  846. return 0;
  847. }
  848. rt_printf("dido_test begin...\r\n");
  849. ustimer_delay(1 * USTIMER_SEC);
  850. #if 1
  851. // 测试开出板、开入板
  852. // 开出板逐一开出
  853. rt_printf("dido_test single.\r\n");
  854. err_once = 0;
  855. for (i = 0; i < 3; i++)
  856. {
  857. slot_do = dido_test_pair[i][0];
  858. slot_di = dido_test_pair[i][1];
  859. for (j = 0; j < equ_get_do_num(slot_do); j++)
  860. {
  861. dido_do(slot_do, (1 << j), 1);
  862. ustimer_delay(250 * USTIMER_MS);
  863. di_num = equ_get_di_num(slot_di);
  864. if (slot_di == equ_get_slot_by_type(BOARD_TYPE_AUX))
  865. {
  866. di_num -= 1;
  867. }
  868. for (k = 0; k < di_num; k++)
  869. {
  870. // 是否对应的开出点
  871. if (dido_test_di[k] == j)
  872. {
  873. if ((g_di[slot_di].value[0] & (1 << k)) == 0)
  874. {
  875. err_once++;
  876. error_count++;
  877. rt_printf("dido_test on err(err=%d):i=%d,j=%d,k=%d,di=0x%08lx.\r\n", err_once, i, j, k, g_di[slot_di].value[0]);
  878. }
  879. }
  880. else
  881. {
  882. if ((g_di[slot_di].value[0] & (1 << k)) != 0)
  883. {
  884. err_once++;
  885. error_count++;
  886. rt_printf("dido_test off err(err=%d):i=%d,j=%d,k=%d,di=0x%08lx.\r\n", err_once, i, j, k, g_di[slot_di].value[0]);
  887. }
  888. }
  889. }
  890. dido_do(slot_do, (1 << j), 0);
  891. ustimer_delay(20 * USTIMER_MS);
  892. watchdog_feed_mainloop();
  893. }
  894. }
  895. #endif
  896. rt_printf("\r\n开入变位时间超出次数:%lu\r\n", di_error_count);
  897. rt_printf("name\t\t\tmin\tmax\tavg\tsum\t\tcnt\n");
  898. rt_stat_printf(&g_stat_di_delay);
  899. rt_printf("\r\n");
  900. for (i = 0; i < EQU_SLOT_NUM_MAX; i++)
  901. {
  902. if (g_do_time[i].slot != 0)
  903. {
  904. rt_printf("do_time[%d]:diff_max=%lu,diff_out_count=%lu\r\n",
  905. g_do_time[i].slot, g_do_time[i].diff_max, g_do_time[i].diff_out_count);
  906. }
  907. }
  908. rt_printf("\r\n");
  909. rt_printf("dido_test end:err_once=%d,error_count=%lu!\r\n\r\n\r\n", err_once, error_count);
  910. return 0;
  911. }
  912. bool check_board_di_valid(u8 slot, u8 index)
  913. {
  914. u32 i;
  915. // 初始化取反标志
  916. index -= 1;
  917. for (i = 0; i < g_equ_config->di_num; i++)
  918. {
  919. if ((g_equ_config_di[i].slot == 1) && (g_equ_config_di[i].index == index))
  920. {
  921. if (g_equ_config_di[i].type == 0)
  922. return true;
  923. }
  924. }
  925. return false;
  926. }
  927. bool check_board_kc6_close(u8 slot, u8 index)
  928. {
  929. u32 i;
  930. // 初始化取反标志
  931. index -= 1;
  932. for (i = 0; i < g_equ_config->do_num; i++)
  933. {
  934. if ((g_equ_config_do[i].slot == slot) && (g_equ_config_do[i].index == index))
  935. {
  936. if (g_equ_config_do[i].type != 0)
  937. return true;
  938. }
  939. }
  940. return false;
  941. }
  942. #define DIGROUP 4
  943. #define V3_OUT_NUM 5
  944. int dido_auto_test_for_v3(char *buf, int num)
  945. {
  946. // 12个开入,用5个开出进行测试
  947. static char test_di[6][DIGROUP] =
  948. {
  949. {1, 6, 11, 0},
  950. {2, 7, 12, 0},
  951. {3, 8, 13, 0},
  952. {4, 9, 14, 0},
  953. {16, 0, 0, 0},
  954. {5, 10, 15, 0},
  955. };
  956. int i, j, k, err_once, flag, flag1, di_num, do_num;
  957. u32 slot_di, slot_do, di_result;
  958. u8 tmp[5], tmp_data[5];
  959. u8 di_ok[4] = {0};
  960. u8 do_errflag = 0;
  961. char dido_test_pair[num][2];
  962. unsigned long us_begin = 0;
  963. bool bkc6close = false;
  964. slot_do = 0;
  965. bkc6close = check_board_kc6_close(1, 6);
  966. dido_stat_reset();
  967. if (bkc6close)
  968. {
  969. dido_do(slot_do, (1 << 5), 1); // 开出6若焊接为常闭输出,先开出,打开节点
  970. ustimer_delay(100 * USTIMER_MS);
  971. }
  972. // 取出板卡槽位
  973. for (i = 0; i < (num * 2);)
  974. {
  975. for (j = 0; j < 2; j++)
  976. {
  977. dido_test_pair[i / 2][j] = buf[i];
  978. i++;
  979. }
  980. }
  981. flag = 0;
  982. flag1 = 0;
  983. memset(tmp, 0, sizeof(tmp));
  984. memset(tmp_data, 0, sizeof(tmp_data));
  985. memset(dido_buf, 0, sizeof(dido_buf));
  986. rt_printf("dido_test begin...\r\n");
  987. err_once = 0;
  988. // 以组为循环
  989. for (i = 0; i < num; i++)
  990. {
  991. slot_do = dido_test_pair[i][0];
  992. slot_di = dido_test_pair[i][1];
  993. // 槽位暂存
  994. dido_buf[i + i * 13] = slot_do;
  995. dido_buf[i + i * 13 + 5] = slot_di;
  996. // 取出对应槽位的板卡中DI和DO的数量
  997. di_num = equ_get_di_num(slot_di);
  998. do_num = equ_get_do_num(slot_do);
  999. // dido_do(slot_do,(1<<5),1); //磁保持继电器
  1000. // 启动记时
  1001. us_begin = ustimer_get_origin();
  1002. // 逐一输出开出点
  1003. for (j = 0; j < V3_OUT_NUM; j++)
  1004. {
  1005. dido_do(slot_do, (1 << j), 1);
  1006. if (j == 5 && bkc6close)
  1007. {
  1008. dido_do(slot_do, (1 << 5), 0); // 开出6若焊接为常闭输出,反操作为闭合
  1009. }
  1010. ustimer_delay(150 * USTIMER_MS);
  1011. memset(di_ok, 0, sizeof(di_ok));
  1012. do_errflag = DIGROUP;
  1013. for (k = 0; k < 4; k++)
  1014. {
  1015. bool bIgnore = false;
  1016. di_result = (g_di[slot_di].value[0] ^ g_di[slot_di].inv[0]);
  1017. k += i * 2;
  1018. if (test_di[j][k] == 17)
  1019. {
  1020. bIgnore = check_board_di_valid(slot_di, 17);
  1021. }
  1022. if (test_di[j][k] == 18)
  1023. {
  1024. bIgnore = check_board_di_valid(slot_di, 18);
  1025. }
  1026. if (!test_di[j][k] || bIgnore)
  1027. {
  1028. do_errflag--;
  1029. }
  1030. else
  1031. {
  1032. if (!(di_result & (1 << (test_di[j][k] - 1)))) // 对应开入无值
  1033. {
  1034. do_errflag--;
  1035. di_ok[k] = test_di[j][k];
  1036. }
  1037. }
  1038. }
  1039. if (!do_errflag) // 开出对应的所有开入都没有值,即代表该开出有问题
  1040. {
  1041. if (j < 8)
  1042. {
  1043. dido_buf[i + i * 13 + 1] |= (1 << j); // 开出错误标志
  1044. }
  1045. else if (j < 16)
  1046. {
  1047. dido_buf[i + i * 13 + 2] |= (1 << (j - 8)); // 开出错误标志
  1048. }
  1049. rt_printf("dido_test do err:i=%d,j=%d,di=%d,di_l=0x%04lx,di_h=0x%04lx.\r\n",
  1050. i, j, di_ok[0], g_di[slot_di].value[0], g_di[slot_di].value[1]);
  1051. }
  1052. else
  1053. {
  1054. for (k = 0; k < 4; k++)
  1055. {
  1056. if (!di_ok[k])
  1057. continue;
  1058. if ((di_ok[k] - 1) < 8)
  1059. {
  1060. dido_buf[i + i * 13 + 6] |= (1 << (di_ok[k] - 1));
  1061. }
  1062. else if ((di_ok[k] - 1) < 16)
  1063. {
  1064. dido_buf[i + i * 13 + 7] |= (1 << (di_ok[k] - 1 - 8));
  1065. }
  1066. else if ((di_ok[k] - 1) < 24)
  1067. {
  1068. dido_buf[i + i * 13 + 8] |= (1 << (di_ok[k] - 1 - 16));
  1069. }
  1070. else if ((di_ok[k] - 1) < 32)
  1071. {
  1072. dido_buf[i + i * 13 + 9] |= (1 << (di_ok[k] - 1 - 24));
  1073. }
  1074. else if ((di_ok[k] - 1) < 40)
  1075. {
  1076. dido_buf[i + i * 13 + 10] |= (1 << (di_ok[k] - 1 - 32));
  1077. }
  1078. else if ((di_ok[k] - 1) < 48)
  1079. {
  1080. dido_buf[i + i * 13 + 11] |= (1 << (di_ok[k] - 1 - 40));
  1081. }
  1082. else if ((di_ok[k] - 1) < 56)
  1083. {
  1084. dido_buf[i + i * 13 + 12] |= (1 << (di_ok[k] - 1 - 48));
  1085. }
  1086. else if ((di_ok[k] - 1) < 64)
  1087. {
  1088. dido_buf[i + i * 13 + 13] |= (1 << (di_ok[k] - 1 - 56));
  1089. }
  1090. rt_printf("dido_test di err:i=%d,j=%d,di=%d,di_l=0x%04lx,di_h=0x%04lx.\r\n",
  1091. i, j, di_ok[k], g_di[slot_di].value[0], g_di[slot_di].value[1]);
  1092. }
  1093. }
  1094. dido_do(slot_do, (1 << j), 0);
  1095. if (j == 5 && bkc6close)
  1096. {
  1097. dido_do(slot_do, (1 << 5), 1); // 开出6若焊接为常闭输出,反操作为闭合
  1098. }
  1099. // if(j==4)
  1100. //{
  1101. // dido_do(slot_do,(1<<5),1); //磁保持继电器
  1102. //}
  1103. ustimer_delay(20 * USTIMER_MS);
  1104. watchdog_feed_mainloop();
  1105. }
  1106. }
  1107. for (i = 0; i < EQU_SLOT_NUM_MAX; i++)
  1108. {
  1109. if (g_do_time[i].slot != 0)
  1110. {
  1111. rt_printf("do_time[%d]:diff_max=%lu,diff_out_count=%lu\r\n",
  1112. g_do_time[i].slot, g_do_time[i].diff_max, g_do_time[i].diff_out_count);
  1113. }
  1114. }
  1115. rt_printf("dido_test end:err_once=%d,error_count=%lu!\r\n\r\n\r\n", err_once, error_count);
  1116. return 0;
  1117. }
  1118. int dido_auto_test(char *buf, int num)
  1119. {
  1120. #define KC_COUNT 11
  1121. // 12个开入,用5个开出进行测试
  1122. #if 0
  1123. static char test_di[6][DIGROUP] =
  1124. {
  1125. {1, 6, 11,0},
  1126. {2, 7, 12,0},
  1127. {3, 8, 13,0},
  1128. {4, 9, 14,0},
  1129. {16, 0, 0,0},
  1130. {5, 10, 15,0},
  1131. };
  1132. #endif
  1133. static char test_di[KC_COUNT][DIGROUP] =
  1134. {
  1135. {1, 12, 23, 0},
  1136. {2, 13, 24, 0},
  1137. {3, 14, 0, 0},
  1138. {4, 15, 0, 0},
  1139. {5, 16, 0, 0},
  1140. {6, 17, 0, 0},
  1141. {7, 18, 0, 0},
  1142. {8, 19, 0, 0},
  1143. {9, 20, 0, 0},
  1144. {10, 21, 0, 0},
  1145. {11, 22, 0, 0},
  1146. // { 0, 0, 0,0}
  1147. };
  1148. u8 brd_v3[] = {22, 23};
  1149. int i, j, k, err_once, flag, flag1, di_num, do_num;
  1150. u32 slot_di, slot_do, di_result;
  1151. u8 tmp[5], tmp_data[5];
  1152. u8 di_ok[4] = {0};
  1153. u8 do_errflag = 0;
  1154. char dido_test_pair[num][2];
  1155. unsigned long us_begin = 0;
  1156. bool bkc6close = false;
  1157. rt_printf("g_brd_type_kz = %lu\r\n ", g_brd_type_kz);
  1158. rt_printf("g_equ_config->do_num = %lu\r\n ", g_equ_config->do_num);
  1159. for (i = 0; i < (int)sizeof(brd_v3); i++)
  1160. {
  1161. if (g_brd_type_kz + BOARD_TYPE_DFTU_KC00 == brd_v3[i]) //
  1162. {
  1163. rt_printf("auto_test_for_v3\r\n ");
  1164. return (dido_auto_test_for_v3(buf, num));
  1165. }
  1166. }
  1167. rt_printf("auto_test_for_v4\r\n ");
  1168. // bkc6close=check_board_kc6_close(1,6);
  1169. dido_stat_reset();
  1170. if (bkc6close)
  1171. {
  1172. dido_do(1, (1 << 5), 1); // 开出6若焊接为常闭输出,先开出,打开节点
  1173. ustimer_delay(100 * USTIMER_MS);
  1174. }
  1175. // 取出板卡槽位
  1176. for (i = 0; i < (num * 2);)
  1177. {
  1178. for (j = 0; j < 2; j++)
  1179. {
  1180. dido_test_pair[i / 2][j] = buf[i];
  1181. i++;
  1182. }
  1183. }
  1184. flag = 0;
  1185. flag1 = 0;
  1186. memset(tmp, 0, sizeof(tmp));
  1187. memset(tmp_data, 0, sizeof(tmp_data));
  1188. memset(dido_buf, 0, sizeof(dido_buf));
  1189. rt_printf("dido_test begin...\r\n");
  1190. err_once = 0;
  1191. // 以组为循环
  1192. for (i = 0; i < num; i++)
  1193. {
  1194. slot_do = dido_test_pair[i][0];
  1195. slot_di = dido_test_pair[i][1];
  1196. // 槽位暂存
  1197. dido_buf[i + i * 13] = slot_do;
  1198. dido_buf[i + i * 13 + 5] = slot_di;
  1199. // 取出对应槽位的板卡中DI和DO的数量
  1200. di_num = equ_get_di_num(slot_di);
  1201. do_num = equ_get_do_num(slot_do);
  1202. // dido_do(slot_do,(1<<5),1); //磁保持继电器
  1203. // 启动记时
  1204. us_begin = ustimer_get_origin();
  1205. // 逐一输出开出点
  1206. for (j = 0; j < KC_COUNT; j++)
  1207. {
  1208. dido_do(slot_do, (1 << j), 1);
  1209. if (j == 5 && bkc6close)
  1210. {
  1211. dido_do(slot_do, (1 << 5), 0); // 开出6若焊接为常闭输出,反操作为闭合
  1212. }
  1213. ustimer_delay(150 * USTIMER_MS);
  1214. memset(di_ok, 0, sizeof(di_ok));
  1215. do_errflag = DIGROUP;
  1216. for (k = 0; k < 4; k++)
  1217. {
  1218. bool bIgnore = false;
  1219. di_result = (g_di[slot_di].value[0] ^ g_di[slot_di].inv[0]);
  1220. k += i * 2;
  1221. if (test_di[j][k] == 17)
  1222. {
  1223. bIgnore = check_board_di_valid(slot_di, 17);
  1224. }
  1225. if (test_di[j][k] == 18)
  1226. {
  1227. bIgnore = check_board_di_valid(slot_di, 18);
  1228. }
  1229. if (!test_di[j][k] || bIgnore)
  1230. {
  1231. do_errflag--;
  1232. }
  1233. else
  1234. {
  1235. if (!(di_result & (1 << (test_di[j][k] - 1)))) // 对应开入无值
  1236. {
  1237. do_errflag--;
  1238. di_ok[k] = test_di[j][k];
  1239. }
  1240. }
  1241. }
  1242. if (!do_errflag) // 开出对应的所有开入都没有值,即代表该开出有问题
  1243. {
  1244. if (j < 8)
  1245. {
  1246. dido_buf[i + i * 13 + 1] |= (1 << j); // 开出错误标志
  1247. }
  1248. else if (j < 16)
  1249. {
  1250. dido_buf[i + i * 13 + 2] |= (1 << (j - 8)); // 开出错误标志
  1251. }
  1252. rt_printf("dido_test do err:i=%d,j=%d,di=%d,di_l=0x%04lx,di_h=0x%04lx.\r\n",
  1253. i, j, di_ok[0], g_di[slot_di].value[0], g_di[slot_di].value[1]);
  1254. }
  1255. else
  1256. {
  1257. for (k = 0; k < 4; k++)
  1258. {
  1259. if (!di_ok[k])
  1260. continue;
  1261. if ((di_ok[k] - 1) < 8)
  1262. {
  1263. dido_buf[i + i * 13 + 6] |= (1 << (di_ok[k] - 1));
  1264. }
  1265. else if ((di_ok[k] - 1) < 16)
  1266. {
  1267. dido_buf[i + i * 13 + 7] |= (1 << (di_ok[k] - 1 - 8));
  1268. }
  1269. else if ((di_ok[k] - 1) < 24)
  1270. {
  1271. dido_buf[i + i * 13 + 8] |= (1 << (di_ok[k] - 1 - 16));
  1272. }
  1273. else if ((di_ok[k] - 1) < 32)
  1274. {
  1275. dido_buf[i + i * 13 + 9] |= (1 << (di_ok[k] - 1 - 24));
  1276. }
  1277. else if ((di_ok[k] - 1) < 40)
  1278. {
  1279. dido_buf[i + i * 13 + 10] |= (1 << (di_ok[k] - 1 - 32));
  1280. }
  1281. else if ((di_ok[k] - 1) < 48)
  1282. {
  1283. dido_buf[i + i * 13 + 11] |= (1 << (di_ok[k] - 1 - 40));
  1284. }
  1285. else if ((di_ok[k] - 1) < 56)
  1286. {
  1287. dido_buf[i + i * 13 + 12] |= (1 << (di_ok[k] - 1 - 48));
  1288. }
  1289. else if ((di_ok[k] - 1) < 64)
  1290. {
  1291. dido_buf[i + i * 13 + 13] |= (1 << (di_ok[k] - 1 - 56));
  1292. }
  1293. rt_printf("dido_test di err:i=%d,j=%d,di=%d,di_l=0x%04lx,di_h=0x%04lx.\r\n",
  1294. i, j, di_ok[k], g_di[slot_di].value[0], g_di[slot_di].value[1]);
  1295. }
  1296. }
  1297. dido_do(slot_do, (1 << j), 0);
  1298. if (j == 5 && bkc6close)
  1299. {
  1300. dido_do(slot_do, (1 << 5), 1); // 开出6若焊接为常闭输出,反操作为闭合
  1301. }
  1302. // if(j==4)
  1303. //{
  1304. // dido_do(slot_do,(1<<5),1); //磁保持继电器
  1305. //}
  1306. ustimer_delay(20 * USTIMER_MS);
  1307. watchdog_feed_mainloop();
  1308. }
  1309. }
  1310. for (i = 0; i < EQU_SLOT_NUM_MAX; i++)
  1311. {
  1312. if (g_do_time[i].slot != 0)
  1313. {
  1314. rt_printf("do_time[%d]:diff_max=%lu,diff_out_count=%lu\r\n",
  1315. g_do_time[i].slot, g_do_time[i].diff_max, g_do_time[i].diff_out_count);
  1316. }
  1317. }
  1318. rt_printf("dido_test end:err_once=%d,error_count=%lu!\r\n\r\n\r\n", err_once, error_count);
  1319. return 0;
  1320. }
  1321. /*------------------------------ 文件结束 -------------------------------------
  1322. */