summaryrefslogtreecommitdiff
path: root/cmd/shutdown.c
blob: 546b076a26a6e8572b8b1c369b0b8a7534b86749 (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
/* SPDX-License-Identifier: ISC */
#include <getopt.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>
#include <stdio.h>
#include <errno.h>

#include <sys/reboot.h>
#include <linux/reboot.h>

#include "util.h"

#define FL_FORCE 0x01
#define FL_NOSYNC 0x02

static const struct option options[] = {
	{ "help", no_argument, NULL, 'h' },
	{ "poweroff", no_argument, NULL, 'p' },
	{ "reboot", no_argument, NULL, 'r' },
	{ "force", no_argument, NULL, 'f' },
	{ "no-sync", no_argument, NULL, 'n' },
	{ NULL, 0, NULL, 0 },
};

static const char *shortopt = "hprfn";

static const char *defact_str = "power-off";
static int defact = RB_POWER_OFF;

static NORETURN void usage(const char *progname, int status)
{
	fprintf(status == EXIT_SUCCESS ? stdout : stderr,
"%s [OPTIONS...]\n\n"
"Perform a system shutdown or reboot.\n\n"
"   -h, --help      Display this help text and exit.\n"
"   -V, --version   Display version information and exit.\n"
"   -p, --poweroff  Power-off the machine.\n"
"   -r, --reboot    Reboot the machine.\n"
"   -f, --force     Force immediate power-off or reboot. Do not contact the\n"
"                   init system.\n"
"   -n, --no-sync   Don't sync storage media before power-off or reboot.\n\n"
"If no option is specified, the default action is %s.\n",
	progname, defact_str);
	exit(status);
}

int main(int argc, char **argv)
{
	int c, ret, flags = 0;
	char *ptr;

	ptr = strrchr(argv[0], '/');
	ptr = (ptr == NULL) ? argv[0] : (ptr + 1);

	if (strcmp(ptr, "reboot") == 0) {
		defact_str = "reboot";
		defact = RB_AUTOBOOT;
	}

	while (1) {
		c = getopt_long(argc, argv, shortopt, options, NULL);
		if (c == -1)
			break;

		switch (c) {
		case 'f':
			flags |= FL_FORCE;
			break;
		case 'n':
			flags |= FL_NOSYNC;
			break;
		case 'p':
			defact = RB_POWER_OFF;
			break;
		case 'r':
			defact = RB_AUTOBOOT;
			break;
		case 'h':
			usage(ptr, EXIT_SUCCESS);
		default:
			usage(ptr, EXIT_FAILURE);
		}
	}

	if (flags & FL_FORCE) {
		if (!(flags & FL_NOSYNC))
			sync();
		reboot(defact);
		perror("reboot system call");
		return EXIT_FAILURE;
	}

	switch (defact) {
	case RB_AUTOBOOT:
		ret = kill(1, SIGINT);
		break;
	case RB_POWER_OFF:
		ret = kill(1, SIGTERM);
		break;
	default:
		return EXIT_SUCCESS;
	}

	if (ret) {
		perror("sending signal to init");
		return EXIT_FAILURE;
	}
	return EXIT_SUCCESS;
}