2021-03-05 14:23:08 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il>
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <AK/Checked.h>
|
|
|
|
#include <AK/Singleton.h>
|
|
|
|
#include <Kernel/CommandLine.h>
|
|
|
|
#include <Kernel/Debug.h>
|
|
|
|
#include <Kernel/Graphics/BochsGraphicsAdapter.h>
|
2021-04-16 22:58:51 +03:00
|
|
|
#include <Kernel/Graphics/Console/FramebufferConsole.h>
|
|
|
|
#include <Kernel/Graphics/Console/TextModeConsole.h>
|
2021-03-05 14:23:08 +02:00
|
|
|
#include <Kernel/Graphics/GraphicsManagement.h>
|
2021-04-12 22:07:30 +03:00
|
|
|
#include <Kernel/Graphics/IntelNativeGraphicsAdapter.h>
|
2021-03-05 14:23:08 +02:00
|
|
|
#include <Kernel/Graphics/VGACompatibleAdapter.h>
|
2021-04-16 22:58:51 +03:00
|
|
|
#include <Kernel/IO.h>
|
2021-03-05 14:23:08 +02:00
|
|
|
#include <Kernel/Multiboot.h>
|
2021-05-21 11:23:41 +03:00
|
|
|
#include <Kernel/Panic.h>
|
2021-04-16 22:58:51 +03:00
|
|
|
#include <Kernel/VM/AnonymousVMObject.h>
|
2021-03-05 14:23:08 +02:00
|
|
|
|
|
|
|
namespace Kernel {
|
|
|
|
|
|
|
|
static AK::Singleton<GraphicsManagement> s_the;
|
|
|
|
|
|
|
|
GraphicsManagement& GraphicsManagement::the()
|
|
|
|
{
|
|
|
|
return *s_the;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GraphicsManagement::is_initialized()
|
|
|
|
{
|
|
|
|
return s_the.is_initialized();
|
|
|
|
}
|
|
|
|
|
|
|
|
UNMAP_AFTER_INIT GraphicsManagement::GraphicsManagement()
|
2021-04-16 22:58:51 +03:00
|
|
|
: 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())
|
2021-03-05 14:23:08 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2021-04-16 22:58:51 +03:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-05 14:23:08 +02:00
|
|
|
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)) {
|
|
|
|
return BochsGraphicsAdapter::initialize(address);
|
|
|
|
}
|
|
|
|
if (PCI::get_class(address) == 0x3 && PCI::get_subclass(address) == 0x0) {
|
2021-04-12 22:07:30 +03:00
|
|
|
if (id.vendor_id == 0x8086) {
|
|
|
|
auto adapter = IntelNativeGraphicsAdapter::initialize(address);
|
|
|
|
if (!adapter.is_null())
|
|
|
|
return adapter;
|
|
|
|
}
|
2021-04-16 22:58:51 +03:00
|
|
|
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_width,
|
2021-05-17 00:02:47 +03:00
|
|
|
multiboot_info_ptr->framebuffer_height,
|
|
|
|
multiboot_info_ptr->framebuffer_pitch);
|
2021-04-16 22:58:51 +03:00
|
|
|
}
|
|
|
|
return VGACompatibleAdapter::initialize(address);
|
2021-03-05 14:23:08 +02:00
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
UNMAP_AFTER_INIT bool GraphicsManagement::initialize()
|
|
|
|
{
|
2021-04-16 22:58:51 +03:00
|
|
|
|
|
|
|
/* 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
|
2021-05-17 17:48:55 +01:00
|
|
|
* available in this situation.
|
2021-04-16 22:58:51 +03:00
|
|
|
* 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
|
2021-05-17 17:48:55 +01:00
|
|
|
* environment is allowed in this case.
|
2021-04-16 22:58:51 +03:00
|
|
|
*
|
|
|
|
* 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");
|
2021-03-05 14:23:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
PCI::enumerate([&](const PCI::Address& address, PCI::ID id) {
|
2021-04-16 22:58:51 +03:00
|
|
|
// 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.
|
2021-03-05 14:23:08 +02:00
|
|
|
auto adapter = determine_graphics_device(address, id);
|
|
|
|
if (!adapter)
|
|
|
|
return;
|
2021-04-16 22:58:51 +03:00
|
|
|
|
|
|
|
// 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();
|
2021-03-05 14:23:08 +02:00
|
|
|
});
|
2021-05-21 11:23:41 +03:00
|
|
|
if (m_graphics_devices.is_empty()) {
|
|
|
|
dbgln("No graphics adapter was initialized.");
|
|
|
|
return false;
|
|
|
|
}
|
2021-03-05 14:23:08 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-16 22:58:51 +03:00
|
|
|
bool GraphicsManagement::framebuffer_devices_exist() const
|
|
|
|
{
|
|
|
|
for (auto& graphics_device : m_graphics_devices) {
|
|
|
|
if (graphics_device.framebuffer_devices_initialized())
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2021-03-05 14:23:08 +02:00
|
|
|
}
|