337 lines
16 KiB
C++
337 lines
16 KiB
C++
/////////////////////////////////////////////////////////////////////////
|
||
// $Id$
|
||
/////////////////////////////////////////////////////////////////////////
|
||
//
|
||
// Copyright (C) 2018-2021 The Bochs Project
|
||
//
|
||
// This library is free software; you can redistribute it and/or
|
||
// modify it under the terms of the GNU Lesser General Public
|
||
// License as published by the Free Software Foundation; either
|
||
// version 2 of the License, or (at your option) any later version.
|
||
//
|
||
// This library 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
|
||
// Lesser General Public License for more details.
|
||
//
|
||
// You should have received a copy of the GNU Lesser General Public
|
||
// License along with this library; if not, write to the Free Software
|
||
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||
|
||
// DDC support (returns the VESA EDID for the Bochs plug&play monitor)
|
||
|
||
// Define BX_PLUGGABLE in files that can be compiled into plugins. For
|
||
// platforms that require a special tag on exported symbols, BX_PLUGGABLE
|
||
// is used to know when we are exporting symbols and when we are importing.
|
||
#define BX_PLUGGABLE
|
||
|
||
#include "bochs.h"
|
||
#include "gui/siminterface.h"
|
||
#include "ddc.h"
|
||
#include "param_names.h"
|
||
|
||
#define LOG_THIS
|
||
|
||
enum {
|
||
DDC_STAGE_START,
|
||
DDC_STAGE_ADDRESS,
|
||
DDC_STAGE_RW,
|
||
DDC_STAGE_DATA_IN,
|
||
DDC_STAGE_DATA_OUT,
|
||
DDC_STAGE_ACK_IN,
|
||
DDC_STAGE_ACK_OUT,
|
||
DDC_STAGE_STOP
|
||
};
|
||
|
||
const Bit8u vesa_EDID[128] = {
|
||
0x00,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0x00,
|
||
/* 0x0000 8-byte header */
|
||
0x04,0x21, /* 0x0008 Vendor ID ("AAA") */
|
||
0xAB,0xCD, /* 0x000A Product ID */
|
||
0x00,0x00,0x00,0x00, /* 0x000C Serial number (none) */
|
||
12, 11, /* 0x0010 Week of manufacture (12) and year of manufacture (2001) */
|
||
0x01, 0x03, /* 0x0012 EDID version number (1.3) */
|
||
0x0F, /* 0x0014 Video signal interface (analogue, 0.700 : 0.300 : 1.000 V p-p,
|
||
Video Setup: Blank Level = Black Level, Separate Sync H & V Signals are
|
||
supported, Composite Sync Signal on Horizontal is supported, Composite
|
||
Sync Signal on Green Video is supported, Serration on the Vertical Sync
|
||
is supported) */
|
||
0x21,0x19, /* 0x0015 Scren size (330 mm * 250 mm) */
|
||
0x78, /* 0x0017 Display gamma (2.2) */
|
||
0x0F, /* 0x0018 Feature flags (no DMPS states, RGB, preferred timing mode, display is continuous frequency) */
|
||
0x78,0xF5, /* 0x0019 Least significant bits for chromaticity and default white point */
|
||
0xA6,0x55,0x48,0x9B,0x26,0x12,0x50,0x54,
|
||
/* 0x001B Most significant bits for chromaticity and default white point */
|
||
|
||
0xFF, /* 0x0023 Established timings 1 (720 x 400 @ 70Hz, 720 x 400 @ 88Hz,
|
||
640 x 480 @ 60Hz, 640 x 480 @ 67Hz, 640 x 480 @ 72Hz, 640 x 480 @ 75Hz,
|
||
800 x 600 @ 56Hz, 800 x 600 @ 60Hz) - historical resolutions */
|
||
0xEF, /* 0x0024 Established timings 2 (800 x 600 @ 72Hz, 800 x 600 @ 75Hz, 832 x 624 @ 75Hz
|
||
not 1024 x 768 @ 87Hz(I), 1024 x 768 @ 60Hz, 1024 x 768 @ 70Hz,
|
||
1024 x 768 @ 75Hz, 1280 x 1024 @ 75Hz) - historical resolutions */
|
||
0x80, /* 0x0025 Established timings 2 (1152 x 870 @ 75Hz and no manufacturer timings) */
|
||
|
||
/* Standard timing */
|
||
/* First byte: X resolution, divided by 8, less 31 (256–2288 pixels) */
|
||
/* bit 7-6, X:Y pixel ratio: 00=16:10; 01=4:3; 10=5:4; 11=16:9 */
|
||
/* bit 5-0, Vertical frequency, less 60 (60–123 Hz), nop 01 01 */
|
||
0x31, 0x59, /* 0x0026 Standard timing #1 (640 x 480 @ 85 Hz) */
|
||
0x45, 0x59, /* 0x0028 Standard timing #2 (800 x 600 @ 85 Hz) */
|
||
0x61, 0x59, /* 0x002A Standard timing #3 (1024 x 768 @ 85 Hz) */
|
||
0x81, 0xCA, /* 0x002C Standard timing #4 (1280 x 720 @ 70 Hz) */
|
||
0x81, 0x0A, /* 0x002E Standard timing #5 (1280 x 800 @ 70 Hz) */
|
||
0xA9, 0xC0, /* 0x0030 Standard timing #6 (1600 x 900 @ 60 Hz) */
|
||
0xA9, 0x40, /* 0x0034 Standard timing #7 (1600 x 1200 @ 60 Hz) */
|
||
0xD1, 0x00, /* 0x0032 Standard timing #8 (1920 x 1080 @ 60 Hz) */
|
||
|
||
/* 0x0036 First 18-byte descriptor (1920 x 1200) */
|
||
0x3C, 0x28, /* Pixel clock = 154000000 Hz */
|
||
0x80, /* 0x0038 Horizontal addressable pixels low byte (0x0780 & 0xFF) */
|
||
0xA0, /* 0x0039 Horizontal blanking low byte (0x00A0 & 0xFF) */
|
||
0x70, /* 0x003A Horizontal addressable pixels high 4 bits ((0x0780 & 0x0F00) >> 4), and */
|
||
/* Horizontal blanking high 4 bits ((0x00A0 & 0x0F00 ) >> 8) as low bits */
|
||
0xB0, /* 0x003B Vertical addressable pixels low byte (0x04B0 & 0xFF) */
|
||
0x23, /* 0x003C Vertical blanking low byte (0x0023 & 0xFF) */
|
||
0x40, /* 0x003D Vertical addressable pixels high 4 bits ((0x04B0 & 0x0F00) >> 4), and */
|
||
/* Vertical blanking high 4 bits ((0x0024 & x0F00) >> 8) */
|
||
0x30, /* 0x003E Horizontal front porch in pixels low byte (0x0030 & 0xFF) */
|
||
0x20, /* 0x003F Horizontal sync pulse width in pixels low byte (0x0020 & 0xFF) */
|
||
0x36, /* 0x0040 Vertical front porch in lines low 4 bits ((0x0003 & 0x0F) << 4), and */
|
||
/* Vertical sync pulse width in lines low 4 bits (0x0006 & 0x0F) */
|
||
0x00, /* 0x0041 Horizontal front porch pixels high 2 bits (0x0030 >> 8), and */
|
||
/* Horizontal sync pulse width in pixels high 2 bits (0x0020 >> 8), and */
|
||
/* Vertical front porch in lines high 2 bits (0x0003 >> 4), and */
|
||
/* Vertical sync pulse width in lines high 2 bits (0x0006 >> 4) */
|
||
0x06, /* 0x0042 Horizontal addressable video image size in mm low 8 bits (0x0206 & 0xFF) */
|
||
0x44, /* 0x0043 Vertical addressable video image size in mm low 8 bits (0x0144 & 0xFF) */
|
||
0x21, /* 0x0044 Horizontal addressable video image size in mm high 8 bits (0x0206 >> 8), and */
|
||
/* Vertical addressable video image size in mm high 8 bits (0x0144 >> 8) */
|
||
0x00, /* 0x0045 Left and right border size in pixels (0x00) */
|
||
0x00, /* 0x0046 Top and bottom border size in lines (0x00) */
|
||
0x1E, /* 0x0047 Flags (non-interlaced, no stereo, analog composite sync, sync on */
|
||
/* all three (RGB) video signals) */
|
||
|
||
|
||
/* 0x0048 Second 18-byte descriptor (1280 x 1024) */
|
||
0x30, 0x2a, /* Pixel clock = 108000000 Hz */
|
||
0x00, /* 0x004A Horizontal addressable pixels low byte (0x0500 & 0xFF) */
|
||
0x98, /* 0x004B Horizontal blanking low byte (0x0198 & 0xFF) */
|
||
0x51, /* 0x004C Horizontal addressable pixels high 4 bits (0x0500 >> 8), and */
|
||
/* Horizontal blanking high 4 bits (0x0198 >> 8) */
|
||
0x00, /* 0x004D Vertical addressable pixels low byte (0x0400 & 0xFF) */
|
||
0x2A, /* 0x004E Vertical blanking low byte (0x002A & 0xFF) */
|
||
0x40, /* 0x004F Vertical addressable pixels high 4 bits (0x0400 >> 8), and */
|
||
/* Vertical blanking high 4 bits (0x002A >> 8) */
|
||
0x30, /* 0x0050 Horizontal front porch in pixels low byte (0x0030 & 0xFF) */
|
||
0x70, /* 0x0051 Horizontal sync pulse width in pixels low byte (0x0070 & 0xFF) */
|
||
0x13, /* 0x0052 Vertical front porch in lines low 4 bits (0x0001 & 0x0F), and */
|
||
/* Vertical sync pulse width in lines low 4 bits (0x0003 & 0x0F) */
|
||
0x00, /* 0x0053 Horizontal front porch pixels high 2 bits (0x0030 >> 8), and */
|
||
/* Horizontal sync pulse width in pixels high 2 bits (0x0070 >> 8), and */
|
||
/* Vertical front porch in lines high 2 bits (0x0001 >> 4), and */
|
||
/* Vertical sync pulse width in lines high 2 bits (0x0003 >> 4) */
|
||
0x2C, /* 0x0054 Horizontal addressable video image size in mm low 8 bits (0x012C & 0xFF) */
|
||
0xE1, /* 0x0055 Vertical addressable video image size in mm low 8 bits (0x00E1 & 0xFF) */
|
||
0x10, /* 0x0056 Horizontal addressable video image size in mm high 8 bits (0x012C >> 8), and */
|
||
/* Vertical addressable video image size in mm high 8 bits (0x00E1 >> 8) */
|
||
0x00, /* 0x0057 Left and right border size in pixels (0x00) */
|
||
0x00, /* 0x0058 Top and bottom border size in lines (0x00) */
|
||
0x1E, /* 0x0059 Flags (non-interlaced, no stereo, analog composite sync, sync on */
|
||
/* all three (RGB) video signals) */
|
||
|
||
|
||
0x00,0x00,0x00,0xFF,0x00, /* 0x005A Third 18-byte descriptor - display product serial number */
|
||
'0','1','2','3','4','5','6','7','8','9',
|
||
0x0A,0x20,0x20,
|
||
|
||
0x00,0x00,0x00,0xFC,0x00, /* 0x006C Fourth 18-byte descriptor - display product name */
|
||
'B','o','c','h','s',' ','S','c','r','e','e','n',
|
||
0x0A,
|
||
|
||
0x00, /* 0x007E Extension block count (none) */
|
||
0x00, /* 0x007F Checksum (set by constructor) */
|
||
};
|
||
|
||
bx_ddc_c::bx_ddc_c(void)
|
||
{
|
||
int fd, ret;
|
||
struct stat stat_buf;
|
||
const char *path;
|
||
Bit8u checksum = 0;
|
||
|
||
put("DDC");
|
||
s.DCKhost = 1;
|
||
s.DDAhost = 1;
|
||
s.DDAmon = 1;
|
||
s.ddc_stage = DDC_STAGE_STOP;
|
||
s.ddc_ack = 1;
|
||
s.ddc_rw = 1;
|
||
s.edid_index = 0;
|
||
s.ddc_mode = SIM->get_param_enum(BXPN_DDC_MODE)->get();
|
||
if (s.ddc_mode == BX_DDC_MODE_BUILTIN) {
|
||
memcpy(s.edid_data, vesa_EDID, 128);
|
||
s.edid_extblock = 0;
|
||
} else if (s.ddc_mode == BX_DDC_MODE_FILE) {
|
||
path = SIM->get_param_string(BXPN_DDC_FILE)->getptr();
|
||
fd = open(path, O_RDONLY
|
||
#ifdef O_BINARY
|
||
| O_BINARY
|
||
#endif
|
||
);
|
||
if (fd < 0) {
|
||
BX_PANIC(("failed to open monitor EDID file '%s'", path));
|
||
}
|
||
ret = fstat(fd, &stat_buf);
|
||
if (ret) {
|
||
BX_PANIC(("could not fstat() monitor EDID file."));
|
||
}
|
||
if ((stat_buf.st_size != 128) && (stat_buf.st_size != 256)) {
|
||
BX_PANIC(("monitor EDID file size must be 128 or 256 bytes"));
|
||
} else {
|
||
s.edid_extblock = (stat_buf.st_size == 256);
|
||
}
|
||
ret = ::read(fd, (bx_ptr_t) s.edid_data, (unsigned)stat_buf.st_size);
|
||
if (ret != stat_buf.st_size) {
|
||
BX_PANIC(("error reading monitor EDID file."));
|
||
}
|
||
close(fd);
|
||
BX_INFO(("Monitor EDID read from image file '%s'.", path));
|
||
}
|
||
s.edid_data[127] = 0;
|
||
for (int i = 0; i < 128; i++) {
|
||
checksum += s.edid_data[i];
|
||
}
|
||
if (checksum != 0) {
|
||
s.edid_data[127] = (Bit8u)-checksum;
|
||
}
|
||
}
|
||
|
||
bx_ddc_c::~bx_ddc_c(void)
|
||
{
|
||
}
|
||
|
||
Bit8u bx_ddc_c::read()
|
||
{
|
||
Bit8u retval = (Bit8u)(((s.DDAmon & s.DDAhost) << 3) | (s.DCKhost << 2) |
|
||
(s.DDAhost << 1) | (Bit8u)s.DCKhost);
|
||
return retval;
|
||
}
|
||
|
||
void bx_ddc_c::write(bool dck, bool dda)
|
||
{
|
||
bool dck_change = 0;
|
||
bool dda_change = 0;
|
||
|
||
if (s.ddc_mode == BX_DDC_MODE_DISABLED)
|
||
return;
|
||
|
||
if ((dck != s.DCKhost) || (dda != s.DDAhost)) {
|
||
dck_change = (dck != s.DCKhost);
|
||
dda_change = (dda != s.DDAhost);
|
||
if (dck_change && dda_change) {
|
||
BX_ERROR(("DDC unknown: DCK=%d DDA=%d", dck, dda));
|
||
} else if (dck_change) {
|
||
if (!dck) {
|
||
switch (s.ddc_stage) {
|
||
case DDC_STAGE_START:
|
||
s.ddc_stage = DDC_STAGE_ADDRESS;
|
||
s.ddc_bitshift = 6;
|
||
s.ddc_byte = 0;
|
||
break;
|
||
case DDC_STAGE_ADDRESS:
|
||
if (s.ddc_bitshift > 0) {
|
||
s.ddc_bitshift--;
|
||
} else {
|
||
s.ddc_ack = !(s.ddc_byte == 0x50);
|
||
BX_DEBUG(("Address = 0x%02x", s.ddc_byte));
|
||
s.ddc_stage = DDC_STAGE_RW;
|
||
}
|
||
break;
|
||
case DDC_STAGE_RW:
|
||
BX_DEBUG(("R/W mode = %d", s.ddc_rw));
|
||
s.ddc_stage = DDC_STAGE_ACK_OUT;
|
||
s.DDAmon = s.ddc_ack;
|
||
break;
|
||
case DDC_STAGE_DATA_IN:
|
||
if (s.ddc_bitshift > 0) {
|
||
s.ddc_bitshift--;
|
||
} else {
|
||
s.ddc_ack = 0;
|
||
BX_DEBUG(("Data = 0x%02x (setting offset address)", s.ddc_byte));
|
||
s.edid_index = s.ddc_byte;
|
||
s.DDAmon = s.ddc_ack;
|
||
s.ddc_stage = DDC_STAGE_ACK_OUT;
|
||
}
|
||
break;
|
||
case DDC_STAGE_DATA_OUT:
|
||
if (s.ddc_bitshift > 0) {
|
||
s.ddc_bitshift--;
|
||
s.DDAmon = ((s.ddc_byte >> s.ddc_bitshift) & 1);
|
||
} else {
|
||
s.ddc_stage = DDC_STAGE_ACK_IN;
|
||
s.DDAmon = 1;
|
||
}
|
||
break;
|
||
case DDC_STAGE_ACK_IN:
|
||
BX_DEBUG(("Received status %s", s.ddc_ack ? "NAK":"ACK"));
|
||
if (s.ddc_ack == 0) {
|
||
s.ddc_bitshift = 7;
|
||
s.ddc_stage = DDC_STAGE_DATA_OUT;
|
||
s.ddc_byte = get_edid_byte();
|
||
s.DDAmon = ((s.ddc_byte >> s.ddc_bitshift) & 1);
|
||
} else {
|
||
s.ddc_stage = DDC_STAGE_STOP;
|
||
}
|
||
break;
|
||
case DDC_STAGE_ACK_OUT:
|
||
BX_DEBUG(("Sent status %s", s.ddc_ack ? "NAK":"ACK"));
|
||
s.ddc_bitshift = 7;
|
||
if (s.ddc_rw) {
|
||
s.ddc_stage = DDC_STAGE_DATA_OUT;
|
||
s.ddc_byte = get_edid_byte();
|
||
s.DDAmon = ((s.ddc_byte >> s.ddc_bitshift) & 1);
|
||
} else {
|
||
s.ddc_stage = DDC_STAGE_DATA_IN;
|
||
s.DDAmon = 1;
|
||
s.ddc_byte = 0;
|
||
}
|
||
break;
|
||
}
|
||
} else {
|
||
switch (s.ddc_stage) {
|
||
case DDC_STAGE_ADDRESS:
|
||
case DDC_STAGE_DATA_IN:
|
||
s.ddc_byte |= (Bit8u)(s.DDAhost << s.ddc_bitshift);
|
||
break;
|
||
case DDC_STAGE_RW:
|
||
s.ddc_rw = s.DDAhost;
|
||
break;
|
||
case DDC_STAGE_ACK_IN:
|
||
s.ddc_ack = s.DDAhost;
|
||
break;
|
||
}
|
||
}
|
||
} else {
|
||
if (s.DCKhost) {
|
||
if (!dda) {
|
||
s.ddc_stage = DDC_STAGE_START;
|
||
BX_DEBUG(("Start detected"));
|
||
} else {
|
||
s.ddc_stage = DDC_STAGE_STOP;
|
||
BX_DEBUG(("Stop detected"));
|
||
}
|
||
}
|
||
}
|
||
s.DCKhost = dck;
|
||
s.DDAhost = dda;
|
||
}
|
||
}
|
||
|
||
Bit8u bx_ddc_c::get_edid_byte()
|
||
{
|
||
Bit8u value = s.edid_data[s.edid_index++];
|
||
BX_DEBUG(("Sending EDID byte 0x%02x (value = 0x%02x)", s.edid_index - 1, value));
|
||
if (!s.edid_extblock) s.edid_index &= 0x7f;
|
||
return value;
|
||
}
|