RelayBoard.c 49 KB

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