2 * Copyright (c) 2015 Baptiste Daroussin <bapt@FreeBSD.org>
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 #include <sys/cdefs.h>
30 #include <sys/param.h>
32 #include <sys/endian.h>
45 MPS_TABLE(top, flash);
48 flash_save(int argc, char **argv)
50 const char *firmware_file;
51 unsigned char *firmware_buffer = NULL;
54 ssize_t written = 0, ret = 0;
57 warnx("missing argument: expecting 'firmware' or bios'");
61 if (strcmp(argv[1], "bios") == 0) {
63 } else if (strcmp(argv[1], "firmware") != 0) {
64 warnx("Invalid argument '%s', expecting 'firmware' or 'bios'",
69 warnx("save %s: extra arguments", argv[1]);
73 firmware_file = argv[1];
75 firmware_file = argv[2];
78 fd = mps_open(mps_unit);
85 if ((size = mps_firmware_get(fd, &firmware_buffer, bios)) < 0) {
86 warnx("Fail to save %s", argv[1]);
93 fd = open(firmware_file, O_CREAT | O_TRUNC | O_RDWR, 0644);
97 free(firmware_buffer);
100 while (written != size) {
101 if ((ret = write(fd, firmware_buffer + written, size - written)) <0) {
104 free(firmware_buffer);
112 free(firmware_buffer);
113 printf("%s successfully saved as %s\n", argv[1], firmware_file);
117 MPS_COMMAND(flash, save, flash_save, "[firmware|bios] [file]",
118 "Save firmware/bios into a file");
121 flash_update(int argc, char **argv)
124 unsigned char *mem = NULL;
127 MPI2_FW_IMAGE_HEADER *fwheader;
128 MPI2_IOC_FACTS_REPLY *facts;
131 warnx("missing argument: expecting 'firmware' or bios'");
135 if (strcmp(argv[1], "bios") == 0) {
137 } else if (strcmp(argv[1], "firmware") != 0) {
138 warnx("Invalid argument '%s', expecting 'firmware' or 'bios'",
143 warnx("update firmware: extra arguments");
148 warnx("no firmware specified");
152 if (stat(argv[2], &st) == -1) {
158 fd = open(argv[2], O_RDONLY);
165 mem = mmap(NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
166 if (mem == MAP_FAILED) {
174 fd = mps_open(mps_unit);
178 munmap(mem, st.st_size);
182 if ((facts = mps_get_iocfacts(fd)) == NULL) {
183 warnx("could not get controller IOCFacts\n");
184 munmap(mem, st.st_size);
190 /* Check boot record magic number */
191 if (((mem[0x01]<<8) + mem[0x00]) != 0xaa55) {
192 warnx("Invalid bios: no boot record magic number");
193 munmap(mem, st.st_size);
198 if ((st.st_size % 512) != 0) {
199 warnx("Invalid bios: size not a multiple of 512");
200 munmap(mem, st.st_size);
206 fwheader = (MPI2_FW_IMAGE_HEADER *)mem;
207 if (le16toh(fwheader->VendorID) != MPI2_MFGPAGE_VENDORID_LSI) {
208 warnx("Invalid firmware:");
209 warnx(" Expected Vendor ID: %04x",
210 MPI2_MFGPAGE_VENDORID_LSI);
211 warnx(" Image Vendor ID: %04x", le16toh(fwheader->VendorID));
212 munmap(mem, st.st_size);
218 if (le16toh(fwheader->ProductID) != facts->ProductID) {
219 warnx("Invalid image:");
220 warnx(" Expected Product ID: %04x", facts->ProductID);
221 warnx(" Image Product ID: %04x", le16toh(fwheader->ProductID));
222 munmap(mem, st.st_size);
229 printf("Updating %s...\n", argv[1]);
230 if (mps_firmware_send(fd, mem, st.st_size, bios) < 0) {
231 warnx("Fail to update %s", argv[1]);
232 munmap(mem, st.st_size);
238 munmap(mem, st.st_size);
241 printf("%s successfully updated\n", argv[1]);
245 MPS_COMMAND(flash, update, flash_update, "[firmware|bios] file",
246 "Update firmware/bios");