diff options
author | Liav A <liavalb@gmail.com> | 2021-04-16 22:58:51 +0300 |
---|---|---|
committer | Andreas Kling <kling@serenityos.org> | 2021-05-16 19:58:33 +0200 |
commit | 20743e8aede1de46195dd61ad18002cd52db7d3a (patch) | |
tree | 49ea710e88949b49c136b0b3b793059c24f6daa1 /Kernel/Graphics | |
parent | dac129e10bdef313dff65b34f1fa17608d3608c2 (diff) | |
download | serenity-20743e8aede1de46195dd61ad18002cd52db7d3a.zip |
Kernel/Graphics + SystemServer: Support text mode properly
As we removed the support of VBE modesetting that was done by GRUB early
on boot, we need to determine if we can modeset the resolution with our
drivers, and if not, we should enable text mode and ensure that
SystemServer knows about it too.
Also, SystemServer should first check if there's a framebuffer device
node, which is an indication that text mode was not even if it was
requested. Then, if it doesn't find it, it should check what boot_mode
argument the user specified (in case it's self-test). This way if we
try to use bochs-display device (which is not VGA compatible) and
request a text mode, it will not honor the request and will continue
with graphical mode.
Also try to print critical messages with mininum memory allocations
possible.
In LibVT, We make the implementation flexible for kernel-specific
methods that are implemented in ConsoleImpl class.
Diffstat (limited to 'Kernel/Graphics')
-rw-r--r-- | Kernel/Graphics/BochsGraphicsAdapter.cpp | 36 | ||||
-rw-r--r-- | Kernel/Graphics/BochsGraphicsAdapter.h | 10 | ||||
-rw-r--r-- | Kernel/Graphics/Console/Console.h | 82 | ||||
-rw-r--r-- | Kernel/Graphics/Console/FramebufferConsole.cpp | 348 | ||||
-rw-r--r-- | Kernel/Graphics/Console/FramebufferConsole.h | 50 | ||||
-rw-r--r-- | Kernel/Graphics/Console/TextModeConsole.cpp | 203 | ||||
-rw-r--r-- | Kernel/Graphics/Console/TextModeConsole.h | 50 | ||||
-rw-r--r-- | Kernel/Graphics/Console/VGAConsole.cpp | 19 | ||||
-rw-r--r-- | Kernel/Graphics/Console/VGAConsole.h | 39 | ||||
-rw-r--r-- | Kernel/Graphics/FramebufferDevice.cpp | 57 | ||||
-rw-r--r-- | Kernel/Graphics/FramebufferDevice.h | 17 | ||||
-rw-r--r-- | Kernel/Graphics/GraphicsDevice.h | 6 | ||||
-rw-r--r-- | Kernel/Graphics/GraphicsManagement.cpp | 104 | ||||
-rw-r--r-- | Kernel/Graphics/GraphicsManagement.h | 30 | ||||
-rw-r--r-- | Kernel/Graphics/IntelNativeGraphicsAdapter.cpp | 42 | ||||
-rw-r--r-- | Kernel/Graphics/IntelNativeGraphicsAdapter.h | 18 | ||||
-rw-r--r-- | Kernel/Graphics/VGACompatibleAdapter.cpp | 42 | ||||
-rw-r--r-- | Kernel/Graphics/VGACompatibleAdapter.h | 17 |
18 files changed, 1127 insertions, 43 deletions
diff --git a/Kernel/Graphics/BochsGraphicsAdapter.cpp b/Kernel/Graphics/BochsGraphicsAdapter.cpp index d6f42b1662..0fb47c36fe 100644 --- a/Kernel/Graphics/BochsGraphicsAdapter.cpp +++ b/Kernel/Graphics/BochsGraphicsAdapter.cpp @@ -11,6 +11,8 @@ #include <Kernel/Graphics/Bochs.h> #include <Kernel/Graphics/BochsFramebufferDevice.h> #include <Kernel/Graphics/BochsGraphicsAdapter.h> +#include <Kernel/Graphics/Console/FramebufferConsole.h> +#include <Kernel/Graphics/GraphicsManagement.h> #include <Kernel/IO.h> #include <Kernel/PCI/Access.h> #include <Kernel/Process.h> @@ -52,19 +54,24 @@ UNMAP_AFTER_INIT BochsGraphicsAdapter::BochsGraphicsAdapter(PCI::Address pci_add , m_mmio_registers(PCI::get_BAR2(pci_address) & 0xfffffff0) { set_safe_resolution(); + // We assume safe resolutio is 1024x768x32 + m_framebuffer_console = Graphics::FramebufferConsole::initialize(PhysicalAddress(PCI::get_BAR0(pci_address) & 0xfffffff0), 1024, 768, 1024 * sizeof(u32)); + // FIXME: This is a very wrong way to do this... + GraphicsManagement::the().m_console = m_framebuffer_console; } UNMAP_AFTER_INIT void BochsGraphicsAdapter::initialize_framebuffer_devices() { // FIXME: Find a better way to determine default resolution... - m_framebuffer = BochsFramebufferDevice::create(*this, PhysicalAddress(PCI::get_BAR0(pci_address()) & 0xfffffff0), 1024 * 4, 1024, 768); + m_framebuffer_device = BochsFramebufferDevice::create(*this, PhysicalAddress(PCI::get_BAR0(pci_address()) & 0xfffffff0), 1024 * 4, 1024, 768); + m_framebuffer_device->initialize(); } GraphicsDevice::Type BochsGraphicsAdapter::type() const { if (PCI::get_class(pci_address()) == 0x3 && PCI::get_subclass(pci_address()) == 0x0) return Type::VGACompatible; - return Type::Bochs; + return Type::Bochs; } void BochsGraphicsAdapter::set_safe_resolution() @@ -119,8 +126,33 @@ bool BochsGraphicsAdapter::validate_setup_resolution(size_t width, size_t height void BochsGraphicsAdapter::set_y_offset(size_t y_offset) { + if (m_console_enabled) + return; auto registers = map_typed_writable<volatile BochsDisplayMMIORegisters>(m_mmio_registers); registers->bochs_regs.y_offset = y_offset; } +void BochsGraphicsAdapter::enable_consoles() +{ + ScopedSpinLock lock(m_console_mode_switch_lock); + VERIFY(m_framebuffer_console); + m_console_enabled = true; + auto registers = map_typed_writable<volatile BochsDisplayMMIORegisters>(m_mmio_registers); + registers->bochs_regs.y_offset = 0; + if (m_framebuffer_device) + m_framebuffer_device->dectivate_writes(); + m_framebuffer_console->enable(); +} +void BochsGraphicsAdapter::disable_consoles() +{ + ScopedSpinLock lock(m_console_mode_switch_lock); + VERIFY(m_framebuffer_console); + VERIFY(m_framebuffer_device); + m_console_enabled = false; + auto registers = map_typed_writable<volatile BochsDisplayMMIORegisters>(m_mmio_registers); + registers->bochs_regs.y_offset = 0; + m_framebuffer_console->disable(); + m_framebuffer_device->activate_writes(); +} + } diff --git a/Kernel/Graphics/BochsGraphicsAdapter.h b/Kernel/Graphics/BochsGraphicsAdapter.h index e2c2b1cdd5..af647e0a87 100644 --- a/Kernel/Graphics/BochsGraphicsAdapter.h +++ b/Kernel/Graphics/BochsGraphicsAdapter.h @@ -8,6 +8,7 @@ #include <AK/String.h> #include <AK/Types.h> +#include <Kernel/Graphics/Console/FramebufferConsole.h> #include <Kernel/Graphics/GraphicsDevice.h> #include <Kernel/PCI/DeviceController.h> #include <Kernel/PhysicalAddress.h> @@ -25,12 +26,16 @@ class BochsGraphicsAdapter final : public GraphicsDevice public: static NonnullRefPtr<BochsGraphicsAdapter> initialize(PCI::Address); virtual ~BochsGraphicsAdapter() = default; + virtual bool framebuffer_devices_initialized() const override { return !m_framebuffer_device.is_null(); } private: // ^GraphicsDevice virtual void initialize_framebuffer_devices() override; virtual Type type() const override; + virtual void enable_consoles() override; + virtual void disable_consoles() override; + explicit BochsGraphicsAdapter(PCI::Address); void set_safe_resolution(); @@ -43,7 +48,10 @@ private: void set_y_offset(size_t); PhysicalAddress m_mmio_registers; - RefPtr<BochsFramebufferDevice> m_framebuffer; + RefPtr<BochsFramebufferDevice> m_framebuffer_device; + RefPtr<Graphics::FramebufferConsole> m_framebuffer_console; + SpinLock<u8> m_console_mode_switch_lock; + bool m_console_enabled { false }; }; } diff --git a/Kernel/Graphics/Console/Console.h b/Kernel/Graphics/Console/Console.h new file mode 100644 index 0000000000..c391dc32fb --- /dev/null +++ b/Kernel/Graphics/Console/Console.h @@ -0,0 +1,82 @@ +/* + * Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#pragma once + +#include <AK/RefCounted.h> +#include <AK/String.h> +#include <AK/Types.h> +#include <Kernel/Graphics/GraphicsDevice.h> + +namespace Kernel::Graphics { + +class Console : public RefCounted<Console> { +public: + // Stanadard VGA text mode colors + enum Color : u8 { + Black = 0, + Blue, + Green, + Cyan, + Red, + Magenta, + Brown, + LightGray, + DarkGray, + BrightBlue, + BrightGreen, + BrightCyan, + BrightRed, + BrightMagenta, + Yellow, + White, + }; + +public: + size_t width() const { return m_width; } + size_t height() const { return m_height; } + size_t pitch() const { return bytes_per_base_glyph() * width(); } + virtual size_t max_column() const { return m_width; } + virtual size_t max_row() const { return m_height; } + virtual size_t bytes_per_base_glyph() const = 0; + virtual size_t chars_per_line() const = 0; + + virtual void enable() = 0; + virtual void disable() = 0; + + virtual bool is_hardware_paged_capable() const = 0; + virtual bool has_hardware_cursor() const = 0; + + virtual void set_cursor(size_t x, size_t y) = 0; + virtual void hide_cursor() = 0; + virtual void show_cursor() = 0; + + virtual void clear(size_t x, size_t y, size_t length) const = 0; + virtual void write(size_t x, size_t y, char ch, Color background, Color foreground) const = 0; + virtual void write(size_t x, size_t y, String, Color background, Color foreground) const = 0; + virtual void write(size_t x, size_t y, char ch) const = 0; + virtual void write(size_t x, size_t y, String) const = 0; + virtual void write(char ch) const = 0; + + virtual ~Console() { } + +protected: + Console(size_t width, size_t height) + : m_width(width) + , m_height(height) + { + m_enabled.store(true); + } + + Atomic<bool> m_enabled; + Color m_default_foreground_color { Color::White }; + Color m_default_background_color { Color::Black }; + const size_t m_width; + const size_t m_height; + mutable size_t m_x { 0 }; + mutable size_t m_y { 0 }; +}; +} diff --git a/Kernel/Graphics/Console/FramebufferConsole.cpp b/Kernel/Graphics/Console/FramebufferConsole.cpp new file mode 100644 index 0000000000..8ae2e3122e --- /dev/null +++ b/Kernel/Graphics/Console/FramebufferConsole.cpp @@ -0,0 +1,348 @@ +/* + * Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#include <Kernel/Graphics/Console/FramebufferConsole.h> + +namespace Kernel::Graphics { + +constexpr unsigned char const font8x8_basic[128][8] = { + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0000 (nul) + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0001 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0002 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0003 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0004 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0005 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0006 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0007 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0008 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0009 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000A + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000B + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000C + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000D + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000E + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+000F + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0010 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0011 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0012 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0013 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0014 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0015 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0016 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0017 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0018 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0019 + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001A + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001B + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001C + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001D + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001E + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+001F + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0020 (space) + { 0x18, 0x3C, 0x3C, 0x18, 0x18, 0x00, 0x18, 0x00 }, // U+0021 (!) + { 0x36, 0x36, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0022 (") + { 0x36, 0x36, 0x7F, 0x36, 0x7F, 0x36, 0x36, 0x00 }, // U+0023 (#) + { 0x0C, 0x3E, 0x03, 0x1E, 0x30, 0x1F, 0x0C, 0x00 }, // U+0024 ($) + { 0x00, 0x63, 0x33, 0x18, 0x0C, 0x66, 0x63, 0x00 }, // U+0025 (%) + { 0x1C, 0x36, 0x1C, 0x6E, 0x3B, 0x33, 0x6E, 0x00 }, // U+0026 (&) + { 0x06, 0x06, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0027 (') + { 0x18, 0x0C, 0x06, 0x06, 0x06, 0x0C, 0x18, 0x00 }, // U+0028 (() + { 0x06, 0x0C, 0x18, 0x18, 0x18, 0x0C, 0x06, 0x00 }, // U+0029 ()) + { 0x00, 0x66, 0x3C, 0xFF, 0x3C, 0x66, 0x00, 0x00 }, // U+002A (*) + { 0x00, 0x0C, 0x0C, 0x3F, 0x0C, 0x0C, 0x00, 0x00 }, // U+002B (+) + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x0C, 0x0C, 0x06 }, // U+002C (,) + { 0x00, 0x00, 0x00, 0x3F, 0x00, 0x00, 0x00, 0x00 }, // U+002D (-) + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x0C, 0x0C, 0x00 }, // U+002E (.) + { 0x60, 0x30, 0x18, 0x0C, 0x06, 0x03, 0x01, 0x00 }, // U+002F (/) + { 0x3E, 0x63, 0x73, 0x7B, 0x6F, 0x67, 0x3E, 0x00 }, // U+0030 (0) + { 0x0C, 0x0E, 0x0C, 0x0C, 0x0C, 0x0C, 0x3F, 0x00 }, // U+0031 (1) + { 0x1E, 0x33, 0x30, 0x1C, 0x06, 0x33, 0x3F, 0x00 }, // U+0032 (2) + { 0x1E, 0x33, 0x30, 0x1C, 0x30, 0x33, 0x1E, 0x00 }, // U+0033 (3) + { 0x38, 0x3C, 0x36, 0x33, 0x7F, 0x30, 0x78, 0x00 }, // U+0034 (4) + { 0x3F, 0x03, 0x1F, 0x30, 0x30, 0x33, 0x1E, 0x00 }, // U+0035 (5) + { 0x1C, 0x06, 0x03, 0x1F, 0x33, 0x33, 0x1E, 0x00 }, // U+0036 (6) + { 0x3F, 0x33, 0x30, 0x18, 0x0C, 0x0C, 0x0C, 0x00 }, // U+0037 (7) + { 0x1E, 0x33, 0x33, 0x1E, 0x33, 0x33, 0x1E, 0x00 }, // U+0038 (8) + { 0x1E, 0x33, 0x33, 0x3E, 0x30, 0x18, 0x0E, 0x00 }, // U+0039 (9) + { 0x00, 0x0C, 0x0C, 0x00, 0x00, 0x0C, 0x0C, 0x00 }, // U+003A (:) + { 0x00, 0x0C, 0x0C, 0x00, 0x00, 0x0C, 0x0C, 0x06 }, // U+003B (;) + { 0x18, 0x0C, 0x06, 0x03, 0x06, 0x0C, 0x18, 0x00 }, // U+003C (<) + { 0x00, 0x00, 0x3F, 0x00, 0x00, 0x3F, 0x00, 0x00 }, // U+003D (=) + { 0x06, 0x0C, 0x18, 0x30, 0x18, 0x0C, 0x06, 0x00 }, // U+003E (>) + { 0x1E, 0x33, 0x30, 0x18, 0x0C, 0x00, 0x0C, 0x00 }, // U+003F (?) + { 0x3E, 0x63, 0x7B, 0x7B, 0x7B, 0x03, 0x1E, 0x00 }, // U+0040 (@) + { 0x0C, 0x1E, 0x33, 0x33, 0x3F, 0x33, 0x33, 0x00 }, // U+0041 (A) + { 0x3F, 0x66, 0x66, 0x3E, 0x66, 0x66, 0x3F, 0x00 }, // U+0042 (B) + { 0x3C, 0x66, 0x03, 0x03, 0x03, 0x66, 0x3C, 0x00 }, // U+0043 (C) + { 0x1F, 0x36, 0x66, 0x66, 0x66, 0x36, 0x1F, 0x00 }, // U+0044 (D) + { 0x7F, 0x46, 0x16, 0x1E, 0x16, 0x46, 0x7F, 0x00 }, // U+0045 (E) + { 0x7F, 0x46, 0x16, 0x1E, 0x16, 0x06, 0x0F, 0x00 }, // U+0046 (F) + { 0x3C, 0x66, 0x03, 0x03, 0x73, 0x66, 0x7C, 0x00 }, // U+0047 (G) + { 0x33, 0x33, 0x33, 0x3F, 0x33, 0x33, 0x33, 0x00 }, // U+0048 (H) + { 0x1E, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x1E, 0x00 }, // U+0049 (I) + { 0x78, 0x30, 0x30, 0x30, 0x33, 0x33, 0x1E, 0x00 }, // U+004A (J) + { 0x67, 0x66, 0x36, 0x1E, 0x36, 0x66, 0x67, 0x00 }, // U+004B (K) + { 0x0F, 0x06, 0x06, 0x06, 0x46, 0x66, 0x7F, 0x00 }, // U+004C (L) + { 0x63, 0x77, 0x7F, 0x7F, 0x6B, 0x63, 0x63, 0x00 }, // U+004D (M) + { 0x63, 0x67, 0x6F, 0x7B, 0x73, 0x63, 0x63, 0x00 }, // U+004E (N) + { 0x1C, 0x36, 0x63, 0x63, 0x63, 0x36, 0x1C, 0x00 }, // U+004F (O) + { 0x3F, 0x66, 0x66, 0x3E, 0x06, 0x06, 0x0F, 0x00 }, // U+0050 (P) + { 0x1E, 0x33, 0x33, 0x33, 0x3B, 0x1E, 0x38, 0x00 }, // U+0051 (Q) + { 0x3F, 0x66, 0x66, 0x3E, 0x36, 0x66, 0x67, 0x00 }, // U+0052 (R) + { 0x1E, 0x33, 0x07, 0x0E, 0x38, 0x33, 0x1E, 0x00 }, // U+0053 (S) + { 0x3F, 0x2D, 0x0C, 0x0C, 0x0C, 0x0C, 0x1E, 0x00 }, // U+0054 (T) + { 0x33, 0x33, 0x33, 0x33, 0x33, 0x33, 0x3F, 0x00 }, // U+0055 (U) + { 0x33, 0x33, 0x33, 0x33, 0x33, 0x1E, 0x0C, 0x00 }, // U+0056 (V) + { 0x63, 0x63, 0x63, 0x6B, 0x7F, 0x77, 0x63, 0x00 }, // U+0057 (W) + { 0x63, 0x63, 0x36, 0x1C, 0x1C, 0x36, 0x63, 0x00 }, // U+0058 (X) + { 0x33, 0x33, 0x33, 0x1E, 0x0C, 0x0C, 0x1E, 0x00 }, // U+0059 (Y) + { 0x7F, 0x63, 0x31, 0x18, 0x4C, 0x66, 0x7F, 0x00 }, // U+005A (Z) + { 0x1E, 0x06, 0x06, 0x06, 0x06, 0x06, 0x1E, 0x00 }, // U+005B ([) + { 0x03, 0x06, 0x0C, 0x18, 0x30, 0x60, 0x40, 0x00 }, // U+005C (\) + { 0x1E, 0x18, 0x18, 0x18, 0x18, 0x18, 0x1E, 0x00 }, // U+005D (]) + { 0x08, 0x1C, 0x36, 0x63, 0x00, 0x00, 0x00, 0x00 }, // U+005E (^) + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF }, // U+005F (_) + { 0x0C, 0x0C, 0x18, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+0060 (`) + { 0x00, 0x00, 0x1E, 0x30, 0x3E, 0x33, 0x6E, 0x00 }, // U+0061 (a) + { 0x07, 0x06, 0x06, 0x3E, 0x66, 0x66, 0x3B, 0x00 }, // U+0062 (b) + { 0x00, 0x00, 0x1E, 0x33, 0x03, 0x33, 0x1E, 0x00 }, // U+0063 (c) + { 0x38, 0x30, 0x30, 0x3e, 0x33, 0x33, 0x6E, 0x00 }, // U+0064 (d) + { 0x00, 0x00, 0x1E, 0x33, 0x3f, 0x03, 0x1E, 0x00 }, // U+0065 (e) + { 0x1C, 0x36, 0x06, 0x0f, 0x06, 0x06, 0x0F, 0x00 }, // U+0066 (f) + { 0x00, 0x00, 0x6E, 0x33, 0x33, 0x3E, 0x30, 0x1F }, // U+0067 (g) + { 0x07, 0x06, 0x36, 0x6E, 0x66, 0x66, 0x67, 0x00 }, // U+0068 (h) + { 0x0C, 0x00, 0x0E, 0x0C, 0x0C, 0x0C, 0x1E, 0x00 }, // U+0069 (i) + { 0x30, 0x00, 0x30, 0x30, 0x30, 0x33, 0x33, 0x1E }, // U+006A (j) + { 0x07, 0x06, 0x66, 0x36, 0x1E, 0x36, 0x67, 0x00 }, // U+006B (k) + { 0x0E, 0x0C, 0x0C, 0x0C, 0x0C, 0x0C, 0x1E, 0x00 }, // U+006C (l) + { 0x00, 0x00, 0x33, 0x7F, 0x7F, 0x6B, 0x63, 0x00 }, // U+006D (m) + { 0x00, 0x00, 0x1F, 0x33, 0x33, 0x33, 0x33, 0x00 }, // U+006E (n) + { 0x00, 0x00, 0x1E, 0x33, 0x33, 0x33, 0x1E, 0x00 }, // U+006F (o) + { 0x00, 0x00, 0x3B, 0x66, 0x66, 0x3E, 0x06, 0x0F }, // U+0070 (p) + { 0x00, 0x00, 0x6E, 0x33, 0x33, 0x3E, 0x30, 0x78 }, // U+0071 (q) + { 0x00, 0x00, 0x3B, 0x6E, 0x66, 0x06, 0x0F, 0x00 }, // U+0072 (r) + { 0x00, 0x00, 0x3E, 0x03, 0x1E, 0x30, 0x1F, 0x00 }, // U+0073 (s) + { 0x08, 0x0C, 0x3E, 0x0C, 0x0C, 0x2C, 0x18, 0x00 }, // U+0074 (t) + { 0x00, 0x00, 0x33, 0x33, 0x33, 0x33, 0x6E, 0x00 }, // U+0075 (u) + { 0x00, 0x00, 0x33, 0x33, 0x33, 0x1E, 0x0C, 0x00 }, // U+0076 (v) + { 0x00, 0x00, 0x63, 0x6B, 0x7F, 0x7F, 0x36, 0x00 }, // U+0077 (w) + { 0x00, 0x00, 0x63, 0x36, 0x1C, 0x36, 0x63, 0x00 }, // U+0078 (x) + { 0x00, 0x00, 0x33, 0x33, 0x33, 0x3E, 0x30, 0x1F }, // U+0079 (y) + { 0x00, 0x00, 0x3F, 0x19, 0x0C, 0x26, 0x3F, 0x00 }, // U+007A (z) + { 0x38, 0x0C, 0x0C, 0x07, 0x0C, 0x0C, 0x38, 0x00 }, // U+007B ({) + { 0x18, 0x18, 0x18, 0x00, 0x18, 0x18, 0x18, 0x00 }, // U+007C (|) + { 0x07, 0x0C, 0x0C, 0x38, 0x0C, 0x0C, 0x07, 0x00 }, // U+007D (}) + { 0x6E, 0x3B, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // U+007E (~) + { 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 } // U+007F +}; + +// FIXME: This assumes 32 bit BGR (Blue-Green-Red) palette +enum BGRColor : u32 { + Black = 0, + Blue = 0x0000FF, + Green = 0x00FF00, + Cyan = 0x0000FFFF, + Red = 0xFF0000, + Magenta = 0x00FF00FF, + Brown = 0x00964B00, + LightGray = 0x00D3D3D3, + DarkGray = 0x00A9A9A9, + BrightBlue = 0x0ADD8E6, + BrightGreen = 0x0090EE90, + BrightCyan = 0x00E0FFFF, + BrightRed = 0x00D70A53, + BrightMagenta = 0x00F984E5, + Yellow = 0x00FFE135, + White = 0x00FFFFFF, +}; + +static inline BGRColor convert_standard_color_to_bgr_color(Console::Color color) +{ + switch (color) { + case Console::Color::Black: + return BGRColor::Black; + case Console::Color::Red: + return BGRColor::Red; + case Console::Color::Brown: + return BGRColor::Brown; + case Console::Color::Blue: + return BGRColor::Blue; + case Console::Color::Magenta: + return BGRColor::Magenta; + case Console::Color::Green: + return BGRColor::Green; + case Console::Color::Cyan: + return BGRColor::Cyan; + case Console::Color::LightGray: + return BGRColor::LightGray; + case Console::Color::DarkGray: + return BGRColor::DarkGray; + case Console::Color::BrightRed: + return BGRColor::BrightRed; + case Console::Color::BrightGreen: + return BGRColor::BrightGreen; + case Console::Color::Yellow: + return BGRColor::Yellow; + case Console::Color::BrightBlue: + return BGRColor::BrightBlue; + case Console::Color::BrightMagenta: + return BGRColor::BrightMagenta; + case Console::Color::BrightCyan: + return BGRColor::BrightCyan; + case Console::Color::White: + return BGRColor::White; + default: + VERIFY_NOT_REACHED(); + } +} + +NonnullRefPtr<FramebufferConsole> FramebufferConsole::initialize(PhysicalAddress framebuffer_address, size_t width, size_t height, size_t pitch) +{ + return adopt_ref(*new FramebufferConsole(framebuffer_address, width, height, pitch)); +} + +FramebufferConsole::FramebufferConsole(PhysicalAddress framebuffer_address, size_t width, size_t height, size_t pitch) + : Console(width, height) + , m_framebuffer_address(framebuffer_address) + , m_pitch(pitch) +{ + dbgln("Framebuffer Console: taking {} bytes", page_round_up(pitch * height)); + m_framebuffer_region = MM.allocate_kernel_region(m_framebuffer_address, page_round_up(pitch * height), "Framebuffer Console", Region::Access::Read | Region::Access::Write, Region::Cacheable::Yes); + VERIFY(m_framebuffer_region); + + // Just to start cleanly, we clean the entire framebuffer + memset(m_framebuffer_region->vaddr().as_ptr(), 0, pitch * height); +} + +size_t FramebufferConsole::bytes_per_base_glyph() const +{ + // FIXME: We assume we have 32 bit bpp framebuffer. + return 8 * 32; +} +size_t FramebufferConsole::chars_per_line() const +{ + return width() / bytes_per_base_glyph(); +} + +void FramebufferConsole::set_cursor(size_t, size_t) +{ +} +void FramebufferConsole::hide_cursor() +{ +} +void FramebufferConsole::show_cursor() +{ +} + +void FramebufferConsole::clear(size_t x, size_t y, size_t length) const +{ + ScopedSpinLock lock(m_lock); + if (x == 0 && length == max_column()) { + // if we need to clear the entire row, just clean it with quick memset :) + auto* offset_in_framebuffer = (u32*)m_framebuffer_region->vaddr().offset(x * sizeof(u32) * 8).offset(y * 8 * sizeof(u32) * width()).as_ptr(); + for (size_t current_x = 0; current_x < 8; current_x++) { + memset(offset_in_framebuffer, 0, width() * sizeof(u32)); + offset_in_framebuffer = (u32*)((u8*)offset_in_framebuffer + width() * 4); + } + return; + } + for (size_t index = 0; index < length; index++) { + if (x >= max_column()) { + x = 0; + y++; + if (y >= max_row()) + y = 0; + } + clear_glyph(x, y); + } +} + +void FramebufferConsole::clear_glyph(size_t x, size_t y) const +{ + VERIFY(m_lock.is_locked()); + auto* offset_in_framebuffer = (u32*)m_framebuffer_region->vaddr().offset(x * sizeof(u32) * 8).offset(y * 8 * sizeof(u32) * width()).as_ptr(); + for (size_t current_x = 0; current_x < 8; current_x++) { + memset(offset_in_framebuffer, 0, 8 * sizeof(u32)); + offset_in_framebuffer = (u32*)((u8*)offset_in_framebuffer + width() * sizeof(u32)); + } +} + +void FramebufferConsole::enable() +{ + ScopedSpinLock lock(m_lock); + memset(m_framebuffer_region->vaddr().as_ptr(), 0, height() * width() * sizeof(u32)); + m_enabled.store(true); +} +void FramebufferConsole::disable() +{ + ScopedSpinLock lock(m_lock); + m_enabled.store(false); +} + +void FramebufferConsole::write(size_t x, size_t y, char ch, Color background, Color foreground) const +{ + ScopedSpinLock lock(m_lock); + if (!m_enabled.load()) + return; + if (ch == '\r' || ch == '\n') { + m_x = 0; + m_y += 1; + if (m_y >= max_row()) + m_y = 0; + return; + } + if ((int)ch < 0x20 || (int)ch == 0x7f) { + // FIXME: There's no point in printing empty glyphs... + // Maybe try to add these special glyphs and print them. + return; + } + clear_glyph(x, y); + auto* offset_in_framebuffer = (u32*)m_framebuffer_region->vaddr().offset(x * sizeof(u32) * 8).offset(y * 8 * sizeof(u32) * width()).as_ptr(); + int current_bitpixels = 0; + int current_bitpixel = 0; + auto bitmap = font8x8_basic[(int)ch]; + bool set; + BGRColor foreground_color = convert_standard_color_to_bgr_color(foreground); + BGRColor background_color = convert_standard_color_to_bgr_color(background); + for (current_bitpixels = 0; current_bitpixels < 8; current_bitpixels++) { + for (current_bitpixel = 0; current_bitpixel < 8; current_bitpixel++) { + set = bitmap[current_bitpixels] & (1 << current_bitpixel); + if (set) { + offset_in_framebuffer[current_bitpixel] = foreground_color; + } else { + offset_in_framebuffer[current_bitpixel] = background_color; + } + } + offset_in_framebuffer = (u32*)((u8*)offset_in_framebuffer + width() * 4); + } + m_x = x + 1; + if (m_x >= max_column()) { + m_x = 0; + m_y = y + 1; + if (m_y >= max_row()) + m_y = 0; + } +} + +void FramebufferConsole::write(size_t, size_t, String, Color, Color) const +{ + TODO(); +} +void FramebufferConsole::write(size_t x, size_t y, char ch) const +{ + write(x, y, ch, m_default_background_color, m_default_foreground_color); +} +void FramebufferConsole::write(size_t, size_t, String) const +{ + TODO(); +} + +void FramebufferConsole::write(char ch) const +{ + write(m_x, m_y, ch, m_default_background_color, m_default_foreground_color); +} + +} diff --git a/Kernel/Graphics/Console/FramebufferConsole.h b/Kernel/Graphics/Console/FramebufferConsole.h new file mode 100644 index 0000000000..336d117367 --- /dev/null +++ b/Kernel/Graphics/Console/FramebufferConsole.h @@ -0,0 +1,50 @@ +/* + * Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#pragma once + +#include <AK/RefCounted.h> +#include <AK/Types.h> +#include <Kernel/Graphics/Console/Console.h> +#include <Kernel/PhysicalAddress.h> + +namespace Kernel::Graphics { +class FramebufferConsole final : public Console { +public: + static NonnullRefPtr<FramebufferConsole> initialize(PhysicalAddress, size_t width, size_t height, size_t bpp); + + virtual size_t bytes_per_base_glyph() const override; + virtual size_t chars_per_line() const override; + + virtual size_t max_column() const { return m_width / 8; } + virtual size_t max_row() const { return m_height / 8; } + + virtual bool is_hardware_paged_capable() const override { return false; } + virtual bool has_hardware_cursor() const override { return false; } + + virtual void set_cursor(size_t x, size_t y) override; + virtual void hide_cursor() override; + virtual void show_cursor() override; + + virtual void clear(size_t x, size_t y, size_t length) const override; + virtual void write(size_t x, size_t y, char ch, Color background, Color foreground) const override; + virtual void write(size_t x, size_t y, String cstring, Color background, Color foreground) const override; + virtual void write(size_t x, size_t y, char ch) const override; + virtual void write(size_t x, size_t y, String) const override; + virtual void write(char ch) const override; + + virtual void enable() override; + virtual void disable() override; + +protected: + void clear_glyph(size_t x, size_t y) const; + FramebufferConsole(PhysicalAddress, size_t width, size_t height, size_t bpp); + OwnPtr<Region> m_framebuffer_region; + PhysicalAddress m_framebuffer_address; + size_t m_pitch; + mutable SpinLock<u8> m_lock; +}; +} diff --git a/Kernel/Graphics/Console/TextModeConsole.cpp b/Kernel/Graphics/Console/TextModeConsole.cpp new file mode 100644 index 0000000000..847f899ffe --- /dev/null +++ b/Kernel/Graphics/Console/TextModeConsole.cpp @@ -0,0 +1,203 @@ +/* + * Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#include <Kernel/Graphics/Console/TextModeConsole.h> +#include <Kernel/Graphics/GraphicsManagement.h> +#include <Kernel/IO.h> + +namespace Kernel::Graphics { + +UNMAP_AFTER_INIT NonnullRefPtr<TextModeConsole> TextModeConsole::initialize(const VGACompatibleAdapter& adapter) +{ + return adopt_ref(*new TextModeConsole(adapter)); +} + +UNMAP_AFTER_INIT TextModeConsole::TextModeConsole(const VGACompatibleAdapter& adapter) + : VGAConsole(adapter, VGAConsole::Mode::TextMode, 80, 25) + , m_current_vga_window(m_vga_region->vaddr().offset(0x18000).as_ptr()) +{ + for (size_t index = 0; index < height(); index++) { + clear_vga_row(index); + } + dbgln("Text mode console initialized!"); +} + +enum VGAColor : u8 { + Black = 0, + Blue, + Green, + Cyan, + Red, + Magenta, + Brown, + LightGray, + DarkGray, + BrightBlue, + BrightGreen, + BrightCyan, + BrightRed, + BrightMagenta, + Yellow, + White, +}; + +static inline VGAColor convert_standard_color_to_vga_color(Console::Color color) +{ + switch (color) { + case Console::Color::Black: + return VGAColor::Black; + case Console::Color::Red: + return VGAColor::Red; + case Console::Color::Brown: + return VGAColor::Brown; + case Console::Color::Blue: + return VGAColor::Blue; + case Console::Color::Magenta: + return VGAColor::Magenta; + case Console::Color::Green: + return VGAColor::Green; + case Console::Color::Cyan: + return VGAColor::Cyan; + case Console::Color::LightGray: + return VGAColor::LightGray; + case Console::Color::DarkGray: + return VGAColor::DarkGray; + case Console::Color::BrightRed: + return VGAColor::BrightRed; + case Console::Color::BrightGreen: + return VGAColor::BrightGreen; + case Console::Color::Yellow: + return VGAColor::Yellow; + case Console::Color::BrightBlue: + return VGAColor::BrightBlue; + case Console::Color::BrightMagenta: + return VGAColor::BrightMagenta; + case Console::Color::BrightCyan: + return VGAColor::BrightCyan; + case Console::Color::White: + return VGAColor::White; + default: + VERIFY_NOT_REACHED(); + } +} + +void TextModeConsole::set_cursor(size_t x, size_t y) +{ + ScopedSpinLock main_lock(GraphicsManagement::the().main_vga_lock()); + ScopedSpinLock lock(m_vga_lock); + m_cursor_x = x; + m_cursor_y = y; + u16 value = m_current_vga_start_address + (y * width() + x); + IO::out8(0x3d4, 0x0e); + IO::out8(0x3d5, MSB(value)); + IO::out8(0x3d4, 0x0f); + IO::out8(0x3d5, LSB(value)); +} +void TextModeConsole::hide_cursor() +{ + ScopedSpinLock main_lock(GraphicsManagement::the().main_vga_lock()); + ScopedSpinLock lock(m_vga_lock); + IO::out8(0x3D4, 0xA); + IO::out8(0x3D5, 0x20); +} +void TextModeConsole::show_cursor() +{ + ScopedSpinLock main_lock(GraphicsManagement::the().main_vga_lock()); + ScopedSpinLock lock(m_vga_lock); + IO::out8(0x3D4, 0xA); + IO::out8(0x3D5, 0x20); +} + +void TextModeConsole::clear(size_t x, size_t y, size_t length) const +{ + ScopedSpinLock lock(m_vga_lock); + auto* buf = (u16*)(m_current_vga_window + (x * 2) + (y * width() * 2)); + for (size_t index = 0; index < length; index++) { + buf[index] = 0x0720; + } +} +void TextModeConsole::write(size_t x, size_t y, char ch) const +{ + ScopedSpinLock lock(m_vga_lock); + auto* buf = (u16*)(m_current_vga_window + (x * 2) + (y * width() * 2)); + *buf = (m_default_foreground_color << 8) | (m_default_background_color << 12) | ch; + m_x = x + 1; + if (m_x >= max_column()) { + m_x = 0; + m_y = y + 1; + if (m_y >= max_row()) + m_y = 0; + } +} +void TextModeConsole::write(size_t x, size_t y, String cstring) const +{ + ScopedSpinLock lock(m_vga_lock); + auto* buf = (u16*)(m_current_vga_window + (x * 2) + (y * width() * 2)); + u16 color_mask = (m_default_foreground_color << 8) | (m_default_background_color << 12); + for (size_t index = 0; index < cstring.length(); index++) { + buf[index] = color_mask | cstring[index]; + } + m_x = x + cstring.length(); + if (m_x >= max_column()) { + m_x = 0; + m_y = y + 1; + if (m_y >= max_row()) + m_y = 0; + } +} +void TextModeConsole::write(size_t x, size_t y, char ch, Color background, Color foreground) const +{ + ScopedSpinLock lock(m_vga_lock); + auto* buf = (u16*)(m_current_vga_window + (x * 2) + (y * width() * 2)); + *buf = foreground << 8 | background << 12 | ch; + m_x = x + 1; + if (m_x >= max_column()) { + m_x = 0; + m_y = y + 1; + if (m_y >= max_row()) + m_y = 0; + } +} +void TextModeConsole::write(size_t x, size_t y, String cstring, Color background, Color foreground) const +{ + ScopedSpinLock lock(m_vga_lock); + auto* buf = (u16*)(m_current_vga_window + (x * 2) + (y * width() * 2)); + u16 color_mask = foreground << 8 | background << 12; + for (size_t index = 0; index < cstring.length(); index++) { + buf[index] = color_mask | cstring[index]; + } + m_x = x + cstring.length(); + if (m_x >= max_column()) { + m_x = 0; + m_y = y + 1; + if (m_y >= max_row()) + m_y = 0; + } +} + +void TextModeConsole::clear_vga_row(u16 row) +{ + clear(row * width(), width(), width()); +} + +void TextModeConsole::set_vga_start_row(u16 row) +{ + ScopedSpinLock lock(m_vga_lock); + m_vga_start_row = row; + m_current_vga_start_address = row * width(); + m_current_vga_window = m_current_vga_window + row * width() * bytes_per_base_glyph(); + IO::out8(0x3d4, 0x0c); + IO::out8(0x3d5, MSB(m_current_vga_start_address)); + IO::out8(0x3d4, 0x0d); + IO::out8(0x3d5, LSB(m_current_vga_start_address)); +} + +void TextModeConsole::write(char ch) const +{ + write(m_x, m_y, ch); +} + +} diff --git a/Kernel/Graphics/Console/TextModeConsole.h b/Kernel/Graphics/Console/TextModeConsole.h new file mode 100644 index 0000000000..c465b63f39 --- /dev/null +++ b/Kernel/Graphics/Console/TextModeConsole.h @@ -0,0 +1,50 @@ +/* + * Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#pragma once + +#include <AK/RefCounted.h> +#include <AK/Types.h> +#include <Kernel/Graphics/Console/VGAConsole.h> +#include <Kernel/SpinLock.h> + +namespace Kernel::Graphics { +class TextModeConsole final : public VGAConsole { +public: + static NonnullRefPtr<TextModeConsole> initialize(const VGACompatibleAdapter& adapter); + virtual size_t chars_per_line() const override { return width(); }; + + virtual bool has_hardware_cursor() const override { return true; } + virtual bool is_hardware_paged_capable() const override { return true; } + + virtual size_t bytes_per_base_glyph() const override { return 2; } + virtual void set_cursor(size_t x, size_t y) override; + virtual void hide_cursor() override; + virtual void show_cursor() override; + virtual void clear(size_t x, size_t y, size_t length) const override; + virtual void write(size_t x, size_t y, char ch) const override; + virtual void write(size_t x, size_t y, String cstring) const override; + virtual void write(size_t x, size_t y, char ch, Color background, Color foreground) const override; + virtual void write(size_t x, size_t y, String, Color background, Color foreground) const override; + virtual void write(char ch) const override; + + virtual void enable() override { } + virtual void disable() override { VERIFY_NOT_REACHED(); } + +private: + void clear_vga_row(u16 row); + void set_vga_start_row(u16 row); + + explicit TextModeConsole(const VGACompatibleAdapter&); + + mutable SpinLock<u8> m_vga_lock; + u16 m_vga_start_row { 0 }; + u16 m_current_vga_start_address { 0 }; + u8* m_current_vga_window { nullptr }; + u16 m_cursor_x { 0 }; + u16 m_cursor_y { 0 }; +}; +} diff --git a/Kernel/Graphics/Console/VGAConsole.cpp b/Kernel/Graphics/Console/VGAConsole.cpp new file mode 100644 index 0000000000..70472b55e7 --- /dev/null +++ b/Kernel/Graphics/Console/VGAConsole.cpp @@ -0,0 +1,19 @@ +/* + * Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#include <Kernel/Graphics/Console/VGAConsole.h> + +namespace Kernel::Graphics { + +UNMAP_AFTER_INIT VGAConsole::VGAConsole(const VGACompatibleAdapter& adapter, Mode mode, size_t width, size_t height) + : Console(width, height) + , m_vga_region(MM.allocate_kernel_region(PhysicalAddress(0xa0000), page_round_up(0xc0000 - 0xa0000), "VGA Display", Region::Access::Read | Region::Access::Write).release_nonnull()) + , m_adapter(adapter) + , m_mode(mode) +{ +} + +} diff --git a/Kernel/Graphics/Console/VGAConsole.h b/Kernel/Graphics/Console/VGAConsole.h new file mode 100644 index 0000000000..e82f1d5e59 --- /dev/null +++ b/Kernel/Graphics/Console/VGAConsole.h @@ -0,0 +1,39 @@ +/* + * Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + +#pragma once + +#include <AK/RefCounted.h> +#include <AK/Types.h> +#include <Kernel/Graphics/Console/Console.h> +#include <Kernel/Graphics/VGACompatibleAdapter.h> + +namespace Kernel::Graphics { +class VGAConsole : public Console { +public: + // Note: these are the modes we will support and only these + enum class Mode { + TextMode = 1, // Text Mode + Colored256, // 320x200 256 color mode + Colored16, // 640x480 16 color mode + }; + +public: + static NonnullRefPtr<VGAConsole> initialize(const VGACompatibleAdapter&, Mode, size_t width, size_t height); + + virtual bool is_hardware_paged_capable() const override { return false; } + virtual bool has_hardware_cursor() const override { return false; } + + virtual ~VGAConsole() = default; + +protected: + VGAConsole(const VGACompatibleAdapter&, Mode, size_t width, size_t height); + + NonnullOwnPtr<Region> m_vga_region; + NonnullRefPtr<VGACompatibleAdapter> m_adapter; + const Mode m_mode; +}; +} diff --git a/Kernel/Graphics/FramebufferDevice.cpp b/Kernel/Graphics/FramebufferDevice.cpp index c6d45774cf..fd1bd27277 100644 --- a/Kernel/Graphics/FramebufferDevice.cpp +++ b/Kernel/Graphics/FramebufferDevice.cpp @@ -16,6 +16,8 @@ #include <LibC/errno_numbers.h> #include <LibC/sys/ioctl_numbers.h> +#include <Kernel/Panic.h> + namespace Kernel { KResultOr<Region*> FramebufferDevice::mmap(Process& process, FileDescription&, const Range& range, u64 offset, int prot, bool shared) @@ -28,16 +30,51 @@ KResultOr<Region*> FramebufferDevice::mmap(Process& process, FileDescription&, c if (range.size() != page_round_up(framebuffer_size_in_bytes())) return EOVERFLOW; - auto vmobject = AnonymousVMObject::create_for_physical_range(m_framebuffer_address, framebuffer_size_in_bytes()); + // FIXME: We rely on the fact that only the WindowServer will mmap the framebuffer + // and only once when starting to work with it. If other program wants to do so, we need to fix this. + VERIFY(!m_userspace_framebuffer_region); + VERIFY(!m_userspace_real_framebuffer_vmobject); + + auto vmobject = AnonymousVMObject::create_for_physical_range(m_framebuffer_address, page_round_up(framebuffer_size_in_bytes())); if (!vmobject) return ENOMEM; - return process.space().allocate_region_with_vmobject( + m_userspace_real_framebuffer_vmobject = vmobject; + + auto result = process.space().allocate_region_with_vmobject( range, vmobject.release_nonnull(), 0, "Framebuffer", prot, shared); + if (!result.is_error()) { + m_userspace_framebuffer_region = result.value(); + } + return result; +} + +void FramebufferDevice::dectivate_writes() +{ + ScopedSpinLock lock(m_activation_lock); + if (!m_userspace_framebuffer_region) + return; + memcpy(m_swapped_framebuffer_region->vaddr().as_ptr(), m_real_framebuffer_region->vaddr().as_ptr(), page_round_up(framebuffer_size_in_bytes())); + auto vmobject = m_swapped_framebuffer_vmobject; + m_userspace_framebuffer_region->set_vmobject(vmobject.release_nonnull()); + m_userspace_framebuffer_region->remap(); +} +void FramebufferDevice::activate_writes() +{ + ScopedSpinLock lock(m_activation_lock); + if (!m_userspace_framebuffer_region || !m_real_framebuffer_vmobject) + return; + // restore the image we had in the void area + // FIXME: if we happen to have multiple Framebuffers that are writing to that location + // we will experience glitches... + memcpy(m_real_framebuffer_region->vaddr().as_ptr(), m_swapped_framebuffer_region->vaddr().as_ptr(), page_round_up(framebuffer_size_in_bytes())); + auto vmobject = m_userspace_real_framebuffer_vmobject; + m_userspace_framebuffer_region->set_vmobject(vmobject.release_nonnull()); + m_userspace_framebuffer_region->remap(); } String FramebufferDevice::device_name() const @@ -45,6 +82,18 @@ String FramebufferDevice::device_name() const return String::formatted("fb{}", minor()); } +UNMAP_AFTER_INIT void FramebufferDevice::initialize() +{ + m_real_framebuffer_vmobject = AnonymousVMObject::create_for_physical_range(m_framebuffer_address, page_round_up(framebuffer_size_in_bytes())); + VERIFY(m_real_framebuffer_vmobject); + m_real_framebuffer_region = MM.allocate_kernel_region_with_vmobject(*m_real_framebuffer_vmobject, framebuffer_size_in_bytes(), "Framebuffer", Region::Access::Read | Region::Access::Write); + VERIFY(m_real_framebuffer_region); + m_swapped_framebuffer_vmobject = AnonymousVMObject::create_with_size(page_round_up(framebuffer_size_in_bytes()), AllocationStrategy::AllocateNow); + VERIFY(m_swapped_framebuffer_vmobject); + m_swapped_framebuffer_region = MM.allocate_kernel_region_with_vmobject(*m_swapped_framebuffer_vmobject, page_round_up(framebuffer_size_in_bytes()), "Framebuffer Swap (Blank)", Region::Access::Read | Region::Access::Write); + VERIFY(m_swapped_framebuffer_region); +} + UNMAP_AFTER_INIT FramebufferDevice::FramebufferDevice(PhysicalAddress addr, size_t pitch, size_t width, size_t height) : BlockDevice(29, GraphicsManagement::the().current_minor_number()) , m_framebuffer_address(addr) @@ -52,6 +101,10 @@ UNMAP_AFTER_INIT FramebufferDevice::FramebufferDevice(PhysicalAddress addr, size , m_framebuffer_width(width) , m_framebuffer_height(height) { + VERIFY(!m_framebuffer_address.is_null()); + VERIFY(m_framebuffer_pitch); + VERIFY(m_framebuffer_width); + VERIFY(m_framebuffer_height); dbgln("Framebuffer {}: address={}, pitch={}, width={}, height={}", minor(), addr, pitch, width, height); } diff --git a/Kernel/Graphics/FramebufferDevice.h b/Kernel/Graphics/FramebufferDevice.h index 104fc5739d..f52f20ff4c 100644 --- a/Kernel/Graphics/FramebufferDevice.h +++ b/Kernel/Graphics/FramebufferDevice.h @@ -6,11 +6,14 @@ #pragma once +#include <AK/NonnullOwnPtr.h> #include <AK/String.h> #include <AK/Types.h> #include <Kernel/Devices/BlockDevice.h> #include <Kernel/Graphics/GraphicsDevice.h> #include <Kernel/PhysicalAddress.h> +#include <Kernel/SpinLock.h> +#include <Kernel/VM/AnonymousVMObject.h> namespace Kernel { @@ -24,9 +27,12 @@ public: virtual mode_t required_mode() const override { return 0660; } virtual String device_name() const override; + virtual void dectivate_writes(); + virtual void activate_writes(); virtual size_t framebuffer_size_in_bytes() const { return m_framebuffer_pitch * m_framebuffer_height; } virtual ~FramebufferDevice() {}; + void initialize(); protected: virtual bool set_resolution(size_t framebuffer_width, size_t framebuffer_height, size_t framebuffer_pitch); @@ -44,6 +50,17 @@ protected: size_t m_framebuffer_pitch { 0 }; size_t m_framebuffer_width { 0 }; size_t m_framebuffer_height { 0 }; + +private: + SpinLock<u8> m_activation_lock; + + RefPtr<AnonymousVMObject> m_real_framebuffer_vmobject; + RefPtr<AnonymousVMObject> m_swapped_framebuffer_vmobject; + OwnPtr<Region> m_real_framebuffer_region; + OwnPtr<Region> m_swapped_framebuffer_region; + + RefPtr<AnonymousVMObject> m_userspace_real_framebuffer_vmobject; + Region* m_userspace_framebuffer_region { nullptr }; }; } diff --git a/Kernel/Graphics/GraphicsDevice.h b/Kernel/Graphics/GraphicsDevice.h index f4d3ae7e29..2279338f4f 100644 --- a/Kernel/Graphics/GraphicsDevice.h +++ b/Kernel/Graphics/GraphicsDevice.h @@ -25,9 +25,15 @@ public: virtual ~GraphicsDevice() = default; virtual void initialize_framebuffer_devices() = 0; virtual Type type() const = 0; + virtual void enable_consoles() = 0; + virtual void disable_consoles() = 0; + bool consoles_enabled() const { return m_consoles_enabled; } + virtual bool framebuffer_devices_initialized() const = 0; protected: GraphicsDevice() = default; + + bool m_consoles_enabled { false }; }; } diff --git a/Kernel/Graphics/GraphicsManagement.cpp b/Kernel/Graphics/GraphicsManagement.cpp index e329d99700..0c5ad939bd 100644 --- a/Kernel/Graphics/GraphicsManagement.cpp +++ b/Kernel/Graphics/GraphicsManagement.cpp @@ -9,10 +9,14 @@ #include <Kernel/CommandLine.h> #include <Kernel/Debug.h> #include <Kernel/Graphics/BochsGraphicsAdapter.h> +#include <Kernel/Graphics/Console/FramebufferConsole.h> +#include <Kernel/Graphics/Console/TextModeConsole.h> #include <Kernel/Graphics/GraphicsManagement.h> #include <Kernel/Graphics/IntelNativeGraphicsAdapter.h> #include <Kernel/Graphics/VGACompatibleAdapter.h> +#include <Kernel/IO.h> #include <Kernel/Multiboot.h> +#include <Kernel/VM/AnonymousVMObject.h> namespace Kernel { @@ -29,10 +33,24 @@ bool GraphicsManagement::is_initialized() } UNMAP_AFTER_INIT GraphicsManagement::GraphicsManagement() - : m_textmode_enabled(kernel_command_line().is_text_mode()) + : m_vga_font_region(MM.allocate_kernel_region(PAGE_SIZE, "VGA font", Region::Access::Read | Region::Access::Write, AllocationStrategy::AllocateNow).release_nonnull()) + , m_framebuffer_devices_allowed(!kernel_command_line().is_no_framebuffer_devices_mode()) { } +void GraphicsManagement::deactivate_graphical_mode() +{ + for (auto& graphics_device : m_graphics_devices) { + graphics_device.enable_consoles(); + } +} +void GraphicsManagement::activate_graphical_mode() +{ + for (auto& graphics_device : m_graphics_devices) { + graphics_device.disable_consoles(); + } +} + UNMAP_AFTER_INIT RefPtr<GraphicsDevice> GraphicsManagement::determine_graphics_device(PCI::Address address, PCI::ID id) const { if ((id.vendor_id == 0x1234 && id.device_id == 0x1111) || (id.vendor_id == 0x80ee && id.device_id == 0xbeef)) { @@ -44,31 +62,93 @@ UNMAP_AFTER_INIT RefPtr<GraphicsDevice> GraphicsManagement::determine_graphics_d if (!adapter.is_null()) return adapter; } - VERIFY(multiboot_info_ptr->framebuffer_type == MULTIBOOT_FRAMEBUFFER_TYPE_RGB || multiboot_info_ptr->framebuffer_type == MULTIBOOT_FRAMEBUFFER_TYPE_EGA_TEXT); - return VGACompatibleAdapter::initialize_with_preset_resolution(address, - PhysicalAddress((u32)(multiboot_info_ptr->framebuffer_addr)), - multiboot_info_ptr->framebuffer_pitch, - multiboot_info_ptr->framebuffer_width, - multiboot_info_ptr->framebuffer_height); + if (multiboot_info_ptr->framebuffer_type == MULTIBOOT_FRAMEBUFFER_TYPE_RGB) { + dmesgln("Graphics: Using a preset resolution from the bootloader"); + return VGACompatibleAdapter::initialize_with_preset_resolution(address, + PhysicalAddress((u32)(multiboot_info_ptr->framebuffer_addr)), + multiboot_info_ptr->framebuffer_pitch, + multiboot_info_ptr->framebuffer_width, + multiboot_info_ptr->framebuffer_height); + } + return VGACompatibleAdapter::initialize(address); } return {}; } UNMAP_AFTER_INIT bool GraphicsManagement::initialize() { - if (kernel_command_line().is_text_mode()) { - dbgln("Text mode enabled"); - return false; + + /* Explanation on the flow when not requesting to force not creating any + * framebuffer devices: + * If the user wants to use a Console instead of the graphical environment, + * they doesn't need to request text mode. + * Graphical mode might not be accessible on bare-metal hardware because + * the bootloader didn't set a framebuffer and we don't have a native driver + * to set a framebuffer for it. We don't have VBE modesetting capabilities + * in the kernel yet, so what will happen is one of the following situations: + * 1. The bootloader didn't specify settings of a pre-set framebuffer. The + * kernel has a native driver for a detected display adapter, therefore + * the kernel can still set a framebuffer. + * 2. The bootloader specified settings of a pre-set framebuffer, and the + * kernel has a native driver for a detected display adapter, therefore + * the kernel can still set a framebuffer and change the settings of it. + * In that situation, the kernel will simply ignore the Multiboot pre-set + * framebuffer. + * 2. The bootloader specified settings of a pre-set framebuffer, and the + * kernel does not have a native driver for a detected display adapter, + * therefore the kernel will use the pre-set framebuffer. Modesetting is not + * availabe in this situation. + * 3. The bootloader didn't specify settings of a pre-set framebuffer, and + * the kernel does not have a native driver for a detected display adapter, + * therefore the kernel will try to initialize a VGA text mode console. + * In that situation, the kernel will assume that VGA text mode was already + * initialized, but will still try to modeset it. No switching to graphical + * enviroment is allowed in this case. + * + * By default, the kernel assumes that no framebuffer was created until it + * was proven that there's an existing framebuffer or we can modeset the + * screen resolution to create a framebuffer. + * + * If the user requests to force no initialization of framebuffer devices + * the same flow above will happen, except that no framebuffer device will + * be created, so SystemServer will not try to initialize WindowServer. + */ + + if (kernel_command_line().is_no_framebuffer_devices_mode()) { + dbgln("Forcing no initialization of framebuffer devices"); } PCI::enumerate([&](const PCI::Address& address, PCI::ID id) { + // Note: Each graphics controller will try to set its native screen resolution + // upon creation. Later on, if we don't want to have framebuffer devices, a + // framebuffer console will take the control instead. auto adapter = determine_graphics_device(address, id); if (!adapter) return; - adapter->initialize_framebuffer_devices(); - m_graphics_devices.append(adapter.release_nonnull()); + + // If IO space is enabled, this VGA adapter is operating in VGA mode. + if (adapter->type() == GraphicsDevice::Type::VGACompatible && PCI::is_io_space_enabled(address)) { + VERIFY(m_vga_adapter.is_null()); + dbgln("Graphics adapter @ {} is operating in VGA mode", address); + m_vga_adapter = adapter; + } + auto display_adapter = adapter.release_nonnull(); + m_graphics_devices.append(display_adapter); + if (!m_framebuffer_devices_allowed) { + display_adapter->enable_consoles(); + return; + } + display_adapter->initialize_framebuffer_devices(); }); return true; } +bool GraphicsManagement::framebuffer_devices_exist() const +{ + for (auto& graphics_device : m_graphics_devices) { + if (graphics_device.framebuffer_devices_initialized()) + return true; + } + return false; +} } diff --git a/Kernel/Graphics/GraphicsManagement.h b/Kernel/Graphics/GraphicsManagement.h index bcf0cded39..da830259a2 100644 --- a/Kernel/Graphics/GraphicsManagement.h +++ b/Kernel/Graphics/GraphicsManagement.h @@ -6,16 +6,26 @@ #pragma once +#include <AK/NonnullOwnPtr.h> #include <AK/NonnullRefPtr.h> #include <AK/NonnullRefPtrVector.h> #include <AK/Types.h> +#include <Kernel/Graphics/Console/Console.h> #include <Kernel/Graphics/GraphicsDevice.h> +#include <Kernel/Graphics/VGACompatibleAdapter.h> #include <Kernel/PCI/Definitions.h> +#include <Kernel/VM/Region.h> namespace Kernel { +class BochsGraphicsAdapter; +class IntelNativeGraphicsAdapter; +class VGACompatibleAdapter; class GraphicsManagement { - AK_MAKE_ETERNAL; + friend class BochsGraphicsAdapter; + friend class IntelNativeGraphicsAdapter; + friend class VGACompatibleAdapter; + AK_MAKE_ETERNAL public: static GraphicsManagement& the(); @@ -25,14 +35,28 @@ public: unsigned current_minor_number() { return m_current_minor_number++; }; GraphicsManagement(); - bool is_text_mode_enabled() const { return m_textmode_enabled; } + bool framebuffer_devices_allowed() const { return m_framebuffer_devices_allowed; } + bool framebuffer_devices_exist() const; + + SpinLock<u8>& main_vga_lock() { return m_main_vga_lock; } + RefPtr<Graphics::Console> console() const { return m_console; } + + void deactivate_graphical_mode(); + void activate_graphical_mode(); private: RefPtr<GraphicsDevice> determine_graphics_device(PCI::Address address, PCI::ID id) const; NonnullRefPtrVector<GraphicsDevice> m_graphics_devices; + NonnullOwnPtr<Region> m_vga_font_region; + RefPtr<Graphics::Console> m_console; + + // Note: there could be multiple VGA adapters, but only one can operate in VGA mode + RefPtr<VGACompatibleAdapter> m_vga_adapter; unsigned m_current_minor_number { 0 }; - bool m_textmode_enabled; + const bool m_framebuffer_devices_allowed; + + SpinLock<u8> m_main_vga_lock; }; } diff --git a/Kernel/Graphics/IntelNativeGraphicsAdapter.cpp b/Kernel/Graphics/IntelNativeGraphicsAdapter.cpp index fa3b0b4792..2ce0266920 100644 --- a/Kernel/Graphics/IntelNativeGraphicsAdapter.cpp +++ b/Kernel/Graphics/IntelNativeGraphicsAdapter.cpp @@ -4,7 +4,9 @@ * SPDX-License-Identifier: BSD-2-Clause */ +#include <Kernel/Graphics/Console/FramebufferConsole.h> #include <Kernel/Graphics/Definitions.h> +#include <Kernel/Graphics/GraphicsManagement.h> #include <Kernel/Graphics/IntelNativeGraphicsAdapter.h> #include <Kernel/IO.h> #include <Kernel/PhysicalAddress.h> @@ -165,7 +167,7 @@ Optional<IntelNativeGraphicsAdapter::PLLSettings> IntelNativeGraphicsAdapter::cr } IntelNativeGraphicsAdapter::IntelNativeGraphicsAdapter(PCI::Address address) - : PCI::DeviceController(address) + : VGACompatibleAdapter(address) , m_registers(PCI::get_BAR0(address) & 0xfffffffc) , m_framebuffer_addr(PCI::get_BAR2(address) & 0xfffffffc) { @@ -182,10 +184,24 @@ IntelNativeGraphicsAdapter::IntelNativeGraphicsAdapter(PCI::Address address) set_gmbus_pin_pair(GMBusPinPair::DedicatedAnalog); } gmbus_read_edid(); - auto modesetting = calculate_modesetting_from_edid(m_crt_edid, 0); - dmesgln("Intel Native Graphics Adapter @ {}, preferred resolution is {:d}x{:d}", address, modesetting.horizontal.active, modesetting.vertical.active); + auto modesetting = calculate_modesetting_from_edid(m_crt_edid, 0); + dmesgln("Intel Native Graphics Adapter @ {}, preferred resolution is {:d}x{:d}", pci_address(), modesetting.horizontal.active, modesetting.vertical.active); set_crt_resolution(modesetting.horizontal.active, modesetting.vertical.active); + auto framebuffer_address = PhysicalAddress(PCI::get_BAR2(pci_address()) & 0xfffffff0); + VERIFY(!framebuffer_address.is_null()); + VERIFY(m_framebuffer_pitch != 0); + VERIFY(m_framebuffer_height != 0); + VERIFY(m_framebuffer_width != 0); + m_framebuffer_console = Graphics::FramebufferConsole::initialize(framebuffer_address, m_framebuffer_width, m_framebuffer_height, m_framebuffer_pitch); + // FIXME: This is a very wrong way to do this... + GraphicsManagement::the().m_console = m_framebuffer_console; +} + +void IntelNativeGraphicsAdapter::enable_vga_plane() +{ + VERIFY(m_control_lock.is_locked()); + VERIFY(m_modeset_lock.is_locked()); } static inline const char* convert_register_index_to_string(IntelGraphics::RegisterIndex index) @@ -269,6 +285,7 @@ bool IntelNativeGraphicsAdapter::pipe_a_enabled() const VERIFY(m_control_lock.is_locked()); return read_from_register(IntelGraphics::RegisterIndex::PipeAConf) & (1 << 30); } + bool IntelNativeGraphicsAdapter::pipe_b_enabled() const { VERIFY(m_control_lock.is_locked()); @@ -396,7 +413,7 @@ bool IntelNativeGraphicsAdapter::set_crt_resolution(size_t width, size_t height) VERIFY_NOT_REACHED(); auto settings = pll_settings.value(); dbgln_if(INTEL_GRAPHICS_DEBUG, "PLL settings for {} {} {} {} {}", settings.n, settings.m1, settings.m2, settings.p1, settings.p2); - set_dpll_registers(pll_settings.value(), dac_multiplier); + enable_dpll_without_vga(pll_settings.value(), dac_multiplier); set_display_timings(modesetting); auto address = PhysicalAddress(PCI::get_BAR2(pci_address()) & 0xfffffff0); VERIFY(!address.is_null()); @@ -404,7 +421,7 @@ bool IntelNativeGraphicsAdapter::set_crt_resolution(size_t width, size_t height) m_framebuffer_width = width; m_framebuffer_height = height; - m_framebuffer_stride = width * 4; + m_framebuffer_pitch = width * 4; return true; } @@ -538,7 +555,7 @@ void IntelNativeGraphicsAdapter::enable_primary_plane(PhysicalAddress fb_address write_to_register(IntelGraphics::RegisterIndex::DisplayPlaneAControl, (read_from_register(IntelGraphics::RegisterIndex::DisplayPlaneAControl) & (~(0b1111 << 26))) | (0b0110 << 26) | (1 << 31)); } -void IntelNativeGraphicsAdapter::set_dpll_registers(const PLLSettings& settings, size_t dac_multiplier) +void IntelNativeGraphicsAdapter::set_dpll_registers(const PLLSettings& settings) { VERIFY(m_control_lock.is_locked()); VERIFY(m_modeset_lock.is_locked()); @@ -546,6 +563,14 @@ void IntelNativeGraphicsAdapter::set_dpll_registers(const PLLSettings& settings, write_to_register(IntelGraphics::RegisterIndex::DPLLDivisorA1, (settings.m2 - 2) | ((settings.m1 - 2) << 8) | ((settings.n - 2) << 16)); write_to_register(IntelGraphics::RegisterIndex::DPLLControlA, read_from_register(IntelGraphics::RegisterIndex::DPLLControlA) & ~0x80000000); +} + +void IntelNativeGraphicsAdapter::enable_dpll_without_vga(const PLLSettings& settings, size_t dac_multiplier) +{ + VERIFY(m_control_lock.is_locked()); + VERIFY(m_modeset_lock.is_locked()); + + set_dpll_registers(settings); IO::delay(200); @@ -597,9 +622,10 @@ void IntelNativeGraphicsAdapter::initialize_framebuffer_devices() { auto address = PhysicalAddress(PCI::get_BAR2(pci_address()) & 0xfffffff0); VERIFY(!address.is_null()); - VERIFY(m_framebuffer_stride != 0); + VERIFY(m_framebuffer_pitch != 0); VERIFY(m_framebuffer_height != 0); VERIFY(m_framebuffer_width != 0); - m_framebuffer = m_framebuffer = RawFramebufferDevice::create(*this, address, m_framebuffer_stride, m_framebuffer_width, m_framebuffer_height); + m_framebuffer_device = RawFramebufferDevice::create(*this, address, m_framebuffer_pitch, m_framebuffer_width, m_framebuffer_height); + m_framebuffer_device->initialize(); } } diff --git a/Kernel/Graphics/IntelNativeGraphicsAdapter.h b/Kernel/Graphics/IntelNativeGraphicsAdapter.h index a221929118..48cd0fac36 100644 --- a/Kernel/Graphics/IntelNativeGraphicsAdapter.h +++ b/Kernel/Graphics/IntelNativeGraphicsAdapter.h @@ -9,8 +9,8 @@ #include <AK/String.h> #include <AK/Types.h> #include <Kernel/Graphics/Definitions.h> -#include <Kernel/Graphics/GraphicsDevice.h> #include <Kernel/Graphics/RawFramebufferDevice.h> +#include <Kernel/Graphics/VGACompatibleAdapter.h> #include <Kernel/PCI/DeviceController.h> #include <Kernel/PhysicalAddress.h> @@ -47,8 +47,7 @@ enum RegisterIndex { } class IntelNativeGraphicsAdapter final - : public GraphicsDevice - , public PCI::DeviceController { + : public VGACompatibleAdapter { AK_MAKE_ETERNAL public: struct PLLSettings { @@ -129,6 +128,7 @@ private: void enable_output(PhysicalAddress fb_address, size_t width); void disable_vga_emulation(); + void enable_vga_plane(); void disable_dac_output(); void enable_dac_output(); @@ -138,7 +138,9 @@ private: void disable_pipe_b(); void disable_dpll(); - void set_dpll_registers(const PLLSettings&, size_t dac_multiplier); + void set_dpll_registers(const PLLSettings&); + + void enable_dpll_without_vga(const PLLSettings&, size_t dac_multiplier); void set_display_timings(const Graphics::Modesetting&); void enable_pipe_a(); void set_framebuffer_parameters(size_t, size_t); @@ -166,15 +168,7 @@ private: Graphics::VideoInfoBlock m_crt_edid; const PhysicalAddress m_registers; const PhysicalAddress m_framebuffer_addr; - OwnPtr<Region> m_registers_region; - - size_t m_framebuffer_width { 0 }; - size_t m_framebuffer_height { 0 }; - size_t m_framebuffer_stride { 0 }; - -protected: - RefPtr<RawFramebufferDevice> m_framebuffer; }; } diff --git a/Kernel/Graphics/VGACompatibleAdapter.cpp b/Kernel/Graphics/VGACompatibleAdapter.cpp index 80b2e8b0d4..0fd8e647ae 100644 --- a/Kernel/Graphics/VGACompatibleAdapter.cpp +++ b/Kernel/Graphics/VGACompatibleAdapter.cpp @@ -4,7 +4,11 @@ * SPDX-License-Identifier: BSD-2-Clause */ +#include <Kernel/Graphics/Console/FramebufferConsole.h> +#include <Kernel/Graphics/Console/TextModeConsole.h> +#include <Kernel/Graphics/GraphicsManagement.h> #include <Kernel/Graphics/VGACompatibleAdapter.h> +#include <Kernel/IO.h> namespace Kernel { @@ -13,19 +17,55 @@ UNMAP_AFTER_INIT NonnullRefPtr<VGACompatibleAdapter> VGACompatibleAdapter::initi return adopt_ref(*new VGACompatibleAdapter(address, m_framebuffer_address, framebuffer_width, framebuffer_height, framebuffer_pitch)); } +UNMAP_AFTER_INIT NonnullRefPtr<VGACompatibleAdapter> VGACompatibleAdapter::initialize(PCI::Address address) +{ + return adopt_ref(*new VGACompatibleAdapter(address)); +} + UNMAP_AFTER_INIT void VGACompatibleAdapter::initialize_framebuffer_devices() { + // We might not have any pre-set framebuffer, so if that's the case - don't try to initialize one. + if (m_framebuffer_address.is_null()) + return; + VERIFY(m_framebuffer_width); + VERIFY(m_framebuffer_width != 0); + VERIFY(m_framebuffer_height != 0); + VERIFY(m_framebuffer_pitch != 0); + m_framebuffer_device = RawFramebufferDevice::create(*this, m_framebuffer_address, m_framebuffer_width, m_framebuffer_height, m_framebuffer_pitch); + m_framebuffer_device->initialize(); } UNMAP_AFTER_INIT VGACompatibleAdapter::VGACompatibleAdapter(PCI::Address address) : PCI::DeviceController(address) { + m_framebuffer_console = Graphics::TextModeConsole::initialize(*this); + // FIXME: This is a very wrong way to do this... + GraphicsManagement::the().m_console = m_framebuffer_console; } UNMAP_AFTER_INIT VGACompatibleAdapter::VGACompatibleAdapter(PCI::Address address, PhysicalAddress framebuffer_address, size_t framebuffer_width, size_t framebuffer_height, size_t framebuffer_pitch) : PCI::DeviceController(address) + , m_framebuffer_address(framebuffer_address) + , m_framebuffer_width(framebuffer_width) + , m_framebuffer_height(framebuffer_height) + , m_framebuffer_pitch(framebuffer_pitch) +{ + m_framebuffer_console = Graphics::FramebufferConsole::initialize(framebuffer_address, framebuffer_width, framebuffer_height, framebuffer_pitch); +} + +void VGACompatibleAdapter::enable_consoles() +{ + VERIFY(m_framebuffer_console); + if (m_framebuffer_device) + m_framebuffer_device->dectivate_writes(); + m_framebuffer_console->enable(); +} +void VGACompatibleAdapter::disable_consoles() { - m_framebuffer = RawFramebufferDevice::create(*this, framebuffer_address, framebuffer_width, framebuffer_height, framebuffer_pitch); + VERIFY(m_framebuffer_device); + VERIFY(m_framebuffer_console); + m_framebuffer_console->disable(); + m_framebuffer_device->activate_writes(); } } diff --git a/Kernel/Graphics/VGACompatibleAdapter.h b/Kernel/Graphics/VGACompatibleAdapter.h index 75ed5ee946..c1b6572e05 100644 --- a/Kernel/Graphics/VGACompatibleAdapter.h +++ b/Kernel/Graphics/VGACompatibleAdapter.h @@ -8,6 +8,7 @@ #include <AK/String.h> #include <AK/Types.h> +#include <Kernel/Graphics/Console/Console.h> #include <Kernel/Graphics/GraphicsDevice.h> #include <Kernel/Graphics/RawFramebufferDevice.h> #include <Kernel/PCI/DeviceController.h> @@ -15,11 +16,14 @@ namespace Kernel { -class VGACompatibleAdapter final : public GraphicsDevice +class VGACompatibleAdapter : public GraphicsDevice , public PCI::DeviceController { AK_MAKE_ETERNAL public: static NonnullRefPtr<VGACompatibleAdapter> initialize_with_preset_resolution(PCI::Address, PhysicalAddress, size_t framebuffer_width, size_t framebuffer_height, size_t framebuffer_pitch); + static NonnullRefPtr<VGACompatibleAdapter> initialize(PCI::Address); + + virtual bool framebuffer_devices_initialized() const override { return !m_framebuffer_device.is_null(); } protected: explicit VGACompatibleAdapter(PCI::Address); @@ -31,8 +35,17 @@ private: virtual void initialize_framebuffer_devices() override; virtual Type type() const override { return Type::VGACompatible; } + virtual void enable_consoles() override; + virtual void disable_consoles() override; + protected: - RefPtr<RawFramebufferDevice> m_framebuffer; + PhysicalAddress m_framebuffer_address; + size_t m_framebuffer_width { 0 }; + size_t m_framebuffer_height { 0 }; + size_t m_framebuffer_pitch { 0 }; + + RefPtr<RawFramebufferDevice> m_framebuffer_device; + RefPtr<Graphics::Console> m_framebuffer_console; }; } |