123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583 |
- /*
- * Infypwr_PsuCommObj.c
- *
- * Created on: 2019年11月26日
- * Author: 7564
- */
- #include "Infypwr_PsuCommObj.h"
- #define DEBUG_LIB 1
- void PRINTF_LIB_FUNC(char *string, ...);
- float IEEE_754_to_float(const byte raw[4]);
- void IEEE_754_to_bytes(float target, byte *bytes2);
- //================================================
- // Private function
- //================================================
- void PRINTF_LIB_FUNC(char *string, ...)
- {
- if (DEBUG_LIB)
- {
- va_list args;
- char buffer[4096];
- va_start(args, string);
- vsnprintf(buffer, sizeof(buffer), string, args);
- va_end(args);
- printf("%s \n", buffer);
- }
- }
- float IEEE_754_to_float(const byte raw[4])
- {
- int sign = (raw[0] >> 7) ? -1 : 1;
- byte exponent = (raw[0] << 1) + (raw[1] >> 7) - 126;
- unsigned int fraction_bits = ((raw[1] & 0x7F) << 16) + (raw[2] << 8) + raw[3];
- float fraction = 0.5f;
- for (byte ii = 0; ii < 24; ++ii)
- fraction += ldexpf((fraction_bits >> (23 - ii)) & 1, -(ii + 1));
- float significand = sign * fraction;
- return ldexpf(significand, exponent);
- }
- void IEEE_754_to_bytes(float target, byte *bytes2)
- {
- int value2 = 0;
- number.f = target;
- int index = 31;
- value2 |= number.raw.sign << index;
- int k;
- for (k = 8 - 1; k >= 0; k--)
- {
- index--;
- if ((number.raw.exponent >> k) & 1)
- value2 |= 1 << index;
- }
- for (k = 23 - 1; k >= 0; k--)
- {
- index--;
- if ((number.raw.mantissa >> k) & 1)
- value2 |= 1 << index;
- }
- *(bytes2) = (value2 >> 24) & 0xFF;
- *(bytes2 + 1) = (value2 >> 16) & 0xFF;
- *(bytes2 + 2) = (value2 >> 8) & 0xFF;
- *(bytes2 + 3) = value2 & 0xFF;
- }
- //================================================
- // Callback function
- //================================================
- void RefreshStatus(void *func)
- {
- return_status = func;
- }
- void RefreshModuleCount(void *func)
- {
- return_module_count = func;
- }
- void RefreshAvailableCap(void *func)
- {
- return_available_cap = func;
- }
- void RefreshFwVersion(void *func)
- {
- return_fw_version = func;
- }
- void RefreshInputVol(void *func)
- {
- return_input_vol = func;
- }
- void RefreshGetOutput(void *func)
- {
- return_get_output = func;
- }
- void RefreshFanInfo(void *func)
- {
- return_fanspeed_info = func;
- }
- void RefreshIavailable(void *func)
- {
- return_iavail_info = func;
- }
- //================================================
- // CANBUS initialization
- //================================================
- int InitCanBus()
- {
- int s0,nbytes;
- struct timeval tv;
- struct ifreq ifr0;
- struct sockaddr_can addr0;
- system("/sbin/ip link set can1 down");
- system("/sbin/ip link set can1 type can bitrate 500000 restart-ms 100");
- system("/sbin/ip link set can1 up");
- s0 = socket(PF_CAN, SOCK_RAW, CAN_RAW);
- tv.tv_sec = 0;
- tv.tv_usec = 10000;
- if (setsockopt(s0, SOL_SOCKET, SO_RCVTIMEO, (char *)&tv, sizeof(struct timeval)) < 0)
- {
- #ifdef SystemLogMessage
- PRINTF_LIB_FUNC("Set SO_RCVTIMEO NG");
- #endif
- }
- nbytes=40960;
- if (setsockopt(s0, SOL_SOCKET, SO_RCVBUF, &nbytes, sizeof(int)) < 0)
- {
- #ifdef SystemLogMessage
- PRINTF_LIB_FUNC("Set SO_RCVBUF NG");
- #endif
- }
- nbytes=40960;
- if (setsockopt(s0, SOL_SOCKET, SO_SNDBUF, &nbytes, sizeof(int)) < 0)
- {
- #ifdef SystemLogMessage
- PRINTF_LIB_FUNC("Set SO_SNDBUF NG");
- #endif
- }
- strcpy(ifr0.ifr_name, "can1" );
- ioctl(s0, SIOCGIFINDEX, &ifr0); /* ifr.ifr_ifindex gets filled with that device's index */
- addr0.can_family = AF_CAN;
- addr0.can_ifindex = ifr0.ifr_ifindex;
- bind(s0, (struct sockaddr *)&addr0, sizeof(addr0));
- return s0;
- }
- //================================================
- // Receive Cmd from canbus
- //================================================
- void ReceiveDataFromCanBus()
- {
- int nbytes;
- struct can_frame frame;
- int intCmd = 0;
- byte group, address;
- while(1)
- {
- memset(&frame, 0, sizeof(struct can_frame));
- nbytes = read(CanFd, &frame, sizeof(struct can_frame));
- if (nbytes > 0)
- {
- frame.can_id = frame.can_id & CAN_EFF_MASK;
- intCmd = frame.can_id & 0x00FF0000;
- intCmd |= INFYPWR_GROUP_SHIFT | intCmd;
- switch (intCmd)
- {
- case INFYPWR_GROUP_SHIFT | STATUS:
- {
- group = frame.data[2];
- address = frame.can_id & 0x000000FF;
- short temp = frame.data[4];
- int status = (frame.data[5] << 16) + (frame.data[6] << 8) + frame.data[7];
- return_status(group, address, temp, status);
- //PRINTF_LIB_FUNC("group = %d, address = %d, temp = %d \n", group, address, temp);
- }
- break;
- case INFYPWR_GROUP_SHIFT | MODULE_COUNT:
- {
- // 回傳模組數量
- group = frame.can_id & 0x000000FF;
- byte count = frame.data[2];
- return_module_count(group, count);
- //PRINTF_LIB_FUNC("group = %d, count = %d \n", group, count);
- }
- break;
- case INFYPWR_GROUP_SHIFT | MODULE_CAP:
- {
- // 回傳輸出能力 : 最大電壓、最小電壓、最大電流、額定功率
- address = frame.can_id & 0x000000FF;
- short maxVol = ((frame.data[0] << 8) + frame.data[1]) * 10;
- short minVol = ((frame.data[2] << 8) + frame.data[3]) * 10;
- short maxCur = (frame.data[4] << 8) + frame.data[5];
- short totalPow = ((frame.data[6] << 8) + frame.data[7]) / 10;
- return_available_cap(address, maxVol, minVol, maxCur, totalPow);
- // PRINTF_LIB_FUNC("address = %d, maxVol = %d, minVol = %d, maxCur = %d, totalPow = %d \n",
- // address, maxVol, minVol, maxCur, totalPow);
- }
- break;
- case INFYPWR_GROUP_SHIFT | MODULE_OUTPUT_VOL_CUR:
- {
- // 回傳當前輸出電壓電流
- address = frame.can_id & 0x000000FF;
- int outputVol = ((frame.data[0] << 24) + (frame.data[1] << 16) + (frame.data[2] << 8) + frame.data[3]) / 100;
- int outputCur = ((frame.data[4] << 24) + (frame.data[5] << 16) + (frame.data[6] << 8) + frame.data[7]) / 100;
- return_get_output(address, outputVol, outputCur);
- //PRINTF_LIB_FUNC("address = %d, outputVol = %d, outputCur = %d \n", address, outputVol, outputCur);
- }
- break;
- case INFYPWR_GROUP_SHIFT | MODULE_IAVAILABLE:
- {
- // 回傳降載後的電流
- address = frame.can_id & 0x000000FF;
- unsigned short vextVol = ((frame.data[0] << 8) + frame.data[1]);
- unsigned short iAvailCur = ((frame.data[2] << 8) + frame.data[3]);
- return_iavail_info(address, iAvailCur, vextVol);
- //PRINTF_LIB_FUNC("address = %d, iAvailCur = %d \n", address, iAvailCur);
- }
- break;
- case INFYPWR_GROUP_SHIFT | MODULE_MIS_INFO:
- {
- address = frame.can_id & 0x000000FF;
- float FanSpeed;
- byte value[4];
- memcpy(value, frame.data + 4, sizeof(value));
- if (frame.data[0] == ((FAN_SPEED_CMD >> 8) & 0xFF) && frame.data[1] == (FAN_SPEED_CMD & 0xFF))
- {
- FanSpeed = IEEE_754_to_float(value);
- return_fanspeed_info(address, FanSpeed);
- //PRINTF_LIB_FUNC("address = %d, FanSpeed = %f \n", address, FanSpeed);
- }
- }
- break;
- case INFYPWR_GROUP_SHIFT | MODULE_VER:
- {
- // 回傳版號 : 無系統回覆功能
- address = frame.can_id & 0x000000FF;
- short dcSwVer = ((frame.data[0] << 8) + frame.data[1]);
- short pfcSwVer = ((frame.data[2] << 8) + frame.data[3]);
- short hwVer = ((frame.data[4] << 8) + frame.data[5]);
- return_fw_version(address, dcSwVer, pfcSwVer, hwVer);
- //PRINTF_LIB_FUNC("address = %d, DC %d, PFC %d, HW %d \n", address, dcSwVer, pfcSwVer, hwVer);
- }
- break;
- case INFYPWR_GROUP_SHIFT | MODULE_BARCODE:
- {
- // 回傳BarCode
- }
- break;
- case INFYPWR_GROUP_SHIFT | MODULE_INPUT:
- {
- // 回傳三向輸入電壓
- address = frame.can_id & 0x000000FF;
- short abVol = ((frame.data[0] << 8) + frame.data[1]) / 10;
- short bcVol = ((frame.data[2] << 8) + frame.data[3]) / 10;
- short caVol = ((frame.data[4] << 8) + frame.data[5]) / 10;
- return_input_vol(address, abVol, bcVol, caVol);
- //PRINTF_LIB_FUNC("address = %d, abVol = %d, bcVol = %d, caVol = %d \n", address, abVol, bcVol, caVol);
- }
- break;
- }
- }
- usleep(10000);
- }
- }
- //================================================
- // Private Function
- //================================================
- void SendCmdToPsu(int cmd, byte *data, byte dataLen)
- {
- struct can_frame frame;
- //設定 CANBSU 2.0B 長封包
- cmd = cmd | 0x80000000;
- frame.can_id = cmd;
- frame.can_dlc = dataLen;
- memcpy(frame.data, data, dataLen);
- write(CanFd, &frame, sizeof(struct can_frame));
- usleep(CMD_DELAY_TIME);
- }
- bool InitialCommunication()
- {
- CanFd = InitCanBus();
- if(CanFd < 0)
- {
- PRINTF_LIB_FUNC("Init can bus fail.\n");
- return false;
- }
- recFork = fork();
- if(recFork > 0)
- {
- ReceiveDataFromCanBus();
- }
- else if(recFork > 0)
- {
- PRINTF_LIB_FUNC("fork fail\n");
- }
- return true;
- }
- //================================================
- // API Function
- //================================================
- void SwitchPower(byte group, byte value)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | SWITCH_POWER;
- memset(data, 0x00, ARRAY_SIZE(data));
- // 1 : 關機
- // 0 : 開機
- data[0] = value;
- if (group == SYSTEM_CMD)
- cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
- else
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void SleepMode(byte group, byte value)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | SLEEP_MODE;
- memset(data, 0x00, ARRAY_SIZE(data));
- // 1 : 休眠
- // 0 : 起床
- data[0] = value;
- if (group == SYSTEM_CMD)
- cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
- else
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void FlashLed(byte group, byte value)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | FLASH_LED;
- memset(data, 0x00, ARRAY_SIZE(data));
- // 1 : 閃爍
- // 0 : 正常
- data[0] = value;
- if (group == SYSTEM_CMD)
- cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
- else
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void PresentOutputVol(byte group, int voltage, int current)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | PRESENT_OUT_VOL;
- int Vol = voltage * 100;
- int Cur = current * 100;
- memset(data, 0x00, ARRAY_SIZE(data));
- // 輸出電壓
- data[0] = (Vol >> 24) & 0xFF;
- data[1] = (Vol >> 16) & 0xFF;
- data[2] = (Vol >> 8) & 0xFF;
- data[3] = Vol & 0xFF;
- // 輸出電流
- data[4] = (Cur >> 24) & 0xFF;
- data[5] = (Cur >> 16) & 0xFF;
- data[6] = (Cur >> 8) & 0xFF;
- data[7] = Cur & 0xFF;
- if (group == SYSTEM_CMD)
- cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
- else
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void FanNoiseInfo(byte group, byte value)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | MIS_INFO;
- memset(data, 0x00, ARRAY_SIZE(data));
- // 風扇低噪音
- data[0] = 0x11;
- data[1] = 0x13;
- // 0xA0 power poriority mode
- // 0xA1 denoise mode
- // 0xA2 quiet mode
- data[7] = value;
- if (group == SYSTEM_CMD)
- cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
- else
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- /**********************************************************************************/
- /*** ***/
- /*** Get ***/
- /*** ***/
- /**********************************************************************************/
- void GetStatus(byte group)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | STATUS;
- memset(data, 0x00, ARRAY_SIZE(data));
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void GetFanSpeed(byte group)
- {
- uint cmd;
- byte data[8];
- cmd = INFYPWR_CMD | MODULE_MIS_INFO;
- memset(data, 0x00, ARRAY_SIZE(data));
- data[0] = (FAN_SPEED_CMD >> 8) & 0xFF;
- data[1] = FAN_SPEED_CMD & 0xFF;
- if (group == (INFYPWR_BROADCAST >> 8))
- cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
- else
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void GetModuleCount(byte group)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | MODULE_COUNT;
- memset(data, 0x00, ARRAY_SIZE(data));
- if (group == SYSTEM_CMD)
- cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
- else
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void GetModuleVer(byte group)
- {
- // 無系統廣播功能
- byte data[8];
- uint cmd = INFYPWR_CMD | MODULE_VER;
- memset(data, 0x00, ARRAY_SIZE(data));
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- //PRINTF_LIB_FUNC("GetModuleVer cmd = %x\n", cmd);
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void GetModuleCap(byte group)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | MODULE_CAP;
- memset(data, 0x00, ARRAY_SIZE(data));
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- //PRINTF_LIB_FUNC("GetModuleCap cmd = %x\n", cmd);
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void GetModuleBarCode(byte group)
- {
- // 無系統廣播功能
- byte data[8];
- uint cmd = INFYPWR_CMD | MODULE_BARCODE;
- memset(data, 0x00, ARRAY_SIZE(data));
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void GetModuleInput(byte group)
- {
- // 無系統廣播功能
- byte data[8];
- uint cmd = INFYPWR_CMD | MODULE_INPUT;
- memset(data, 0x00, ARRAY_SIZE(data));
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void GetModuleIavailable(byte group)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | MODULE_IAVAILABLE;
- memset(data, 0x00, ARRAY_SIZE(data));
- if (group == SYSTEM_CMD)
- cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
- else
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
- void GetModuleOutput(byte group)
- {
- byte data[8];
- uint cmd = INFYPWR_CMD | MODULE_OUTPUT_VOL_CUR;
- memset(data, 0x00, ARRAY_SIZE(data));
- if (group == SYSTEM_CMD)
- cmd |= INFYPWR_BROADCAST | INFYPWR_DEFAULT;
- else
- cmd |= INFYPWR_GROUP_SHIFT | (group << 8) | INFYPWR_DEFAULT;
- SendCmdToPsu(cmd, data, sizeof(data));
- }
|