summaryrefslogtreecommitdiff
path: root/util/libreboot-utils/nvmutil.c
blob: d78ab0c8a6dff9ee5bf1f0510278657692ac3175 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
/* SPDX-License-Identifier: MIT
 * Copyright (c) 2022-2026 Leah Rowe <leah@libreboot.org>
 *
 * This tool lets you modify Intel GbE NVM (Gigabit Ethernet
 * Non-Volatile Memory) images, e.g. change the MAC address.
 * These images configure your Intel Gigabit Ethernet adapter.
 */

#include <sys/types.h>
#include <sys/stat.h>

#include <errno.h>
#include <fcntl.h>
#include <limits.h>
#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#include "include/common.h"

int
main(int argc, char *argv[])
{
	struct xstate *x;

	struct commands *cmd;
	struct xfile *f;

	size_t c;

	if (lbgetprogname(argv[0]) == NULL)
		err_no_cleanup(0, errno, "could not set progname");

	xpledgex("stdio flock rpath wpath cpath unveil", NULL);
	xunveilx("/dev/urandom", "r");

#ifndef S_ISREG
	err_no_cleanup(0, ECANCELED,
	    "Can't determine file types (S_ISREG undefined)");
#endif
#if ((CHAR_BIT) != 8)
	err_no_cleanup(0, ECANCELED, "Unsupported char size");
#endif

	if ((x = xstart(argc, argv)) == NULL)
		err_no_cleanup(0, ECANCELED, "NULL state on init");

	/* parse user command */
/* TODO: CHECK ACCESSES VIA xstatus() */
	set_cmd(argc, argv);
	set_cmd_args(argc, argv);

	cmd = &x->cmd[x->i];
	f = &x->f;

	if ((cmd->flags & O_ACCMODE) == O_RDONLY)
		xunveilx(f->fname, "r");
	else
		xunveilx(f->fname, "rwc");

	xunveilx(f->tname, "rwc");
	xunveilx(NULL, NULL);
	xpledgex("stdio flock rpath wpath cpath", NULL);

	if (cmd->run == NULL)
		b0rk(errno, "Command not set");

	sanitize_command_list();

	open_gbe_file();

	copy_gbe();
	read_checksums();

	cmd->run();

	for (c = 0; c < items(x->cmd); c++)
		x->cmd[c].run = cmd_helper_err;

	if ((cmd->flags & O_ACCMODE) == O_RDWR)
		write_to_gbe_bin();

	if (exit_cleanup() == -1)
		b0rk(EIO, "%s: close", f->fname);

	if (f->io_err_gbe_bin)
		b0rk(EIO, "%s: error writing final file");

	free_and_set_null(&f->tname);

	return EXIT_SUCCESS;
}