Skip to content

Commit

Permalink
Kernel: Only allow superuser to halt() the system (SerenityOS#342)
Browse files Browse the repository at this point in the history
Following the discussion in SerenityOS#334, shutdown must also have root-only
run permissions.
  • Loading branch information
Quaker762 authored and awesomekling committed Jul 19, 2019
1 parent cd76b69 commit a5d80f7
Show file tree
Hide file tree
Showing 6 changed files with 31 additions and 10 deletions.
15 changes: 15 additions & 0 deletions Kernel/Process.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -2639,6 +2639,21 @@ int Process::sys$systrace(pid_t pid)
return fd;
}

int Process::sys$halt()
{
if (!is_superuser())
return -EPERM;

dbgprintf("acquiring FS locks...\n");
FS::lock_all();
dbgprintf("syncing mounted filesystems...\n");
FS::sync();
dbgprintf("attempting system shutdown...\n");
IO::out16(0x604, 0x2000);

return ESUCCESS;
}

int Process::sys$reboot()
{
if (!is_superuser())
Expand Down
1 change: 1 addition & 0 deletions Kernel/Process.h
Original file line number Diff line number Diff line change
Expand Up @@ -203,6 +203,7 @@ class Process : public InlineLinkedListNode<Process>
int sys$release_shared_buffer(int shared_buffer_id);
int sys$seal_shared_buffer(int shared_buffer_id);
int sys$get_shared_buffer_size(int shared_buffer_id);
int sys$halt();
int sys$reboot();

static void initialize();
Expand Down
9 changes: 2 additions & 7 deletions Kernel/Syscall.cpp
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
#include <Kernel/Arch/i386/CPU.h>
#include <Kernel/Console.h>
#include <Kernel/Process.h>
#include <Kernel/IO.h>
#include <Kernel/Process.h>
#include <Kernel/ProcessTracer.h>
#include <Kernel/Scheduler.h>
#include <Kernel/Syscall.h>
Expand Down Expand Up @@ -285,12 +285,7 @@ static u32 handle(RegisterDump& regs, u32 function, u32 arg1, u32 arg2, u32 arg3
case Syscall::SC_sched_getparam:
return current->process().sys$sched_setparam((pid_t)arg1, (struct sched_param*)arg2);
case Syscall::SC_halt: {
dbgprintf("<%u> halting! acquiring locks...\n");
FS::lock_all();
dbgprintf("<%u> halting! syncing...\n");
FS::sync();
dbgprintf("<%u> halting! bye, friends...\n");
IO::out16(0x604, 0x2000);
return current->process().sys$halt();
break;
}
case Syscall::SC_reboot: {
Expand Down
6 changes: 6 additions & 0 deletions Libraries/LibC/unistd.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -515,6 +515,12 @@ int fsync(int fd)
return 0;
}

int halt()
{
int rc = syscall(SC_halt);
__RETURN_WITH_ERRNO(rc, rc, -1);
}

int reboot()
{
int rc = syscall(SC_reboot);
Expand Down
1 change: 1 addition & 0 deletions Libraries/LibC/unistd.h
Original file line number Diff line number Diff line change
Expand Up @@ -93,6 +93,7 @@ char* getlogin();
int chown(const char* pathname, uid_t, gid_t);
int fchown(int fd, uid_t, gid_t);
int ftruncate(int fd, off_t length);
int halt();
int reboot();

enum {
Expand Down
9 changes: 6 additions & 3 deletions Userland/shutdown.cpp
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
#include <Kernel/Syscall.h>
#include <LibCore/CArgsParser.h>
#include <stdio.h>
#include <unistd.h>

int main(int argc, char** argv)
{
Expand All @@ -8,8 +9,10 @@ int main(int argc, char** argv)
CArgsParserResult args = args_parser.parse(argc, argv);

if (args.is_present("n")) {
syscall(SC_halt);
return 0;
if (halt() < 0) {
perror("shutdown");
return 1;
}
} else {
args_parser.print_usage();
return 0;
Expand Down

0 comments on commit a5d80f7

Please sign in to comment.