Module_PsuComm.c 61 KB

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