dingusppc/devices/video/videoctrl.h
Mihai Parparita dab9334c3a Pass through draw_fb optimization metadata to Display class
The JS implementation does content hashing to not blit unchanged
framebuffer contents (see mihaip/dingusppc@171ff2d407).
However, that is not necessary for the ATI adapters that already track
this and only set draw_fb if the framebuffer has actually changed.

Pass through a fb_known_to_be_changed for these cases, and also add an
optional update_skipped method (since the JS still wants to know when
the last logical screen update was).
2024-07-27 11:38:59 -07:00

120 lines
4.3 KiB
C++

/*
DingusPPC - The Experimental PowerPC Macintosh emulator
Copyright (C) 2018-23 divingkatae and maximum
(theweirdo) spatium
(Contact divingkatae#1017 or powermax#2286 on Discord for more info)
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
/** @file Video Conroller base class definitions. */
#ifndef VIDEO_CTRL_H
#define VIDEO_CTRL_H
#include <devices/common/hwinterrupt.h>
#include <devices/video/display.h>
#include <cinttypes>
#include <functional>
class WindowEvent;
class VideoCtrlBase {
public:
VideoCtrlBase(int width = 640, int height = 480);
~VideoCtrlBase();
void handle_events(const WindowEvent& wnd_event);
void create_display_window(int width, int height);
void blank_display();
void update_screen(void);
void start_refresh_task();
void stop_refresh_task();
void get_palette_color(uint8_t index, uint8_t& r, uint8_t& g, uint8_t& b,
uint8_t& a);
void set_palette_color(uint8_t index, uint8_t r, uint8_t g, uint8_t b, uint8_t a);
// HW cursor support
void setup_hw_cursor(int cursor_width=64, int cur_height=64);
virtual void draw_hw_cursor(uint8_t *dst_buf, int dst_pitch) {};
virtual void get_cursor_position(int& x, int& y) { x = 0; y = 0; };
// converters for various framebuffer pixel depths
virtual void convert_frame_1bpp_indexed(uint8_t *dst_buf, int dst_pitch);
virtual void convert_frame_2bpp_indexed(uint8_t *dst_buf, int dst_pitch);
virtual void convert_frame_4bpp_indexed(uint8_t *dst_buf, int dst_pitch);
virtual void convert_frame_8bpp_indexed(uint8_t *dst_buf, int dst_pitch);
#if 0
virtual void convert_frame_8bpp_32LE_indexed(uint8_t *dst_buf, int dst_pitch);
#endif
virtual void convert_frame_8bpp(uint8_t *dst_buf, int dst_pitch);
virtual void convert_frame_15bpp(uint8_t *dst_buf, int dst_pitch);
virtual void convert_frame_15bpp_BE(uint8_t *dst_buf, int dst_pitch);
virtual void convert_frame_16bpp(uint8_t *dst_buf, int dst_pitch);
virtual void convert_frame_24bpp(uint8_t *dst_buf, int dst_pitch);
virtual void convert_frame_32bpp(uint8_t *dst_buf, int dst_pitch);
virtual void convert_frame_32bpp_BE(uint8_t *dst_buf, int dst_pitch);
protected:
// CRT controller parameters
bool crtc_on = false;
bool blank_on = true;
bool cursor_on = false;
bool cursor_dirty = false;
int active_width; // width of the visible display area
int active_height; // height of the visible display area
int hori_total = 0;
int vert_total = 0;
int hori_blank = 0;
int vert_blank = 0;
int pixel_depth;
int pixel_format;
float pixel_clock;
float refresh_rate;
// Implementations may choose to track framebuffer writes and set draw_fb
// to false if updates can be skipped. If the do this, they should set
// draw_fb_is_dynamic at initialization time.
bool draw_fb = true;
bool draw_fb_is_dynamic = false;
uint32_t palette[256] = {0}; // internal DAC palette in RGBA format
// Framebuffer parameters
uint8_t* fb_ptr = nullptr;
int fb_pitch = 0;
uint32_t refresh_task_id = 0;
uint32_t vbl_end_task_id = 0;
// interrupt suff
InterruptCtrl* int_ctrl = nullptr;
uint32_t irq_id = 0;
std::function<void(uint8_t irq_line_state)> vbl_cb = [this](uint8_t irq_line_state) {
if (this->int_ctrl)
this->int_ctrl->ack_int(this->irq_id, irq_line_state);
};
std::function<void(uint8_t *dst_buf, int dst_pitch)> convert_fb_cb = nullptr;
std::function<void(uint8_t *dst_buf, int dst_pitch)> cursor_ovl_cb = nullptr;
private:
Display display;
};
#endif // VIDEO_CTRL_H