mirror of
https://codeberg.org/libreboot/lbmk.git
synced 2026-03-28 23:19:03 +02:00
This prevents hogging the CPU in a tight loop, while waiting for access. I've also reduced the number of tries to 30, rather than 200. This is more conservative, while still being somewhat permissive. The addition of the usleep delay probably makes this more reliable than the previous behaviour of quickly spinning through 200 tries, but without hogging CPU resources. I *could* allow this loop to be infinite, but I regard infinite spin-lock as an error state. Signed-off-by: Leah Rowe <leah@libreboot.org>
783 lines
15 KiB
C
783 lines
15 KiB
C
/* SPDX-License-Identifier: MIT */
|
|
/* Copyright (c) 2022-2026 Leah Rowe <leah@libreboot.org> */
|
|
/* Copyright (c) 2023 Riku Viitanen <riku.viitanen@protonmail.com> */
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <stdarg.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
|
|
static void reset_global_state(void);
|
|
static void set_cmd(int, char **);
|
|
static void check_cmd_args(int, char **);
|
|
static void set_io_flags(int, char **);
|
|
static void open_files(void);
|
|
static void xopen(int *, const char *, int, struct stat *);
|
|
static void read_gbe(void);
|
|
static void read_gbe_part(int, int);
|
|
static void cmd_setmac(void);
|
|
static void parse_mac_string(void);
|
|
static void set_mac_byte(int);
|
|
static void check_mac_separator(int);
|
|
static void set_mac_nib(int, int, uint8_t *);
|
|
static uint8_t hextonum(char);
|
|
static uint8_t rhex(void);
|
|
static void read_urandom(uint8_t *, size_t);
|
|
static int check_read_or_die(const char *,
|
|
ssize_t, size_t, int, const char *);
|
|
static int write_mac_part(int);
|
|
static void cmd_dump(void);
|
|
static void print_mac_address(int);
|
|
static void hexdump(int);
|
|
static void cmd_setchecksum(void);
|
|
static void cmd_brick(void);
|
|
static void cmd_copy(void);
|
|
static void cmd_swap(void);
|
|
static int good_checksum(int);
|
|
static uint16_t word(int, int);
|
|
static void set_word(int, int, uint16_t);
|
|
static void check_bound(int, int);
|
|
static void write_gbe(void);
|
|
static void write_gbe_part(int);
|
|
static void swap(int);
|
|
static void usage(void);
|
|
static void err(int, const char *, ...);
|
|
static const char *getnvmprogname(void);
|
|
static void set_err(int);
|
|
|
|
#define NVM_CHECKSUM 0xBABA
|
|
#define NVM_CHECKSUM_WORD 0x3F
|
|
#define NVM_SIZE 128
|
|
|
|
#define SIZE_4KB 0x1000
|
|
#define SIZE_8KB 0x2000
|
|
#define SIZE_16KB 0x4000
|
|
#define SIZE_128KB 0x20000
|
|
|
|
#define MAX_RETRY_READ 30
|
|
|
|
#define items(x) (sizeof((x)) / sizeof((x)[0]))
|
|
|
|
static uint8_t buf[SIZE_8KB];
|
|
static uint16_t macbuf[3];
|
|
static off_t partsize;
|
|
|
|
static int flags;
|
|
static int rfd = -1;
|
|
static int fd = -1;
|
|
static int part;
|
|
static int invert;
|
|
static int part_modified[2];
|
|
|
|
static const char *mac = NULL;
|
|
static const char *rmac = "xx:xx:xx:xx:xx:xx";
|
|
static const char *fname = "";
|
|
static const char *argv0;
|
|
|
|
struct op {
|
|
const char *str;
|
|
void (*cmd)(void);
|
|
int args;
|
|
};
|
|
static const struct op ops[] = {
|
|
{ "dump", cmd_dump, 3 },
|
|
{ "setmac", cmd_setmac, 3 },
|
|
{ "swap", cmd_swap, 3 },
|
|
{ "copy", cmd_copy, 4 },
|
|
{ "brick", cmd_brick, 4 },
|
|
{ "setchecksum", cmd_setchecksum, 4 },
|
|
};
|
|
|
|
static void (*cmd)(void) = NULL;
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
argv0 = argv[0];
|
|
if (argc < 2)
|
|
usage();
|
|
|
|
reset_global_state();
|
|
fname = argv[1];
|
|
|
|
#ifdef __OpenBSD__
|
|
if (pledge("stdio rpath wpath unveil", NULL) == -1)
|
|
err(ECANCELED, "pledge");
|
|
if (unveil("/dev/urandom", "r") == -1)
|
|
err(ECANCELED, "unveil '/dev/urandom'");
|
|
#endif
|
|
|
|
set_cmd(argc, argv);
|
|
check_cmd_args(argc, argv);
|
|
set_io_flags(argc, argv);
|
|
|
|
#ifdef __OpenBSD__
|
|
if (flags == O_RDONLY) {
|
|
if (unveil(fname, "r") == -1)
|
|
err(ECANCELED, "unveil ro '%s'", fname);
|
|
if (unveil(NULL, NULL) == -1)
|
|
err(ECANCELED, "unveil block (ro)");
|
|
if (pledge("stdio rpath", NULL) == -1)
|
|
err(ECANCELED, "pledge ro (kill unveil)");
|
|
} else {
|
|
if (unveil(fname, "rw") == -1)
|
|
err(ECANCELED, "unveil rw '%s'", fname);
|
|
if (unveil(NULL, NULL) == -1)
|
|
err(ECANCELED, "unveil block (rw)");
|
|
if (pledge("stdio rpath wpath", NULL) == -1)
|
|
err(ECANCELED, "pledge rw (kill unveil)");
|
|
}
|
|
#endif
|
|
|
|
open_files();
|
|
|
|
#ifdef __OpenBSD__
|
|
if (pledge("stdio", NULL) == -1)
|
|
err(ECANCELED, "pledge stdio (main)");
|
|
#endif
|
|
|
|
read_gbe();
|
|
(*cmd)();
|
|
write_gbe();
|
|
|
|
if (close(fd) == -1)
|
|
err(ECANCELED, "close '%s'", fname);
|
|
if (close(rfd) == -1)
|
|
err(ECANCELED, "close '/dev/urandom'");
|
|
|
|
if (cmd != cmd_dump) {
|
|
if (errno)
|
|
err(ECANCELED, "Unhandled error on exit");
|
|
}
|
|
|
|
if (errno)
|
|
return EXIT_FAILURE;
|
|
else
|
|
return EXIT_SUCCESS;
|
|
}
|
|
|
|
static void
|
|
reset_global_state(void)
|
|
{
|
|
errno = 0;
|
|
|
|
mac = NULL;
|
|
invert = 0;
|
|
part_modified[0] = 0;
|
|
part_modified[1] = 0;
|
|
fname = "";
|
|
cmd = NULL;
|
|
fd = -1;
|
|
rfd = -1;
|
|
part = 0;
|
|
|
|
memset(macbuf, 0, sizeof(macbuf));
|
|
memset(buf, 0, sizeof(buf));
|
|
}
|
|
|
|
static void
|
|
set_cmd(int argc, char *argv[])
|
|
{
|
|
size_t i;
|
|
|
|
if (argc == 2) {
|
|
cmd = cmd_setmac;
|
|
return;
|
|
}
|
|
|
|
for (i = 0; (i < items(ops)) && (cmd == NULL); i++) {
|
|
if (strcmp(argv[2], ops[i].str) != 0)
|
|
continue;
|
|
if (argc >= ops[i].args) {
|
|
cmd = ops[i].cmd;
|
|
break;
|
|
}
|
|
err(EINVAL, "Too few args: command '%s'", ops[i].str);
|
|
}
|
|
}
|
|
|
|
static void
|
|
check_cmd_args(int argc, char *argv[])
|
|
{
|
|
if ((cmd == NULL) && (argc > 2)) { /* nvm gbe [MAC] */
|
|
mac = argv[2];
|
|
cmd = cmd_setmac;
|
|
} else if (cmd == cmd_setmac) { /* nvm gbe setmac [MAC] */
|
|
mac = rmac; /* random MAC */
|
|
if (argc > 3)
|
|
mac = argv[3];
|
|
} else if ((cmd != NULL) && (argc > 3)) { /* user-supplied partnum */
|
|
part = argv[3][0] - '0';
|
|
if (!((part == 0 || part == 1) && argv[3][1] == '\0'))
|
|
err(EINVAL, "Bad partnum: %s", argv[3]);
|
|
}
|
|
|
|
if (cmd == NULL)
|
|
err(EINVAL, "Bad command");
|
|
}
|
|
|
|
static void
|
|
set_io_flags(int argc, char *argv[])
|
|
{
|
|
flags = O_RDWR;
|
|
if (argc > 2) {
|
|
if (strcmp(argv[2], "dump") == 0)
|
|
flags = O_RDONLY;
|
|
}
|
|
}
|
|
|
|
static void
|
|
open_files(void)
|
|
{
|
|
struct stat st;
|
|
struct stat st_rfd;
|
|
|
|
xopen(&rfd, "/dev/urandom", O_RDONLY, &st_rfd);
|
|
xopen(&fd, fname, flags, &st);
|
|
|
|
switch(st.st_size) {
|
|
case SIZE_8KB:
|
|
case SIZE_16KB:
|
|
case SIZE_128KB:
|
|
partsize = st.st_size >> 1;
|
|
break;
|
|
default:
|
|
err(ECANCELED, "File size must be 8KB, 16KB or 128KB");
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
xopen(int *f, const char *l, int p, struct stat *st)
|
|
{
|
|
if ((*f = open(l, p)) == -1)
|
|
err(ECANCELED, "%s", l);
|
|
if (fstat(*f, st) == -1)
|
|
err(ECANCELED, "%s", l);
|
|
}
|
|
|
|
static void
|
|
read_gbe(void)
|
|
{
|
|
int p;
|
|
int do_read[2] = {1, 1};
|
|
|
|
if ((cmd == cmd_copy) || (cmd == cmd_brick) ||
|
|
(cmd == cmd_setchecksum))
|
|
do_read[part ^ 1] = 0;
|
|
|
|
/*
|
|
* speedhack: if copy/swap, flip where data gets written to memory,
|
|
* so that cmd_copy and cmd_swap don't have to work on every word
|
|
*/
|
|
if ((cmd == cmd_copy) || (cmd == cmd_swap))
|
|
invert = 1;
|
|
|
|
for (p = 0; p < 2; p++) {
|
|
if (do_read[p])
|
|
read_gbe_part(p, invert);
|
|
}
|
|
}
|
|
|
|
static void
|
|
read_gbe_part(int p, int invert)
|
|
{
|
|
int retry;
|
|
ssize_t rval;
|
|
|
|
for (retry = 0; retry < MAX_RETRY_READ; retry++) {
|
|
rval = pread(fd, buf + (SIZE_4KB * (p ^ invert)),
|
|
SIZE_4KB, ((off_t ) p) * partsize);
|
|
|
|
if (check_read_or_die(fname, rval, SIZE_4KB, retry, "pread"))
|
|
break;
|
|
}
|
|
|
|
swap(p ^ invert); /* handle big-endian host CPU */
|
|
}
|
|
|
|
static void
|
|
cmd_setmac(void)
|
|
{
|
|
int partnum;
|
|
int mac_updated = 0;
|
|
|
|
parse_mac_string();
|
|
printf("MAC address to be written: %s\n", mac);
|
|
|
|
for (partnum = 0; partnum < 2; partnum++)
|
|
mac_updated |= write_mac_part(partnum);
|
|
if (mac_updated)
|
|
errno = 0;
|
|
}
|
|
|
|
static void
|
|
parse_mac_string(void)
|
|
{
|
|
size_t c;
|
|
int mac_pos;
|
|
uint64_t mac_total;
|
|
|
|
if (strnlen(mac, 20) != 17)
|
|
err(EINVAL, "MAC address is the wrong length");
|
|
|
|
for (mac_pos = 0; mac_pos < 16; mac_pos += 3)
|
|
set_mac_byte(mac_pos);
|
|
|
|
mac_total = 0;
|
|
for (c = 0; c < 3; c++)
|
|
mac_total += macbuf[c];
|
|
|
|
if (mac_total == 0)
|
|
err(EINVAL, "Must not specify all-zeroes MAC address");
|
|
if (macbuf[0] & 1)
|
|
err(EINVAL, "Must not specify multicast MAC address");
|
|
}
|
|
|
|
static void
|
|
set_mac_byte(int mac_pos)
|
|
{
|
|
int nib;
|
|
uint8_t h = 0;
|
|
check_mac_separator(mac_pos);
|
|
|
|
for (nib = 0; nib < 2; nib++)
|
|
set_mac_nib(mac_pos, nib, &h);
|
|
}
|
|
|
|
static void
|
|
check_mac_separator(int mac_pos)
|
|
{
|
|
char separator;
|
|
|
|
if (mac_pos == 15)
|
|
return;
|
|
if ((separator = mac[mac_pos + 2]) == ':')
|
|
return;
|
|
|
|
err(EINVAL, "Invalid MAC address separator '%c'", separator);
|
|
}
|
|
|
|
static void
|
|
set_mac_nib(int mac_pos, int nib, uint8_t *h)
|
|
{
|
|
int byte = mac_pos / 3;
|
|
|
|
if ((*h = hextonum(mac[mac_pos + nib])) > 15)
|
|
err(EINVAL, "Invalid character '%c'",
|
|
mac[mac_pos + nib]);
|
|
|
|
/* If random, ensure that local/unicast bits are set */
|
|
if ((byte == 0) && (nib == 1)) {
|
|
if ((mac[mac_pos + nib] == '?') ||
|
|
(mac[mac_pos + nib] == 'x') ||
|
|
(mac[mac_pos + nib] == 'X')) /* random */
|
|
*h = (*h & 0xE) | 2; /* local, unicast */
|
|
}
|
|
|
|
macbuf[byte >> 1] |= ((uint16_t ) *h) << ((8 * (byte % 2)) +
|
|
(4 * (nib ^ 1)));
|
|
}
|
|
|
|
static uint8_t
|
|
hextonum(char ch)
|
|
{
|
|
if ((ch >= '0') && (ch <= '9'))
|
|
return ch - '0';
|
|
else if ((ch >= 'A') && (ch <= 'F'))
|
|
return ch - 'A' + 10;
|
|
else if ((ch >= 'a') && (ch <= 'f'))
|
|
return ch - 'a' + 10;
|
|
else if ((ch == '?') || (ch == 'x') || (ch == 'X'))
|
|
return rhex(); /* random hex value */
|
|
else
|
|
return 16; /* error: invalid character */
|
|
}
|
|
|
|
static uint8_t
|
|
rhex(void)
|
|
{
|
|
static uint8_t n = 0;
|
|
static uint8_t rnum[12];
|
|
|
|
if (!n) {
|
|
n = sizeof(rnum) - 1;
|
|
read_urandom(rnum, sizeof(rnum));
|
|
}
|
|
|
|
return rnum[n--] & 0xf;
|
|
}
|
|
|
|
static void
|
|
read_urandom(uint8_t *rnum, size_t rsize)
|
|
{
|
|
int retry = 0;
|
|
ssize_t rval;
|
|
|
|
for (retry = 0; retry < MAX_RETRY_READ; retry++) {
|
|
rval = read(rfd, rnum, rsize);
|
|
if (check_read_or_die("/dev/urandom", rval,
|
|
rsize, retry, "read"))
|
|
break;
|
|
}
|
|
}
|
|
|
|
static int
|
|
check_read_or_die(const char *rpath, ssize_t rval, size_t rsize,
|
|
int retry, const char *readtype)
|
|
{
|
|
if (rval == (ssize_t) rsize) {
|
|
errno = 0;
|
|
return 1; /* Successful read */
|
|
}
|
|
|
|
if (rval != -1)
|
|
err(ECANCELED, "Short %s, %zd bytes, on file: %s",
|
|
readtype, rval, rpath);
|
|
if (errno != EINTR)
|
|
err(ECANCELED, "Could not %s file: '%s'", readtype, rpath);
|
|
if (retry == MAX_RETRY_READ - 1)
|
|
err(EINTR, "%s: max retries exceeded on file: %s",
|
|
readtype, rpath);
|
|
|
|
/* Prevent CPU hog when on spin-locked reads. */
|
|
usleep(1);
|
|
|
|
/*
|
|
* Bad read, with errno EINTR (syscall interrupted).
|
|
* Reset the error state and try again.
|
|
*/
|
|
errno = 0;
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
write_mac_part(int partnum)
|
|
{
|
|
int w;
|
|
|
|
part = partnum;
|
|
if (!good_checksum(partnum))
|
|
return 0;
|
|
|
|
for (w = 0; w < 3; w++)
|
|
set_word(w, partnum, macbuf[w]);
|
|
|
|
printf("Wrote MAC address to part %d: ", partnum);
|
|
print_mac_address(partnum);
|
|
|
|
cmd_setchecksum();
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
cmd_dump(void)
|
|
{
|
|
int partnum;
|
|
int num_invalid = 0;
|
|
|
|
for (partnum = 0; partnum < 2; partnum++) {
|
|
if (!good_checksum(partnum))
|
|
++num_invalid;
|
|
|
|
printf("MAC (part %d): ", partnum);
|
|
print_mac_address(partnum);
|
|
hexdump(partnum);
|
|
}
|
|
|
|
if ((num_invalid < 2))
|
|
errno = 0;
|
|
}
|
|
|
|
static void
|
|
print_mac_address(int partnum)
|
|
{
|
|
int c;
|
|
for (c = 0; c < 3; c++) {
|
|
uint16_t val16 = word(c, partnum);
|
|
printf("%02x:%02x", val16 & 0xff, val16 >> 8);
|
|
if (c == 2)
|
|
printf("\n");
|
|
else
|
|
printf(":");
|
|
}
|
|
}
|
|
|
|
static void
|
|
hexdump(int partnum)
|
|
{
|
|
int c;
|
|
int row;
|
|
|
|
for (row = 0; row < 8; row++) {
|
|
printf("%08x ", row << 4);
|
|
for (c = 0; c < 8; c++) {
|
|
uint16_t val16 = word((row << 3) + c, partnum);
|
|
if (c == 4)
|
|
printf(" ");
|
|
printf(" %02x %02x", val16 & 0xff, val16 >> 8);
|
|
}
|
|
printf("\n");
|
|
}
|
|
}
|
|
|
|
static void
|
|
cmd_setchecksum(void)
|
|
{
|
|
int c;
|
|
uint16_t val16 = 0;
|
|
|
|
for (c = 0; c < NVM_CHECKSUM_WORD; c++)
|
|
val16 += word(c, part);
|
|
|
|
set_word(NVM_CHECKSUM_WORD, part, NVM_CHECKSUM - val16);
|
|
}
|
|
|
|
static void
|
|
cmd_brick(void)
|
|
{
|
|
if (!good_checksum(part))
|
|
err(ECANCELED, "brick p%d, file '%s'", part, fname);
|
|
|
|
set_word(NVM_CHECKSUM_WORD, part,
|
|
((word(NVM_CHECKSUM_WORD, part)) ^ 0xFF));
|
|
}
|
|
|
|
static void
|
|
cmd_copy(void)
|
|
{
|
|
if (!good_checksum(part ^ 1))
|
|
err(ECANCELED, "copy p%d, file '%s'", part ^ 1, fname);
|
|
|
|
/*
|
|
* SPEED HACK:
|
|
*
|
|
* read_gbe() already performed the copy,
|
|
* by virtue of inverted read. We need
|
|
* only set the other part as changed.
|
|
*
|
|
* THIS IS NOT A BUG!
|
|
*/
|
|
part_modified[part ^ 1] = 1;
|
|
}
|
|
|
|
static void
|
|
cmd_swap(void)
|
|
{
|
|
if (!(good_checksum(0) || good_checksum(1)))
|
|
err(ECANCELED, "swap parts, file '%s'", fname);
|
|
|
|
/*
|
|
* good_checksum() can set errno, if one
|
|
* of the parts is bad. We will reset it.
|
|
*/
|
|
errno = 0;
|
|
|
|
/*
|
|
* SPEED HACK:
|
|
*
|
|
* read_gbe() already performed the swap,
|
|
* by virtue of inverted read. We need
|
|
* only set both parts as changed.
|
|
*
|
|
* THIS IS NOT A BUG!
|
|
*/
|
|
part_modified[1] = part_modified[0] = 1;
|
|
}
|
|
|
|
static int
|
|
good_checksum(int partnum)
|
|
{
|
|
int w;
|
|
uint16_t total = 0;
|
|
for (w = 0; w <= NVM_CHECKSUM_WORD; w++)
|
|
total += word(w, partnum);
|
|
|
|
if (total == NVM_CHECKSUM)
|
|
return 1;
|
|
|
|
fprintf(stderr, "WARNING: BAD checksum in part %d\n",
|
|
partnum ^ invert);
|
|
|
|
set_err(ECANCELED);
|
|
return 0;
|
|
}
|
|
|
|
static uint16_t
|
|
word(int pos16, int p)
|
|
{
|
|
uint16_t rval = 0;
|
|
|
|
check_bound(pos16, p);
|
|
memcpy(&rval, buf + (SIZE_4KB * p) + (pos16 << 1), sizeof(uint16_t));
|
|
|
|
return rval;
|
|
}
|
|
|
|
static void
|
|
set_word(int pos16, int p, uint16_t val16)
|
|
{
|
|
check_bound(pos16, p);
|
|
memcpy(buf + (SIZE_4KB * p) + (pos16 << 1), &val16, sizeof(uint16_t));
|
|
|
|
part_modified[p] = 1;
|
|
}
|
|
|
|
static void
|
|
check_bound(int c, int p)
|
|
{
|
|
/*
|
|
* NVM_SIZE assumed as the limit, because the
|
|
* current design assumes that we will only
|
|
* ever modified the NVM area.
|
|
*
|
|
* The only exception is copy/swap, but these
|
|
* do not use word/set_word and therefore do
|
|
* not cause check_bound() to be called.
|
|
*
|
|
* TODO:
|
|
* This should be adjusted in the future, if
|
|
* we ever wish to work on the Extented NVM.
|
|
*/
|
|
|
|
if ((p != 0) && (p != 1))
|
|
err(EINVAL, "check_bound: invalid partnum %d", p);
|
|
if ((c < 0) || (c >= (NVM_SIZE >> 1)))
|
|
err(EINVAL, "check_bound: out of bounds %d", c);
|
|
}
|
|
|
|
static void
|
|
write_gbe(void)
|
|
{
|
|
int p;
|
|
|
|
if (flags == O_RDONLY)
|
|
return;
|
|
|
|
for (p = 0; p < 2; p++) {
|
|
if (part_modified[p])
|
|
write_gbe_part(p);
|
|
}
|
|
}
|
|
|
|
static void
|
|
write_gbe_part(int p)
|
|
{
|
|
swap(p); /* swap bytes on big-endian host CPUs */
|
|
|
|
if (pwrite(fd, buf + (SIZE_4KB * p),
|
|
SIZE_4KB, ((off_t) p) * partsize) != SIZE_4KB) {
|
|
err(ECANCELED,
|
|
"Can't write %d b to '%s' p%d", SIZE_4KB, fname, p);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* GbE files store bytes in little-endian order.
|
|
* This function ensures big-endian host CPU support.
|
|
*/
|
|
static void
|
|
swap(int partnum)
|
|
{
|
|
/*
|
|
* NVM_SIZE assumed as the limit; see notes in
|
|
* check_bound().
|
|
*
|
|
* TODO:
|
|
* This should be adjusted in the future, if
|
|
* we ever wish to work on the Extended NVM.
|
|
*/
|
|
|
|
size_t w;
|
|
size_t x;
|
|
|
|
int e = 1;
|
|
uint8_t *n = buf + (SIZE_4KB * partnum);
|
|
|
|
if (((uint8_t *) &e)[0] == 1)
|
|
return; /* Little-endian host CPU; no swap needed. */
|
|
|
|
/*
|
|
* The host CPU stores bytes in big-endian order.
|
|
* We will therefore reverse the order in memory:
|
|
*/
|
|
for (w = 0, x = 1; w < NVM_SIZE; w += 2, x += 2) {
|
|
uint8_t chg = n[w];
|
|
n[w] = n[x];
|
|
n[x] = chg;
|
|
}
|
|
}
|
|
|
|
static void
|
|
usage(void)
|
|
{
|
|
const char *util = getnvmprogname();
|
|
|
|
#ifdef __OpenBSD__
|
|
if (pledge("stdio", NULL) == -1)
|
|
err(ECANCELED, "pledge");
|
|
#endif
|
|
fprintf(stderr,
|
|
"Modify Intel GbE NVM images e.g. set MAC\n"
|
|
"USAGE:\n"
|
|
"\t%s FILE dump\n"
|
|
"\t%s FILE # same as setmac without [MAC]\n"
|
|
"\t%s FILE setmac [MAC]\n"
|
|
"\t%s FILE swap\n"
|
|
"\t%s FILE copy 0|1\n"
|
|
"\t%s FILE brick 0|1\n"
|
|
"\t%s FILE setchecksum 0|1\n",
|
|
util, util, util, util, util, util, util);
|
|
|
|
err(ECANCELED, "Too few arguments");
|
|
}
|
|
|
|
static void
|
|
err(int nvm_errval, const char *msg, ...)
|
|
{
|
|
va_list args;
|
|
|
|
fprintf(stderr, "%s: ", getnvmprogname());
|
|
|
|
va_start(args, msg);
|
|
vfprintf(stderr, msg, args);
|
|
va_end(args);
|
|
|
|
set_err(nvm_errval);
|
|
fprintf(stderr, ": %s", strerror(errno));
|
|
|
|
fprintf(stderr, "\n");
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
static const char *
|
|
getnvmprogname(void)
|
|
{
|
|
const char *p;
|
|
|
|
if ((argv0 == NULL) || (*argv0 == '\0'))
|
|
return "";
|
|
|
|
if ((p = strrchr(argv0, '/')))
|
|
return p + 1;
|
|
else
|
|
return argv0;
|
|
}
|
|
|
|
static void
|
|
set_err(int x)
|
|
{
|
|
if (errno)
|
|
return;
|
|
if (x)
|
|
errno = x;
|
|
else
|
|
errno = ECANCELED;
|
|
}
|