IPMBIfc.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  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. //printf("ipmbSelect: %d\n", ipmbSelect);
  55. //Primary IPMB
  56. if((ipmbSelect == 0) && g_BMCInfo.IpmiConfig.PrimaryIPMBSupport)
  57. {
  58. printf("Primary IPMBIfcTask start...\n");
  59. strcpy(IPMB_IFC_Q, PRIMARY_IPMB_IFC_Q);
  60. strcpy(IPMB_RES_Q, PRIMARY_IPMB_RES_Q);
  61. sprintf(dev_name, "/dev/i2c%d", g_BMCInfo.IpmiConfig.PrimaryIPMBBus);
  62. OwnerAddr = g_BMCInfo.IpmiConfig.PrimaryIPMBAddr;
  63. }
  64. //Secondary IPMB
  65. else if((ipmbSelect == 1) && g_BMCInfo.IpmiConfig.SecondaryIPMBSupport)
  66. {
  67. printf("Secondary IPMBIfcTask start...\n");
  68. strcpy(IPMB_IFC_Q, SECONDARY_IPMB_IFC_Q);
  69. strcpy(IPMB_RES_Q, SECONDARY_IPMB_RES_Q);
  70. sprintf(dev_name, "/dev/i2c%d", g_BMCInfo.IpmiConfig.SecondaryIPMBBus);
  71. OwnerAddr = g_BMCInfo.IpmiConfig.SecondaryIPMBAddr;
  72. }
  73. else
  74. {
  75. printf("Invalid param! %d\n", ipmbSelect);
  76. return (void *)-1;
  77. }
  78. //create IPMB_IFC_Q
  79. if(-1 != access(IPMB_IFC_Q, F_OK))
  80. {
  81. remove(IPMB_IFC_Q);
  82. }
  83. if(0 != mkfifo (IPMB_IFC_Q, 0777))
  84. {
  85. printf("%s: Create %s fifo failed! %s\n", __FUNCTION__, IPMB_IFC_Q, strerror(errno));
  86. return (void*)-1;
  87. }
  88. fd_IpmbIfcQ = open (IPMB_IFC_Q, O_RDWR);
  89. if(-1 == fd_IpmbIfcQ)
  90. {
  91. printf("%s: Open %s fifo failed! %s\n", __FUNCTION__, IPMB_IFC_Q, strerror(errno));
  92. return (void*)-1;
  93. }
  94. //create IPMB_RES_Q
  95. if(-1 != access(IPMB_RES_Q, F_OK))
  96. {
  97. remove(IPMB_RES_Q);
  98. }
  99. if(0 != mkfifo (IPMB_RES_Q, 0777))
  100. {
  101. printf("%s: Create %s fifo failed! %s\n", __FUNCTION__, IPMB_RES_Q, strerror(errno));
  102. return (void*)-1;
  103. }
  104. fd_IpmbResQ = open (IPMB_RES_Q, O_RDWR);
  105. if(-1 == fd_IpmbResQ)
  106. {
  107. printf("%s: Open %s fifo failed! %s\n", __FUNCTION__, IPMB_RES_Q, strerror(errno));
  108. return (void*)-1;
  109. }
  110. fd_IpmbDev = open (dev_name, O_RDWR);
  111. if(-1 == fd_IpmbDev)
  112. {
  113. printf("%s: Open IPMB failed! %s\n", __FUNCTION__, strerror(errno));
  114. return (void*)-1;
  115. }
  116. //set IPMB address
  117. if(0 != stm32_i2c_set_addr(fd_IpmbDev, OwnerAddr))
  118. printf("Set ipmb address fail.\n");
  119. if(ipmbSelect == 0)
  120. {
  121. gFd_PrimaryIpmbIfcQ = fd_IpmbIfcQ;
  122. gFd_PrimaryIpmbResQ = fd_IpmbResQ;
  123. gFd_Primary = fd_IpmbDev;
  124. }
  125. else if(ipmbSelect == 1)
  126. {
  127. gFd_SecondaryIpmbIfcQ = fd_IpmbIfcQ;
  128. gFd_SecondaryIpmbResQ = fd_IpmbResQ;
  129. gFd_Secondary = fd_IpmbDev;
  130. }
  131. /*Create a thread to recv IPMB Pkt */
  132. gThreadIndex++;
  133. if(0 != pthread_create(&gThreadIDs[gThreadIndex],NULL,RecvIPMBPkt,Param))
  134. {
  135. printf("%s: Create RecvIPMBPkt thread failed! %s\n", __FUNCTION__, strerror(errno));
  136. return (void *)-1;
  137. }
  138. while(1)
  139. {
  140. while(GetMsg(fd_IpmbIfcQ, &RcvMsgPkt, WAIT_INFINITE) != 0);
  141. switch(RcvMsgPkt.Param)
  142. {
  143. case PARAM_REQUEST:
  144. ProcessIPMBReq (&RcvMsgPkt);
  145. break;
  146. case PARAM_BRIDGE:
  147. break;
  148. default:
  149. printf("Unknow message param %#x\r\n", RcvMsgPkt.Param);
  150. break;
  151. }
  152. }
  153. }
  154. /**
  155. * @brief Process the request posted by IPMB driver.
  156. * @param pReq - Request message pointer.
  157. **/
  158. static void
  159. ProcessIPMBReq ( MsgPkt_T* pReq)
  160. {
  161. MsgPkt_T ResPkt;
  162. uint8_t ReqLen;
  163. IPMIMsgHdr_T* pIPMIMsgReq = (IPMIMsgHdr_T*) pReq->Data;
  164. int RetVal = 0;
  165. ReqLen = pReq->Size;
  166. /* Validate Checksums */
  167. if (TRUE != ValidateIPMBChkSums ((uint8_t*)pIPMIMsgReq, ReqLen))
  168. {
  169. printf ("IPMBIfc.c : IPMB Checksum validation failed\r\n");
  170. return;
  171. }
  172. pReq->Param = PARAM_IFC;
  173. pReq->Cmd = pIPMIMsgReq->Cmd;
  174. pReq->NetFnLUN = pIPMIMsgReq->NetFnLUN;
  175. if(pReq->Channel == PRIMARY_IPMB_CHANNEL)
  176. pReq->SrcQ = gFd_PrimaryIpmbResQ;
  177. else
  178. pReq->SrcQ = gFd_SecondaryIpmbResQ;
  179. /* Post the message to message handler Task */
  180. PostMsg (gFd_MsgHndlrIfc, pReq);
  181. if(pReq->ResTimeOut < 1) pReq->ResTimeOut = 10;
  182. do
  183. {
  184. if(pReq->Channel == PRIMARY_IPMB_CHANNEL)
  185. RetVal = GetMsg (gFd_PrimaryIpmbResQ, &ResPkt, pReq->ResTimeOut);
  186. else
  187. RetVal = GetMsg (gFd_SecondaryIpmbResQ, &ResPkt, pReq->ResTimeOut);
  188. }while(!RetVal && !IsResponseMatch(pReq, &ResPkt) );
  189. if (0 != RetVal)
  190. {
  191. printf("Warning: Process IPMB request fail, return timeout.\r\n");
  192. ResPkt.Param = PARAM_NORMAL_RESPONSE;
  193. ResPkt.Size = sizeof(IPMIMsgHdr_T)+2;
  194. SwapIPMIMsgHdr(((IPMIMsgHdr_T*)pReq->Data), ((IPMIMsgHdr_T*)ResPkt.Data));
  195. ResPkt.Data[sizeof(IPMIMsgHdr_T)] = CC_TIMEOUT;
  196. ResPkt.Data[sizeof(IPMIMsgHdr_T)+1] = CalculateCheckSum2(ResPkt.Data, sizeof(IPMIMsgHdr_T)+1);
  197. }
  198. /* If its not normal IPMI Response just return */
  199. if (PARAM_NO_RESPONSE == ResPkt.Param)
  200. {
  201. printf ("IPMBIfc.c : IPMB request packet dropped, not processed\n");
  202. return;
  203. }
  204. /* Send the response */
  205. if(pReq->Channel == PRIMARY_IPMB_CHANNEL)
  206. {
  207. RetVal = stm32_i2c_master_write(gFd_Primary, ResPkt.Data[0], &ResPkt.Data[1], ResPkt.Size);
  208. }
  209. else if(pReq->Channel == SECONDARY_IPMB_CHANNEL)
  210. {
  211. RetVal = stm32_i2c_master_write(gFd_Secondary, ResPkt.Data[0], &ResPkt.Data[1], ResPkt.Size);
  212. }
  213. else
  214. {
  215. printf("IPMBIfc.c: IPMB channel error. %#x\r\n", pReq->Channel);
  216. }
  217. if (0 != RetVal)
  218. {
  219. printf ("IPMBIfc.c : Unable to send a IPMI Response\n");
  220. }
  221. return;
  222. }
  223. /**
  224. * @brief Validate IPMB Checksums.
  225. * @param Data - Buffer to be validated.
  226. * @param Size - Size of the buffer.
  227. * @return TRUE if valid, FALSE if error.
  228. **/
  229. static uint8_t ValidateIPMBChkSums ( uint8_t* Data, uint8_t Size)
  230. {
  231. uint8_t i;
  232. uint8_t chksum;
  233. /* Check IPMB message min size */
  234. if (Size < MIN_IPMB_MSG_LENGTH)
  235. {
  236. printf("IPMBIfc.c: Invalid IPMB Message Length\r\n");
  237. return FALSE;
  238. }
  239. /* Validate the checksum1 */
  240. chksum = 0;
  241. for (i = 0; i < 3; i++)
  242. {
  243. chksum += *(Data + i);
  244. }
  245. if (chksum != 0)
  246. {
  247. printf ("IPMBIfc.c: Invalid checksum 1, size = %d\n",Size);
  248. for(i = 0; i < Size;i++)
  249. printf("%x ",Data[i]);
  250. printf("\n");
  251. return FALSE;
  252. }
  253. /* Validate the checksum2 */
  254. chksum = 0;
  255. for (i = 3; i < Size; i++)
  256. {
  257. chksum += *(Data + i);
  258. }
  259. if (chksum != 0)
  260. {
  261. printf("IPMBIfc.c: Invalid checksum 2, size = %d\n",Size);
  262. for(i = 0; i < Size;i++)
  263. printf("%x ",Data[i]);
  264. printf("\n");
  265. return FALSE;
  266. }
  267. return TRUE;
  268. }
  269. /*-----------------------------------------------------
  270. * @fn IsResponseMatch
  271. * @brief Checks if the Response Message corresponds to
  272. * the Request Message
  273. *
  274. * @param pReq : IPMI Request Message Packet
  275. * @param pRes : IPMI Response Message Packet
  276. *
  277. * @return 1 if match
  278. * 0 if mismatch
  279. *----------------------------------------------------*/
  280. uint8_t IsResponseMatch (MsgPkt_T* pReq, MsgPkt_T* pRes)
  281. {
  282. if( ( ((((IPMIMsgHdr_T*)pRes->Data)->RqSeqLUN) >> 2) != ((((IPMIMsgHdr_T*)pReq->Data)->RqSeqLUN) >> 2) ) &&
  283. ( (pReq->NetFnLUN & 0x4) && (PARAM_NO_RESPONSE != pRes->Param) ) )
  284. {
  285. return FALSE;
  286. }
  287. return TRUE;
  288. }
  289. /**
  290. * @fn RecvIPMBPkt
  291. * @brief This function receives IPMB request packet and post the message to
  292. * IPMB interface queue
  293. **/
  294. static void* RecvIPMBPkt (void *Param)
  295. {
  296. MsgPkt_T IPMBReqPkt;
  297. int IPMBSlaveFd;
  298. int retval;
  299. uint8_t ipmbSelect = *(uint8_t*)Param;
  300. prctl(PR_SET_NAME, __FUNCTION__, 0, 0, 0);
  301. int curThreadIndex = 0;
  302. memset(&IPMBReqPkt,0,sizeof(MsgPkt_T));
  303. if(0 == ipmbSelect)
  304. {
  305. printf ("Primary IPMB Receiver Started...\n");
  306. IPMBReqPkt.Data [0] = g_BMCInfo.IpmiConfig.PrimaryIPMBAddr;
  307. IPMBReqPkt.Channel = PRIMARY_IPMB_CHANNEL;
  308. IPMBSlaveFd = gFd_Primary;
  309. }
  310. else if(1 == ipmbSelect)
  311. {
  312. printf ("Secondary IPMB Receiver Started...\n");
  313. IPMBReqPkt.Data [0] = g_BMCInfo.IpmiConfig.SecondaryIPMBAddr;
  314. IPMBReqPkt.Channel = SECONDARY_IPMB_CHANNEL;
  315. IPMBSlaveFd = gFd_Secondary;
  316. }
  317. /* Loop forever */
  318. while (1)
  319. {
  320. retval = stm32_i2c_slave_recv(IPMBSlaveFd, &IPMBReqPkt.Data[1]);
  321. if( retval > 5)
  322. {
  323. IPMBReqPkt.Param = PARAM_REQUEST;
  324. IPMBReqPkt.Size = retval + 1; /* +1 to include BMC Slave address */
  325. /* Post the IPMB Request message to IPMBIfc Queue */
  326. if(ipmbSelect == 0)
  327. {
  328. if (0 != PostMsg (gFd_PrimaryIpmbIfcQ, &IPMBReqPkt))
  329. {
  330. printf ("IPMBIfc.c : Error posting message to IPMBIfc_Q\n");
  331. }
  332. }
  333. else
  334. {
  335. if (0 != PostMsg (gFd_SecondaryIpmbIfcQ, &IPMBReqPkt))
  336. {
  337. printf ("IPMBIfc.c : Error posting message to IPMBIfc_Q\n");
  338. }
  339. }
  340. }
  341. int cnt;
  342. if(0 == ipmbSelect)
  343. printf("---> Primary IPMB Recv: ");
  344. else
  345. printf("---> Secondary IPMB Recv: ");
  346. for(cnt=0;cnt < IPMBReqPkt.Size; cnt++)
  347. printf("%#x ", IPMBReqPkt.Data[cnt]);
  348. printf("\n");
  349. }
  350. return (void*)-1;
  351. }