IPMBIfc.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. /*
  2. * Brief: Implement IPMB send and receive function in this file, include primary IPMB and secondary IPMB.
  3. * Author: Jimbo_Zhang@outlook.com
  4. * Date: 2019-9-9
  5. */
  6. /* Standard includes. */
  7. #include <stdio.h>
  8. #include <stdint.h>
  9. #include <string.h>
  10. #include <errno.h>
  11. #include <pthread.h>
  12. #include <fcntl.h>
  13. #include <sys/socket.h>
  14. #include "com_IPMI_LANIfc.h"
  15. #include "com_IPMI_RMCP.h"
  16. #include "RMCP.h"
  17. #include "LANIfc.h"
  18. #include "com_IPMIDefs.h"
  19. #include "RMCP.h"
  20. #include "MD.h"
  21. #include <netdb.h> /* getaddrinfo(3) et al. */
  22. #include <netinet/in.h> /* sockaddr_in & sockaddr_in6 definition. */
  23. #include <net/if.h>
  24. #include <sys/prctl.h>
  25. #include "main.h"
  26. #include "Session.h"
  27. #include "com_IPMI_SDR.h"
  28. #include "com_IPMI_Storage.h"
  29. #include "SDR.h"
  30. //static int sendIPMBPkt(uint32_t ipmb_bus, uint8_t* i2c_send_buf, uint8_t size);
  31. static uint8_t ValidateIPMBChkSums ( uint8_t* Data, uint8_t Size);
  32. static void ProcessIPMBReq ( MsgPkt_T* pReq);
  33. uint8_t IsResponseMatch (MsgPkt_T* pReq, MsgPkt_T* pRes);
  34. static void* RecvIPMBPkt (void *pArg);
  35. int gFd_PrimaryIpmbIfcQ, gFd_PrimaryIpmbResQ;
  36. int gFd_SecondaryIpmbIfcQ, gFd_SecondaryIpmbResQ;
  37. int gFd_Primary, gFd_Secondary;
  38. /*!
  39. \brief IPMB Task. Generating primary_IPMB task and secondary_IPMB task in this task.
  40. \param: 0: primary ipmb, 1: secondary ipmb
  41. \retval none
  42. */
  43. void *IPMBIfcTask(void *Param)
  44. {
  45. // int i;
  46. MsgPkt_T RcvMsgPkt;
  47. uint8_t dev_name[10] = {0};
  48. uint8_t IPMB_IFC_Q[32] = {0};
  49. uint8_t IPMB_RES_Q[32] = {0};
  50. uint8_t OwnerAddr;
  51. int fd_IpmbDev;
  52. int fd_IpmbIfcQ, fd_IpmbResQ;
  53. uint8_t ipmbSelect = *(uint8_t*)Param;
  54. int RetVal;
  55. //printf("ipmbSelect: %d\n", ipmbSelect);
  56. //Primary IPMB
  57. if((ipmbSelect == 0) && g_BMCInfo.IpmiConfig.PrimaryIPMBSupport)
  58. {
  59. printf("Primary IPMBIfcTask start...\n");
  60. strcpy(IPMB_IFC_Q, PRIMARY_IPMB_IFC_Q);
  61. strcpy(IPMB_RES_Q, PRIMARY_IPMB_RES_Q);
  62. sprintf(dev_name, "/dev/i2c%d", g_BMCInfo.IpmiConfig.PrimaryIPMBBus);
  63. OwnerAddr = g_BMCInfo.IpmiConfig.PrimaryIPMBAddr;
  64. }
  65. //Secondary IPMB
  66. else if((ipmbSelect == 1) && g_BMCInfo.IpmiConfig.SecondaryIPMBSupport)
  67. {
  68. printf("Secondary IPMBIfcTask start...\n");
  69. strcpy(IPMB_IFC_Q, SECONDARY_IPMB_IFC_Q);
  70. strcpy(IPMB_RES_Q, SECONDARY_IPMB_RES_Q);
  71. sprintf(dev_name, "/dev/i2c%d", g_BMCInfo.IpmiConfig.SecondaryIPMBBus);
  72. OwnerAddr = g_BMCInfo.IpmiConfig.SecondaryIPMBAddr;
  73. }
  74. else
  75. {
  76. printf("Invalid param! %d\n", ipmbSelect);
  77. return (void *)-1;
  78. }
  79. //create IPMB_IFC_Q
  80. if(-1 != access(IPMB_IFC_Q, F_OK))
  81. {
  82. remove(IPMB_IFC_Q);
  83. }
  84. if(0 != mkfifo (IPMB_IFC_Q, 0777))
  85. {
  86. printf("%s: Create %s fifo failed! %s\n", __FUNCTION__, IPMB_IFC_Q, strerror(errno));
  87. return (void*)-1;
  88. }
  89. fd_IpmbIfcQ = open (IPMB_IFC_Q, O_RDWR);
  90. if(-1 == fd_IpmbIfcQ)
  91. {
  92. printf("%s: Open %s fifo failed! %s\n", __FUNCTION__, IPMB_IFC_Q, strerror(errno));
  93. return (void*)-1;
  94. }
  95. //create IPMB_RES_Q
  96. if(-1 != access(IPMB_RES_Q, F_OK))
  97. {
  98. remove(IPMB_RES_Q);
  99. }
  100. if(0 != mkfifo (IPMB_RES_Q, 0777))
  101. {
  102. printf("%s: Create %s fifo failed! %s\n", __FUNCTION__, IPMB_RES_Q, strerror(errno));
  103. return (void*)-1;
  104. }
  105. fd_IpmbResQ = open (IPMB_RES_Q, O_RDWR);
  106. if(-1 == fd_IpmbResQ)
  107. {
  108. printf("%s: Open %s fifo failed! %s\n", __FUNCTION__, IPMB_RES_Q, strerror(errno));
  109. return (void*)-1;
  110. }
  111. fd_IpmbDev = open (dev_name, O_RDWR);
  112. if(-1 == fd_IpmbDev)
  113. {
  114. printf("%s: Open IPMB failed! %s\n", __FUNCTION__, strerror(errno));
  115. return (void*)-1;
  116. }
  117. //set IPMB address
  118. if(0 != stm32_i2c_set_addr(fd_IpmbDev, OwnerAddr))
  119. printf("Set ipmb address fail.\n");
  120. if(ipmbSelect == 0)
  121. {
  122. gFd_PrimaryIpmbIfcQ = fd_IpmbIfcQ;
  123. gFd_PrimaryIpmbResQ = fd_IpmbResQ;
  124. gFd_Primary = fd_IpmbDev;
  125. }
  126. else if(ipmbSelect == 1)
  127. {
  128. gFd_SecondaryIpmbIfcQ = fd_IpmbIfcQ;
  129. gFd_SecondaryIpmbResQ = fd_IpmbResQ;
  130. gFd_Secondary = fd_IpmbDev;
  131. }
  132. /*Create a thread to recv IPMB Pkt */
  133. gThreadIndex++;
  134. if(0 != pthread_create(&gThreadIDs[gThreadIndex],NULL,RecvIPMBPkt,Param))
  135. {
  136. printf("%s: Create RecvIPMBPkt thread failed! %s\n", __FUNCTION__, strerror(errno));
  137. return (void *)-1;
  138. }
  139. while(1)
  140. {
  141. while(GetMsg(fd_IpmbIfcQ, &RcvMsgPkt, WAIT_INFINITE) != 0);
  142. switch(RcvMsgPkt.Param)
  143. {
  144. case PARAM_REQUEST:
  145. ProcessIPMBReq (&RcvMsgPkt);
  146. break;
  147. case PARAM_BRIDGE:
  148. /* Send the response */
  149. if(RcvMsgPkt.Channel == PRIMARY_IPMB_CHANNEL)
  150. {
  151. RetVal = stm32_i2c_master_write(gFd_Primary, RcvMsgPkt.Data[0], &RcvMsgPkt.Data[1], RcvMsgPkt.Size);
  152. }
  153. else if(RcvMsgPkt.Channel == SECONDARY_IPMB_CHANNEL)
  154. {
  155. RetVal = stm32_i2c_master_write(gFd_Secondary, RcvMsgPkt.Data[0], &RcvMsgPkt.Data[1], RcvMsgPkt.Size);
  156. }
  157. else
  158. {
  159. printf("IPMBIfc.c: IPMB channel error. %#x\r\n", RcvMsgPkt.Channel);
  160. }
  161. if (RetVal < 0)
  162. {
  163. printf ("IPMBIfc.c : Unable to send a IPMI Bridge Message\n");
  164. RespondSendMessage (&RcvMsgPkt, STATUS_FAIL);
  165. }
  166. else
  167. {
  168. RespondSendMessage (&RcvMsgPkt, STATUS_OK);
  169. }
  170. printf("---> IPMBIfcTask/BRIDGING_REQUEST\n");
  171. break;
  172. default:
  173. printf("Unknow message param %#x\r\n", RcvMsgPkt.Param);
  174. break;
  175. }
  176. }
  177. }
  178. /**
  179. * @brief Process the request posted by IPMB driver.
  180. * @param pReq - Request message pointer.
  181. **/
  182. static void
  183. ProcessIPMBReq ( MsgPkt_T* pReq)
  184. {
  185. MsgPkt_T ResPkt;
  186. uint8_t ReqLen;
  187. IPMIMsgHdr_T* pIPMIMsgReq = (IPMIMsgHdr_T*) pReq->Data;
  188. int RetVal = 0;
  189. ReqLen = pReq->Size;
  190. /* Validate Checksums */
  191. if (TRUE != ValidateIPMBChkSums ((uint8_t*)pIPMIMsgReq, ReqLen))
  192. {
  193. printf ("IPMBIfc.c : IPMB Checksum validation failed\r\n");
  194. return;
  195. }
  196. pReq->Param = PARAM_IFC;
  197. pReq->Cmd = pIPMIMsgReq->Cmd;
  198. pReq->NetFnLUN = pIPMIMsgReq->NetFnLUN;
  199. if(pReq->Channel == PRIMARY_IPMB_CHANNEL)
  200. pReq->SrcQ = gFd_PrimaryIpmbResQ;
  201. else
  202. pReq->SrcQ = gFd_SecondaryIpmbResQ;
  203. /* Post the message to message handler Task */
  204. PostMsg (gFd_MsgHndlrIfc, pReq);
  205. if(pReq->ResTimeOut < 1) pReq->ResTimeOut = 10;
  206. do
  207. {
  208. if(pReq->Channel == PRIMARY_IPMB_CHANNEL)
  209. RetVal = GetMsg (gFd_PrimaryIpmbResQ, &ResPkt, pReq->ResTimeOut);
  210. else
  211. RetVal = GetMsg (gFd_SecondaryIpmbResQ, &ResPkt, pReq->ResTimeOut);
  212. }while(!RetVal && !IsResponseMatch(pReq, &ResPkt) );
  213. if (0 != RetVal)
  214. {
  215. printf("Warning: Process IPMB request fail, return timeout.\r\n");
  216. ResPkt.Param = PARAM_NORMAL_RESPONSE;
  217. ResPkt.Size = sizeof(IPMIMsgHdr_T)+2;
  218. SwapIPMIMsgHdr(((IPMIMsgHdr_T*)pReq->Data), ((IPMIMsgHdr_T*)ResPkt.Data));
  219. ResPkt.Data[sizeof(IPMIMsgHdr_T)] = CC_TIMEOUT;
  220. ResPkt.Data[sizeof(IPMIMsgHdr_T)+1] = CalculateCheckSum2(ResPkt.Data, sizeof(IPMIMsgHdr_T)+1);
  221. }
  222. /* If its not normal IPMI Response just return */
  223. if (PARAM_NO_RESPONSE == ResPkt.Param)
  224. {
  225. printf ("IPMBIfc.c : IPMB request packet dropped, not processed\n");
  226. return;
  227. }
  228. /* Send the response */
  229. if(pReq->Channel == PRIMARY_IPMB_CHANNEL)
  230. {
  231. RetVal = stm32_i2c_master_write(gFd_Primary, ResPkt.Data[0], &ResPkt.Data[1], ResPkt.Size);
  232. }
  233. else if(pReq->Channel == SECONDARY_IPMB_CHANNEL)
  234. {
  235. RetVal = stm32_i2c_master_write(gFd_Secondary, ResPkt.Data[0], &ResPkt.Data[1], ResPkt.Size);
  236. }
  237. else
  238. {
  239. printf("IPMBIfc.c: IPMB channel error. %#x\r\n", pReq->Channel);
  240. }
  241. if (0 != RetVal)
  242. {
  243. printf ("IPMBIfc.c : Unable to send a IPMI Response\n");
  244. }
  245. return;
  246. }
  247. /**
  248. * @brief Validate IPMB Checksums.
  249. * @param Data - Buffer to be validated.
  250. * @param Size - Size of the buffer.
  251. * @return TRUE if valid, FALSE if error.
  252. **/
  253. static uint8_t ValidateIPMBChkSums ( uint8_t* Data, uint8_t Size)
  254. {
  255. uint8_t i;
  256. uint8_t chksum;
  257. /* Check IPMB message min size */
  258. if (Size < MIN_IPMB_MSG_LENGTH)
  259. {
  260. printf("IPMBIfc.c: Invalid IPMB Message Length\r\n");
  261. return FALSE;
  262. }
  263. /* Validate the checksum1 */
  264. chksum = 0;
  265. for (i = 0; i < 3; i++)
  266. {
  267. chksum += *(Data + i);
  268. }
  269. if (chksum != 0)
  270. {
  271. printf ("IPMBIfc.c: Invalid checksum 1, size = %d\n",Size);
  272. for(i = 0; i < Size;i++)
  273. printf("%x ",Data[i]);
  274. printf("\n");
  275. return FALSE;
  276. }
  277. /* Validate the checksum2 */
  278. chksum = 0;
  279. for (i = 3; i < Size; i++)
  280. {
  281. chksum += *(Data + i);
  282. }
  283. if (chksum != 0)
  284. {
  285. printf("IPMBIfc.c: Invalid checksum 2, size = %d\n",Size);
  286. for(i = 0; i < Size;i++)
  287. printf("%x ",Data[i]);
  288. printf("\n");
  289. return FALSE;
  290. }
  291. return TRUE;
  292. }
  293. /*-----------------------------------------------------
  294. * @fn IsResponseMatch
  295. * @brief Checks if the Response Message corresponds to
  296. * the Request Message
  297. *
  298. * @param pReq : IPMI Request Message Packet
  299. * @param pRes : IPMI Response Message Packet
  300. *
  301. * @return 1 if match
  302. * 0 if mismatch
  303. *----------------------------------------------------*/
  304. uint8_t IsResponseMatch (MsgPkt_T* pReq, MsgPkt_T* pRes)
  305. {
  306. if( ( ((((IPMIMsgHdr_T*)pRes->Data)->RqSeqLUN) >> 2) != ((((IPMIMsgHdr_T*)pReq->Data)->RqSeqLUN) >> 2) ) &&
  307. ( (pReq->NetFnLUN & 0x4) && (PARAM_NO_RESPONSE != pRes->Param) ) )
  308. {
  309. return FALSE;
  310. }
  311. return TRUE;
  312. }
  313. /**
  314. * @fn RecvIPMBPkt
  315. * @brief This function receives IPMB request packet and post the message to
  316. * IPMB interface queue
  317. **/
  318. static void* RecvIPMBPkt (void *Param)
  319. {
  320. MsgPkt_T IPMBReqPkt;
  321. int IPMBSlaveFd;
  322. int retval;
  323. uint8_t ipmbSelect = *(uint8_t*)Param;
  324. prctl(PR_SET_NAME, __FUNCTION__, 0, 0, 0);
  325. int curThreadIndex = 0;
  326. memset(&IPMBReqPkt,0,sizeof(MsgPkt_T));
  327. if(0 == ipmbSelect)
  328. {
  329. printf ("Primary IPMB Receiver Started...\n");
  330. IPMBReqPkt.Data [0] = g_BMCInfo.IpmiConfig.PrimaryIPMBAddr;
  331. IPMBReqPkt.Channel = PRIMARY_IPMB_CHANNEL;
  332. IPMBSlaveFd = gFd_Primary;
  333. }
  334. else if(1 == ipmbSelect)
  335. {
  336. printf ("Secondary IPMB Receiver Started...\n");
  337. IPMBReqPkt.Data [0] = g_BMCInfo.IpmiConfig.SecondaryIPMBAddr;
  338. IPMBReqPkt.Channel = SECONDARY_IPMB_CHANNEL;
  339. IPMBSlaveFd = gFd_Secondary;
  340. }
  341. /* Loop forever */
  342. while (1)
  343. {
  344. retval = stm32_i2c_slave_recv(IPMBSlaveFd, &IPMBReqPkt.Data[1]);
  345. if( retval > 5)
  346. {
  347. IPMBReqPkt.Param = PARAM_REQUEST;
  348. IPMBReqPkt.Size = retval + 1; /* +1 to include BMC Slave address */
  349. /* Post the IPMB Request message to IPMBIfc Queue */
  350. if(ipmbSelect == 0)
  351. {
  352. if (0 != PostMsg (gFd_PrimaryIpmbIfcQ, &IPMBReqPkt))
  353. {
  354. printf ("IPMBIfc.c : Error posting message to IPMBIfc_Q\n");
  355. }
  356. }
  357. else
  358. {
  359. if (0 != PostMsg (gFd_SecondaryIpmbIfcQ, &IPMBReqPkt))
  360. {
  361. printf ("IPMBIfc.c : Error posting message to IPMBIfc_Q\n");
  362. }
  363. }
  364. }
  365. int cnt;
  366. if(0 == ipmbSelect)
  367. printf("---> Primary IPMB Recv: ");
  368. else
  369. printf("---> Secondary IPMB Recv: ");
  370. for(cnt=0;cnt < IPMBReqPkt.Size; cnt++)
  371. printf("%#x ", IPMBReqPkt.Data[cnt]);
  372. printf("\n");
  373. }
  374. return (void*)-1;
  375. }