ipmi_fwum.c 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128
  1. /*
  2. * Copyright (c) 2004 Kontron Canada, Inc. All Rights Reserved.
  3. *
  4. * Base on code from
  5. * Copyright (c) 2003 Sun Microsystems, Inc. All Rights Reserved.
  6. *
  7. * Redistribution and use in source and binary forms, with or without
  8. * modification, are permitted provided that the following conditions
  9. * are met:
  10. *
  11. * Redistribution of source code must retain the above copyright
  12. * notice, this list of conditions and the following disclaimer.
  13. *
  14. * Redistribution in binary form must reproduce the above copyright
  15. * notice, this list of conditions and the following disclaimer in the
  16. * documentation and/or other materials provided with the distribution.
  17. *
  18. * Neither the name of Sun Microsystems, Inc. or the names of
  19. * contributors may be used to endorse or promote products derived
  20. * from this software without specific prior written permission.
  21. *
  22. * This software is provided "AS IS," without a warranty of any kind.
  23. * ALL EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES,
  24. * INCLUDING ANY IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A
  25. * PARTICULAR PURPOSE OR NON-INFRINGEMENT, ARE HEREBY EXCLUDED.
  26. * SUN MICROSYSTEMS, INC. ("SUN") AND ITS LICENSORS SHALL NOT BE LIABLE
  27. * FOR ANY DAMAGES SUFFERED BY LICENSEE AS A RESULT OF USING, MODIFYING
  28. * OR DISTRIBUTING THIS SOFTWARE OR ITS DERIVATIVES. IN NO EVENT WILL
  29. * SUN OR ITS LICENSORS BE LIABLE FOR ANY LOST REVENUE, PROFIT OR DATA,
  30. * OR FOR DIRECT, INDIRECT, SPECIAL, CONSEQUENTIAL, INCIDENTAL OR
  31. * PUNITIVE DAMAGES, HOWEVER CAUSED AND REGARDLESS OF THE THEORY OF
  32. * LIABILITY, ARISING OUT OF THE USE OF OR INABILITY TO USE THIS SOFTWARE,
  33. * EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES.
  34. */
  35. #include <string.h>
  36. #include <math.h>
  37. #include <time.h>
  38. #include <unistd.h>
  39. #include <ipmitool/log.h>
  40. #include <ipmitool/helper.h>
  41. #include <ipmitool/ipmi.h>
  42. #include <ipmitool/ipmi_fwum.h>
  43. #include <ipmitool/ipmi_intf.h>
  44. #include <ipmitool/ipmi_mc.h>
  45. extern int verbose;
  46. unsigned char firmBuf[1024*512];
  47. tKFWUM_SaveFirmwareInfo save_fw_nfo;
  48. int KfwumGetFileSize(const char *pFileName,
  49. unsigned long *pFileSize);
  50. int KfwumSetupBuffersFromFile(const char *pFileName,
  51. unsigned long fileSize);
  52. void KfwumShowProgress(const char *task, unsigned long current,
  53. unsigned long total);
  54. unsigned short KfwumCalculateChecksumPadding(unsigned char *pBuffer,
  55. unsigned long totalSize);
  56. int KfwumGetInfo(struct ipmi_intf *intf, unsigned char output,
  57. unsigned char *pNumBank);
  58. int KfwumGetDeviceInfo(struct ipmi_intf *intf,
  59. unsigned char output, tKFWUM_BoardInfo *pBoardInfo);
  60. int KfwumGetStatus(struct ipmi_intf *intf);
  61. int KfwumManualRollback(struct ipmi_intf *intf);
  62. int KfwumStartFirmwareImage(struct ipmi_intf *intf,
  63. unsigned long length, unsigned short padding);
  64. int KfwumSaveFirmwareImage(struct ipmi_intf *intf,
  65. unsigned char sequenceNumber, unsigned long address,
  66. unsigned char *pFirmBuf, unsigned char *pInBufLength);
  67. int KfwumFinishFirmwareImage(struct ipmi_intf *intf,
  68. tKFWUM_InFirmwareInfo firmInfo);
  69. int KfwumUploadFirmware(struct ipmi_intf *intf,
  70. unsigned char *pBuffer, unsigned long totalSize);
  71. int KfwumStartFirmwareUpgrade(struct ipmi_intf *intf);
  72. int KfwumGetInfoFromFirmware(unsigned char *pBuf,
  73. unsigned long bufSize, tKFWUM_InFirmwareInfo *pInfo);
  74. void KfwumFixTableVersionForOldFirmware(tKFWUM_InFirmwareInfo *pInfo);
  75. int KfwumGetTraceLog(struct ipmi_intf *intf);
  76. int ipmi_kfwum_checkfwcompat(tKFWUM_BoardInfo boardInfo,
  77. tKFWUM_InFirmwareInfo firmInfo);
  78. int ipmi_fwum_fwupgrade(struct ipmi_intf *intf, char *file, int action);
  79. int ipmi_fwum_info(struct ipmi_intf *intf);
  80. int ipmi_fwum_status(struct ipmi_intf *intf);
  81. void printf_kfwum_help(void);
  82. void printf_kfwum_info(tKFWUM_BoardInfo boardInfo,
  83. tKFWUM_InFirmwareInfo firmInfo);
  84. /* String table */
  85. /* Must match eFWUM_CmdId */
  86. const char *CMD_ID_STRING[] = {
  87. "GetFwInfo",
  88. "KickWatchdog",
  89. "GetLastAnswer",
  90. "BootHandshake",
  91. "ReportStatus",
  92. "CtrlIPMBLine",
  93. "SetFwState",
  94. "GetFwStatus",
  95. "GetSpiMemStatus",
  96. "StartFwUpdate",
  97. "StartFwImage",
  98. "SaveFwImage",
  99. "FinishFwImage",
  100. "ReadFwImage",
  101. "ManualRollback",
  102. "GetTraceLog"
  103. };
  104. const char *EXT_CMD_ID_STRING[] = {
  105. "FwUpgradeLock",
  106. "ProcessFwUpg",
  107. "ProcessFwRb",
  108. "WaitHSAfterUpg",
  109. "WaitFirstHSUpg",
  110. "FwInfoStateChange"
  111. };
  112. const char *CMD_STATE_STRING[] = {
  113. "Invalid",
  114. "Begin",
  115. "Progress",
  116. "Completed"
  117. };
  118. const struct valstr bankStateValS[] = {
  119. { 0x00, "Not programmed" },
  120. { 0x01, "New firmware" },
  121. { 0x02, "Wait for validation" },
  122. { 0x03, "Last Known Good" },
  123. { 0x04, "Previous Good" }
  124. };
  125. /* ipmi_fwum_main - entry point for this ipmitool mode
  126. *
  127. * @intf: ipmi interface
  128. * @arc: number of arguments
  129. * @argv: point to argument array
  130. *
  131. * returns 0 on success
  132. * returns -1 on error
  133. */
  134. int
  135. ipmi_fwum_main(struct ipmi_intf *intf, int argc, char **argv)
  136. {
  137. int rc = 0;
  138. printf("FWUM extension Version %d.%d\n", VER_MAJOR, VER_MINOR);
  139. if (argc < 1) {
  140. lprintf(LOG_ERR, "Not enough parameters given.");
  141. printf_kfwum_help();
  142. return (-1);
  143. }
  144. if (strncmp(argv[0], "help", 4) == 0) {
  145. printf_kfwum_help();
  146. rc = 0;
  147. } else if (strncmp(argv[0], "info", 4) == 0) {
  148. rc = ipmi_fwum_info(intf);
  149. } else if (strncmp(argv[0], "status", 6) == 0) {
  150. rc = ipmi_fwum_status(intf);
  151. } else if (strncmp(argv[0], "rollback", 8) == 0) {
  152. rc = KfwumManualRollback(intf);
  153. } else if (strncmp(argv[0], "download", 8) == 0) {
  154. if ((argc < 2) || (strlen(argv[1]) < 1)) {
  155. lprintf(LOG_ERR,
  156. "Path and file name must be specified.");
  157. return (-1);
  158. }
  159. printf("Firmware File Name : %s\n", argv[1]);
  160. rc = ipmi_fwum_fwupgrade(intf, argv[1], 0);
  161. } else if (strncmp(argv[0], "upgrade", 7) == 0) {
  162. if ((argc >= 2) && (strlen(argv[1]) > 0)) {
  163. printf("Upgrading using file name %s\n", argv[1]);
  164. rc = ipmi_fwum_fwupgrade(intf, argv[1], 1);
  165. } else {
  166. rc = KfwumStartFirmwareUpgrade(intf);
  167. }
  168. } else if (strncmp(argv[0], "tracelog", 8) == 0) {
  169. rc = KfwumGetTraceLog(intf);
  170. } else {
  171. lprintf(LOG_ERR, "Invalid KFWUM command: %s", argv[0]);
  172. printf_kfwum_help();
  173. rc = (-1);
  174. }
  175. return rc;
  176. }
  177. void
  178. printf_kfwum_help(void)
  179. {
  180. lprintf(LOG_NOTICE,
  181. "KFWUM Commands: info status download upgrade rollback tracelog");
  182. }
  183. /* private definitions and macros */
  184. typedef enum eFWUM_CmdId
  185. {
  186. KFWUM_CMD_ID_GET_FIRMWARE_INFO = 0,
  187. KFWUM_CMD_ID_KICK_IPMC_WATCHDOG = 1,
  188. KFWUM_CMD_ID_GET_LAST_ANSWER = 2,
  189. KFWUM_CMD_ID_BOOT_HANDSHAKE = 3,
  190. KFWUM_CMD_ID_REPORT_STATUS = 4,
  191. KFWUM_CMD_ID_GET_FIRMWARE_STATUS = 7,
  192. KFWUM_CMD_ID_START_FIRMWARE_UPDATE = 9,
  193. KFWUM_CMD_ID_START_FIRMWARE_IMAGE = 0x0a,
  194. KFWUM_CMD_ID_SAVE_FIRMWARE_IMAGE = 0x0b,
  195. KFWUM_CMD_ID_FINISH_FIRMWARE_IMAGE = 0x0c,
  196. KFWUM_CMD_ID_READ_FIRMWARE_IMAGE = 0x0d,
  197. KFWUM_CMD_ID_MANUAL_ROLLBACK = 0x0e,
  198. KFWUM_CMD_ID_GET_TRACE_LOG = 0x0f,
  199. KFWUM_CMD_ID_STD_MAX_CMD,
  200. KFWUM_CMD_ID_EXTENDED_CMD = 0xC0
  201. } tKFWUM_CmdId;
  202. int
  203. ipmi_fwum_info(struct ipmi_intf *intf)
  204. {
  205. tKFWUM_BoardInfo b_info;
  206. int rc = 0;
  207. unsigned char not_used;
  208. if (verbose) {
  209. printf("Getting Kontron FWUM Info\n");
  210. }
  211. if (KfwumGetDeviceInfo(intf, 1, &b_info) != 0) {
  212. rc = (-1);
  213. }
  214. if (KfwumGetInfo(intf, 1, &not_used) != 0) {
  215. rc = (-1);
  216. }
  217. return rc;
  218. }
  219. int
  220. ipmi_fwum_status(struct ipmi_intf *intf)
  221. {
  222. if (verbose) {
  223. printf("Getting Kontron FWUM Status\n");
  224. }
  225. if (KfwumGetStatus(intf) != 0) {
  226. return (-1);
  227. }
  228. return 0;
  229. }
  230. /* ipmi_fwum_fwupgrade - function implements download/upload of the firmware
  231. * data received as parameters
  232. *
  233. * @file: fw file
  234. * @action: 0 = download, 1 = upload/start upload
  235. *
  236. * returns 0 on success, otherwise (-1)
  237. */
  238. int
  239. ipmi_fwum_fwupgrade(struct ipmi_intf *intf, char *file, int action)
  240. {
  241. tKFWUM_BoardInfo b_info;
  242. tKFWUM_InFirmwareInfo fw_info = { 0 };
  243. unsigned short padding;
  244. unsigned long fsize = 0;
  245. unsigned char not_used;
  246. if (file == NULL) {
  247. lprintf(LOG_ERR, "No file given.");
  248. return (-1);
  249. }
  250. if (KfwumGetFileSize(file, &fsize) != 0) {
  251. return (-1);
  252. }
  253. if (KfwumSetupBuffersFromFile(file, fsize) != 0) {
  254. return (-1);
  255. }
  256. padding = KfwumCalculateChecksumPadding(firmBuf, fsize);
  257. if (KfwumGetInfoFromFirmware(firmBuf, fsize, &fw_info) != 0) {
  258. return (-1);
  259. }
  260. if (KfwumGetDeviceInfo(intf, 0, &b_info) != 0) {
  261. return (-1);
  262. }
  263. if (ipmi_kfwum_checkfwcompat(b_info, fw_info) != 0) {
  264. return (-1);
  265. }
  266. KfwumGetInfo(intf, 0, &not_used);
  267. printf_kfwum_info(b_info, fw_info);
  268. if (KfwumStartFirmwareImage(intf, fsize, padding) != 0) {
  269. return (-1);
  270. }
  271. if (KfwumUploadFirmware(intf, firmBuf, fsize) != 0) {
  272. return (-1);
  273. }
  274. if (KfwumFinishFirmwareImage(intf, fw_info) != 0) {
  275. return (-1);
  276. }
  277. if (KfwumGetStatus(intf) != 0) {
  278. return (-1);
  279. }
  280. if (action != 0) {
  281. if (KfwumStartFirmwareUpgrade(intf) != 0) {
  282. return (-1);
  283. }
  284. }
  285. return 0;
  286. }
  287. /* KfwumGetFileSize - gets the file size
  288. *
  289. * @pFileName : filename ptr
  290. * @pFileSize : output ptr for filesize
  291. *
  292. * returns 0 on success, otherwise (-1)
  293. */
  294. int
  295. KfwumGetFileSize(const char *pFileName, unsigned long *pFileSize)
  296. {
  297. FILE *pFileHandle = NULL;
  298. pFileHandle = fopen(pFileName, "rb");
  299. if (pFileHandle == NULL) {
  300. return (-1);
  301. }
  302. if (fseek(pFileHandle, 0L , SEEK_END) == 0) {
  303. *pFileSize = ftell(pFileHandle);
  304. }
  305. fclose(pFileHandle);
  306. if (*pFileSize != 0) {
  307. return 0;
  308. }
  309. return (-1);
  310. }
  311. /* KfwumSetupBuffersFromFile - small buffers are used to store the file data
  312. *
  313. * @pFileName : filename ptr
  314. * unsigned long : filesize
  315. *
  316. * returns 0 on success, otherwise (-1)
  317. */
  318. int
  319. KfwumSetupBuffersFromFile(const char *pFileName, unsigned long fileSize)
  320. {
  321. int rc = (-1);
  322. FILE *pFileHandle = NULL;
  323. int count;
  324. int modulus;
  325. int qty = 0;
  326. pFileHandle = fopen(pFileName, "rb");
  327. if (pFileHandle == NULL) {
  328. lprintf(LOG_ERR, "Failed to open '%s' for reading.",
  329. pFileName);
  330. return (-1);
  331. }
  332. count = fileSize / MAX_BUFFER_SIZE;
  333. modulus = fileSize % MAX_BUFFER_SIZE;
  334. rewind(pFileHandle);
  335. for (qty = 0; qty < count; qty++) {
  336. KfwumShowProgress("Reading Firmware from File",
  337. qty, count);
  338. if (fread(&firmBuf[qty * MAX_BUFFER_SIZE], 1,
  339. MAX_BUFFER_SIZE,
  340. pFileHandle) == MAX_BUFFER_SIZE) {
  341. rc = 0;
  342. }
  343. }
  344. if (modulus) {
  345. if (fread(&firmBuf[qty * MAX_BUFFER_SIZE], 1,
  346. modulus, pFileHandle) == modulus) {
  347. rc = 0;
  348. }
  349. }
  350. if (rc == 0) {
  351. KfwumShowProgress("Reading Firmware from File", 100, 100);
  352. }
  353. fclose(pFileHandle);
  354. return rc;
  355. }
  356. /* KfwumShowProgress - helper routine to display progress bar
  357. *
  358. * Converts current/total in percent
  359. *
  360. * *task : string identifying current operation
  361. * current: progress
  362. * total : limit
  363. */
  364. void
  365. KfwumShowProgress(const char *task, unsigned long current, unsigned long total)
  366. {
  367. # define PROG_LENGTH 42
  368. static unsigned long staticProgress=0xffffffff;
  369. unsigned char spaces[PROG_LENGTH + 1];
  370. unsigned short hash;
  371. float percent = ((float)current / total);
  372. unsigned long progress = 100 * (percent);
  373. if (staticProgress == progress) {
  374. /* We displayed the same last time.. so don't do it */
  375. return;
  376. }
  377. staticProgress = progress;
  378. printf("%-25s : ", task); /* total 20 bytes */
  379. hash = (percent * PROG_LENGTH);
  380. memset(spaces, '#', hash);
  381. spaces[hash] = '\0';
  382. printf("%s", spaces);
  383. memset(spaces, ' ', (PROG_LENGTH - hash));
  384. spaces[(PROG_LENGTH - hash)] = '\0';
  385. printf("%s", spaces );
  386. printf(" %3ld %%\r", progress); /* total 7 bytes */
  387. if (progress == 100) {
  388. printf("\n");
  389. }
  390. fflush(stdout);
  391. }
  392. /* KfwumCalculateChecksumPadding - TBD
  393. */
  394. unsigned short
  395. KfwumCalculateChecksumPadding(unsigned char *pBuffer, unsigned long totalSize)
  396. {
  397. unsigned short sumOfBytes = 0;
  398. unsigned short padding;
  399. unsigned long counter;
  400. for (counter = 0; counter < totalSize; counter ++) {
  401. sumOfBytes += pBuffer[counter];
  402. }
  403. padding = 0 - sumOfBytes;
  404. return padding;
  405. }
  406. /* KfwumGetInfo - Get Firmware Update Manager (FWUM) information
  407. *
  408. * *intf : IPMI interface
  409. * output : when set to non zero, queried information is displayed
  410. * pNumBank: output ptr for number of banks
  411. *
  412. * returns 0 on success, otherwise (-1)
  413. */
  414. int
  415. KfwumGetInfo(struct ipmi_intf *intf, unsigned char output,
  416. unsigned char *pNumBank)
  417. {
  418. int rc = 0;
  419. static struct KfwumGetInfoResp *pGetInfo;
  420. struct ipmi_rs *rsp;
  421. struct ipmi_rq req;
  422. memset(&req, 0, sizeof(req));
  423. req.msg.netfn = IPMI_NETFN_FIRMWARE;
  424. req.msg.cmd = KFWUM_CMD_ID_GET_FIRMWARE_INFO;
  425. req.msg.data_len = 0;
  426. rsp = intf->sendrecv(intf, &req);
  427. if (!rsp) {
  428. lprintf(LOG_ERR, "Error in FWUM Firmware Get Info Command.");
  429. return (-1);
  430. } else if (rsp->ccode != 0) {
  431. lprintf(LOG_ERR, "FWUM Firmware Get Info returned %x",
  432. rsp->ccode);
  433. return (-1);
  434. }
  435. pGetInfo = (struct KfwumGetInfoResp *)rsp->data;
  436. if (output) {
  437. printf("\nFWUM info\n");
  438. printf("=========\n");
  439. printf("Protocol Revision : %02Xh\n",
  440. pGetInfo->protocolRevision);
  441. printf("Controller Device Id : %02Xh\n",
  442. pGetInfo->controllerDeviceId);
  443. printf("Firmware Revision : %u.%u%u",
  444. pGetInfo->firmRev1, pGetInfo->firmRev2 >> 4,
  445. pGetInfo->firmRev2 & 0x0f);
  446. if (pGetInfo->byte.mode != 0) {
  447. printf(" - DEBUG BUILD\n");
  448. } else {
  449. printf("\n");
  450. }
  451. printf("Number Of Memory Bank : %u\n", pGetInfo->numBank);
  452. }
  453. *pNumBank = pGetInfo->numBank;
  454. /* Determine wich type of download to use: */
  455. /* Old FWUM or Old IPMC fw (data_len < 7)
  456. * --> Address with small buffer size
  457. */
  458. if ((pGetInfo->protocolRevision) <= 0x05 || (rsp->data_len < 7 )) {
  459. save_fw_nfo.downloadType = KFWUM_DOWNLOAD_TYPE_ADDRESS;
  460. save_fw_nfo.bufferSize = KFWUM_SMALL_BUFFER;
  461. save_fw_nfo.overheadSize = KFWUM_OLD_CMD_OVERHEAD;
  462. if (verbose) {
  463. printf("Protocol Revision :");
  464. printf(" <= 5 detected, adjusting buffers\n");
  465. }
  466. } else {
  467. /* Both fw are using the new protocol */
  468. save_fw_nfo.downloadType = KFWUM_DOWNLOAD_TYPE_SEQUENCE;
  469. save_fw_nfo.overheadSize = KFWUM_NEW_CMD_OVERHEAD;
  470. /* Buffer size depending on access type (Local or remote) */
  471. /* Look if we run remote or locally */
  472. if (verbose) {
  473. printf("Protocol Revision :");
  474. printf(" > 5 optimizing buffers\n");
  475. }
  476. if (strstr(intf->name,"lan") != NULL) {
  477. /* also covers lanplus */
  478. save_fw_nfo.bufferSize = KFWUM_SMALL_BUFFER;
  479. if (verbose) {
  480. printf("IOL payload size : %d\n",
  481. save_fw_nfo.bufferSize);
  482. }
  483. } else if ((strstr(intf->name,"open")!= NULL)
  484. && intf->target_addr != IPMI_BMC_SLAVE_ADDR
  485. && (intf->target_addr != intf->my_addr)) {
  486. save_fw_nfo.bufferSize = KFWUM_SMALL_BUFFER;
  487. if (verbose) {
  488. printf("IPMB payload size : %d\n",
  489. save_fw_nfo.bufferSize);
  490. }
  491. } else {
  492. save_fw_nfo.bufferSize = KFWUM_BIG_BUFFER;
  493. if (verbose) {
  494. printf("SMI payload size : %d\n",
  495. save_fw_nfo.bufferSize);
  496. }
  497. }
  498. }
  499. return rc;
  500. }
  501. /* KfwumGetDeviceInfo - Get IPMC/Board information
  502. *
  503. * *intf: IPMI interface
  504. * output: when set to non zero, queried information is displayed
  505. * tKFWUM_BoardInfo: output ptr for IPMC/Board information
  506. *
  507. * returns 0 on success, otherwise (-1)
  508. */
  509. int
  510. KfwumGetDeviceInfo(struct ipmi_intf *intf, unsigned char output,
  511. tKFWUM_BoardInfo *pBoardInfo)
  512. {
  513. struct ipm_devid_rsp *pGetDevId;
  514. struct ipmi_rs *rsp;
  515. struct ipmi_rq req;
  516. /* Send Get Device Id */
  517. memset(&req, 0, sizeof(req));
  518. req.msg.netfn = IPMI_NETFN_APP;
  519. req.msg.cmd = BMC_GET_DEVICE_ID;
  520. req.msg.data_len = 0;
  521. rsp = intf->sendrecv(intf, &req);
  522. if (rsp == NULL) {
  523. lprintf(LOG_ERR, "Error in Get Device Id Command");
  524. return (-1);
  525. } else if (rsp->ccode != 0) {
  526. lprintf(LOG_ERR, "Get Device Id returned %x",
  527. rsp->ccode);
  528. return (-1);
  529. }
  530. pGetDevId = (struct ipm_devid_rsp *)rsp->data;
  531. pBoardInfo->iana = IPM_DEV_MANUFACTURER_ID(pGetDevId->manufacturer_id);
  532. pBoardInfo->boardId = buf2short(pGetDevId->product_id);
  533. if (output) {
  534. printf("\nIPMC Info\n");
  535. printf("=========\n");
  536. printf("Manufacturer Id : %u\n",
  537. pBoardInfo->iana);
  538. printf("Board Id : %u\n",
  539. pBoardInfo->boardId);
  540. printf("Firmware Revision : %u.%u%u",
  541. pGetDevId->fw_rev1, pGetDevId->fw_rev2 >> 4,
  542. pGetDevId->fw_rev2 & 0x0f);
  543. if (((pBoardInfo->iana == IPMI_OEM_KONTRON)
  544. && (pBoardInfo->boardId == KFWUM_BOARD_KONTRON_5002))) {
  545. printf(" SDR %u", pGetDevId->aux_fw_rev[0]);
  546. }
  547. printf("\n");
  548. }
  549. return 0;
  550. }
  551. /* KfwumGetStatus - Get (and prints) FWUM banks information
  552. *
  553. * *intf : IPMI interface
  554. *
  555. * returns 0 on success, otherwise (-1)
  556. */
  557. int
  558. KfwumGetStatus(struct ipmi_intf * intf)
  559. {
  560. int rc = 0;
  561. struct ipmi_rs *rsp;
  562. struct ipmi_rq req;
  563. struct KfwumGetStatusResp *pGetStatus;
  564. unsigned char numBank;
  565. unsigned char counter;
  566. unsigned long firmLength;
  567. if (verbose) {
  568. printf(" Getting Status!\n");
  569. }
  570. /* Retreive the number of bank */
  571. rc = KfwumGetInfo(intf, 0, &numBank);
  572. for(counter = 0;
  573. (counter < numBank) && (rc == 0);
  574. counter ++) {
  575. /* Retreive the status of each bank */
  576. memset(&req, 0, sizeof(req));
  577. req.msg.netfn = IPMI_NETFN_FIRMWARE;
  578. req.msg.cmd = KFWUM_CMD_ID_GET_FIRMWARE_STATUS;
  579. req.msg.data = &counter;
  580. req.msg.data_len = 1;
  581. rsp = intf->sendrecv(intf, &req);
  582. if (rsp == NULL) {
  583. lprintf(LOG_ERR,
  584. "Error in FWUM Firmware Get Status Command.");
  585. rc = (-1);
  586. break;
  587. } else if (rsp->ccode) {
  588. lprintf(LOG_ERR,
  589. "FWUM Firmware Get Status returned %x",
  590. rsp->ccode);
  591. rc = (-1);
  592. break;
  593. }
  594. pGetStatus = (struct KfwumGetStatusResp *) rsp->data;
  595. printf("\nBank State %d : %s\n",
  596. counter,
  597. val2str(pGetStatus->bankState, bankStateValS));
  598. if (!pGetStatus->bankState) {
  599. continue;
  600. }
  601. firmLength = pGetStatus->firmLengthMSB;
  602. firmLength = firmLength << 8;
  603. firmLength |= pGetStatus->firmLengthMid;
  604. firmLength = firmLength << 8;
  605. firmLength |= pGetStatus->firmLengthLSB;
  606. printf("Firmware Length : %ld bytes\n",
  607. firmLength);
  608. printf("Firmware Revision : %u.%u%u SDR %u\n",
  609. pGetStatus->firmRev1,
  610. pGetStatus->firmRev2 >> 4,
  611. pGetStatus->firmRev2 & 0x0f,
  612. pGetStatus->firmRev3);
  613. }
  614. printf("\n");
  615. return rc;
  616. }
  617. /* KfwumManualRollback - Ask IPMC to rollback to previous version
  618. *
  619. * *intf : IPMI interface
  620. *
  621. * returns 0 on success
  622. * returns (-1) on error
  623. */
  624. int
  625. KfwumManualRollback(struct ipmi_intf *intf)
  626. {
  627. struct ipmi_rs *rsp;
  628. struct ipmi_rq req;
  629. struct KfwumManualRollbackReq thisReq;
  630. memset(&req, 0, sizeof(req));
  631. req.msg.netfn = IPMI_NETFN_FIRMWARE;
  632. req.msg.cmd = KFWUM_CMD_ID_MANUAL_ROLLBACK;
  633. thisReq.type = 0; /* Wait BMC shutdown */
  634. req.msg.data = (unsigned char *)&thisReq;
  635. req.msg.data_len = 1;
  636. rsp = intf->sendrecv(intf, &req);
  637. if (rsp == NULL) {
  638. lprintf(LOG_ERR, "Error in FWUM Manual Rollback Command.");
  639. return (-1);
  640. } else if (rsp->ccode != 0) {
  641. lprintf(LOG_ERR,
  642. "Error in FWUM Manual Rollback Command returned %x",
  643. rsp->ccode);
  644. return (-1);
  645. }
  646. printf("FWUM Starting Manual Rollback \n");
  647. return 0;
  648. }
  649. int
  650. KfwumStartFirmwareImage(struct ipmi_intf *intf, unsigned long length,
  651. unsigned short padding)
  652. {
  653. struct ipmi_rs *rsp;
  654. struct ipmi_rq req;
  655. struct KfwumStartFirmwareDownloadResp *pResp;
  656. struct KfwumStartFirmwareDownloadReq thisReq;
  657. thisReq.lengthLSB = length & 0x000000ff;
  658. thisReq.lengthMid = (length >> 8) & 0x000000ff;
  659. thisReq.lengthMSB = (length >> 16) & 0x000000ff;
  660. thisReq.paddingLSB = padding & 0x00ff;
  661. thisReq.paddingMSB = (padding>> 8) & 0x00ff;
  662. thisReq.useSequence = 0x01;
  663. memset(&req, 0, sizeof(req));
  664. req.msg.netfn = IPMI_NETFN_FIRMWARE;
  665. req.msg.cmd = KFWUM_CMD_ID_START_FIRMWARE_IMAGE;
  666. req.msg.data = (unsigned char *) &thisReq;
  667. /* Look for download type */
  668. if (save_fw_nfo.downloadType == KFWUM_DOWNLOAD_TYPE_ADDRESS) {
  669. req.msg.data_len = 5;
  670. } else {
  671. req.msg.data_len = 6;
  672. }
  673. rsp = intf->sendrecv(intf, &req);
  674. if (rsp == NULL) {
  675. lprintf(LOG_ERR,
  676. "Error in FWUM Firmware Start Firmware Image Download Command.");
  677. return (-1);
  678. } else if (rsp->ccode) {
  679. lprintf(LOG_ERR,
  680. "FWUM Firmware Start Firmware Image Download returned %x",
  681. rsp->ccode);
  682. return (-1);
  683. }
  684. pResp = (struct KfwumStartFirmwareDownloadResp *)rsp->data;
  685. printf("Bank holding new firmware : %d\n", pResp->bank);
  686. sleep(5);
  687. return 0;
  688. }
  689. int
  690. KfwumSaveFirmwareImage(struct ipmi_intf *intf, unsigned char sequenceNumber,
  691. unsigned long address, unsigned char *pFirmBuf,
  692. unsigned char *pInBufLength)
  693. {
  694. int rc = 0;
  695. struct ipmi_rs *rsp;
  696. struct ipmi_rq req;
  697. struct KfwumSaveFirmwareAddressReq addr_req;
  698. struct KfwumSaveFirmwareSequenceReq seq_req;
  699. int retry = 0;
  700. int no_rsp = 0;
  701. do {
  702. memset(&req, 0, sizeof(req));
  703. req.msg.netfn = IPMI_NETFN_FIRMWARE;
  704. req.msg.cmd = KFWUM_CMD_ID_SAVE_FIRMWARE_IMAGE;
  705. if (save_fw_nfo.downloadType == KFWUM_DOWNLOAD_TYPE_ADDRESS) {
  706. addr_req.addressLSB = address & 0x000000ff;
  707. addr_req.addressMid = (address >> 8) & 0x000000ff;
  708. addr_req.addressMSB = (address >> 16) & 0x000000ff;
  709. addr_req.numBytes = *pInBufLength;
  710. memcpy(addr_req.txBuf, pFirmBuf, *pInBufLength);
  711. req.msg.data = (unsigned char *)&addr_req;
  712. req.msg.data_len = *pInBufLength + 4;
  713. } else {
  714. seq_req.sequenceNumber = sequenceNumber;
  715. memcpy(seq_req.txBuf, pFirmBuf, *pInBufLength);
  716. req.msg.data = (unsigned char *)&seq_req;
  717. req.msg.data_len = *pInBufLength + sizeof(unsigned char);
  718. /* + 1 => sequenceNumber*/
  719. }
  720. rsp = intf->sendrecv(intf, &req);
  721. if (rsp == NULL) {
  722. lprintf(LOG_ERR,
  723. "Error in FWUM Firmware Save Firmware Image Download Command.");
  724. /* We don't receive "C7" on errors with IOL,
  725. * instead we receive nothing
  726. */
  727. if (strstr(intf->name, "lan") != NULL) {
  728. no_rsp++;
  729. if (no_rsp < FWUM_SAVE_FIRMWARE_NO_RESPONSE_LIMIT) {
  730. *pInBufLength -= 1;
  731. continue;
  732. }
  733. lprintf(LOG_ERR,
  734. "Error, too many commands without response.");
  735. *pInBufLength = 0;
  736. break;
  737. } /* For other interface keep trying */
  738. } else if (rsp->ccode != 0) {
  739. if (rsp->ccode == 0xc0) {
  740. sleep(1);
  741. } else if ((rsp->ccode == 0xc7)
  742. || ((rsp->ccode == 0xc3)
  743. && (sequenceNumber == 0))) {
  744. *pInBufLength -= 1;
  745. retry = 1;
  746. } else if (rsp->ccode == 0x82) {
  747. /* Double sent, continue */
  748. rc = 0;
  749. break;
  750. } else if (rsp->ccode == 0x83) {
  751. if (retry == 0) {
  752. retry = 1;
  753. continue;
  754. }
  755. rc = (-1);
  756. break;
  757. } else if (rsp->ccode == 0xcf) {
  758. /* Ok if receive duplicated request */
  759. retry = 1;
  760. } else if (rsp->ccode == 0xc3) {
  761. if (retry == 0) {
  762. retry = 1;
  763. continue;
  764. }
  765. rc = (-1);
  766. break;
  767. } else {
  768. lprintf(LOG_ERR,
  769. "FWUM Firmware Save Firmware Image Download returned %x",
  770. rsp->ccode);
  771. rc = (-1);
  772. break;
  773. }
  774. } else {
  775. break;
  776. }
  777. } while (1);
  778. return rc;
  779. }
  780. int
  781. KfwumFinishFirmwareImage(struct ipmi_intf *intf, tKFWUM_InFirmwareInfo firmInfo)
  782. {
  783. struct ipmi_rs *rsp;
  784. struct ipmi_rq req;
  785. struct KfwumFinishFirmwareDownloadReq thisReq;
  786. thisReq.versionMaj = firmInfo.versMajor;
  787. thisReq.versionMinSub = ((firmInfo.versMinor <<4)
  788. | firmInfo.versSubMinor);
  789. thisReq.versionSdr = firmInfo.sdrRev;
  790. thisReq.reserved = 0;
  791. /* Byte 4 reserved, write 0 */
  792. memset(&req, 0, sizeof(req));
  793. req.msg.netfn = IPMI_NETFN_FIRMWARE;
  794. req.msg.cmd = KFWUM_CMD_ID_FINISH_FIRMWARE_IMAGE;
  795. req.msg.data = (unsigned char *)&thisReq;
  796. req.msg.data_len = 4;
  797. /* Infinite loop if BMC doesn't reply or replies 0xc0 every time. */
  798. do {
  799. rsp = intf->sendrecv(intf, &req);
  800. } while (rsp == NULL || rsp->ccode == 0xc0);
  801. if (rsp->ccode != 0) {
  802. lprintf(LOG_ERR,
  803. "FWUM Firmware Finish Firmware Image Download returned %x",
  804. rsp->ccode);
  805. return (-1);
  806. }
  807. return 0;
  808. }
  809. int
  810. KfwumUploadFirmware(struct ipmi_intf *intf, unsigned char *pBuffer,
  811. unsigned long totalSize)
  812. {
  813. int rc = (-1);
  814. unsigned long address = 0x0;
  815. unsigned char writeSize;
  816. unsigned char oldWriteSize;
  817. unsigned long lastAddress = 0;
  818. unsigned char sequenceNumber = 0;
  819. unsigned char retry = FWUM_MAX_UPLOAD_RETRY;
  820. do {
  821. writeSize = save_fw_nfo.bufferSize - save_fw_nfo.overheadSize;
  822. /* Reach the end */
  823. if (address + writeSize > totalSize) {
  824. writeSize = (totalSize - address);
  825. } else if (((address % KFWUM_PAGE_SIZE)
  826. + writeSize) > KFWUM_PAGE_SIZE) {
  827. /* Reach boundary end */
  828. writeSize = (KFWUM_PAGE_SIZE - (address % KFWUM_PAGE_SIZE));
  829. }
  830. oldWriteSize = writeSize;
  831. rc = KfwumSaveFirmwareImage(intf, sequenceNumber,
  832. address, &pBuffer[address], &writeSize);
  833. if ((rc != 0) && (retry-- != 0)) {
  834. address = lastAddress;
  835. rc = 0;
  836. } else if ( writeSize == 0) {
  837. rc = (-1);
  838. } else {
  839. if (writeSize != oldWriteSize) {
  840. printf("Adjusting length to %d bytes \n",
  841. writeSize);
  842. save_fw_nfo.bufferSize -= (oldWriteSize - writeSize);
  843. }
  844. retry = FWUM_MAX_UPLOAD_RETRY;
  845. lastAddress = address;
  846. address+= writeSize;
  847. }
  848. if (rc == 0) {
  849. if ((address % 1024) == 0) {
  850. KfwumShowProgress("Writing Firmware in Flash",
  851. address, totalSize);
  852. }
  853. sequenceNumber++;
  854. }
  855. } while ((rc == 0) && (address < totalSize));
  856. if (rc == 0) {
  857. KfwumShowProgress("Writing Firmware in Flash",
  858. 100, 100);
  859. }
  860. return rc;
  861. }
  862. int
  863. KfwumStartFirmwareUpgrade(struct ipmi_intf *intf)
  864. {
  865. int rc = 0;
  866. struct ipmi_rs *rsp;
  867. struct ipmi_rq req;
  868. /* Upgrade type, wait BMC shutdown */
  869. unsigned char upgType = 0 ;
  870. memset(&req, 0, sizeof(req));
  871. req.msg.netfn = IPMI_NETFN_FIRMWARE;
  872. req.msg.cmd = KFWUM_CMD_ID_START_FIRMWARE_UPDATE;
  873. req.msg.data = (unsigned char *) &upgType;
  874. req.msg.data_len = 1;
  875. rsp = intf->sendrecv(intf, &req);
  876. if (rsp == NULL) {
  877. lprintf(LOG_ERR,
  878. "Error in FWUM Firmware Start Firmware Upgrade Command");
  879. rc = (-1);
  880. } else if (rsp->ccode) {
  881. if (rsp->ccode == 0xd5) {
  882. lprintf(LOG_ERR,
  883. "No firmware available for upgrade. Download Firmware first.");
  884. } else {
  885. lprintf(LOG_ERR,
  886. "FWUM Firmware Start Firmware Upgrade returned %x",
  887. rsp->ccode);
  888. }
  889. rc = (-1);
  890. }
  891. return rc;
  892. }
  893. int
  894. KfwumGetTraceLog(struct ipmi_intf *intf)
  895. {
  896. int rc = 0;
  897. struct ipmi_rs *rsp;
  898. struct ipmi_rq req;
  899. unsigned char chunkIdx;
  900. unsigned char cmdIdx;
  901. if (verbose) {
  902. printf(" Getting Trace Log!\n");
  903. }
  904. for (chunkIdx = 0;
  905. (chunkIdx < TRACE_LOG_CHUNK_COUNT)
  906. && (rc == 0);
  907. chunkIdx++) {
  908. /* Retreive each log chunk and print it */
  909. memset(&req, 0, sizeof(req));
  910. req.msg.netfn = IPMI_NETFN_FIRMWARE;
  911. req.msg.cmd = KFWUM_CMD_ID_GET_TRACE_LOG;
  912. req.msg.data = &chunkIdx;
  913. req.msg.data_len = 1;
  914. rsp = intf->sendrecv(intf, &req);
  915. if (rsp == NULL) {
  916. lprintf(LOG_ERR,
  917. "Error in FWUM Firmware Get Trace Log Command");
  918. rc = (-1);
  919. break;
  920. } else if (rsp->ccode) {
  921. lprintf(LOG_ERR,
  922. "FWUM Firmware Get Trace Log returned %x",
  923. rsp->ccode);
  924. rc = (-1);
  925. break;
  926. }
  927. for (cmdIdx=0; cmdIdx < TRACE_LOG_CHUNK_SIZE; cmdIdx++) {
  928. /* Don't diplay commands with an invalid state */
  929. if ((rsp->data[TRACE_LOG_ATT_COUNT * cmdIdx + 1] != 0)
  930. && (rsp->data[TRACE_LOG_ATT_COUNT * cmdIdx] < KFWUM_CMD_ID_STD_MAX_CMD)) {
  931. printf(" Cmd ID: %17s -- CmdState: %10s -- CompCode: %2x\n",
  932. CMD_ID_STRING[rsp->data[TRACE_LOG_ATT_COUNT * cmdIdx]],
  933. CMD_STATE_STRING[rsp->data[TRACE_LOG_ATT_COUNT * cmdIdx + 1]],
  934. rsp->data[TRACE_LOG_ATT_COUNT * cmdIdx + 2]);
  935. } else if ((rsp->data[TRACE_LOG_ATT_COUNT * cmdIdx + 1] != 0)
  936. && (rsp->data[TRACE_LOG_ATT_COUNT*cmdIdx] >= KFWUM_CMD_ID_EXTENDED_CMD)) {
  937. printf(" Cmd ID: %17s -- CmdState: %10s -- CompCode: %2x\n",
  938. EXT_CMD_ID_STRING[rsp->data[TRACE_LOG_ATT_COUNT * cmdIdx] - KFWUM_CMD_ID_EXTENDED_CMD],
  939. CMD_STATE_STRING[rsp->data[TRACE_LOG_ATT_COUNT * cmdIdx + 1]],
  940. rsp->data[TRACE_LOG_ATT_COUNT * cmdIdx + 2]);
  941. }
  942. }
  943. }
  944. printf("\n");
  945. return rc;
  946. }
  947. int
  948. KfwumGetInfoFromFirmware(unsigned char *pBuf, unsigned long bufSize,
  949. tKFWUM_InFirmwareInfo *pInfo)
  950. {
  951. unsigned long offset = 0;
  952. if (bufSize < (IN_FIRMWARE_INFO_OFFSET_LOCATION + IN_FIRMWARE_INFO_SIZE)) {
  953. return (-1);
  954. }
  955. offset = IN_FIRMWARE_INFO_OFFSET_LOCATION;
  956. /* Now, fill the structure with read informations */
  957. pInfo->checksum = (unsigned short)KWUM_GET_BYTE_AT_OFFSET(pBuf,
  958. offset + 0 + IN_FIRMWARE_INFO_OFFSET_CHECKSUM ) << 8;
  959. pInfo->checksum|= (unsigned short)KWUM_GET_BYTE_AT_OFFSET(pBuf,
  960. offset + 1 + IN_FIRMWARE_INFO_OFFSET_CHECKSUM);
  961. pInfo->sumToRemoveFromChecksum = KWUM_GET_BYTE_AT_OFFSET(pBuf,
  962. offset + IN_FIRMWARE_INFO_OFFSET_CHECKSUM);
  963. pInfo->sumToRemoveFromChecksum+= KWUM_GET_BYTE_AT_OFFSET(pBuf,
  964. offset + IN_FIRMWARE_INFO_OFFSET_CHECKSUM + 1);
  965. pInfo->fileSize = KWUM_GET_BYTE_AT_OFFSET(pBuf,
  966. offset + IN_FIRMWARE_INFO_OFFSET_FILE_SIZE + 0) << 24;
  967. pInfo->fileSize|= (unsigned long)KWUM_GET_BYTE_AT_OFFSET(pBuf,
  968. offset + IN_FIRMWARE_INFO_OFFSET_FILE_SIZE + 1) << 16;
  969. pInfo->fileSize|= (unsigned long)KWUM_GET_BYTE_AT_OFFSET(pBuf,
  970. offset + IN_FIRMWARE_INFO_OFFSET_FILE_SIZE + 2) << 8;
  971. pInfo->fileSize|= (unsigned long)KWUM_GET_BYTE_AT_OFFSET(pBuf,
  972. offset + IN_FIRMWARE_INFO_OFFSET_FILE_SIZE + 3);
  973. pInfo->boardId = KWUM_GET_BYTE_AT_OFFSET(pBuf,
  974. offset + IN_FIRMWARE_INFO_OFFSET_BOARD_ID + 0) << 8;
  975. pInfo->boardId|= KWUM_GET_BYTE_AT_OFFSET(pBuf,
  976. offset + IN_FIRMWARE_INFO_OFFSET_BOARD_ID + 1);
  977. pInfo->deviceId = KWUM_GET_BYTE_AT_OFFSET(pBuf,
  978. offset + IN_FIRMWARE_INFO_OFFSET_DEVICE_ID);
  979. pInfo->tableVers = KWUM_GET_BYTE_AT_OFFSET(pBuf,
  980. offset + IN_FIRMWARE_INFO_OFFSET_TABLE_VERSION);
  981. pInfo->implRev = KWUM_GET_BYTE_AT_OFFSET(pBuf,
  982. offset + IN_FIRMWARE_INFO_OFFSET_IMPLEMENT_REV);
  983. pInfo->versMajor = (KWUM_GET_BYTE_AT_OFFSET(pBuf,
  984. offset
  985. + IN_FIRMWARE_INFO_OFFSET_VER_MAJOROR)) & 0x0f;
  986. pInfo->versMinor = (KWUM_GET_BYTE_AT_OFFSET(pBuf,
  987. offset
  988. + IN_FIRMWARE_INFO_OFFSET_VER_MINORSUB) >> 4) & 0x0f;
  989. pInfo->versSubMinor = (KWUM_GET_BYTE_AT_OFFSET(pBuf,
  990. offset + IN_FIRMWARE_INFO_OFFSET_VER_MINORSUB)) & 0x0f;
  991. pInfo->sdrRev = KWUM_GET_BYTE_AT_OFFSET(pBuf,
  992. offset + IN_FIRMWARE_INFO_OFFSET_SDR_REV);
  993. pInfo->iana = KWUM_GET_BYTE_AT_OFFSET(pBuf,
  994. offset + IN_FIRMWARE_INFO_OFFSET_IANA2) << 16;
  995. pInfo->iana|= (unsigned long)KWUM_GET_BYTE_AT_OFFSET(pBuf,
  996. offset + IN_FIRMWARE_INFO_OFFSET_IANA1) << 8;
  997. pInfo->iana|= (unsigned long)KWUM_GET_BYTE_AT_OFFSET(pBuf,
  998. offset + IN_FIRMWARE_INFO_OFFSET_IANA0);
  999. KfwumFixTableVersionForOldFirmware(pInfo);
  1000. return 0;
  1001. }
  1002. void
  1003. KfwumFixTableVersionForOldFirmware(tKFWUM_InFirmwareInfo * pInfo)
  1004. {
  1005. switch(pInfo->boardId) {
  1006. case KFWUM_BOARD_KONTRON_UNKNOWN:
  1007. pInfo->tableVers = 0xff;
  1008. break;
  1009. default:
  1010. /* pInfo->tableVers is already set for
  1011. * the right version
  1012. */
  1013. break;
  1014. }
  1015. }
  1016. /* ipmi_kfwum_checkfwcompat - check whether firmware we're about to upload is
  1017. * compatible with board.
  1018. *
  1019. * @boardInfo:
  1020. * @firmInfo:
  1021. *
  1022. * returns 0 if compatible, otherwise (-1)
  1023. */
  1024. int
  1025. ipmi_kfwum_checkfwcompat(tKFWUM_BoardInfo boardInfo,
  1026. tKFWUM_InFirmwareInfo firmInfo)
  1027. {
  1028. int compatible = 0;
  1029. if (boardInfo.iana != firmInfo.iana) {
  1030. lprintf(LOG_ERR,
  1031. "Board IANA does not match firmware IANA.");
  1032. compatible = (-1);
  1033. }
  1034. if (boardInfo.boardId != firmInfo.boardId) {
  1035. lprintf(LOG_ERR,
  1036. "Board IANA does not match firmware IANA.");
  1037. compatible = (-1);
  1038. }
  1039. if (compatible != 0) {
  1040. lprintf(LOG_ERR,
  1041. "Firmware invalid for target board. Download of upgrade aborted.");
  1042. }
  1043. return compatible;
  1044. }
  1045. void
  1046. printf_kfwum_info(tKFWUM_BoardInfo boardInfo, tKFWUM_InFirmwareInfo firmInfo)
  1047. {
  1048. printf(
  1049. "Target Board Id : %u\n", boardInfo.boardId);
  1050. printf(
  1051. "Target IANA number : %u\n", boardInfo.iana);
  1052. printf(
  1053. "File Size : %lu bytes\n", firmInfo.fileSize);
  1054. printf(
  1055. "Firmware Version : %d.%d%d SDR %d\n", firmInfo.versMajor,
  1056. firmInfo.versMinor, firmInfo.versSubMinor, firmInfo.sdrRev);
  1057. }