Infypwr_PsuCommObj.c 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991
  1. /*
  2. * Infypwr_PsuCommObj.c
  3. *
  4. * Created on: 2019年11月26日
  5. * Author: 7564
  6. */
  7. #include "Infypwr_PsuCommObj.h"
  8. #define ARRAY_SIZE(A) (sizeof(A) / sizeof(A[0]))
  9. #define NO 0
  10. #define YES 1
  11. #define DEBUG_LIB 1
  12. void PRINTF_LIB_FUNC(char *string, ...);
  13. float IEEE_754_to_float(const byte raw[4]);
  14. void IEEE_754_to_bytes(float target, byte *bytes2);
  15. // 設備號 0x09 與 0x0C 似乎是模塊
  16. unsigned int filter[3] = { 0x87570000, 0x8E9B0000, 0x829B0000 };
  17. //================================================
  18. // Private function
  19. //================================================
  20. void PRINTF_LIB_FUNC(char *string, ...)
  21. {
  22. if (DEBUG_LIB)
  23. {
  24. va_list args;
  25. char buffer[4096];
  26. va_start(args, string);
  27. vsnprintf(buffer, sizeof(buffer), string, args);
  28. va_end(args);
  29. printf("%s \n", buffer);
  30. }
  31. }
  32. float IEEE_754_to_float(const byte raw[4])
  33. {
  34. int sign = (raw[0] >> 7) ? -1 : 1;
  35. byte exponent = (raw[0] << 1) + (raw[1] >> 7) - 126;
  36. unsigned int fraction_bits = ((raw[1] & 0x7F) << 16) + (raw[2] << 8) + raw[3];
  37. float fraction = 0.5f;
  38. for (byte ii = 0; ii < 24; ++ii)
  39. fraction += ldexpf((fraction_bits >> (23 - ii)) & 1, -(ii + 1));
  40. float significand = sign * fraction;
  41. return ldexpf(significand, exponent);
  42. }
  43. void IEEE_754_to_bytes(float target, byte *bytes2)
  44. {
  45. int value2 = 0;
  46. number.f = target;
  47. int index = 31;
  48. value2 |= number.raw.sign << index;
  49. int k;
  50. for (k = 8 - 1; k >= 0; k--)
  51. {
  52. index--;
  53. if ((number.raw.exponent >> k) & 1)
  54. value2 |= 1 << index;
  55. }
  56. for (k = 23 - 1; k >= 0; k--)
  57. {
  58. index--;
  59. if ((number.raw.mantissa >> k) & 1)
  60. value2 |= 1 << index;
  61. }
  62. *(bytes2) = (value2 >> 24) & 0xFF;
  63. *(bytes2 + 1) = (value2 >> 16) & 0xFF;
  64. *(bytes2 + 2) = (value2 >> 8) & 0xFF;
  65. *(bytes2 + 3) = value2 & 0xFF;
  66. }
  67. //================================================
  68. // Callback function
  69. //================================================
  70. void RefreshStatus(void *func)
  71. {
  72. return_status = func;
  73. }
  74. void RefreshModuleCount(void *func)
  75. {
  76. return_module_count = func;
  77. }
  78. void RefreshAvailableCap(void *func)
  79. {
  80. return_available_cap = func;
  81. }
  82. void RefreshFwVersion(void *func)
  83. {
  84. return_fw_version = func;
  85. }
  86. void RefreshInputVol(void *func)
  87. {
  88. return_input_vol = func;
  89. }
  90. void RefreshGetOutput(void *func)
  91. {
  92. return_get_output = func;
  93. }
  94. void RefreshGetOutputF(void *func)
  95. {
  96. return_get_output_float = func;
  97. }
  98. void RefreshMisInfo(void *func)
  99. {
  100. return_mis_info = func;
  101. }
  102. void RefreshIavailable(void *func)
  103. {
  104. return_iavail_info = func;
  105. }
  106. void AutoMode_RefreshOutputAndTemp(void *func)
  107. {
  108. return_output_temp = func;
  109. }
  110. void AutoMode_RefreshModuleStatus(void *func)
  111. {
  112. return_module_status = func;
  113. }
  114. void AutoMode_RefreshModuleInput(void *func)
  115. {
  116. return_module_input = func;
  117. }
  118. //================================================
  119. // CANBUS initialization
  120. //================================================
  121. int InitCanBus()
  122. {
  123. int s0,nbytes;
  124. struct timeval tv;
  125. struct ifreq ifr0;
  126. struct sockaddr_can addr0;
  127. //struct can_filter rfilter[2];
  128. system("/sbin/ip link set can1 down");
  129. system("/sbin/ip link set can1 type can bitrate 500000 restart-ms 100");
  130. system("/sbin/ip link set can1 up");
  131. s0 = socket(PF_CAN, SOCK_RAW, CAN_RAW);
  132. tv.tv_sec = 0;
  133. tv.tv_usec = 10000;
  134. if (setsockopt(s0, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(struct timeval)) < 0)
  135. {
  136. #ifdef SystemLogMessage
  137. PRINTF_LIB_FUNC("Set SO_RCVTIMEO NG");
  138. #endif
  139. }
  140. nbytes=40960;
  141. if (setsockopt(s0, SOL_SOCKET, SO_RCVBUF, &nbytes, sizeof(int)) < 0)
  142. {
  143. #ifdef SystemLogMessage
  144. PRINTF_LIB_FUNC("Set SO_RCVBUF NG");
  145. #endif
  146. }
  147. nbytes=40960;
  148. if (setsockopt(s0, SOL_SOCKET, SO_SNDBUF, &nbytes, sizeof(int)) < 0)
  149. {
  150. #ifdef SystemLogMessage
  151. PRINTF_LIB_FUNC("Set SO_SNDBUF NG");
  152. #endif
  153. }
  154. nbytes=40960;
  155. // rfilter[0].can_id = 0x0757F803 | CAN_EFF_FLAG;
  156. // rfilter[0].can_mask = (CAN_EFF_FLAG | CAN_RTR_FLAG | CAN_EFF_MASK);
  157. // rfilter[1].can_id = 0x0757F80B | CAN_EFF_FLAG;
  158. // rfilter[1].can_mask = (CAN_EFF_FLAG | CAN_RTR_FLAG | CAN_EFF_MASK);
  159. //
  160. // if (setsockopt(s0, CAN_INV_FILTER, CAN_RAW_FILTER_MAX, &rfilter, sizeof(rfilter)) < 0)
  161. // {
  162. // #ifdef SystemLogMessage
  163. // printf("*********************************Set SOL_CAN_RAW NG");
  164. // #endif
  165. // }
  166. strcpy(ifr0.ifr_name, "can1" );
  167. ioctl(s0, SIOCGIFINDEX, &ifr0); /* ifr.ifr_ifindex gets filled with that device's index */
  168. addr0.can_family = AF_CAN;
  169. addr0.can_ifindex = ifr0.ifr_ifindex;
  170. bind(s0, (struct sockaddr *)&addr0, sizeof(addr0));
  171. return s0;
  172. }
  173. //================================================
  174. // Receive Cmd from canbus
  175. //================================================
  176. byte isFilterValue(unsigned int value)
  177. {
  178. byte result = NO;
  179. for(byte i = 0; i < ARRAY_SIZE(filter); i++)
  180. {
  181. if (value == filter[i])
  182. {
  183. result = YES;
  184. break;
  185. }
  186. }
  187. return result;
  188. }
  189. bool GetRealIndexByGroup(byte *_address)
  190. {
  191. for (byte i = 0; i < 2; i++)
  192. {
  193. for (byte j = 0; j < infy_pow_info[i].psuCount; j++)
  194. {
  195. if (infy_pow_info[i].serialNumber[j] == *_address)
  196. {
  197. *_address = infy_pow_info[i].targetNumber[j];
  198. if (i > 0)
  199. *_address += infy_pow_info[0].psuCount;
  200. return true;
  201. }
  202. }
  203. }
  204. return false;
  205. }
  206. void ReceiveDataFromCanBus()
  207. {
  208. int nbytes;
  209. struct can_frame frame;
  210. int intCmd = 0;
  211. byte group, address;
  212. byte _totalModuleCount = 0;
  213. bool colFinished = false;
  214. for (byte i = 0; i < 2; i++)
  215. {
  216. infy_pow_info[i].psuCount = 0;
  217. for (byte j = 0; j < 12; j++)
  218. {
  219. infy_pow_info[i].serialNumber[j] = 0xFF;
  220. infy_pow_info[i].targetNumber[j] = 0xFF;
  221. }
  222. }
  223. while(1)
  224. {
  225. memset(&frame, 0, sizeof(struct can_frame));
  226. nbytes = read(CanFd, &frame, sizeof(struct can_frame));
  227. if (nbytes > 0)
  228. {
  229. if (isFilterValue(frame.can_id & 0xFFFF0000) == YES)
  230. continue;
  231. // if (frame.can_id == 0x82caf000 || frame.can_id == 0x82caf001 ||
  232. // frame.can_id == 0x82c1f000 || frame.can_id == 0x82c1f001 ||
  233. // frame.can_id == 0x82ccf000 || frame.can_id == 0x82ccf001 ||
  234. // frame.can_id == 0x9901ff00 || frame.can_id == 0x9902ff00 || frame.can_id == 0x9903ff00 ||
  235. // frame.can_id == 0x9901ff01 || frame.can_id == 0x9902ff01 || frame.can_id == 0x9903ff01 ||
  236. // frame.can_id == 0x82c7f000 || frame.can_id == 0x82c7f001)
  237. // {}
  238. // else
  239. // printf("can_id = %x \n", frame.can_id);
  240. frame.can_id = frame.can_id & CAN_EFF_MASK;
  241. intCmd = frame.can_id & 0x00FF0000;
  242. intCmd |= INFYPWR_GROUP_SHIFT | intCmd;
  243. switch (intCmd)
  244. {
  245. case WALK_IN_MODE:
  246. case INFYPWR_GROUP_SHIFT | WALK_IN_MODE:
  247. {
  248. //address = frame.can_id & 0x000000FF;
  249. // printf("walk in response address = %d, [0] = %d, [1] = %d, [2] = %d, [3] = %d, [4] = %d, [5] = %d, [6] = %d, [7] = %d \n"
  250. // , address,
  251. // frame.data[0], frame.data[1],
  252. // frame.data[2], frame.data[3],
  253. // frame.data[4], frame.data[5],
  254. // frame.data[6], frame.data[7]);
  255. }
  256. break;
  257. case INFYPWR_GROUP_SHIFT | STATUS:
  258. {
  259. group = frame.data[2];
  260. byte SN = frame.can_id & 0x000000FF;
  261. bool isfind = false;
  262. if (group < 2 && !colFinished)
  263. {
  264. for(byte _index = 0; _index < infy_pow_info[group].psuCount; _index++)
  265. {
  266. if (infy_pow_info[group].serialNumber[_index] == SN)
  267. {
  268. isfind = true;
  269. break;
  270. }
  271. }
  272. if (!isfind)
  273. {
  274. infy_pow_info[group].serialNumber[infy_pow_info[group].psuCount] = SN;
  275. infy_pow_info[group].targetNumber[infy_pow_info[group].psuCount] = infy_pow_info[group].psuCount;
  276. infy_pow_info[group].psuCount++;
  277. }
  278. byte subPcount = 0;
  279. for (byte i = 0; i < 2; i++)
  280. {
  281. subPcount += infy_pow_info[i].psuCount;
  282. }
  283. if (subPcount > 0 && subPcount == _totalModuleCount)
  284. colFinished = true;
  285. }
  286. short temp = frame.data[4];
  287. int status = (frame.data[5] << 16) + (frame.data[6] << 8) + frame.data[7];
  288. return_status(group, SN, temp, status);
  289. //PRINTF_LIB_FUNC("group = %d, address = %d, temp = %d \n", group, address, temp);
  290. }
  291. break;
  292. case INFYPWR_GROUP_SHIFT | MODULE_COUNT:
  293. {
  294. // 回傳模組數量
  295. group = frame.can_id & 0x000000FF;
  296. byte count = frame.data[2];
  297. if (group == SYSTEM_CMD)
  298. {
  299. _totalModuleCount = count;
  300. }
  301. return_module_count(group, count);
  302. //PRINTF_LIB_FUNC("group = %d, count = %d \n", group, count);
  303. }
  304. break;
  305. case MODULE_CAP:
  306. case INFYPWR_GROUP_SHIFT | MODULE_CAP:
  307. {
  308. if (!colFinished)
  309. break;
  310. // 回傳輸出能力 : 最大電壓、最小電壓、最大電流、額定功率
  311. address = frame.can_id & 0x000000FF;
  312. if(!GetRealIndexByGroup(&address))
  313. break;
  314. short maxVol = ((frame.data[0] << 8) + frame.data[1]) * 10;
  315. short minVol = ((frame.data[2] << 8) + frame.data[3]) * 10;
  316. short maxCur = (frame.data[4] << 8) + frame.data[5];
  317. short totalPow = ((frame.data[6] << 8) + frame.data[7]) / 10;
  318. return_available_cap(address, maxVol, minVol, maxCur, totalPow);
  319. // PRINTF_LIB_FUNC("address = %d, maxVol = %d, minVol = %d, maxCur = %d, totalPow = %d \n",
  320. // address, maxVol, minVol, maxCur, totalPow);
  321. }
  322. break;
  323. case INFYPWR_GROUP_SHIFT | MODULE_OUTPUT_VOL_CUR:
  324. {
  325. // 回傳當前輸出電壓電流
  326. address = frame.can_id & 0x000000FF;
  327. int outputVol = ((frame.data[0] << 24) + (frame.data[1] << 16) + (frame.data[2] << 8) + frame.data[3]) / 100;
  328. int outputCur = ((frame.data[4] << 24) + (frame.data[5] << 16) + (frame.data[6] << 8) + frame.data[7]) / 100;
  329. return_get_output(address, outputVol, outputCur);
  330. //PRINTF_LIB_FUNC("address = %d, outputVol = %d, outputCur = %d \n", address, outputVol, outputCur);
  331. }
  332. break;
  333. case INFYPWR_GROUP_SHIFT | MODULE_OUTPUT_VOL_CUR_FLOAT:
  334. {
  335. group = frame.can_id & 0x000000FF;
  336. byte vol[4], cur[4];
  337. memcpy(vol, frame.data, 4);
  338. memcpy(cur, frame.data + 4, 4);
  339. float _Vol = IEEE_754_to_float(vol);
  340. float _Cur = IEEE_754_to_float(cur);
  341. return_get_output_float(group, _Vol, _Cur);
  342. }
  343. break;
  344. case INFYPWR_GROUP_SHIFT | MODULE_IAVAILABLE:
  345. case MODULE_IAVAILABLE:
  346. {
  347. if (!colFinished)
  348. break;
  349. // 回傳降載後的電流
  350. address = frame.can_id & 0x000000FF;
  351. if(!GetRealIndexByGroup(&address))
  352. break;
  353. unsigned short vextVol = ((frame.data[0] << 8) + frame.data[1]);
  354. unsigned short iAvailCur = ((frame.data[2] << 8) + frame.data[3]);
  355. return_iavail_info(address, iAvailCur, vextVol);
  356. //PRINTF_LIB_FUNC("address = %d, iAvailCur = %d \n", address, iAvailCur);
  357. }
  358. break;
  359. case INFYPWR_GROUP_SHIFT | MODULE_MIS_INFO:
  360. {
  361. if (!colFinished)
  362. break;
  363. address = frame.can_id & 0x000000FF;
  364. float ReturnValue;
  365. byte value[4];
  366. byte type;
  367. if(!GetRealIndexByGroup(&address))
  368. break;
  369. memcpy(value, frame.data + 4, sizeof(value));
  370. ReturnValue = IEEE_754_to_float(value);
  371. if (frame.data[0] == ((FAN_SPEED_CMD >> 8) & 0xFF) && frame.data[1] == (FAN_SPEED_CMD & 0xFF))
  372. {
  373. type = 1;
  374. return_mis_info(address, ReturnValue, type);
  375. //PRINTF_LIB_FUNC("address = %d, FanSpeed = %f \n", address, FanSpeed);
  376. }
  377. else if (frame.data[0] == ((TEMP_DC_CMD >> 8) & 0xFF) && frame.data[1] == (TEMP_DC_CMD & 0xFF))
  378. {
  379. type = 2;
  380. return_mis_info(address, ReturnValue, type);
  381. //PRINTF_LIB_FUNC("address = %d, FanSpeed = %f \n", address, FanSpeed);
  382. }
  383. else if (frame.data[0] == ((TEMP_DC_CMD >> 8) & 0xFF) && frame.data[1] == (TEMP_DC_CMD & 0xFF))
  384. {
  385. type = 3;
  386. return_mis_info(address, ReturnValue, type);
  387. //PRINTF_LIB_FUNC("address = %d, FanSpeed = %f \n", address, FanSpeed);
  388. }
  389. }
  390. break;
  391. case INFYPWR_GROUP_SHIFT | MODULE_VER:
  392. {
  393. if (!colFinished)
  394. break;
  395. // 回傳版號 : 無系統回覆功能
  396. address = frame.can_id & 0x000000FF;
  397. if(!GetRealIndexByGroup(&address))
  398. break;
  399. short dcSwVer = ((frame.data[0] << 8) + frame.data[1]);
  400. short pfcSwVer = ((frame.data[2] << 8) + frame.data[3]);
  401. short hwVer = ((frame.data[4] << 8) + frame.data[5]);
  402. return_fw_version(address, dcSwVer, pfcSwVer, hwVer);
  403. //PRINTF_LIB_FUNC("address = %d, DC %d, PFC %d, HW %d \n", address, dcSwVer, pfcSwVer, hwVer);
  404. }
  405. break;
  406. case INFYPWR_GROUP_SHIFT | MODULE_BARCODE:
  407. {
  408. // 回傳BarCode
  409. }
  410. break;
  411. case INFYPWR_GROUP_SHIFT | MODULE_INPUT:
  412. {
  413. if (!colFinished)
  414. break;
  415. // 回傳三向輸入電壓
  416. address = frame.can_id & 0x000000FF;
  417. if(!GetRealIndexByGroup(&address))
  418. break;
  419. short abVol = ((frame.data[0] << 8) + frame.data[1]) / 10;
  420. short bcVol = ((frame.data[2] << 8) + frame.data[3]) / 10;
  421. short caVol = ((frame.data[4] << 8) + frame.data[5]) / 10;
  422. return_input_vol(address, abVol, bcVol, caVol);
  423. //PRINTF_LIB_FUNC("address = %d, abVol = %d, bcVol = %d, caVol = %d \n", address, abVol, bcVol, caVol);
  424. }
  425. break;
  426. case INFYPWR_GROUP_SHIFT | AUTO_OUTPUT_TEMP:
  427. {
  428. if (!colFinished)
  429. break;
  430. /*Test mode used*/
  431. // 回傳輸出值與入風口溫度
  432. address = frame.can_id & 0x000000FF;
  433. if(!GetRealIndexByGroup(&address))
  434. break;
  435. short outputVol = ((frame.data[0] << 8) + frame.data[1]);
  436. short outputCur = ((frame.data[2] << 8) + frame.data[3]);
  437. short outputPow = ((frame.data[4] << 8) + frame.data[5]);
  438. byte temp = frame.data[6];
  439. return_output_temp(address, outputVol, outputCur, outputPow, temp);
  440. //PRINTF_LIB_FUNC("address = %d, abVol = %d, bcVol = %d, caVol = %d \n", address, abVol, bcVol, caVol);
  441. }
  442. break;
  443. case INFYPWR_GROUP_SHIFT | AUTO_MODULE_STATUS:
  444. {
  445. if (!colFinished)
  446. break;
  447. /*Test mode used*/
  448. // 回傳輸出值與入風口溫度
  449. address = frame.can_id & 0x000000FF;
  450. if(!GetRealIndexByGroup(&address))
  451. break;
  452. byte isErr = (frame.data[0] >> 0) & 0x01;
  453. byte status = (frame.data[0] >> 1) & 0x01;
  454. byte err1 = frame.data[2];
  455. byte err2 = frame.data[3];
  456. byte err3 = frame.data[4];
  457. byte err4 = frame.data[5];
  458. return_module_status(address, isErr, status, err1, err2, err3, err4);
  459. //PRINTF_LIB_FUNC("address = %d, abVol = %d, bcVol = %d, caVol = %d \n", address, abVol, bcVol, caVol);
  460. }
  461. break;
  462. case INFYPWR_GROUP_SHIFT | AUTO_MODULE_INPUT:
  463. {
  464. if (!colFinished)
  465. break;
  466. /*Test mode used*/
  467. // 回傳輸出值與入風口溫度
  468. address = frame.can_id & 0x000000FF;
  469. if(!GetRealIndexByGroup(&address))
  470. break;
  471. short vR = ((frame.data[0] << 8) + frame.data[1]);
  472. short vS = ((frame.data[2] << 8) + frame.data[3]);
  473. short vT = ((frame.data[4] << 8) + frame.data[5]);
  474. return_module_input(address, vR, vS, vT);
  475. //PRINTF_LIB_FUNC("address = %d, abVol = %d, bcVol = %d, caVol = %d \n", address, abVol, bcVol, caVol);
  476. }
  477. break;
  478. }
  479. }
  480. else
  481. usleep(10000);
  482. }
  483. }
  484. //================================================
  485. // Private Function
  486. //================================================
  487. void SendCmdToPsu(int cmd, byte *data, byte dataLen)
  488. {
  489. struct can_frame frame;
  490. //設定 CANBSU 2.0B 長封包
  491. cmd = cmd | 0x80000000;
  492. frame.can_id = cmd;
  493. frame.can_dlc = dataLen;
  494. memcpy(frame.data, data, dataLen);
  495. write(CanFd, &frame, sizeof(struct can_frame));
  496. // 群命令才 delay
  497. if ((cmd & 0x0000FF00) == INFYPWR_BROADCAST)
  498. usleep(CMD_DELAY_TIME);
  499. }
  500. bool InitialCommunication()
  501. {
  502. CanFd = InitCanBus();
  503. if(CanFd < 0)
  504. {
  505. PRINTF_LIB_FUNC("Init can bus fail.\n");
  506. return false;
  507. }
  508. recFork = fork();
  509. if(recFork > 0)
  510. {
  511. ReceiveDataFromCanBus();
  512. }
  513. else if(recFork > 0)
  514. {
  515. PRINTF_LIB_FUNC("fork fail\n");
  516. }
  517. return true;
  518. }
  519. //================================================
  520. // API Function
  521. //================================================
  522. void SwitchPower(byte group, byte value)
  523. {
  524. byte data[8];
  525. uint cmd = INFYPWR_CMD | SWITCH_POWER;
  526. memset(data, 0x00, ARRAY_SIZE(data));
  527. // 1 : 關機
  528. // 0 : 開機
  529. data[0] = value;
  530. if (group == SYSTEM_CMD)
  531. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  532. else
  533. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  534. SendCmdToPsu(cmd, data, sizeof(data));
  535. }
  536. void SleepMode(byte group, byte value)
  537. {
  538. byte data[8];
  539. uint cmd = INFYPWR_CMD | SLEEP_MODE;
  540. memset(data, 0x00, ARRAY_SIZE(data));
  541. // 1 : 休眠
  542. // 0 : 起床
  543. data[0] = value;
  544. if (group == SYSTEM_CMD)
  545. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  546. else
  547. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  548. SendCmdToPsu(cmd, data, sizeof(data));
  549. }
  550. void FlashLed(byte group, byte value)
  551. {
  552. byte data[8];
  553. uint cmd = INFYPWR_CMD | FLASH_LED;
  554. memset(data, 0x00, ARRAY_SIZE(data));
  555. // 1 : 閃爍
  556. // 0 : 正常
  557. data[0] = value;
  558. if (group == SYSTEM_CMD)
  559. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  560. else
  561. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  562. SendCmdToPsu(cmd, data, sizeof(data));
  563. }
  564. void PresentOutputVol(byte group, int voltage, int current)
  565. {
  566. byte data[8];
  567. uint cmd = INFYPWR_CMD | PRESENT_OUT_VOL;
  568. int Vol = voltage * 100;
  569. int Cur = current * 100;
  570. memset(data, 0x00, ARRAY_SIZE(data));
  571. // 輸出電壓
  572. data[0] = (Vol >> 24) & 0xFF;
  573. data[1] = (Vol >> 16) & 0xFF;
  574. data[2] = (Vol >> 8) & 0xFF;
  575. data[3] = Vol & 0xFF;
  576. // 輸出電流
  577. data[4] = (Cur >> 24) & 0xFF;
  578. data[5] = (Cur >> 16) & 0xFF;
  579. data[6] = (Cur >> 8) & 0xFF;
  580. data[7] = Cur & 0xFF;
  581. if (group == SYSTEM_CMD)
  582. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  583. else
  584. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  585. SendCmdToPsu(cmd, data, sizeof(data));
  586. }
  587. void FanNoiseInfo(byte group, byte value)
  588. {
  589. byte data[8];
  590. uint cmd = INFYPWR_CMD | MIS_INFO;
  591. memset(data, 0x00, ARRAY_SIZE(data));
  592. // 風扇低噪音
  593. data[0] = 0x11;
  594. data[1] = 0x13;
  595. // 0xA0 power poriority mode
  596. // 0xA1 denoise mode
  597. // 0xA2 quiet mode
  598. data[7] = value;
  599. if (group == SYSTEM_CMD)
  600. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  601. else
  602. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  603. SendCmdToPsu(cmd, data, sizeof(data));
  604. }
  605. void SetWalkInConfig(byte group, byte enable, byte sec)
  606. {
  607. byte data[8];
  608. uint cmd = INFYPWR_CMD | WALK_IN_MODE;
  609. memset(data, 0x00, ARRAY_SIZE(data));
  610. unsigned short _Sec = sec * 100;
  611. // Walk-in mode enable
  612. data[0] = enable;
  613. // Walk-in time (default == 5s)
  614. data[6] = (_Sec >> 8) & 0xFF;
  615. data[7] = _Sec & 0xFF;
  616. if (group == SYSTEM_CMD)
  617. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  618. else
  619. cmd |= (group << 8) | INFYPWR_DEFAULT;
  620. //printf("walk in cmd = %x \n", cmd);
  621. SendCmdToPsu(cmd, data, sizeof(data));
  622. }
  623. void SetDirModulePresentOutput(byte group, int voltage, int current, byte _switch, byte _interRelay)
  624. {
  625. byte data[8];
  626. uint cmd = TEST_PRESENT_OUT; //0x180100E5
  627. memset(data, 0x00, ARRAY_SIZE(data));
  628. // 輸出電壓
  629. data[0] = (voltage >> 8) & 0xFF;
  630. data[1] = voltage & 0xFF;
  631. // 輸出電流
  632. data[2] = (current >> 8) & 0xFF;
  633. data[3] = current & 0xFF;
  634. // 開 / 關
  635. data[4] = _switch;
  636. // Internal Relay
  637. data[5] = _interRelay;
  638. if (group == SYSTEM_CMD)
  639. cmd |= INFYPWR_BROADCAST;
  640. else
  641. cmd |= (group << 8);
  642. SendCmdToPsu(cmd, data, sizeof(data));
  643. }
  644. void SetDipSwitchMode()
  645. {
  646. byte data[8];
  647. uint cmd = INFYPWR_CMD | DIP_SWITCH_MODE;
  648. memset(data, 0x00, ARRAY_SIZE(data));
  649. data[0] = 0x01;
  650. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  651. SendCmdToPsu(cmd, data, sizeof(data));
  652. }
  653. /**********************************************************************************/
  654. /*** ***/
  655. /*** Get ***/
  656. /*** ***/
  657. /**********************************************************************************/
  658. void GetStatus(byte group)
  659. {
  660. byte data[8];
  661. uint cmd = INFYPWR_CMD | STATUS;
  662. memset(data, 0x00, ARRAY_SIZE(data));
  663. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  664. SendCmdToPsu(cmd, data, sizeof(data));
  665. }
  666. void GetFanSpeed(byte group)
  667. {
  668. uint cmd;
  669. byte data[8];
  670. cmd = INFYPWR_CMD | MODULE_MIS_INFO;
  671. memset(data, 0x00, ARRAY_SIZE(data));
  672. data[0] = (FAN_SPEED_CMD >> 8) & 0xFF;
  673. data[1] = FAN_SPEED_CMD & 0xFF;
  674. if (group == (INFYPWR_BROADCAST >> 8))
  675. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  676. else
  677. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  678. SendCmdToPsu(cmd, data, sizeof(data));
  679. }
  680. void GetDcTemperature(byte group)
  681. {
  682. uint cmd;
  683. byte data[8];
  684. cmd = INFYPWR_CMD | MODULE_MIS_INFO;
  685. memset(data, 0x00, ARRAY_SIZE(data));
  686. data[0] = (TEMP_DC_CMD >> 8) & 0xFF;
  687. data[1] = TEMP_DC_CMD & 0xFF;
  688. if (group == (INFYPWR_BROADCAST >> 8))
  689. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  690. else
  691. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  692. SendCmdToPsu(cmd, data, sizeof(data));
  693. }
  694. void GetPfcTemperature(byte group)
  695. {
  696. uint cmd;
  697. byte data[8];
  698. cmd = INFYPWR_CMD | MODULE_MIS_INFO;
  699. memset(data, 0x00, ARRAY_SIZE(data));
  700. data[0] = (TEMP_PFC_CMD >> 8) & 0xFF;
  701. data[1] = TEMP_PFC_CMD & 0xFF;
  702. if (group == (INFYPWR_BROADCAST >> 8))
  703. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  704. else
  705. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  706. SendCmdToPsu(cmd, data, sizeof(data));
  707. }
  708. void GetModuleCount(byte group)
  709. {
  710. byte data[8];
  711. uint cmd = INFYPWR_CMD | MODULE_COUNT;
  712. memset(data, 0x00, ARRAY_SIZE(data));
  713. if (group == SYSTEM_CMD)
  714. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  715. else
  716. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  717. SendCmdToPsu(cmd, data, sizeof(data));
  718. }
  719. void GetModuleVer(byte group)
  720. {
  721. // 無系統廣播功能
  722. byte data[8];
  723. uint cmd = INFYPWR_CMD | MODULE_VER;
  724. memset(data, 0x00, ARRAY_SIZE(data));
  725. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  726. //PRINTF_LIB_FUNC("GetModuleVer cmd = %x\n", cmd);
  727. SendCmdToPsu(cmd, data, sizeof(data));
  728. }
  729. void GetModuleCap(byte group)
  730. {
  731. byte data[8];
  732. uint cmd = INFYPWR_CMD | MODULE_CAP;
  733. memset(data, 0x00, ARRAY_SIZE(data));
  734. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  735. //PRINTF_LIB_FUNC("GetModuleCap cmd = %x\n", cmd);
  736. SendCmdToPsu(cmd, data, sizeof(data));
  737. }
  738. void GetModuleBarCode(byte group)
  739. {
  740. // 無系統廣播功能
  741. byte data[8];
  742. uint cmd = INFYPWR_CMD | MODULE_BARCODE;
  743. memset(data, 0x00, ARRAY_SIZE(data));
  744. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  745. SendCmdToPsu(cmd, data, sizeof(data));
  746. }
  747. void GetModuleInput(byte group)
  748. {
  749. // 無系統廣播功能
  750. byte data[8];
  751. uint cmd = INFYPWR_CMD | MODULE_INPUT;
  752. memset(data, 0x00, ARRAY_SIZE(data));
  753. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  754. SendCmdToPsu(cmd, data, sizeof(data));
  755. }
  756. void GetModuleIavailable(byte group)
  757. {
  758. byte data[8];
  759. uint cmd = INFYPWR_CMD | MODULE_IAVAILABLE;
  760. memset(data, 0x00, ARRAY_SIZE(data));
  761. if (group == SYSTEM_CMD)
  762. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  763. else
  764. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  765. SendCmdToPsu(cmd, data, sizeof(data));
  766. }
  767. void GetModuleOutput(byte group)
  768. {
  769. byte data[8];
  770. uint cmd = INFYPWR_CMD | MODULE_OUTPUT_VOL_CUR;
  771. memset(data, 0x00, ARRAY_SIZE(data));
  772. if (group == SYSTEM_CMD)
  773. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  774. else
  775. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  776. SendCmdToPsu(cmd, data, sizeof(data));
  777. }
  778. void GetModuleOutputF(byte group)
  779. {
  780. byte data[8];
  781. uint cmd = INFYPWR_CMD | MODULE_OUTPUT_VOL_CUR_FLOAT;
  782. memset(data, 0x00, ARRAY_SIZE(data));
  783. if (group == SYSTEM_CMD)
  784. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  785. else
  786. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  787. SendCmdToPsu(cmd, data, sizeof(data));
  788. }
  789. /**********************************************************************************/
  790. /*** ***/
  791. /*** Upgrate ***/
  792. /*** ***/
  793. /**********************************************************************************/
  794. void ChangePsuBaudrate(short baudrate)
  795. {
  796. byte data[8];
  797. uint cmd = CHANGE_BAUDRATE; //0x180100E5
  798. memset(data, 0x00, ARRAY_SIZE(data));
  799. data[0] = 0x11;
  800. data[1] = 0x26;
  801. if (baudrate == 125)
  802. data[7] = 0xA0;
  803. else if (baudrate == 250)
  804. data[7] = 0xA1;
  805. else if (baudrate == 500)
  806. data[7] = 0xA2;
  807. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  808. SendCmdToPsu(cmd, data, sizeof(data));
  809. }