2023-08-31 13:37:17 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin <stenzek@gmail.com>
|
2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2019-09-11 04:01:19 +00:00
|
|
|
#include "gpu.h"
|
2019-09-11 04:59:41 +00:00
|
|
|
#include "dma.h"
|
2023-08-31 13:37:17 +00:00
|
|
|
#include "gpu_shadergen.h"
|
2022-07-11 13:03:29 +00:00
|
|
|
#include "host.h"
|
2022-03-26 13:09:28 +00:00
|
|
|
#include "imgui.h"
|
2019-09-17 14:22:41 +00:00
|
|
|
#include "interrupt_controller.h"
|
2020-12-12 05:37:53 +00:00
|
|
|
#include "settings.h"
|
2019-09-13 16:07:31 +00:00
|
|
|
#include "system.h"
|
2019-09-20 13:40:19 +00:00
|
|
|
#include "timers.h"
|
2023-08-27 06:00:06 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
#include "util/gpu_device.h"
|
2023-08-27 06:00:06 +00:00
|
|
|
#include "util/imgui_manager.h"
|
2023-08-27 12:48:40 +00:00
|
|
|
#include "util/postprocessing.h"
|
2023-08-27 08:13:50 +00:00
|
|
|
#include "util/shadergen.h"
|
2022-07-08 12:43:38 +00:00
|
|
|
#include "util/state_wrapper.h"
|
2023-08-27 06:00:06 +00:00
|
|
|
|
2023-08-27 08:13:50 +00:00
|
|
|
#include "common/align.h"
|
2023-08-27 06:00:06 +00:00
|
|
|
#include "common/file_system.h"
|
|
|
|
#include "common/heap_array.h"
|
|
|
|
#include "common/log.h"
|
|
|
|
#include "common/string_util.h"
|
|
|
|
|
2023-08-27 08:13:50 +00:00
|
|
|
#include "stb_image_resize.h"
|
2023-08-27 06:00:06 +00:00
|
|
|
#include "stb_image_write.h"
|
|
|
|
|
2019-10-22 13:07:51 +00:00
|
|
|
#include <cmath>
|
2023-08-27 08:13:50 +00:00
|
|
|
#include <thread>
|
2023-08-27 06:00:06 +00:00
|
|
|
|
2019-09-11 04:01:19 +00:00
|
|
|
Log_SetChannel(GPU);
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
std::unique_ptr<GPU> g_gpu;
|
|
|
|
|
2019-10-22 13:07:51 +00:00
|
|
|
const GPU::GP0CommandHandlerTable GPU::s_GP0_command_handler_table = GPU::GenerateGP0CommandHandlerTable();
|
|
|
|
|
2019-09-11 04:01:19 +00:00
|
|
|
GPU::GPU() = default;
|
|
|
|
|
2022-09-03 04:15:15 +00:00
|
|
|
GPU::~GPU()
|
|
|
|
{
|
2023-08-13 03:42:02 +00:00
|
|
|
if (g_gpu_device)
|
|
|
|
g_gpu_device->SetGPUTimingEnabled(false);
|
2022-09-03 04:15:15 +00:00
|
|
|
}
|
2019-09-11 04:01:19 +00:00
|
|
|
|
2022-07-11 13:03:29 +00:00
|
|
|
bool GPU::Initialize()
|
2019-09-11 04:01:19 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
m_force_progressive_scan = g_settings.gpu_disable_interlacing;
|
|
|
|
m_force_ntsc_timings = g_settings.gpu_force_ntsc_timings;
|
|
|
|
m_crtc_tick_event = TimingEvents::CreateTimingEvent(
|
2021-01-09 15:43:59 +00:00
|
|
|
"GPU CRTC Tick", 1, 1,
|
|
|
|
[](void* param, TickCount ticks, TickCount ticks_late) { static_cast<GPU*>(param)->CRTCTickEvent(ticks); }, this,
|
|
|
|
true);
|
2020-07-31 07:09:18 +00:00
|
|
|
m_command_tick_event = TimingEvents::CreateTimingEvent(
|
2021-01-09 15:43:59 +00:00
|
|
|
"GPU Command Tick", 1, 1,
|
|
|
|
[](void* param, TickCount ticks, TickCount ticks_late) { static_cast<GPU*>(param)->CommandTickEvent(ticks); }, this,
|
|
|
|
true);
|
2020-07-31 07:09:18 +00:00
|
|
|
m_fifo_size = g_settings.gpu_fifo_size;
|
|
|
|
m_max_run_ahead = g_settings.gpu_max_run_ahead;
|
|
|
|
m_console_is_pal = System::IsPALRegion();
|
2020-06-07 07:36:45 +00:00
|
|
|
UpdateCRTCConfig();
|
2022-07-11 13:03:29 +00:00
|
|
|
|
2023-08-31 13:37:17 +00:00
|
|
|
if (!CompileDisplayPipeline())
|
2022-07-11 13:03:29 +00:00
|
|
|
{
|
2023-08-27 08:13:50 +00:00
|
|
|
Host::ReportErrorAsync("Error", "Failed to compile base GPU pipelines.");
|
|
|
|
return false;
|
2022-07-11 13:03:29 +00:00
|
|
|
}
|
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
g_gpu_device->SetGPUTimingEnabled(g_settings.display_show_gpu);
|
2022-09-03 04:15:15 +00:00
|
|
|
|
2019-09-11 04:01:19 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-08-31 13:37:17 +00:00
|
|
|
void GPU::UpdateSettings(const Settings& old_settings)
|
2019-12-10 12:52:46 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
m_force_progressive_scan = g_settings.gpu_disable_interlacing;
|
|
|
|
m_fifo_size = g_settings.gpu_fifo_size;
|
|
|
|
m_max_run_ahead = g_settings.gpu_max_run_ahead;
|
2020-03-28 15:14:37 +00:00
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
if (m_force_ntsc_timings != g_settings.gpu_force_ntsc_timings || m_console_is_pal != System::IsPALRegion())
|
2020-04-10 03:34:12 +00:00
|
|
|
{
|
2020-07-31 07:09:18 +00:00
|
|
|
m_force_ntsc_timings = g_settings.gpu_force_ntsc_timings;
|
|
|
|
m_console_is_pal = System::IsPALRegion();
|
2020-04-10 03:34:12 +00:00
|
|
|
UpdateCRTCConfig();
|
|
|
|
}
|
|
|
|
|
2020-03-28 15:14:37 +00:00
|
|
|
// Crop mode calls this, so recalculate the display area
|
|
|
|
UpdateCRTCDisplayParameters();
|
2022-07-11 13:03:29 +00:00
|
|
|
|
2023-08-31 13:37:17 +00:00
|
|
|
if (g_settings.display_scaling != old_settings.display_scaling)
|
|
|
|
{
|
|
|
|
if (!CompileDisplayPipeline())
|
|
|
|
Panic("Failed to compile display pipeline on settings change.");
|
|
|
|
}
|
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
g_gpu_device->SetGPUTimingEnabled(g_settings.display_show_gpu);
|
2021-03-28 03:47:10 +00:00
|
|
|
}
|
|
|
|
|
2020-09-29 13:29:28 +00:00
|
|
|
void GPU::CPUClockChanged()
|
|
|
|
{
|
|
|
|
UpdateCRTCConfig();
|
|
|
|
}
|
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
void GPU::UpdateResolutionScale()
|
|
|
|
{
|
|
|
|
}
|
2020-08-02 17:06:03 +00:00
|
|
|
|
2021-08-24 01:57:09 +00:00
|
|
|
std::tuple<u32, u32> GPU::GetEffectiveDisplayResolution(bool scaled /* = true */)
|
2020-08-15 14:17:10 +00:00
|
|
|
{
|
|
|
|
return std::tie(m_crtc_state.display_vram_width, m_crtc_state.display_vram_height);
|
|
|
|
}
|
|
|
|
|
2021-08-24 01:57:09 +00:00
|
|
|
std::tuple<u32, u32> GPU::GetFullDisplayResolution(bool scaled /* = true */)
|
|
|
|
{
|
|
|
|
return std::tie(m_crtc_state.display_width, m_crtc_state.display_height);
|
|
|
|
}
|
|
|
|
|
2021-01-23 09:00:54 +00:00
|
|
|
void GPU::Reset(bool clear_vram)
|
2019-09-11 04:01:19 +00:00
|
|
|
{
|
2021-09-29 00:48:25 +00:00
|
|
|
m_GPUSTAT.bits = 0x14802000;
|
2019-12-11 06:47:49 +00:00
|
|
|
m_set_texture_disable_mask = false;
|
2019-11-14 12:17:09 +00:00
|
|
|
m_GPUREAD_latch = 0;
|
2021-09-29 00:48:25 +00:00
|
|
|
m_crtc_state.fractional_ticks = 0;
|
|
|
|
m_crtc_state.fractional_dot_ticks = 0;
|
|
|
|
m_crtc_state.current_tick_in_scanline = 0;
|
|
|
|
m_crtc_state.current_scanline = 0;
|
|
|
|
m_crtc_state.in_hblank = false;
|
|
|
|
m_crtc_state.in_vblank = false;
|
|
|
|
m_crtc_state.interlaced_field = 0;
|
|
|
|
m_crtc_state.interlaced_display_field = 0;
|
|
|
|
SoftReset();
|
2021-08-24 01:57:12 +00:00
|
|
|
UpdateDisplay();
|
2019-09-11 04:01:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::SoftReset()
|
|
|
|
{
|
2020-04-03 14:10:55 +00:00
|
|
|
FlushRender();
|
2020-11-18 10:35:02 +00:00
|
|
|
if (m_blitter_state == BlitterState::WritingVRAM)
|
|
|
|
FinishVRAMWrite();
|
2020-04-03 14:10:55 +00:00
|
|
|
|
2021-09-29 00:48:25 +00:00
|
|
|
m_GPUSTAT.texture_page_x_base = 0;
|
|
|
|
m_GPUSTAT.texture_page_y_base = 0;
|
|
|
|
m_GPUSTAT.semi_transparency_mode = GPUTransparencyMode::HalfBackgroundPlusHalfForeground;
|
|
|
|
m_GPUSTAT.texture_color_mode = GPUTextureMode::Palette4Bit;
|
|
|
|
m_GPUSTAT.dither_enable = false;
|
|
|
|
m_GPUSTAT.draw_to_displayed_field = false;
|
|
|
|
m_GPUSTAT.set_mask_while_drawing = false;
|
|
|
|
m_GPUSTAT.check_mask_before_draw = false;
|
|
|
|
m_GPUSTAT.reverse_flag = false;
|
|
|
|
m_GPUSTAT.texture_disable = false;
|
|
|
|
m_GPUSTAT.horizontal_resolution_2 = 0;
|
|
|
|
m_GPUSTAT.horizontal_resolution_1 = 0;
|
|
|
|
m_GPUSTAT.vertical_resolution = false;
|
2020-07-31 07:09:18 +00:00
|
|
|
m_GPUSTAT.pal_mode = System::IsPALRegion();
|
2021-09-29 00:48:25 +00:00
|
|
|
m_GPUSTAT.display_area_color_depth_24 = false;
|
|
|
|
m_GPUSTAT.vertical_interlace = false;
|
|
|
|
m_GPUSTAT.display_disable = true;
|
|
|
|
m_GPUSTAT.dma_direction = DMADirection::Off;
|
2019-11-05 09:19:49 +00:00
|
|
|
m_drawing_area.Set(0, 0, 0, 0);
|
|
|
|
m_drawing_area_changed = true;
|
2019-10-04 10:33:37 +00:00
|
|
|
m_drawing_offset = {};
|
2020-04-10 05:12:16 +00:00
|
|
|
std::memset(&m_crtc_state.regs, 0, sizeof(m_crtc_state.regs));
|
2019-09-17 04:25:25 +00:00
|
|
|
m_crtc_state.regs.horizontal_display_range = 0xC60260;
|
|
|
|
m_crtc_state.regs.vertical_display_range = 0x3FC10;
|
2020-04-18 15:16:58 +00:00
|
|
|
m_blitter_state = BlitterState::Idle;
|
2020-06-12 15:28:49 +00:00
|
|
|
m_pending_command_ticks = 0;
|
2019-11-14 12:17:09 +00:00
|
|
|
m_command_total_words = 0;
|
|
|
|
m_vram_transfer = {};
|
2020-04-18 15:16:58 +00:00
|
|
|
m_fifo.Clear();
|
|
|
|
m_blit_buffer.clear();
|
|
|
|
m_blit_remaining_words = 0;
|
2020-09-20 11:28:45 +00:00
|
|
|
m_draw_mode.texture_window_value = 0xFFFFFFFFu;
|
2019-12-11 06:35:14 +00:00
|
|
|
SetDrawMode(0);
|
|
|
|
SetTexturePalette(0);
|
2020-04-03 14:10:55 +00:00
|
|
|
SetTextureWindow(0);
|
2020-03-01 07:06:38 +00:00
|
|
|
UpdateDMARequest();
|
2019-09-17 04:25:25 +00:00
|
|
|
UpdateCRTCConfig();
|
2020-06-12 15:28:49 +00:00
|
|
|
UpdateCRTCTickEvent();
|
|
|
|
UpdateCommandTickEvent();
|
2021-09-29 00:48:25 +00:00
|
|
|
UpdateGPUIdle();
|
2019-09-11 04:59:41 +00:00
|
|
|
}
|
|
|
|
|
2022-10-03 06:44:34 +00:00
|
|
|
bool GPU::DoState(StateWrapper& sw, GPUTexture** host_texture, bool update_display)
|
2019-09-14 10:28:47 +00:00
|
|
|
{
|
2023-08-13 03:42:02 +00:00
|
|
|
FlushRender();
|
|
|
|
|
2019-09-14 10:28:47 +00:00
|
|
|
if (sw.IsReading())
|
2019-10-04 10:33:37 +00:00
|
|
|
{
|
|
|
|
// perform a reset to discard all pending draws/fb state
|
2021-01-23 09:00:54 +00:00
|
|
|
Reset(host_texture == nullptr);
|
2019-10-04 10:33:37 +00:00
|
|
|
}
|
2019-09-14 10:28:47 +00:00
|
|
|
|
|
|
|
sw.Do(&m_GPUSTAT.bits);
|
2019-09-25 14:15:21 +00:00
|
|
|
|
2019-12-11 10:28:14 +00:00
|
|
|
sw.Do(&m_draw_mode.mode_reg.bits);
|
|
|
|
sw.Do(&m_draw_mode.palette_reg);
|
|
|
|
sw.Do(&m_draw_mode.texture_window_value);
|
2019-12-11 06:35:14 +00:00
|
|
|
sw.Do(&m_draw_mode.texture_page_x);
|
|
|
|
sw.Do(&m_draw_mode.texture_page_y);
|
|
|
|
sw.Do(&m_draw_mode.texture_palette_x);
|
|
|
|
sw.Do(&m_draw_mode.texture_palette_y);
|
2020-11-21 03:32:58 +00:00
|
|
|
sw.Do(&m_draw_mode.texture_window.and_x);
|
|
|
|
sw.Do(&m_draw_mode.texture_window.and_y);
|
|
|
|
sw.Do(&m_draw_mode.texture_window.or_x);
|
|
|
|
sw.Do(&m_draw_mode.texture_window.or_y);
|
2019-12-11 06:35:14 +00:00
|
|
|
sw.Do(&m_draw_mode.texture_x_flip);
|
|
|
|
sw.Do(&m_draw_mode.texture_y_flip);
|
2019-09-25 14:15:21 +00:00
|
|
|
|
2019-09-26 13:33:20 +00:00
|
|
|
sw.Do(&m_drawing_area.left);
|
|
|
|
sw.Do(&m_drawing_area.top);
|
|
|
|
sw.Do(&m_drawing_area.right);
|
|
|
|
sw.Do(&m_drawing_area.bottom);
|
2019-09-14 10:28:47 +00:00
|
|
|
sw.Do(&m_drawing_offset.x);
|
|
|
|
sw.Do(&m_drawing_offset.y);
|
|
|
|
sw.Do(&m_drawing_offset.x);
|
|
|
|
|
2020-06-07 07:36:45 +00:00
|
|
|
sw.Do(&m_console_is_pal);
|
2019-12-11 06:47:49 +00:00
|
|
|
sw.Do(&m_set_texture_disable_mask);
|
|
|
|
|
2019-09-17 04:40:23 +00:00
|
|
|
sw.Do(&m_crtc_state.regs.display_address_start);
|
|
|
|
sw.Do(&m_crtc_state.regs.horizontal_display_range);
|
|
|
|
sw.Do(&m_crtc_state.regs.vertical_display_range);
|
|
|
|
sw.Do(&m_crtc_state.dot_clock_divider);
|
2020-03-28 15:14:37 +00:00
|
|
|
sw.Do(&m_crtc_state.display_width);
|
|
|
|
sw.Do(&m_crtc_state.display_height);
|
|
|
|
sw.Do(&m_crtc_state.display_origin_left);
|
|
|
|
sw.Do(&m_crtc_state.display_origin_top);
|
|
|
|
sw.Do(&m_crtc_state.display_vram_left);
|
|
|
|
sw.Do(&m_crtc_state.display_vram_top);
|
|
|
|
sw.Do(&m_crtc_state.display_vram_width);
|
|
|
|
sw.Do(&m_crtc_state.display_vram_height);
|
2019-11-11 13:36:11 +00:00
|
|
|
sw.Do(&m_crtc_state.horizontal_total);
|
2020-12-12 07:59:09 +00:00
|
|
|
sw.Do(&m_crtc_state.horizontal_visible_start);
|
|
|
|
sw.Do(&m_crtc_state.horizontal_visible_end);
|
2019-11-11 13:36:11 +00:00
|
|
|
sw.Do(&m_crtc_state.horizontal_display_start);
|
|
|
|
sw.Do(&m_crtc_state.horizontal_display_end);
|
|
|
|
sw.Do(&m_crtc_state.vertical_total);
|
2020-12-12 07:59:09 +00:00
|
|
|
sw.Do(&m_crtc_state.vertical_visible_start);
|
|
|
|
sw.Do(&m_crtc_state.vertical_visible_end);
|
2019-11-11 13:36:11 +00:00
|
|
|
sw.Do(&m_crtc_state.vertical_display_start);
|
|
|
|
sw.Do(&m_crtc_state.vertical_display_end);
|
2019-09-17 04:40:23 +00:00
|
|
|
sw.Do(&m_crtc_state.fractional_ticks);
|
|
|
|
sw.Do(&m_crtc_state.current_tick_in_scanline);
|
|
|
|
sw.Do(&m_crtc_state.current_scanline);
|
2020-12-05 05:59:47 +00:00
|
|
|
sw.DoEx(&m_crtc_state.fractional_dot_ticks, 46, 0);
|
2019-09-17 04:40:23 +00:00
|
|
|
sw.Do(&m_crtc_state.in_hblank);
|
|
|
|
sw.Do(&m_crtc_state.in_vblank);
|
2020-05-26 12:57:56 +00:00
|
|
|
sw.Do(&m_crtc_state.interlaced_field);
|
2020-08-15 04:56:20 +00:00
|
|
|
sw.Do(&m_crtc_state.interlaced_display_field);
|
2020-05-26 12:57:56 +00:00
|
|
|
sw.Do(&m_crtc_state.active_line_lsb);
|
2019-09-17 04:40:23 +00:00
|
|
|
|
2020-04-18 15:16:58 +00:00
|
|
|
sw.Do(&m_blitter_state);
|
2020-06-12 15:28:49 +00:00
|
|
|
sw.Do(&m_pending_command_ticks);
|
2020-03-01 07:06:38 +00:00
|
|
|
sw.Do(&m_command_total_words);
|
2019-11-14 12:17:09 +00:00
|
|
|
sw.Do(&m_GPUREAD_latch);
|
|
|
|
|
|
|
|
sw.Do(&m_vram_transfer.x);
|
|
|
|
sw.Do(&m_vram_transfer.y);
|
|
|
|
sw.Do(&m_vram_transfer.width);
|
|
|
|
sw.Do(&m_vram_transfer.height);
|
|
|
|
sw.Do(&m_vram_transfer.col);
|
|
|
|
sw.Do(&m_vram_transfer.row);
|
2019-09-17 04:40:23 +00:00
|
|
|
|
2020-04-18 15:16:58 +00:00
|
|
|
sw.Do(&m_fifo);
|
|
|
|
sw.Do(&m_blit_buffer);
|
|
|
|
sw.Do(&m_blit_remaining_words);
|
|
|
|
sw.Do(&m_render_command.bits);
|
|
|
|
|
|
|
|
sw.Do(&m_max_run_ahead);
|
|
|
|
sw.Do(&m_fifo_size);
|
2019-09-14 10:28:47 +00:00
|
|
|
|
|
|
|
if (sw.IsReading())
|
|
|
|
{
|
2019-12-11 06:35:14 +00:00
|
|
|
m_draw_mode.texture_page_changed = true;
|
|
|
|
m_draw_mode.texture_window_changed = true;
|
2019-11-05 09:19:49 +00:00
|
|
|
m_drawing_area_changed = true;
|
2020-03-01 07:06:38 +00:00
|
|
|
UpdateDMARequest();
|
2019-09-14 10:28:47 +00:00
|
|
|
}
|
|
|
|
|
2021-01-23 09:00:54 +00:00
|
|
|
if (!host_texture)
|
|
|
|
{
|
|
|
|
if (!sw.DoMarker("GPU-VRAM"))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (sw.IsReading())
|
|
|
|
{
|
|
|
|
// Still need a temporary here.
|
2023-08-19 03:52:51 +00:00
|
|
|
FixedHeapArray<u16, VRAM_WIDTH * VRAM_HEIGHT> temp;
|
2021-01-23 09:00:54 +00:00
|
|
|
sw.DoBytes(temp.data(), VRAM_WIDTH * VRAM_HEIGHT * sizeof(u16));
|
|
|
|
UpdateVRAM(0, 0, VRAM_WIDTH, VRAM_HEIGHT, temp.data(), false, false);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ReadVRAM(0, 0, VRAM_WIDTH, VRAM_HEIGHT);
|
|
|
|
sw.DoBytes(m_vram_ptr, VRAM_WIDTH * VRAM_HEIGHT * sizeof(u16));
|
|
|
|
}
|
|
|
|
}
|
2019-09-14 10:45:26 +00:00
|
|
|
|
|
|
|
if (sw.IsReading())
|
|
|
|
{
|
2020-02-28 07:01:01 +00:00
|
|
|
UpdateCRTCConfig();
|
2020-11-25 13:06:39 +00:00
|
|
|
if (update_display)
|
|
|
|
UpdateDisplay();
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
UpdateCRTCTickEvent();
|
|
|
|
UpdateCommandTickEvent();
|
2019-09-14 10:45:26 +00:00
|
|
|
}
|
|
|
|
|
2019-09-14 10:28:47 +00:00
|
|
|
return !sw.HasError();
|
|
|
|
}
|
|
|
|
|
2023-09-03 03:13:17 +00:00
|
|
|
void GPU::RestoreDeviceContext()
|
2023-08-13 03:42:02 +00:00
|
|
|
{
|
|
|
|
}
|
2019-10-04 12:10:43 +00:00
|
|
|
|
2020-03-01 07:06:38 +00:00
|
|
|
void GPU::UpdateDMARequest()
|
2019-09-11 04:59:41 +00:00
|
|
|
{
|
2020-04-18 15:16:58 +00:00
|
|
|
switch (m_blitter_state)
|
|
|
|
{
|
|
|
|
case BlitterState::Idle:
|
|
|
|
m_GPUSTAT.ready_to_send_vram = false;
|
2020-11-15 11:54:07 +00:00
|
|
|
m_GPUSTAT.ready_to_recieve_dma = (m_fifo.IsEmpty() || m_fifo.GetSize() < m_command_total_words);
|
2020-04-18 15:16:58 +00:00
|
|
|
break;
|
2020-03-01 07:06:38 +00:00
|
|
|
|
2020-04-18 15:16:58 +00:00
|
|
|
case BlitterState::WritingVRAM:
|
|
|
|
m_GPUSTAT.ready_to_send_vram = false;
|
|
|
|
m_GPUSTAT.ready_to_recieve_dma = (m_fifo.GetSize() < m_fifo_size);
|
|
|
|
break;
|
2020-03-01 07:06:38 +00:00
|
|
|
|
2020-04-18 15:16:58 +00:00
|
|
|
case BlitterState::ReadingVRAM:
|
|
|
|
m_GPUSTAT.ready_to_send_vram = true;
|
2023-07-24 09:23:58 +00:00
|
|
|
m_GPUSTAT.ready_to_recieve_dma = m_fifo.IsEmpty();
|
2020-04-18 15:16:58 +00:00
|
|
|
break;
|
2023-09-03 04:30:26 +00:00
|
|
|
|
|
|
|
case BlitterState::DrawingPolyLine:
|
|
|
|
m_GPUSTAT.ready_to_send_vram = false;
|
|
|
|
m_GPUSTAT.ready_to_recieve_dma = (m_fifo.GetSize() < m_fifo_size);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
UnreachableCode();
|
|
|
|
break;
|
2020-04-18 15:16:58 +00:00
|
|
|
}
|
2019-09-13 16:07:31 +00:00
|
|
|
|
|
|
|
bool dma_request;
|
|
|
|
switch (m_GPUSTAT.dma_direction)
|
|
|
|
{
|
|
|
|
case DMADirection::Off:
|
|
|
|
dma_request = false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DMADirection::FIFO:
|
2020-04-18 15:16:58 +00:00
|
|
|
dma_request = m_GPUSTAT.ready_to_recieve_dma;
|
2019-09-13 16:07:31 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DMADirection::CPUtoGP0:
|
2020-04-18 15:16:58 +00:00
|
|
|
dma_request = m_GPUSTAT.ready_to_recieve_dma;
|
2019-09-13 16:07:31 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DMADirection::GPUREADtoCPU:
|
2020-04-18 15:16:58 +00:00
|
|
|
dma_request = m_GPUSTAT.ready_to_send_vram;
|
2019-09-13 16:07:31 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
dma_request = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
m_GPUSTAT.dma_data_request = dma_request;
|
2023-01-11 08:51:38 +00:00
|
|
|
DMA::SetRequest(DMA::Channel::GPU, dma_request);
|
2019-09-11 04:01:19 +00:00
|
|
|
}
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
void GPU::UpdateGPUIdle()
|
|
|
|
{
|
|
|
|
switch (m_blitter_state)
|
|
|
|
{
|
|
|
|
case BlitterState::Idle:
|
2020-09-24 11:45:14 +00:00
|
|
|
m_GPUSTAT.gpu_idle = (m_pending_command_ticks <= 0 && m_fifo.IsEmpty());
|
2020-06-12 15:28:49 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BlitterState::WritingVRAM:
|
|
|
|
m_GPUSTAT.gpu_idle = false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case BlitterState::ReadingVRAM:
|
|
|
|
m_GPUSTAT.gpu_idle = false;
|
|
|
|
break;
|
2023-09-03 04:30:26 +00:00
|
|
|
|
|
|
|
case BlitterState::DrawingPolyLine:
|
|
|
|
m_GPUSTAT.gpu_idle = false;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
UnreachableCode();
|
|
|
|
break;
|
2020-06-12 15:28:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-11 04:01:19 +00:00
|
|
|
u32 GPU::ReadRegister(u32 offset)
|
|
|
|
{
|
2019-09-11 04:59:41 +00:00
|
|
|
switch (offset)
|
2019-09-11 04:01:19 +00:00
|
|
|
{
|
2019-09-11 04:59:41 +00:00
|
|
|
case 0x00:
|
|
|
|
return ReadGPUREAD();
|
|
|
|
|
|
|
|
case 0x04:
|
2020-03-01 07:07:17 +00:00
|
|
|
{
|
|
|
|
// code can be dependent on the odd/even bit, so update the GPU state when reading.
|
|
|
|
// we can mitigate this slightly by only updating when the raster is actually hitting a new line
|
2020-06-12 15:28:49 +00:00
|
|
|
if (IsCRTCScanlinePending())
|
|
|
|
SynchronizeCRTC();
|
|
|
|
if (IsCommandCompletionPending())
|
|
|
|
m_command_tick_event->InvokeEarly();
|
2020-03-01 07:07:17 +00:00
|
|
|
|
2019-12-07 14:40:52 +00:00
|
|
|
return m_GPUSTAT.bits;
|
2020-03-01 07:07:17 +00:00
|
|
|
}
|
2019-09-11 04:59:41 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
Log_ErrorPrintf("Unhandled register read: %02X", offset);
|
|
|
|
return UINT32_C(0xFFFFFFFF);
|
2019-09-11 04:01:19 +00:00
|
|
|
}
|
2019-09-11 04:59:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::WriteRegister(u32 offset, u32 value)
|
|
|
|
{
|
|
|
|
switch (offset)
|
2019-09-11 04:01:19 +00:00
|
|
|
{
|
2019-09-11 04:59:41 +00:00
|
|
|
case 0x00:
|
2020-04-18 15:16:58 +00:00
|
|
|
m_fifo.Push(value);
|
|
|
|
ExecuteCommands();
|
2020-06-12 15:28:49 +00:00
|
|
|
UpdateCommandTickEvent();
|
2019-09-11 04:59:41 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
case 0x04:
|
|
|
|
WriteGP1(value);
|
|
|
|
return;
|
|
|
|
|
|
|
|
default:
|
|
|
|
Log_ErrorPrintf("Unhandled register write: %02X <- %08X", offset, value);
|
|
|
|
return;
|
2019-09-11 04:01:19 +00:00
|
|
|
}
|
2019-09-11 04:59:41 +00:00
|
|
|
}
|
2019-09-11 04:01:19 +00:00
|
|
|
|
2019-10-13 06:48:11 +00:00
|
|
|
void GPU::DMARead(u32* words, u32 word_count)
|
2019-09-11 04:59:41 +00:00
|
|
|
{
|
|
|
|
if (m_GPUSTAT.dma_direction != DMADirection::GPUREADtoCPU)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Invalid DMA direction from GPU DMA read");
|
2019-10-13 06:48:11 +00:00
|
|
|
std::fill_n(words, word_count, UINT32_C(0xFFFFFFFF));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-14 12:17:09 +00:00
|
|
|
for (u32 i = 0; i < word_count; i++)
|
|
|
|
words[i] = ReadGPUREAD();
|
2019-09-11 04:01:19 +00:00
|
|
|
}
|
|
|
|
|
2020-08-01 14:25:07 +00:00
|
|
|
void GPU::EndDMAWrite()
|
2019-09-11 04:59:41 +00:00
|
|
|
{
|
2020-08-01 14:25:07 +00:00
|
|
|
m_fifo_pushed = true;
|
|
|
|
if (!m_syncing)
|
2019-09-11 06:04:31 +00:00
|
|
|
{
|
2020-08-01 14:25:07 +00:00
|
|
|
ExecuteCommands();
|
|
|
|
UpdateCommandTickEvent();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
UpdateDMARequest();
|
2019-09-11 06:04:31 +00:00
|
|
|
}
|
2019-09-11 04:59:41 +00:00
|
|
|
}
|
|
|
|
|
2020-06-07 07:36:45 +00:00
|
|
|
/**
|
|
|
|
* NTSC GPU clock 53.693175 MHz
|
|
|
|
* PAL GPU clock 53.203425 MHz
|
|
|
|
* courtesy of @ggrtk
|
|
|
|
*
|
|
|
|
* NTSC - sysclk * 715909 / 451584
|
|
|
|
* PAL - sysclk * 709379 / 451584
|
|
|
|
*/
|
|
|
|
|
2020-07-01 16:51:22 +00:00
|
|
|
TickCount GPU::GetCRTCFrequency() const
|
|
|
|
{
|
|
|
|
return m_console_is_pal ? 53203425 : 53693175;
|
|
|
|
}
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
TickCount GPU::CRTCTicksToSystemTicks(TickCount gpu_ticks, TickCount fractional_ticks) const
|
2020-06-07 07:36:45 +00:00
|
|
|
{
|
|
|
|
// convert to master clock, rounding up as we want to overshoot not undershoot
|
|
|
|
if (!m_console_is_pal)
|
|
|
|
return static_cast<TickCount>((u64(gpu_ticks) * u64(451584) + fractional_ticks + u64(715908)) / u64(715909));
|
|
|
|
else
|
|
|
|
return static_cast<TickCount>((u64(gpu_ticks) * u64(451584) + fractional_ticks + u64(709378)) / u64(709379));
|
|
|
|
}
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
TickCount GPU::SystemTicksToCRTCTicks(TickCount sysclk_ticks, TickCount* fractional_ticks) const
|
2020-06-07 07:36:45 +00:00
|
|
|
{
|
2020-12-16 03:23:07 +00:00
|
|
|
u64 mul = u64(sysclk_ticks);
|
|
|
|
mul *= !m_console_is_pal ? u64(715909) : u64(709379);
|
|
|
|
mul += u64(*fractional_ticks);
|
|
|
|
|
|
|
|
const TickCount ticks = static_cast<TickCount>(mul / u64(451584));
|
|
|
|
*fractional_ticks = static_cast<TickCount>(mul % u64(451584));
|
|
|
|
return ticks;
|
2020-06-07 07:36:45 +00:00
|
|
|
}
|
|
|
|
|
2020-04-18 15:16:58 +00:00
|
|
|
void GPU::AddCommandTicks(TickCount ticks)
|
|
|
|
{
|
2020-06-12 15:28:49 +00:00
|
|
|
m_pending_command_ticks += ticks;
|
2020-04-18 15:16:58 +00:00
|
|
|
}
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
void GPU::SynchronizeCRTC()
|
2020-01-24 04:53:40 +00:00
|
|
|
{
|
2020-06-12 15:28:49 +00:00
|
|
|
m_crtc_tick_event->InvokeEarly();
|
2020-01-24 04:53:40 +00:00
|
|
|
}
|
|
|
|
|
2020-06-07 07:36:45 +00:00
|
|
|
float GPU::ComputeHorizontalFrequency() const
|
|
|
|
{
|
|
|
|
const CRTCState& cs = m_crtc_state;
|
|
|
|
TickCount fractional_ticks = 0;
|
2020-09-29 13:29:28 +00:00
|
|
|
return static_cast<float>(
|
|
|
|
static_cast<double>(SystemTicksToCRTCTicks(System::GetTicksPerSecond(), &fractional_ticks)) /
|
|
|
|
static_cast<double>(cs.horizontal_total));
|
2020-06-07 07:36:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
float GPU::ComputeVerticalFrequency() const
|
|
|
|
{
|
|
|
|
const CRTCState& cs = m_crtc_state;
|
|
|
|
const TickCount ticks_per_frame = cs.horizontal_total * cs.vertical_total;
|
|
|
|
TickCount fractional_ticks = 0;
|
2020-09-29 13:29:28 +00:00
|
|
|
return static_cast<float>(
|
|
|
|
static_cast<double>(SystemTicksToCRTCTicks(System::GetTicksPerSecond(), &fractional_ticks)) /
|
|
|
|
static_cast<double>(ticks_per_frame));
|
2020-06-07 07:36:45 +00:00
|
|
|
}
|
|
|
|
|
2023-08-27 08:13:50 +00:00
|
|
|
float GPU::ComputeDisplayAspectRatio() const
|
2020-09-26 05:11:45 +00:00
|
|
|
{
|
|
|
|
if (g_settings.display_force_4_3_for_24bit && m_GPUSTAT.display_area_color_depth_24)
|
2020-12-12 05:37:53 +00:00
|
|
|
{
|
2020-09-26 05:11:45 +00:00
|
|
|
return 4.0f / 3.0f;
|
2020-12-12 05:37:53 +00:00
|
|
|
}
|
|
|
|
else if (g_settings.display_aspect_ratio == DisplayAspectRatio::Auto)
|
|
|
|
{
|
|
|
|
const CRTCState& cs = m_crtc_state;
|
|
|
|
float relative_width = static_cast<float>(cs.horizontal_visible_end - cs.horizontal_visible_start);
|
|
|
|
float relative_height = static_cast<float>(cs.vertical_visible_end - cs.vertical_visible_start);
|
|
|
|
|
|
|
|
if (relative_width <= 0 || relative_height <= 0)
|
|
|
|
return 4.0f / 3.0f;
|
|
|
|
|
|
|
|
if (m_GPUSTAT.pal_mode)
|
|
|
|
{
|
|
|
|
relative_width /= static_cast<float>(PAL_HORIZONTAL_ACTIVE_END - PAL_HORIZONTAL_ACTIVE_START);
|
|
|
|
relative_height /= static_cast<float>(PAL_VERTICAL_ACTIVE_END - PAL_VERTICAL_ACTIVE_START);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
relative_width /= static_cast<float>(NTSC_HORIZONTAL_ACTIVE_END - NTSC_HORIZONTAL_ACTIVE_START);
|
|
|
|
relative_height /= static_cast<float>(NTSC_VERTICAL_ACTIVE_END - NTSC_VERTICAL_ACTIVE_START);
|
|
|
|
}
|
|
|
|
return (relative_width / relative_height) * (4.0f / 3.0f);
|
|
|
|
}
|
|
|
|
else if (g_settings.display_aspect_ratio == DisplayAspectRatio::PAR1_1)
|
|
|
|
{
|
|
|
|
if (m_crtc_state.display_width == 0 || m_crtc_state.display_height == 0)
|
|
|
|
return 4.0f / 3.0f;
|
|
|
|
|
|
|
|
return static_cast<float>(m_crtc_state.display_width) / static_cast<float>(m_crtc_state.display_height);
|
|
|
|
}
|
2020-09-26 05:11:45 +00:00
|
|
|
else
|
2020-12-12 05:37:53 +00:00
|
|
|
{
|
2021-04-28 16:42:08 +00:00
|
|
|
return g_settings.GetDisplayAspectRatioValue();
|
2020-12-12 05:37:53 +00:00
|
|
|
}
|
2020-09-26 05:11:45 +00:00
|
|
|
}
|
|
|
|
|
2019-09-17 04:25:25 +00:00
|
|
|
void GPU::UpdateCRTCConfig()
|
|
|
|
{
|
2020-02-28 07:01:01 +00:00
|
|
|
static constexpr std::array<u16, 8> dot_clock_dividers = {{10, 8, 5, 4, 7, 7, 7, 7}};
|
2019-09-17 04:25:25 +00:00
|
|
|
CRTCState& cs = m_crtc_state;
|
|
|
|
|
|
|
|
if (m_GPUSTAT.pal_mode)
|
|
|
|
{
|
2020-03-19 15:12:41 +00:00
|
|
|
cs.vertical_total = PAL_TOTAL_LINES;
|
|
|
|
cs.current_scanline %= PAL_TOTAL_LINES;
|
|
|
|
cs.horizontal_total = PAL_TICKS_PER_LINE;
|
2020-08-16 15:21:52 +00:00
|
|
|
cs.horizontal_sync_start = PAL_HSYNC_TICKS;
|
2020-09-29 13:29:28 +00:00
|
|
|
cs.current_tick_in_scanline %= System::ScaleTicksToOverclock(PAL_TICKS_PER_LINE);
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-03-19 15:12:41 +00:00
|
|
|
cs.vertical_total = NTSC_TOTAL_LINES;
|
|
|
|
cs.current_scanline %= NTSC_TOTAL_LINES;
|
|
|
|
cs.horizontal_total = NTSC_TICKS_PER_LINE;
|
2020-08-16 15:21:52 +00:00
|
|
|
cs.horizontal_sync_start = NTSC_HSYNC_TICKS;
|
2020-09-29 13:29:28 +00:00
|
|
|
cs.current_tick_in_scanline %= System::ScaleTicksToOverclock(NTSC_TICKS_PER_LINE);
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
|
2020-08-16 15:21:52 +00:00
|
|
|
cs.in_hblank = (cs.current_tick_in_scanline >= cs.horizontal_sync_start);
|
|
|
|
|
2019-10-20 13:19:26 +00:00
|
|
|
const u8 horizontal_resolution_index = m_GPUSTAT.horizontal_resolution_1 | (m_GPUSTAT.horizontal_resolution_2 << 2);
|
|
|
|
cs.dot_clock_divider = dot_clock_dividers[horizontal_resolution_index];
|
2020-09-27 14:33:51 +00:00
|
|
|
cs.horizontal_display_start =
|
|
|
|
(std::min<u16>(cs.regs.X1, cs.horizontal_total) / cs.dot_clock_divider) * cs.dot_clock_divider;
|
|
|
|
cs.horizontal_display_end =
|
|
|
|
(std::min<u16>(cs.regs.X2, cs.horizontal_total) / cs.dot_clock_divider) * cs.dot_clock_divider;
|
2020-03-28 15:14:37 +00:00
|
|
|
cs.vertical_display_start = std::min<u16>(cs.regs.Y1, cs.vertical_total);
|
|
|
|
cs.vertical_display_end = std::min<u16>(cs.regs.Y2, cs.vertical_total);
|
2020-02-28 07:01:01 +00:00
|
|
|
|
2020-04-10 03:34:12 +00:00
|
|
|
if (m_GPUSTAT.pal_mode && m_force_ntsc_timings)
|
|
|
|
{
|
|
|
|
// scale to NTSC parameters
|
|
|
|
cs.horizontal_display_start =
|
|
|
|
static_cast<u16>((static_cast<u32>(cs.horizontal_display_start) * NTSC_TICKS_PER_LINE) / PAL_TICKS_PER_LINE);
|
|
|
|
cs.horizontal_display_end = static_cast<u16>(
|
|
|
|
((static_cast<u32>(cs.horizontal_display_end) * NTSC_TICKS_PER_LINE) + (PAL_TICKS_PER_LINE - 1)) /
|
|
|
|
PAL_TICKS_PER_LINE);
|
|
|
|
cs.vertical_display_start =
|
|
|
|
static_cast<u16>((static_cast<u32>(cs.vertical_display_start) * NTSC_TOTAL_LINES) / PAL_TOTAL_LINES);
|
|
|
|
cs.vertical_display_end = static_cast<u16>(
|
|
|
|
((static_cast<u32>(cs.vertical_display_end) * NTSC_TOTAL_LINES) + (PAL_TOTAL_LINES - 1)) / PAL_TOTAL_LINES);
|
|
|
|
|
|
|
|
cs.vertical_total = NTSC_TOTAL_LINES;
|
|
|
|
cs.current_scanline %= NTSC_TOTAL_LINES;
|
|
|
|
cs.horizontal_total = NTSC_TICKS_PER_LINE;
|
|
|
|
cs.current_tick_in_scanline %= NTSC_TICKS_PER_LINE;
|
|
|
|
}
|
|
|
|
|
2020-09-29 13:29:28 +00:00
|
|
|
cs.horizontal_display_start =
|
|
|
|
static_cast<u16>(System::ScaleTicksToOverclock(static_cast<TickCount>(cs.horizontal_display_start)));
|
|
|
|
cs.horizontal_display_end =
|
|
|
|
static_cast<u16>(System::ScaleTicksToOverclock(static_cast<TickCount>(cs.horizontal_display_end)));
|
|
|
|
cs.horizontal_total = static_cast<u16>(System::ScaleTicksToOverclock(static_cast<TickCount>(cs.horizontal_total)));
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
System::SetThrottleFrequency(ComputeVerticalFrequency());
|
2020-04-10 03:34:12 +00:00
|
|
|
|
2020-03-28 15:14:37 +00:00
|
|
|
UpdateCRTCDisplayParameters();
|
2020-06-12 15:28:49 +00:00
|
|
|
UpdateCRTCTickEvent();
|
2020-03-28 15:14:37 +00:00
|
|
|
}
|
2020-02-28 07:01:01 +00:00
|
|
|
|
2020-03-28 15:14:37 +00:00
|
|
|
void GPU::UpdateCRTCDisplayParameters()
|
|
|
|
{
|
|
|
|
CRTCState& cs = m_crtc_state;
|
2020-07-31 07:09:18 +00:00
|
|
|
const DisplayCropMode crop_mode = g_settings.display_crop_mode;
|
2020-03-28 15:14:37 +00:00
|
|
|
|
2020-04-10 03:34:12 +00:00
|
|
|
const u16 horizontal_total = m_GPUSTAT.pal_mode ? PAL_TICKS_PER_LINE : NTSC_TICKS_PER_LINE;
|
|
|
|
const u16 vertical_total = m_GPUSTAT.pal_mode ? PAL_TOTAL_LINES : NTSC_TOTAL_LINES;
|
2020-09-27 14:33:51 +00:00
|
|
|
const u16 horizontal_display_start =
|
|
|
|
(std::min<u16>(cs.regs.X1, horizontal_total) / cs.dot_clock_divider) * cs.dot_clock_divider;
|
|
|
|
const u16 horizontal_display_end =
|
|
|
|
(std::min<u16>(cs.regs.X2, horizontal_total) / cs.dot_clock_divider) * cs.dot_clock_divider;
|
2020-04-10 03:34:12 +00:00
|
|
|
const u16 vertical_display_start = std::min<u16>(cs.regs.Y1, vertical_total);
|
|
|
|
const u16 vertical_display_end = std::min<u16>(cs.regs.Y2, vertical_total);
|
|
|
|
|
2020-03-28 15:14:37 +00:00
|
|
|
if (m_GPUSTAT.pal_mode)
|
2019-10-20 14:18:52 +00:00
|
|
|
{
|
2020-03-28 15:14:37 +00:00
|
|
|
// TODO: Verify PAL numbers.
|
|
|
|
switch (crop_mode)
|
2020-02-28 07:01:01 +00:00
|
|
|
{
|
2020-03-28 15:14:37 +00:00
|
|
|
case DisplayCropMode::None:
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.horizontal_visible_start = PAL_HORIZONTAL_ACTIVE_START;
|
|
|
|
cs.horizontal_visible_end = PAL_HORIZONTAL_ACTIVE_END;
|
|
|
|
cs.vertical_visible_start = PAL_VERTICAL_ACTIVE_START;
|
|
|
|
cs.vertical_visible_end = PAL_VERTICAL_ACTIVE_END;
|
2020-03-28 15:14:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayCropMode::Overscan:
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.horizontal_visible_start = static_cast<u16>(std::max<int>(0, 628 + g_settings.display_active_start_offset));
|
|
|
|
cs.horizontal_visible_end =
|
|
|
|
static_cast<u16>(std::max<int>(cs.horizontal_visible_start, 3188 + g_settings.display_active_end_offset));
|
|
|
|
cs.vertical_visible_start = static_cast<u16>(std::max<int>(0, 30 + g_settings.display_line_start_offset));
|
|
|
|
cs.vertical_visible_end =
|
|
|
|
static_cast<u16>(std::max<int>(cs.vertical_visible_start, 298 + g_settings.display_line_end_offset));
|
2020-03-28 15:14:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayCropMode::Borders:
|
|
|
|
default:
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.horizontal_visible_start = horizontal_display_start;
|
|
|
|
cs.horizontal_visible_end = horizontal_display_end;
|
|
|
|
cs.vertical_visible_start = vertical_display_start;
|
|
|
|
cs.vertical_visible_end = vertical_display_end;
|
2020-03-28 15:14:37 +00:00
|
|
|
break;
|
2020-02-28 07:01:01 +00:00
|
|
|
}
|
2020-12-12 08:02:24 +00:00
|
|
|
cs.horizontal_visible_start =
|
|
|
|
std::clamp<u16>(cs.horizontal_visible_start, PAL_HORIZONTAL_ACTIVE_START, PAL_HORIZONTAL_ACTIVE_END);
|
|
|
|
cs.horizontal_visible_end =
|
|
|
|
std::clamp<u16>(cs.horizontal_visible_end, cs.horizontal_visible_start, PAL_HORIZONTAL_ACTIVE_END);
|
|
|
|
cs.vertical_visible_start =
|
|
|
|
std::clamp<u16>(cs.vertical_visible_start, PAL_VERTICAL_ACTIVE_START, PAL_VERTICAL_ACTIVE_END);
|
|
|
|
cs.vertical_visible_end =
|
|
|
|
std::clamp<u16>(cs.vertical_visible_end, cs.vertical_visible_start, PAL_VERTICAL_ACTIVE_END);
|
2020-03-28 15:14:37 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (crop_mode)
|
2020-02-28 07:01:01 +00:00
|
|
|
{
|
2020-03-28 15:14:37 +00:00
|
|
|
case DisplayCropMode::None:
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.horizontal_visible_start = NTSC_HORIZONTAL_ACTIVE_START;
|
|
|
|
cs.horizontal_visible_end = NTSC_HORIZONTAL_ACTIVE_END;
|
|
|
|
cs.vertical_visible_start = NTSC_VERTICAL_ACTIVE_START;
|
|
|
|
cs.vertical_visible_end = NTSC_VERTICAL_ACTIVE_END;
|
2020-03-28 15:14:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayCropMode::Overscan:
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.horizontal_visible_start = static_cast<u16>(std::max<int>(0, 608 + g_settings.display_active_start_offset));
|
|
|
|
cs.horizontal_visible_end =
|
|
|
|
static_cast<u16>(std::max<int>(cs.horizontal_visible_start, 3168 + g_settings.display_active_end_offset));
|
|
|
|
cs.vertical_visible_start = static_cast<u16>(std::max<int>(0, 24 + g_settings.display_line_start_offset));
|
|
|
|
cs.vertical_visible_end =
|
|
|
|
static_cast<u16>(std::max<int>(cs.vertical_visible_start, 248 + g_settings.display_line_end_offset));
|
2020-03-28 15:14:37 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayCropMode::Borders:
|
|
|
|
default:
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.horizontal_visible_start = horizontal_display_start;
|
|
|
|
cs.horizontal_visible_end = horizontal_display_end;
|
|
|
|
cs.vertical_visible_start = vertical_display_start;
|
|
|
|
cs.vertical_visible_end = vertical_display_end;
|
2020-03-28 15:14:37 +00:00
|
|
|
break;
|
2020-02-28 07:01:01 +00:00
|
|
|
}
|
2020-12-12 08:02:24 +00:00
|
|
|
cs.horizontal_visible_start =
|
|
|
|
std::clamp<u16>(cs.horizontal_visible_start, NTSC_HORIZONTAL_ACTIVE_START, NTSC_HORIZONTAL_ACTIVE_END);
|
|
|
|
cs.horizontal_visible_end =
|
|
|
|
std::clamp<u16>(cs.horizontal_visible_end, cs.horizontal_visible_start, NTSC_HORIZONTAL_ACTIVE_END);
|
|
|
|
cs.vertical_visible_start =
|
|
|
|
std::clamp<u16>(cs.vertical_visible_start, NTSC_VERTICAL_ACTIVE_START, NTSC_VERTICAL_ACTIVE_END);
|
|
|
|
cs.vertical_visible_end =
|
|
|
|
std::clamp<u16>(cs.vertical_visible_end, cs.vertical_visible_start, NTSC_VERTICAL_ACTIVE_END);
|
2020-03-28 15:14:37 +00:00
|
|
|
}
|
|
|
|
|
2020-06-02 10:34:33 +00:00
|
|
|
// If force-progressive is enabled, we only double the height in 480i mode. This way non-interleaved 480i framebuffers
|
|
|
|
// won't be broken when displayed.
|
|
|
|
const u8 y_shift = BoolToUInt8(m_GPUSTAT.vertical_interlace && m_GPUSTAT.vertical_resolution);
|
|
|
|
const u8 height_shift = m_force_progressive_scan ? y_shift : BoolToUInt8(m_GPUSTAT.vertical_interlace);
|
2020-02-28 07:01:01 +00:00
|
|
|
|
2020-03-28 15:14:37 +00:00
|
|
|
// Determine screen size.
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.display_width = (cs.horizontal_visible_end - cs.horizontal_visible_start) / cs.dot_clock_divider;
|
|
|
|
cs.display_height = (cs.vertical_visible_end - cs.vertical_visible_start) << height_shift;
|
2020-03-28 15:14:37 +00:00
|
|
|
|
2020-09-27 14:35:55 +00:00
|
|
|
// Determine number of pixels outputted from VRAM (in general, round to 4-pixel multiple).
|
|
|
|
// TODO: Verify behavior if values are outside of the active video portion of scanline.
|
|
|
|
const u16 horizontal_display_ticks =
|
|
|
|
(horizontal_display_end < horizontal_display_start) ? 0 : (horizontal_display_end - horizontal_display_start);
|
|
|
|
|
|
|
|
const u16 horizontal_display_pixels = horizontal_display_ticks / cs.dot_clock_divider;
|
|
|
|
if (horizontal_display_pixels == 1u)
|
|
|
|
cs.display_vram_width = 4u;
|
|
|
|
else
|
|
|
|
cs.display_vram_width = (horizontal_display_pixels + 2u) & ~3u;
|
2020-07-11 06:58:56 +00:00
|
|
|
|
2020-03-28 15:14:37 +00:00
|
|
|
// Determine if we need to adjust the VRAM rectangle (because the display is starting outside the visible area) or add
|
|
|
|
// padding.
|
2020-05-17 09:23:46 +00:00
|
|
|
u16 horizontal_skip_pixels;
|
2020-12-12 07:59:09 +00:00
|
|
|
if (horizontal_display_start >= cs.horizontal_visible_start)
|
2020-03-28 15:14:37 +00:00
|
|
|
{
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.display_origin_left = (horizontal_display_start - cs.horizontal_visible_start) / cs.dot_clock_divider;
|
2020-12-15 01:30:02 +00:00
|
|
|
cs.display_vram_left = cs.regs.X;
|
2020-05-17 09:23:46 +00:00
|
|
|
horizontal_skip_pixels = 0;
|
2020-03-28 15:14:37 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-12-12 07:59:09 +00:00
|
|
|
horizontal_skip_pixels = (cs.horizontal_visible_start - horizontal_display_start) / cs.dot_clock_divider;
|
2020-03-28 15:14:37 +00:00
|
|
|
cs.display_origin_left = 0;
|
2020-12-15 01:30:02 +00:00
|
|
|
cs.display_vram_left = (cs.regs.X + horizontal_skip_pixels) % VRAM_WIDTH;
|
2019-10-20 14:18:52 +00:00
|
|
|
}
|
|
|
|
|
2020-05-17 09:23:46 +00:00
|
|
|
// apply the crop from the start (usually overscan)
|
|
|
|
cs.display_vram_width -= std::min(cs.display_vram_width, horizontal_skip_pixels);
|
|
|
|
|
2020-07-11 06:58:56 +00:00
|
|
|
// Apply crop from the end by shrinking VRAM rectangle width if display would end outside the visible area.
|
|
|
|
cs.display_vram_width = std::min<u16>(cs.display_vram_width, cs.display_width - cs.display_origin_left);
|
|
|
|
|
2020-12-12 07:59:09 +00:00
|
|
|
if (vertical_display_start >= cs.vertical_visible_start)
|
2020-03-28 15:14:37 +00:00
|
|
|
{
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.display_origin_top = (vertical_display_start - cs.vertical_visible_start) << y_shift;
|
|
|
|
cs.display_vram_top = cs.regs.Y;
|
2020-03-28 15:14:37 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cs.display_origin_top = 0;
|
2020-12-15 01:30:02 +00:00
|
|
|
cs.display_vram_top = (cs.regs.Y + ((cs.vertical_visible_start - vertical_display_start) << y_shift)) % VRAM_HEIGHT;
|
2020-03-28 15:14:37 +00:00
|
|
|
}
|
|
|
|
|
2020-12-12 07:59:09 +00:00
|
|
|
if (vertical_display_end <= cs.vertical_visible_end)
|
2020-03-28 15:14:37 +00:00
|
|
|
{
|
2020-12-12 07:59:09 +00:00
|
|
|
cs.display_vram_height =
|
|
|
|
(vertical_display_end -
|
|
|
|
std::min(vertical_display_end, std::max(vertical_display_start, cs.vertical_visible_start)))
|
|
|
|
<< height_shift;
|
2020-03-28 15:14:37 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-04-27 11:49:34 +00:00
|
|
|
cs.display_vram_height =
|
2020-12-12 07:59:09 +00:00
|
|
|
(cs.vertical_visible_end -
|
|
|
|
std::min(cs.vertical_visible_end, std::max(vertical_display_start, cs.vertical_visible_start)))
|
2020-04-27 11:49:34 +00:00
|
|
|
<< height_shift;
|
2020-03-28 15:14:37 +00:00
|
|
|
}
|
2020-01-24 04:53:40 +00:00
|
|
|
}
|
2019-10-20 14:18:52 +00:00
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
TickCount GPU::GetPendingCRTCTicks() const
|
2020-03-01 07:04:36 +00:00
|
|
|
{
|
2020-06-12 15:28:49 +00:00
|
|
|
const TickCount pending_sysclk_ticks = m_crtc_tick_event->GetTicksSinceLastExecution();
|
2020-06-07 07:36:45 +00:00
|
|
|
TickCount fractional_ticks = m_crtc_state.fractional_ticks;
|
2020-06-12 15:28:49 +00:00
|
|
|
return SystemTicksToCRTCTicks(pending_sysclk_ticks, &fractional_ticks);
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
TickCount GPU::GetPendingCommandTicks() const
|
|
|
|
{
|
|
|
|
if (!m_command_tick_event->IsActive())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return SystemTicksToGPUTicks(m_command_tick_event->GetTicksSinceLastExecution());
|
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::UpdateCRTCTickEvent()
|
2019-09-14 13:49:55 +00:00
|
|
|
{
|
2020-04-28 10:30:44 +00:00
|
|
|
// figure out how many GPU ticks until the next vblank or event
|
2020-12-05 09:31:43 +00:00
|
|
|
TickCount lines_until_event;
|
2023-01-11 08:58:25 +00:00
|
|
|
if (Timers::IsSyncEnabled(HBLANK_TIMER_INDEX))
|
2020-12-05 09:31:43 +00:00
|
|
|
{
|
|
|
|
// when the timer sync is enabled we need to sync at vblank start and end
|
|
|
|
lines_until_event =
|
|
|
|
(m_crtc_state.current_scanline >= m_crtc_state.vertical_display_end) ?
|
|
|
|
(m_crtc_state.vertical_total - m_crtc_state.current_scanline + m_crtc_state.vertical_display_start) :
|
|
|
|
(m_crtc_state.vertical_display_end - m_crtc_state.current_scanline);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
lines_until_event =
|
|
|
|
(m_crtc_state.current_scanline >= m_crtc_state.vertical_display_end ?
|
|
|
|
(m_crtc_state.vertical_total - m_crtc_state.current_scanline + m_crtc_state.vertical_display_end) :
|
|
|
|
(m_crtc_state.vertical_display_end - m_crtc_state.current_scanline));
|
|
|
|
}
|
2023-01-11 08:58:25 +00:00
|
|
|
if (Timers::IsExternalIRQEnabled(HBLANK_TIMER_INDEX))
|
|
|
|
lines_until_event = std::min(lines_until_event, Timers::GetTicksUntilIRQ(HBLANK_TIMER_INDEX));
|
2020-12-05 05:59:47 +00:00
|
|
|
|
|
|
|
TickCount ticks_until_event =
|
2020-04-28 10:30:44 +00:00
|
|
|
lines_until_event * m_crtc_state.horizontal_total - m_crtc_state.current_tick_in_scanline;
|
2023-01-11 08:58:25 +00:00
|
|
|
if (Timers::IsExternalIRQEnabled(DOT_TIMER_INDEX))
|
2020-12-05 05:59:47 +00:00
|
|
|
{
|
2023-01-11 08:58:25 +00:00
|
|
|
const TickCount dots_until_irq = Timers::GetTicksUntilIRQ(DOT_TIMER_INDEX);
|
2020-12-05 09:31:43 +00:00
|
|
|
const TickCount ticks_until_irq =
|
|
|
|
(dots_until_irq * m_crtc_state.dot_clock_divider) - m_crtc_state.fractional_dot_ticks;
|
2020-12-05 05:59:47 +00:00
|
|
|
ticks_until_event = std::min(ticks_until_event, std::max<TickCount>(ticks_until_irq, 0));
|
|
|
|
}
|
2020-04-28 10:30:44 +00:00
|
|
|
|
|
|
|
#if 0
|
2020-03-01 07:06:38 +00:00
|
|
|
const TickCount ticks_until_hblank =
|
2020-01-24 04:53:40 +00:00
|
|
|
(m_crtc_state.current_tick_in_scanline >= m_crtc_state.horizontal_display_end) ?
|
2020-04-28 10:30:44 +00:00
|
|
|
(m_crtc_state.horizontal_total - m_crtc_state.current_tick_in_scanline + m_crtc_state.horizontal_display_end) :
|
|
|
|
(m_crtc_state.horizontal_display_end - m_crtc_state.current_tick_in_scanline);
|
|
|
|
#endif
|
2020-01-24 04:53:40 +00:00
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
m_crtc_tick_event->Schedule(CRTCTicksToSystemTicks(ticks_until_event, m_crtc_state.fractional_ticks));
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
bool GPU::IsCRTCScanlinePending() const
|
2020-03-01 07:07:17 +00:00
|
|
|
{
|
2021-01-28 16:46:15 +00:00
|
|
|
const TickCount ticks = (GetPendingCRTCTicks() + m_crtc_state.current_tick_in_scanline);
|
|
|
|
return (ticks >= (m_crtc_state.in_hblank ? m_crtc_state.horizontal_total : m_crtc_state.horizontal_sync_start));
|
2020-03-01 07:07:17 +00:00
|
|
|
}
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
bool GPU::IsCommandCompletionPending() const
|
2020-04-18 15:16:58 +00:00
|
|
|
{
|
2020-06-12 15:28:49 +00:00
|
|
|
return (m_pending_command_ticks > 0 && GetPendingCommandTicks() >= m_pending_command_ticks);
|
2020-04-18 15:16:58 +00:00
|
|
|
}
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
void GPU::CRTCTickEvent(TickCount ticks)
|
2019-09-17 04:25:25 +00:00
|
|
|
{
|
|
|
|
// convert cpu/master clock to GPU ticks, accounting for partial cycles because of the non-integer divider
|
|
|
|
{
|
2020-06-12 15:28:49 +00:00
|
|
|
const TickCount gpu_ticks = SystemTicksToCRTCTicks(ticks, &m_crtc_state.fractional_ticks);
|
2020-03-01 07:06:38 +00:00
|
|
|
m_crtc_state.current_tick_in_scanline += gpu_ticks;
|
2020-12-05 05:59:47 +00:00
|
|
|
|
2023-01-11 08:58:25 +00:00
|
|
|
if (Timers::IsUsingExternalClock(DOT_TIMER_INDEX))
|
2020-12-05 05:59:47 +00:00
|
|
|
{
|
|
|
|
m_crtc_state.fractional_dot_ticks += gpu_ticks;
|
|
|
|
const TickCount dots = m_crtc_state.fractional_dot_ticks / m_crtc_state.dot_clock_divider;
|
|
|
|
m_crtc_state.fractional_dot_ticks = m_crtc_state.fractional_dot_ticks % m_crtc_state.dot_clock_divider;
|
|
|
|
if (dots > 0)
|
2023-01-11 08:58:25 +00:00
|
|
|
Timers::AddTicks(DOT_TIMER_INDEX, dots);
|
2020-12-05 05:59:47 +00:00
|
|
|
}
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
|
2020-01-24 04:53:40 +00:00
|
|
|
if (m_crtc_state.current_tick_in_scanline < m_crtc_state.horizontal_total)
|
2019-09-17 04:25:25 +00:00
|
|
|
{
|
2020-01-24 04:53:40 +00:00
|
|
|
// short path when we execute <1 line.. this shouldn't occur often.
|
|
|
|
const bool old_hblank = m_crtc_state.in_hblank;
|
2020-08-16 15:21:52 +00:00
|
|
|
const bool new_hblank = (m_crtc_state.current_tick_in_scanline >= m_crtc_state.horizontal_sync_start);
|
|
|
|
m_crtc_state.in_hblank = new_hblank;
|
2023-01-11 08:58:25 +00:00
|
|
|
if (!old_hblank && new_hblank && Timers::IsUsingExternalClock(HBLANK_TIMER_INDEX))
|
|
|
|
Timers::AddTicks(HBLANK_TIMER_INDEX, 1);
|
2019-09-17 04:25:25 +00:00
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
UpdateCRTCTickEvent();
|
2020-01-24 04:53:40 +00:00
|
|
|
return;
|
|
|
|
}
|
2019-09-26 14:03:28 +00:00
|
|
|
|
2020-01-24 04:53:40 +00:00
|
|
|
u32 lines_to_draw = m_crtc_state.current_tick_in_scanline / m_crtc_state.horizontal_total;
|
|
|
|
m_crtc_state.current_tick_in_scanline %= m_crtc_state.horizontal_total;
|
|
|
|
#if 0
|
|
|
|
Log_WarningPrintf("Old line: %u, new line: %u, drawing %u", m_crtc_state.current_scanline,
|
|
|
|
m_crtc_state.current_scanline + lines_to_draw, lines_to_draw);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
const bool old_hblank = m_crtc_state.in_hblank;
|
2020-08-16 15:21:52 +00:00
|
|
|
const bool new_hblank = (m_crtc_state.current_tick_in_scanline >= m_crtc_state.horizontal_sync_start);
|
2020-01-24 04:53:40 +00:00
|
|
|
m_crtc_state.in_hblank = new_hblank;
|
2023-01-11 08:58:25 +00:00
|
|
|
if (Timers::IsUsingExternalClock(HBLANK_TIMER_INDEX))
|
2020-01-24 04:53:40 +00:00
|
|
|
{
|
|
|
|
const u32 hblank_timer_ticks = BoolToUInt32(!old_hblank) + BoolToUInt32(new_hblank) + (lines_to_draw - 1);
|
2023-01-11 08:58:25 +00:00
|
|
|
Timers::AddTicks(HBLANK_TIMER_INDEX, static_cast<TickCount>(hblank_timer_ticks));
|
2020-01-24 04:53:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
while (lines_to_draw > 0)
|
|
|
|
{
|
|
|
|
const u32 lines_to_draw_this_loop =
|
|
|
|
std::min(lines_to_draw, m_crtc_state.vertical_total - m_crtc_state.current_scanline);
|
|
|
|
const u32 prev_scanline = m_crtc_state.current_scanline;
|
|
|
|
m_crtc_state.current_scanline += lines_to_draw_this_loop;
|
|
|
|
DebugAssert(m_crtc_state.current_scanline <= m_crtc_state.vertical_total);
|
|
|
|
lines_to_draw -= lines_to_draw_this_loop;
|
|
|
|
|
|
|
|
// clear the vblank flag if the beam would pass through the display area
|
|
|
|
if (prev_scanline < m_crtc_state.vertical_display_start &&
|
|
|
|
m_crtc_state.current_scanline >= m_crtc_state.vertical_display_end)
|
|
|
|
{
|
2023-01-11 08:58:25 +00:00
|
|
|
Timers::SetGate(HBLANK_TIMER_INDEX, false);
|
2020-01-24 04:53:40 +00:00
|
|
|
m_crtc_state.in_vblank = false;
|
2019-09-26 14:03:28 +00:00
|
|
|
}
|
|
|
|
|
2019-11-11 13:36:11 +00:00
|
|
|
const bool new_vblank = m_crtc_state.current_scanline < m_crtc_state.vertical_display_start ||
|
|
|
|
m_crtc_state.current_scanline >= m_crtc_state.vertical_display_end;
|
|
|
|
if (m_crtc_state.in_vblank != new_vblank)
|
2019-09-17 04:25:25 +00:00
|
|
|
{
|
2019-11-11 13:36:11 +00:00
|
|
|
if (new_vblank)
|
2019-09-20 13:40:19 +00:00
|
|
|
{
|
2020-01-30 06:18:31 +00:00
|
|
|
Log_DebugPrintf("Now in v-blank");
|
2023-01-11 09:01:20 +00:00
|
|
|
InterruptController::InterruptRequest(InterruptController::IRQ::VBLANK);
|
2019-11-11 13:36:11 +00:00
|
|
|
|
|
|
|
// flush any pending draws and "scan out" the image
|
2023-08-15 13:12:21 +00:00
|
|
|
// TODO: move present in here I guess
|
2019-11-11 13:36:11 +00:00
|
|
|
FlushRender();
|
|
|
|
UpdateDisplay();
|
2023-08-15 13:12:21 +00:00
|
|
|
TimingEvents::SetFrameDone();
|
2020-05-16 15:02:20 +00:00
|
|
|
|
|
|
|
// switch fields early. this is needed so we draw to the correct one.
|
2020-08-15 04:56:20 +00:00
|
|
|
if (m_GPUSTAT.InInterleaved480iMode())
|
|
|
|
m_crtc_state.interlaced_display_field = m_crtc_state.interlaced_field ^ 1u;
|
2020-05-16 15:02:20 +00:00
|
|
|
else
|
2020-08-15 04:56:20 +00:00
|
|
|
m_crtc_state.interlaced_display_field = 0;
|
2019-09-20 13:40:19 +00:00
|
|
|
}
|
|
|
|
|
2023-01-11 08:58:25 +00:00
|
|
|
Timers::SetGate(HBLANK_TIMER_INDEX, new_vblank);
|
2020-01-30 06:18:31 +00:00
|
|
|
m_crtc_state.in_vblank = new_vblank;
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
|
2020-01-24 04:53:40 +00:00
|
|
|
// past the end of vblank?
|
|
|
|
if (m_crtc_state.current_scanline == m_crtc_state.vertical_total)
|
2019-12-07 14:40:52 +00:00
|
|
|
{
|
2020-01-24 04:53:40 +00:00
|
|
|
// start the new frame
|
|
|
|
m_crtc_state.current_scanline = 0;
|
2020-08-15 04:56:20 +00:00
|
|
|
if (m_GPUSTAT.vertical_interlace)
|
2020-08-18 14:02:35 +00:00
|
|
|
{
|
2020-08-15 04:56:20 +00:00
|
|
|
m_crtc_state.interlaced_field ^= 1u;
|
2021-09-29 00:48:53 +00:00
|
|
|
m_GPUSTAT.interlaced_field = !m_crtc_state.interlaced_field;
|
2020-08-18 14:02:35 +00:00
|
|
|
}
|
2020-08-15 04:56:20 +00:00
|
|
|
else
|
2020-08-18 14:02:35 +00:00
|
|
|
{
|
2020-08-15 04:56:20 +00:00
|
|
|
m_crtc_state.interlaced_field = 0;
|
2020-08-20 11:30:11 +00:00
|
|
|
m_GPUSTAT.interlaced_field = 0u; // new GPU = 1, old GPU = 0
|
2020-08-18 14:02:35 +00:00
|
|
|
}
|
2019-12-07 14:40:52 +00:00
|
|
|
}
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
|
2020-01-24 04:53:40 +00:00
|
|
|
// alternating even line bit in 240-line mode
|
2020-08-15 04:56:20 +00:00
|
|
|
if (m_GPUSTAT.InInterleaved480iMode())
|
2019-11-11 13:36:11 +00:00
|
|
|
{
|
2020-05-26 12:57:56 +00:00
|
|
|
m_crtc_state.active_line_lsb =
|
2020-08-15 04:56:20 +00:00
|
|
|
Truncate8((m_crtc_state.regs.Y + BoolToUInt32(m_crtc_state.interlaced_display_field)) & u32(1));
|
|
|
|
m_GPUSTAT.display_line_lsb = ConvertToBoolUnchecked(
|
2021-04-14 15:32:05 +00:00
|
|
|
(m_crtc_state.regs.Y + (BoolToUInt8(!m_crtc_state.in_vblank) & m_crtc_state.interlaced_display_field)) & u32(1));
|
2020-01-24 04:53:40 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-05-26 12:57:56 +00:00
|
|
|
m_crtc_state.active_line_lsb = 0;
|
|
|
|
m_GPUSTAT.display_line_lsb = ConvertToBoolUnchecked((m_crtc_state.regs.Y + m_crtc_state.current_scanline) & u32(1));
|
2019-11-11 13:36:11 +00:00
|
|
|
}
|
|
|
|
|
2020-06-12 15:28:49 +00:00
|
|
|
UpdateCRTCTickEvent();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::CommandTickEvent(TickCount ticks)
|
|
|
|
{
|
|
|
|
m_pending_command_ticks -= SystemTicksToGPUTicks(ticks);
|
2020-08-02 04:09:19 +00:00
|
|
|
m_command_tick_event->Deactivate();
|
2020-06-12 15:28:49 +00:00
|
|
|
|
|
|
|
// we can be syncing if this came from a DMA write. recursively executing commands would be bad.
|
|
|
|
if (!m_syncing)
|
|
|
|
ExecuteCommands();
|
|
|
|
|
|
|
|
UpdateGPUIdle();
|
|
|
|
|
|
|
|
if (m_pending_command_ticks <= 0)
|
|
|
|
m_pending_command_ticks = 0;
|
|
|
|
else
|
|
|
|
m_command_tick_event->SetIntervalAndSchedule(GPUTicksToSystemTicks(m_pending_command_ticks));
|
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::UpdateCommandTickEvent()
|
|
|
|
{
|
|
|
|
if (m_pending_command_ticks <= 0)
|
|
|
|
m_command_tick_event->Deactivate();
|
|
|
|
else if (!m_command_tick_event->IsActive())
|
|
|
|
m_command_tick_event->SetIntervalAndSchedule(GPUTicksToSystemTicks(m_pending_command_ticks));
|
2019-09-14 13:49:55 +00:00
|
|
|
}
|
|
|
|
|
2023-09-20 06:56:12 +00:00
|
|
|
void GPU::ConvertScreenCoordinatesToDisplayCoordinates(float window_x, float window_y, float* display_x,
|
|
|
|
float* display_y) const
|
2020-04-25 15:10:46 +00:00
|
|
|
{
|
2023-09-20 06:56:12 +00:00
|
|
|
float scale;
|
|
|
|
const Common::Rectangle<float> draw_rc = CalculateDrawRect(
|
|
|
|
g_gpu_device->GetWindowWidth(), g_gpu_device->GetWindowHeight(), nullptr, nullptr, &scale, nullptr);
|
2023-08-27 08:13:50 +00:00
|
|
|
|
|
|
|
// convert coordinates to active display region, then to full display region
|
2023-09-20 06:56:12 +00:00
|
|
|
const float scaled_display_x = (window_x - draw_rc.left) / static_cast<float>(draw_rc.GetWidth());
|
|
|
|
const float scaled_display_y = (window_y - draw_rc.top) / static_cast<float>(draw_rc.GetHeight());
|
2023-08-27 08:13:50 +00:00
|
|
|
|
|
|
|
// scale back to internal resolution
|
2023-09-20 06:56:12 +00:00
|
|
|
*display_x = scaled_display_x * static_cast<float>(m_crtc_state.display_width);
|
|
|
|
*display_y = scaled_display_y * static_cast<float>(m_crtc_state.display_height);
|
2020-12-19 04:22:59 +00:00
|
|
|
|
2023-09-20 06:56:12 +00:00
|
|
|
Log_DebugPrintf("win %.0f,%.0f -> disp %.2f,%.2f (size %u,%u frac %f,%f)", window_x, window_y, *display_x, *display_y,
|
|
|
|
m_crtc_state.display_width, m_crtc_state.display_height,
|
|
|
|
*display_x / static_cast<float>(m_crtc_state.display_width),
|
|
|
|
*display_y / static_cast<float>(m_crtc_state.display_height));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GPU::ConvertDisplayCoordinatesToBeamTicksAndLines(float display_x, float display_y, float x_scale, u32* out_tick,
|
|
|
|
u32* out_line) const
|
|
|
|
{
|
2020-12-19 04:22:59 +00:00
|
|
|
if (x_scale != 1.0f)
|
|
|
|
{
|
|
|
|
const float dw = static_cast<float>(m_crtc_state.display_width);
|
2020-12-19 04:29:35 +00:00
|
|
|
float scaled_x = ((display_x / dw) * 2.0f) - 1.0f; // 0..1 -> -1..1
|
2020-12-19 04:22:59 +00:00
|
|
|
scaled_x *= x_scale;
|
2020-12-19 04:29:35 +00:00
|
|
|
display_x = (((scaled_x + 1.0f) * 0.5f) * dw); // -1..1 -> 0..1
|
2020-12-19 04:22:59 +00:00
|
|
|
}
|
|
|
|
|
2020-04-25 15:10:46 +00:00
|
|
|
if (display_x < 0 || static_cast<u32>(display_x) >= m_crtc_state.display_width || display_y < 0 ||
|
|
|
|
static_cast<u32>(display_y) >= m_crtc_state.display_height)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-12-19 04:29:35 +00:00
|
|
|
*out_line = (static_cast<u32>(std::round(display_y)) >> BoolToUInt8(m_GPUSTAT.vertical_interlace)) +
|
|
|
|
m_crtc_state.vertical_visible_start;
|
|
|
|
*out_tick = static_cast<u32>(std::round(display_x * static_cast<float>(m_crtc_state.dot_clock_divider))) +
|
|
|
|
m_crtc_state.horizontal_visible_start;
|
2020-04-25 15:10:46 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-09-11 04:59:41 +00:00
|
|
|
u32 GPU::ReadGPUREAD()
|
2019-09-11 04:01:19 +00:00
|
|
|
{
|
2020-04-18 15:16:58 +00:00
|
|
|
if (m_blitter_state != BlitterState::ReadingVRAM)
|
2019-11-14 12:17:09 +00:00
|
|
|
return m_GPUREAD_latch;
|
|
|
|
|
|
|
|
// Read two pixels out of VRAM and combine them. Zero fill odd pixel counts.
|
|
|
|
u32 value = 0;
|
|
|
|
for (u32 i = 0; i < 2; i++)
|
2019-09-13 16:07:31 +00:00
|
|
|
{
|
2019-11-14 12:17:09 +00:00
|
|
|
// Read with correct wrap-around behavior.
|
|
|
|
const u16 read_x = (m_vram_transfer.x + m_vram_transfer.col) % VRAM_WIDTH;
|
|
|
|
const u16 read_y = (m_vram_transfer.y + m_vram_transfer.row) % VRAM_HEIGHT;
|
2020-03-29 14:41:31 +00:00
|
|
|
value |= ZeroExtend32(m_vram_ptr[read_y * VRAM_WIDTH + read_x]) << (i * 16);
|
2019-11-14 12:17:09 +00:00
|
|
|
|
|
|
|
if (++m_vram_transfer.col == m_vram_transfer.width)
|
|
|
|
{
|
|
|
|
m_vram_transfer.col = 0;
|
|
|
|
|
|
|
|
if (++m_vram_transfer.row == m_vram_transfer.height)
|
|
|
|
{
|
|
|
|
Log_DebugPrintf("End of VRAM->CPU transfer");
|
|
|
|
m_vram_transfer = {};
|
2020-04-18 15:16:58 +00:00
|
|
|
m_blitter_state = BlitterState::Idle;
|
2019-11-14 12:17:09 +00:00
|
|
|
|
|
|
|
// end of transfer, catch up on any commands which were written (unlikely)
|
|
|
|
ExecuteCommands();
|
2020-06-12 15:28:49 +00:00
|
|
|
UpdateCommandTickEvent();
|
2019-11-14 12:17:09 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-09-13 16:07:31 +00:00
|
|
|
}
|
|
|
|
|
2019-11-14 12:17:09 +00:00
|
|
|
m_GPUREAD_latch = value;
|
2019-09-13 16:07:31 +00:00
|
|
|
return value;
|
2019-09-11 04:01:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::WriteGP1(u32 value)
|
|
|
|
{
|
2020-07-03 15:28:18 +00:00
|
|
|
const u32 command = (value >> 24) & 0x3Fu;
|
2019-09-11 04:59:41 +00:00
|
|
|
const u32 param = value & UINT32_C(0x00FFFFFF);
|
|
|
|
switch (command)
|
|
|
|
{
|
2019-10-15 12:36:26 +00:00
|
|
|
case 0x00: // Reset GPU
|
|
|
|
{
|
|
|
|
Log_DebugPrintf("GP1 reset GPU");
|
2020-07-04 16:25:55 +00:00
|
|
|
m_command_tick_event->InvokeEarly();
|
2020-06-12 15:28:49 +00:00
|
|
|
SynchronizeCRTC();
|
2019-10-15 12:36:26 +00:00
|
|
|
SoftReset();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-09-14 13:49:55 +00:00
|
|
|
case 0x01: // Clear FIFO
|
|
|
|
{
|
|
|
|
Log_DebugPrintf("GP1 clear FIFO");
|
2020-07-04 16:25:55 +00:00
|
|
|
m_command_tick_event->InvokeEarly();
|
2020-06-12 15:28:49 +00:00
|
|
|
SynchronizeCRTC();
|
2020-11-18 10:35:02 +00:00
|
|
|
|
|
|
|
// flush partial writes
|
|
|
|
if (m_blitter_state == BlitterState::WritingVRAM)
|
|
|
|
FinishVRAMWrite();
|
|
|
|
|
2020-04-18 15:16:58 +00:00
|
|
|
m_blitter_state = BlitterState::Idle;
|
2019-11-14 12:17:09 +00:00
|
|
|
m_command_total_words = 0;
|
|
|
|
m_vram_transfer = {};
|
2020-04-18 15:16:58 +00:00
|
|
|
m_fifo.Clear();
|
|
|
|
m_blit_buffer.clear();
|
|
|
|
m_blit_remaining_words = 0;
|
2020-06-12 15:28:49 +00:00
|
|
|
m_pending_command_ticks = 0;
|
|
|
|
m_command_tick_event->Deactivate();
|
2020-03-01 07:06:38 +00:00
|
|
|
UpdateDMARequest();
|
2020-06-12 15:28:49 +00:00
|
|
|
UpdateGPUIdle();
|
2019-09-14 13:49:55 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-10-05 12:22:49 +00:00
|
|
|
case 0x02: // Acknowledge Interrupt
|
|
|
|
{
|
|
|
|
Log_DebugPrintf("Acknowledge interrupt");
|
|
|
|
m_GPUSTAT.interrupt_request = false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-10-28 07:43:34 +00:00
|
|
|
case 0x03: // Display on/off
|
|
|
|
{
|
|
|
|
const bool disable = ConvertToBoolUnchecked(value & 0x01);
|
|
|
|
Log_DebugPrintf("Display %s", disable ? "disabled" : "enabled");
|
2020-06-12 15:28:49 +00:00
|
|
|
SynchronizeCRTC();
|
2021-01-17 17:28:12 +00:00
|
|
|
|
|
|
|
if (!m_GPUSTAT.display_disable && disable && m_GPUSTAT.vertical_interlace && !m_force_progressive_scan)
|
|
|
|
ClearDisplay();
|
|
|
|
|
2019-10-28 07:43:34 +00:00
|
|
|
m_GPUSTAT.display_disable = disable;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-09-11 04:59:41 +00:00
|
|
|
case 0x04: // DMA Direction
|
|
|
|
{
|
2020-04-28 07:55:43 +00:00
|
|
|
Log_DebugPrintf("DMA direction <- 0x%02X", static_cast<u32>(param));
|
2020-04-18 15:16:58 +00:00
|
|
|
if (m_GPUSTAT.dma_direction != static_cast<DMADirection>(param))
|
|
|
|
{
|
|
|
|
m_GPUSTAT.dma_direction = static_cast<DMADirection>(param);
|
|
|
|
UpdateDMARequest();
|
|
|
|
}
|
2019-09-13 16:07:31 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x05: // Set display start address
|
|
|
|
{
|
2020-10-23 04:41:42 +00:00
|
|
|
const u32 new_value = param & CRTCState::Regs::DISPLAY_ADDRESS_START_MASK;
|
|
|
|
Log_DebugPrintf("Display address start <- 0x%08X", new_value);
|
|
|
|
|
2020-07-31 07:09:18 +00:00
|
|
|
System::IncrementInternalFrameNumber();
|
2020-10-23 04:41:42 +00:00
|
|
|
if (m_crtc_state.regs.display_address_start != new_value)
|
|
|
|
{
|
|
|
|
SynchronizeCRTC();
|
|
|
|
m_crtc_state.regs.display_address_start = new_value;
|
2020-10-23 09:37:56 +00:00
|
|
|
UpdateCRTCDisplayParameters();
|
2020-10-23 04:41:42 +00:00
|
|
|
}
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x06: // Set horizontal display range
|
|
|
|
{
|
2020-01-24 04:53:40 +00:00
|
|
|
const u32 new_value = param & CRTCState::Regs::HORIZONTAL_DISPLAY_RANGE_MASK;
|
|
|
|
Log_DebugPrintf("Horizontal display range <- 0x%08X", new_value);
|
|
|
|
|
|
|
|
if (m_crtc_state.regs.horizontal_display_range != new_value)
|
|
|
|
{
|
2020-06-12 15:28:49 +00:00
|
|
|
SynchronizeCRTC();
|
2020-01-24 04:53:40 +00:00
|
|
|
m_crtc_state.regs.horizontal_display_range = new_value;
|
|
|
|
UpdateCRTCConfig();
|
|
|
|
}
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2020-10-21 15:31:15 +00:00
|
|
|
case 0x07: // Set vertical display range
|
2019-09-17 04:25:25 +00:00
|
|
|
{
|
2020-01-24 04:53:40 +00:00
|
|
|
const u32 new_value = param & CRTCState::Regs::VERTICAL_DISPLAY_RANGE_MASK;
|
|
|
|
Log_DebugPrintf("Vertical display range <- 0x%08X", new_value);
|
|
|
|
|
|
|
|
if (m_crtc_state.regs.vertical_display_range != new_value)
|
|
|
|
{
|
2020-06-12 15:28:49 +00:00
|
|
|
SynchronizeCRTC();
|
2020-01-24 04:53:40 +00:00
|
|
|
m_crtc_state.regs.vertical_display_range = new_value;
|
|
|
|
UpdateCRTCConfig();
|
|
|
|
}
|
2019-09-17 04:25:25 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x08: // Set display mode
|
|
|
|
{
|
|
|
|
union GP1_08h
|
|
|
|
{
|
|
|
|
u32 bits;
|
|
|
|
|
|
|
|
BitField<u32, u8, 0, 2> horizontal_resolution_1;
|
2019-10-15 12:36:10 +00:00
|
|
|
BitField<u32, bool, 2, 1> vertical_resolution;
|
2019-09-17 04:25:25 +00:00
|
|
|
BitField<u32, bool, 3, 1> pal_mode;
|
|
|
|
BitField<u32, bool, 4, 1> display_area_color_depth;
|
|
|
|
BitField<u32, bool, 5, 1> vertical_interlace;
|
|
|
|
BitField<u32, bool, 6, 1> horizontal_resolution_2;
|
|
|
|
BitField<u32, bool, 7, 1> reverse_flag;
|
|
|
|
};
|
|
|
|
|
|
|
|
const GP1_08h dm{param};
|
2020-01-24 04:53:40 +00:00
|
|
|
GPUSTAT new_GPUSTAT{m_GPUSTAT.bits};
|
|
|
|
new_GPUSTAT.horizontal_resolution_1 = dm.horizontal_resolution_1;
|
|
|
|
new_GPUSTAT.vertical_resolution = dm.vertical_resolution;
|
|
|
|
new_GPUSTAT.pal_mode = dm.pal_mode;
|
|
|
|
new_GPUSTAT.display_area_color_depth_24 = dm.display_area_color_depth;
|
|
|
|
new_GPUSTAT.vertical_interlace = dm.vertical_interlace;
|
|
|
|
new_GPUSTAT.horizontal_resolution_2 = dm.horizontal_resolution_2;
|
|
|
|
new_GPUSTAT.reverse_flag = dm.reverse_flag;
|
2019-09-17 04:25:25 +00:00
|
|
|
Log_DebugPrintf("Set display mode <- 0x%08X", dm.bits);
|
2020-01-24 04:53:40 +00:00
|
|
|
|
2020-08-02 17:26:11 +00:00
|
|
|
if (!m_GPUSTAT.vertical_interlace && dm.vertical_interlace && !m_force_progressive_scan)
|
|
|
|
{
|
|
|
|
// bit of a hack, technically we should pull the previous frame in, but this may not exist anymore
|
|
|
|
ClearDisplay();
|
|
|
|
}
|
|
|
|
|
2020-01-24 04:53:40 +00:00
|
|
|
if (m_GPUSTAT.bits != new_GPUSTAT.bits)
|
|
|
|
{
|
2020-04-18 15:16:58 +00:00
|
|
|
// Have to be careful when setting this because Synchronize() can modify GPUSTAT.
|
|
|
|
static constexpr u32 SET_MASK = UINT32_C(0b00000000011111110100000000000000);
|
2020-07-04 16:25:55 +00:00
|
|
|
m_command_tick_event->InvokeEarly();
|
2020-06-12 15:28:49 +00:00
|
|
|
SynchronizeCRTC();
|
2020-04-18 15:16:58 +00:00
|
|
|
m_GPUSTAT.bits = (m_GPUSTAT.bits & ~SET_MASK) | (new_GPUSTAT.bits & SET_MASK);
|
2020-01-24 04:53:40 +00:00
|
|
|
UpdateCRTCConfig();
|
|
|
|
}
|
2019-09-11 04:59:41 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-12-11 06:47:49 +00:00
|
|
|
case 0x09: // Allow texture disable
|
|
|
|
{
|
|
|
|
m_set_texture_disable_mask = ConvertToBoolUnchecked(param & 0x01);
|
|
|
|
Log_DebugPrintf("Set texture disable mask <- %s", m_set_texture_disable_mask ? "allowed" : "ignored");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-10-05 12:22:49 +00:00
|
|
|
case 0x10:
|
|
|
|
case 0x11:
|
|
|
|
case 0x12:
|
|
|
|
case 0x13:
|
|
|
|
case 0x14:
|
|
|
|
case 0x15:
|
|
|
|
case 0x16:
|
|
|
|
case 0x17:
|
|
|
|
case 0x18:
|
|
|
|
case 0x19:
|
|
|
|
case 0x1A:
|
|
|
|
case 0x1B:
|
|
|
|
case 0x1C:
|
|
|
|
case 0x1D:
|
|
|
|
case 0x1E:
|
|
|
|
case 0x1F:
|
|
|
|
{
|
2019-10-05 14:37:31 +00:00
|
|
|
HandleGetGPUInfoCommand(value);
|
2019-10-05 12:22:49 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-09-11 04:59:41 +00:00
|
|
|
default:
|
|
|
|
Log_ErrorPrintf("Unimplemented GP1 command 0x%02X", command);
|
|
|
|
break;
|
2019-10-05 12:22:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-05 14:37:31 +00:00
|
|
|
void GPU::HandleGetGPUInfoCommand(u32 value)
|
2019-10-05 12:22:49 +00:00
|
|
|
{
|
2019-10-05 14:37:31 +00:00
|
|
|
const u8 subcommand = Truncate8(value & 0x07);
|
2019-10-05 12:22:49 +00:00
|
|
|
switch (subcommand)
|
|
|
|
{
|
|
|
|
case 0x00:
|
|
|
|
case 0x01:
|
|
|
|
case 0x06:
|
|
|
|
case 0x07:
|
|
|
|
// leave GPUREAD intact
|
|
|
|
break;
|
|
|
|
|
2019-10-06 13:12:17 +00:00
|
|
|
case 0x02: // Get Texture Window
|
|
|
|
{
|
|
|
|
Log_DebugPrintf("Get texture window");
|
2019-12-11 06:35:14 +00:00
|
|
|
m_GPUREAD_latch = m_draw_mode.texture_window_value;
|
2019-10-06 13:12:17 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x03: // Get Draw Area Top Left
|
|
|
|
{
|
|
|
|
Log_DebugPrintf("Get drawing area top left");
|
2019-11-14 12:17:09 +00:00
|
|
|
m_GPUREAD_latch =
|
|
|
|
((m_drawing_area.left & UINT32_C(0b1111111111)) | ((m_drawing_area.top & UINT32_C(0b1111111111)) << 10));
|
2019-10-06 13:12:17 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x04: // Get Draw Area Bottom Right
|
|
|
|
{
|
|
|
|
Log_DebugPrintf("Get drawing area bottom right");
|
2019-11-14 12:17:09 +00:00
|
|
|
m_GPUREAD_latch =
|
|
|
|
((m_drawing_area.right & UINT32_C(0b1111111111)) | ((m_drawing_area.bottom & UINT32_C(0b1111111111)) << 10));
|
2019-10-06 13:12:17 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x05: // Get Drawing Offset
|
|
|
|
{
|
|
|
|
Log_DebugPrintf("Get drawing offset");
|
2019-11-14 12:17:09 +00:00
|
|
|
m_GPUREAD_latch =
|
|
|
|
((m_drawing_offset.x & INT32_C(0b11111111111)) | ((m_drawing_offset.y & INT32_C(0b11111111111)) << 11));
|
2019-10-06 13:12:17 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-10-05 12:22:49 +00:00
|
|
|
default:
|
|
|
|
Log_WarningPrintf("Unhandled GetGPUInfo(0x%02X)", ZeroExtend32(subcommand));
|
|
|
|
break;
|
2019-09-11 04:59:41 +00:00
|
|
|
}
|
2019-09-11 04:01:19 +00:00
|
|
|
}
|
2019-09-11 06:04:31 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
void GPU::ClearDisplay()
|
|
|
|
{
|
|
|
|
}
|
2020-08-02 17:26:11 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
void GPU::UpdateDisplay()
|
|
|
|
{
|
|
|
|
}
|
2019-09-13 16:07:31 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
void GPU::ReadVRAM(u32 x, u32 y, u32 width, u32 height)
|
|
|
|
{
|
|
|
|
}
|
2019-09-14 10:45:26 +00:00
|
|
|
|
2020-02-28 14:18:33 +00:00
|
|
|
void GPU::FillVRAM(u32 x, u32 y, u32 width, u32 height, u32 color)
|
|
|
|
{
|
2021-03-18 02:55:02 +00:00
|
|
|
const u16 color16 = VRAMRGBA8888ToRGBA5551(color);
|
2020-04-03 14:11:39 +00:00
|
|
|
if ((x + width) <= VRAM_WIDTH && !IsInterlacedRenderingEnabled())
|
2020-03-20 14:15:35 +00:00
|
|
|
{
|
|
|
|
for (u32 yoffs = 0; yoffs < height; yoffs++)
|
|
|
|
{
|
|
|
|
const u32 row = (y + yoffs) % VRAM_HEIGHT;
|
|
|
|
std::fill_n(&m_vram_ptr[row * VRAM_WIDTH + x], width, color16);
|
|
|
|
}
|
|
|
|
}
|
2020-04-03 14:11:39 +00:00
|
|
|
else if (IsInterlacedRenderingEnabled())
|
|
|
|
{
|
|
|
|
// Hardware tests show that fills seem to break on the first two lines when the offset matches the displayed field.
|
2020-06-12 15:28:49 +00:00
|
|
|
if (IsCRTCScanlinePending())
|
|
|
|
SynchronizeCRTC();
|
2020-05-16 15:02:20 +00:00
|
|
|
|
2020-05-26 12:57:56 +00:00
|
|
|
const u32 active_field = GetActiveLineLSB();
|
2020-04-03 14:11:39 +00:00
|
|
|
for (u32 yoffs = 0; yoffs < height; yoffs++)
|
|
|
|
{
|
|
|
|
const u32 row = (y + yoffs) % VRAM_HEIGHT;
|
|
|
|
if ((row & u32(1)) == active_field)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
u16* row_ptr = &m_vram_ptr[row * VRAM_WIDTH];
|
|
|
|
for (u32 xoffs = 0; xoffs < width; xoffs++)
|
|
|
|
{
|
|
|
|
const u32 col = (x + xoffs) % VRAM_WIDTH;
|
|
|
|
row_ptr[col] = color16;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-03-20 14:15:35 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
for (u32 yoffs = 0; yoffs < height; yoffs++)
|
|
|
|
{
|
|
|
|
const u32 row = (y + yoffs) % VRAM_HEIGHT;
|
|
|
|
u16* row_ptr = &m_vram_ptr[row * VRAM_WIDTH];
|
|
|
|
for (u32 xoffs = 0; xoffs < width; xoffs++)
|
|
|
|
{
|
|
|
|
const u32 col = (x + xoffs) % VRAM_WIDTH;
|
|
|
|
row_ptr[col] = color16;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-02-28 14:18:33 +00:00
|
|
|
}
|
2019-09-14 06:43:39 +00:00
|
|
|
|
2020-12-14 16:19:28 +00:00
|
|
|
void GPU::UpdateVRAM(u32 x, u32 y, u32 width, u32 height, const void* data, bool set_mask, bool check_mask)
|
2019-11-14 10:31:48 +00:00
|
|
|
{
|
|
|
|
// Fast path when the copy is not oversized.
|
2020-12-14 16:19:28 +00:00
|
|
|
if ((x + width) <= VRAM_WIDTH && (y + height) <= VRAM_HEIGHT && !set_mask && !check_mask)
|
2019-11-14 10:31:48 +00:00
|
|
|
{
|
|
|
|
const u16* src_ptr = static_cast<const u16*>(data);
|
|
|
|
u16* dst_ptr = &m_vram_ptr[y * VRAM_WIDTH + x];
|
|
|
|
for (u32 yoffs = 0; yoffs < height; yoffs++)
|
|
|
|
{
|
|
|
|
std::copy_n(src_ptr, width, dst_ptr);
|
|
|
|
src_ptr += width;
|
|
|
|
dst_ptr += VRAM_WIDTH;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Slow path when we need to handle wrap-around.
|
2020-12-14 16:19:28 +00:00
|
|
|
// During transfer/render operations, if ((dst_pixel & mask_and) == 0) { pixel = src_pixel | mask_or }
|
2019-11-14 10:31:48 +00:00
|
|
|
const u16* src_ptr = static_cast<const u16*>(data);
|
2020-12-14 16:19:28 +00:00
|
|
|
const u16 mask_and = check_mask ? 0x8000 : 0;
|
|
|
|
const u16 mask_or = set_mask ? 0x8000 : 0;
|
2019-11-24 08:47:40 +00:00
|
|
|
|
2019-11-14 10:31:48 +00:00
|
|
|
for (u32 row = 0; row < height;)
|
|
|
|
{
|
|
|
|
u16* dst_row_ptr = &m_vram_ptr[((y + row++) % VRAM_HEIGHT) * VRAM_WIDTH];
|
|
|
|
for (u32 col = 0; col < width;)
|
|
|
|
{
|
|
|
|
// TODO: Handle unaligned reads...
|
2019-11-24 08:47:40 +00:00
|
|
|
u16* pixel_ptr = &dst_row_ptr[(x + col++) % VRAM_WIDTH];
|
2020-03-05 14:29:35 +00:00
|
|
|
if (((*pixel_ptr) & mask_and) == 0)
|
2019-11-24 08:47:40 +00:00
|
|
|
*pixel_ptr = *(src_ptr++) | mask_or;
|
2019-11-14 10:31:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-09-12 15:10:08 +00:00
|
|
|
|
2020-02-28 14:18:33 +00:00
|
|
|
void GPU::CopyVRAM(u32 src_x, u32 src_y, u32 dst_x, u32 dst_y, u32 width, u32 height)
|
|
|
|
{
|
2020-04-19 14:54:46 +00:00
|
|
|
// Break up oversized copies. This behavior has not been verified on console.
|
|
|
|
if ((src_x + width) > VRAM_WIDTH || (dst_x + width) > VRAM_WIDTH)
|
|
|
|
{
|
|
|
|
u32 remaining_rows = height;
|
|
|
|
u32 current_src_y = src_y;
|
|
|
|
u32 current_dst_y = dst_y;
|
|
|
|
while (remaining_rows > 0)
|
|
|
|
{
|
|
|
|
const u32 rows_to_copy =
|
|
|
|
std::min<u32>(remaining_rows, std::min<u32>(VRAM_HEIGHT - current_src_y, VRAM_HEIGHT - current_dst_y));
|
|
|
|
|
|
|
|
u32 remaining_columns = width;
|
|
|
|
u32 current_src_x = src_x;
|
|
|
|
u32 current_dst_x = dst_x;
|
|
|
|
while (remaining_columns > 0)
|
|
|
|
{
|
|
|
|
const u32 columns_to_copy =
|
|
|
|
std::min<u32>(remaining_columns, std::min<u32>(VRAM_WIDTH - current_src_x, VRAM_WIDTH - current_dst_x));
|
|
|
|
CopyVRAM(current_src_x, current_src_y, current_dst_x, current_dst_y, columns_to_copy, rows_to_copy);
|
|
|
|
current_src_x = (current_src_x + columns_to_copy) % VRAM_WIDTH;
|
|
|
|
current_dst_x = (current_dst_x + columns_to_copy) % VRAM_WIDTH;
|
|
|
|
remaining_columns -= columns_to_copy;
|
|
|
|
}
|
|
|
|
|
|
|
|
current_src_y = (current_src_y + rows_to_copy) % VRAM_HEIGHT;
|
|
|
|
current_dst_y = (current_dst_y + rows_to_copy) % VRAM_HEIGHT;
|
|
|
|
remaining_rows -= rows_to_copy;
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-28 14:18:33 +00:00
|
|
|
// This doesn't have a fast path, but do we really need one? It's not common.
|
|
|
|
const u16 mask_and = m_GPUSTAT.GetMaskAND();
|
|
|
|
const u16 mask_or = m_GPUSTAT.GetMaskOR();
|
|
|
|
|
2020-04-19 14:54:46 +00:00
|
|
|
// Copy in reverse when src_x < dst_x, this is verified on console.
|
|
|
|
if (src_x < dst_x || ((src_x + width - 1) % VRAM_WIDTH) < ((dst_x + width - 1) % VRAM_WIDTH))
|
2020-02-28 14:18:33 +00:00
|
|
|
{
|
2020-04-19 14:54:46 +00:00
|
|
|
for (u32 row = 0; row < height; row++)
|
|
|
|
{
|
|
|
|
const u16* src_row_ptr = &m_vram_ptr[((src_y + row) % VRAM_HEIGHT) * VRAM_WIDTH];
|
|
|
|
u16* dst_row_ptr = &m_vram_ptr[((dst_y + row) % VRAM_HEIGHT) * VRAM_WIDTH];
|
2020-02-28 14:18:33 +00:00
|
|
|
|
2020-04-19 14:54:46 +00:00
|
|
|
for (s32 col = static_cast<s32>(width - 1); col >= 0; col--)
|
|
|
|
{
|
|
|
|
const u16 src_pixel = src_row_ptr[(src_x + static_cast<u32>(col)) % VRAM_WIDTH];
|
|
|
|
u16* dst_pixel_ptr = &dst_row_ptr[(dst_x + static_cast<u32>(col)) % VRAM_WIDTH];
|
|
|
|
if ((*dst_pixel_ptr & mask_and) == 0)
|
|
|
|
*dst_pixel_ptr = src_pixel | mask_or;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (u32 row = 0; row < height; row++)
|
2020-02-28 14:18:33 +00:00
|
|
|
{
|
2020-04-19 14:54:46 +00:00
|
|
|
const u16* src_row_ptr = &m_vram_ptr[((src_y + row) % VRAM_HEIGHT) * VRAM_WIDTH];
|
|
|
|
u16* dst_row_ptr = &m_vram_ptr[((dst_y + row) % VRAM_HEIGHT) * VRAM_WIDTH];
|
|
|
|
|
|
|
|
for (u32 col = 0; col < width; col++)
|
|
|
|
{
|
|
|
|
const u16 src_pixel = src_row_ptr[(src_x + col) % VRAM_WIDTH];
|
|
|
|
u16* dst_pixel_ptr = &dst_row_ptr[(dst_x + col) % VRAM_WIDTH];
|
|
|
|
if ((*dst_pixel_ptr & mask_and) == 0)
|
|
|
|
*dst_pixel_ptr = src_pixel | mask_or;
|
|
|
|
}
|
2020-02-28 14:18:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-09-17 14:58:30 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
void GPU::DispatchRenderCommand()
|
|
|
|
{
|
|
|
|
}
|
2019-09-12 02:53:04 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
void GPU::FlushRender()
|
|
|
|
{
|
|
|
|
}
|
2019-09-13 16:07:31 +00:00
|
|
|
|
2019-12-11 06:35:14 +00:00
|
|
|
void GPU::SetDrawMode(u16 value)
|
2019-09-13 16:07:31 +00:00
|
|
|
{
|
2020-10-22 09:31:28 +00:00
|
|
|
GPUDrawModeReg new_mode_reg{static_cast<u16>(value & GPUDrawModeReg::MASK)};
|
2019-12-11 06:47:49 +00:00
|
|
|
if (!m_set_texture_disable_mask)
|
|
|
|
new_mode_reg.texture_disable = false;
|
|
|
|
|
|
|
|
if (new_mode_reg.bits == m_draw_mode.mode_reg.bits)
|
2019-09-13 16:07:31 +00:00
|
|
|
return;
|
|
|
|
|
2020-10-22 09:31:28 +00:00
|
|
|
if ((new_mode_reg.bits & GPUDrawModeReg::TEXTURE_PAGE_MASK) !=
|
|
|
|
(m_draw_mode.mode_reg.bits & GPUDrawModeReg::TEXTURE_PAGE_MASK))
|
2019-12-11 06:35:14 +00:00
|
|
|
{
|
2020-10-22 09:31:28 +00:00
|
|
|
m_draw_mode.texture_page_x = new_mode_reg.GetTexturePageBaseX();
|
|
|
|
m_draw_mode.texture_page_y = new_mode_reg.GetTexturePageBaseY();
|
2019-12-11 06:35:14 +00:00
|
|
|
m_draw_mode.texture_page_changed = true;
|
|
|
|
}
|
|
|
|
|
2019-12-11 06:47:49 +00:00
|
|
|
m_draw_mode.mode_reg.bits = new_mode_reg.bits;
|
2019-09-27 11:20:35 +00:00
|
|
|
|
2020-04-03 14:11:33 +00:00
|
|
|
if (m_GPUSTAT.draw_to_displayed_field != new_mode_reg.draw_to_displayed_field)
|
|
|
|
FlushRender();
|
|
|
|
|
2019-12-11 06:35:14 +00:00
|
|
|
// Bits 0..10 are returned in the GPU status register.
|
2020-12-05 09:31:43 +00:00
|
|
|
m_GPUSTAT.bits = (m_GPUSTAT.bits & ~(GPUDrawModeReg::GPUSTAT_MASK)) |
|
|
|
|
(ZeroExtend32(new_mode_reg.bits) & GPUDrawModeReg::GPUSTAT_MASK);
|
2019-12-11 06:35:14 +00:00
|
|
|
m_GPUSTAT.texture_disable = m_draw_mode.mode_reg.texture_disable;
|
2019-09-13 16:07:31 +00:00
|
|
|
}
|
|
|
|
|
2019-12-11 06:35:14 +00:00
|
|
|
void GPU::SetTexturePalette(u16 value)
|
2019-09-13 16:07:31 +00:00
|
|
|
{
|
2019-12-11 06:35:14 +00:00
|
|
|
value &= DrawMode::PALETTE_MASK;
|
|
|
|
if (m_draw_mode.palette_reg == value)
|
2019-09-13 16:07:31 +00:00
|
|
|
return;
|
|
|
|
|
2019-12-11 06:35:14 +00:00
|
|
|
m_draw_mode.texture_palette_x = ZeroExtend32(value & 0x3F) * 16;
|
|
|
|
m_draw_mode.texture_palette_y = ZeroExtend32(value >> 6);
|
|
|
|
m_draw_mode.palette_reg = value;
|
|
|
|
m_draw_mode.texture_page_changed = true;
|
2019-09-13 16:07:31 +00:00
|
|
|
}
|
2019-09-18 05:43:25 +00:00
|
|
|
|
2020-04-03 14:10:55 +00:00
|
|
|
void GPU::SetTextureWindow(u32 value)
|
2019-10-05 13:25:06 +00:00
|
|
|
{
|
2020-04-03 14:10:55 +00:00
|
|
|
value &= DrawMode::TEXTURE_WINDOW_MASK;
|
|
|
|
if (m_draw_mode.texture_window_value == value)
|
2019-10-05 13:25:06 +00:00
|
|
|
return;
|
|
|
|
|
2020-04-03 14:10:55 +00:00
|
|
|
FlushRender();
|
|
|
|
|
2020-09-20 11:28:45 +00:00
|
|
|
const u8 mask_x = Truncate8(value & UINT32_C(0x1F));
|
|
|
|
const u8 mask_y = Truncate8((value >> 5) & UINT32_C(0x1F));
|
|
|
|
const u8 offset_x = Truncate8((value >> 10) & UINT32_C(0x1F));
|
|
|
|
const u8 offset_y = Truncate8((value >> 15) & UINT32_C(0x1F));
|
|
|
|
Log_DebugPrintf("Set texture window %02X %02X %02X %02X", mask_x, mask_y, offset_x, offset_y);
|
|
|
|
|
2020-11-21 03:32:58 +00:00
|
|
|
m_draw_mode.texture_window.and_x = ~(mask_x * 8);
|
|
|
|
m_draw_mode.texture_window.and_y = ~(mask_y * 8);
|
|
|
|
m_draw_mode.texture_window.or_x = (offset_x & mask_x) * 8u;
|
|
|
|
m_draw_mode.texture_window.or_y = (offset_y & mask_y) * 8u;
|
2020-04-03 14:10:55 +00:00
|
|
|
m_draw_mode.texture_window_value = value;
|
|
|
|
m_draw_mode.texture_window_changed = true;
|
2019-10-05 13:25:06 +00:00
|
|
|
}
|
|
|
|
|
2023-08-31 13:37:17 +00:00
|
|
|
bool GPU::CompileDisplayPipeline()
|
2023-08-27 08:13:50 +00:00
|
|
|
{
|
2023-08-31 13:37:17 +00:00
|
|
|
GPUShaderGen shadergen(g_gpu_device->GetRenderAPI(), g_gpu_device->GetFeatures().dual_source_blend);
|
2023-08-27 08:13:50 +00:00
|
|
|
|
|
|
|
GPUPipeline::GraphicsConfig plconfig;
|
|
|
|
plconfig.layout = GPUPipeline::Layout::SingleTextureAndPushConstants;
|
|
|
|
plconfig.input_layout.vertex_stride = 0;
|
|
|
|
plconfig.primitive = GPUPipeline::Primitive::Triangles;
|
|
|
|
plconfig.rasterization = GPUPipeline::RasterizationState::GetNoCullState();
|
|
|
|
plconfig.depth = GPUPipeline::DepthState::GetNoTestsState();
|
|
|
|
plconfig.blend = GPUPipeline::BlendState::GetNoBlendingState();
|
|
|
|
plconfig.color_format = g_gpu_device->HasSurface() ? g_gpu_device->GetWindowFormat() : GPUTexture::Format::RGBA8;
|
|
|
|
plconfig.depth_format = GPUTexture::Format::Unknown;
|
|
|
|
plconfig.samples = 1;
|
|
|
|
plconfig.per_sample_shading = false;
|
|
|
|
|
2023-08-31 13:37:17 +00:00
|
|
|
std::string vs = shadergen.GenerateDisplayVertexShader();
|
|
|
|
std::string fs;
|
|
|
|
switch (g_settings.display_scaling)
|
|
|
|
{
|
|
|
|
case DisplayScalingMode::BilinearSharp:
|
|
|
|
fs = shadergen.GenerateDisplaySharpBilinearFragmentShader();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayScalingMode::BilinearSmooth:
|
2023-09-02 09:22:29 +00:00
|
|
|
fs = shadergen.GenerateDisplayFragmentShader(true);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayScalingMode::Nearest:
|
2023-08-31 13:37:17 +00:00
|
|
|
case DisplayScalingMode::NearestInteger:
|
|
|
|
default:
|
2023-09-02 09:22:29 +00:00
|
|
|
fs = shadergen.GenerateDisplayFragmentShader(false);
|
2023-08-31 13:37:17 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<GPUShader> vso = g_gpu_device->CreateShader(GPUShaderStage::Vertex, vs);
|
|
|
|
std::unique_ptr<GPUShader> fso = g_gpu_device->CreateShader(GPUShaderStage::Fragment, fs);
|
|
|
|
if (!vso || !fso)
|
2023-08-27 08:13:50 +00:00
|
|
|
return false;
|
2023-08-31 13:37:17 +00:00
|
|
|
GL_OBJECT_NAME(vso, "Display Vertex Shader");
|
2023-09-20 14:42:27 +00:00
|
|
|
GL_OBJECT_NAME_FMT(fso, "Display Fragment Shader [{}]", Settings::GetDisplayScalingName(g_settings.display_scaling));
|
2023-08-27 08:13:50 +00:00
|
|
|
|
2023-08-31 13:37:17 +00:00
|
|
|
plconfig.vertex_shader = vso.get();
|
|
|
|
plconfig.fragment_shader = fso.get();
|
2023-09-02 12:09:20 +00:00
|
|
|
plconfig.geometry_shader = nullptr;
|
2023-08-27 08:13:50 +00:00
|
|
|
if (!(m_display_pipeline = g_gpu_device->CreatePipeline(plconfig)))
|
|
|
|
return false;
|
2023-09-20 14:42:27 +00:00
|
|
|
GL_OBJECT_NAME_FMT(m_display_pipeline, "Display Pipeline [{}]",
|
|
|
|
Settings::GetDisplayScalingName(g_settings.display_scaling));
|
2023-08-27 08:13:50 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::ClearDisplayTexture()
|
|
|
|
{
|
|
|
|
m_display_texture = nullptr;
|
|
|
|
m_display_texture_view_x = 0;
|
|
|
|
m_display_texture_view_y = 0;
|
|
|
|
m_display_texture_view_width = 0;
|
|
|
|
m_display_texture_view_height = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::SetDisplayTexture(GPUTexture* texture, s32 view_x, s32 view_y, s32 view_width, s32 view_height)
|
|
|
|
{
|
|
|
|
DebugAssert(texture);
|
|
|
|
m_display_texture = texture;
|
|
|
|
m_display_texture_view_x = view_x;
|
|
|
|
m_display_texture_view_y = view_y;
|
|
|
|
m_display_texture_view_width = view_width;
|
|
|
|
m_display_texture_view_height = view_height;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::SetDisplayTextureRect(s32 view_x, s32 view_y, s32 view_width, s32 view_height)
|
|
|
|
{
|
|
|
|
m_display_texture_view_x = view_x;
|
|
|
|
m_display_texture_view_y = view_y;
|
|
|
|
m_display_texture_view_width = view_width;
|
|
|
|
m_display_texture_view_height = view_height;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GPU::SetDisplayParameters(s32 display_width, s32 display_height, s32 active_left, s32 active_top, s32 active_width,
|
|
|
|
s32 active_height, float display_aspect_ratio)
|
|
|
|
{
|
|
|
|
m_display_width = display_width;
|
|
|
|
m_display_height = display_height;
|
|
|
|
m_display_active_left = active_left;
|
|
|
|
m_display_active_top = active_top;
|
|
|
|
m_display_active_width = active_width;
|
|
|
|
m_display_active_height = active_height;
|
|
|
|
m_display_aspect_ratio = display_aspect_ratio;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GPU::PresentDisplay()
|
|
|
|
{
|
|
|
|
if (!HasDisplayTexture())
|
|
|
|
return g_gpu_device->BeginPresent(false);
|
|
|
|
|
|
|
|
const Common::Rectangle<s32> draw_rect =
|
|
|
|
CalculateDrawRect(g_gpu_device->GetWindowWidth(), g_gpu_device->GetWindowHeight());
|
2023-08-31 13:37:17 +00:00
|
|
|
return RenderDisplay(nullptr, draw_rect, true);
|
2023-08-27 08:13:50 +00:00
|
|
|
}
|
|
|
|
|
2023-08-31 13:37:17 +00:00
|
|
|
bool GPU::RenderDisplay(GPUFramebuffer* target, const Common::Rectangle<s32>& draw_rect, bool postfx)
|
2023-08-27 08:13:50 +00:00
|
|
|
{
|
2023-09-20 14:42:27 +00:00
|
|
|
GL_SCOPE_FMT("RenderDisplay: {}x{} at {},{}", draw_rect.left, draw_rect.top, draw_rect.GetWidth(), draw_rect.GetHeight());
|
2023-08-27 08:13:50 +00:00
|
|
|
|
|
|
|
if (m_display_texture)
|
|
|
|
m_display_texture->MakeReadyForSampling();
|
|
|
|
|
2023-08-31 13:37:17 +00:00
|
|
|
bool texture_filter_linear = false;
|
|
|
|
|
|
|
|
struct Uniforms
|
|
|
|
{
|
|
|
|
float src_rect[4];
|
|
|
|
float src_size[4];
|
2023-09-02 09:22:29 +00:00
|
|
|
float clamp_rect[4];
|
2023-08-31 13:37:17 +00:00
|
|
|
float params[4];
|
|
|
|
} uniforms;
|
|
|
|
std::memset(uniforms.params, 0, sizeof(uniforms.params));
|
|
|
|
|
|
|
|
switch (g_settings.display_scaling)
|
|
|
|
{
|
|
|
|
case DisplayScalingMode::Nearest:
|
|
|
|
case DisplayScalingMode::NearestInteger:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayScalingMode::BilinearSmooth:
|
|
|
|
texture_filter_linear = true;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayScalingMode::BilinearSharp:
|
|
|
|
{
|
|
|
|
texture_filter_linear = true;
|
|
|
|
uniforms.params[0] = std::max(
|
|
|
|
std::floor(static_cast<float>(draw_rect.GetWidth()) / static_cast<float>(m_display_texture_view_width)), 1.0f);
|
|
|
|
uniforms.params[1] = std::max(
|
|
|
|
std::floor(static_cast<float>(draw_rect.GetHeight()) / static_cast<float>(m_display_texture_view_height)),
|
|
|
|
1.0f);
|
|
|
|
uniforms.params[2] = 0.5f - 0.5f / uniforms.params[0];
|
|
|
|
uniforms.params[3] = 0.5f - 0.5f / uniforms.params[1];
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
UnreachableCode();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2023-08-27 08:13:50 +00:00
|
|
|
const GPUTexture::Format hdformat =
|
|
|
|
(target && target->GetRT()) ? target->GetRT()->GetFormat() : g_gpu_device->GetWindowFormat();
|
|
|
|
const u32 target_width = target ? target->GetWidth() : g_gpu_device->GetWindowWidth();
|
|
|
|
const u32 target_height = target ? target->GetHeight() : g_gpu_device->GetWindowHeight();
|
2023-08-27 12:48:40 +00:00
|
|
|
const bool really_postfx = (postfx && HasDisplayTexture() && PostProcessing::IsActive() &&
|
|
|
|
PostProcessing::CheckTargets(hdformat, target_width, target_height));
|
2023-08-27 08:13:50 +00:00
|
|
|
if (really_postfx)
|
|
|
|
{
|
2023-08-27 12:48:40 +00:00
|
|
|
g_gpu_device->ClearRenderTarget(PostProcessing::GetInputTexture(), 0);
|
|
|
|
g_gpu_device->SetFramebuffer(PostProcessing::GetInputFramebuffer());
|
2023-08-27 08:13:50 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (target)
|
|
|
|
g_gpu_device->SetFramebuffer(target);
|
|
|
|
else if (!g_gpu_device->BeginPresent(false))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!HasDisplayTexture())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
g_gpu_device->SetPipeline(m_display_pipeline.get());
|
2023-08-31 13:37:17 +00:00
|
|
|
g_gpu_device->SetTextureSampler(
|
|
|
|
0, m_display_texture, texture_filter_linear ? g_gpu_device->GetLinearSampler() : g_gpu_device->GetNearestSampler());
|
|
|
|
|
2023-09-02 09:22:29 +00:00
|
|
|
// For bilinear, clamp to 0.5/SIZE-0.5 to avoid bleeding from the adjacent texels in VRAM. This is because
|
|
|
|
// 1.0 in UV space is not the bottom-right texel, but a mix of the bottom-right and wrapped/next texel.
|
2023-08-31 13:37:17 +00:00
|
|
|
const float rcp_width = 1.0f / static_cast<float>(m_display_texture->GetWidth());
|
|
|
|
const float rcp_height = 1.0f / static_cast<float>(m_display_texture->GetHeight());
|
2023-09-02 09:22:29 +00:00
|
|
|
uniforms.src_rect[0] = static_cast<float>(m_display_texture_view_x) * rcp_width;
|
|
|
|
uniforms.src_rect[1] = static_cast<float>(m_display_texture_view_y) * rcp_height;
|
|
|
|
uniforms.src_rect[2] = static_cast<float>(m_display_texture_view_width) * rcp_width;
|
|
|
|
uniforms.src_rect[3] = static_cast<float>(m_display_texture_view_height) * rcp_height;
|
|
|
|
uniforms.clamp_rect[0] = (static_cast<float>(m_display_texture_view_x) + 0.5f) * rcp_width;
|
|
|
|
uniforms.clamp_rect[1] = (static_cast<float>(m_display_texture_view_y) + 0.5f) * rcp_height;
|
|
|
|
uniforms.clamp_rect[2] =
|
|
|
|
(static_cast<float>(m_display_texture_view_x + m_display_texture_view_width) - 0.5f) * rcp_width;
|
|
|
|
uniforms.clamp_rect[3] =
|
|
|
|
(static_cast<float>(m_display_texture_view_y + m_display_texture_view_height) - 0.5f) * rcp_height;
|
2023-08-31 13:37:17 +00:00
|
|
|
uniforms.src_size[0] = static_cast<float>(m_display_texture->GetWidth());
|
|
|
|
uniforms.src_size[1] = static_cast<float>(m_display_texture->GetHeight());
|
|
|
|
uniforms.src_size[2] = rcp_width;
|
|
|
|
uniforms.src_size[3] = rcp_height;
|
|
|
|
g_gpu_device->PushUniformBuffer(&uniforms, sizeof(uniforms));
|
2023-08-27 08:13:50 +00:00
|
|
|
|
|
|
|
g_gpu_device->SetViewportAndScissor(draw_rect.left, draw_rect.top, draw_rect.GetWidth(), draw_rect.GetHeight());
|
|
|
|
g_gpu_device->Draw(3, 0);
|
|
|
|
|
|
|
|
if (really_postfx)
|
|
|
|
{
|
2023-08-31 13:37:17 +00:00
|
|
|
return PostProcessing::Apply(target, draw_rect.left, draw_rect.top, draw_rect.GetWidth(), draw_rect.GetHeight(),
|
|
|
|
m_display_texture_view_width, m_display_texture_view_height);
|
2023-08-27 08:13:50 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Rectangle<float> GPU::CalculateDrawRect(s32 window_width, s32 window_height, float* out_left_padding,
|
|
|
|
float* out_top_padding, float* out_scale, float* out_x_scale,
|
|
|
|
bool apply_aspect_ratio /* = true */) const
|
|
|
|
{
|
|
|
|
const float window_ratio = static_cast<float>(window_width) / static_cast<float>(window_height);
|
|
|
|
const float x_scale =
|
|
|
|
apply_aspect_ratio ?
|
2023-08-31 13:37:17 +00:00
|
|
|
(m_display_aspect_ratio / (static_cast<float>(m_display_width) / static_cast<float>(m_display_height))) :
|
2023-08-27 08:13:50 +00:00
|
|
|
1.0f;
|
|
|
|
const float display_width = g_settings.display_stretch_vertically ? static_cast<float>(m_display_width) :
|
|
|
|
static_cast<float>(m_display_width) * x_scale;
|
|
|
|
const float display_height = g_settings.display_stretch_vertically ? static_cast<float>(m_display_height) / x_scale :
|
|
|
|
static_cast<float>(m_display_height);
|
|
|
|
const float active_left = g_settings.display_stretch_vertically ? static_cast<float>(m_display_active_left) :
|
|
|
|
static_cast<float>(m_display_active_left) * x_scale;
|
|
|
|
const float active_top = g_settings.display_stretch_vertically ? static_cast<float>(m_display_active_top) / x_scale :
|
|
|
|
static_cast<float>(m_display_active_top);
|
|
|
|
const float active_width = g_settings.display_stretch_vertically ?
|
|
|
|
static_cast<float>(m_display_active_width) :
|
|
|
|
static_cast<float>(m_display_active_width) * x_scale;
|
|
|
|
const float active_height = g_settings.display_stretch_vertically ?
|
|
|
|
static_cast<float>(m_display_active_height) / x_scale :
|
|
|
|
static_cast<float>(m_display_active_height);
|
|
|
|
if (out_x_scale)
|
|
|
|
*out_x_scale = x_scale;
|
|
|
|
|
|
|
|
// now fit it within the window
|
|
|
|
float scale;
|
|
|
|
if ((display_width / display_height) >= window_ratio)
|
|
|
|
{
|
|
|
|
// align in middle vertically
|
|
|
|
scale = static_cast<float>(window_width) / display_width;
|
2023-08-31 13:37:17 +00:00
|
|
|
if (g_settings.display_scaling == DisplayScalingMode::NearestInteger)
|
2023-08-27 08:13:50 +00:00
|
|
|
scale = std::max(std::floor(scale), 1.0f);
|
|
|
|
|
|
|
|
if (out_left_padding)
|
|
|
|
{
|
2023-08-31 13:37:17 +00:00
|
|
|
if (g_settings.display_scaling == DisplayScalingMode::NearestInteger)
|
2023-08-27 08:13:50 +00:00
|
|
|
*out_left_padding = std::max<float>((static_cast<float>(window_width) - display_width * scale) / 2.0f, 0.0f);
|
|
|
|
else
|
|
|
|
*out_left_padding = 0.0f;
|
|
|
|
}
|
|
|
|
if (out_top_padding)
|
|
|
|
{
|
|
|
|
switch (g_settings.display_alignment)
|
|
|
|
{
|
|
|
|
case DisplayAlignment::RightOrBottom:
|
|
|
|
*out_top_padding = std::max<float>(static_cast<float>(window_height) - (display_height * scale), 0.0f);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayAlignment::Center:
|
|
|
|
*out_top_padding =
|
|
|
|
std::max<float>((static_cast<float>(window_height) - (display_height * scale)) / 2.0f, 0.0f);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayAlignment::LeftOrTop:
|
|
|
|
default:
|
|
|
|
*out_top_padding = 0.0f;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// align in middle horizontally
|
|
|
|
scale = static_cast<float>(window_height) / display_height;
|
2023-08-31 13:37:17 +00:00
|
|
|
if (g_settings.display_scaling == DisplayScalingMode::NearestInteger)
|
2023-08-27 08:13:50 +00:00
|
|
|
scale = std::max(std::floor(scale), 1.0f);
|
|
|
|
|
|
|
|
if (out_left_padding)
|
|
|
|
{
|
|
|
|
switch (g_settings.display_alignment)
|
|
|
|
{
|
|
|
|
case DisplayAlignment::RightOrBottom:
|
|
|
|
*out_left_padding = std::max<float>(static_cast<float>(window_width) - (display_width * scale), 0.0f);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayAlignment::Center:
|
|
|
|
*out_left_padding =
|
|
|
|
std::max<float>((static_cast<float>(window_width) - (display_width * scale)) / 2.0f, 0.0f);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DisplayAlignment::LeftOrTop:
|
|
|
|
default:
|
|
|
|
*out_left_padding = 0.0f;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (out_top_padding)
|
|
|
|
{
|
2023-08-31 13:37:17 +00:00
|
|
|
if (g_settings.display_scaling == DisplayScalingMode::NearestInteger)
|
2023-08-27 08:13:50 +00:00
|
|
|
*out_top_padding = std::max<float>((static_cast<float>(window_height) - (display_height * scale)) / 2.0f, 0.0f);
|
|
|
|
else
|
|
|
|
*out_top_padding = 0.0f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (out_scale)
|
|
|
|
*out_scale = scale;
|
|
|
|
|
|
|
|
return Common::Rectangle<float>::FromExtents(active_left * scale, active_top * scale, active_width * scale,
|
|
|
|
active_height * scale);
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Rectangle<s32> GPU::CalculateDrawRect(s32 window_width, s32 window_height,
|
|
|
|
bool apply_aspect_ratio /* = true */) const
|
|
|
|
{
|
|
|
|
float left_padding, top_padding;
|
|
|
|
const Common::Rectangle<float> draw_rc =
|
|
|
|
CalculateDrawRect(window_width, window_height, &left_padding, &top_padding, nullptr, nullptr, apply_aspect_ratio);
|
|
|
|
|
|
|
|
// TODO: This should be a float rectangle. But because GL is lame, it only has integer viewports...
|
|
|
|
return Common::Rectangle<s32>::FromExtents(
|
|
|
|
static_cast<s32>(draw_rc.left + left_padding), static_cast<s32>(draw_rc.top + top_padding),
|
|
|
|
static_cast<s32>(draw_rc.GetWidth()), static_cast<s32>(draw_rc.GetHeight()));
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool CompressAndWriteTextureToFile(u32 width, u32 height, std::string filename, FileSystem::ManagedCFilePtr fp,
|
|
|
|
bool clear_alpha, bool flip_y, u32 resize_width, u32 resize_height,
|
|
|
|
std::vector<u32> texture_data, u32 texture_data_stride,
|
|
|
|
GPUTexture::Format texture_format)
|
|
|
|
{
|
|
|
|
|
|
|
|
const char* extension = std::strrchr(filename.c_str(), '.');
|
|
|
|
if (!extension)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Unable to determine file extension for '%s'", filename.c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!GPUTexture::ConvertTextureDataToRGBA8(width, height, texture_data, texture_data_stride, texture_format))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (clear_alpha)
|
|
|
|
{
|
|
|
|
for (u32& pixel : texture_data)
|
|
|
|
pixel |= 0xFF000000;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flip_y)
|
|
|
|
GPUTexture::FlipTextureDataRGBA8(width, height, texture_data, texture_data_stride);
|
|
|
|
|
|
|
|
if (resize_width > 0 && resize_height > 0 && (resize_width != width || resize_height != height))
|
|
|
|
{
|
|
|
|
std::vector<u32> resized_texture_data(resize_width * resize_height);
|
|
|
|
u32 resized_texture_stride = sizeof(u32) * resize_width;
|
|
|
|
if (!stbir_resize_uint8(reinterpret_cast<u8*>(texture_data.data()), width, height, texture_data_stride,
|
|
|
|
reinterpret_cast<u8*>(resized_texture_data.data()), resize_width, resize_height,
|
|
|
|
resized_texture_stride, 4))
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to resize texture data from %ux%u to %ux%u", width, height, resize_width, resize_height);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
width = resize_width;
|
|
|
|
height = resize_height;
|
|
|
|
texture_data = std::move(resized_texture_data);
|
|
|
|
texture_data_stride = resized_texture_stride;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto write_func = [](void* context, void* data, int size) {
|
|
|
|
std::fwrite(data, 1, size, static_cast<std::FILE*>(context));
|
|
|
|
};
|
|
|
|
|
|
|
|
bool result = false;
|
|
|
|
if (StringUtil::Strcasecmp(extension, ".png") == 0)
|
|
|
|
{
|
|
|
|
result =
|
|
|
|
(stbi_write_png_to_func(write_func, fp.get(), width, height, 4, texture_data.data(), texture_data_stride) != 0);
|
|
|
|
}
|
|
|
|
else if (StringUtil::Strcasecmp(extension, ".jpg") == 0)
|
|
|
|
{
|
|
|
|
result = (stbi_write_jpg_to_func(write_func, fp.get(), width, height, 4, texture_data.data(), 95) != 0);
|
|
|
|
}
|
|
|
|
else if (StringUtil::Strcasecmp(extension, ".tga") == 0)
|
|
|
|
{
|
|
|
|
result = (stbi_write_tga_to_func(write_func, fp.get(), width, height, 4, texture_data.data()) != 0);
|
|
|
|
}
|
|
|
|
else if (StringUtil::Strcasecmp(extension, ".bmp") == 0)
|
|
|
|
{
|
|
|
|
result = (stbi_write_bmp_to_func(write_func, fp.get(), width, height, 4, texture_data.data()) != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!result)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Unknown extension in filename '%s' or save error: '%s'", filename.c_str(), extension);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GPU::WriteDisplayTextureToFile(std::string filename, bool full_resolution /* = true */,
|
|
|
|
bool apply_aspect_ratio /* = true */, bool compress_on_thread /* = false */)
|
|
|
|
{
|
|
|
|
if (!m_display_texture)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
s32 resize_width = 0;
|
|
|
|
s32 resize_height = std::abs(m_display_texture_view_height);
|
|
|
|
if (apply_aspect_ratio)
|
|
|
|
{
|
|
|
|
const float ss_width_scale = static_cast<float>(m_display_active_width) / static_cast<float>(m_display_width);
|
|
|
|
const float ss_height_scale = static_cast<float>(m_display_active_height) / static_cast<float>(m_display_height);
|
|
|
|
const float ss_aspect_ratio = m_display_aspect_ratio * ss_width_scale / ss_height_scale;
|
|
|
|
resize_width = g_settings.display_stretch_vertically ?
|
|
|
|
m_display_texture_view_width :
|
|
|
|
static_cast<s32>(static_cast<float>(resize_height) * ss_aspect_ratio);
|
|
|
|
resize_height = g_settings.display_stretch_vertically ?
|
|
|
|
static_cast<s32>(static_cast<float>(resize_height) /
|
|
|
|
(m_display_aspect_ratio /
|
|
|
|
(static_cast<float>(m_display_width) / static_cast<float>(m_display_height)))) :
|
|
|
|
resize_height;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
resize_width = m_display_texture_view_width;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!full_resolution)
|
|
|
|
{
|
|
|
|
const s32 resolution_scale = std::abs(m_display_texture_view_height) / m_display_active_height;
|
|
|
|
resize_height /= resolution_scale;
|
|
|
|
resize_width /= resolution_scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (resize_width <= 0 || resize_height <= 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const u32 read_x = static_cast<u32>(m_display_texture_view_x);
|
|
|
|
const u32 read_y = static_cast<u32>(m_display_texture_view_y);
|
|
|
|
const u32 read_width = static_cast<u32>(m_display_texture_view_width);
|
|
|
|
const u32 read_height = static_cast<u32>(m_display_texture_view_height);
|
|
|
|
|
|
|
|
std::vector<u32> texture_data(read_width * read_height);
|
|
|
|
const u32 texture_data_stride =
|
|
|
|
Common::AlignUpPow2(GPUTexture::GetPixelSize(m_display_texture->GetFormat()) * read_width, 4);
|
|
|
|
if (!g_gpu_device->DownloadTexture(m_display_texture, read_x, read_y, read_width, read_height, texture_data.data(),
|
|
|
|
texture_data_stride))
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Texture download failed");
|
2023-09-03 03:13:17 +00:00
|
|
|
RestoreDeviceContext();
|
2023-08-27 08:13:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-09-03 03:13:17 +00:00
|
|
|
RestoreDeviceContext();
|
2023-08-27 08:13:50 +00:00
|
|
|
|
|
|
|
auto fp = FileSystem::OpenManagedCFile(filename.c_str(), "wb");
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Can't open file '%s': errno %d", filename.c_str(), errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr bool clear_alpha = true;
|
|
|
|
const bool flip_y = g_gpu_device->UsesLowerLeftOrigin();
|
|
|
|
|
|
|
|
if (!compress_on_thread)
|
|
|
|
{
|
|
|
|
return CompressAndWriteTextureToFile(read_width, read_height, std::move(filename), std::move(fp), clear_alpha,
|
|
|
|
flip_y, resize_width, resize_height, std::move(texture_data),
|
|
|
|
texture_data_stride, m_display_texture->GetFormat());
|
|
|
|
}
|
|
|
|
|
|
|
|
std::thread compress_thread(CompressAndWriteTextureToFile, read_width, read_height, std::move(filename),
|
|
|
|
std::move(fp), clear_alpha, flip_y, resize_width, resize_height, std::move(texture_data),
|
|
|
|
texture_data_stride, m_display_texture->GetFormat());
|
|
|
|
compress_thread.detach();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GPU::RenderScreenshotToBuffer(u32 width, u32 height, const Common::Rectangle<s32>& draw_rect, bool postfx,
|
|
|
|
std::vector<u32>* out_pixels, u32* out_stride, GPUTexture::Format* out_format)
|
|
|
|
{
|
|
|
|
const GPUTexture::Format hdformat =
|
|
|
|
g_gpu_device->HasSurface() ? g_gpu_device->GetWindowFormat() : GPUTexture::Format::RGBA8;
|
|
|
|
|
|
|
|
std::unique_ptr<GPUTexture> render_texture =
|
|
|
|
g_gpu_device->CreateTexture(width, height, 1, 1, 1, GPUTexture::Type::RenderTarget, hdformat);
|
|
|
|
if (!render_texture)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
std::unique_ptr<GPUFramebuffer> render_fb = g_gpu_device->CreateFramebuffer(render_texture.get());
|
|
|
|
if (!render_fb)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
g_gpu_device->ClearRenderTarget(render_texture.get(), 0);
|
|
|
|
|
2023-08-31 13:37:17 +00:00
|
|
|
// TODO: this should use copy shader instead.
|
|
|
|
RenderDisplay(render_fb.get(), draw_rect, postfx);
|
2023-08-27 08:13:50 +00:00
|
|
|
|
|
|
|
g_gpu_device->SetFramebuffer(nullptr);
|
|
|
|
|
|
|
|
const u32 stride = GPUTexture::GetPixelSize(hdformat) * width;
|
|
|
|
out_pixels->resize(width * height);
|
|
|
|
if (!g_gpu_device->DownloadTexture(render_texture.get(), 0, 0, width, height, out_pixels->data(), stride))
|
|
|
|
{
|
2023-09-03 03:13:17 +00:00
|
|
|
RestoreDeviceContext();
|
2023-08-27 08:13:50 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_stride = stride;
|
|
|
|
*out_format = hdformat;
|
2023-09-03 03:13:17 +00:00
|
|
|
RestoreDeviceContext();
|
2023-08-27 08:13:50 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GPU::RenderScreenshotToFile(std::string filename, bool internal_resolution /* = false */,
|
|
|
|
bool compress_on_thread /* = false */)
|
|
|
|
{
|
|
|
|
u32 width = g_gpu_device->GetWindowWidth();
|
|
|
|
u32 height = g_gpu_device->GetWindowHeight();
|
2023-09-02 09:29:58 +00:00
|
|
|
Common::Rectangle<s32> draw_rect = CalculateDrawRect(width, height);
|
2023-08-27 08:13:50 +00:00
|
|
|
|
|
|
|
if (internal_resolution && m_display_texture_view_width != 0 && m_display_texture_view_height != 0)
|
|
|
|
{
|
2023-09-02 09:29:58 +00:00
|
|
|
const u32 draw_width = static_cast<u32>(draw_rect.GetWidth());
|
|
|
|
const u32 draw_height = static_cast<u32>(draw_rect.GetHeight());
|
|
|
|
|
2023-08-27 08:13:50 +00:00
|
|
|
// If internal res, scale the computed draw rectangle to the internal res.
|
|
|
|
// We re-use the draw rect because it's already been AR corrected.
|
|
|
|
const float sar =
|
|
|
|
static_cast<float>(m_display_texture_view_width) / static_cast<float>(m_display_texture_view_height);
|
|
|
|
const float dar = static_cast<float>(draw_width) / static_cast<float>(draw_height);
|
|
|
|
if (sar >= dar)
|
|
|
|
{
|
|
|
|
// stretch height, preserve width
|
|
|
|
const float scale = static_cast<float>(m_display_texture_view_width) / static_cast<float>(draw_width);
|
|
|
|
width = m_display_texture_view_width;
|
|
|
|
height = static_cast<u32>(std::round(static_cast<float>(draw_height) * scale));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// stretch width, preserve height
|
|
|
|
const float scale = static_cast<float>(m_display_texture_view_height) / static_cast<float>(draw_height);
|
|
|
|
width = static_cast<u32>(std::round(static_cast<float>(draw_width) * scale));
|
|
|
|
height = m_display_texture_view_height;
|
|
|
|
}
|
|
|
|
|
|
|
|
// DX11 won't go past 16K texture size.
|
|
|
|
constexpr u32 MAX_TEXTURE_SIZE = 16384;
|
|
|
|
if (width > MAX_TEXTURE_SIZE)
|
|
|
|
{
|
|
|
|
height = static_cast<u32>(static_cast<float>(height) /
|
|
|
|
(static_cast<float>(width) / static_cast<float>(MAX_TEXTURE_SIZE)));
|
|
|
|
width = MAX_TEXTURE_SIZE;
|
|
|
|
}
|
|
|
|
if (height > MAX_TEXTURE_SIZE)
|
|
|
|
{
|
|
|
|
height = MAX_TEXTURE_SIZE;
|
|
|
|
width = static_cast<u32>(static_cast<float>(width) /
|
|
|
|
(static_cast<float>(height) / static_cast<float>(MAX_TEXTURE_SIZE)));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove padding, it's not part of the framebuffer.
|
2023-09-02 09:29:58 +00:00
|
|
|
draw_rect.Set(0, 0, static_cast<s32>(width), static_cast<s32>(height));
|
2023-08-27 08:13:50 +00:00
|
|
|
}
|
|
|
|
if (width == 0 || height == 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
std::vector<u32> pixels;
|
|
|
|
u32 pixels_stride;
|
|
|
|
GPUTexture::Format pixels_format;
|
2023-09-02 09:29:58 +00:00
|
|
|
if (!RenderScreenshotToBuffer(width, height, draw_rect, !internal_resolution, &pixels, &pixels_stride,
|
|
|
|
&pixels_format))
|
2023-08-27 08:13:50 +00:00
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to render %ux%u screenshot", width, height);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto fp = FileSystem::OpenManagedCFile(filename.c_str(), "wb");
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Can't open file '%s': errno %d", filename.c_str(), errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!compress_on_thread)
|
|
|
|
{
|
|
|
|
return CompressAndWriteTextureToFile(width, height, std::move(filename), std::move(fp), true,
|
|
|
|
g_gpu_device->UsesLowerLeftOrigin(), width, height, std::move(pixels),
|
|
|
|
pixels_stride, pixels_format);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::thread compress_thread(CompressAndWriteTextureToFile, width, height, std::move(filename), std::move(fp), true,
|
|
|
|
g_gpu_device->UsesLowerLeftOrigin(), width, height, std::move(pixels), pixels_stride,
|
|
|
|
pixels_format);
|
|
|
|
compress_thread.detach();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-01-13 09:24:41 +00:00
|
|
|
bool GPU::DumpVRAMToFile(const char* filename)
|
|
|
|
{
|
|
|
|
ReadVRAM(0, 0, VRAM_WIDTH, VRAM_HEIGHT);
|
|
|
|
|
|
|
|
const char* extension = std::strrchr(filename, '.');
|
|
|
|
if (extension && StringUtil::Strcasecmp(extension, ".png") == 0)
|
|
|
|
{
|
|
|
|
return DumpVRAMToFile(filename, VRAM_WIDTH, VRAM_HEIGHT, sizeof(u16) * VRAM_WIDTH, m_vram_ptr, true);
|
|
|
|
}
|
|
|
|
else if (extension && StringUtil::Strcasecmp(extension, ".bin") == 0)
|
|
|
|
{
|
|
|
|
return FileSystem::WriteBinaryFile(filename, m_vram_ptr, VRAM_WIDTH * VRAM_HEIGHT * sizeof(u16));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Unknown extension: '%s'", filename);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-18 05:43:25 +00:00
|
|
|
bool GPU::DumpVRAMToFile(const char* filename, u32 width, u32 height, u32 stride, const void* buffer, bool remove_alpha)
|
|
|
|
{
|
2020-09-14 19:27:22 +00:00
|
|
|
auto fp = FileSystem::OpenManagedCFile(filename, "wb");
|
|
|
|
if (!fp)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Can't open file '%s'", filename);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto rgba8_buf = std::make_unique<u32[]>(width * height);
|
2019-09-18 05:43:25 +00:00
|
|
|
|
|
|
|
const char* ptr_in = static_cast<const char*>(buffer);
|
2020-09-14 19:27:22 +00:00
|
|
|
u32* ptr_out = rgba8_buf.get();
|
2019-09-18 05:43:25 +00:00
|
|
|
for (u32 row = 0; row < height; row++)
|
|
|
|
{
|
|
|
|
const char* row_ptr_in = ptr_in;
|
|
|
|
|
|
|
|
for (u32 col = 0; col < width; col++)
|
|
|
|
{
|
|
|
|
u16 src_col;
|
|
|
|
std::memcpy(&src_col, row_ptr_in, sizeof(u16));
|
|
|
|
row_ptr_in += sizeof(u16);
|
2021-03-18 02:55:02 +00:00
|
|
|
*(ptr_out++) = VRAMRGBA5551ToRGBA8888(remove_alpha ? (src_col | u16(0x8000)) : src_col);
|
2019-09-18 05:43:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ptr_in += stride;
|
|
|
|
}
|
2020-09-14 19:27:22 +00:00
|
|
|
|
|
|
|
const auto write_func = [](void* context, void* data, int size) {
|
|
|
|
std::fwrite(data, 1, size, static_cast<std::FILE*>(context));
|
|
|
|
};
|
|
|
|
return (stbi_write_png_to_func(write_func, fp.get(), width, height, 4, rgba8_buf.get(), sizeof(u32) * width) != 0);
|
2019-09-18 05:43:25 +00:00
|
|
|
}
|
2019-10-12 12:15:38 +00:00
|
|
|
|
|
|
|
void GPU::DrawDebugStateWindow()
|
|
|
|
{
|
2022-07-11 13:03:29 +00:00
|
|
|
const float framebuffer_scale = Host::GetOSDScale();
|
2020-02-28 06:59:51 +00:00
|
|
|
|
|
|
|
ImGui::SetNextWindowSize(ImVec2(450.0f * framebuffer_scale, 550.0f * framebuffer_scale), ImGuiCond_FirstUseEver);
|
2021-02-11 14:47:32 +00:00
|
|
|
if (!ImGui::Begin("GPU", nullptr))
|
2019-10-12 12:15:38 +00:00
|
|
|
{
|
|
|
|
ImGui::End();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-05 09:44:17 +00:00
|
|
|
const bool is_idle_frame = m_stats.num_polygons == 0;
|
|
|
|
if (!is_idle_frame)
|
|
|
|
{
|
|
|
|
m_last_stats = m_stats;
|
|
|
|
m_stats = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::CollapsingHeader("Statistics", ImGuiTreeNodeFlags_DefaultOpen))
|
|
|
|
{
|
|
|
|
const Stats& stats = m_last_stats;
|
|
|
|
|
|
|
|
ImGui::Columns(2);
|
2020-02-28 06:59:51 +00:00
|
|
|
ImGui::SetColumnWidth(0, 200.0f * framebuffer_scale);
|
2019-11-05 09:44:17 +00:00
|
|
|
|
|
|
|
ImGui::TextUnformatted("Idle Frame: ");
|
|
|
|
ImGui::NextColumn();
|
|
|
|
ImGui::Text("%s", is_idle_frame ? "Yes" : "No");
|
|
|
|
ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::TextUnformatted("VRAM Reads: ");
|
|
|
|
ImGui::NextColumn();
|
|
|
|
ImGui::Text("%u", stats.num_vram_reads);
|
|
|
|
ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::TextUnformatted("VRAM Fills: ");
|
|
|
|
ImGui::NextColumn();
|
|
|
|
ImGui::Text("%u", stats.num_vram_fills);
|
|
|
|
ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::TextUnformatted("VRAM Writes: ");
|
|
|
|
ImGui::NextColumn();
|
|
|
|
ImGui::Text("%u", stats.num_vram_writes);
|
|
|
|
ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::TextUnformatted("VRAM Copies: ");
|
|
|
|
ImGui::NextColumn();
|
|
|
|
ImGui::Text("%u", stats.num_vram_copies);
|
|
|
|
ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::TextUnformatted("Vertices Processed: ");
|
|
|
|
ImGui::NextColumn();
|
|
|
|
ImGui::Text("%u", stats.num_vertices);
|
|
|
|
ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::TextUnformatted("Polygons Drawn: ");
|
|
|
|
ImGui::NextColumn();
|
|
|
|
ImGui::Text("%u", stats.num_polygons);
|
|
|
|
ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::Columns(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
DrawRendererStats(is_idle_frame);
|
|
|
|
|
2020-05-17 04:11:42 +00:00
|
|
|
if (ImGui::CollapsingHeader("GPU", ImGuiTreeNodeFlags_DefaultOpen))
|
|
|
|
{
|
|
|
|
static constexpr std::array<const char*, 5> state_strings = {
|
|
|
|
{"Idle", "Reading VRAM", "Writing VRAM", "Drawing Polyline"}};
|
|
|
|
|
|
|
|
ImGui::Text("State: %s", state_strings[static_cast<u8>(m_blitter_state)]);
|
|
|
|
ImGui::Text("Dither: %s", m_GPUSTAT.dither_enable ? "Enabled" : "Disabled");
|
|
|
|
ImGui::Text("Draw To Displayed Field: %s", m_GPUSTAT.draw_to_displayed_field ? "Enabled" : "Disabled");
|
|
|
|
ImGui::Text("Draw Set Mask Bit: %s", m_GPUSTAT.set_mask_while_drawing ? "Yes" : "No");
|
|
|
|
ImGui::Text("Draw To Masked Pixels: %s", m_GPUSTAT.check_mask_before_draw ? "Yes" : "No");
|
|
|
|
ImGui::Text("Reverse Flag: %s", m_GPUSTAT.reverse_flag ? "Yes" : "No");
|
|
|
|
ImGui::Text("Texture Disable: %s", m_GPUSTAT.texture_disable ? "Yes" : "No");
|
|
|
|
ImGui::Text("PAL Mode: %s", m_GPUSTAT.pal_mode ? "Yes" : "No");
|
|
|
|
ImGui::Text("Interrupt Request: %s", m_GPUSTAT.interrupt_request ? "Yes" : "No");
|
|
|
|
ImGui::Text("DMA Request: %s", m_GPUSTAT.dma_data_request ? "Yes" : "No");
|
|
|
|
}
|
|
|
|
|
2019-10-12 12:15:38 +00:00
|
|
|
if (ImGui::CollapsingHeader("CRTC", ImGuiTreeNodeFlags_DefaultOpen))
|
|
|
|
{
|
|
|
|
const auto& cs = m_crtc_state;
|
2020-06-07 07:36:45 +00:00
|
|
|
ImGui::Text("Clock: %s", (m_console_is_pal ? (m_GPUSTAT.pal_mode ? "PAL-on-PAL" : "NTSC-on-PAL") :
|
|
|
|
(m_GPUSTAT.pal_mode ? "PAL-on-NTSC" : "NTSC-on-NTSC")));
|
|
|
|
ImGui::Text("Horizontal Frequency: %.3f KHz", ComputeHorizontalFrequency() / 1000.0f);
|
|
|
|
ImGui::Text("Vertical Frequency: %.3f Hz", ComputeVerticalFrequency());
|
2019-10-12 12:15:38 +00:00
|
|
|
ImGui::Text("Dot Clock Divider: %u", cs.dot_clock_divider);
|
|
|
|
ImGui::Text("Vertical Interlace: %s (%s field)", m_GPUSTAT.vertical_interlace ? "Yes" : "No",
|
2020-10-21 15:31:15 +00:00
|
|
|
cs.interlaced_field ? "odd" : "even");
|
|
|
|
ImGui::Text("Current Scanline: %u (tick %u)", cs.current_scanline, cs.current_tick_in_scanline);
|
2019-10-28 07:43:34 +00:00
|
|
|
ImGui::Text("Display Disable: %s", m_GPUSTAT.display_disable ? "Yes" : "No");
|
2020-10-21 15:31:15 +00:00
|
|
|
ImGui::Text("Displaying Odd Lines: %s", cs.active_line_lsb ? "Yes" : "No");
|
2019-10-12 12:15:38 +00:00
|
|
|
ImGui::Text("Color Depth: %u-bit", m_GPUSTAT.display_area_color_depth_24 ? 24 : 15);
|
2020-10-21 15:31:15 +00:00
|
|
|
ImGui::Text("Start Offset in VRAM: (%u, %u)", cs.regs.X.GetValue(), cs.regs.Y.GetValue());
|
2019-11-11 13:36:11 +00:00
|
|
|
ImGui::Text("Display Total: %u (%u) horizontal, %u vertical", cs.horizontal_total,
|
|
|
|
cs.horizontal_total / cs.dot_clock_divider, cs.vertical_total);
|
2020-10-21 15:31:15 +00:00
|
|
|
ImGui::Text("Configured Display Range: %u-%u (%u-%u), %u-%u", cs.regs.X1.GetValue(), cs.regs.X2.GetValue(),
|
2019-11-11 13:36:11 +00:00
|
|
|
cs.regs.X1.GetValue() / cs.dot_clock_divider, cs.regs.X2.GetValue() / cs.dot_clock_divider,
|
|
|
|
cs.regs.Y1.GetValue(), cs.regs.Y2.GetValue());
|
2020-10-21 15:31:15 +00:00
|
|
|
ImGui::Text("Output Display Range: %u-%u (%u-%u), %u-%u", cs.horizontal_display_start, cs.horizontal_display_end,
|
|
|
|
cs.horizontal_display_start / cs.dot_clock_divider, cs.horizontal_display_end / cs.dot_clock_divider,
|
|
|
|
cs.vertical_display_start, cs.vertical_display_end);
|
|
|
|
ImGui::Text("Cropping: %s", Settings::GetDisplayCropModeName(g_settings.display_crop_mode));
|
2020-12-12 07:59:09 +00:00
|
|
|
ImGui::Text("Visible Display Range: %u-%u (%u-%u), %u-%u", cs.horizontal_visible_start, cs.horizontal_visible_end,
|
|
|
|
cs.horizontal_visible_start / cs.dot_clock_divider, cs.horizontal_visible_end / cs.dot_clock_divider,
|
|
|
|
cs.vertical_visible_start, cs.vertical_visible_end);
|
2020-10-21 15:31:15 +00:00
|
|
|
ImGui::Text("Display Resolution: %ux%u", cs.display_width, cs.display_height);
|
|
|
|
ImGui::Text("Display Origin: %u, %u", cs.display_origin_left, cs.display_origin_top);
|
|
|
|
ImGui::Text("Displayed/Visible VRAM Portion: %ux%u @ (%u, %u)", cs.display_vram_width, cs.display_vram_height,
|
|
|
|
cs.display_vram_left, cs.display_vram_top);
|
2020-07-04 05:11:05 +00:00
|
|
|
ImGui::Text("Padding: Left=%d, Top=%d, Right=%d, Bottom=%d", cs.display_origin_left, cs.display_origin_top,
|
2020-03-28 15:14:37 +00:00
|
|
|
cs.display_width - cs.display_vram_width - cs.display_origin_left,
|
|
|
|
cs.display_height - cs.display_vram_height - cs.display_origin_top);
|
2019-10-12 12:15:38 +00:00
|
|
|
}
|
|
|
|
|
2019-10-26 02:55:56 +00:00
|
|
|
ImGui::End();
|
2019-10-22 13:07:51 +00:00
|
|
|
}
|
2019-10-26 02:55:56 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
void GPU::DrawRendererStats(bool is_idle_frame)
|
|
|
|
{
|
|
|
|
}
|