2021-12-07 22:48:33 +01:00
|
|
|
/*
|
|
|
|
DingusPPC - The Experimental PowerPC Macintosh emulator
|
2023-04-01 12:08:25 +02:00
|
|
|
Copyright (C) 2018-23 divingkatae and maximum
|
2021-12-07 22:48:33 +01:00
|
|
|
(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 PDM on-board video emulation. */
|
|
|
|
|
|
|
|
#include <cpu/ppc/ppcmmu.h>
|
|
|
|
#include <devices/memctrl/hmc.h>
|
|
|
|
#include <devices/video/pdmonboard.h>
|
|
|
|
#include <devices/video/videoctrl.h>
|
|
|
|
#include <machines/machinebase.h>
|
2023-12-02 20:41:30 -08:00
|
|
|
#include <memaccess.h>
|
2021-12-07 22:48:33 +01:00
|
|
|
#include <loguru.hpp>
|
|
|
|
|
|
|
|
#include <cinttypes>
|
|
|
|
|
|
|
|
PdmOnboardVideo::PdmOnboardVideo()
|
|
|
|
: VideoCtrlBase()
|
|
|
|
{
|
2022-01-10 17:09:21 +01:00
|
|
|
this->video_mode = PDM_VMODE_OFF;
|
|
|
|
this->blanking = 0x80;
|
|
|
|
this->crtc_on = false;
|
|
|
|
this->vdac_mode = 8;
|
|
|
|
this->pixel_depth = 0;
|
2021-12-07 22:48:33 +01:00
|
|
|
|
|
|
|
// get pointer to the Highspeed Memory controller
|
|
|
|
this->hmc_obj = dynamic_cast<HMC*>(gMachineObj->get_comp_by_name("HMC"));
|
|
|
|
}
|
|
|
|
|
|
|
|
void PdmOnboardVideo::set_video_mode(uint8_t new_mode)
|
|
|
|
{
|
2023-04-01 17:38:11 +02:00
|
|
|
if (this->blanking != (new_mode & 0x80)) {
|
|
|
|
if (new_mode & 0x80) {
|
|
|
|
this->disable_video_internal();
|
|
|
|
} else {
|
|
|
|
this->blank_on = false;
|
|
|
|
this->blanking = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-07 22:48:33 +01:00
|
|
|
switch(new_mode & 0x7F) {
|
|
|
|
case PdmVideoMode::Portrait:
|
|
|
|
case PdmVideoMode::Rgb12in:
|
|
|
|
case PdmVideoMode::Rgb13in:
|
|
|
|
case PdmVideoMode::Rgb16in:
|
|
|
|
case PdmVideoMode::VGA:
|
|
|
|
if (this->video_mode != (new_mode & 0x1F)) {
|
|
|
|
this->video_mode = new_mode & 0x1F;
|
|
|
|
LOG_F(INFO, "PDM-Video: video mode set to 0x%X", this->video_mode);
|
|
|
|
}
|
2023-04-01 17:38:11 +02:00
|
|
|
if (!this->blanking) {
|
|
|
|
this->enable_video_internal();
|
2021-12-07 22:48:33 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
LOG_F(9, "PDM-Video: video disabled, new mode = 0x%X", new_mode & 0x1F);
|
|
|
|
this->video_mode = new_mode & 0x1F;
|
2023-04-01 17:38:11 +02:00
|
|
|
this->disable_video_internal();
|
2021-12-07 22:48:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PdmOnboardVideo::set_pixel_depth(uint8_t depth)
|
|
|
|
{
|
|
|
|
static uint8_t pix_depths[8] = {1, 2, 4, 8, 16, 0xFF, 0xFF, 0xFF};
|
|
|
|
|
2022-01-10 17:09:21 +01:00
|
|
|
uint8_t new_pix_depth = pix_depths[depth];
|
|
|
|
if (new_pix_depth == 0xFF) {
|
2021-12-07 22:48:33 +01:00
|
|
|
ABORT_F("PDM-Video: invalid pixel depth code %d specified!", depth);
|
|
|
|
}
|
2022-01-10 17:09:21 +01:00
|
|
|
|
|
|
|
if (new_pix_depth != this->pixel_depth) {
|
|
|
|
this->pixel_depth = new_pix_depth;
|
|
|
|
this->set_depth_internal(this->active_width);
|
|
|
|
LOG_F(INFO, "PDM-Video: pixel depth changed to %dbpp", new_pix_depth);
|
|
|
|
}
|
2021-12-07 22:48:33 +01:00
|
|
|
}
|
|
|
|
|
2022-01-10 17:09:21 +01:00
|
|
|
void PdmOnboardVideo::set_vdac_config(uint8_t mode)
|
2021-12-07 22:48:33 +01:00
|
|
|
{
|
2022-01-10 17:09:21 +01:00
|
|
|
this->vdac_mode = mode;
|
|
|
|
if ((mode & 0xF8) != 8) {
|
|
|
|
LOG_F(WARNING, "Ariel control changed to unknown value 0x%X", mode);
|
2021-12-07 22:48:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PdmOnboardVideo::set_clut_index(uint8_t index)
|
|
|
|
{
|
|
|
|
this->clut_index = index;
|
|
|
|
this->comp_index = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PdmOnboardVideo::set_clut_color(uint8_t color)
|
|
|
|
{
|
|
|
|
this->clut_color[this->comp_index++] = color;
|
|
|
|
if (this->comp_index >= 3) {
|
2023-04-01 12:08:25 +02:00
|
|
|
this->set_palette_color(this->clut_index, clut_color[0],
|
|
|
|
clut_color[1], clut_color[2], 0xFF);
|
|
|
|
this->clut_index++;
|
|
|
|
this->comp_index = 0;
|
2021-12-07 22:48:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-02 20:41:30 -08:00
|
|
|
void PdmOnboardVideo::set_depth_internal(int width)
|
2022-01-10 17:09:21 +01:00
|
|
|
{
|
|
|
|
switch (this->pixel_depth) {
|
|
|
|
case 1:
|
|
|
|
this->convert_fb_cb = [this](uint8_t* dst_buf, int dst_pitch) {
|
2023-09-18 17:34:23 -07:00
|
|
|
this->convert_frame_1bpp_indexed(dst_buf, dst_pitch);
|
2022-01-10 17:09:21 +01:00
|
|
|
};
|
2023-12-02 20:41:30 -08:00
|
|
|
this->fb_pitch = width >> 3; // one byte contains 8 pixels
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
this->convert_fb_cb = [this](uint8_t* dst_buf, int dst_pitch) {
|
|
|
|
this->convert_frame_2bpp_indexed(dst_buf, dst_pitch);
|
|
|
|
};
|
|
|
|
this->fb_pitch = width >> 2; // one byte contains 4 pixels
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
this->convert_fb_cb = [this](uint8_t* dst_buf, int dst_pitch) {
|
|
|
|
this->convert_frame_4bpp_indexed(dst_buf, dst_pitch);
|
|
|
|
};
|
|
|
|
this->fb_pitch = width >> 1; // one byte contains 2 pixels
|
2022-01-10 17:09:21 +01:00
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
this->convert_fb_cb = [this](uint8_t* dst_buf, int dst_pitch) {
|
2023-09-18 17:34:23 -07:00
|
|
|
this->convert_frame_8bpp_indexed(dst_buf, dst_pitch);
|
2022-01-10 17:09:21 +01:00
|
|
|
};
|
2023-12-02 20:41:30 -08:00
|
|
|
this->fb_pitch = width; // one byte contains 1 pixel
|
|
|
|
break;
|
|
|
|
case 16:
|
|
|
|
this->convert_fb_cb = [this](uint8_t* dst_buf, int dst_pitch) {
|
|
|
|
this->convert_frame_15bpp_BE(dst_buf, dst_pitch);
|
|
|
|
};
|
|
|
|
this->fb_pitch = width << 1; // 1 pixel is 2 bytes
|
2022-01-10 17:09:21 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ABORT_F("PDM-Video: pixel depth %d not implemented yet!", this->pixel_depth);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-07 22:48:33 +01:00
|
|
|
void PdmOnboardVideo::enable_video_internal()
|
|
|
|
{
|
2022-01-10 17:09:21 +01:00
|
|
|
int new_width, new_height, hori_blank, vert_blank;
|
2021-12-07 22:48:33 +01:00
|
|
|
|
|
|
|
// ensure all video parameters contain safe values
|
|
|
|
switch(this->video_mode) {
|
|
|
|
case PdmVideoMode::Portrait:
|
|
|
|
case PdmVideoMode::Rgb16in:
|
|
|
|
case PdmVideoMode::VGA:
|
|
|
|
if (this->pixel_depth > 8) {
|
|
|
|
ABORT_F("PDM-Video: no 16bpp support in mode %d!", this->video_mode);
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// set video mode parameters
|
|
|
|
switch(this->video_mode) {
|
|
|
|
case PdmVideoMode::Portrait:
|
|
|
|
new_width = 640;
|
|
|
|
new_height = 870;
|
2022-01-10 17:09:21 +01:00
|
|
|
hori_blank = 192;
|
|
|
|
vert_blank = 48;
|
2021-12-07 22:48:33 +01:00
|
|
|
this->pixel_clock = 57283200;
|
|
|
|
break;
|
|
|
|
case PdmVideoMode::Rgb12in:
|
|
|
|
new_width = 512;
|
|
|
|
new_height = 384;
|
2022-01-10 17:09:21 +01:00
|
|
|
hori_blank = 128;
|
|
|
|
vert_blank = 23;
|
2021-12-07 22:48:33 +01:00
|
|
|
this->pixel_clock = 15667200;
|
|
|
|
break;
|
|
|
|
case PdmVideoMode::Rgb13in:
|
|
|
|
new_width = 640;
|
|
|
|
new_height = 480;
|
2022-01-10 17:09:21 +01:00
|
|
|
hori_blank = 256;
|
|
|
|
vert_blank = 45;
|
2021-12-07 22:48:33 +01:00
|
|
|
this->pixel_clock = 31334400;
|
|
|
|
break;
|
|
|
|
case PdmVideoMode::Rgb16in:
|
|
|
|
new_width = 832;
|
|
|
|
new_height = 624;
|
2022-01-10 17:09:21 +01:00
|
|
|
hori_blank = 320;
|
|
|
|
vert_blank = 43;
|
2021-12-07 22:48:33 +01:00
|
|
|
this->pixel_clock = 57283200;
|
|
|
|
break;
|
|
|
|
case PdmVideoMode::VGA:
|
|
|
|
new_width = 640;
|
|
|
|
new_height = 480;
|
2022-01-10 17:09:21 +01:00
|
|
|
hori_blank = 160;
|
|
|
|
vert_blank = 45;
|
2021-12-07 22:48:33 +01:00
|
|
|
this->pixel_clock = 25175000;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ABORT_F("PDM-Video: invalid video mode %d", this->video_mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (new_width != this->active_width || new_height != this->active_height) {
|
|
|
|
LOG_F(WARNING, "Display window resizing not implemented yet!");
|
|
|
|
}
|
|
|
|
|
|
|
|
// figure out where our framebuffer is located
|
|
|
|
uint64_t hmc_control = this->hmc_obj->get_control_reg();
|
|
|
|
uint32_t fb_base_phys = ((hmc_control >> HMC_VBASE_BIT) & 1) ? 0 : 0x100000;
|
|
|
|
LOG_F(INFO, "PDM-Video: framebuffer phys base addr = 0x%X", fb_base_phys);
|
|
|
|
|
2023-08-01 17:41:02 +02:00
|
|
|
// set CRTC parameters
|
2023-10-02 15:00:12 +02:00
|
|
|
MapDmaResult res = mmu_map_dma_mem(fb_base_phys, PDM_FB_SIZE_MAX, false);
|
|
|
|
this->fb_ptr = res.host_va;
|
2021-12-07 22:48:33 +01:00
|
|
|
this->active_width = new_width;
|
|
|
|
this->active_height = new_height;
|
2023-08-01 17:41:02 +02:00
|
|
|
this->hori_blank = hori_blank;
|
|
|
|
this->vert_blank = vert_blank;
|
|
|
|
this->hori_total = new_width + hori_blank;
|
|
|
|
this->vert_total = new_height + vert_blank;
|
2021-12-07 22:48:33 +01:00
|
|
|
|
2022-01-10 17:09:21 +01:00
|
|
|
this->set_depth_internal(new_width);
|
|
|
|
|
2023-07-31 03:36:17 +02:00
|
|
|
this->stop_refresh_task();
|
2023-04-01 17:38:11 +02:00
|
|
|
|
2022-01-10 17:09:21 +01:00
|
|
|
// set up video refresh timer
|
2023-08-01 17:41:02 +02:00
|
|
|
this->refresh_rate = (double)(this->pixel_clock) / hori_total / vert_total;
|
2023-07-31 03:36:17 +02:00
|
|
|
LOG_F(INFO, "PDM-Video: refresh rate set to %f Hz", this->refresh_rate);
|
|
|
|
this->start_refresh_task();
|
2021-12-07 22:48:33 +01:00
|
|
|
|
2023-04-01 17:38:11 +02:00
|
|
|
LOG_F(9, "PDM-Video: video enabled");
|
2021-12-07 22:48:33 +01:00
|
|
|
this->crtc_on = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PdmOnboardVideo::disable_video_internal()
|
|
|
|
{
|
2023-08-01 17:41:02 +02:00
|
|
|
this->stop_refresh_task();
|
2023-04-01 17:38:11 +02:00
|
|
|
this->blank_on = true;
|
|
|
|
this->blank_display();
|
|
|
|
this->blanking = 0x80;
|
2021-12-07 22:48:33 +01:00
|
|
|
this->crtc_on = false;
|
2023-04-01 17:38:11 +02:00
|
|
|
LOG_F(9, "PDM-Video: video disabled");
|
2021-12-07 22:48:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Ariel II has a weird 1bpp mode where a white pixel is mapped to
|
|
|
|
CLUT entry #127 (%01111111) and a black pixel to #255 (%11111111).
|
2023-07-31 03:36:17 +02:00
|
|
|
It requres a non-standard conversion routine implemented below.
|
2021-12-07 22:48:33 +01:00
|
|
|
*/
|
2023-09-18 17:34:23 -07:00
|
|
|
void PdmOnboardVideo::convert_frame_1bpp_indexed(uint8_t *dst_buf, int dst_pitch)
|
2021-12-07 22:48:33 +01:00
|
|
|
{
|
2023-12-02 20:41:30 -08:00
|
|
|
uint8_t *src_row, *dst_row;
|
|
|
|
int src_pitch;
|
|
|
|
uint64_t pixels;
|
2021-12-07 22:48:33 +01:00
|
|
|
|
|
|
|
// prepare cached ARGB values for white & black pixels
|
2023-12-02 20:41:30 -08:00
|
|
|
pixels = ((uint64_t)this->palette[127] << 32) | this->palette[255];
|
|
|
|
|
|
|
|
src_pitch = this->fb_pitch - ((this->active_width + 7) >> 3);
|
|
|
|
dst_pitch = dst_pitch - 4 * this->active_width;
|
|
|
|
|
|
|
|
src_row = this->fb_ptr - 1;
|
|
|
|
dst_row = dst_buf;
|
|
|
|
for (int h = this->active_height; h > 0; h--) {
|
|
|
|
uint8_t bit = 0x00;
|
|
|
|
uint8_t c;
|
|
|
|
for (int x = this->active_width; x > 0; x--) {
|
|
|
|
if (!bit) {
|
|
|
|
src_row += 1;
|
|
|
|
bit = 0x80;
|
|
|
|
c = *src_row;
|
|
|
|
}
|
|
|
|
WRITE_DWORD_LE_A(dst_row, (uint32_t)(pixels >> (!(c & bit) << 5)));
|
|
|
|
bit >>= 1;
|
|
|
|
dst_row += 4;
|
|
|
|
}
|
|
|
|
src_row += src_pitch;
|
|
|
|
dst_row += dst_pitch;
|
|
|
|
}
|
|
|
|
}
|
2021-12-07 22:48:33 +01:00
|
|
|
|
2023-12-02 20:41:30 -08:00
|
|
|
void PdmOnboardVideo::convert_frame_2bpp_indexed(uint8_t *dst_buf, int dst_pitch)
|
|
|
|
{
|
|
|
|
uint8_t *src_row, *dst_row;
|
|
|
|
int src_pitch;
|
|
|
|
|
|
|
|
src_pitch = this->fb_pitch - (this->active_width >> 2);
|
|
|
|
dst_pitch = dst_pitch - 4 * this->active_width;
|
|
|
|
|
|
|
|
src_row = this->fb_ptr;
|
|
|
|
dst_row = dst_buf;
|
|
|
|
for (int h = this->active_height; h > 0; h--) {
|
|
|
|
uint8_t c;
|
|
|
|
for (int x = this->active_width >> 2; x > 0; x--) {
|
|
|
|
c = *src_row;
|
|
|
|
WRITE_DWORD_LE_A(dst_row, this->palette[c & 0xc0 | 0x3f]);
|
|
|
|
dst_row += 4;
|
|
|
|
WRITE_DWORD_LE_A(dst_row, this->palette[(c << 2) & 0xc0 | 0x3f]);
|
|
|
|
dst_row += 4;
|
|
|
|
WRITE_DWORD_LE_A(dst_row, this->palette[(c << 4) & 0xc0 | 0x3f]);
|
|
|
|
dst_row += 4;
|
|
|
|
WRITE_DWORD_LE_A(dst_row, this->palette[(uint8_t)(c << 6) | 0x3f]);
|
|
|
|
dst_row += 4;
|
|
|
|
src_row += 1;
|
|
|
|
}
|
|
|
|
src_row += src_pitch;
|
|
|
|
dst_row += dst_pitch;
|
|
|
|
}
|
|
|
|
}
|
2021-12-07 22:48:33 +01:00
|
|
|
|
2023-12-02 20:41:30 -08:00
|
|
|
void PdmOnboardVideo::convert_frame_4bpp_indexed(uint8_t *dst_buf, int dst_pitch)
|
|
|
|
{
|
|
|
|
uint8_t *src_row, *dst_row;
|
|
|
|
int src_pitch;
|
|
|
|
|
|
|
|
src_pitch = this->fb_pitch - (this->active_width >> 1);
|
|
|
|
dst_pitch = dst_pitch - 4 * this->active_width;
|
|
|
|
|
|
|
|
src_row = this->fb_ptr;
|
|
|
|
dst_row = dst_buf;
|
|
|
|
for (int h = this->active_height; h > 0; h--) {
|
|
|
|
uint8_t c;
|
|
|
|
for (int x = this->active_width >> 1; x > 0; x--) {
|
|
|
|
c = *src_row;
|
|
|
|
WRITE_DWORD_LE_A(dst_row, this->palette[c & 0xf0 | 0x0f]);
|
|
|
|
dst_row += 4;
|
|
|
|
WRITE_DWORD_LE_A(dst_row, this->palette[(uint8_t)(c << 4) | 0x0f]);
|
|
|
|
dst_row += 4;
|
|
|
|
src_row += 1;
|
2021-12-07 22:48:33 +01:00
|
|
|
}
|
2023-12-02 20:41:30 -08:00
|
|
|
src_row += src_pitch;
|
|
|
|
dst_row += dst_pitch;
|
2021-12-07 22:48:33 +01:00
|
|
|
}
|
|
|
|
}
|