RelayBoard.c 75 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863
  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 struct WARNING_CODE_INFO *pSysWarning = NULL;
  23. static struct LedModuleData *ShmLedModuleData = NULL;
  24. static struct FanModuleData *ShmFanModuleData = NULL;
  25. static Relay outputRelay = {0};
  26. static Relay regRelay = {0};
  27. static int Uart5Fd = 0;
  28. static struct timeval gFanBoardRunTimer;
  29. static uint16_t _setFanSpeed = 0;
  30. static uint16_t fanSpeedSmoothValue = 500;
  31. static Led_Color cur_led_color = {COLOR_MIN_LV};
  32. static Led_Color led_color;
  33. static struct timeval _led_priority_time;
  34. //static bool _isRelayWelding[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY];
  35. //static struct timeval _checkRelayWeldingTimer[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY];
  36. //static bool _isOutputNoneMatch[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY];
  37. //static struct timeval _checkOutputNoneMatchTimer[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY];
  38. static bool _isOvpChkTimeFlag[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY]; //DS60-120 add
  39. static struct timeval _checkOutputVolProtectTimer[CHAdeMO_QUANTITY + CCS_QUANTITY + GB_QUANTITY]; //DS60-120 add
  40. static struct timeval _close_ac_contactor;
  41. //------------------------------------------------------------------------------
  42. static void RunForceStopProcess(void)
  43. {
  44. static bool isCriticalStop = NO;
  45. static struct timeval _psuCriticalStop;
  46. uint32_t _timebuf;
  47. if (isCriticalStop == NO) {
  48. isCriticalStop = YES;
  49. gettimeofday(&_psuCriticalStop, NULL);
  50. } else {
  51. _timebuf = GetTimeoutValue(_psuCriticalStop);
  52. if (_timebuf < 0) {
  53. gettimeofday(&_psuCriticalStop, NULL);
  54. } else {
  55. if (_timebuf / 1000 >= (FORCE_STOP_TIME * 1000)) {
  56. isCriticalStop = NO;
  57. pAlarmCode->AlarmEvents.bits.PsuFailureAlarm = NORMAL;
  58. }
  59. }
  60. }
  61. }
  62. static void StopCheckRelayInfo(uint8_t _chkIndex)
  63. {
  64. if (ShmDcCommonData->CheckRelayStatus[_chkIndex] != STOP) {
  65. ShmDcCommonData->CheckRelayStatus[_chkIndex] = STOP;
  66. }
  67. }
  68. static void StartCheckRelayInfo(uint8_t _chkIndex, uint8_t toState)
  69. {
  70. // SMR1 *2 + SMR2 * 2 + Parallel * 2
  71. static struct timeval lastCheckRelayStateTimer[6] = {0};
  72. //uint8_t *pCheckRelayState = (uint8_t *)ShmDcCommonData->CheckRelayStatus[_chkIndex];
  73. if (ShmDcCommonData->CheckRelayStatus[_chkIndex] == STOP) {
  74. gettimeofday(&lastCheckRelayStateTimer[_chkIndex], NULL);
  75. ShmDcCommonData->CheckRelayStatus[_chkIndex] = START;
  76. } else {
  77. if ((GetTimeoutValue(lastCheckRelayStateTimer[_chkIndex]) / 1000000) >= 1) {
  78. //log_info("relay welding or driving fault = %d \n", _chkIndex);
  79. if (toState == 1) {
  80. ShmDcCommonData->CheckRelayStatus[_chkIndex] = RELAY_STATUS_ERROR_DRIVING;
  81. } else {
  82. ShmDcCommonData->CheckRelayStatus[_chkIndex] = RELAY_STATUS_ERROR_WELDING;
  83. }
  84. gettimeofday(&lastCheckRelayStateTimer[_chkIndex], NULL);
  85. }
  86. }
  87. }
  88. static uint8_t getCommTargetID(uint8_t index)
  89. {
  90. uint8_t targetID = 0;
  91. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  92. if (pSysConfig->TotalConnectorCount == 1) {
  93. if (strncmp((char *)&pSysConfig->ModelName[7], "0", 1) != 0) {
  94. targetID = 0x01;
  95. } else if (strncmp((char *)&pSysConfig->ModelName[9], "0", 1) != 0) {
  96. targetID = 0x02;
  97. }
  98. } else {
  99. targetID = pDcChargingInfo->Evboard_id;
  100. }
  101. return targetID;
  102. }
  103. /*static void MatchRelayStatus(void)
  104. {
  105. // 因為 AC Contactor 沒有 Feedback,所以暫時先這樣處理
  106. //regRelay.relay_event.bits.AC_Contactor = outputRelay.relay_event.bits.AC_Contactor;
  107. //pSysInfo->AcContactorStatus =
  108. // regRelay.relay_event.bits.AC_Contactor =
  109. // outputRelay.relay_event.bits.AC_Contactor;
  110. regRelay.relay_event.bits.CCS_Precharge = outputRelay.relay_event.bits.CCS_Precharge;
  111. regRelay.relay_event.bits.Gun1_P = outputRelay.relay_event.bits.Gun1_P;
  112. regRelay.relay_event.bits.Gun1_N = outputRelay.relay_event.bits.Gun1_N;
  113. regRelay.relay_event.bits.Gun2_P = outputRelay.relay_event.bits.Gun2_P;
  114. regRelay.relay_event.bits.Gun2_N = outputRelay.relay_event.bits.Gun2_N;
  115. regRelay.relay_event.bits.Gun1_Parallel_P = outputRelay.relay_event.bits.Gun1_Parallel_P;
  116. regRelay.relay_event.bits.Gun1_Parallel_N = outputRelay.relay_event.bits.Gun1_Parallel_N;
  117. }
  118. */
  119. static bool IsNoneMatchRelayStatus(void)
  120. {
  121. bool result = false;
  122. if (
  123. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  124. (regRelay.relay_event.bits.AC_Contactor != outputRelay.relay_event.bits.AC_Contactor) ||
  125. (regRelay.relay_event.bits.CCS_Precharge != outputRelay.relay_event.bits.CCS_Precharge) ||
  126. #endif //!defined DD360 && !defined DD360Audi
  127. (regRelay.relay_event.bits.Gun1_P != outputRelay.relay_event.bits.Gun1_P) ||
  128. (regRelay.relay_event.bits.Gun1_N != outputRelay.relay_event.bits.Gun1_N) ||
  129. (regRelay.relay_event.bits.Gun2_P != outputRelay.relay_event.bits.Gun2_P) ||
  130. (regRelay.relay_event.bits.Gun2_N != outputRelay.relay_event.bits.Gun2_N)
  131. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  132. ||
  133. (regRelay.relay_event.bits.Gun1_Parallel_P != outputRelay.relay_event.bits.Gun1_Parallel_P) ||
  134. (regRelay.relay_event.bits.Gun1_Parallel_N != outputRelay.relay_event.bits.Gun1_Parallel_N)
  135. #endif //!defined DD360 && !defined DD360Audi
  136. ) {
  137. result = true;
  138. }
  139. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  140. if (regRelay.relay_event.bits.AC_Contactor != outputRelay.relay_event.bits.AC_Contactor) {
  141. log_info("AC Contact Relay none match. \n");
  142. }
  143. if (regRelay.relay_event.bits.CCS_Precharge != outputRelay.relay_event.bits.CCS_Precharge) {
  144. log_info("CCS Precharge Relay none match. \n");
  145. }
  146. #endif //
  147. if (regRelay.relay_event.bits.Gun1_P != outputRelay.relay_event.bits.Gun1_P) {
  148. //log_info("SMR1:D+ Relay none match. \n");
  149. StartCheckRelayInfo(RELAY_SMR1_P_STATUS, outputRelay.relay_event.bits.Gun1_P);
  150. } else {
  151. StopCheckRelayInfo(RELAY_SMR1_P_STATUS);
  152. }
  153. if (regRelay.relay_event.bits.Gun1_N != outputRelay.relay_event.bits.Gun1_N) {
  154. //log_info("SMR1:D- Relay none match. \n");
  155. StartCheckRelayInfo(RELAY_SMR1_N_STATUS, outputRelay.relay_event.bits.Gun1_N);
  156. } else {
  157. StopCheckRelayInfo(RELAY_SMR1_N_STATUS);
  158. }
  159. if (regRelay.relay_event.bits.Gun2_P != outputRelay.relay_event.bits.Gun2_P) {
  160. //log_info("SMR2:D+ Relay none match. \n");
  161. StartCheckRelayInfo(RELAY_SMR2_P_STATUS, outputRelay.relay_event.bits.Gun2_P);
  162. } else {
  163. StopCheckRelayInfo(RELAY_SMR2_P_STATUS);
  164. }
  165. if (regRelay.relay_event.bits.Gun2_N != outputRelay.relay_event.bits.Gun2_N) {
  166. //log_info("SMR2:D- Relay none match. \n");
  167. StartCheckRelayInfo(RELAY_SMR2_N_STATUS, outputRelay.relay_event.bits.Gun2_N);
  168. } else {
  169. StopCheckRelayInfo(RELAY_SMR2_N_STATUS);
  170. }
  171. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  172. if (regRelay.relay_event.bits.Gun1_Parallel_P != outputRelay.relay_event.bits.Gun1_Parallel_P) {
  173. //log_info("Parallel:D+ Relay none match. \n");
  174. StartCheckRelayInfo(RELAY_PARA_P_STATUS, outputRelay.relay_event.bits.Gun1_Parallel_P);
  175. } else {
  176. StopCheckRelayInfo(RELAY_PARA_P_STATUS);
  177. }
  178. if (regRelay.relay_event.bits.Gun1_Parallel_N != outputRelay.relay_event.bits.Gun1_Parallel_N) {
  179. //log_info("Parallel:D- Relay none match. \n");
  180. StartCheckRelayInfo(RELAY_PARA_N_STATUS, outputRelay.relay_event.bits.Gun1_Parallel_N);
  181. } else {
  182. StopCheckRelayInfo(RELAY_PARA_N_STATUS);
  183. }
  184. #endif //
  185. return result;
  186. }
  187. static void SetParalleRelayStatus(void)
  188. {
  189. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  190. return;
  191. #endif //!defined DD360 || !defined DD360Audi || !defined DD360ComBox
  192. struct ChargingInfoData *pDcChargingInfo0 = (struct ChargingInfoData *)GetDcChargingInfoData(0);
  193. struct ChargingInfoData *pDcChargingInfo1 = (struct ChargingInfoData *)GetDcChargingInfoData(1);
  194. // 之後雙槍單模機種,橋接都會上
  195. if (pSysConfig->TotalConnectorCount >= 2) {
  196. if (pDcChargingInfo0->SystemStatus == S_BOOTING || pDcChargingInfo1->SystemStatus == S_BOOTING ||
  197. (pDcChargingInfo0->SystemStatus == S_IDLE && pDcChargingInfo1->SystemStatus == S_IDLE)) {
  198. // 初始化~ 不搭橋接
  199. if (regRelay.relay_event.bits.Gun1_Parallel_P == YES) {
  200. outputRelay.relay_event.bits.Gun1_Parallel_P = NO;
  201. } else if (regRelay.relay_event.bits.Gun1_Parallel_N == YES) {
  202. outputRelay.relay_event.bits.Gun1_Parallel_N = NO;
  203. }
  204. } else {
  205. if (pDcChargingInfo0->IsReadyToCharging == YES ||
  206. pDcChargingInfo1->IsReadyToCharging == YES) {
  207. // ************需考慮在切換中 - 切開 relay 與搭回 relay 的時機點************
  208. if (pSysInfo->MainChargingMode == _MAIN_CHARGING_MODE_MAX) {
  209. if (pSysInfo->ReAssignedFlag < _REASSIGNED_RELAY_M_TO_A) {
  210. // 最大充 - 搭上橋接
  211. if (regRelay.relay_event.bits.Gun1_Parallel_N == NO) {
  212. outputRelay.relay_event.bits.Gun1_Parallel_N = YES;
  213. } else if (regRelay.relay_event.bits.Gun1_Parallel_P == NO) {
  214. outputRelay.relay_event.bits.Gun1_Parallel_P = YES;
  215. }
  216. } else {
  217. // 平均充 - 不搭
  218. if (regRelay.relay_event.bits.Gun1_Parallel_P == YES) {
  219. outputRelay.relay_event.bits.Gun1_Parallel_P = NO;
  220. } else if (regRelay.relay_event.bits.Gun1_Parallel_N == YES) {
  221. outputRelay.relay_event.bits.Gun1_Parallel_N = NO;
  222. }
  223. }
  224. } else if (pSysInfo->MainChargingMode == _MAIN_CHARGING_MODE_AVER) {
  225. if (pSysInfo->ReAssignedFlag < _REASSIGNED_RELAY_A_TO_M) {
  226. // 平均充 - 不搭
  227. if (regRelay.relay_event.bits.Gun1_Parallel_P == YES) {
  228. outputRelay.relay_event.bits.Gun1_Parallel_P = NO;
  229. } else if (regRelay.relay_event.bits.Gun1_Parallel_N == YES) {
  230. outputRelay.relay_event.bits.Gun1_Parallel_N = NO;
  231. }
  232. } else {
  233. // 最大充 - 搭上橋接
  234. if (regRelay.relay_event.bits.Gun1_Parallel_N == NO) {
  235. outputRelay.relay_event.bits.Gun1_Parallel_N = YES;
  236. } else if (regRelay.relay_event.bits.Gun1_Parallel_P == NO) {
  237. outputRelay.relay_event.bits.Gun1_Parallel_P = YES;
  238. }
  239. }
  240. }
  241. }
  242. }
  243. }
  244. }
  245. static void GetGfdAdc(void)
  246. {
  247. int gunIndex = 0;
  248. uint8_t targetID = 0;
  249. struct ChargingInfoData *pDcChargingInfo = NULL;
  250. Gfd gfd_adc = {0};
  251. // define : 每 0.2 ~ 1 秒一次
  252. // occur : <= 75k 歐姆 @ 150 - 750 Vdc
  253. // warning : >= 100 歐姆 && <= 500 歐姆 @ 150-750 Vdc
  254. if (Query_Gfd_Adc(Uart5Fd, ADDR_RELAY, &gfd_adc) == PASS) {
  255. for (gunIndex = 0; gunIndex < pSysConfig->TotalConnectorCount; gunIndex++) {
  256. pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(gunIndex);
  257. if (pDcChargingInfo->Type == 0x09 &&
  258. !pSysConfig->AlwaysGfdFlag
  259. ) {
  260. if ((pDcChargingInfo->PresentChargingVoltage * 10) >= VOUT_MIN_VOLTAGE) {
  261. pDcChargingInfo->GroundFaultStatus = GFD_PASS;
  262. }
  263. continue;
  264. }
  265. targetID = getCommTargetID(gunIndex);
  266. if (targetID == 0x01) {
  267. //if (gfd_adc.result_conn1 == GFD_WARNING) {
  268. // gfd_adc.result_conn1 = GFD_PASS;
  269. //}
  270. pDcChargingInfo->GroundFaultStatus = gfd_adc.result_conn1;
  271. //log_info("GFD ******** Result = %d, Step = %d, R = %d, Vol = %d \n",
  272. // pDcChargingInfo->GroundFaultStatus,
  273. // gfd_adc.rb_step_1,
  274. // gfd_adc.Resister_conn1,
  275. // gfd_adc.voltage_conn1);
  276. if (pDcChargingInfo->GroundFaultStatus == GFD_FAIL) {
  277. log_info("GFD Fail. index = %d, Step = %d, R = %d, Vol = %d \n",
  278. gunIndex,
  279. gfd_adc.rb_step_1,
  280. gfd_adc.Resister_conn1,
  281. gfd_adc.voltage_conn1);
  282. } else if (pDcChargingInfo->GroundFaultStatus == GFD_PASS ||
  283. pDcChargingInfo->GroundFaultStatus == GFD_WARNING
  284. ) {
  285. if (pDcChargingInfo->GroundFaultStatus == GFD_WARNING) {
  286. log_info("GFD Warning. index = %d, Result = %d, R = %d, Vol = %d \n",
  287. gunIndex,
  288. pDcChargingInfo->GroundFaultStatus,
  289. gfd_adc.Resister_conn1,
  290. gfd_adc.voltage_conn1);
  291. }
  292. }
  293. } else if (targetID == 0x02) {
  294. //if (gfd_adc.result_conn2 == GFD_WARNING) {
  295. // gfd_adc.result_conn2 = GFD_PASS;
  296. //}
  297. pDcChargingInfo->GroundFaultStatus = gfd_adc.result_conn2;
  298. if (pDcChargingInfo->GroundFaultStatus == GFD_FAIL) {
  299. log_info("GFD Fail. index = %d, Step = %d, R = %d, Vol = %d \n",
  300. gunIndex,
  301. gfd_adc.rb_step_2,
  302. gfd_adc.Resister_conn2,
  303. gfd_adc.voltage_conn2);
  304. } else if (pDcChargingInfo->GroundFaultStatus == GFD_PASS ||
  305. pDcChargingInfo->GroundFaultStatus == GFD_WARNING
  306. ) {
  307. if (pDcChargingInfo->GroundFaultStatus == GFD_WARNING) {
  308. log_info("GFD Warning. index = %d, Result = %d, R = %d, Vol = %d \n",
  309. gunIndex,
  310. pDcChargingInfo->GroundFaultStatus,
  311. gfd_adc.Resister_conn1,
  312. gfd_adc.voltage_conn1);
  313. }
  314. }
  315. }
  316. }
  317. }
  318. }
  319. void CheckOutputPowerOverCarReq(uint8_t index)
  320. {
  321. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  322. float fireV = pDcChargingInfo->FireChargingVoltage;
  323. float carV = pDcChargingInfo->EvBatteryMaxVoltage * 10;
  324. if ((pDcChargingInfo->EvBatterytargetVoltage * 10) > 1500 &&
  325. (pDcChargingInfo->Type == _Type_Chademo ||
  326. pDcChargingInfo->Type == _Type_CCS_2 ||
  327. pDcChargingInfo->Type == _Type_GB)) {
  328. if (fireV >= (carV + (carV * 0.02))) {
  329. if (!_isOvpChkTimeFlag[index]) {
  330. if ((pDcChargingInfo->PresentChargingVoltage * 10) >= VOUT_MIN_VOLTAGE * 10) {
  331. gettimeofday(&_checkOutputVolProtectTimer[index], NULL);
  332. _isOvpChkTimeFlag[index] = YES;
  333. }
  334. } else {
  335. log_info("[Module_InternalComm]CheckOutputPowerOverCarReq NG : fire = %f, battery = %f \n",
  336. pDcChargingInfo->FireChargingVoltage,
  337. (pDcChargingInfo->EvBatterytargetVoltage * 10));
  338. log_error("[Module_InternalComm]CheckOutputPowerOverCarReq NG : fire = %f, battery = %f \n",
  339. pDcChargingInfo->FireChargingVoltage,
  340. (pDcChargingInfo->EvBatterytargetVoltage * 10));
  341. if ((GetTimeoutValue(_checkOutputVolProtectTimer[index]) / 1000) >= OUTPUT_VOL_CHK_TIME) {
  342. if (pDcChargingInfo->Type == _Type_Chademo) {
  343. //pAlarmCode->AlarmEvents.bits.SystemChademoOutputOVP = YES;
  344. ShmDcCommonData->ConnectErrList[index].GunBits.ChaConnectOVP = YES;
  345. } else if (pDcChargingInfo->Type == _Type_CCS_2) {
  346. //pAlarmCode->AlarmEvents.bits.SystemCcsOutputOVP = YES;
  347. ShmDcCommonData->ConnectErrList[index].GunBits.CCSConnectOVP = YES;
  348. } else if (pDcChargingInfo->Type == _Type_GB) {
  349. //pAlarmCode->AlarmEvents.bits.SystemGbOutputOVP = YES;
  350. ShmDcCommonData->ConnectErrList[index].GunBits.GBTConnectOVP = YES;
  351. }
  352. //pDcChargingInfo->StopChargeFlag = YES;
  353. }
  354. }
  355. } else {
  356. if (_isOvpChkTimeFlag[index] == YES) {
  357. _isOvpChkTimeFlag[index] = NO;
  358. }
  359. }
  360. }
  361. }
  362. void ResetDetAlarmStatus(uint8_t gun)
  363. {
  364. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(gun);
  365. if (pDcChargingInfo->Type == _Type_Chademo) {
  366. if (pAlarmCode->AlarmEvents.bits.SystemChademoOutputOVP == YES) {
  367. pAlarmCode->AlarmEvents.bits.SystemChademoOutputOVP = NO;
  368. }
  369. } else if (pDcChargingInfo->Type == _Type_GB) {
  370. if (pAlarmCode->AlarmEvents.bits.SystemGbOutputOVP == YES) {
  371. pAlarmCode->AlarmEvents.bits.SystemGbOutputOVP = NO;
  372. }
  373. } else if (pDcChargingInfo->Type == _Type_CCS_2) {
  374. if (pAlarmCode->AlarmEvents.bits.SystemCcsOutputOVP == YES) {
  375. pAlarmCode->AlarmEvents.bits.SystemCcsOutputOVP = NO;
  376. }
  377. }
  378. }
  379. void CheckAcInputOvpStatus(uint8_t index)
  380. {
  381. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  382. if (pAlarmCode->AlarmEvents.bits.SystemL1InputOVP == YES ||
  383. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP == YES ||
  384. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP == YES) {
  385. // if ((pDcChargingInfo->SystemStatus >= S_PREPARNING && pDcChargingInfo->SystemStatus <= S_CHARGING) ||
  386. // (pDcChargingInfo->SystemStatus >= S_CCS_PRECHARGE_ST0 && pDcChargingInfo->SystemStatus <= S_CCS_PRECHARGE_ST1))
  387. // {
  388. // if (pSysInfo->ChargerType == _CHARGER_TYPE_IEC)
  389. // {
  390. // if (_psuInputVolR > VIN_MAX_VOLTAGE_IEC ||
  391. // _psuInputVolS > VIN_MAX_VOLTAGE_IEC ||
  392. // _psuInputVolT > VIN_MAX_VOLTAGE_IEC)
  393. // {
  394. // log_info("IEC _psuInputVolR = %f, _psuInputVolS = %f, _psuInputVolT = %f \n",
  395. // _psuInputVolR, _psuInputVolS, _psuInputVolT);
  396. // pDcChargingInfo->StopChargeFlag = YES;
  397. // }
  398. //
  399. // }
  400. // else if (pSysInfo->ChargerType == _CHARGER_TYPE_UL)
  401. // {
  402. // if (_psuInputVolR > VIN_MAX_VOLTAGE_UL ||
  403. // _psuInputVolS > VIN_MAX_VOLTAGE_UL ||
  404. // _psuInputVolT > VIN_MAX_VOLTAGE_UL)
  405. // {
  406. // log_info("UL _psuInputVolR = %f, _psuInputVolS = %f, _psuInputVolT = %f \n",
  407. // _psuInputVolR, _psuInputVolS, _psuInputVolT);
  408. // pDcChargingInfo->StopChargeFlag = YES;
  409. // }
  410. // }
  411. // }
  412. // else
  413. //log_info("CheckAcInputOvpStatus\r\n");
  414. pDcChargingInfo->StopChargeFlag = YES;
  415. }
  416. }
  417. //void CheckOutputVolNoneMatchFire(uint8_t index)
  418. //{
  419. // struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  420. //
  421. // if ((pDcChargingInfo->EvBatterytargetVoltage * 10) > 1500 &&
  422. // (pDcChargingInfo->Type == _Type_Chademo ||
  423. // pDcChargingInfo->Type == _Type_CCS_2 ||
  424. // pDcChargingInfo->Type == _Type_GB)) {
  425. // if (((pDcChargingInfo->PresentChargingVoltage * 10) < pDcChargingInfo->FireChargingVoltage - 300) ||
  426. // ((pDcChargingInfo->PresentChargingVoltage * 10) > pDcChargingInfo->FireChargingVoltage + 300)) {
  427. // if (!_isOutputNoneMatch[index]) {
  428. // _isOutputNoneMatch[index] = YES;
  429. // gettimeofday(&_checkOutputNoneMatchTimer[index], NULL);
  430. // } else {
  431. // if ((GetTimeoutValue(_checkOutputNoneMatchTimer[index]) / 1000) >= 5000) {
  432. // /*log_info("[Module_InternalComm]CheckOutputVolNoneMatchFire NG (%d) : pre = %f, fire = %f \n",
  433. // index, (pDcChargingInfo->PresentChargingVoltage * 10), pDcChargingInfo->FireChargingVoltage);
  434. // log_error("[Module_InternalComm]CheckOutputVolNoneMatchFire NG (%d): pre = %f, fire = %f \n",
  435. // index, (pDcChargingInfo->PresentChargingVoltage * 10), pDcChargingInfo->FireChargingVoltage);
  436. // pDcChargingInfo->StopChargeFlag = YES;*/
  437. // }
  438. // }
  439. // } else {
  440. // _isOutputNoneMatch[index] = NO;
  441. // }
  442. // }
  443. //}
  444. void CheckPhaseLossStatus(uint8_t index)
  445. {
  446. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  447. if (pAlarmCode->AlarmEvents.bits.SystemL1InputUVP == YES ||
  448. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP == YES ||
  449. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP == YES) {
  450. //log_info("CheckPhaseLossStatus\r\n");
  451. pDcChargingInfo->StopChargeFlag = YES;
  452. }
  453. }
  454. void SetK1K2RelayStatus(uint8_t index)
  455. {
  456. uint8_t targetID = 0;
  457. PreChargingState *pRegPreChargingState = NULL;
  458. PreChargingState *pOutputPreChargingState = NULL;
  459. GunPNState *pRegGunPNState = NULL;
  460. GunPNState *pOutputGunPNState = NULL;
  461. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  462. if (ShmPsuData->Work_Step >= _TEST_MODE &&
  463. ShmPsuData->Work_Step <= _TEST_MODE) {
  464. if (regRelay.relay_event.bits.Gun1_N == NO) {
  465. outputRelay.relay_event.bits.Gun1_N = YES;
  466. } else if (regRelay.relay_event.bits.Gun1_P == NO) {
  467. outputRelay.relay_event.bits.Gun1_P = YES;
  468. }
  469. return;
  470. }
  471. targetID = getCommTargetID(index);
  472. pRegPreChargingState = (PreChargingState *)&regRelay.relay_event.relay_status[0];
  473. pOutputPreChargingState = (PreChargingState *)&outputRelay.relay_event.relay_status[0];
  474. if (targetID == 0x01) {
  475. pRegGunPNState = (GunPNState *)&regRelay.relay_event.relay_status[1];
  476. pOutputGunPNState = (GunPNState *)&outputRelay.relay_event.relay_status[1];
  477. } else if (targetID == 0x02) {
  478. pRegGunPNState = (GunPNState *)&regRelay.relay_event.relay_status[2];
  479. pOutputGunPNState = (GunPNState *)&outputRelay.relay_event.relay_status[2];
  480. }
  481. switch (pDcChargingInfo->SystemStatus) {
  482. case S_BOOTING:
  483. case S_IDLE:
  484. case S_AUTHORIZING:
  485. case S_REASSIGN_CHECK:
  486. case S_REASSIGN:
  487. case S_PREPARNING:
  488. case S_PREPARING_FOR_EV:
  489. if (pRegGunPNState->GunP == YES) {
  490. pOutputGunPNState->GunP = NO;
  491. } else if (pRegGunPNState->GunN == YES) {
  492. pOutputGunPNState->GunN = NO;
  493. }
  494. if (targetID == 0x02 && pDcChargingInfo->Type == _Type_CCS_2) {
  495. if (pRegPreChargingState->CcsPrecharge == YES) {
  496. pOutputPreChargingState->CcsPrecharge = NO;
  497. }
  498. }
  499. break;
  500. case S_PREPARING_FOR_EVSE:
  501. case S_CHARGING:
  502. //if (pDcChargingInfo->RelayWeldingCheck != YES) {
  503. // break;
  504. //}
  505. if (pRegGunPNState->GunN == NO) {
  506. if (pDcChargingInfo->GroundFaultStatus != GFD_FAIL) {
  507. pOutputGunPNState->GunN = YES;
  508. } else {
  509. pOutputGunPNState->GunN = NO;
  510. }
  511. } else {
  512. if (pDcChargingInfo->GroundFaultStatus == GFD_FAIL) {
  513. pOutputGunPNState->GunN = NO;
  514. }
  515. }
  516. if (pRegGunPNState->GunP == NO) {
  517. if (pDcChargingInfo->GroundFaultStatus != GFD_FAIL) {
  518. pOutputGunPNState->GunP = YES;
  519. } else {
  520. pOutputGunPNState->GunP = NO;
  521. }
  522. } else {
  523. if (pDcChargingInfo->GroundFaultStatus == GFD_FAIL) {
  524. pOutputGunPNState->GunP = NO;
  525. }
  526. }
  527. break;
  528. case S_TERMINATING:
  529. case S_COMPLETE:
  530. case S_ALARM:
  531. if ((pDcChargingInfo->PresentChargingCurrent * 10) <= SEFETY_SWITCH_RELAY_CUR) {
  532. pOutputGunPNState->GunP = NO;
  533. pOutputGunPNState->GunN = NO;
  534. }
  535. if (pDcChargingInfo->GroundFaultStatus == GFD_FAIL) {
  536. pOutputGunPNState->GunP = NO;
  537. pOutputGunPNState->GunN = NO;
  538. }
  539. break;
  540. case S_CCS_PRECHARGE_ST0:
  541. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  542. break;
  543. #endif //defined DD360 || defined DD360Audi || defined DD360ComBox
  544. //if (pDcChargingInfo->Type == _Type_CCS_2 && targetID == 0x02) {
  545. // if (pRegPreChargingState->CcsPrecharge == NO) {
  546. // pOutputPreChargingState->CcsPrecharge = YES;
  547. // } else if (pRegPreChargingState->CcsPrecharge == YES) {
  548. // pRegGunPNState->GunP = NO;
  549. // }
  550. //}
  551. if (pDcChargingInfo->GroundFaultStatus == GFD_FAIL) {
  552. pOutputGunPNState->GunP = NO;
  553. pOutputGunPNState->GunN = NO;
  554. }
  555. break;
  556. case S_CCS_PRECHARGE_ST1:
  557. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  558. break;
  559. #endif //defined DD360 || defined DD360Audi || defined DD360ComBox
  560. //if (pDcChargingInfo->Type == _Type_CCS_2 && targetID == 0x02) {
  561. // if (pRegGunPNState->GunP == NO) {
  562. // pOutputGunPNState->GunP = YES;
  563. // } else if (pRegGunPNState->GunP == YES) {
  564. // pOutputPreChargingState->CcsPrecharge = NO;
  565. // }
  566. //}
  567. if (pDcChargingInfo->GroundFaultStatus == GFD_FAIL) {
  568. pOutputGunPNState->GunP = NO;
  569. pOutputGunPNState->GunN = NO;
  570. }
  571. break;
  572. }
  573. }
  574. // 確認 K1 K2 relay 的狀態
  575. void CheckK1K2RelayOutput(uint8_t index)
  576. {
  577. uint8_t targetID = 0;
  578. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  579. targetID = getCommTargetID(index);
  580. switch (targetID) {
  581. case 0x01:
  582. if (regRelay.relay_event.bits.Gun1_N == YES && regRelay.relay_event.bits.Gun1_P == YES) {
  583. pDcChargingInfo->RelayK1K2Status = YES;
  584. } else {
  585. pDcChargingInfo->RelayK1K2Status = NO;
  586. }
  587. if (pDcChargingInfo->Type == _Type_CCS_2) {
  588. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  589. if (regRelay.relay_event.bits.Gun1_N == YES && regRelay.relay_event.bits.CCS_Precharge == YES) {
  590. pDcChargingInfo->RelayKPK2Status = YES;
  591. } else {
  592. pDcChargingInfo->RelayKPK2Status = NO;
  593. }
  594. #else
  595. if (pDcChargingInfo->SystemStatus == S_CCS_PRECHARGE_ST0) {
  596. pDcChargingInfo->RelayKPK2Status = YES;
  597. } else {
  598. pDcChargingInfo->RelayKPK2Status = NO;
  599. }
  600. #endif //!defined DD360 && !defined DD360Audi
  601. }
  602. break;
  603. case 0x02:
  604. if (regRelay.relay_event.bits.Gun2_N == YES &&
  605. regRelay.relay_event.bits.Gun2_P == YES) {
  606. pDcChargingInfo->RelayK1K2Status = YES;
  607. } else {
  608. pDcChargingInfo->RelayK1K2Status = NO;
  609. }
  610. if (pDcChargingInfo->Type == _Type_CCS_2) {
  611. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  612. if (regRelay.relay_event.bits.Gun2_N == YES &&
  613. regRelay.relay_event.bits.CCS_Precharge == YES) {
  614. pDcChargingInfo->RelayKPK2Status = YES;
  615. } else {
  616. pDcChargingInfo->RelayKPK2Status = NO;
  617. }
  618. #else
  619. if (pDcChargingInfo->SystemStatus == S_CCS_PRECHARGE_ST0) {
  620. pDcChargingInfo->RelayKPK2Status = YES;
  621. } else {
  622. pDcChargingInfo->RelayKPK2Status = NO;
  623. }
  624. #endif //!defined DD360 && !defined DD360Audi
  625. }
  626. break;
  627. }
  628. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  629. //DS60-120 add
  630. if (pSysInfo->BridgeRelayStatus == YES) {
  631. if (regRelay.relay_event.bits.Gun1_Parallel_N == NO &&
  632. regRelay.relay_event.bits.Gun1_Parallel_P == NO) {
  633. pSysInfo->BridgeRelayStatus = NO;
  634. }
  635. } else if (pSysInfo->BridgeRelayStatus == NO) {
  636. if (regRelay.relay_event.bits.Gun1_Parallel_N == YES &&
  637. regRelay.relay_event.bits.Gun1_Parallel_P == YES) {
  638. pSysInfo->BridgeRelayStatus = YES;
  639. }
  640. }
  641. #else
  642. pSysInfo->BridgeRelayStatus = YES;
  643. #endif //!defined DD360 && !defined DD360Audi
  644. }
  645. void SetGfdConfig(uint8_t index, uint8_t resister)
  646. {
  647. Gfd_config gfd_config = {
  648. .index = index,
  649. .state = resister,
  650. };
  651. //log_info("************************GFD Vol = %d, GFD Res = %d \n", gfd_config.reqVol, gfd_config.resister);
  652. if (Config_Gfd_Value(Uart5Fd, ADDR_RELAY, &gfd_config) == PASS) {
  653. // log_info("Set reqVol = %f, resister = %d \n",
  654. // gfd_config.reqVol,
  655. // gfd_config.resister);
  656. }
  657. }
  658. void CableCheckDetected(uint8_t index)
  659. {
  660. uint8_t targetID = 0;
  661. struct ChargingInfoData *pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  662. // Cable Check
  663. // 當火線上的電壓 = 車端要求的電壓電流
  664. // _chargingData[targetGun]->EvBatterytargetVoltage
  665. // 才可以開始偵測 1s
  666. // Warning : Rgfd <= 150 歐/V 假設電壓為 500V 則~ Rgfd <= 75000 歐
  667. // Pre-Warning : 150 歐/V < Rgfd <= 500 歐/V 假設電壓為 500V 則 75000 歐 < Rgfd <= 250000
  668. // SO Normal : Rgfd > 500 歐/V 假設電壓為 500 V 則 Rgfd > 250000 歐
  669. if (pSysConfig->TotalConnectorCount == 1) {
  670. if (strncmp((char *)&pSysConfig->ModelName[7], "0", 1) != 0) {
  671. targetID = 0;
  672. } else if (strncmp((char *)&pSysConfig->ModelName[9], "0", 1) != 0) {
  673. targetID = 1;
  674. }
  675. } else {
  676. targetID = index;
  677. }
  678. if ((pDcChargingInfo->Type >= _Type_Chademo &&
  679. pDcChargingInfo->Type <= _Type_GB) ||
  680. (pDcChargingInfo->Type == 0x09 &&
  681. pSysConfig->AlwaysGfdFlag)
  682. ) {
  683. if ((pDcChargingInfo->SystemStatus >= S_PREPARING_FOR_EVSE &&
  684. pDcChargingInfo->SystemStatus < S_TERMINATING) ||
  685. (pDcChargingInfo->SystemStatus >= S_CCS_PRECHARGE_ST0 &&
  686. pDcChargingInfo->SystemStatus <= S_CCS_PRECHARGE_ST1)
  687. ) {
  688. //if ((pDcChargingInfo->SystemStatus == S_PREPARING_FOR_EVSE) &&
  689. // (pDcChargingInfo->RelayWeldingCheck == YES)
  690. // ) {
  691. if (pDcChargingInfo->SystemStatus == S_PREPARING_FOR_EVSE) {
  692. SetGfdConfig(targetID, GFD_CABLECHK);
  693. } else if ((pDcChargingInfo->SystemStatus >= S_CCS_PRECHARGE_ST0) &&
  694. (pDcChargingInfo->SystemStatus <= S_CCS_PRECHARGE_ST1)
  695. ) {
  696. SetGfdConfig(targetID, GFD_PRECHARGE);
  697. } else if ((pDcChargingInfo->SystemStatus >= S_CHARGING) &&
  698. (pDcChargingInfo->SystemStatus < S_TERMINATING)
  699. ) {
  700. if ((pDcChargingInfo->Type == _Type_GB) ||
  701. (pDcChargingInfo->Type == _Type_Chademo)
  702. ) {
  703. SetGfdConfig(targetID, GFD_IDLE);
  704. } else {
  705. SetGfdConfig(targetID, GFD_CHARGING);
  706. }
  707. }
  708. } else {
  709. SetGfdConfig(targetID, GFD_IDLE);
  710. }
  711. }
  712. }
  713. // 讀取 Relay 狀態
  714. void GetRelayOutputStatus(void)
  715. {
  716. if (Query_Relay_Output(Uart5Fd, ADDR_RELAY, &regRelay) == PASS) {
  717. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  718. regRelay.relay_event.bits.AC_Contactor = pSysInfo->AcContactorStatus;
  719. #endif //!defined DD360 && !defined DD360Audi
  720. }
  721. }
  722. // AC 三相輸入電壓
  723. void GetPresentInputVol(void)
  724. {
  725. static uint8_t _threePhaseOvp[3] = {0, 0, 0}; //DS60-120 add
  726. static uint8_t _threePhaseUvp[3] = {0, 0, 0}; //DS60-120 add
  727. PresentInputVoltage inputVoltage = {0};
  728. if (Query_Present_InputVoltage(Uart5Fd, ADDR_RELAY, &inputVoltage) == PASS) {
  729. // resolution : 0.1
  730. pSysInfo->InputVoltageR = ShmRelayModuleData->InputL1Volt = inputVoltage.L1N_L12;
  731. pSysInfo->InputVoltageS = ShmRelayModuleData->InputL2Volt = inputVoltage.L2N_L23;
  732. pSysInfo->InputVoltageT = ShmRelayModuleData->InputL3Volt = inputVoltage.L3N_L31;
  733. //********************************************************************************************************//
  734. // Vin (UVP)
  735. if (pSysInfo->ChargerType == _CHARGER_TYPE_IEC) {
  736. if (pAlarmCode->AlarmEvents.bits.SystemL1InputUVP == NO) {
  737. if (inputVoltage.L1N_L12 < VIN_MIN_VOLTAGE_IEC) {
  738. log_info("In Uvp L1N_L12 = %f \n", inputVoltage.L1N_L12);
  739. if (_threePhaseUvp[0] >= OVP_UVP_CHK_COUNT) {
  740. pAlarmCode->AlarmEvents.bits.SystemL1InputUVP = YES;
  741. } else {
  742. _threePhaseUvp[0] += 1;
  743. }
  744. }
  745. } else {
  746. if (inputVoltage.L1N_L12 > VIN_MIN_REV_VOLTAGE_IEC) {
  747. pAlarmCode->AlarmEvents.bits.SystemL1InputUVP = NO;
  748. _threePhaseUvp[0] = 0;
  749. }
  750. }
  751. if (pAlarmCode->AlarmEvents.bits.SystemL2InputUVP == NO) {
  752. if (inputVoltage.L2N_L23 < VIN_MIN_VOLTAGE_IEC) {
  753. log_info("In Uvp L2N_L23 = %f \n", inputVoltage.L2N_L23);
  754. if (_threePhaseUvp[1] >= OVP_UVP_CHK_COUNT) {
  755. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP = YES;
  756. } else {
  757. _threePhaseUvp[1] += 1;
  758. }
  759. }
  760. } else {
  761. if (inputVoltage.L2N_L23 > VIN_MIN_REV_VOLTAGE_IEC) {
  762. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP = NO;
  763. _threePhaseUvp[1] = 0;
  764. }
  765. }
  766. if (pAlarmCode->AlarmEvents.bits.SystemL3InputUVP == NO) {
  767. if (inputVoltage.L3N_L31 < VIN_MIN_VOLTAGE_IEC) {
  768. log_info("In Uvp L3N_L31 = %f \n", inputVoltage.L3N_L31);
  769. if (_threePhaseUvp[2] >= OVP_UVP_CHK_COUNT) {
  770. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP = YES;
  771. } else {
  772. _threePhaseUvp[2] += 1;
  773. }
  774. }
  775. } else {
  776. if (inputVoltage.L3N_L31 > VIN_MIN_REV_VOLTAGE_IEC) {
  777. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP = NO;
  778. _threePhaseUvp[2] = 0;
  779. }
  780. }
  781. } else if (pSysInfo->ChargerType == _CHARGER_TYPE_UL) {
  782. if (pAlarmCode->AlarmEvents.bits.SystemL1InputUVP == NO) {
  783. if (inputVoltage.L1N_L12 < VIN_MIN_VOLTAGE_UL) {
  784. log_info("In Uvp L1N_L12 = %f \n", inputVoltage.L1N_L12);
  785. if (_threePhaseUvp[0] >= OVP_UVP_CHK_COUNT) {
  786. pAlarmCode->AlarmEvents.bits.SystemL1InputUVP = YES;
  787. } else {
  788. _threePhaseUvp[0] += 1;
  789. }
  790. }
  791. } else {
  792. if (inputVoltage.L1N_L12 > VIN_MIN_REV_VOLTAGE_UL) {
  793. pAlarmCode->AlarmEvents.bits.SystemL1InputUVP = NO;
  794. _threePhaseUvp[0] = 0;
  795. }
  796. }
  797. if (pAlarmCode->AlarmEvents.bits.SystemL2InputUVP == NO) {
  798. if (inputVoltage.L2N_L23 < VIN_MIN_VOLTAGE_UL) {
  799. log_info("In Uvp L2N_L23 = %f \n", inputVoltage.L2N_L23);
  800. if (_threePhaseUvp[1] >= OVP_UVP_CHK_COUNT) {
  801. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP = YES;
  802. } else {
  803. _threePhaseUvp[1] += 1;
  804. }
  805. }
  806. } else {
  807. if (inputVoltage.L2N_L23 > VIN_MIN_REV_VOLTAGE_UL) {
  808. pAlarmCode->AlarmEvents.bits.SystemL2InputUVP = NO;
  809. _threePhaseUvp[1] = 0;
  810. }
  811. }
  812. if (pAlarmCode->AlarmEvents.bits.SystemL3InputUVP == NO) {
  813. if (inputVoltage.L3N_L31 < VIN_MIN_VOLTAGE_UL) {
  814. log_info("In Uvp L3N_L31 = %f \n", inputVoltage.L3N_L31);
  815. if (_threePhaseUvp[2] >= OVP_UVP_CHK_COUNT) {
  816. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP = YES;
  817. } else {
  818. _threePhaseUvp[2] += 1;
  819. }
  820. }
  821. } else {
  822. if (inputVoltage.L3N_L31 > VIN_MIN_REV_VOLTAGE_UL) {
  823. pAlarmCode->AlarmEvents.bits.SystemL3InputUVP = NO;
  824. _threePhaseUvp[2] = 0;
  825. }
  826. }
  827. }
  828. //********************************************************************************************************//
  829. // Vin (OVP)
  830. if (pSysInfo->ChargerType == _CHARGER_TYPE_IEC) {
  831. if (pAlarmCode->AlarmEvents.bits.SystemL1InputOVP == NO) {
  832. if (inputVoltage.L1N_L12 > VIN_MAX_VOLTAGE_IEC) {
  833. log_info("In Ovp L1N_L12 = %f \n", inputVoltage.L1N_L12);
  834. if (_threePhaseOvp[0] >= OVP_UVP_CHK_COUNT) {
  835. pAlarmCode->AlarmEvents.bits.SystemL1InputOVP = YES;
  836. } else {
  837. _threePhaseOvp[0] += 1;
  838. }
  839. }
  840. } else {
  841. if (inputVoltage.L1N_L12 < VIN_MAX_REV_VOLTAGE_IEC) {
  842. pAlarmCode->AlarmEvents.bits.SystemL1InputOVP = NO;
  843. _threePhaseOvp[0] = 0;
  844. }
  845. }
  846. if (pAlarmCode->AlarmEvents.bits.SystemL2InputOVP == NO) {
  847. if (inputVoltage.L2N_L23 > VIN_MAX_VOLTAGE_IEC) {
  848. log_info("In Ovp L2N_L23 = %f \n", inputVoltage.L2N_L23);
  849. if (_threePhaseOvp[1] >= OVP_UVP_CHK_COUNT) {
  850. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP = YES;
  851. } else {
  852. _threePhaseOvp[1] += 1;
  853. }
  854. }
  855. } else {
  856. if (inputVoltage.L2N_L23 < VIN_MAX_REV_VOLTAGE_IEC) {
  857. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP = NO;
  858. _threePhaseOvp[1] = 0;
  859. }
  860. }
  861. if (pAlarmCode->AlarmEvents.bits.SystemL3InputOVP == NO) {
  862. if (inputVoltage.L3N_L31 > VIN_MAX_VOLTAGE_IEC) {
  863. log_info("In Ovp L3N_L31 = %f \n", inputVoltage.L3N_L31);
  864. if (_threePhaseOvp[2] >= OVP_UVP_CHK_COUNT) {
  865. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP = YES;
  866. } else {
  867. _threePhaseOvp[2] += 1;
  868. }
  869. }
  870. } else {
  871. if (inputVoltage.L3N_L31 < VIN_MAX_REV_VOLTAGE_IEC) {
  872. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP = NO;
  873. _threePhaseOvp[2] = 0;
  874. }
  875. }
  876. } else if (pSysInfo->ChargerType == _CHARGER_TYPE_UL) {
  877. if (pAlarmCode->AlarmEvents.bits.SystemL1InputOVP == NO) {
  878. if (inputVoltage.L1N_L12 > VIN_MAX_VOLTAGE_UL) {
  879. log_info("In Ovp L1N_L12 = %f \n", inputVoltage.L1N_L12);
  880. if (_threePhaseOvp[0] >= OVP_UVP_CHK_COUNT) {
  881. pAlarmCode->AlarmEvents.bits.SystemL1InputOVP = YES;
  882. } else {
  883. _threePhaseOvp[0] += 0;
  884. }
  885. }
  886. } else {
  887. if (inputVoltage.L1N_L12 < VIN_MAX_REV_VOLTAGE_UL) {
  888. pAlarmCode->AlarmEvents.bits.SystemL1InputOVP = NO;
  889. _threePhaseOvp[0] = 0;
  890. }
  891. }
  892. if (pAlarmCode->AlarmEvents.bits.SystemL2InputOVP == NO) {
  893. if (inputVoltage.L2N_L23 > VIN_MAX_VOLTAGE_UL) {
  894. log_info("In Ovp L2N_L23 = %f \n", inputVoltage.L2N_L23);
  895. if (_threePhaseOvp[1] >= OVP_UVP_CHK_COUNT) {
  896. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP = YES;
  897. } else {
  898. _threePhaseOvp[1] += 0;
  899. }
  900. }
  901. } else {
  902. if (inputVoltage.L2N_L23 < VIN_MAX_REV_VOLTAGE_UL) {
  903. pAlarmCode->AlarmEvents.bits.SystemL2InputOVP = NO;
  904. _threePhaseOvp[1] = 0;
  905. }
  906. }
  907. if (pAlarmCode->AlarmEvents.bits.SystemL2InputOVP == NO) {
  908. if (inputVoltage.L3N_L31 > VIN_MAX_VOLTAGE_UL) {
  909. log_info("In Ovp L3N_L31 = %f \n", inputVoltage.L3N_L31);
  910. if (_threePhaseOvp[2] >= OVP_UVP_CHK_COUNT) {
  911. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP = YES;
  912. } else {
  913. _threePhaseOvp[2] += 1;
  914. }
  915. }
  916. } else {
  917. if (inputVoltage.L3N_L31 < VIN_MAX_REV_VOLTAGE_UL) {
  918. pAlarmCode->AlarmEvents.bits.SystemL3InputOVP = NO;
  919. _threePhaseOvp[2] = 0;
  920. }
  921. }
  922. }
  923. }
  924. }
  925. // 左右槍的 Relay 前後的輸出電壓
  926. void GetPersentOutputVol(void)
  927. {
  928. uint8_t index = 0;
  929. uint8_t targetID = 0;
  930. struct ChargingInfoData *pDcChargingInfo = NULL;
  931. PresentOutputVoltage outputVoltage = {0};
  932. if (Query_Present_OutputVoltage(Uart5Fd, ADDR_RELAY, &outputVoltage) != PASS) {
  933. return;
  934. }
  935. //log_info("Conn1 fuse 1 = %f \n", outputVoltage.behindFuse_Voltage_C1);
  936. //log_info("Conn1 relay 1 = %f \n", outputVoltage.behindRelay_Voltage_C1);
  937. //log_info("Conn2 fuse 2 = %f \n", outputVoltage.behindFuse_Voltage_C2);
  938. //log_info("Conn2 relay 2 = %f \n", outputVoltage.behindRelay_Voltage_C2);
  939. //log_info("outputVoltage.behindFuse_Voltage_C1 = %f \n", outputVoltage.behindFuse_Voltage_C1);
  940. //log_info("outputVoltage.behindFuse_Voltage_C2 = %f \n", outputVoltage.behindFuse_Voltage_C2);
  941. ShmRelayModuleData->Gun1FuseOutputVolt = outputVoltage.behindFuse_Voltage_C1;
  942. ShmRelayModuleData->Gun1RelayOutputVolt = outputVoltage.behindRelay_Voltage_C1;
  943. ShmRelayModuleData->Gun2FuseOutputVolt = outputVoltage.behindFuse_Voltage_C2;
  944. ShmRelayModuleData->Gun2RelayOutputVolt = outputVoltage.behindRelay_Voltage_C2;
  945. for (index = 0; index < pSysConfig->TotalConnectorCount; index++) {
  946. pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(index);
  947. targetID = getCommTargetID(index);
  948. switch (targetID) {
  949. case 0x01:
  950. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  951. pDcChargingInfo->FireChargingVoltage = ShmRelayModuleData->Gun1RelayOutputVolt;
  952. pDcChargingInfo->PresentChargingCurrent = ((float)ShmRelayModuleData->Gun1FuseOutputVolt) / 10;
  953. pDcChargingInfo->PresentChargingVoltage = ((float)pDcChargingInfo->FireChargingVoltage) / 10;
  954. pDcChargingInfo->FuseChargingVoltage = pDcChargingInfo->FireChargingVoltage;
  955. break;
  956. #endif //defined DD360 || defined DD360Audi || defined DD360ComBox
  957. pDcChargingInfo->FireChargingVoltage = ShmRelayModuleData->Gun1RelayOutputVolt;
  958. pDcChargingInfo->FuseChargingVoltage = ShmRelayModuleData->Gun1FuseOutputVolt;
  959. break;
  960. case 0x02:
  961. #if defined DD360 || defined DD360Audi || defined DD360ComBox
  962. pDcChargingInfo->FireChargingVoltage = ShmRelayModuleData->Gun2RelayOutputVolt;
  963. pDcChargingInfo->PresentChargingCurrent = ((float)ShmRelayModuleData->Gun2FuseOutputVolt) / 10;
  964. pDcChargingInfo->PresentChargingVoltage = ((float)pDcChargingInfo->FireChargingVoltage) / 10;
  965. pDcChargingInfo->FuseChargingVoltage = pDcChargingInfo->FireChargingVoltage;
  966. break;
  967. #endif //defined DD360 || defined DD360Audi || defined DD360ComBox
  968. pDcChargingInfo->FireChargingVoltage = ShmRelayModuleData->Gun2RelayOutputVolt;
  969. pDcChargingInfo->FuseChargingVoltage = ShmRelayModuleData->Gun2FuseOutputVolt;
  970. break;
  971. }
  972. //log_info("%d persent vol = %f, cur = %f\r\n",
  973. // index,
  974. // pDcChargingInfo->PresentChargingVoltage,
  975. // pDcChargingInfo->PresentChargingCurrent);
  976. //unsigned short Ovp = 0;
  977. //unsigned short Ocp = 0;
  978. //Ovp = MIN [VOUT_MAX_VOLTAGE, EV_BATTERY_VOLTAGE] // 最大輸出電壓與電池電壓最大值
  979. //Ocp = MIN [IOUT_MAX_CURRENT, EV_CURRENT_REQ] // 最大輸出電流與需求電流最小值
  980. //if (pDcChargingInfo->Type == _Type_Chademo) {
  981. // //Ovp = MaxValue(pDcChargingInfo->MaximumChargingVoltage, pDcChargingInfo->EvBatteryMaxVoltage);
  982. // //Ocp = MaxValue(pDcChargingInfo->PresentChargingCurrent, ShmCHAdeMOData->ev[pDcChargingInfo->type_index].ChargingCurrentRequest);
  983. //} else if (pDcChargingInfo->Type == _Type_CCS_2) {
  984. //}
  985. }
  986. }
  987. void SetRtcData_Relay(void)
  988. {
  989. struct timeb csuTime;
  990. struct tm *tmCSU;
  991. Rtc rtc = {0};
  992. ftime(&csuTime);
  993. tmCSU = localtime(&csuTime.time);
  994. // log_info("Time : %04d-%02d-%02d %02d:%02d:%02d \n", tmCSU->tm_year + 1900,
  995. // tmCSU->tm_mon + 1, tmCSU->tm_mday, tmCSU->tm_hour, tmCSU->tm_min,
  996. // tmCSU->tm_sec);
  997. rtc.RtcData[0] = '0' + (tmCSU->tm_year + 1900) / 1000 % 10;
  998. rtc.RtcData[1] = '0' + (tmCSU->tm_year + 1900) / 100 % 10;
  999. rtc.RtcData[2] = '0' + (tmCSU->tm_year + 1900) / 10 % 10;
  1000. rtc.RtcData[3] = '0' + (tmCSU->tm_year + 1900) / 1 % 10;
  1001. rtc.RtcData[4] = '0' + (tmCSU->tm_mon + 1) / 10 % 10;
  1002. rtc.RtcData[5] = '0' + (tmCSU->tm_mon + 1) / 1 % 10;
  1003. rtc.RtcData[6] = '0' + (tmCSU->tm_mday) / 10 % 10;
  1004. rtc.RtcData[7] = '0' + (tmCSU->tm_mday) / 1 % 10;
  1005. rtc.RtcData[8] = '0' + (tmCSU->tm_hour) / 10 % 10;
  1006. rtc.RtcData[9] = '0' + (tmCSU->tm_hour) / 1 % 10;
  1007. rtc.RtcData[10] = '0' + (tmCSU->tm_min) / 10 % 10;
  1008. rtc.RtcData[11] = '0' + (tmCSU->tm_min) / 1 % 10;
  1009. rtc.RtcData[12] = '0' + (tmCSU->tm_sec) / 10 % 10;
  1010. rtc.RtcData[13] = '0' + (tmCSU->tm_sec) / 1 % 10;
  1011. if (Config_Rtc_Data(Uart5Fd, ADDR_RELAY, &rtc) == PASS) {
  1012. //log_info("SetRtc (RB) sucessfully. \n");
  1013. }
  1014. }
  1015. void SetModelName_Relay(void)
  1016. {
  1017. if (Config_Model_Name(Uart5Fd, ADDR_RELAY, pSysConfig->ModelName) == PASS) {
  1018. //log_info("Set Model name (RB) PASS = %s \n", pSysConfig->ModelName);
  1019. }
  1020. }
  1021. void GetFwAndHwVersion_Relay(void)
  1022. {
  1023. Ver ver = {0};
  1024. if (Query_FW_Ver(Uart5Fd, ADDR_RELAY, &ver) == PASS) {
  1025. // RelayModuleData
  1026. strcpy((char *)ShmRelayModuleData->version, ver.Version_FW);
  1027. // SystemInfo
  1028. strcpy((char *)pSysInfo->RelayModuleFwRev, ver.Version_FW);
  1029. //log_info("GetFwAndHwVersion_Relay s1 = %s \n", ver.Version_FW);
  1030. }
  1031. if (Query_HW_Ver(Uart5Fd, ADDR_RELAY, &ver) == PASS) {
  1032. // SystemInfo
  1033. strcpy((char *)pSysInfo->RelayModuleHwRev, ver.Version_FW);
  1034. //log_info("GetFwAndHwVersion_Relay s2 = %s \n", ver.Version_HW);
  1035. }
  1036. }
  1037. static void outputRelayInit(int fd)
  1038. {
  1039. memset((uint8_t *)&outputRelay, 0, sizeof(Relay));
  1040. if (Config_Relay_Output(fd, ADDR_RELAY, &outputRelay) != PASS) {
  1041. log_info("Config_Relay_Output fail \n");
  1042. }
  1043. }
  1044. static bool IsRelayProcessNeedPause(void)
  1045. {
  1046. bool _pause = false;
  1047. static bool isPause = false;
  1048. struct ChargingInfoData *pDcChargingInfo = NULL;
  1049. for (uint8_t i = 0; i < pSysConfig->TotalConnectorCount; i++)
  1050. {
  1051. pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(i);
  1052. if(pDcChargingInfo->SystemStatus == S_UPDATE)
  1053. {
  1054. _pause = true;
  1055. }
  1056. }
  1057. if(isPause != _pause)
  1058. {
  1059. log_info("Relay Process Now Is %s \n", _pause == true ? "Paused" : "Continued");
  1060. }
  1061. isPause = _pause;
  1062. return _pause;
  1063. }
  1064. static void SetFanModuleSpeed(void)
  1065. {
  1066. {
  1067. FanSpeed _fanSpeed = {0};
  1068. _setFanSpeed += fanSpeedSmoothValue;
  1069. if (_setFanSpeed >= ShmFanModuleData->SetFan1Speed) {
  1070. _setFanSpeed = ShmFanModuleData->SetFan1Speed;
  1071. }
  1072. //printf("_setFanSpeed = %d \n", _setFanSpeed);
  1073. _fanSpeed.speed[0] = _setFanSpeed;
  1074. _fanSpeed.speed[1] = _setFanSpeed;
  1075. _fanSpeed.speed[2] = _setFanSpeed;
  1076. _fanSpeed.speed[3] = _setFanSpeed;
  1077. if (Config_Fan_Speed(Uart5Fd, ADDR_FAN, &_fanSpeed) == PASS) {
  1078. //log_info("successfully Fan\n");
  1079. }
  1080. }
  1081. }
  1082. // 風扇速度
  1083. static void GetFanSpeed(void)
  1084. {
  1085. FanSpeed fanSpeed = {0};
  1086. //log_info("Get fan board speed \n");
  1087. if (Query_Fan_Speed(Uart5Fd, ADDR_FAN, &fanSpeed) == PASS) {
  1088. ShmFanModuleData->PresentFan1Speed = fanSpeed.speed[0];
  1089. ShmFanModuleData->PresentFan2Speed = fanSpeed.speed[1];
  1090. ShmFanModuleData->PresentFan3Speed = fanSpeed.speed[2];
  1091. ShmFanModuleData->PresentFan4Speed = fanSpeed.speed[3];
  1092. // log_info("SystemFanRotaSpeed_1 = %d \n", fanSpeed.speed[0]);
  1093. // log_info("SystemFanRotaSpeed_2 = %d \n", fanSpeed.speed[1]);
  1094. // log_info("SystemFanRotaSpeed_3 = %d \n", fanSpeed.speed[2]);
  1095. // log_info("SystemFanRotaSpeed_4 = %d \n", fanSpeed.speed[3]);
  1096. // Config_Fan_Speed(Uart5Fd, ADDR_FAN, &fanSpeed[0]);
  1097. //SysInfoData (SystemFanRotaSpeed)
  1098. }
  1099. }
  1100. static void GetFanSpeedByFunction(void)
  1101. {
  1102. if (pSysConfig->SwitchDebugFlag == YES) {
  1103. return;
  1104. }
  1105. // 風控修改 :
  1106. // ******************************************************* //
  1107. //
  1108. // 當前PSU輸出總 KW PSU Temp
  1109. // 30 x -------------------- x ---------- + 14 x (PSU Temp - 45)
  1110. // 當前樁最大功率 KW 45
  1111. //
  1112. // ******************************************************* //
  1113. // 當前樁最大功率 KW : ShmPsuData->SystemAvailablePower
  1114. uint32_t _maxPower = ShmPsuData->SystemAvailablePower;
  1115. // 當前PSU輸出總 KW & PSU Temp :
  1116. uint8_t temp = 0;
  1117. uint8_t index = 0;
  1118. uint8_t count = 0;
  1119. uint8_t gunIndex = 0;
  1120. uint8_t _temp_diff = 0;
  1121. float power = 0;
  1122. double _pw_rate = 0;
  1123. double _temp_rate = 0;
  1124. struct ChargingInfoData *pDcChargingInfo = NULL;
  1125. for (index = 0; index < ShmPsuData->GroupCount; index++) {
  1126. for (count = 0; count < ShmPsuData->PsuGroup[index].GroupPresentPsuQuantity; count++) {
  1127. if (temp < ShmPsuData->PsuGroup[index].PsuModule[count].ExletTemp) {
  1128. temp = ShmPsuData->PsuGroup[index].PsuModule[count].ExletTemp;
  1129. }
  1130. }
  1131. }
  1132. for (gunIndex = 0; gunIndex < pSysConfig->TotalConnectorCount; gunIndex++) {
  1133. pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(gunIndex);
  1134. power += (pDcChargingInfo->PresentChargingPower * 10);
  1135. }
  1136. if (_maxPower > 0) {
  1137. _pw_rate = power / (double)_maxPower;
  1138. }
  1139. if (temp > 0) {
  1140. _temp_rate = (double)temp / 50;
  1141. }
  1142. if (temp > 45) {
  1143. _temp_diff = temp - 70;
  1144. }
  1145. ShmFanModuleData->TestFanSpeed = (((50 * _pw_rate * _temp_rate) + (0.5 * _temp_diff)) / 100) * MAX_FAN_SPEED;
  1146. if (ShmFanModuleData->TestFanSpeed > MAX_FAN_SPEED) {
  1147. ShmFanModuleData->TestFanSpeed = MAX_FAN_SPEED;
  1148. }
  1149. if (ShmFanModuleData->TestFanSpeed < 0) {
  1150. ShmFanModuleData->TestFanSpeed = 0;
  1151. }
  1152. //
  1153. // printf("power = %f \n", power);
  1154. // printf("_maxPower = %d \n", _maxPower);
  1155. // printf("temp = %d \n", temp);
  1156. //
  1157. // printf("_pw_rate = %f \n", _pw_rate);
  1158. // printf("_temp_rate = %f \n", _temp_rate);
  1159. // printf("_temp_diff = %d \n", _temp_diff);
  1160. // printf("fan rate = %f \n", (30 * _pw_rate * _temp_rate + 14 * _temp_diff));
  1161. // printf("ShmFanModuleData->TestFanSpeed = %d \n", ShmFanModuleData->TestFanSpeed);
  1162. }
  1163. static void SetRtcData_Fan(void)
  1164. {
  1165. struct timeb csuTime;
  1166. struct tm *tmCSU;
  1167. Rtc rtc = {0};
  1168. ftime(&csuTime);
  1169. tmCSU = localtime(&csuTime.time);
  1170. // log_info("Time : %04d-%02d-%02d %02d:%02d:%02d \n", tmCSU->tm_year + 1900,
  1171. // tmCSU->tm_mon + 1, tmCSU->tm_mday, tmCSU->tm_hour, tmCSU->tm_min,
  1172. // tmCSU->tm_sec);
  1173. rtc.RtcData[0] = '0' + (tmCSU->tm_year + 1900) / 1000 % 10;
  1174. rtc.RtcData[1] = '0' + (tmCSU->tm_year + 1900) / 100 % 10;
  1175. rtc.RtcData[2] = '0' + (tmCSU->tm_year + 1900) / 10 % 10;
  1176. rtc.RtcData[3] = '0' + (tmCSU->tm_year + 1900) / 1 % 10;
  1177. rtc.RtcData[4] = '0' + (tmCSU->tm_mon + 1) / 10 % 10;
  1178. rtc.RtcData[5] = '0' + (tmCSU->tm_mon + 1) / 1 % 10;
  1179. rtc.RtcData[6] = '0' + (tmCSU->tm_mday) / 10 % 10;
  1180. rtc.RtcData[7] = '0' + (tmCSU->tm_mday) / 1 % 10;
  1181. rtc.RtcData[8] = '0' + (tmCSU->tm_hour) / 10 % 10;
  1182. rtc.RtcData[9] = '0' + (tmCSU->tm_hour) / 1 % 10;
  1183. rtc.RtcData[10] = '0' + (tmCSU->tm_min) / 10 % 10;
  1184. rtc.RtcData[11] = '0' + (tmCSU->tm_min) / 1 % 10;
  1185. rtc.RtcData[12] = '0' + (tmCSU->tm_sec) / 10 % 10;
  1186. rtc.RtcData[13] = '0' + (tmCSU->tm_sec) / 1 % 10;
  1187. if (Config_Rtc_Data(Uart5Fd, ADDR_FAN, &rtc) == PASS) {
  1188. //log_info("SetRtc (FB) sucessfully. \n");
  1189. }
  1190. }
  1191. static void SetModelName_Fan(void)
  1192. {
  1193. if (Config_Model_Name(Uart5Fd, ADDR_FAN, pSysConfig->ModelName) == PASS) {
  1194. log_info("Set Model name PASS = %s \n", pSysConfig->ModelName);
  1195. }
  1196. }
  1197. static void GetFwAndHwVersion_Fan(void)
  1198. {
  1199. Ver ver = {0};
  1200. if (Query_FW_Ver(Uart5Fd, ADDR_FAN, &ver) == PASS) {
  1201. // FanModuleData
  1202. strcpy((char *)ShmFanModuleData->version, ver.Version_FW);
  1203. // SystemInfo
  1204. strcpy((char *)pSysInfo->FanModuleFwRev, ver.Version_FW);
  1205. //log_info("GetFwAndHwVersion_Fan s1 = %s \n", ver.Version_FW);
  1206. }
  1207. if (Query_HW_Ver(Uart5Fd, ADDR_FAN, &ver) == PASS) {
  1208. // SystemInfo
  1209. strcpy((char *)pSysInfo->FanModuleHwRev, ver.Version_FW);
  1210. //log_info("GetFwAndHwVersion_Fan s2 = %s \n", ver.Version_HW);
  1211. }
  1212. }
  1213. static void fanBoardSelfTest(void)
  1214. {
  1215. if (ShmFanModuleData->SelfTest_Comp == YES) {
  1216. return;
  1217. }
  1218. GetFwAndHwVersion_Fan();
  1219. SetModelName_Fan();
  1220. SetRtcData_Fan();
  1221. sleep(1);
  1222. gettimeofday(&gFanBoardRunTimer, NULL);
  1223. }
  1224. static void fanBoardPorcess(void)
  1225. {
  1226. if (ShmFanModuleData->SelfTest_Comp == NO) {
  1227. return;
  1228. }
  1229. if (ShmFanModuleData->SelfTest_Comp == YES ||
  1230. strlen((char *)pSysInfo->FanModuleFwRev) != 0 ||
  1231. pSysInfo->FanModuleFwRev[0] != '\0') {
  1232. ShmFanModuleData->SelfTest_Comp = YES;
  1233. if (GetTimeoutValue(gFanBoardRunTimer) / 1000 >= 1000) {
  1234. //GetPsuTempForFanSpeed();
  1235. GetFanSpeedByFunction();
  1236. GetFanSpeed();
  1237. pSysInfo->SystemFanRotaSpeed = _setFanSpeed;
  1238. gettimeofday(&gFanBoardRunTimer, NULL);
  1239. ShmFanModuleData->SetFan1Speed = ShmFanModuleData->TestFanSpeed;
  1240. ShmFanModuleData->SetFan2Speed = ShmFanModuleData->TestFanSpeed;
  1241. ShmFanModuleData->SetFan3Speed = ShmFanModuleData->TestFanSpeed;
  1242. ShmFanModuleData->SetFan4Speed = ShmFanModuleData->TestFanSpeed;
  1243. //log_info("set fan = %d \n", ShmFanModuleData->SetFan1Speed);
  1244. SetFanModuleSpeed();
  1245. }
  1246. }
  1247. }
  1248. static void GetFwAndHwVersion_Led(void)
  1249. {
  1250. Ver ver = {0};
  1251. if (Query_FW_Ver(Uart5Fd, ADDR_LED, &ver) == PASS) {
  1252. // LedModuleData
  1253. strcpy((char *) ShmLedModuleData->version, ver.Version_FW);
  1254. // SystemInfo
  1255. strcpy((char *) pSysInfo->LedModuleFwRev, ver.Version_FW);
  1256. log_info("GetFwAndHwVersion_Led s1 = %s \n", ver.Version_FW);
  1257. ShmLedModuleData->SelfTest_Comp = YES;
  1258. } else {
  1259. //log_info("GetFwAndHwVersion_Led fail \n");
  1260. }
  1261. // if (Query_HW_Ver(Uart5Fd, ADDR_LED, &ver) == PASS)
  1262. // {
  1263. // // SystemInfo
  1264. // strcpy((char *) pSysInfo->RelayModuleHwRev, ver.Version_FW);
  1265. // //log_info("GetFwAndHwVersion_Relay s2 = %s \n", ver.Version_HW);
  1266. // }
  1267. }
  1268. static bool IsNoneMatchLedColor(void)
  1269. {
  1270. bool result = false;
  1271. if (cur_led_color.Connect_1_Red != led_color.Connect_1_Red ||
  1272. cur_led_color.Connect_1_Green != led_color.Connect_1_Green ||
  1273. cur_led_color.Connect_1_Blue != led_color.Connect_1_Blue ||
  1274. cur_led_color.Connect_2_Red != led_color.Connect_2_Red ||
  1275. cur_led_color.Connect_2_Green != led_color.Connect_2_Green ||
  1276. cur_led_color.Connect_2_Blue != led_color.Connect_2_Blue) {
  1277. result = true;
  1278. }
  1279. return result;
  1280. }
  1281. //static void SetLedColor(struct ChargingInfoData *chargingData_1, struct ChargingInfoData *chargingData_2)
  1282. static void SetLedColor(void)
  1283. {
  1284. static uint8_t _checkLedChanged = 3;
  1285. struct ChargingInfoData *chargingData_1 = NULL;
  1286. struct ChargingInfoData *chargingData_2 = NULL;
  1287. uint8_t _colorBuf = COLOR_MAX_LV * LED_INTENSITY_BRIGHTEST;
  1288. if (pSysConfig->TotalConnectorCount == 1) {
  1289. chargingData_1 = (struct ChargingInfoData *)GetDcChargingInfoData(0);
  1290. chargingData_2 = (struct ChargingInfoData *)GetDcChargingInfoData(0);
  1291. } else if (pSysConfig->TotalConnectorCount == 2) {
  1292. chargingData_1 = (struct ChargingInfoData *)GetDcChargingInfoData(0);
  1293. chargingData_2 = (struct ChargingInfoData *)GetDcChargingInfoData(1);
  1294. }
  1295. if (pSysConfig->LedInfo.Intensity == _LED_INTENSITY_DARKEST) {
  1296. _colorBuf = COLOR_MAX_LV * LED_INTENSITY_DARKEST;
  1297. } else if (pSysConfig->LedInfo.Intensity == _LED_INTENSITY_MEDIUM) {
  1298. _colorBuf = COLOR_MAX_LV * LED_INTENSITY_MEDIUM;
  1299. }
  1300. //printf("chargingData_1->SystemStatus=%d\n",chargingData_1->SystemStatus);
  1301. //printf("chargingData_2->SystemStatus=%d\n",chargingData_2->SystemStatus);
  1302. //printf("pSysWarning->Level=%d\n",pSysWarning->Level);
  1303. if (pSysWarning->Level == 2) {
  1304. led_color.Connect_1_Green = COLOR_MIN_LV;
  1305. led_color.Connect_1_Blue = COLOR_MIN_LV;
  1306. led_color.Connect_1_Red = _colorBuf;
  1307. led_color.Connect_2_Green = COLOR_MIN_LV;
  1308. led_color.Connect_2_Blue = COLOR_MIN_LV;
  1309. led_color.Connect_2_Red = _colorBuf;
  1310. } else {
  1311. if (pSysInfo->IsAlternatvieConf) {
  1312. if ((chargingData_1->SystemStatus == S_BOOTING ||
  1313. chargingData_1->SystemStatus == S_IDLE ||
  1314. chargingData_1->SystemStatus == S_RESERVATION) &&
  1315. (chargingData_2->SystemStatus == S_BOOTING ||
  1316. chargingData_2->SystemStatus == S_IDLE ||
  1317. chargingData_2->SystemStatus == S_RESERVATION)) {
  1318. #if defined DD360Audi
  1319. led_color.Connect_1_Green = _colorBuf;
  1320. led_color.Connect_1_Blue = _colorBuf;
  1321. led_color.Connect_1_Red = _colorBuf;
  1322. led_color.Connect_2_Green = _colorBuf;
  1323. led_color.Connect_2_Blue = _colorBuf;
  1324. led_color.Connect_2_Red = _colorBuf;
  1325. #else
  1326. led_color.Connect_1_Green = _colorBuf;
  1327. led_color.Connect_1_Blue = COLOR_MIN_LV;
  1328. led_color.Connect_1_Red = COLOR_MIN_LV;
  1329. led_color.Connect_2_Green = _colorBuf;
  1330. led_color.Connect_2_Blue = COLOR_MIN_LV;
  1331. led_color.Connect_2_Red = COLOR_MIN_LV;
  1332. #endif
  1333. } else if ((chargingData_1->SystemStatus >= S_AUTHORIZING &&
  1334. chargingData_1->SystemStatus <= S_COMPLETE) ||
  1335. (chargingData_1->SystemStatus >= S_CCS_PRECHARGE_ST0 &&
  1336. chargingData_1->SystemStatus <= S_CCS_PRECHARGE_ST1) ||
  1337. (chargingData_2->SystemStatus >= S_AUTHORIZING &&
  1338. chargingData_2->SystemStatus <= S_COMPLETE) ||
  1339. (chargingData_2->SystemStatus >= S_CCS_PRECHARGE_ST0 &&
  1340. chargingData_2->SystemStatus <= S_CCS_PRECHARGE_ST1)) {
  1341. led_color.Connect_1_Green = COLOR_MIN_LV;
  1342. led_color.Connect_1_Blue = _colorBuf;
  1343. led_color.Connect_1_Red = COLOR_MIN_LV;
  1344. led_color.Connect_2_Green = COLOR_MIN_LV;
  1345. led_color.Connect_2_Blue = _colorBuf;
  1346. led_color.Connect_2_Red = COLOR_MIN_LV;
  1347. }
  1348. } else {
  1349. if (chargingData_1->SystemStatus == S_BOOTING ||
  1350. chargingData_1->SystemStatus == S_IDLE ||
  1351. chargingData_1->SystemStatus == S_RESERVATION ||
  1352. chargingData_1->SystemStatus == S_MAINTAIN) {
  1353. if (chargingData_1->IsAvailable == NO) { //For Audi
  1354. led_color.Connect_1_Green = COLOR_MIN_LV;
  1355. led_color.Connect_1_Blue = COLOR_MIN_LV;
  1356. led_color.Connect_1_Red = _colorBuf;
  1357. } else {
  1358. #if defined DD360Audi
  1359. led_color.Connect_1_Green = _colorBuf;
  1360. led_color.Connect_1_Blue = _colorBuf;
  1361. led_color.Connect_1_Red = _colorBuf;
  1362. #else
  1363. led_color.Connect_1_Green = _colorBuf;
  1364. led_color.Connect_1_Blue = COLOR_MIN_LV;
  1365. led_color.Connect_1_Red = COLOR_MIN_LV;
  1366. #endif
  1367. }
  1368. } else if ((chargingData_1->SystemStatus >= S_AUTHORIZING &&
  1369. chargingData_1->SystemStatus <= S_COMPLETE) ||
  1370. (chargingData_1->SystemStatus >= S_CCS_PRECHARGE_ST0 &&
  1371. chargingData_1->SystemStatus <= S_CCS_PRECHARGE_ST1)) {
  1372. led_color.Connect_1_Green = COLOR_MIN_LV;
  1373. led_color.Connect_1_Blue = _colorBuf;
  1374. led_color.Connect_1_Red = COLOR_MIN_LV;
  1375. }
  1376. // --------------------------------------------------------------------------
  1377. if (chargingData_2->SystemStatus == S_BOOTING ||
  1378. chargingData_2->SystemStatus == S_IDLE ||
  1379. chargingData_2->SystemStatus == S_RESERVATION ||
  1380. chargingData_2->SystemStatus == S_MAINTAIN) {
  1381. if (chargingData_2->IsAvailable == NO) {
  1382. led_color.Connect_2_Green = COLOR_MIN_LV;
  1383. led_color.Connect_2_Blue = COLOR_MIN_LV;
  1384. led_color.Connect_2_Red = _colorBuf;
  1385. } else {
  1386. #if defined DD360Audi
  1387. led_color.Connect_2_Green = _colorBuf;
  1388. led_color.Connect_2_Blue = _colorBuf;
  1389. led_color.Connect_2_Red = _colorBuf;
  1390. #else
  1391. led_color.Connect_2_Green = _colorBuf;
  1392. led_color.Connect_2_Blue = COLOR_MIN_LV;
  1393. led_color.Connect_2_Red = COLOR_MIN_LV;
  1394. #endif
  1395. }
  1396. } else if ((chargingData_2->SystemStatus >= S_AUTHORIZING &&
  1397. chargingData_2->SystemStatus <= S_COMPLETE) ||
  1398. (chargingData_2->SystemStatus >= S_CCS_PRECHARGE_ST0 &&
  1399. chargingData_2->SystemStatus <= S_CCS_PRECHARGE_ST1)) {
  1400. led_color.Connect_2_Green = COLOR_MIN_LV;
  1401. led_color.Connect_2_Blue = _colorBuf;
  1402. led_color.Connect_2_Red = COLOR_MIN_LV;
  1403. }
  1404. }
  1405. }
  1406. if (_checkLedChanged > 0) {
  1407. if (Config_Led_Color(Uart5Fd, ADDR_LED, &led_color) == PASS) {
  1408. _checkLedChanged--;
  1409. cur_led_color.Connect_1_Red = led_color.Connect_1_Red;
  1410. cur_led_color.Connect_1_Green = led_color.Connect_1_Green;
  1411. cur_led_color.Connect_1_Blue = led_color.Connect_1_Blue;
  1412. cur_led_color.Connect_2_Red = led_color.Connect_2_Red;
  1413. cur_led_color.Connect_2_Green = led_color.Connect_2_Green;
  1414. cur_led_color.Connect_2_Blue = led_color.Connect_2_Blue;
  1415. }
  1416. } else if (IsNoneMatchLedColor()) {
  1417. _checkLedChanged = 3;
  1418. }
  1419. }
  1420. static void LEDBoardSelfTest(void)
  1421. {
  1422. // 自檢階段處理,自檢階段如果讀不到版號則代表該系統沒有掛燈板
  1423. if (ShmLedModuleData->SelfTest_Comp == YES) {
  1424. return;
  1425. }
  1426. #if defined DD360 ||defined DD360Audi
  1427. GetFwAndHwVersion_Led();
  1428. sleep(1);
  1429. gettimeofday(&_led_priority_time, NULL);
  1430. return;
  1431. #endif //defined DD360 || defined DD360Audi
  1432. // 自檢階段
  1433. if (pSysInfo->SelfTestSeq <= _STEST_PSU_CAP) {
  1434. GetFwAndHwVersion_Led();
  1435. sleep(1);
  1436. gettimeofday(&_led_priority_time, NULL);
  1437. } else {
  1438. // 自檢階段沒有問到版號
  1439. if (pAlarmCode->AlarmEvents.bits.LedboardStestFail == NO) {
  1440. pAlarmCode->AlarmEvents.bits.LedboardStestFail = YES;
  1441. }
  1442. }
  1443. }
  1444. static void LEDBoardProcess(void)
  1445. {
  1446. //struct ChargingInfoData *pDcChargingInfo0 = NULL;
  1447. //struct ChargingInfoData *pDcChargingInfo1 = NULL;
  1448. if (ShmLedModuleData->SelfTest_Comp == NO) {
  1449. return;
  1450. }
  1451. if (GetTimeoutValue(_led_priority_time) / 1000 >= 1000) {
  1452. //if (pSysConfig->TotalConnectorCount == 1) {
  1453. // pDcChargingInfo0 = (struct ChargeingInfoData *)GetDcChargingInfoData(0);
  1454. // SetLedColor(pDcChargingInfo0, pDcChargingInfo0);
  1455. //} else if (pSysConfig->TotalConnectorCount == 2) {
  1456. // pDcChargingInfo0 = (struct ChargeingInfoData *)GetDcChargingInfoData(0);
  1457. // pDcChargingInfo1 = (struct ChargeingInfoData *)GetDcChargingInfoData(1);
  1458. // SetLedColor(pDcChargingInfo0, pDcChargingInfo1);
  1459. //}
  1460. SetLedColor();
  1461. gettimeofday(&_led_priority_time, NULL);
  1462. }
  1463. }
  1464. void RelayBoardTask(int uartFD)
  1465. {
  1466. pid_t pid = fork();
  1467. if (pid == 0) {
  1468. bool isCharging = false;
  1469. bool isStopChargingCount = false;
  1470. uint8_t i = 0;
  1471. int isContinue = 1;
  1472. struct ChargingInfoData *pDcChargingInfo = NULL;
  1473. //share memory mapping
  1474. pSysConfig = (struct SysConfigData *)GetShmSysConfigData();
  1475. pSysInfo = (struct SysInfoData *)GetShmSysInfoData();
  1476. pAlarmCode = (struct AlarmCodeData *)GetShmAlarmCodeData();
  1477. ShmRelayModuleData = (struct RelayModuleData *)GetShmRelayModuleData();
  1478. ShmPsuData = (struct PsuData *)GetShmPsuData();
  1479. ShmDcCommonData = (DcCommonInfo *)GetShmDcCommonData();
  1480. ShmPrimaryMcuData = (struct PrimaryMcuData *)GetShmPrimaryMcuData();
  1481. pSysWarning = (struct WARNING_CODE_INFO *)GetShmSysWarningInfo();
  1482. ShmFanModuleData = (struct FanModuleData *)GetShmFanModuleData();
  1483. ShmLedModuleData = (struct LedModuleData *)GetShmLedModuleData();
  1484. Uart5Fd = uartFD;
  1485. //relay init
  1486. outputRelayInit(uartFD);
  1487. while (isContinue) {
  1488. if(IsRelayProcessNeedPause() == true)
  1489. {
  1490. sleep(1);
  1491. continue;
  1492. }
  1493. // 程序開始之前~ 必須先確定 FW 版本與硬體版本,確認後!!~ 該模組才算是真正的 Initial Comp.
  1494. if (ShmRelayModuleData->SelfTest_Comp == NO) {
  1495. GetFwAndHwVersion_Relay();
  1496. SetModelName_Relay(); //DS60-120 add
  1497. SetRtcData_Relay();
  1498. sleep(1);
  1499. }
  1500. #if !defined NO_FAN_BOARD && !defined DD360ComBox
  1501. fanBoardSelfTest();
  1502. #endif //NO_FAN_BOARD
  1503. #if !defined DD360ComBox
  1504. LEDBoardSelfTest();
  1505. #endif //defined DD360ComBox
  1506. if (ShmRelayModuleData->SelfTest_Comp == YES)
  1507. {
  1508. // ==============優先權最高 10 ms ==============
  1509. // 輸出電壓
  1510. GetPersentOutputVol();
  1511. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  1512. // 三相輸入電壓
  1513. GetPresentInputVol();
  1514. #endif //!defined DD360 && !defined DD360Audi
  1515. // 讀取當前 AC relay 狀態
  1516. regRelay.relay_event.bits.AC_Contactor = pSysInfo->AcContactorStatus;
  1517. GetRelayOutputStatus();
  1518. // Cable check (Get)
  1519. GetGfdAdc();
  1520. for (i = 0; i < pSysConfig->TotalConnectorCount; i++) {
  1521. pDcChargingInfo = (struct ChargingInfoData *)GetDcChargingInfoData(i);
  1522. // Cable check (Set)
  1523. CableCheckDetected(i);
  1524. // check k1 k2 relay 狀態
  1525. CheckK1K2RelayOutput(i);
  1526. // 依據當前各槍的狀態選擇 搭上/放開 Relay
  1527. SetK1K2RelayStatus(i);
  1528. #if !defined DD360 && !defined DD360Audi && !defined DD360ComBox
  1529. if (pSysConfig->PhaseLossPolicy == YES) {
  1530. CheckPhaseLossStatus(i);
  1531. }
  1532. CheckAcInputOvpStatus(i);
  1533. #endif //!defined DD360 && !defined DD360Audi
  1534. if (pDcChargingInfo->SystemStatus == S_IDLE ||
  1535. pDcChargingInfo->SystemStatus == S_RESERVATION ||
  1536. pDcChargingInfo->SystemStatus == S_MAINTAIN) {
  1537. //pDcChargingInfo->RelayWeldingCheck = NO;
  1538. //_isRelayWelding[i] = NO;
  1539. _isOvpChkTimeFlag[i] = NO;
  1540. //ResetDetAlarmStatus(i); //DS60-120 add
  1541. }
  1542. if (pDcChargingInfo->SystemStatus == S_BOOTING ||
  1543. (pDcChargingInfo->SystemStatus >= S_REASSIGN_CHECK &&
  1544. pDcChargingInfo->SystemStatus <= S_COMPLETE) ||
  1545. (pDcChargingInfo->SystemStatus >= S_CCS_PRECHARGE_ST0 &&
  1546. pDcChargingInfo->SystemStatus <= S_CCS_PRECHARGE_ST1) ||
  1547. pSysInfo->WaitForPlugit == YES ||
  1548. (pSysInfo->PageIndex >= _LCM_AUTHORIZING &&
  1549. pSysInfo->PageIndex <= _LCM_WAIT_FOR_PLUG)
  1550. ) {
  1551. pDcChargingInfo->IsReadyToCharging = YES;
  1552. isCharging = true;
  1553. // 限定只有在槍類別為 GBT 的時候才做 relay welding 的判斷
  1554. //if (pDcChargingInfo->Type == _Type_GB) {
  1555. // if (pDcChargingInfo->SystemStatus >= S_PREPARING_FOR_EVSE &&
  1556. // pDcChargingInfo->RelayWeldingCheck == NO) {
  1557. // CheckRelayWeldingStatus(i);
  1558. // }
  1559. //} else {
  1560. //pDcChargingInfo->RelayWeldingCheck = YES;
  1561. //}
  1562. if (pDcChargingInfo->SystemStatus == S_CHARGING) {
  1563. CheckOutputPowerOverCarReq(i);
  1564. //CheckOutputVolNoneMatchFire(i);
  1565. }
  1566. /*else {
  1567. _isOutputNoneMatch[i] = NO;
  1568. }*/
  1569. } else {
  1570. pDcChargingInfo->IsReadyToCharging = NO;
  1571. }
  1572. }
  1573. // 橋接 relay
  1574. SetParalleRelayStatus();
  1575. // 搭上 AC Contactor
  1576. //if (isCharging) {
  1577. // outputRelay.relay_event.bits.AC_Contactor = YES;
  1578. //} else {
  1579. // outputRelay.relay_event.bits.AC_Contactor = NO;
  1580. //}
  1581. if (isCharging ||
  1582. (ShmPsuData->Work_Step >= _TEST_MODE &&
  1583. ShmPsuData->Work_Step <= _TEST_MODE)) {
  1584. isStopChargingCount = false;
  1585. outputRelay.relay_event.bits.AC_Contactor = YES;
  1586. } else {
  1587. if (!isStopChargingCount) {
  1588. gettimeofday(&_close_ac_contactor, NULL);
  1589. isStopChargingCount = true;
  1590. } else {
  1591. if ((outputRelay.relay_event.bits.AC_Contactor == YES &&
  1592. GetTimeoutValue(_close_ac_contactor) / 1000 >= (TEN_MINUTES * 1000))) {
  1593. outputRelay.relay_event.bits.AC_Contactor = NO;
  1594. }
  1595. }
  1596. }
  1597. if (ShmPrimaryMcuData->InputDet.bits.EmergencyButton == ABNORMAL) {
  1598. outputRelay.relay_event.bits.AC_Contactor = NO;
  1599. }
  1600. if (pAlarmCode->AlarmEvents.bits.PsuFailureAlarm == ABNORMAL) {
  1601. RunForceStopProcess();
  1602. outputRelay.relay_event.bits.AC_Contactor = NO;
  1603. }
  1604. if (ShmPsuData->Work_Step >= _TEST_MODE && ShmPsuData->Work_Step <= _TEST_MODE) {
  1605. outputRelay.relay_event.bits.Gun1_N = outputRelay.relay_event.bits.Gun1_P = YES;
  1606. }
  1607. // 搭上/鬆開 Relay
  1608. if (IsNoneMatchRelayStatus()) {
  1609. if (Config_Relay_Output(Uart5Fd, ADDR_RELAY, &outputRelay)) {
  1610. //regRelay.relay_event.bits.AC_Contactor = pSysInfo->AcContactorStatus;
  1611. //regRelay.relay_event.bits.CCS_Precharge = outputRelay.relay_event.bits.CCS_Precharge;
  1612. //regRelay.relay_event.bits.Gun1_P = outputRelay.relay_event.bits.Gun1_P;
  1613. //regRelay.relay_event.bits.Gun1_N = outputRelay.relay_event.bits.Gun1_N;
  1614. //regRelay.relay_event.bits.Gun2_P = outputRelay.relay_event.bits.Gun2_P;
  1615. //regRelay.relay_event.bits.Gun2_N = outputRelay.relay_event.bits.Gun2_N;
  1616. //regRelay.relay_event.bits.Gun1_Parallel_P = outputRelay.relay_event.bits.Gun1_Parallel_P;
  1617. //regRelay.relay_event.bits.Gun1_Parallel_N = outputRelay.relay_event.bits.Gun1_Parallel_N;
  1618. //MatchRelayStatus();
  1619. //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",
  1620. // regRelay.relay_event.bits.AC_Contactor,
  1621. // regRelay.relay_event.bits.Gun1_P,
  1622. // regRelay.relay_event.bits.Gun1_N,
  1623. // regRelay.relay_event.bits.Gun2_P,
  1624. // regRelay.relay_event.bits.Gun2_N,
  1625. // regRelay.relay_event.bits.CCS_Precharge,
  1626. // regRelay.relay_event.bits.Gun1_Parallel_P,
  1627. // regRelay.relay_event.bits.Gun1_Parallel_N);
  1628. }
  1629. }
  1630. }
  1631. #if !defined NO_FAN_BOARD && !defined DD360ComBox
  1632. fanBoardPorcess();
  1633. #endif //NO_FAN_BOARD
  1634. #if !defined DD360ComBox
  1635. LEDBoardProcess();
  1636. #endif //defined DD360ComBox
  1637. usleep(10000);
  1638. }
  1639. }
  1640. }