Infypwr_PsuCommObj.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584
  1. /*
  2. * Infypwr_PsuCommObj.c
  3. *
  4. * Created on: 2019年11月26日
  5. * Author: 7564
  6. */
  7. #include "Infypwr_PsuCommObj.h"
  8. #define DEBUG_LIB 1
  9. void PRINTF_LIB_FUNC(char *string, ...);
  10. float IEEE_754_to_float(const byte raw[4]);
  11. void IEEE_754_to_bytes(float target, byte *bytes2);
  12. //================================================
  13. // Private function
  14. //================================================
  15. void PRINTF_LIB_FUNC(char *string, ...)
  16. {
  17. if (DEBUG_LIB)
  18. {
  19. va_list args;
  20. char buffer[4096];
  21. va_start(args, string);
  22. vsnprintf(buffer, sizeof(buffer), string, args);
  23. va_end(args);
  24. printf("%s \n", buffer);
  25. }
  26. }
  27. float IEEE_754_to_float(const byte raw[4])
  28. {
  29. int sign = (raw[0] >> 7) ? -1 : 1;
  30. byte exponent = (raw[0] << 1) + (raw[1] >> 7) - 126;
  31. unsigned int fraction_bits = ((raw[1] & 0x7F) << 16) + (raw[2] << 8) + raw[3];
  32. float fraction = 0.5f;
  33. for (byte ii = 0; ii < 24; ++ii)
  34. fraction += ldexpf((fraction_bits >> (23 - ii)) & 1, -(ii + 1));
  35. float significand = sign * fraction;
  36. return ldexpf(significand, exponent);
  37. }
  38. void IEEE_754_to_bytes(float target, byte *bytes2)
  39. {
  40. int value2 = 0;
  41. number.f = target;
  42. int index = 31;
  43. value2 |= number.raw.sign << index;
  44. int k;
  45. for (k = 8 - 1; k >= 0; k--)
  46. {
  47. index--;
  48. if ((number.raw.exponent >> k) & 1)
  49. value2 |= 1 << index;
  50. }
  51. for (k = 23 - 1; k >= 0; k--)
  52. {
  53. index--;
  54. if ((number.raw.mantissa >> k) & 1)
  55. value2 |= 1 << index;
  56. }
  57. *(bytes2) = (value2 >> 24) & 0xFF;
  58. *(bytes2 + 1) = (value2 >> 16) & 0xFF;
  59. *(bytes2 + 2) = (value2 >> 8) & 0xFF;
  60. *(bytes2 + 3) = value2 & 0xFF;
  61. }
  62. //================================================
  63. // Callback function
  64. //================================================
  65. void RefreshStatus(void *func)
  66. {
  67. return_status = func;
  68. }
  69. void RefreshModuleCount(void *func)
  70. {
  71. return_module_count = func;
  72. }
  73. void RefreshAvailableCap(void *func)
  74. {
  75. return_available_cap = func;
  76. }
  77. void RefreshFwVersion(void *func)
  78. {
  79. return_fw_version = func;
  80. }
  81. void RefreshInputVol(void *func)
  82. {
  83. return_input_vol = func;
  84. }
  85. void RefreshGetOutput(void *func)
  86. {
  87. return_get_output = func;
  88. }
  89. void RefreshFanInfo(void *func)
  90. {
  91. return_fanspeed_info = func;
  92. }
  93. void RefreshIavailable(void *func)
  94. {
  95. return_iavail_info = func;
  96. }
  97. //================================================
  98. // CANBUS initialization
  99. //================================================
  100. int InitCanBus()
  101. {
  102. int s0,nbytes;
  103. struct timeval tv;
  104. struct ifreq ifr0;
  105. struct sockaddr_can addr0;
  106. system("/sbin/ip link set can1 down");
  107. system("/sbin/ip link set can1 type can bitrate 500000 restart-ms 100");
  108. system("/sbin/ip link set can1 up");
  109. s0 = socket(PF_CAN, SOCK_RAW, CAN_RAW);
  110. tv.tv_sec = 0;
  111. tv.tv_usec = 10000;
  112. if (setsockopt(s0, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(struct timeval)) < 0)
  113. {
  114. #ifdef SystemLogMessage
  115. PRINTF_LIB_FUNC("Set SO_RCVTIMEO NG");
  116. #endif
  117. }
  118. nbytes=40960;
  119. if (setsockopt(s0, SOL_SOCKET, SO_RCVBUF, &nbytes, sizeof(int)) < 0)
  120. {
  121. #ifdef SystemLogMessage
  122. PRINTF_LIB_FUNC("Set SO_RCVBUF NG");
  123. #endif
  124. }
  125. nbytes=40960;
  126. if (setsockopt(s0, SOL_SOCKET, SO_SNDBUF, &nbytes, sizeof(int)) < 0)
  127. {
  128. #ifdef SystemLogMessage
  129. PRINTF_LIB_FUNC("Set SO_SNDBUF NG");
  130. #endif
  131. }
  132. strcpy(ifr0.ifr_name, "can1" );
  133. ioctl(s0, SIOCGIFINDEX, &ifr0); /* ifr.ifr_ifindex gets filled with that device's index */
  134. addr0.can_family = AF_CAN;
  135. addr0.can_ifindex = ifr0.ifr_ifindex;
  136. bind(s0, (struct sockaddr *)&addr0, sizeof(addr0));
  137. return s0;
  138. }
  139. //================================================
  140. // Receive Cmd from canbus
  141. //================================================
  142. void ReceiveDataFromCanBus()
  143. {
  144. int nbytes;
  145. struct can_frame frame;
  146. int intCmd = 0;
  147. byte group, address;
  148. while(1)
  149. {
  150. memset(&frame, 0, sizeof(struct can_frame));
  151. nbytes = read(CanFd, &frame, sizeof(struct can_frame));
  152. if (nbytes > 0)
  153. {
  154. frame.can_id = frame.can_id & CAN_EFF_MASK;
  155. intCmd = frame.can_id & 0x00FF0000;
  156. intCmd |= INFYPWR_GROUP_SHIFT | intCmd;
  157. switch (intCmd)
  158. {
  159. case INFYPWR_GROUP_SHIFT | STATUS:
  160. {
  161. group = frame.data[2];
  162. address = frame.can_id & 0x000000FF;
  163. short temp = frame.data[4];
  164. int status = (frame.data[5] << 16) + (frame.data[6] << 8) + frame.data[7];
  165. return_status(group, address, temp, status);
  166. //PRINTF_LIB_FUNC("group = %d, address = %d, temp = %d \n", group, address, temp);
  167. }
  168. break;
  169. case INFYPWR_GROUP_SHIFT | MODULE_COUNT:
  170. {
  171. // 回傳模組數量
  172. group = frame.can_id & 0x000000FF;
  173. byte count = frame.data[2];
  174. return_module_count(group, count);
  175. //PRINTF_LIB_FUNC("group = %d, count = %d \n", group, count);
  176. }
  177. break;
  178. case INFYPWR_GROUP_SHIFT | MODULE_CAP:
  179. {
  180. // 回傳輸出能力 : 最大電壓、最小電壓、最大電流、額定功率
  181. address = frame.can_id & 0x000000FF;
  182. short maxVol = ((frame.data[0] << 8) + frame.data[1]) * 10;
  183. short minVol = ((frame.data[2] << 8) + frame.data[3]) * 10;
  184. short maxCur = (frame.data[4] << 8) + frame.data[5];
  185. short totalPow = ((frame.data[6] << 8) + frame.data[7]) / 10;
  186. return_available_cap(address, maxVol, minVol, maxCur, totalPow);
  187. // PRINTF_LIB_FUNC("address = %d, maxVol = %d, minVol = %d, maxCur = %d, totalPow = %d \n",
  188. // address, maxVol, minVol, maxCur, totalPow);
  189. }
  190. break;
  191. case INFYPWR_GROUP_SHIFT | MODULE_OUTPUT_VOL_CUR:
  192. {
  193. // 回傳當前輸出電壓電流
  194. address = frame.can_id & 0x000000FF;
  195. int outputVol = ((frame.data[0] << 24) + (frame.data[1] << 16) + (frame.data[2] << 8) + frame.data[3]) / 100;
  196. int outputCur = ((frame.data[4] << 24) + (frame.data[5] << 16) + (frame.data[6] << 8) + frame.data[7]) / 100;
  197. return_get_output(address, outputVol, outputCur);
  198. //PRINTF_LIB_FUNC("address = %d, outputVol = %d, outputCur = %d \n", address, outputVol, outputCur);
  199. }
  200. break;
  201. case INFYPWR_GROUP_SHIFT | MODULE_IAVAILABLE:
  202. case MODULE_IAVAILABLE:
  203. {
  204. // 回傳降載後的電流
  205. address = frame.can_id & 0x000000FF;
  206. unsigned short vextVol = ((frame.data[0] << 8) + frame.data[1]);
  207. unsigned short iAvailCur = ((frame.data[2] << 8) + frame.data[3]);
  208. return_iavail_info(address, iAvailCur, vextVol);
  209. //PRINTF_LIB_FUNC("address = %d, iAvailCur = %d \n", address, iAvailCur);
  210. }
  211. break;
  212. case INFYPWR_GROUP_SHIFT | MODULE_MIS_INFO:
  213. {
  214. address = frame.can_id & 0x000000FF;
  215. float FanSpeed;
  216. byte value[4];
  217. memcpy(value, frame.data + 4, sizeof(value));
  218. if (frame.data[0] == ((FAN_SPEED_CMD >> 8) & 0xFF) && frame.data[1] == (FAN_SPEED_CMD & 0xFF))
  219. {
  220. FanSpeed = IEEE_754_to_float(value);
  221. return_fanspeed_info(address, FanSpeed);
  222. //PRINTF_LIB_FUNC("address = %d, FanSpeed = %f \n", address, FanSpeed);
  223. }
  224. }
  225. break;
  226. case INFYPWR_GROUP_SHIFT | MODULE_VER:
  227. {
  228. // 回傳版號 : 無系統回覆功能
  229. address = frame.can_id & 0x000000FF;
  230. short dcSwVer = ((frame.data[0] << 8) + frame.data[1]);
  231. short pfcSwVer = ((frame.data[2] << 8) + frame.data[3]);
  232. short hwVer = ((frame.data[4] << 8) + frame.data[5]);
  233. return_fw_version(address, dcSwVer, pfcSwVer, hwVer);
  234. //PRINTF_LIB_FUNC("address = %d, DC %d, PFC %d, HW %d \n", address, dcSwVer, pfcSwVer, hwVer);
  235. }
  236. break;
  237. case INFYPWR_GROUP_SHIFT | MODULE_BARCODE:
  238. {
  239. // 回傳BarCode
  240. }
  241. break;
  242. case INFYPWR_GROUP_SHIFT | MODULE_INPUT:
  243. {
  244. // 回傳三向輸入電壓
  245. address = frame.can_id & 0x000000FF;
  246. short abVol = ((frame.data[0] << 8) + frame.data[1]) / 10;
  247. short bcVol = ((frame.data[2] << 8) + frame.data[3]) / 10;
  248. short caVol = ((frame.data[4] << 8) + frame.data[5]) / 10;
  249. return_input_vol(address, abVol, bcVol, caVol);
  250. //PRINTF_LIB_FUNC("address = %d, abVol = %d, bcVol = %d, caVol = %d \n", address, abVol, bcVol, caVol);
  251. }
  252. break;
  253. }
  254. }
  255. usleep(10000);
  256. }
  257. }
  258. //================================================
  259. // Private Function
  260. //================================================
  261. void SendCmdToPsu(int cmd, byte *data, byte dataLen)
  262. {
  263. struct can_frame frame;
  264. //設定 CANBSU 2.0B 長封包
  265. cmd = cmd | 0x80000000;
  266. frame.can_id = cmd;
  267. frame.can_dlc = dataLen;
  268. memcpy(frame.data, data, dataLen);
  269. write(CanFd, &frame, sizeof(struct can_frame));
  270. usleep(CMD_DELAY_TIME);
  271. }
  272. bool InitialCommunication()
  273. {
  274. CanFd = InitCanBus();
  275. if(CanFd < 0)
  276. {
  277. PRINTF_LIB_FUNC("Init can bus fail.\n");
  278. return false;
  279. }
  280. recFork = fork();
  281. if(recFork > 0)
  282. {
  283. ReceiveDataFromCanBus();
  284. }
  285. else if(recFork > 0)
  286. {
  287. PRINTF_LIB_FUNC("fork fail\n");
  288. }
  289. return true;
  290. }
  291. //================================================
  292. // API Function
  293. //================================================
  294. void SwitchPower(byte group, byte value)
  295. {
  296. byte data[8];
  297. uint cmd = INFYPWR_CMD | SWITCH_POWER;
  298. memset(data, 0x00, ARRAY_SIZE(data));
  299. // 1 : 關機
  300. // 0 : 開機
  301. data[0] = value;
  302. if (group == SYSTEM_CMD)
  303. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  304. else
  305. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  306. SendCmdToPsu(cmd, data, sizeof(data));
  307. }
  308. void SleepMode(byte group, byte value)
  309. {
  310. byte data[8];
  311. uint cmd = INFYPWR_CMD | SLEEP_MODE;
  312. memset(data, 0x00, ARRAY_SIZE(data));
  313. // 1 : 休眠
  314. // 0 : 起床
  315. data[0] = value;
  316. if (group == SYSTEM_CMD)
  317. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  318. else
  319. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  320. SendCmdToPsu(cmd, data, sizeof(data));
  321. }
  322. void FlashLed(byte group, byte value)
  323. {
  324. byte data[8];
  325. uint cmd = INFYPWR_CMD | FLASH_LED;
  326. memset(data, 0x00, ARRAY_SIZE(data));
  327. // 1 : 閃爍
  328. // 0 : 正常
  329. data[0] = value;
  330. if (group == SYSTEM_CMD)
  331. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  332. else
  333. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  334. SendCmdToPsu(cmd, data, sizeof(data));
  335. }
  336. void PresentOutputVol(byte group, int voltage, int current)
  337. {
  338. byte data[8];
  339. uint cmd = INFYPWR_CMD | PRESENT_OUT_VOL;
  340. int Vol = voltage * 100;
  341. int Cur = current * 100;
  342. memset(data, 0x00, ARRAY_SIZE(data));
  343. // 輸出電壓
  344. data[0] = (Vol >> 24) & 0xFF;
  345. data[1] = (Vol >> 16) & 0xFF;
  346. data[2] = (Vol >> 8) & 0xFF;
  347. data[3] = Vol & 0xFF;
  348. // 輸出電流
  349. data[4] = (Cur >> 24) & 0xFF;
  350. data[5] = (Cur >> 16) & 0xFF;
  351. data[6] = (Cur >> 8) & 0xFF;
  352. data[7] = Cur & 0xFF;
  353. if (group == SYSTEM_CMD)
  354. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  355. else
  356. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  357. SendCmdToPsu(cmd, data, sizeof(data));
  358. }
  359. void FanNoiseInfo(byte group, byte value)
  360. {
  361. byte data[8];
  362. uint cmd = INFYPWR_CMD | MIS_INFO;
  363. memset(data, 0x00, ARRAY_SIZE(data));
  364. // 風扇低噪音
  365. data[0] = 0x11;
  366. data[1] = 0x13;
  367. // 0xA0 power poriority mode
  368. // 0xA1 denoise mode
  369. // 0xA2 quiet mode
  370. data[7] = value;
  371. if (group == SYSTEM_CMD)
  372. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  373. else
  374. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  375. SendCmdToPsu(cmd, data, sizeof(data));
  376. }
  377. /**********************************************************************************/
  378. /*** ***/
  379. /*** Get ***/
  380. /*** ***/
  381. /**********************************************************************************/
  382. void GetStatus(byte group)
  383. {
  384. byte data[8];
  385. uint cmd = INFYPWR_CMD | STATUS;
  386. memset(data, 0x00, ARRAY_SIZE(data));
  387. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  388. SendCmdToPsu(cmd, data, sizeof(data));
  389. }
  390. void GetFanSpeed(byte group)
  391. {
  392. uint cmd;
  393. byte data[8];
  394. cmd = INFYPWR_CMD | MODULE_MIS_INFO;
  395. memset(data, 0x00, ARRAY_SIZE(data));
  396. data[0] = (FAN_SPEED_CMD >> 8) & 0xFF;
  397. data[1] = FAN_SPEED_CMD & 0xFF;
  398. if (group == (INFYPWR_BROADCAST >> 8))
  399. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  400. else
  401. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  402. SendCmdToPsu(cmd, data, sizeof(data));
  403. }
  404. void GetModuleCount(byte group)
  405. {
  406. byte data[8];
  407. uint cmd = INFYPWR_CMD | MODULE_COUNT;
  408. memset(data, 0x00, ARRAY_SIZE(data));
  409. if (group == SYSTEM_CMD)
  410. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  411. else
  412. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  413. SendCmdToPsu(cmd, data, sizeof(data));
  414. }
  415. void GetModuleVer(byte group)
  416. {
  417. // 無系統廣播功能
  418. byte data[8];
  419. uint cmd = INFYPWR_CMD | MODULE_VER;
  420. memset(data, 0x00, ARRAY_SIZE(data));
  421. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  422. //PRINTF_LIB_FUNC("GetModuleVer cmd = %x\n", cmd);
  423. SendCmdToPsu(cmd, data, sizeof(data));
  424. }
  425. void GetModuleCap(byte group)
  426. {
  427. byte data[8];
  428. uint cmd = INFYPWR_CMD | MODULE_CAP;
  429. memset(data, 0x00, ARRAY_SIZE(data));
  430. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  431. //PRINTF_LIB_FUNC("GetModuleCap cmd = %x\n", cmd);
  432. SendCmdToPsu(cmd, data, sizeof(data));
  433. }
  434. void GetModuleBarCode(byte group)
  435. {
  436. // 無系統廣播功能
  437. byte data[8];
  438. uint cmd = INFYPWR_CMD | MODULE_BARCODE;
  439. memset(data, 0x00, ARRAY_SIZE(data));
  440. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  441. SendCmdToPsu(cmd, data, sizeof(data));
  442. }
  443. void GetModuleInput(byte group)
  444. {
  445. // 無系統廣播功能
  446. byte data[8];
  447. uint cmd = INFYPWR_CMD | MODULE_INPUT;
  448. memset(data, 0x00, ARRAY_SIZE(data));
  449. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  450. SendCmdToPsu(cmd, data, sizeof(data));
  451. }
  452. void GetModuleIavailable(byte group)
  453. {
  454. byte data[8];
  455. uint cmd = INFYPWR_CMD | MODULE_IAVAILABLE;
  456. memset(data, 0x00, ARRAY_SIZE(data));
  457. if (group == SYSTEM_CMD)
  458. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  459. else
  460. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  461. SendCmdToPsu(cmd, data, sizeof(data));
  462. }
  463. void GetModuleOutput(byte group)
  464. {
  465. byte data[8];
  466. uint cmd = INFYPWR_CMD | MODULE_OUTPUT_VOL_CUR;
  467. memset(data, 0x00, ARRAY_SIZE(data));
  468. if (group == SYSTEM_CMD)
  469. cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
  470. else
  471. cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
  472. SendCmdToPsu(cmd, data, sizeof(data));
  473. }