RelayBoard.c 56 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358
  1. #include <stdio.h> /*標準輸入輸出定義*/
  2. #include <stdlib.h> /*標準函數庫定義*/
  3. #include <string.h>
  4. #include <stdint.h>
  5. #include <time.h>
  6. #include <unistd.h>
  7. #include <sys/time.h>
  8. #include <sys/timeb.h>
  9. #include "../ShareMemory/shmMem.h"
  10. #include "../Config.h"
  11. #include "../Log/log.h"
  12. #include "Module_InternalComm.h"
  13. #include "internalComm.h"
  14. //------------------------------------------------------------------------------
  15. static struct SysConfigData *pSysConfig = NULL;
  16. static struct SysInfoData *pSysInfo = NULL;
  17. static struct AlarmCodeData *pAlarmCode = NULL;
  18. static struct RelayModuleData *ShmRelayModuleData = NULL;
  19. static struct PsuData *ShmPsuData = NULL;
  20. static struct PrimaryMcuData *ShmPrimaryMcuData = NULL;
  21. static DcCommonInfo *ShmDcCommonData = NULL;
  22. static Relay outputRelay = {0};
  23. static Relay regRelay = {0};
  24. static int Uart5Fd = 0;
  25. //static bool _isRelayWelding[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY];
  26. //static struct timeval _checkRelayWeldingTimer[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY];
  27. //static bool _isOutputNoneMatch[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY];
  28. //static struct timeval _checkOutputNoneMatchTimer[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY];
  29. static bool _isOvpChkTimeFlag[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY]; //DS60-120 add
  30. static struct timeval _checkOutputVolProtectTimer[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY]; //DS60-120 add
  31. static struct timeval _close_ac_contactor;
  32. //------------------------------------------------------------------------------
  33. static void RunForceStopProcess(void)
  34. {
  35. static bool isCriticalStop = NO;
  36. static struct timeval _psuCriticalStop;
  37. uint32_t _timebuf;
  38. if (isCriticalStop == NO) {
  39. isCriticalStop = YES;
  40. gettimeofday(&_psuCriticalStop, NULL);
  41. } else {
  42. _timebuf = GetTimeoutValue(_psuCriticalStop);
  43. if (_timebuf < 0) {
  44. gettimeofday(&_psuCriticalStop, NULL);
  45. } else {
  46. if (_timebuf / 1000 >= (FORCE_STOP_TIME * 1000)) {
  47. isCriticalStop = NO;
  48. pAlarmCode->AlarmEvents.bits.PsuFailureAlarm = NORMAL;
  49. }
  50. }
  51. }
  52. }
  53. static void StopCheckRelayInfo(uint8_t _chkIndex)
  54. {
  55. if (ShmDcCommonData->CheckRelayStatus[_chkIndex] != STOP) {
  56. ShmDcCommonData->CheckRelayStatus[_chkIndex] = STOP;
  57. }
  58. }
  59. static void StartCheckRelayInfo(uint8_t _chkIndex, uint8_t toState)
  60. {
  61. // SMR1 *2 + SMR2 * 2 + Parallel * 2
  62. static time_t lastCheckRelayStateTimer[6] = {0};
  63. time_t nowTime = {0};
  64. //uint8_t *pCheckRelayState = (uint8_t *)ShmDcCommonData->CheckRelayStatus[_chkIndex];
  65. if (ShmDcCommonData->CheckRelayStatus[_chkIndex] == STOP) {
  66. time(&lastCheckRelayStateTimer[_chkIndex]);
  67. ShmDcCommonData->CheckRelayStatus[_chkIndex] = START;
  68. } else {
  69. time(&nowTime);
  70. if (nowTime - lastCheckRelayStateTimer[_chkIndex] >= 1) {
  71. //log_info("relay welding or driving fault = %d \n", _chkIndex);
  72. if (toState == 1) {
  73. ShmDcCommonData->CheckRelayStatus[_chkIndex] = RELAY_STATUS_ERROR_DRIVING;
  74. } else {
  75. ShmDcCommonData->CheckRelayStatus[_chkIndex] = RELAY_STATUS_ERROR_WELDING;
  76. }
  77. lastCheckRelayStateTimer[_chkIndex] = nowTime;
  78. }
  79. }
  80. }
  81. static uint8_t getCommTargetID(uint8_t index)
  82. {
  83. uint8_t targetID = 0;
  84. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  85. if (pSysConfig->TotalConnectorCount == 1) {
  86. if (strncmp((char *)&pSysConfig->ModelName[7], "0", 1) != 0) {
  87. targetID = 0x01;
  88. } else if (strncmp((char *)&pSysConfig->ModelName[9], "0", 1) != 0) {
  89. targetID = 0x02;
  90. }
  91. } else {
  92. targetID = pDcChargingInfo->Evboard_id;
  93. }
  94. return targetID;
  95. }
  96. /*static void MatchRelayStatus(void)
  97. {
  98. // 因為 AC Contactor 沒有 Feedback,所以暫時先這樣處理
  99. //regRelay.relay_event.bits.AC_Contactor = outputRelay.relay_event.bits.AC_Contactor;
  100. //pSysInfo->AcContactorStatus =
  101. // regRelay.relay_event.bits.AC_Contactor =
  102. // outputRelay.relay_event.bits.AC_Contactor;
  103. regRelay.relay_event.bits.CCS_Precharge = outputRelay.relay_event.bits.CCS_Precharge;
  104. regRelay.relay_event.bits.Gun1_P = outputRelay.relay_event.bits.Gun1_P;
  105. regRelay.relay_event.bits.Gun1_N = outputRelay.relay_event.bits.Gun1_N;
  106. regRelay.relay_event.bits.Gun2_P = outputRelay.relay_event.bits.Gun2_P;
  107. regRelay.relay_event.bits.Gun2_N = outputRelay.relay_event.bits.Gun2_N;
  108. regRelay.relay_event.bits.Gun1_Parallel_P = outputRelay.relay_event.bits.Gun1_Parallel_P;
  109. regRelay.relay_event.bits.Gun1_Parallel_N = outputRelay.relay_event.bits.Gun1_Parallel_N;
  110. }
  111. */
  112. static bool IsNoneMatchRelayStatus(void)
  113. {
  114. bool result = false;
  115. if (
  116. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  117. (regRelay.relay_event.bits.AC_Contactor != outputRelay.relay_event.bits.AC_Contactor) ||
  118. (regRelay.relay_event.bits.CCS_Precharge != outputRelay.relay_event.bits.CCS_Precharge) ||
  119. #endif //!defined DD360 && !defined DD360Audi
  120. (regRelay.relay_event.bits.Gun1_P != outputRelay.relay_event.bits.Gun1_P) ||
  121. (regRelay.relay_event.bits.Gun1_N != outputRelay.relay_event.bits.Gun1_N) ||
  122. (regRelay.relay_event.bits.Gun2_P != outputRelay.relay_event.bits.Gun2_P) ||
  123. (regRelay.relay_event.bits.Gun2_N != outputRelay.relay_event.bits.Gun2_N)
  124. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  125. ||
  126. (regRelay.relay_event.bits.Gun1_Parallel_P != outputRelay.relay_event.bits.Gun1_Parallel_P) ||
  127. (regRelay.relay_event.bits.Gun1_Parallel_N != outputRelay.relay_event.bits.Gun1_Parallel_N)
  128. #endif //!defined DD360 && !defined DD360Audi
  129. ) {
  130. result = true;
  131. }
  132. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  133. if (regRelay.relay_event.bits.AC_Contactor != outputRelay.relay_event.bits.AC_Contactor) {
  134. log_info("AC Contact Relay none match. \n");
  135. }
  136. if (regRelay.relay_event.bits.CCS_Precharge != outputRelay.relay_event.bits.CCS_Precharge) {
  137. log_info("CCS Precharge Relay none match. \n");
  138. }
  139. #endif //
  140. if (regRelay.relay_event.bits.Gun1_P != outputRelay.relay_event.bits.Gun1_P) {
  141. //log_info("SMR1:D+ Relay none match. \n");
  142. StartCheckRelayInfo(RELAY_SMR1_P_STATUS, outputRelay.relay_event.bits.Gun1_P);
  143. } else {
  144. StopCheckRelayInfo(RELAY_SMR1_P_STATUS);
  145. }
  146. if (regRelay.relay_event.bits.Gun1_N != outputRelay.relay_event.bits.Gun1_N) {
  147. //log_info("SMR1:D- Relay none match. \n");
  148. StartCheckRelayInfo(RELAY_SMR1_N_STATUS, outputRelay.relay_event.bits.Gun1_N);
  149. } else {
  150. StopCheckRelayInfo(RELAY_SMR1_N_STATUS);
  151. }
  152. if (regRelay.relay_event.bits.Gun2_P != outputRelay.relay_event.bits.Gun2_P) {
  153. //log_info("SMR2:D+ Relay none match. \n");
  154. StartCheckRelayInfo(RELAY_SMR2_P_STATUS, outputRelay.relay_event.bits.Gun2_P);
  155. } else {
  156. StopCheckRelayInfo(RELAY_SMR2_P_STATUS);
  157. }
  158. if (regRelay.relay_event.bits.Gun2_N != outputRelay.relay_event.bits.Gun2_N) {
  159. //log_info("SMR2:D- Relay none match. \n");
  160. StartCheckRelayInfo(RELAY_SMR2_N_STATUS, outputRelay.relay_event.bits.Gun2_N);
  161. } else {
  162. StopCheckRelayInfo(RELAY_SMR2_N_STATUS);
  163. }
  164. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  165. if (regRelay.relay_event.bits.Gun1_Parallel_P != outputRelay.relay_event.bits.Gun1_Parallel_P) {
  166. //log_info("Parallel:D+ Relay none match. \n");
  167. StartCheckRelayInfo(RELAY_PARA_P_STATUS, outputRelay.relay_event.bits.Gun1_Parallel_P);
  168. } else {
  169. StopCheckRelayInfo(RELAY_PARA_P_STATUS);
  170. }
  171. if (regRelay.relay_event.bits.Gun1_Parallel_N != outputRelay.relay_event.bits.Gun1_Parallel_N) {
  172. //log_info("Parallel:D- Relay none match. \n");
  173. StartCheckRelayInfo(RELAY_PARA_N_STATUS, outputRelay.relay_event.bits.Gun1_Parallel_N);
  174. } else {
  175. StopCheckRelayInfo(RELAY_PARA_N_STATUS);
  176. }
  177. #endif //
  178. return result;
  179. }
  180. static void SetParalleRelayStatus(void)
  181. {
  182. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  183. return;
  184. #endif //!defined DD360 || !defined DD360Audi || !defined DD360ComBox
  185. struct ChargingInfoData *pDcChargingInfo0 = (struct ChargingInfoData *)GetDcChargingInfoData(0);
  186. struct ChargingInfoData *pDcChargingInfo1 = (struct ChargingInfoData *)GetDcChargingInfoData(1);
  187. // 之後雙槍單模機種,橋接都會上
  188. if (pSysConfig->TotalConnectorCount >= 2) {
  189. if (pDcChargingInfo0->SystemStatus == S_BOOTING || pDcChargingInfo1->SystemStatus == S_BOOTING ||
  190. (pDcChargingInfo0->SystemStatus == S_IDLE && pDcChargingInfo1->SystemStatus == S_IDLE)) {
  191. // 初始化~ 不搭橋接
  192. if (regRelay.relay_event.bits.Gun1_Parallel_P == YES) {
  193. outputRelay.relay_event.bits.Gun1_Parallel_P = NO;
  194. } else if (regRelay.relay_event.bits.Gun1_Parallel_N == YES) {
  195. outputRelay.relay_event.bits.Gun1_Parallel_N = NO;
  196. }
  197. } else {
  198. if (pDcChargingInfo0->IsReadyToCharging == YES ||
  199. pDcChargingInfo1->IsReadyToCharging == YES) {
  200. // ************需考慮在切換中 - 切開 relay 與搭回 relay 的時機點************
  201. if (pSysInfo->MainChargingMode == _MAIN_CHARGING_MODE_MAX) {
  202. if (pSysInfo->ReAssignedFlag < _REASSIGNED_RELAY_M_TO_A) {
  203. // 最大充 - 搭上橋接
  204. if (regRelay.relay_event.bits.Gun1_Parallel_N == NO) {
  205. outputRelay.relay_event.bits.Gun1_Parallel_N = YES;
  206. } else if (regRelay.relay_event.bits.Gun1_Parallel_P == NO) {
  207. outputRelay.relay_event.bits.Gun1_Parallel_P = YES;
  208. }
  209. } else {
  210. // 平均充 - 不搭
  211. if (regRelay.relay_event.bits.Gun1_Parallel_P == YES) {
  212. outputRelay.relay_event.bits.Gun1_Parallel_P = NO;
  213. } else if (regRelay.relay_event.bits.Gun1_Parallel_N == YES) {
  214. outputRelay.relay_event.bits.Gun1_Parallel_N = NO;
  215. }
  216. }
  217. } else if (pSysInfo->MainChargingMode == _MAIN_CHARGING_MODE_AVER) {
  218. if (pSysInfo->ReAssignedFlag < _REASSIGNED_RELAY_A_TO_M) {
  219. // 平均充 - 不搭
  220. if (regRelay.relay_event.bits.Gun1_Parallel_P == YES) {
  221. outputRelay.relay_event.bits.Gun1_Parallel_P = NO;
  222. } else if (regRelay.relay_event.bits.Gun1_Parallel_N == YES) {
  223. outputRelay.relay_event.bits.Gun1_Parallel_N = NO;
  224. }
  225. } else {
  226. // 最大充 - 搭上橋接
  227. if (regRelay.relay_event.bits.Gun1_Parallel_N == NO) {
  228. outputRelay.relay_event.bits.Gun1_Parallel_N = YES;
  229. } else if (regRelay.relay_event.bits.Gun1_Parallel_P == NO) {
  230. outputRelay.relay_event.bits.Gun1_Parallel_P = YES;
  231. }
  232. }
  233. }
  234. }
  235. }
  236. }
  237. }
  238. static void GetGfdAdc(void)
  239. {
  240. int gunIndex = 0;
  241. uint8_t targetID = 0;
  242. struct ChargingInfoData *pDcChargingInfo = NULL;
  243. Gfd gfd_adc = {0};
  244. // define : 每 0.2 ~ 1 秒一次
  245. // occur : <= 75k 歐姆 @ 150 - 750 Vdc
  246. // warning : >= 100 歐姆 && <= 500 歐姆 @ 150-750 Vdc
  247. if (Query_Gfd_Adc(Uart5Fd, ADDR_RELAY, &gfd_adc) == PASS) {
  248. for (gunIndex = 0; gunIndex < pSysConfig->TotalConnectorCount; gunIndex++) {
  249. pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(gunIndex);
  250. if (pDcChargingInfo->Type == 0x09 &&
  251. !pSysConfig->AlwaysGfdFlag
  252. ) {
  253. if ((pDcChargingInfo->PresentChargingVoltage * 10) >= VOUT_MIN_VOLTAGE) {
  254. pDcChargingInfo->GroundFaultStatus = GFD_PASS;
  255. }
  256. continue;
  257. }
  258. targetID = getCommTargetID(gunIndex);
  259. if (targetID == 0x01) {
  260. //if (gfd_adc.result_conn1 == GFD_WARNING) {
  261. // gfd_adc.result_conn1 = GFD_PASS;
  262. //}
  263. pDcChargingInfo->GroundFaultStatus = gfd_adc.result_conn1;
  264. //log_info("GFD ******** Result = %d, Step = %d, R = %d, Vol = %d \n",
  265. // pDcChargingInfo->GroundFaultStatus,
  266. // gfd_adc.rb_step_1,
  267. // gfd_adc.Resister_conn1,
  268. // gfd_adc.voltage_conn1);
  269. if (pDcChargingInfo->GroundFaultStatus == GFD_FAIL) {
  270. log_info("GFD Fail. index = %d, Step = %d, R = %d, Vol = %d \n",
  271. gunIndex,
  272. gfd_adc.rb_step_1,
  273. gfd_adc.Resister_conn1,
  274. gfd_adc.voltage_conn1);
  275. } else if (pDcChargingInfo->GroundFaultStatus == GFD_PASS ||
  276. pDcChargingInfo->GroundFaultStatus == GFD_WARNING
  277. ) {
  278. if (pDcChargingInfo->GroundFaultStatus == GFD_WARNING) {
  279. log_info("GFD Warning. index = %d, Result = %d, R = %d, Vol = %d \n",
  280. gunIndex,
  281. pDcChargingInfo->GroundFaultStatus,
  282. gfd_adc.Resister_conn1,
  283. gfd_adc.voltage_conn1);
  284. }
  285. }
  286. } else if (targetID == 0x02) {
  287. //if (gfd_adc.result_conn2 == GFD_WARNING) {
  288. // gfd_adc.result_conn2 = GFD_PASS;
  289. //}
  290. pDcChargingInfo->GroundFaultStatus = gfd_adc.result_conn2;
  291. if (pDcChargingInfo->GroundFaultStatus == GFD_FAIL) {
  292. log_info("GFD Fail. index = %d, Step = %d, R = %d, Vol = %d \n",
  293. gunIndex,
  294. gfd_adc.rb_step_2,
  295. gfd_adc.Resister_conn2,
  296. gfd_adc.voltage_conn2);
  297. } else if (pDcChargingInfo->GroundFaultStatus == GFD_PASS ||
  298. pDcChargingInfo->GroundFaultStatus == GFD_WARNING
  299. ) {
  300. if (pDcChargingInfo->GroundFaultStatus == GFD_WARNING) {
  301. log_info("GFD Warning. index = %d, Result = %d, R = %d, Vol = %d \n",
  302. gunIndex,
  303. pDcChargingInfo->GroundFaultStatus,
  304. gfd_adc.Resister_conn1,
  305. gfd_adc.voltage_conn1);
  306. }
  307. }
  308. }
  309. }
  310. }
  311. }
  312. void CheckOutputPowerOverCarReq(uint8_t index)
  313. {
  314. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  315. float fireV = pDcChargingInfo->FireChargingVoltage;
  316. float carV = pDcChargingInfo->EvBatteryMaxVoltage * 10;
  317. if ((pDcChargingInfo->EvBatterytargetVoltage * 10) > 1500 &&
  318. (pDcChargingInfo->Type == _Type_Chademo ||
  319. pDcChargingInfo->Type == _Type_CCS_2 ||
  320. pDcChargingInfo->Type == _Type_GB)) {
  321. if (fireV >= (carV + (carV * 0.02))) {
  322. if (!_isOvpChkTimeFlag[index]) {
  323. if ((pDcChargingInfo->PresentChargingVoltage * 10) >= VOUT_MIN_VOLTAGE * 10) {
  324. gettimeofday(&_checkOutputVolProtectTimer[index], NULL);
  325. _isOvpChkTimeFlag[index] = YES;
  326. }
  327. } else {
  328. log_info("[Module_InternalComm]CheckOutputPowerOverCarReq NG : fire = %f, battery = %f \n",
  329. pDcChargingInfo->FireChargingVoltage,
  330. (pDcChargingInfo->EvBatterytargetVoltage * 10));
  331. log_error("[Module_InternalComm]CheckOutputPowerOverCarReq NG : fire = %f, battery = %f \n",
  332. pDcChargingInfo->FireChargingVoltage,
  333. (pDcChargingInfo->EvBatterytargetVoltage * 10));
  334. if ((GetTimeoutValue(_checkOutputVolProtectTimer[index]) / 1000) >= OUTPUT_VOL_CHK_TIME) {
  335. if (pDcChargingInfo->Type == _Type_Chademo) {
  336. pAlarmCode->AlarmEvents.bits.SystemChademoOutputOVP = YES;
  337. } else if (pDcChargingInfo->Type == _Type_CCS_2) {
  338. pAlarmCode->AlarmEvents.bits.SystemCcsOutputOVP = YES;
  339. } else if (pDcChargingInfo->Type == _Type_GB) {
  340. pAlarmCode->AlarmEvents.bits.SystemGbOutputOVP = YES;
  341. }
  342. pDcChargingInfo->StopChargeFlag = YES;
  343. }
  344. }
  345. } else {
  346. if (_isOvpChkTimeFlag[index] == YES) {
  347. _isOvpChkTimeFlag[index] = NO;
  348. }
  349. }
  350. }
  351. }
  352. void ResetDetAlarmStatus(uint8_t gun)
  353. {
  354. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(gun);
  355. if (pDcChargingInfo->Type == _Type_Chademo) {
  356. if (pAlarmCode->AlarmEvents.bits.SystemChademoOutputOVP == YES) {
  357. pAlarmCode->AlarmEvents.bits.SystemChademoOutputOVP = NO;
  358. }
  359. } else if (pDcChargingInfo->Type == _Type_GB) {
  360. if (pAlarmCode->AlarmEvents.bits.SystemGbOutputOVP == YES) {
  361. pAlarmCode->AlarmEvents.bits.SystemGbOutputOVP = NO;
  362. }
  363. } else if (pDcChargingInfo->Type == _Type_CCS_2) {
  364. if (pAlarmCode->AlarmEvents.bits.SystemCcsOutputOVP == YES) {
  365. pAlarmCode->AlarmEvents.bits.SystemCcsOutputOVP = NO;
  366. }
  367. }
  368. }
  369. void CheckAcInputOvpStatus(uint8_t index)
  370. {
  371. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  372. if (pAlarmCode->AlarmEvents.bits.SystemL1InputOVP == YES ||
  373. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP == YES ||
  374. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP == YES) {
  375. // if ((pDcChargingInfo->SystemStatus >= S_PREPARNING && pDcChargingInfo->SystemStatus <= S_CHARGING) ||
  376. // (pDcChargingInfo->SystemStatus >= S_CCS_PRECHARGE_ST0 && pDcChargingInfo->SystemStatus <= S_CCS_PRECHARGE_ST1))
  377. // {
  378. // if (pSysInfo->ChargerType == _CHARGER_TYPE_IEC)
  379. // {
  380. // if (_psuInputVolR > VIN_MAX_VOLTAGE_IEC ||
  381. // _psuInputVolS > VIN_MAX_VOLTAGE_IEC ||
  382. // _psuInputVolT > VIN_MAX_VOLTAGE_IEC)
  383. // {
  384. // log_info("IEC _psuInputVolR = %f, _psuInputVolS = %f, _psuInputVolT = %f \n",
  385. // _psuInputVolR, _psuInputVolS, _psuInputVolT);
  386. // pDcChargingInfo->StopChargeFlag = YES;
  387. // }
  388. //
  389. // }
  390. // else if (pSysInfo->ChargerType == _CHARGER_TYPE_UL)
  391. // {
  392. // if (_psuInputVolR > VIN_MAX_VOLTAGE_UL ||
  393. // _psuInputVolS > VIN_MAX_VOLTAGE_UL ||
  394. // _psuInputVolT > VIN_MAX_VOLTAGE_UL)
  395. // {
  396. // log_info("UL _psuInputVolR = %f, _psuInputVolS = %f, _psuInputVolT = %f \n",
  397. // _psuInputVolR, _psuInputVolS, _psuInputVolT);
  398. // pDcChargingInfo->StopChargeFlag = YES;
  399. // }
  400. // }
  401. // }
  402. // else
  403. //log_info("CheckAcInputOvpStatus\r\n");
  404. pDcChargingInfo->StopChargeFlag = YES;
  405. }
  406. }
  407. //void CheckOutputVolNoneMatchFire(uint8_t index)
  408. //{
  409. // struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  410. //
  411. // if ((pDcChargingInfo->EvBatterytargetVoltage * 10) > 1500 &&
  412. // (pDcChargingInfo->Type == _Type_Chademo ||
  413. // pDcChargingInfo->Type == _Type_CCS_2 ||
  414. // pDcChargingInfo->Type == _Type_GB)) {
  415. // if (((pDcChargingInfo->PresentChargingVoltage * 10) < pDcChargingInfo->FireChargingVoltage - 300) ||
  416. // ((pDcChargingInfo->PresentChargingVoltage * 10) > pDcChargingInfo->FireChargingVoltage + 300)) {
  417. // if (!_isOutputNoneMatch[index]) {
  418. // _isOutputNoneMatch[index] = YES;
  419. // gettimeofday(&_checkOutputNoneMatchTimer[index], NULL);
  420. // } else {
  421. // if ((GetTimeoutValue(_checkOutputNoneMatchTimer[index]) / 1000) >= 5000) {
  422. // /*log_info("[Module_InternalComm]CheckOutputVolNoneMatchFire NG (%d) : pre = %f, fire = %f \n",
  423. // index, (pDcChargingInfo->PresentChargingVoltage * 10), pDcChargingInfo->FireChargingVoltage);
  424. // log_error("[Module_InternalComm]CheckOutputVolNoneMatchFire NG (%d): pre = %f, fire = %f \n",
  425. // index, (pDcChargingInfo->PresentChargingVoltage * 10), pDcChargingInfo->FireChargingVoltage);
  426. // pDcChargingInfo->StopChargeFlag = YES;*/
  427. // }
  428. // }
  429. // } else {
  430. // _isOutputNoneMatch[index] = NO;
  431. // }
  432. // }
  433. //}
  434. void CheckPhaseLossStatus(uint8_t index)
  435. {
  436. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  437. if (pAlarmCode->AlarmEvents.bits.SystemL1InputUVP == YES ||
  438. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP == YES ||
  439. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP == YES) {
  440. //log_info("CheckPhaseLossStatus\r\n");
  441. pDcChargingInfo->StopChargeFlag = YES;
  442. }
  443. }
  444. void SetK1K2RelayStatus(uint8_t index)
  445. {
  446. uint8_t targetID = 0;
  447. PreChargingState *pRegPreChargingState = NULL;
  448. PreChargingState *pOutputPreChargingState = NULL;
  449. GunPNState *pRegGunPNState = NULL;
  450. GunPNState *pOutputGunPNState = NULL;
  451. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  452. if (ShmPsuData->Work_Step >= _TEST_MODE &&
  453. ShmPsuData->Work_Step <= _TEST_MODE) {
  454. if (regRelay.relay_event.bits.Gun1_N == NO) {
  455. outputRelay.relay_event.bits.Gun1_N = YES;
  456. } else if (regRelay.relay_event.bits.Gun1_P == NO) {
  457. outputRelay.relay_event.bits.Gun1_P = YES;
  458. }
  459. return;
  460. }
  461. targetID = getCommTargetID(index);
  462. pRegPreChargingState = (PreChargingState *)&regRelay.relay_event.relay_status[0];
  463. pOutputPreChargingState = (PreChargingState *)&outputRelay.relay_event.relay_status[0];
  464. if (targetID == 0x01) {
  465. pRegGunPNState = (GunPNState *)&regRelay.relay_event.relay_status[1];
  466. pOutputGunPNState = (GunPNState *)&outputRelay.relay_event.relay_status[1];
  467. } else if (targetID == 0x02) {
  468. pRegGunPNState = (GunPNState *)&regRelay.relay_event.relay_status[2];
  469. pOutputGunPNState = (GunPNState *)&outputRelay.relay_event.relay_status[2];
  470. }
  471. switch (pDcChargingInfo->SystemStatus) {
  472. case S_BOOTING:
  473. case S_IDLE:
  474. case S_AUTHORIZING:
  475. case S_REASSIGN_CHECK:
  476. case S_REASSIGN:
  477. case S_PREPARNING:
  478. case S_PREPARING_FOR_EV:
  479. if (pRegGunPNState->GunP == YES) {
  480. pOutputGunPNState->GunP = NO;
  481. } else if (pRegGunPNState->GunN == YES) {
  482. pOutputGunPNState->GunN = NO;
  483. }
  484. if (targetID == 0x02 && pDcChargingInfo->Type == _Type_CCS_2) {
  485. if (pRegPreChargingState->CcsPrecharge == YES) {
  486. pOutputPreChargingState->CcsPrecharge = NO;
  487. }
  488. }
  489. break;
  490. case S_PREPARING_FOR_EVSE:
  491. case S_CHARGING:
  492. //if (pDcChargingInfo->RelayWeldingCheck != YES) {
  493. // break;
  494. //}
  495. if (pRegGunPNState->GunN == NO) {
  496. pOutputGunPNState->GunN = YES;
  497. } else if (pRegGunPNState->GunP == NO) {
  498. pOutputGunPNState->GunP = YES;
  499. }
  500. break;
  501. case S_TERMINATING:
  502. case S_COMPLETE:
  503. case S_ALARM:
  504. if ((pDcChargingInfo->PresentChargingCurrent * 10) <= SEFETY_SWITCH_RELAY_CUR) {
  505. if (pRegGunPNState->GunP == YES) {
  506. pOutputGunPNState->GunP = NO;
  507. } else if (pRegGunPNState->GunN == YES) {
  508. pOutputGunPNState->GunN = NO;
  509. }
  510. }
  511. break;
  512. case S_CCS_PRECHARGE_ST0:
  513. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  514. break;
  515. #endif //defined DD360 || defined DD360Audi || defined DD360ComBox
  516. //if (pDcChargingInfo->Type == _Type_CCS_2 && targetID == 0x02) {
  517. // if (pRegPreChargingState->CcsPrecharge == NO) {
  518. // pOutputPreChargingState->CcsPrecharge = YES;
  519. // } else if (pRegPreChargingState->CcsPrecharge == YES) {
  520. // pRegGunPNState->GunP = NO;
  521. // }
  522. //}
  523. break;
  524. case S_CCS_PRECHARGE_ST1:
  525. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  526. break;
  527. #endif //defined DD360 || defined DD360Audi || defined DD360ComBox
  528. //if (pDcChargingInfo->Type == _Type_CCS_2 && targetID == 0x02) {
  529. // if (pRegGunPNState->GunP == NO) {
  530. // pOutputGunPNState->GunP = YES;
  531. // } else if (pRegGunPNState->GunP == YES) {
  532. // pOutputPreChargingState->CcsPrecharge = NO;
  533. // }
  534. //}
  535. break;
  536. }
  537. }
  538. // 確認 K1 K2 relay 的狀態
  539. void CheckK1K2RelayOutput(uint8_t index)
  540. {
  541. uint8_t targetID = 0;
  542. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  543. targetID = getCommTargetID(index);
  544. switch (targetID) {
  545. case 0x01:
  546. if (regRelay.relay_event.bits.Gun1_N == YES && regRelay.relay_event.bits.Gun1_P == YES) {
  547. pDcChargingInfo->RelayK1K2Status = YES;
  548. } else {
  549. pDcChargingInfo->RelayK1K2Status = NO;
  550. }
  551. if (pDcChargingInfo->Type == _Type_CCS_2) {
  552. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  553. if (regRelay.relay_event.bits.Gun1_N == YES && regRelay.relay_event.bits.CCS_Precharge == YES) {
  554. pDcChargingInfo->RelayKPK2Status = YES;
  555. } else {
  556. pDcChargingInfo->RelayKPK2Status = NO;
  557. }
  558. #else
  559. if (pDcChargingInfo->SystemStatus == S_CCS_PRECHARGE_ST0) {
  560. pDcChargingInfo->RelayKPK2Status = YES;
  561. } else {
  562. pDcChargingInfo->RelayKPK2Status = NO;
  563. }
  564. #endif //!defined DD360 && !defined DD360Audi
  565. }
  566. break;
  567. case 0x02:
  568. if (regRelay.relay_event.bits.Gun2_N == YES &&
  569. regRelay.relay_event.bits.Gun2_P == YES) {
  570. pDcChargingInfo->RelayK1K2Status = YES;
  571. } else {
  572. pDcChargingInfo->RelayK1K2Status = NO;
  573. }
  574. if (pDcChargingInfo->Type == _Type_CCS_2) {
  575. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  576. if (regRelay.relay_event.bits.Gun2_N == YES &&
  577. regRelay.relay_event.bits.CCS_Precharge == YES) {
  578. pDcChargingInfo->RelayKPK2Status = YES;
  579. } else {
  580. pDcChargingInfo->RelayKPK2Status = NO;
  581. }
  582. #else
  583. if (pDcChargingInfo->SystemStatus == S_CCS_PRECHARGE_ST0) {
  584. pDcChargingInfo->RelayKPK2Status = YES;
  585. } else {
  586. pDcChargingInfo->RelayKPK2Status = NO;
  587. }
  588. #endif //!defined DD360 && !defined DD360Audi
  589. }
  590. break;
  591. }
  592. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  593. //DS60-120 add
  594. if (pSysInfo->BridgeRelayStatus == YES) {
  595. if (regRelay.relay_event.bits.Gun1_Parallel_N == NO &&
  596. regRelay.relay_event.bits.Gun1_Parallel_P == NO) {
  597. pSysInfo->BridgeRelayStatus = NO;
  598. }
  599. } else if (pSysInfo->BridgeRelayStatus == NO) {
  600. if (regRelay.relay_event.bits.Gun1_Parallel_N == YES &&
  601. regRelay.relay_event.bits.Gun1_Parallel_P == YES) {
  602. pSysInfo->BridgeRelayStatus = YES;
  603. }
  604. }
  605. #else
  606. pSysInfo->BridgeRelayStatus = YES;
  607. #endif //!defined DD360 && !defined DD360Audi
  608. }
  609. void SetGfdConfig(uint8_t index, uint8_t resister)
  610. {
  611. Gfd_config gfd_config = {
  612. .index = index,
  613. .state = resister,
  614. };
  615. //log_info("************************GFD Vol = %d, GFD Res = %d \n", gfd_config.reqVol, gfd_config.resister);
  616. if (Config_Gfd_Value(Uart5Fd, ADDR_RELAY, &gfd_config) == PASS) {
  617. // log_info("Set reqVol = %f, resister = %d \n",
  618. // gfd_config.reqVol,
  619. // gfd_config.resister);
  620. }
  621. }
  622. void CableCheckDetected(uint8_t index)
  623. {
  624. uint8_t targetID = 0;
  625. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  626. // Cable Check
  627. // 當火線上的電壓 = 車端要求的電壓電流
  628. // _chargingData[targetGun]->EvBatterytargetVoltage
  629. // 才可以開始偵測 1s
  630. // Warning : Rgfd <= 150 歐/V 假設電壓為 500V 則~ Rgfd <= 75000 歐
  631. // Pre-Warning : 150 歐/V < Rgfd <= 500 歐/V 假設電壓為 500V 則 75000 歐 < Rgfd <= 250000
  632. // SO Normal : Rgfd > 500 歐/V 假設電壓為 500 V 則 Rgfd > 250000 歐
  633. if (pSysConfig->TotalConnectorCount == 1) {
  634. if (strncmp((char *)&pSysConfig->ModelName[7], "0", 1) != 0) {
  635. targetID = 0;
  636. } else if (strncmp((char *)&pSysConfig->ModelName[9], "0", 1) != 0) {
  637. targetID = 1;
  638. }
  639. } else {
  640. targetID = index;
  641. }
  642. if ((pDcChargingInfo->Type >= _Type_Chademo &&
  643. pDcChargingInfo->Type <= _Type_GB) ||
  644. (pDcChargingInfo->Type == 0x09 &&
  645. pSysConfig->AlwaysGfdFlag)
  646. ) {
  647. if ((pDcChargingInfo->SystemStatus >= S_PREPARING_FOR_EVSE &&
  648. pDcChargingInfo->SystemStatus <= S_TERMINATING) ||
  649. (pDcChargingInfo->SystemStatus >= S_CCS_PRECHARGE_ST0 &&
  650. pDcChargingInfo->SystemStatus <= S_CCS_PRECHARGE_ST1)
  651. ) {
  652. //if ((pDcChargingInfo->SystemStatus == S_PREPARING_FOR_EVSE) &&
  653. // (pDcChargingInfo->RelayWeldingCheck == YES)
  654. // ) {
  655. if (pDcChargingInfo->SystemStatus == S_PREPARING_FOR_EVSE) {
  656. SetGfdConfig(targetID, GFD_CABLECHK);
  657. } else if ((pDcChargingInfo->SystemStatus >= S_CCS_PRECHARGE_ST0) &&
  658. (pDcChargingInfo->SystemStatus <= S_CCS_PRECHARGE_ST1)
  659. ) {
  660. SetGfdConfig(targetID, GFD_PRECHARGE);
  661. } else if ((pDcChargingInfo->SystemStatus >= S_CHARGING) &&
  662. (pDcChargingInfo->SystemStatus <= S_TERMINATING)
  663. ) {
  664. if ((pDcChargingInfo->Type == _Type_GB) ||
  665. (pDcChargingInfo->Type == _Type_Chademo)
  666. ) {
  667. SetGfdConfig(targetID, GFD_IDLE);
  668. } else {
  669. SetGfdConfig(targetID, GFD_CHARGING);
  670. }
  671. }
  672. } else if (pDcChargingInfo->SystemStatus == S_COMPLETE ||
  673. pDcChargingInfo->SystemStatus == S_PREPARNING ||
  674. pDcChargingInfo->SystemStatus == S_IDLE) {
  675. SetGfdConfig(targetID, GFD_IDLE);
  676. }
  677. }
  678. }
  679. // 讀取 Relay 狀態
  680. void GetRelayOutputStatus(void)
  681. {
  682. if (Query_Relay_Output(Uart5Fd, ADDR_RELAY, &regRelay) == PASS) {
  683. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  684. regRelay.relay_event.bits.AC_Contactor = pSysInfo->AcContactorStatus;
  685. #endif //!defined DD360 && !defined DD360Audi
  686. }
  687. }
  688. // AC 三相輸入電壓
  689. void GetPresentInputVol(void)
  690. {
  691. static uint8_t _threePhaseOvp[3] = {0, 0, 0}; //DS60-120 add
  692. static uint8_t _threePhaseUvp[3] = {0, 0, 0}; //DS60-120 add
  693. PresentInputVoltage inputVoltage = {0};
  694. if (Query_Present_InputVoltage(Uart5Fd, ADDR_RELAY, &inputVoltage) == PASS) {
  695. // resolution : 0.1
  696. pSysInfo->InputVoltageR = ShmRelayModuleData->InputL1Volt = inputVoltage.L1N_L12;
  697. pSysInfo->InputVoltageS = ShmRelayModuleData->InputL2Volt = inputVoltage.L2N_L23;
  698. pSysInfo->InputVoltageT = ShmRelayModuleData->InputL3Volt = inputVoltage.L3N_L31;
  699. //********************************************************************************************************//
  700. // Vin (UVP)
  701. if (pSysInfo->ChargerType == _CHARGER_TYPE_IEC) {
  702. if (pAlarmCode->AlarmEvents.bits.SystemL1InputUVP == NO) {
  703. if (inputVoltage.L1N_L12 < VIN_MIN_VOLTAGE_IEC) {
  704. log_info("In Uvp L1N_L12 = %f \n", inputVoltage.L1N_L12);
  705. if (_threePhaseUvp[0] >= OVP_UVP_CHK_COUNT) {
  706. pAlarmCode->AlarmEvents.bits.SystemL1InputUVP = YES;
  707. } else {
  708. _threePhaseUvp[0] += 1;
  709. }
  710. }
  711. } else {
  712. if (inputVoltage.L1N_L12 > VIN_MIN_REV_VOLTAGE_IEC) {
  713. pAlarmCode->AlarmEvents.bits.SystemL1InputUVP = NO;
  714. _threePhaseUvp[0] = 0;
  715. }
  716. }
  717. if (pAlarmCode->AlarmEvents.bits.SystemL2InputUVP == NO) {
  718. if (inputVoltage.L2N_L23 < VIN_MIN_VOLTAGE_IEC) {
  719. log_info("In Uvp L2N_L23 = %f \n", inputVoltage.L2N_L23);
  720. if (_threePhaseUvp[1] >= OVP_UVP_CHK_COUNT) {
  721. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP = YES;
  722. } else {
  723. _threePhaseUvp[1] += 1;
  724. }
  725. }
  726. } else {
  727. if (inputVoltage.L2N_L23 > VIN_MIN_REV_VOLTAGE_IEC) {
  728. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP = NO;
  729. _threePhaseUvp[1] = 0;
  730. }
  731. }
  732. if (pAlarmCode->AlarmEvents.bits.SystemL3InputUVP == NO) {
  733. if (inputVoltage.L3N_L31 < VIN_MIN_VOLTAGE_IEC) {
  734. log_info("In Uvp L3N_L31 = %f \n", inputVoltage.L3N_L31);
  735. if (_threePhaseUvp[2] >= OVP_UVP_CHK_COUNT) {
  736. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP = YES;
  737. } else {
  738. _threePhaseUvp[2] += 1;
  739. }
  740. }
  741. } else {
  742. if (inputVoltage.L3N_L31 > VIN_MIN_REV_VOLTAGE_IEC) {
  743. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP = NO;
  744. _threePhaseUvp[2] = 0;
  745. }
  746. }
  747. } else if (pSysInfo->ChargerType == _CHARGER_TYPE_UL) {
  748. if (pAlarmCode->AlarmEvents.bits.SystemL1InputUVP == NO) {
  749. if (inputVoltage.L1N_L12 < VIN_MIN_VOLTAGE_UL) {
  750. log_info("In Uvp L1N_L12 = %f \n", inputVoltage.L1N_L12);
  751. if (_threePhaseUvp[0] >= OVP_UVP_CHK_COUNT) {
  752. pAlarmCode->AlarmEvents.bits.SystemL1InputUVP = YES;
  753. } else {
  754. _threePhaseUvp[0] += 1;
  755. }
  756. }
  757. } else {
  758. if (inputVoltage.L1N_L12 > VIN_MIN_REV_VOLTAGE_UL) {
  759. pAlarmCode->AlarmEvents.bits.SystemL1InputUVP = NO;
  760. _threePhaseUvp[0] = 0;
  761. }
  762. }
  763. if (pAlarmCode->AlarmEvents.bits.SystemL2InputUVP == NO) {
  764. if (inputVoltage.L2N_L23 < VIN_MIN_VOLTAGE_UL) {
  765. log_info("In Uvp L2N_L23 = %f \n", inputVoltage.L2N_L23);
  766. if (_threePhaseUvp[1] >= OVP_UVP_CHK_COUNT) {
  767. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP = YES;
  768. } else {
  769. _threePhaseUvp[1] += 1;
  770. }
  771. }
  772. } else {
  773. if (inputVoltage.L2N_L23 > VIN_MIN_REV_VOLTAGE_UL) {
  774. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP = NO;
  775. _threePhaseUvp[1] = 0;
  776. }
  777. }
  778. if (pAlarmCode->AlarmEvents.bits.SystemL3InputUVP == NO) {
  779. if (inputVoltage.L3N_L31 < VIN_MIN_VOLTAGE_UL) {
  780. log_info("In Uvp L3N_L31 = %f \n", inputVoltage.L3N_L31);
  781. if (_threePhaseUvp[2] >= OVP_UVP_CHK_COUNT) {
  782. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP = YES;
  783. } else {
  784. _threePhaseUvp[2] += 1;
  785. }
  786. }
  787. } else {
  788. if (inputVoltage.L3N_L31 > VIN_MIN_REV_VOLTAGE_UL) {
  789. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP = NO;
  790. _threePhaseUvp[2] = 0;
  791. }
  792. }
  793. }
  794. //********************************************************************************************************//
  795. // Vin (OVP)
  796. if (pSysInfo->ChargerType == _CHARGER_TYPE_IEC) {
  797. if (pAlarmCode->AlarmEvents.bits.SystemL1InputOVP == NO) {
  798. if (inputVoltage.L1N_L12 > VIN_MAX_VOLTAGE_IEC) {
  799. log_info("In Ovp L1N_L12 = %f \n", inputVoltage.L1N_L12);
  800. if (_threePhaseOvp[0] >= OVP_UVP_CHK_COUNT) {
  801. pAlarmCode->AlarmEvents.bits.SystemL1InputOVP = YES;
  802. } else {
  803. _threePhaseOvp[0] += 1;
  804. }
  805. }
  806. } else {
  807. if (inputVoltage.L1N_L12 < VIN_MAX_REV_VOLTAGE_IEC) {
  808. pAlarmCode->AlarmEvents.bits.SystemL1InputOVP = NO;
  809. _threePhaseOvp[0] = 0;
  810. }
  811. }
  812. if (pAlarmCode->AlarmEvents.bits.SystemL2InputOVP == NO) {
  813. if (inputVoltage.L2N_L23 > VIN_MAX_VOLTAGE_IEC) {
  814. log_info("In Ovp L2N_L23 = %f \n", inputVoltage.L2N_L23);
  815. if (_threePhaseOvp[1] >= OVP_UVP_CHK_COUNT) {
  816. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP = YES;
  817. } else {
  818. _threePhaseOvp[1] += 1;
  819. }
  820. }
  821. } else {
  822. if (inputVoltage.L2N_L23 < VIN_MAX_REV_VOLTAGE_IEC) {
  823. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP = NO;
  824. _threePhaseOvp[1] = 0;
  825. }
  826. }
  827. if (pAlarmCode->AlarmEvents.bits.SystemL3InputOVP == NO) {
  828. if (inputVoltage.L3N_L31 > VIN_MAX_VOLTAGE_IEC) {
  829. log_info("In Ovp L3N_L31 = %f \n", inputVoltage.L3N_L31);
  830. if (_threePhaseOvp[2] >= OVP_UVP_CHK_COUNT) {
  831. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP = YES;
  832. } else {
  833. _threePhaseOvp[2] += 1;
  834. }
  835. }
  836. } else {
  837. if (inputVoltage.L3N_L31 < VIN_MAX_REV_VOLTAGE_IEC) {
  838. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP = NO;
  839. _threePhaseOvp[2] = 0;
  840. }
  841. }
  842. } else if (pSysInfo->ChargerType == _CHARGER_TYPE_UL) {
  843. if (pAlarmCode->AlarmEvents.bits.SystemL1InputOVP == NO) {
  844. if (inputVoltage.L1N_L12 > VIN_MAX_VOLTAGE_UL) {
  845. log_info("In Ovp L1N_L12 = %f \n", inputVoltage.L1N_L12);
  846. if (_threePhaseOvp[0] >= OVP_UVP_CHK_COUNT) {
  847. pAlarmCode->AlarmEvents.bits.SystemL1InputOVP = YES;
  848. } else {
  849. _threePhaseOvp[0] += 0;
  850. }
  851. }
  852. } else {
  853. if (inputVoltage.L1N_L12 < VIN_MAX_REV_VOLTAGE_UL) {
  854. pAlarmCode->AlarmEvents.bits.SystemL1InputOVP = NO;
  855. _threePhaseOvp[0] = 0;
  856. }
  857. }
  858. if (pAlarmCode->AlarmEvents.bits.SystemL2InputOVP == NO) {
  859. if (inputVoltage.L2N_L23 > VIN_MAX_VOLTAGE_UL) {
  860. log_info("In Ovp L2N_L23 = %f \n", inputVoltage.L2N_L23);
  861. if (_threePhaseOvp[1] >= OVP_UVP_CHK_COUNT) {
  862. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP = YES;
  863. } else {
  864. _threePhaseOvp[1] += 0;
  865. }
  866. }
  867. } else {
  868. if (inputVoltage.L2N_L23 < VIN_MAX_REV_VOLTAGE_UL) {
  869. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP = NO;
  870. _threePhaseOvp[1] = 0;
  871. }
  872. }
  873. if (pAlarmCode->AlarmEvents.bits.SystemL2InputOVP == NO) {
  874. if (inputVoltage.L3N_L31 > VIN_MAX_VOLTAGE_UL) {
  875. log_info("In Ovp L3N_L31 = %f \n", inputVoltage.L3N_L31);
  876. if (_threePhaseOvp[2] >= OVP_UVP_CHK_COUNT) {
  877. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP = YES;
  878. } else {
  879. _threePhaseOvp[2] += 1;
  880. }
  881. }
  882. } else {
  883. if (inputVoltage.L3N_L31 < VIN_MAX_REV_VOLTAGE_UL) {
  884. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP = NO;
  885. _threePhaseOvp[2] = 0;
  886. }
  887. }
  888. }
  889. }
  890. }
  891. // 左右槍的 Relay 前後的輸出電壓
  892. void GetPersentOutputVol(void)
  893. {
  894. uint8_t index = 0;
  895. uint8_t targetID = 0;
  896. struct ChargingInfoData *pDcChargingInfo = NULL;
  897. PresentOutputVoltage outputVoltage = {0};
  898. if (Query_Present_OutputVoltage(Uart5Fd, ADDR_RELAY, &outputVoltage) != PASS) {
  899. return;
  900. }
  901. //log_info("Conn1 fuse 1 = %f \n", outputVoltage.behindFuse_Voltage_C1);
  902. //log_info("Conn1 relay 1 = %f \n", outputVoltage.behindRelay_Voltage_C1);
  903. //log_info("Conn2 fuse 2 = %f \n", outputVoltage.behindFuse_Voltage_C2);
  904. //log_info("Conn2 relay 2 = %f \n", outputVoltage.behindRelay_Voltage_C2);
  905. //log_info("outputVoltage.behindFuse_Voltage_C1 = %f \n", outputVoltage.behindFuse_Voltage_C1);
  906. //log_info("outputVoltage.behindFuse_Voltage_C2 = %f \n", outputVoltage.behindFuse_Voltage_C2);
  907. ShmRelayModuleData->Gun1FuseOutputVolt = outputVoltage.behindFuse_Voltage_C1;
  908. ShmRelayModuleData->Gun1RelayOutputVolt = outputVoltage.behindRelay_Voltage_C1;
  909. ShmRelayModuleData->Gun2FuseOutputVolt = outputVoltage.behindFuse_Voltage_C2;
  910. ShmRelayModuleData->Gun2RelayOutputVolt = outputVoltage.behindRelay_Voltage_C2;
  911. for (index = 0; index < pSysConfig->TotalConnectorCount; index++) {
  912. pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  913. targetID = getCommTargetID(index);
  914. switch (targetID) {
  915. case 0x01:
  916. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  917. pDcChargingInfo->FireChargingVoltage = ShmRelayModuleData->Gun1RelayOutputVolt;
  918. pDcChargingInfo->PresentChargingCurrent = ShmRelayModuleData->Gun1FuseOutputVolt / 10;
  919. pDcChargingInfo->PresentChargingVoltage = pDcChargingInfo->FireChargingVoltage / 10;
  920. pDcChargingInfo->FuseChargingVoltage = pDcChargingInfo->FireChargingVoltage;
  921. break;
  922. #endif //defined DD360 || defined DD360Audi || defined DD360ComBox
  923. pDcChargingInfo->FireChargingVoltage = ShmRelayModuleData->Gun1RelayOutputVolt;
  924. pDcChargingInfo->FuseChargingVoltage = ShmRelayModuleData->Gun1FuseOutputVolt;
  925. break;
  926. case 0x02:
  927. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  928. pDcChargingInfo->FireChargingVoltage = ShmRelayModuleData->Gun2RelayOutputVolt;
  929. pDcChargingInfo->PresentChargingCurrent = ShmRelayModuleData->Gun2FuseOutputVolt / 10;
  930. pDcChargingInfo->PresentChargingVoltage = pDcChargingInfo->FireChargingVoltage / 10;
  931. pDcChargingInfo->FuseChargingVoltage = pDcChargingInfo->FireChargingVoltage;
  932. break;
  933. #endif //defined DD360 || defined DD360Audi || defined DD360ComBox
  934. pDcChargingInfo->FireChargingVoltage = ShmRelayModuleData->Gun2RelayOutputVolt;
  935. pDcChargingInfo->FuseChargingVoltage = ShmRelayModuleData->Gun2FuseOutputVolt;
  936. break;
  937. }
  938. //unsigned short Ovp = 0;
  939. //unsigned short Ocp = 0;
  940. //Ovp = MIN [VOUT_MAX_VOLTAGE, EV_BATTERY_VOLTAGE] // 最大輸出電壓與電池電壓最大值
  941. //Ocp = MIN [IOUT_MAX_CURRENT, EV_CURRENT_REQ] // 最大輸出電流與需求電流最小值
  942. //if (pDcChargingInfo->Type == _Type_Chademo) {
  943. // //Ovp = MaxValue(pDcChargingInfo->MaximumChargingVoltage, pDcChargingInfo->EvBatteryMaxVoltage);
  944. // //Ocp = MaxValue(pDcChargingInfo->PresentChargingCurrent, ShmCHAdeMOData->ev[pDcChargingInfo->type_index].ChargingCurrentRequest);
  945. //} else if (pDcChargingInfo->Type == _Type_CCS_2) {
  946. //}
  947. }
  948. }
  949. void SetRtcData_Relay(void)
  950. {
  951. struct timeb csuTime;
  952. struct tm *tmCSU;
  953. Rtc rtc = {0};
  954. ftime(&csuTime);
  955. tmCSU = localtime(&csuTime.time);
  956. // log_info("Time : %04d-%02d-%02d %02d:%02d:%02d \n", tmCSU->tm_year + 1900,
  957. // tmCSU->tm_mon + 1, tmCSU->tm_mday, tmCSU->tm_hour, tmCSU->tm_min,
  958. // tmCSU->tm_sec);
  959. rtc.RtcData[0] = '0' + (tmCSU->tm_year + 1900) / 1000 % 10;
  960. rtc.RtcData[1] = '0' + (tmCSU->tm_year + 1900) / 100 % 10;
  961. rtc.RtcData[2] = '0' + (tmCSU->tm_year + 1900) / 10 % 10;
  962. rtc.RtcData[3] = '0' + (tmCSU->tm_year + 1900) / 1 % 10;
  963. rtc.RtcData[4] = '0' + (tmCSU->tm_mon + 1) / 10 % 10;
  964. rtc.RtcData[5] = '0' + (tmCSU->tm_mon + 1) / 1 % 10;
  965. rtc.RtcData[6] = '0' + (tmCSU->tm_mday) / 10 % 10;
  966. rtc.RtcData[7] = '0' + (tmCSU->tm_mday) / 1 % 10;
  967. rtc.RtcData[8] = '0' + (tmCSU->tm_hour) / 10 % 10;
  968. rtc.RtcData[9] = '0' + (tmCSU->tm_hour) / 1 % 10;
  969. rtc.RtcData[10] = '0' + (tmCSU->tm_min) / 10 % 10;
  970. rtc.RtcData[11] = '0' + (tmCSU->tm_min) / 1 % 10;
  971. rtc.RtcData[12] = '0' + (tmCSU->tm_sec) / 10 % 10;
  972. rtc.RtcData[13] = '0' + (tmCSU->tm_sec) / 1 % 10;
  973. if (Config_Rtc_Data(Uart5Fd, ADDR_RELAY, &rtc) == PASS) {
  974. //log_info("SetRtc (RB) sucessfully. \n");
  975. }
  976. }
  977. void SetModelName_Relay(void)
  978. {
  979. if (Config_Model_Name(Uart5Fd, ADDR_RELAY, pSysConfig->ModelName) == PASS) {
  980. //log_info("Set Model name (RB) PASS = %s \n", pSysConfig->ModelName);
  981. }
  982. }
  983. void GetFwAndHwVersion_Relay(void)
  984. {
  985. Ver ver = {0};
  986. if (Query_FW_Ver(Uart5Fd, ADDR_RELAY, &ver) == PASS) {
  987. // RelayModuleData
  988. strcpy((char *)ShmRelayModuleData->version, ver.Version_FW);
  989. // SystemInfo
  990. strcpy((char *)pSysInfo->RelayModuleFwRev, ver.Version_FW);
  991. //log_info("GetFwAndHwVersion_Relay s1 = %s \n", ver.Version_FW);
  992. }
  993. if (Query_HW_Ver(Uart5Fd, ADDR_RELAY, &ver) == PASS) {
  994. // SystemInfo
  995. strcpy((char *)pSysInfo->RelayModuleHwRev, ver.Version_FW);
  996. //log_info("GetFwAndHwVersion_Relay s2 = %s \n", ver.Version_HW);
  997. }
  998. }
  999. static void outputRelayInit(int fd)
  1000. {
  1001. memset((uint8_t *)&outputRelay, 0, sizeof(Relay));
  1002. if (Config_Relay_Output(fd, ADDR_RELAY, &outputRelay) != PASS) {
  1003. log_info("Config_Relay_Output fail \n");
  1004. }
  1005. }
  1006. void RelayBoardTask(int uartFD)
  1007. {
  1008. pid_t pid = fork();
  1009. if (pid == 0) {
  1010. bool isCharging = false;
  1011. bool isStopChargingCount = false;
  1012. uint8_t i = 0;
  1013. int isContinue = 1;
  1014. struct ChargingInfoData *pDcChargingInfo = NULL;
  1015. //share memory mapping
  1016. pSysConfig = (struct SysConfigData *)GetShmSysConfigData();
  1017. pSysInfo = (struct SysInfoData *)GetShmSysInfoData();
  1018. pAlarmCode = (struct AlarmCodeData *)GetShmAlarmCodeData();
  1019. ShmRelayModuleData = (struct RelayModuleData *)GetShmRelayModuleData();
  1020. ShmPsuData = (struct PsuData *)GetShmPsuData();
  1021. ShmDcCommonData = (DcCommonInfo *)GetShmDcCommonData();
  1022. ShmPrimaryMcuData = (struct PrimaryMcuData *)GetShmPrimaryMcuData();
  1023. Uart5Fd = uartFD;
  1024. //relay init
  1025. outputRelayInit(uartFD);
  1026. while (isContinue) {
  1027. // 程序開始之前~ 必須先確定 FW 版本與硬體版本,確認後!!~ 該模組才算是真正的 Initial Comp.
  1028. if (ShmRelayModuleData->SelfTest_Comp == NO) {
  1029. GetFwAndHwVersion_Relay();
  1030. SetModelName_Relay(); //DS60-120 add
  1031. SetRtcData_Relay();
  1032. sleep(1);
  1033. continue;
  1034. }
  1035. // ==============優先權最高 10 ms ==============
  1036. // 輸出電壓
  1037. GetPersentOutputVol();
  1038. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  1039. // 三相輸入電壓
  1040. GetPresentInputVol();
  1041. #endif //!defined DD360 && !defined DD360Audi
  1042. // 讀取當前 AC relay 狀態
  1043. regRelay.relay_event.bits.AC_Contactor = pSysInfo->AcContactorStatus;
  1044. GetRelayOutputStatus();
  1045. for (i = 0; i < pSysConfig->TotalConnectorCount; i++) {
  1046. pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(i);
  1047. // Cable check (Set)
  1048. CableCheckDetected(i);
  1049. // check k1 k2 relay 狀態
  1050. CheckK1K2RelayOutput(i);
  1051. // 依據當前各槍的狀態選擇 搭上/放開 Relay
  1052. SetK1K2RelayStatus(i);
  1053. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  1054. if (pSysConfig->PhaseLossPolicy == YES) {
  1055. CheckPhaseLossStatus(i);
  1056. }
  1057. CheckAcInputOvpStatus(i);
  1058. #endif //!defined DD360 && !defined DD360Audi
  1059. if (pDcChargingInfo->SystemStatus == S_IDLE ||
  1060. pDcChargingInfo->SystemStatus == S_RESERVATION ||
  1061. pDcChargingInfo->SystemStatus == S_MAINTAIN) {
  1062. //pDcChargingInfo->RelayWeldingCheck = NO;
  1063. //_isRelayWelding[i] = NO;
  1064. _isOvpChkTimeFlag[i] = NO;
  1065. //ResetDetAlarmStatus(i); //DS60-120 add
  1066. }
  1067. if (pDcChargingInfo->SystemStatus == S_BOOTING ||
  1068. (pDcChargingInfo->SystemStatus >= S_REASSIGN_CHECK &&
  1069. pDcChargingInfo->SystemStatus <= S_COMPLETE) ||
  1070. (pDcChargingInfo->SystemStatus >= S_CCS_PRECHARGE_ST0 &&
  1071. pDcChargingInfo->SystemStatus <= S_CCS_PRECHARGE_ST1) ||
  1072. pSysInfo->WaitForPlugit == YES ||
  1073. (pSysInfo->PageIndex >= _LCM_AUTHORIZING &&
  1074. pSysInfo->PageIndex <= _LCM_WAIT_FOR_PLUG)
  1075. ) {
  1076. pDcChargingInfo->IsReadyToCharging = YES;
  1077. isCharging = true;
  1078. // 限定只有在槍類別為 GBT 的時候才做 relay welding 的判斷
  1079. //if (pDcChargingInfo->Type == _Type_GB) {
  1080. // if (pDcChargingInfo->SystemStatus >= S_PREPARING_FOR_EVSE &&
  1081. // pDcChargingInfo->RelayWeldingCheck == NO) {
  1082. // CheckRelayWeldingStatus(i);
  1083. // }
  1084. //} else {
  1085. //pDcChargingInfo->RelayWeldingCheck = YES;
  1086. //}
  1087. if (pDcChargingInfo->SystemStatus == S_CHARGING) {
  1088. CheckOutputPowerOverCarReq(i);
  1089. //CheckOutputVolNoneMatchFire(i);
  1090. }
  1091. /*else {
  1092. _isOutputNoneMatch[i] = NO;
  1093. }*/
  1094. } else {
  1095. pDcChargingInfo->IsReadyToCharging = NO;
  1096. }
  1097. }
  1098. // Cable check (Get)
  1099. GetGfdAdc();
  1100. // 橋接 relay
  1101. SetParalleRelayStatus();
  1102. // 搭上 AC Contactor
  1103. //if (isCharging) {
  1104. // outputRelay.relay_event.bits.AC_Contactor = YES;
  1105. //} else {
  1106. // outputRelay.relay_event.bits.AC_Contactor = NO;
  1107. //}
  1108. if (isCharging ||
  1109. (ShmPsuData->Work_Step >= _TEST_MODE &&
  1110. ShmPsuData->Work_Step <= _TEST_MODE)) {
  1111. isStopChargingCount = false;
  1112. outputRelay.relay_event.bits.AC_Contactor = YES;
  1113. } else {
  1114. if (!isStopChargingCount) {
  1115. gettimeofday(&_close_ac_contactor, NULL);
  1116. isStopChargingCount = true;
  1117. } else {
  1118. if ((outputRelay.relay_event.bits.AC_Contactor == YES &&
  1119. GetTimeoutValue(_close_ac_contactor) / 1000 >= (TEN_MINUTES * 1000))) {
  1120. outputRelay.relay_event.bits.AC_Contactor = NO;
  1121. }
  1122. }
  1123. }
  1124. if (ShmPrimaryMcuData->InputDet.bits.EmergencyButton == ABNORMAL) {
  1125. outputRelay.relay_event.bits.AC_Contactor = NO;
  1126. }
  1127. if (pAlarmCode->AlarmEvents.bits.PsuFailureAlarm == ABNORMAL) {
  1128. RunForceStopProcess();
  1129. outputRelay.relay_event.bits.AC_Contactor = NO;
  1130. }
  1131. if (ShmPsuData->Work_Step >= _TEST_MODE && ShmPsuData->Work_Step <= _TEST_MODE) {
  1132. outputRelay.relay_event.bits.Gun1_N = outputRelay.relay_event.bits.Gun1_P = YES;
  1133. }
  1134. // 搭上/鬆開 Relay
  1135. if (IsNoneMatchRelayStatus()) {
  1136. if (Config_Relay_Output(Uart5Fd, ADDR_RELAY, &outputRelay)) {
  1137. //regRelay.relay_event.bits.AC_Contactor = pSysInfo->AcContactorStatus;
  1138. //regRelay.relay_event.bits.CCS_Precharge = outputRelay.relay_event.bits.CCS_Precharge;
  1139. //regRelay.relay_event.bits.Gun1_P = outputRelay.relay_event.bits.Gun1_P;
  1140. //regRelay.relay_event.bits.Gun1_N = outputRelay.relay_event.bits.Gun1_N;
  1141. //regRelay.relay_event.bits.Gun2_P = outputRelay.relay_event.bits.Gun2_P;
  1142. //regRelay.relay_event.bits.Gun2_N = outputRelay.relay_event.bits.Gun2_N;
  1143. //regRelay.relay_event.bits.Gun1_Parallel_P = outputRelay.relay_event.bits.Gun1_Parallel_P;
  1144. //regRelay.relay_event.bits.Gun1_Parallel_N = outputRelay.relay_event.bits.Gun1_Parallel_N;
  1145. //MatchRelayStatus();
  1146. //log_info("Match Relay, AC = %x, g1_p = %x, g1_n = %x, g2_p = %x, g2_n = %x, pre = %x, bri_p = %x, bri_n = %x \n",
  1147. // regRelay.relay_event.bits.AC_Contactor,
  1148. // regRelay.relay_event.bits.Gun1_P,
  1149. // regRelay.relay_event.bits.Gun1_N,
  1150. // regRelay.relay_event.bits.Gun2_P,
  1151. // regRelay.relay_event.bits.Gun2_N,
  1152. // regRelay.relay_event.bits.CCS_Precharge,
  1153. // regRelay.relay_event.bits.Gun1_Parallel_P,
  1154. // regRelay.relay_event.bits.Gun1_Parallel_N);
  1155. }
  1156. } /*else {
  1157. log_info("======== Relay Status Start========\n");
  1158. if (regRelay.relay_event.bits.AC_Contactor == YES) {
  1159. log_info("AC Power : ON \n");
  1160. } else {
  1161. log_info("AC Power : OFF \n");
  1162. }
  1163. if (regRelay.relay_event.bits.Gun1_P == YES) {
  1164. log_info("Conn1(+) : ON \n");
  1165. } else {
  1166. log_info("Conn1(+) : OFF \n");
  1167. }
  1168. if (regRelay.relay_event.bits.Gun1_N == YES) {
  1169. log_info("Conn1(-) : ON \n");
  1170. } else {
  1171. log_info("Conn1(-) : OFF \n");
  1172. }
  1173. if (regRelay.relay_event.bits.Gun2_P == YES) {
  1174. log_info("Conn2(+) : ON \n");
  1175. } else {
  1176. log_info("Conn2(+) : OFF \n");
  1177. }
  1178. if (regRelay.relay_event.bits.Gun2_N == YES) {
  1179. log_info("Conn2(-) : ON \n");
  1180. } else {
  1181. log_info("Conn2(-) : OFF \n");
  1182. }
  1183. if (regRelay.relay_event.bits.CCS_Precharge == YES) {
  1184. log_info("Precharge : ON \n");
  1185. } else {
  1186. log_info("Precharge : OFF \n");
  1187. }
  1188. if (regRelay.relay_event.bits.Gun1_Parallel_P == YES) {
  1189. log_info("Parallel(+) : ON \n");
  1190. } else {
  1191. log_info("Parallel(+) : OFF \n");
  1192. }
  1193. if (regRelay.relay_event.bits.Gun1_Parallel_N == YES) {
  1194. log_info("Parallel(-) : ON \n");
  1195. } else {
  1196. log_info("Parallel(-) : OFF \n");
  1197. }
  1198. log_info("======== Relay Status End========\n");
  1199. }*/
  1200. }
  1201. usleep(100000);
  1202. }
  1203. }