Module_PsuComm.c 53 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260
  1. #include "Module_PsuComm.h"
  2. #define ARRAY_SIZE(A) (sizeof(A) / sizeof(A[0]))
  3. #define PASS 1
  4. #define FAIL -1
  5. #define YES 1
  6. #define NO 0
  7. #define DERATING 30
  8. #define ELEMENT_NOT_FIND 255
  9. #define CHK_VOL_RANGE 20
  10. #define CHK_CUR_RANGE 10
  11. #define DERATING_RANGE 100
  12. #define ZERO_CURRENT 0
  13. #define ZERO_VOLTAGE 10
  14. #define STOP_CURRENT 30
  15. #define PSU_MIN_CUR 100
  16. struct SysConfigAndInfo *ShmSysConfigAndInfo;
  17. struct StatusCodeData *ShmStatusCodeData;
  18. struct PsuData *ShmPsuData;
  19. bool libInitialize = false;
  20. byte getAvailableCapOffset = 5;
  21. byte deratingKeepCount = 0;
  22. float carReqVol = 0;
  23. float carReqCur = 0;
  24. float evseOutVol = 0;
  25. float evseOutCur = 0;
  26. void PRINTF_FUNC(char *string, ...);
  27. int StoreLogMsg(const char *fmt, ...);
  28. #define DEBUG_INFO(format, args...) StoreLogMsg("[%s:%d][%s][Info] "format, __FILE__, __LINE__, __FUNCTION__, ##args)
  29. #define DEBUG_WARN(format, args...) StoreLogMsg("[%s:%d][%s][Warn] "format, __FILE__, __LINE__, __FUNCTION__, ##args)
  30. #define DEBUG_ERROR(format, args...) StoreLogMsg("[%s:%d][%s][Error] "format, __FILE__, __LINE__, __FUNCTION__, ##args)
  31. unsigned long GetTimeoutValue(struct timeval _sour_time);
  32. unsigned long GetTimeoutValue(struct timeval _sour_time)
  33. {
  34. struct timeval _end_time;
  35. gettimeofday(&_end_time, NULL);
  36. return 1000000 * (_end_time.tv_sec - _sour_time.tv_sec) + _end_time.tv_usec - _sour_time.tv_usec;
  37. }
  38. int StoreLogMsg(const char *fmt, ...)
  39. {
  40. char Buf[4096+256];
  41. char buffer[4096];
  42. time_t CurrentTime;
  43. struct tm *tm;
  44. va_list args;
  45. va_start(args, fmt);
  46. int rc = vsnprintf(buffer, sizeof(buffer), fmt, args);
  47. va_end(args);
  48. memset(Buf,0,sizeof(Buf));
  49. CurrentTime = time(NULL);
  50. tm=localtime(&CurrentTime);
  51. sprintf(Buf,"echo \"%04d-%02d-%02d %02d:%02d:%02d - %s\" >> /Storage/SystemLog/[%04d.%02d]SystemLog",
  52. tm->tm_year+1900,tm->tm_mon+1,tm->tm_mday,tm->tm_hour,tm->tm_min,tm->tm_sec,
  53. buffer,
  54. tm->tm_year+1900,tm->tm_mon+1);
  55. system(Buf);
  56. return rc;
  57. }
  58. void PRINTF_FUNC(char *string, ...)
  59. {
  60. va_list args;
  61. char buffer[4096];
  62. va_start(args, string);
  63. vsnprintf(buffer, sizeof(buffer), string, args);
  64. va_end(args);
  65. if (ShmSysConfigAndInfo->SysConfig.SwitchDebugFlag == YES)
  66. printf("%s \n", buffer);
  67. else
  68. DEBUG_INFO("%s \n", buffer);
  69. }
  70. //=================================
  71. // Common routine
  72. //=================================
  73. size_t FindIndex(const int a[], size_t size, int value, byte group)
  74. {
  75. size_t index = 0;
  76. while ( index < size && a[index] != value ) ++index;
  77. return (index == size ? ELEMENT_NOT_FIND : group);
  78. }
  79. byte FindTargetGroup(byte address)
  80. {
  81. byte _group = ELEMENT_NOT_FIND;
  82. if (ShmPsuData->GroupCount == 1)
  83. _group = 0;
  84. else
  85. {
  86. _group = FindIndex(connector_1, ShmPsuData->PsuGroup[0].GroupPresentPsuQuantity, address, 0);
  87. if (_group == ELEMENT_NOT_FIND)
  88. _group = FindIndex(connector_2, ShmPsuData->PsuGroup[1].GroupPresentPsuQuantity, address, 1);
  89. }
  90. return _group;
  91. }
  92. bool IsOverModuleCount(byte count)
  93. {
  94. bool result = false;
  95. if (count >= ShmPsuData->SystemPresentPsuQuantity)
  96. result = true;
  97. return result;
  98. }
  99. //=================================
  100. // Save data to share memory Function
  101. //=================================
  102. bool FindChargingInfoData(byte target, struct ChargingInfoData **chargingData)
  103. {
  104. for (byte index = 0; index < CHAdeMO_QUANTITY; index++)
  105. {
  106. if (ShmSysConfigAndInfo->SysInfo.ChademoChargingData[index].Index == target)
  107. {
  108. chargingData[target] = &ShmSysConfigAndInfo->SysInfo.ChademoChargingData[index];
  109. return true;
  110. }
  111. }
  112. for (byte index = 0; index < CCS_QUANTITY; index++)
  113. {
  114. if (ShmSysConfigAndInfo->SysInfo.CcsChargingData[index].Index == target)
  115. {
  116. chargingData[target] = &ShmSysConfigAndInfo->SysInfo.CcsChargingData[index];
  117. return true;
  118. }
  119. }
  120. for (byte index = 0; index < GB_QUANTITY; index++)
  121. {
  122. if (ShmSysConfigAndInfo->SysInfo.GbChargingData[index].Index == target)
  123. {
  124. chargingData[target] = &ShmSysConfigAndInfo->SysInfo.GbChargingData[index];
  125. return true;
  126. }
  127. }
  128. return false;
  129. }
  130. //=================================
  131. // Alarm code mapping to share memory Function
  132. //=================================
  133. // 檢查 Byte 中某個 Bit 的值
  134. // _byte : 欲改變的 byte
  135. // _bit : 該 byte 的第幾個 bit
  136. unsigned char mask_table[] = { 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80 };
  137. unsigned char DetectBitValue(unsigned char _byte, unsigned char _bit)
  138. {
  139. return ( _byte & mask_table[_bit] ) != 0x00;
  140. }
  141. void AbnormalStopAnalysis(byte gun_index, int errCode)
  142. {
  143. for (char i = 0; i < 3; i++)
  144. {
  145. unsigned char byteIndex = (errCode >> (8 * i)) & 0xff;
  146. for (char bitIndex = 0; bitIndex < 8; bitIndex++)
  147. {
  148. if(DetectBitValue(byteIndex , bitIndex) == 1)
  149. {
  150. switch(byteIndex)
  151. {
  152. case 0:
  153. {
  154. if (bitIndex == 0)
  155. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuOutputShortCircuit = YES;
  156. else if (bitIndex == 5)
  157. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuDcSideShutDown = YES;
  158. }
  159. break;
  160. case 1:
  161. {
  162. if (bitIndex == 1)
  163. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuFailureAlarm = YES;
  164. else if (bitIndex == 2)
  165. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuProtectionAlarm = YES;
  166. else if (bitIndex == 3)
  167. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuFanFailureAlarm = YES;
  168. else if (bitIndex == 4)
  169. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuCriticalPointOTP = YES;
  170. else if (bitIndex == 5)
  171. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuDcSideShutDown = YES;
  172. }
  173. break;
  174. case 2:
  175. {
  176. if (bitIndex == 0)
  177. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuPowerLimitedState = YES;
  178. else if (bitIndex == 1)
  179. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuDuplicateID = YES;
  180. else if (bitIndex == 2)
  181. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuThreePhaseOnputImbalance = YES;
  182. else if (bitIndex == 3)
  183. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuThreePhaseInputInadequate = YES;
  184. else if (bitIndex == 4)
  185. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuThreePhaseInputInadequate = YES;
  186. else if (bitIndex == 5)
  187. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuInputUVP = YES;
  188. else if (bitIndex == 6)
  189. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuInputOVP = YES;
  190. }
  191. break;
  192. }
  193. }
  194. // else
  195. // {
  196. // switch (byteIndex) {
  197. // case 0: {
  198. // if (bitIndex == 0)
  199. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuOutputShortCircuit = NO;
  200. // else if (bitIndex == 5)
  201. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuDcSideShutDown = NO;
  202. // }
  203. // break;
  204. // case 1: {
  205. // if (bitIndex == 1)
  206. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuFailureAlarm = NO;
  207. // else if (bitIndex == 2)
  208. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuProtectionAlarm = NO;
  209. // else if (bitIndex == 3)
  210. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuFanFailureAlarm = NO;
  211. // else if (bitIndex == 4)
  212. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuCriticalPointOTP = NO;
  213. // else if (bitIndex == 5)
  214. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuDcSideShutDown = NO;
  215. // }
  216. // break;
  217. // case 2: {
  218. // if (bitIndex == 1)
  219. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuDuplicateID = NO;
  220. // if (bitIndex == 2)
  221. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuThreePhaseOnputImbalance = NO;
  222. // else if (bitIndex == 3)
  223. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuThreePhaseInputInadequate = NO;
  224. // else if (bitIndex == 4)
  225. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuThreePhaseInputInadequate = NO;
  226. // else if (bitIndex == 5)
  227. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuInputUVP = NO;
  228. // else if (bitIndex == 6)
  229. // ShmStatusCodeData->AlarmCode.AlarmEvents.bits.PsuInputOVP = NO;
  230. // }
  231. // break;
  232. // }
  233. // }
  234. }
  235. }
  236. }
  237. //=================================
  238. // Callback Function
  239. //=================================
  240. void GetStatusCallback(byte group, byte address, byte temp, int alarm)
  241. {
  242. if (IsOverModuleCount(address))
  243. return;
  244. byte group1 = FindTargetGroup(address);
  245. if (group1 == 1)
  246. address -= ShmPsuData->PsuGroup[group1 - 1].GroupPresentPsuQuantity;
  247. ShmPsuData->PsuGroup[group1].PsuModule[address].CriticalTemp1 = temp;
  248. ShmPsuData->PsuGroup[group1].PsuModule[address].CriticalTemp2 = temp;
  249. ShmPsuData->PsuGroup[group1].PsuModule[address].CriticalTemp3 = temp;
  250. ShmPsuData->PsuGroup[group1].PsuModule[address].ExletTemp = temp;
  251. ShmPsuData->PsuGroup[group1].PsuModule[address].AlarmCode = alarm;
  252. AbnormalStopAnalysis(group1, alarm);
  253. //printf("alarm = %d \n", alarm);
  254. }
  255. void GetModuleCountCallback(byte group, byte count)
  256. {
  257. if (group == SYSTEM_CMD)
  258. ShmPsuData->SystemPresentPsuQuantity = count;
  259. else
  260. {
  261. ShmPsuData->PsuGroup[group].GroupPresentPsuQuantity = count;
  262. }
  263. }
  264. void GetMaxPowerAndCur(unsigned char mode, int ratingCur, int *pow, int *cur)
  265. {
  266. unsigned short maxCurrent = ShmSysConfigAndInfo->SysConfig.MaxChargingCurrent * 10;
  267. unsigned short maxPower = ShmSysConfigAndInfo->SysConfig.MaxChargingPower * 10;
  268. if (mode == _MAIN_CHARGING_MODE_AVER)
  269. {
  270. maxCurrent /= 2;
  271. maxPower /= 2;
  272. }
  273. if (maxPower != 0 && maxPower <= *pow)
  274. *pow = maxPower;
  275. if (maxCurrent != 0 && maxCurrent <= *cur)
  276. *cur = maxCurrent;
  277. if (ratingCur != 0 && ratingCur <= *cur)
  278. *cur = ratingCur;
  279. }
  280. void GetAvailableCapCallback(byte address, short maxVol, short minVol, short maxCur, short totalPow)
  281. {
  282. int _groupPower = 0, _groupCurrent = 0;
  283. byte group = FindTargetGroup(address);
  284. if (group == 1)
  285. address -= ShmPsuData->PsuGroup[group - 1].GroupPresentPsuQuantity;
  286. if (chargingInfo[group]->DeratingChargingCurrent == 0)
  287. ShmPsuData->PsuGroup[group].PsuModule[address].AvailableCurrent = PSU_MIN_CUR;
  288. else
  289. ShmPsuData->PsuGroup[group].PsuModule[address].AvailableCurrent = maxCur;
  290. ShmPsuData->PsuGroup[group].PsuModule[address].AvailablePower = totalPow;
  291. for (byte index = 0; index < ShmPsuData->PsuGroup[group].GroupPresentPsuQuantity; index++)
  292. {
  293. _groupCurrent += ShmPsuData->PsuGroup[group].PsuModule[index].AvailableCurrent;
  294. _groupPower += ShmPsuData->PsuGroup[group].PsuModule[index].AvailablePower;
  295. }
  296. // 各群得到最大輸出能力 (電流、Power)
  297. ShmPsuData->PsuGroup[group].GroupAvailableCurrent = _groupCurrent;
  298. ShmPsuData->PsuGroup[group].GroupAvailablePower = _groupPower;
  299. chargingInfo[group]->MaximumChargingVoltage = maxVol;
  300. int _power = 0, _current = 0, _ratingcurrent = 0;
  301. for (byte index = 0; index < ShmPsuData->GroupCount; index++)
  302. {
  303. _power += ShmPsuData->PsuGroup[index].GroupAvailablePower;
  304. _current += ShmPsuData->PsuGroup[index].GroupAvailableCurrent;
  305. _ratingcurrent += chargingInfo[index]->DeratingChargingCurrent;
  306. }
  307. ShmPsuData->SystemAvailableCurrent = _current;
  308. ShmPsuData->SystemAvailablePower = _power;
  309. if (ShmSysConfigAndInfo->SysInfo.MainChargingMode == _MAIN_CHARGING_MODE_AVER ||
  310. (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag >= _REASSIGNED_GET_NEW_CAP &&
  311. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag <= _REASSIGNED_RELAY_M_TO_A))
  312. {
  313. int halfPow = ShmPsuData->PsuGroup[group].GroupAvailablePower;
  314. int halfCur = ShmPsuData->PsuGroup[group].GroupAvailableCurrent;
  315. int ratingCur = chargingInfo[group]->DeratingChargingCurrent;
  316. GetMaxPowerAndCur(_MAIN_CHARGING_MODE_AVER, ratingCur, &halfPow, &halfCur);
  317. // if ((ShmSysConfigAndInfo->SysInfo.ReAssignedFlag >= _REASSIGNED_GET_NEW_CAP &&
  318. // ShmSysConfigAndInfo->SysInfo.ReAssignedFlag <= _REASSIGNED_RELAY_M_TO_A))
  319. // {
  320. // chargingInfo[group]->AvailableChargingCurrent = DERATING_RANGE;
  321. // chargingInfo[group]->AvailableChargingPower = ShmPsuData->PsuGroup[group].GroupAvailablePower;
  322. // }
  323. // else
  324. {
  325. // 以下狀況 -> 槍資訊中的最大輸出能力,為該群的輸出能力
  326. // 1. 如不是最大充
  327. // 2. 智能切換成均充過程
  328. chargingInfo[group]->AvailableChargingCurrent = halfCur;
  329. chargingInfo[group]->AvailableChargingPower = halfPow;
  330. }
  331. }
  332. else if (ShmSysConfigAndInfo->SysInfo.MainChargingMode == _MAIN_CHARGING_MODE_MAX)
  333. {
  334. GetMaxPowerAndCur(_MAIN_CHARGING_MODE_MAX, _ratingcurrent, &_power, &_current);
  335. if (ShmSysConfigAndInfo->SysInfo.IsAlternatvieConf == YES)
  336. {
  337. for (byte count = 0; count < ShmSysConfigAndInfo->SysConfig.TotalConnectorCount; count++)
  338. {
  339. chargingInfo[count]->MaximumChargingVoltage = maxVol;
  340. chargingInfo[count]->AvailableChargingCurrent = _current;
  341. chargingInfo[count]->AvailableChargingPower = _power;
  342. }
  343. }
  344. else
  345. {
  346. // 如果是最大充,該槍資訊中的輸出能力為各群輸出能力的和
  347. chargingInfo[group]->AvailableChargingCurrent = _current;
  348. chargingInfo[group]->AvailableChargingPower = _power;
  349. }
  350. }
  351. }
  352. void GetFwCallback(byte address, short dcSwVer, short pfcSwVer, short hwVer)
  353. {
  354. if (IsOverModuleCount(address))
  355. return;
  356. byte group = FindTargetGroup(address);
  357. if (group == 1)
  358. address -= ShmPsuData->PsuGroup[group - 1].GroupPresentPsuQuantity;
  359. sprintf((char *)ShmPsuData->PsuGroup[group].PsuModule[address].FwVersion, "DC %d.%02d", (dcSwVer & 0xFF00) >> 8, dcSwVer & 0xFF);
  360. //DEBUG_INFO("fw Ver. = %s \n", ShmPsuData->PsuGroup[group].PsuModule[address].FwVersion);
  361. }
  362. void GetInputVoltageCallback(byte address, unsigned short vol1, unsigned short vol2, unsigned short vol3)
  363. {
  364. if (IsOverModuleCount(address))
  365. return;
  366. byte group = FindTargetGroup(address);
  367. if (group == 1)
  368. address -= ShmPsuData->PsuGroup[group - 1].GroupPresentPsuQuantity;
  369. ShmPsuData->PsuGroup[group].PsuModule[address].InputVoltageL1 = vol1;
  370. ShmPsuData->PsuGroup[group].PsuModule[address].InputVoltageL2 = vol2;
  371. ShmPsuData->PsuGroup[group].PsuModule[address].InputVoltageL3 = vol3;
  372. }
  373. void GetPresentOutputCallback(byte group, unsigned short outVol, unsigned short outCur)
  374. {
  375. unsigned short outputVol = outVol;
  376. unsigned short outputCur = outCur;
  377. // PSU Group - 電壓
  378. ShmPsuData->PsuGroup[group].GroupPresentOutputVoltage = outputVol;
  379. // PSU Group - 電流
  380. ShmPsuData->PsuGroup[group].GroupPresentOutputCurrent = outputCur;
  381. if (ShmSysConfigAndInfo->SysInfo.MainChargingMode == _MAIN_CHARGING_MODE_MAX)
  382. {
  383. outputVol = 0;
  384. outputCur = 0;
  385. for (byte index = 0; index < ShmPsuData->GroupCount; index++)
  386. {
  387. if (ShmPsuData->PsuGroup[index].GroupPresentOutputVoltage > outputVol)
  388. outputVol = ShmPsuData->PsuGroup[index].GroupPresentOutputVoltage;
  389. outputCur += ShmPsuData->PsuGroup[index].GroupPresentOutputCurrent;
  390. }
  391. // 黑白機
  392. if (ShmSysConfigAndInfo->SysInfo.IsAlternatvieConf == YES)
  393. {
  394. for (byte count = 0; count < ShmSysConfigAndInfo->SysConfig.TotalConnectorCount; count++)
  395. {
  396. float _vol_buf = outputVol;
  397. float _cur_buf = outputCur;
  398. // EVSE - 電壓
  399. _vol_buf /= 10;
  400. chargingInfo[count]->PresentChargingVoltage = _vol_buf;
  401. // EVSE - 電流
  402. _cur_buf /= 10;
  403. chargingInfo[count]->PresentChargingCurrent = _cur_buf;
  404. }
  405. }
  406. if ((chargingInfo[group]->SystemStatus >= S_PREPARING_FOR_EVSE && chargingInfo[group]->SystemStatus <= S_COMPLETE) ||
  407. (chargingInfo[group]->SystemStatus >= S_CCS_PRECHARGE_ST0 && chargingInfo[group]->SystemStatus <= S_CCS_PRECHARGE_ST1))
  408. {
  409. float _vol_buf = outputVol;
  410. float _cur_buf = outputCur;
  411. // EVSE - 電壓
  412. _vol_buf /= 10;
  413. chargingInfo[group]->PresentChargingVoltage = _vol_buf;
  414. // EVSE - 電流
  415. _cur_buf /= 10;
  416. chargingInfo[group]->PresentChargingCurrent = _cur_buf;
  417. }
  418. }
  419. else
  420. {
  421. float _vol_buf = ShmPsuData->PsuGroup[group].GroupPresentOutputVoltage;
  422. float _cur_buf = ShmPsuData->PsuGroup[group].GroupPresentOutputCurrent;
  423. // EVSE - 電壓
  424. _vol_buf /= 10;
  425. chargingInfo[group]->PresentChargingVoltage = _vol_buf;
  426. // EVSE - 電流
  427. _cur_buf /= 10;
  428. chargingInfo[group]->PresentChargingCurrent = _cur_buf;
  429. }
  430. // PRINTF_FUNC("Gun_%d, PresentChargingCurrent = %f \n", group,
  431. // chargingInfo[group]->PresentChargingCurrent);
  432. }
  433. void GetFanSpeedCallback(byte address, unsigned int fanSpeed)
  434. {
  435. if (IsOverModuleCount(address))
  436. return;
  437. byte group = FindTargetGroup(address);
  438. if (group == 1)
  439. address -= ShmPsuData->PsuGroup[group - 1].GroupPresentPsuQuantity;
  440. ShmPsuData->PsuGroup[group].PsuModule[address].FanSpeed_1 = fanSpeed;
  441. ShmPsuData->PsuGroup[group].PsuModule[address].FanSpeed_2 = fanSpeed;
  442. ShmPsuData->PsuGroup[group].PsuModule[address].FanSpeed_3 = fanSpeed;
  443. ShmPsuData->PsuGroup[group].PsuModule[address].FanSpeed_4 = fanSpeed;
  444. }
  445. void GetIavailableCallback(byte address, unsigned short Iavail, unsigned short Vext)
  446. {
  447. if (IsOverModuleCount(address))
  448. return;
  449. byte group = FindTargetGroup(address);
  450. if (group == 1)
  451. address -= ShmPsuData->PsuGroup[group - 1].GroupPresentPsuQuantity;
  452. //PRINTF_FUNC("group = %d, address_%d, Iavail = %d \n", group, address, Iavail);
  453. // PSU Group - 電壓
  454. ShmPsuData->PsuGroup[group].PsuModule[address].IAvailableCurrent = Iavail;
  455. bool isPass = true;
  456. int totalCur = 0;
  457. if (Iavail == 0)
  458. {
  459. for (byte count = 0; count < 2; count++)
  460. {
  461. chargingInfo[group]->SampleChargingCur[count] = Iavail;
  462. }
  463. }
  464. else
  465. {
  466. // 該群的可輸出電流
  467. for (byte index = 0; index < ShmPsuData->PsuGroup[group].GroupPresentPsuQuantity; index++)
  468. {
  469. totalCur += ShmPsuData->PsuGroup[group].PsuModule[index].IAvailableCurrent;
  470. }
  471. for (byte count = 0; count < 2; count++)
  472. {
  473. if (chargingInfo[group]->SampleChargingCur[count] == 0)
  474. {
  475. chargingInfo[group]->SampleChargingCur[count] = totalCur;
  476. return;
  477. }
  478. else
  479. {
  480. if (chargingInfo[group]->SampleChargingCur[count] != totalCur)
  481. {
  482. chargingInfo[group]->SampleChargingCur[count] = totalCur;
  483. isPass = false;
  484. continue;
  485. }
  486. }
  487. }
  488. }
  489. if (isPass)
  490. {
  491. chargingInfo[group]->DeratingChargingCurrent = totalCur;
  492. }
  493. }
  494. //==========================================
  495. // Init all share memory
  496. //==========================================
  497. int InitShareMemory()
  498. {
  499. int result = PASS;
  500. int MeterSMId;
  501. //creat ShmSysConfigAndInfo
  502. if ((MeterSMId = shmget(ShmSysConfigAndInfoKey, sizeof(struct SysConfigAndInfo), 0777)) < 0)
  503. {
  504. #ifdef SystemLogMessage
  505. DEBUG_ERROR("shmget ShmSysConfigAndInfo NG %d \n");
  506. #endif
  507. result = FAIL;
  508. }
  509. else if ((ShmSysConfigAndInfo = shmat(MeterSMId, NULL, 0)) == (void *) -1)
  510. {
  511. #ifdef SystemLogMessage
  512. DEBUG_ERROR("shmat ShmSysConfigAndInfo NG \n");
  513. #endif
  514. result = FAIL;
  515. }
  516. else
  517. {}
  518. //creat ShmStatusCodeData
  519. if ((MeterSMId = shmget(ShmStatusCodeKey, sizeof(struct StatusCodeData), 0777)) < 0)
  520. {
  521. #ifdef SystemLogMessage
  522. DEBUG_ERROR("shmget ShmStatusCodeData NG \n");
  523. #endif
  524. result = FAIL;
  525. }
  526. else if ((ShmStatusCodeData = shmat(MeterSMId, NULL, 0)) == (void *) -1)
  527. {
  528. #ifdef SystemLogMessage
  529. DEBUG_ERROR("shmat ShmStatusCodeData NG \n");
  530. #endif
  531. result = FAIL;
  532. }
  533. else
  534. {}
  535. //creat ShmPsuData
  536. if ((MeterSMId = shmget(ShmPsuKey, sizeof(struct PsuData), 0777)) < 0)
  537. {
  538. #ifdef SystemLogMessage
  539. DEBUG_ERROR("shmget ShmPsuData NG \n");
  540. #endif
  541. result = FAIL;
  542. }
  543. else if ((ShmPsuData = shmat(MeterSMId, NULL, 0)) == (void *) -1)
  544. {
  545. #ifdef SystemLogMessage
  546. DEBUG_ERROR("shmat ShmPsuData NG \n");
  547. #endif
  548. result = FAIL;
  549. }
  550. memset(ShmPsuData,0,sizeof(struct PsuData));
  551. return result;
  552. }
  553. //================================================
  554. // Main process
  555. //================================================
  556. void InitialPsuData()
  557. {
  558. ShmPsuData->SystemPresentPsuQuantity = 0;
  559. for (byte _groupCount = 0; _groupCount < ARRAY_SIZE(ShmPsuData->PsuGroup); _groupCount++)
  560. {
  561. ShmPsuData->PsuGroup[_groupCount].GroupPresentPsuQuantity = 0;
  562. ShmPsuData->PsuGroup[_groupCount].GroupAvailablePower = 0;
  563. ShmPsuData->PsuGroup[_groupCount].GroupAvailableCurrent = 0;
  564. }
  565. ShmPsuData->Work_Step = _INIT_PSU_STATUS;
  566. }
  567. void Initialization()
  568. {
  569. bool isPass = false;
  570. while(!isPass)
  571. {
  572. isPass = true;
  573. for (byte _index = 0; _index < _gunCount; _index++)
  574. {
  575. if (!FindChargingInfoData(_index, &chargingInfo[0]))
  576. {
  577. DEBUG_ERROR("EvComm (main) : FindChargingInfoData false \n");
  578. isPass = false;
  579. break;
  580. }
  581. }
  582. }
  583. if (ShmSysConfigAndInfo->SysInfo.IsAlternatvieConf == YES)
  584. ShmPsuData->GroupCount = 1;
  585. else
  586. ShmPsuData->GroupCount = _gunCount;
  587. }
  588. void CheckSmartChargingStep(bool isCharging)
  589. {
  590. if (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag == _REASSIGNED_PREPARE_M_TO_A)
  591. {
  592. if (ShmSysConfigAndInfo->SysInfo.MainChargingMode == _MAIN_CHARGING_MODE_MAX)
  593. {
  594. PRINTF_FUNC("=============Smart Charging : _REASSIGNED_GET_NEW_CAP============= Step 2 \n");
  595. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag = _REASSIGNED_GET_NEW_CAP;
  596. }
  597. else
  598. {
  599. PRINTF_FUNC("=============Smart Charging : _REASSIGNED_NONE============= Step 0 \n");
  600. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag = _REASSIGNED_NONE;
  601. }
  602. }
  603. else if (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag == _REASSIGNED_PREPARE_A_TO_M)
  604. {
  605. if (isCharging)
  606. {
  607. if (ShmSysConfigAndInfo->SysInfo.MainChargingMode == _MAIN_CHARGING_MODE_AVER)
  608. {
  609. PRINTF_FUNC("=============Smart Charging : _REASSIGNED_ADJUST_A_TO_M============= Step 12 \n");
  610. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag = _REASSIGNED_ADJUST_A_TO_M;
  611. }
  612. else
  613. {
  614. PRINTF_FUNC("=============Smart Charging : _REASSIGNED_COMP============= Step 15 \n");
  615. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag = _REASSIGNED_COMP;
  616. }
  617. }
  618. }
  619. }
  620. int main(void)
  621. {
  622. if(InitShareMemory() == FAIL)
  623. {
  624. #ifdef SystemLogMessage
  625. DEBUG_ERROR("InitShareMemory NG\n");
  626. #endif
  627. if(ShmStatusCodeData != NULL)
  628. {
  629. ShmStatusCodeData->AlarmCode.AlarmEvents.bits.FailToCreateShareMemory = 1;
  630. }
  631. sleep(5);
  632. return 0;
  633. }
  634. PRINTF_FUNC("InitShareMemory OK\n");
  635. // register callback function
  636. RefreshStatus(&GetStatusCallback);
  637. RefreshModuleCount(&GetModuleCountCallback);
  638. RefreshAvailableCap(&GetAvailableCapCallback);
  639. RefreshFwVersion(&GetFwCallback);
  640. RefreshInputVol(&GetInputVoltageCallback);
  641. RefreshGetOutput(&GetPresentOutputCallback);
  642. RefreshFanInfo(&GetFanSpeedCallback);
  643. RefreshIavailable(&GetIavailableCallback);
  644. sleep(2);
  645. _gunCount = ShmSysConfigAndInfo->SysConfig.TotalConnectorCount;
  646. // initial object
  647. InitialPsuData();
  648. Initialization();
  649. libInitialize = InitialCommunication();
  650. byte isInitialComp = NO;
  651. PRINTF_FUNC("ALTERNATIVE_CONG = %d \n", ShmSysConfigAndInfo->SysInfo.IsAlternatvieConf);
  652. //main loop
  653. while (libInitialize)
  654. {
  655. // 斷電狀態
  656. if (ShmSysConfigAndInfo->SysInfo.AcContactorStatus == NO)
  657. {
  658. //一但 AC Off PSU 斷電全部的 PSU Group ID 會全部清 0
  659. if (!isInitialComp)
  660. {
  661. InitialPsuData();
  662. ShmPsuData->Work_Step = INITIAL_START;
  663. isInitialComp = YES;
  664. }
  665. sleep(1);
  666. continue;
  667. }
  668. else
  669. isInitialComp = NO;
  670. // 自檢失敗
  671. if (ShmPsuData->Work_Step == _NO_WORKING)
  672. {
  673. PRINTF_FUNC("== PSU == self test fail. \n");
  674. sleep(5);
  675. }
  676. switch(ShmPsuData->Work_Step)
  677. {
  678. case INITIAL_START:
  679. {
  680. PRINTF_FUNC("== PSU == INITIAL_START \n");
  681. gettimeofday(&_cmdSubPriority_time, NULL);
  682. sleep(5);
  683. ShmPsuData->Work_Step = GET_PSU_COUNT;
  684. }
  685. break;
  686. case GET_PSU_COUNT:
  687. {
  688. int time = GetTimeoutValue(_cmdSubPriority_time) / 1000;
  689. byte moduleCount = 0;
  690. if (time > 2000)
  691. {
  692. PRINTF_FUNC("== PSU == %d \n", ShmPsuData->GroupCount);
  693. // if (ShmPsuData->GroupCount == 0)
  694. // ShmPsuData->GroupCount = ShmSysConfigAndInfo->SysConfig.TotalConnectorCount;
  695. // 分別取各群模組數量
  696. for (byte index = 0; index < ShmPsuData->GroupCount; index++)
  697. {
  698. // 總和各群模組數量
  699. moduleCount += ShmPsuData->PsuGroup[index].GroupPresentPsuQuantity;
  700. // 取各群模組數量
  701. GetModuleCount(index);
  702. // 取版號
  703. GetModuleVer(index);
  704. }
  705. // 發送取得目前全部模組數量
  706. GetModuleCount(SYSTEM_CMD);
  707. // 判斷系統數量與各群數量一致
  708. if(moduleCount == ShmPsuData->SystemPresentPsuQuantity && moduleCount > 0)
  709. {
  710. PRINTF_FUNC("Psu Count = %d \n", moduleCount);
  711. if (ShmSysConfigAndInfo->SysInfo.BootingStatus == BOOTTING)
  712. {
  713. // 電樁在 Booting 的狀態 - 自檢
  714. PRINTF_FUNC("== PSU == GET_SYS_CAP \n");
  715. ShmPsuData->Work_Step = GET_SYS_CAP;
  716. }
  717. else
  718. {
  719. PRINTF_FUNC("== PSU == _WORK_CHARGING \n");
  720. ShmPsuData->Work_Step = _WORK_CHARGING;
  721. //sdlu test
  722. gettimeofday(&_test_time, NULL);
  723. }
  724. }
  725. _getCapDelayCount = 3;
  726. gettimeofday(&_cmdSubPriority_time, NULL);
  727. }
  728. }
  729. break;
  730. case GET_SYS_CAP:
  731. {
  732. int time = GetTimeoutValue(_cmdSubPriority_time) / 1000;
  733. if (time > 1000)
  734. {
  735. for (byte index = 0; index < ShmPsuData->GroupCount; index++)
  736. {
  737. // Pooling Status
  738. GetStatus(index);
  739. // 取系統總輸出能力
  740. GetModuleCap(index);
  741. }
  742. _getCapDelayCount--;
  743. gettimeofday(&_cmdSubPriority_time, NULL);
  744. }
  745. // 判斷系統輸出額定功率與電流
  746. if (ShmPsuData->SystemAvailablePower > 0 && ShmPsuData->SystemAvailableCurrent > 0 &&
  747. _getCapDelayCount <= 0)
  748. {
  749. PRINTF_FUNC("SystemAvailableCurrent = %d, SystemAvailablePower = %d \n",
  750. ShmPsuData->SystemAvailableCurrent, ShmPsuData->SystemAvailablePower);
  751. PRINTF_FUNC("== PSU == BOOTING_COMPLETE \n");
  752. ShmPsuData->Work_Step = BOOTING_COMPLETE;
  753. }
  754. }
  755. break;
  756. case BOOTING_COMPLETE:
  757. {
  758. sleep(1);
  759. }
  760. break;
  761. case _WORK_CHARGING:
  762. {
  763. int time = GetTimeoutValue(_cmdSubPriority_time) / 1000;
  764. // sdlu - test
  765. int testtime = GetTimeoutValue(_test_time) / 1000;
  766. bool isCharging = false;
  767. // 低 Priority 的指令
  768. if (time > 1500)
  769. {
  770. for (byte index = 0; index < ShmPsuData->GroupCount; index++)
  771. {
  772. // Pooling Status
  773. GetStatus(index);
  774. // 取系統總輸出能力
  775. GetModuleCap(index);
  776. // 取得模塊輸入電壓
  777. GetModuleInput(index);
  778. // 取得模塊輸出額定電流能力
  779. GetModuleIavailable(index);
  780. if (chargingInfo[index]->SystemStatus == S_CHARGING)
  781. isCharging = true;
  782. }
  783. for (byte gunIndex = 0; gunIndex < _gunCount; gunIndex++)
  784. {
  785. if (chargingInfo[gunIndex]->FireChargingVoltage > 0 &&
  786. evseOutVol != (chargingInfo[gunIndex]->FireChargingVoltage / 10))
  787. {
  788. evseOutVol = (chargingInfo[gunIndex]->FireChargingVoltage / 10);
  789. PRINTF_FUNC("groupIndex = %d, evse output vol = %f \n", gunIndex,
  790. chargingInfo[gunIndex]->FireChargingVoltage);
  791. }
  792. if (chargingInfo[gunIndex]->PresentChargingCurrent > 0 &&
  793. evseOutCur != chargingInfo[gunIndex]->PresentChargingCurrent)
  794. {
  795. evseOutCur = chargingInfo[gunIndex]->PresentChargingCurrent;
  796. PRINTF_FUNC("groupIndex = %d, evse output cur = %f \n", gunIndex,
  797. chargingInfo[gunIndex]->PresentChargingCurrent);
  798. }
  799. }
  800. gettimeofday(&_cmdSubPriority_time, NULL);
  801. }
  802. CheckSmartChargingStep(isCharging);
  803. for (byte groupIndex = 0; groupIndex < _gunCount; groupIndex++)
  804. {
  805. GetModuleOutput(groupIndex);
  806. // 針對各槍當前狀態,傳送需要回傳的資料指令
  807. if (((chargingInfo[groupIndex]->SystemStatus >= S_PREPARING_FOR_EVSE && chargingInfo[groupIndex]->SystemStatus <= S_CHARGING) && chargingInfo[groupIndex]->RelayK1K2Status) ||
  808. chargingInfo[groupIndex]->SystemStatus >= S_PREPARING_FOR_EVSE ||
  809. (chargingInfo[groupIndex]->SystemStatus >= S_CCS_PRECHARGE_ST0 && chargingInfo[groupIndex]->SystemStatus <= S_CCS_PRECHARGE_ST1))
  810. {
  811. if (chargingInfo[groupIndex]->EvBatterytargetVoltage > 0 &&
  812. carReqVol != chargingInfo[groupIndex]->EvBatterytargetVoltage)
  813. {
  814. carReqVol = chargingInfo[groupIndex]->EvBatterytargetVoltage;
  815. DEBUG_INFO("ev need vol = %f \n", chargingInfo[groupIndex]->EvBatterytargetVoltage);
  816. }
  817. if (chargingInfo[groupIndex]->EvBatterytargetCurrent > 0 &&
  818. carReqCur != chargingInfo[groupIndex]->EvBatterytargetCurrent)
  819. {
  820. carReqCur = chargingInfo[groupIndex]->EvBatterytargetCurrent;
  821. DEBUG_INFO("ev need cur = %f \n", chargingInfo[groupIndex]->EvBatterytargetCurrent);
  822. }
  823. if (ShmSysConfigAndInfo->SysInfo.MainChargingMode == _MAIN_CHARGING_MODE_MAX)
  824. {
  825. // 智能判斷 Start -----------------------------------------------------------
  826. if (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag == _REASSIGNED_GET_NEW_CAP)
  827. {
  828. if (ShmPsuData->SystemAvailableCurrent != chargingInfo[groupIndex]->AvailableChargingCurrent)
  829. {
  830. // 車端要求電流為該充電槍的額定輸出電流的範圍內
  831. if (chargingInfo[groupIndex]->EvBatterytargetCurrent <= chargingInfo[groupIndex]->AvailableChargingCurrent ||
  832. deratingKeepCount >= DERATING)
  833. {
  834. // 車端降載完成
  835. PRINTF_FUNC("=============Smart Charging : _REASSIGNED_ADJUST_M_TO_A============= Step 3 \n");
  836. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag = _REASSIGNED_ADJUST_M_TO_A;
  837. gettimeofday(&_derating_time, NULL);
  838. deratingKeepCount = 0;
  839. }
  840. else
  841. {
  842. deratingKeepCount++;
  843. }
  844. }
  845. }
  846. else if (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag == _REASSIGNED_ADJUST_M_TO_A)
  847. {
  848. bool isChanged = false;
  849. // 需求電流不降低的情況下 -> 依然要切
  850. if (chargingInfo[groupIndex]->AvailableChargingCurrent < chargingInfo[groupIndex]->EvBatterytargetCurrent)
  851. {
  852. PRINTF_FUNC("** _REASSIGNED_ADJUST_M_TO_A ** Gun_%d, AvailableChargingCurrent = %f, EvBatterytargetCurrent = %f \n", groupIndex,
  853. chargingInfo[groupIndex]->PresentChargingCurrent,
  854. chargingInfo[groupIndex]->AvailableChargingCurrent);
  855. for (byte subIndex = 0; subIndex < ShmPsuData->GroupCount; subIndex++)
  856. {
  857. if (chargingInfo[subIndex]->SystemStatus == S_REASSIGN)
  858. {
  859. if (chargingInfo[subIndex]->PresentChargingCurrent <= CHK_CUR_RANGE)
  860. isChanged = true;
  861. break;
  862. }
  863. }
  864. // 這狀況下輸出端的電流載滿載衝的狀況下,並不會降電流
  865. // 所以只能拉載到該槍端的最大輸出能力
  866. if (chargingInfo[groupIndex]->PresentChargingCurrent >= chargingInfo[groupIndex]->AvailableChargingCurrent - CHK_CUR_RANGE ||
  867. chargingInfo[groupIndex]->PresentChargingCurrent <= CHK_CUR_RANGE)
  868. {
  869. isChanged = true;
  870. }
  871. }
  872. else if ((chargingInfo[groupIndex]->PresentChargingCurrent >= ShmPsuData->PsuGroup[groupIndex].GroupPresentOutputCurrent - CHK_CUR_RANGE) &&
  873. (chargingInfo[groupIndex]->PresentChargingCurrent <= ShmPsuData->PsuGroup[groupIndex].GroupPresentOutputCurrent + CHK_CUR_RANGE))
  874. {
  875. isChanged = true;
  876. }
  877. if (isChanged)
  878. {
  879. PRINTF_FUNC("** _REASSIGNED_ADJUST_M_TO_A ** Gun_%d, PresentChargingCurrent = %f, GroupPresentOutputCurrent = %d \n", groupIndex,
  880. chargingInfo[groupIndex]->PresentChargingCurrent,
  881. ShmPsuData->PsuGroup[groupIndex].GroupPresentOutputCurrent);
  882. // 輸出端與車端要求電流接近
  883. PRINTF_FUNC("=============Smart Charging : _REASSIGNED_RELAY_M_TO_A============= Step 4 \n");
  884. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag = _REASSIGNED_RELAY_M_TO_A;
  885. }
  886. }
  887. // 智能判斷 End -----------------------------------------------------------
  888. if (testtime > 500 &&
  889. (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag >= _REASSIGNED_GET_NEW_CAP && ShmSysConfigAndInfo->SysInfo.ReAssignedFlag <= _REASSIGNED_RELAY_M_TO_A))
  890. {
  891. PRINTF_FUNC("Gun_%d, AvailableChargingCurrent = %f, AvailableChargingPower = %f \n", groupIndex,
  892. chargingInfo[groupIndex]->AvailableChargingCurrent,
  893. chargingInfo[groupIndex]->AvailableChargingPower);
  894. PRINTF_FUNC("Gun_%d, NeedVol = %f, NeedCur = %f \n", groupIndex,
  895. chargingInfo[groupIndex]->EvBatterytargetVoltage,
  896. chargingInfo[groupIndex]->EvBatterytargetCurrent);
  897. PRINTF_FUNC("Gun_%d OutputVol = %f, OutputCur = %f \n", groupIndex,
  898. chargingInfo[groupIndex]->PresentChargingVoltage,
  899. chargingInfo[groupIndex]->PresentChargingCurrent);
  900. gettimeofday(&_test_time, NULL);
  901. }
  902. if (ShmPsuData->SystemAvailablePower > 0)
  903. {
  904. // 調整輸出電流 : 漸進調整方式
  905. if (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag >= _REASSIGNED_ADJUST_M_TO_A)
  906. {
  907. // 當前充電中的目標電壓
  908. float targetVol = chargingInfo[groupIndex]->EvBatterytargetVoltage;
  909. // 當前充電中的目標電流
  910. float targetCur = 0;
  911. // 準備切出去的模塊電流
  912. float deratingCur = 0;
  913. byte reassignIndex = ELEMENT_NOT_FIND;
  914. // 找到等待分配的槍
  915. for (byte subIndex = 0; subIndex < ShmPsuData->GroupCount; subIndex++)
  916. {
  917. if (chargingInfo[subIndex]->SystemStatus == S_REASSIGN)
  918. {
  919. reassignIndex = subIndex;
  920. }
  921. }
  922. if (reassignIndex != ELEMENT_NOT_FIND)
  923. {
  924. //int derating = GetTimeoutValue(_derating_time) / 1000;
  925. //if (derating > 1000)
  926. {
  927. if (ShmPsuData->PsuGroup[reassignIndex].GroupPresentOutputCurrent > 0)
  928. {
  929. deratingCur = ShmPsuData->PsuGroup[reassignIndex].GroupPresentOutputCurrent - DERATING_RANGE;
  930. if (deratingCur <= CHK_CUR_RANGE)
  931. deratingCur = CHK_CUR_RANGE;
  932. PresentOutputVol(reassignIndex, targetVol, deratingCur);
  933. gettimeofday(&_derating_time, NULL);
  934. }
  935. }
  936. // 因為爬的速度沒有降的速度快,所以採兩倍速度爬升
  937. targetCur = ShmPsuData->PsuGroup[groupIndex].GroupPresentOutputCurrent + (DERATING_RANGE * 2);
  938. if (targetCur >= chargingInfo[groupIndex]->EvBatterytargetCurrent)
  939. targetCur = chargingInfo[groupIndex]->EvBatterytargetCurrent;
  940. if (targetVol == 0)
  941. {
  942. SwitchPower(SYSTEM_CMD, PSU_POWER_OFF);
  943. FlashLed(SYSTEM_CMD, PSU_FLASH_NORMAL);
  944. }
  945. else
  946. {
  947. SwitchPower(SYSTEM_CMD, PSU_POWER_ON);
  948. FlashLed(SYSTEM_CMD, PSU_FLASH_ON);
  949. }
  950. }
  951. }
  952. else
  953. {
  954. // 該充電槍的目標電壓與目標電流
  955. PresentOutputVol(SYSTEM_CMD,
  956. chargingInfo[groupIndex]->EvBatterytargetVoltage,
  957. chargingInfo[groupIndex]->EvBatterytargetCurrent);
  958. if (chargingInfo[groupIndex]->EvBatterytargetVoltage == 0)
  959. {
  960. SwitchPower(SYSTEM_CMD, PSU_POWER_OFF);
  961. FlashLed(SYSTEM_CMD, PSU_FLASH_NORMAL);
  962. }
  963. else
  964. {
  965. SwitchPower(SYSTEM_CMD, PSU_POWER_ON);
  966. FlashLed(SYSTEM_CMD, PSU_FLASH_ON);
  967. }
  968. }
  969. }
  970. }
  971. else if (ShmSysConfigAndInfo->SysInfo.MainChargingMode == _MAIN_CHARGING_MODE_AVER)
  972. {
  973. // 智能判斷 Start -----------------------------------------------------------
  974. if (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag == _REASSIGNED_ADJUST_A_TO_M)
  975. {
  976. bool balanceVol = true;
  977. for (byte subIndex = 0; subIndex < ShmPsuData->GroupCount; subIndex++)
  978. {
  979. if (chargingInfo[subIndex]->SystemStatus == S_IDLE ||
  980. chargingInfo[subIndex]->SystemStatus == S_RESERVATION)
  981. {
  982. // 各群電壓接近平衡
  983. if (((chargingInfo[subIndex]->PresentChargingVoltage * 10) < (chargingInfo[groupIndex]->PresentChargingVoltage * 10) - ZERO_VOLTAGE) ||
  984. ((chargingInfo[subIndex]->PresentChargingVoltage * 10) < chargingInfo[groupIndex]->EvBatterytargetVoltage - CHK_VOL_RANGE))
  985. {
  986. PRINTF_FUNC("** _REASSIGNED_ADJUST_A_TO_M ** Gun_%d, PresentChargingVoltage = %f, PresentChargingVoltage_V = %f, EvBatterytargetVoltage = %f \n", subIndex,
  987. chargingInfo[subIndex]->PresentChargingVoltage,
  988. (chargingInfo[groupIndex]->PresentChargingVoltage - ZERO_VOLTAGE),
  989. (chargingInfo[groupIndex]->EvBatterytargetVoltage - CHK_VOL_RANGE));
  990. balanceVol = false;
  991. }
  992. break;
  993. }
  994. }
  995. if (balanceVol)
  996. {
  997. // 閒置端與車端要求電壓接近
  998. PRINTF_FUNC("=============Smart Charging : _REASSIGNED_RELAY_A_TO_M============= Step 13 \n");
  999. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag = _REASSIGNED_RELAY_A_TO_M;
  1000. GetTimeoutValue(_averageComp_time);
  1001. }
  1002. }
  1003. else if(ShmSysConfigAndInfo->SysInfo.ReAssignedFlag == _REASSIGNED_WAITING)
  1004. {
  1005. int avrTime = GetTimeoutValue(_averageComp_time) / 1000;
  1006. if (avrTime > 3000)
  1007. {
  1008. // 閒置端與車端要求電壓接近
  1009. PRINTF_FUNC("=============Smart Charging : _REASSIGNED_COMP============= Step 15 \n");
  1010. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag = _REASSIGNED_COMP;
  1011. }
  1012. }
  1013. // 智能判斷 End -----------------------------------------------------------
  1014. if (testtime > 500 &&
  1015. (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag >= _REASSIGNED_ADJUST_A_TO_M && ShmSysConfigAndInfo->SysInfo.ReAssignedFlag <= _REASSIGNED_WAITING))
  1016. {
  1017. PRINTF_FUNC("Gun_%d, AvailableChargingCurrent = %f, AvailableChargingPower = %f \n", groupIndex,
  1018. chargingInfo[groupIndex]->AvailableChargingCurrent,
  1019. chargingInfo[groupIndex]->AvailableChargingPower);
  1020. PRINTF_FUNC("Gun_%d, NeedVol = %f, NeedCur = %f \n", groupIndex,
  1021. chargingInfo[groupIndex]->EvBatterytargetVoltage,
  1022. chargingInfo[groupIndex]->EvBatterytargetCurrent);
  1023. PRINTF_FUNC("Gun_%d OutputVol = %f, OutputCur = %f \n", groupIndex,
  1024. chargingInfo[groupIndex]->PresentChargingVoltage,
  1025. chargingInfo[groupIndex]->PresentChargingCurrent);
  1026. gettimeofday(&_test_time, NULL);
  1027. }
  1028. if (chargingInfo[groupIndex]->AvailableChargingCurrent > 0)
  1029. {
  1030. if (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag == _REASSIGNED_ADJUST_A_TO_M)
  1031. {
  1032. for (byte subIndex = 0; subIndex < ShmPsuData->GroupCount; subIndex++)
  1033. {
  1034. if (chargingInfo[subIndex]->SystemStatus == S_IDLE ||
  1035. chargingInfo[subIndex]->SystemStatus == S_RESERVATION)
  1036. {
  1037. // 閒置模塊升壓
  1038. PresentOutputVol(subIndex,
  1039. chargingInfo[groupIndex]->EvBatterytargetVoltage,
  1040. ZERO_CURRENT);
  1041. }
  1042. else
  1043. {
  1044. // 充電中的模塊維持輸出
  1045. PresentOutputVol(subIndex,
  1046. chargingInfo[subIndex]->EvBatterytargetVoltage,
  1047. chargingInfo[subIndex]->EvBatterytargetCurrent);
  1048. }
  1049. if (chargingInfo[groupIndex]->EvBatterytargetVoltage == 0)
  1050. {
  1051. SwitchPower(subIndex, PSU_POWER_OFF);
  1052. FlashLed(subIndex, PSU_FLASH_NORMAL);
  1053. }
  1054. else
  1055. {
  1056. SwitchPower(subIndex, PSU_POWER_ON);
  1057. FlashLed(subIndex, PSU_FLASH_ON);
  1058. }
  1059. }
  1060. }
  1061. else
  1062. {
  1063. PresentOutputVol(groupIndex,
  1064. chargingInfo[groupIndex]->EvBatterytargetVoltage,
  1065. chargingInfo[groupIndex]->EvBatterytargetCurrent);
  1066. if (chargingInfo[groupIndex]->EvBatterytargetVoltage == 0)
  1067. {
  1068. SwitchPower(groupIndex, PSU_POWER_OFF);
  1069. FlashLed(groupIndex, PSU_FLASH_NORMAL);
  1070. }
  1071. else
  1072. {
  1073. SwitchPower(groupIndex, PSU_POWER_ON);
  1074. FlashLed(groupIndex, PSU_FLASH_ON);
  1075. }
  1076. }
  1077. }
  1078. }
  1079. }
  1080. else if (chargingInfo[groupIndex]->SystemStatus >= S_TERMINATING &&
  1081. chargingInfo[groupIndex]->SystemStatus <= S_COMPLETE)
  1082. {
  1083. if (ShmSysConfigAndInfo->SysInfo.MainChargingMode == _MAIN_CHARGING_MODE_MAX)
  1084. {
  1085. SwitchPower(SYSTEM_CMD, PSU_POWER_OFF);
  1086. FlashLed(SYSTEM_CMD, PSU_FLASH_NORMAL);
  1087. if (chargingInfo[groupIndex]->SystemStatus == S_TERMINATING)
  1088. {
  1089. if (ShmSysConfigAndInfo->SysInfo.ReAssignedFlag >= _REASSIGNED_PREPARE_M_TO_A &&
  1090. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag <= _REASSIGNED_RELAY_M_TO_A)
  1091. {
  1092. // 代表在切換的過程中,停止充電了
  1093. if (chargingInfo[groupIndex]->PresentChargingCurrent <= STOP_CURRENT)
  1094. ShmSysConfigAndInfo->SysInfo.ReAssignedFlag = _REASSIGNED_RELAY_M_TO_A;
  1095. }
  1096. }
  1097. }
  1098. else if (ShmSysConfigAndInfo->SysInfo.MainChargingMode == _MAIN_CHARGING_MODE_AVER)
  1099. {
  1100. SwitchPower(groupIndex, PSU_POWER_OFF);
  1101. FlashLed(groupIndex, PSU_FLASH_NORMAL);
  1102. }
  1103. }
  1104. }
  1105. break;
  1106. }
  1107. }
  1108. usleep(20000);
  1109. }
  1110. return FAIL;
  1111. }