Files
lbmk/util/nvmutil/nvmutil.c
Leah Rowe 0d3a8749fe util/nvmutil: usleep 1 on check_read_or_die
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>
2026-03-04 00:57:48 +00:00

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;
}