2023-05-02 08:50:39 +00:00
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
2023-04-22 11:56:08 +00:00
|
|
|
#include <luna/PathParser.h>
|
2023-05-02 18:55:41 +00:00
|
|
|
#include <luna/Sort.h>
|
2023-04-22 11:56:08 +00:00
|
|
|
#include <luna/String.h>
|
|
|
|
#include <luna/Vector.h>
|
2023-04-28 19:15:41 +00:00
|
|
|
#include <os/Directory.h>
|
2023-04-22 11:56:08 +00:00
|
|
|
#include <os/File.h>
|
|
|
|
#include <os/Process.h>
|
2023-03-18 18:23:18 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2023-04-22 11:56:08 +00:00
|
|
|
#include <string.h>
|
2023-06-02 19:45:31 +00:00
|
|
|
#include <sys/mount.h>
|
2023-03-18 18:23:18 +00:00
|
|
|
#include <sys/stat.h>
|
2023-03-18 20:55:16 +00:00
|
|
|
#include <sys/syscall.h>
|
2023-03-18 18:23:18 +00:00
|
|
|
#include <sys/sysmacros.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
FILE* g_init_log = nullptr;
|
2023-04-22 11:56:08 +00:00
|
|
|
|
|
|
|
struct Service
|
|
|
|
{
|
|
|
|
String name;
|
|
|
|
String command;
|
|
|
|
bool restart { false };
|
|
|
|
String environment;
|
2023-04-25 19:00:12 +00:00
|
|
|
String standard_output;
|
|
|
|
String standard_error;
|
|
|
|
String standard_input;
|
|
|
|
bool wait { false };
|
2023-04-22 11:56:08 +00:00
|
|
|
Option<pid_t> pid {};
|
|
|
|
};
|
|
|
|
|
|
|
|
Vector<Service> g_services;
|
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
static void do_log(const char* format, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, format);
|
|
|
|
|
|
|
|
if (g_init_log) vfprintf(g_init_log, format, ap);
|
|
|
|
|
|
|
|
va_end(ap);
|
|
|
|
}
|
|
|
|
|
2023-04-25 19:00:12 +00:00
|
|
|
static Result<void> service_child(const Service& service, SharedPtr<os::File> output, SharedPtr<os::File> error,
|
|
|
|
SharedPtr<os::File> input)
|
2023-04-22 11:56:08 +00:00
|
|
|
{
|
|
|
|
auto args = TRY(service.command.split(" \n"));
|
|
|
|
|
2023-04-25 19:00:12 +00:00
|
|
|
if (output) dup2(output->fd(), STDOUT_FILENO);
|
|
|
|
if (error) dup2(error->fd(), STDERR_FILENO);
|
|
|
|
if (input) dup2(input->fd(), STDIN_FILENO);
|
|
|
|
|
2023-04-22 11:56:08 +00:00
|
|
|
if (service.environment.is_empty()) { TRY(os::Process::exec(args[0].view(), args.slice(), false)); }
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto env = TRY(service.environment.split(",\n"));
|
|
|
|
TRY(os::Process::exec(args[0].view(), args.slice(), env.slice(), false));
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
static Result<void> try_start_service(Service& service)
|
|
|
|
{
|
2023-04-25 19:00:12 +00:00
|
|
|
SharedPtr<os::File> new_stdout = {};
|
|
|
|
SharedPtr<os::File> new_stderr = {};
|
|
|
|
SharedPtr<os::File> new_stdin = {};
|
|
|
|
|
|
|
|
if (!service.standard_output.is_empty())
|
|
|
|
{
|
|
|
|
new_stdout = TRY(os::File::open_or_create(service.standard_output.view(), os::File::Append, 0600));
|
|
|
|
new_stdout->set_close_on_exec();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!service.standard_error.is_empty())
|
|
|
|
{
|
|
|
|
new_stderr = TRY(os::File::open_or_create(service.standard_error.view(), os::File::Append, 0600));
|
|
|
|
new_stderr->set_close_on_exec();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!service.standard_input.is_empty())
|
|
|
|
{
|
|
|
|
new_stdin = TRY(os::File::open(service.standard_input.view(), os::File::ReadOnly));
|
|
|
|
new_stdin->set_close_on_exec();
|
|
|
|
}
|
|
|
|
|
2023-04-22 11:56:08 +00:00
|
|
|
pid_t pid = TRY(os::Process::fork());
|
|
|
|
if (pid == 0)
|
|
|
|
{
|
2023-04-25 19:00:12 +00:00
|
|
|
auto rc = service_child(service, new_stdout, new_stderr, new_stdin);
|
2023-04-22 11:56:08 +00:00
|
|
|
if (rc.has_error())
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[child %d] failed to start service %s due to error: %s\n", getpid(), service.name.chars(),
|
|
|
|
rc.error_string());
|
2023-04-22 11:56:08 +00:00
|
|
|
}
|
|
|
|
fclose(g_init_log);
|
|
|
|
exit(127);
|
|
|
|
}
|
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] created new child process %d for service %s\n", pid, service.name.chars());
|
2023-04-22 11:56:08 +00:00
|
|
|
|
2023-04-25 19:00:12 +00:00
|
|
|
if (service.wait)
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] waiting for child process %d to finish\n", pid);
|
2023-04-25 19:00:12 +00:00
|
|
|
|
|
|
|
int status;
|
2023-06-09 21:12:31 +00:00
|
|
|
TRY(os::Process::wait(pid, &status));
|
2023-04-25 19:00:12 +00:00
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] child process %d exited with code %d\n", pid, WEXITSTATUS(status));
|
2023-04-25 19:00:12 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
service.pid = pid;
|
2023-04-22 11:56:08 +00:00
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
static void start_service(Service& service)
|
|
|
|
{
|
|
|
|
auto rc = try_start_service(service);
|
|
|
|
if (rc.has_error())
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] failed to start service %s due to error: %s\n", service.name.chars(), rc.error_string());
|
2023-04-22 11:56:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-03 15:37:26 +00:00
|
|
|
static Result<void> load_service(const os::Path& path)
|
2023-04-22 11:56:08 +00:00
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] reading service file: %s\n", path.name().chars());
|
2023-04-22 11:56:08 +00:00
|
|
|
|
|
|
|
auto file = TRY(os::File::open(path, os::File::ReadOnly));
|
|
|
|
|
|
|
|
Service service;
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
2023-04-22 13:19:37 +00:00
|
|
|
auto line = TRY(file->read_line());
|
2023-04-22 11:56:08 +00:00
|
|
|
if (line.is_empty()) break;
|
|
|
|
|
2023-04-22 13:19:37 +00:00
|
|
|
line.trim("\n");
|
|
|
|
if (line.is_empty()) continue;
|
|
|
|
|
2023-04-22 11:56:08 +00:00
|
|
|
auto parts = TRY(line.split_once('='));
|
|
|
|
if (parts.size() < 2 || parts[0].is_empty() || parts[1].is_empty())
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] file contains invalid line, aborting: '%s'\n", line.chars());
|
2023-04-22 11:56:08 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parts[0].view() == "Name")
|
|
|
|
{
|
|
|
|
service.name = move(parts[1]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parts[0].view() == "Command")
|
|
|
|
{
|
2023-06-03 09:34:53 +00:00
|
|
|
if (!service.command.is_empty())
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] 'Command' cannot be specified after 'Script' has already been set! (%s)\n",
|
|
|
|
line.chars());
|
2023-06-03 09:34:53 +00:00
|
|
|
return {};
|
|
|
|
}
|
2023-04-22 11:56:08 +00:00
|
|
|
service.command = move(parts[1]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-06-03 09:34:53 +00:00
|
|
|
if (parts[0].view() == "Script")
|
|
|
|
{
|
|
|
|
if (!service.command.is_empty())
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] 'Script' cannot be specified after 'Command' has already been set! (%s)\n",
|
|
|
|
line.chars());
|
2023-06-03 09:34:53 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
service.command = TRY(String::format("/bin/sh -- %s"_sv, parts[1].chars()));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-04-22 11:56:08 +00:00
|
|
|
if (parts[0].view() == "Restart")
|
|
|
|
{
|
|
|
|
if (parts[1].view() == "true" || parts[1].view().to_uint().value_or(0) == 1)
|
|
|
|
{
|
|
|
|
service.restart = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
service.restart = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parts[0].view() == "Environment")
|
|
|
|
{
|
|
|
|
service.environment = move(parts[1]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-04-25 19:00:12 +00:00
|
|
|
if (parts[0].view() == "StandardOutput")
|
|
|
|
{
|
|
|
|
service.standard_output = move(parts[1]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parts[0].view() == "StandardError")
|
|
|
|
{
|
|
|
|
service.standard_error = move(parts[1]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parts[0].view() == "StandardInput")
|
|
|
|
{
|
|
|
|
service.standard_input = move(parts[1]);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parts[0].view() == "Wait")
|
|
|
|
{
|
|
|
|
if (parts[1].view() == "true" || parts[1].view().to_uint().value_or(0) == 1)
|
|
|
|
{
|
|
|
|
service.wait = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
service.wait = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] skipping unknown entry name %s\n", parts[0].chars());
|
2023-04-22 11:56:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (service.name.is_empty())
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] service file is missing 'Name' entry, aborting!\n");
|
2023-04-22 11:56:08 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (service.command.is_empty())
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] service file is missing 'Command' or 'Script' entry, aborting!\n");
|
2023-04-22 11:56:08 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] loaded service %s into memory\n", service.name.chars());
|
2023-04-22 11:56:08 +00:00
|
|
|
|
|
|
|
TRY(g_services.try_append(move(service)));
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
static Result<void> load_services()
|
|
|
|
{
|
2023-04-28 19:15:41 +00:00
|
|
|
auto dir = TRY(os::Directory::open("/etc/init"));
|
2023-04-22 11:56:08 +00:00
|
|
|
|
2023-05-28 19:50:48 +00:00
|
|
|
auto services = TRY(dir->list_names(os::Directory::Filter::ParentAndBase));
|
2023-05-02 18:55:41 +00:00
|
|
|
sort(services.begin(), services.end(), String::compare);
|
2023-04-28 20:41:44 +00:00
|
|
|
|
2023-05-03 15:37:26 +00:00
|
|
|
for (const auto& entry : services) TRY(load_service({ dir->fd(), entry.view() }));
|
2023-04-22 11:56:08 +00:00
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
static Result<void> start_services()
|
|
|
|
{
|
|
|
|
TRY(load_services());
|
|
|
|
for (auto& service : g_services)
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] starting service %s\n", service.name.chars());
|
2023-04-22 11:56:08 +00:00
|
|
|
start_service(service);
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-04-24 19:20:44 +00:00
|
|
|
static Result<void> set_hostname()
|
|
|
|
{
|
|
|
|
auto file = TRY(os::File::open("/etc/hostname", os::File::ReadOnly));
|
|
|
|
|
|
|
|
auto hostname = TRY(file->read_line());
|
|
|
|
hostname.trim("\n");
|
|
|
|
|
|
|
|
if (sethostname(hostname.chars(), hostname.length()) < 0) return {};
|
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] successfully set system hostname to '%s'\n", hostname.chars());
|
2023-04-24 19:20:44 +00:00
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
static void mount_tmpfs()
|
2023-06-02 19:45:31 +00:00
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
if (mount("/tmp", "tmpfs", "tmpfs") < 0) exit(255);
|
2023-06-02 19:45:31 +00:00
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
if (chmod("/tmp", 01777) < 0) exit(255);
|
2023-06-02 19:45:31 +00:00
|
|
|
}
|
|
|
|
|
2023-03-18 18:23:18 +00:00
|
|
|
int main()
|
|
|
|
{
|
|
|
|
if (getpid() != 1)
|
|
|
|
{
|
2023-03-19 18:19:20 +00:00
|
|
|
fprintf(stderr, "error: init not running as PID 1.\n");
|
2023-03-18 18:23:18 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2023-03-19 18:19:20 +00:00
|
|
|
// Before this point, we don't even have an stdin, stdout and stderr. Set it up now so that child processes (and us)
|
|
|
|
// can print stuff.
|
|
|
|
stdin = fopen("/dev/console", "r");
|
2023-03-18 18:23:18 +00:00
|
|
|
stdout = fopen("/dev/console", "w");
|
|
|
|
stderr = fopen("/dev/console", "w");
|
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
mount_tmpfs();
|
|
|
|
|
2023-05-26 20:27:49 +00:00
|
|
|
umask(022);
|
|
|
|
|
2023-07-10 11:05:06 +00:00
|
|
|
g_init_log = fopen("/tmp/init.log", "w+");
|
2023-04-22 11:56:08 +00:00
|
|
|
fcntl(fileno(g_init_log), F_SETFD, FD_CLOEXEC);
|
|
|
|
|
2023-04-24 19:20:44 +00:00
|
|
|
set_hostname();
|
|
|
|
|
2023-04-22 11:56:08 +00:00
|
|
|
start_services();
|
2023-03-18 22:45:48 +00:00
|
|
|
|
2023-04-22 11:56:08 +00:00
|
|
|
while (1)
|
2023-03-18 22:45:48 +00:00
|
|
|
{
|
2023-04-22 11:56:08 +00:00
|
|
|
int status;
|
2023-06-09 21:12:31 +00:00
|
|
|
auto rc = os::Process::wait(os::Process::ANY_CHILD, &status);
|
|
|
|
if (rc.has_error()) continue;
|
|
|
|
|
|
|
|
pid_t child = rc.release_value();
|
2023-03-23 21:19:54 +00:00
|
|
|
|
2023-04-22 11:56:08 +00:00
|
|
|
for (auto& service : g_services)
|
|
|
|
{
|
|
|
|
if (service.pid.has_value() && service.pid.value() == child)
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] service %s exited with status %d\n", service.name.chars(), WEXITSTATUS(status));
|
2023-04-22 11:56:08 +00:00
|
|
|
|
|
|
|
if (service.restart)
|
|
|
|
{
|
2023-07-10 11:05:06 +00:00
|
|
|
do_log("[init] restarting service %s\n", service.name.chars());
|
2023-04-22 11:56:08 +00:00
|
|
|
|
|
|
|
start_service(service);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-03-18 18:23:18 +00:00
|
|
|
}
|