summaryrefslogtreecommitdiff
path: root/util/libreboot-utils/nvmutil.c
blob: cb08ec43de6b20341bdb583b24c41d2d08084c59 (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
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
/* 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.
 */

#ifdef __OpenBSD__
/* for pledge/unveil test:
 */
#include <sys/param.h>
#endif

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

/* https://man.openbsd.org/pledge.2
   https://man.openbsd.org/unveil.2     */
#if defined(__OpenBSD__) && defined(OpenBSD)
#if (OpenBSD) >= 604
	if (pledge("stdio flock rpath wpath cpath unveil", NULL) == -1)
		err_no_cleanup(errno, "pledge plus unveil, main");
	if (unveil("/dev/null", "r") == -1)
		err_no_cleanup(errno, "unveil r: /dev/null");
#elif (OpenBSD) >= 509
	if (pledge("stdio flock rpath wpath cpath", NULL) == -1)
		err_no_cleanup(errno, "pledge, main");
#endif
#endif

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

	x = xstart(argc, argv);

	if (x == NULL)
		err_no_cleanup(ECANCELED, "NULL state on init");

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

/* https://man.openbsd.org/pledge.2
   https://man.openbsd.org/unveil.2     */
#if defined(__OpenBSD__) && defined(OpenBSD)
#if (OpenBSD) >= 604

	if ((us.cmd[i].flags & O_ACCMODE) == O_RDONLY) {
		if (unveil(us.f.fname, "r") == -1)
			err(errno, "%s: unveil r", us.f.fname);
	} else {
		if (unveil(us.f.fname, "rwc") == -1)
			err(errno, "%s: unveil rw", us.f.fname);
	}

	if (unveil(us.f.tname, "rwc") == -1)
		err(errno, "unveil rwc: %s", us.f.tname);

	if (unveil(NULL, NULL) == -1)
		err(errno, "unveil block (rw)");

	if (pledge("stdio flock rpath wpath cpath", NULL) == -1)
		err(errno, "pledge (kill unveil)");

#elif (OpenBSD) >= 509
	if (pledge("stdio flock rpath wpath cpath", NULL) == -1)
		err(errno, "pledge");
#endif
#endif

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

	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)
		err(EIO, "%s: close", f->fname);

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

	if (f->tname != NULL) {
		free(f->tname);
		f->tname = NULL;
	}

	return EXIT_SUCCESS;
}