dido.c 40 KB

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