| // SPDX-License-Identifier: Apache-2.0 |
| // Copyright (C) 2021 IBM Corp. |
| |
| #include <err.h> |
| #include <fcntl.h> |
| #include <getopt.h> |
| #include <libgen.h> |
| #include <limits.h> |
| #include <linux/reboot.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <sys/reboot.h> |
| #include <sys/stat.h> |
| #include <sys/types.h> |
| #include <unistd.h> |
| |
| static void process_debug(int sink) |
| { |
| static const char action = 'c'; |
| ssize_t rc; |
| |
| sync(); |
| |
| if ((rc = write(sink, &action, sizeof(action))) == sizeof(action)) |
| return; |
| |
| if (rc == -1) { |
| warn("Failed to execute debug command"); |
| } else { |
| warnx("Failed to execute debug command: %zd", rc); |
| } |
| } |
| |
| static void process_reboot(int sink __attribute__((unused))) |
| { |
| ssize_t rc; |
| |
| sync(); |
| |
| if ((rc = reboot(LINUX_REBOOT_CMD_RESTART))) { |
| if (rc == -1) |
| warn("Failed to reboot BMC"); |
| else |
| warnx("Failed to reboot BMC: %zd", rc); |
| } |
| } |
| |
| static int process(int source, int sink) |
| { |
| ssize_t ingress; |
| char command; |
| |
| while ((ingress = read(source, &command, sizeof(command))) == sizeof(command)) { |
| switch (command) { |
| case 'D': |
| process_debug(sink); |
| break; |
| case 'R': |
| process_reboot(sink); |
| break; |
| default: |
| warnx("Unexpected command: 0x%02x (%c)", command, command); |
| } |
| } |
| |
| if (ingress == -1) |
| warn("Failed to read from source"); |
| |
| return ingress; |
| } |
| |
| int main(int argc, char * const argv[]) |
| { |
| char devnode[PATH_MAX]; |
| char *devid; |
| int source; |
| int sink; |
| |
| while (1) { |
| static struct option long_options[] = { |
| {0, 0, 0, 0}, |
| }; |
| int c; |
| |
| c = getopt_long(argc, argv, "", long_options, NULL); |
| if (c == -1) |
| break; |
| } |
| |
| source = 0; |
| sink = 1; |
| |
| if (optind < argc) { |
| char devpath[PATH_MAX]; |
| |
| strncpy(devpath, argv[optind], sizeof(devpath)); |
| devpath[PATH_MAX - 1] = '\0'; |
| devid = basename(devpath); |
| |
| strncpy(devnode, "/dev/", sizeof(devnode)); |
| strncat(devnode, devid, sizeof(devnode)); |
| devnode[PATH_MAX - 1] = '\0'; |
| |
| if ((source = open(devnode, O_RDONLY)) == -1) |
| err(EXIT_FAILURE, "Failed to open %s", devnode); |
| |
| optind++; |
| } |
| |
| if (optind < argc) { |
| if ((sink = open(argv[optind], O_WRONLY)) == -1) |
| err(EXIT_FAILURE, "Failed to open %s", argv[optind]); |
| |
| optind++; |
| } |
| |
| if (optind < argc) |
| err(EXIT_FAILURE, "Found %d unexpected arguments", argc - optind); |
| |
| if (process(source, sink) < 0) |
| errx(EXIT_FAILURE, "Failure while processing command stream"); |
| |
| return 0; |
| } |