/* * Copyright (c) 2018-2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include namespace Kernel { static char s_cmd_line[1024]; static constexpr StringView s_embedded_cmd_line = ""; static CommandLine* s_the; UNMAP_AFTER_INIT void CommandLine::early_initialize(const char* cmd_line) { if (!cmd_line) return; size_t length = strlen(cmd_line); if (length >= sizeof(s_cmd_line)) length = sizeof(s_cmd_line) - 1; memcpy(s_cmd_line, cmd_line, length); s_cmd_line[length] = '\0'; } const CommandLine& kernel_command_line() { VERIFY(s_the); return *s_the; } UNMAP_AFTER_INIT void CommandLine::initialize() { VERIFY(!s_the); s_the = new CommandLine(s_cmd_line); dmesgln("Kernel Commandline: {}", kernel_command_line().string()); } UNMAP_AFTER_INIT void CommandLine::build_commandline(const String& cmdline_from_bootloader) { StringBuilder builder; builder.append(cmdline_from_bootloader); if (!s_embedded_cmd_line.is_empty()) { builder.append(" "); builder.append(s_embedded_cmd_line); } m_string = builder.to_string(); } UNMAP_AFTER_INIT void CommandLine::add_arguments(const Vector& args) { for (auto&& str : args) { if (str == "") { continue; } auto pair = str.split_limit('=', 2); if (pair.size() == 1) { m_params.set(move(pair[0]), ""); } else { m_params.set(move(pair[0]), move(pair[1])); } } } UNMAP_AFTER_INIT CommandLine::CommandLine(const String& cmdline_from_bootloader) { s_the = this; build_commandline(cmdline_from_bootloader); const auto& args = m_string.split(' '); m_params.ensure_capacity(args.size()); add_arguments(args); } Optional CommandLine::lookup(const String& key) const { return m_params.get(key); } bool CommandLine::contains(const String& key) const { return m_params.contains(key); } UNMAP_AFTER_INIT bool CommandLine::is_boot_profiling_enabled() const { return contains("boot_prof"); } UNMAP_AFTER_INIT bool CommandLine::is_ide_enabled() const { return !contains("disable_ide"); } UNMAP_AFTER_INIT bool CommandLine::is_smp_enabled() const { return lookup("smp").value_or("off") == "on"; } UNMAP_AFTER_INIT bool CommandLine::is_vmmouse_enabled() const { return lookup("vmmouse").value_or("on") == "on"; } UNMAP_AFTER_INIT PCIAccessLevel CommandLine::pci_access_level() const { auto value = lookup("pci_ecam").value_or("off"); if (value == "on") return PCIAccessLevel::MappingPerBus; if (value == "per-device") return PCIAccessLevel::MappingPerDevice; if (value == "off") return PCIAccessLevel::IOAddressing; PANIC("Unknown PCI ECAM setting: {}", value); } UNMAP_AFTER_INIT bool CommandLine::is_legacy_time_enabled() const { return lookup("time").value_or("modern") == "legacy"; } UNMAP_AFTER_INIT bool CommandLine::is_force_pio() const { return contains("force_pio"); } UNMAP_AFTER_INIT String CommandLine::root_device() const { return lookup("root").value_or("/dev/hda"); } UNMAP_AFTER_INIT AcpiFeatureLevel CommandLine::acpi_feature_level() const { auto value = kernel_command_line().lookup("acpi").value_or("on"); if (value == "limited") return AcpiFeatureLevel::Limited; if (value == "off") return AcpiFeatureLevel::Disabled; return AcpiFeatureLevel::Enabled; } UNMAP_AFTER_INIT HPETMode CommandLine::hpet_mode() const { auto hpet_mode = lookup("hpet").value_or("periodic"); if (hpet_mode == "periodic") return HPETMode::Periodic; if (hpet_mode == "nonperiodic") return HPETMode::NonPeriodic; PANIC("Unknown HPETMode: {}", hpet_mode); } UNMAP_AFTER_INIT bool CommandLine::disable_ps2_controller() const { return contains("disable_ps2_controller"); } UNMAP_AFTER_INIT bool CommandLine::disable_physical_storage() const { return contains("disable_physical_storage"); } UNMAP_AFTER_INIT bool CommandLine::disable_uhci_controller() const { return contains("disable_uhci_controller"); } UNMAP_AFTER_INIT bool CommandLine::disable_virtio() const { return contains("disable_virtio"); } UNMAP_AFTER_INIT AHCIResetMode CommandLine::ahci_reset_mode() const { const auto ahci_reset_mode = lookup("ahci_reset_mode").value_or("controller"); if (ahci_reset_mode == "controller") { return AHCIResetMode::ControllerOnly; } else if (ahci_reset_mode == "aggressive") { return AHCIResetMode::Aggressive; } PANIC("Unknown AHCIResetMode: {}", ahci_reset_mode); } UNMAP_AFTER_INIT BootMode CommandLine::boot_mode() const { const auto boot_mode = lookup("boot_mode").value_or("graphical"); if (boot_mode == "no-fbdev") { return BootMode::NoFramebufferDevices; } else if (boot_mode == "self-test") { return BootMode::SelfTest; } else if (boot_mode == "graphical") { return BootMode::Graphical; } PANIC("Unknown BootMode: {}", boot_mode); } UNMAP_AFTER_INIT bool CommandLine::is_no_framebuffer_devices_mode() const { const auto mode = boot_mode(); return mode == BootMode::NoFramebufferDevices || mode == BootMode::SelfTest; } String CommandLine::userspace_init() const { return lookup("init").value_or("/bin/SystemServer"); } Vector CommandLine::userspace_init_args() const { auto init_args = lookup("init_args").value_or("").split(','); if (!init_args.is_empty()) init_args.prepend(userspace_init()); return init_args; } UNMAP_AFTER_INIT size_t CommandLine::switch_to_tty() const { const auto default_tty = lookup("switch_to_tty").value_or("1"); auto switch_tty_number = default_tty.to_uint(); if (switch_tty_number.has_value() && switch_tty_number.value() >= 1) { return switch_tty_number.value() - 1; } PANIC("Invalid default tty value: {}", default_tty); } }