1117 lines
37 KiB
C
1117 lines
37 KiB
C
/*
|
|
* BIOS interrupt 10h handler
|
|
*
|
|
* Copyright 1998 Ove Kåven
|
|
* Copyright 1998 Joseph Pranevich
|
|
*
|
|
* 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.1 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "miscemu.h"
|
|
#include "vga.h"
|
|
#include "wine/debug.h"
|
|
#include "dosexe.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(int);
|
|
|
|
|
|
/*
|
|
* Wine internal information about video modes.
|
|
* If depth is zero, the mode is considered to
|
|
* be a text mode.
|
|
*/
|
|
typedef struct {
|
|
WORD Mode;
|
|
WORD Width;
|
|
WORD Height;
|
|
WORD Depth;
|
|
} INT10_MODE;
|
|
|
|
|
|
/*
|
|
* List of supported video modes.
|
|
*/
|
|
static const INT10_MODE INT10_modelist[] =
|
|
{
|
|
{0x0000, 40, 25, 0},
|
|
{0x0001, 40, 25, 0},
|
|
{0x0002, 80, 25, 0},
|
|
{0x0003, 80, 25, 0},
|
|
{0x0007, 80, 25, 0},
|
|
{0x000d, 320, 200, 4},
|
|
{0x000e, 640, 200, 4},
|
|
{0x0010, 640, 350, 4},
|
|
{0x0012, 640, 480, 4},
|
|
{0x0013, 320, 200, 8},
|
|
{0x0100, 640, 400, 8},
|
|
{0x0101, 640, 480, 8},
|
|
{0x0102, 800, 600, 4},
|
|
{0x0103, 800, 600, 8},
|
|
{0x0104, 1024, 768, 4},
|
|
{0x0105, 1024, 768, 8},
|
|
{0x0106, 1280, 1024, 4},
|
|
{0x0107, 1280, 1024, 8},
|
|
{0x0108, 80, 60, 0},
|
|
{0x0109, 132, 25, 0},
|
|
{0x010a, 132, 43, 0},
|
|
{0x010b, 132, 50, 0},
|
|
{0x010c, 132, 60, 0},
|
|
{0x010d, 320, 200, 15},
|
|
{0x010e, 320, 200, 16},
|
|
{0x010f, 320, 200, 24},
|
|
{0x0110, 640, 480, 15},
|
|
{0x0111, 640, 480, 16},
|
|
{0x0112, 640, 480, 24},
|
|
{0x0113, 800, 600, 15},
|
|
{0x0114, 800, 600, 16},
|
|
{0x0115, 800, 600, 24},
|
|
{0x0116, 1024, 768, 15},
|
|
{0x0117, 1024, 768, 16},
|
|
{0x0118, 1024, 768, 24},
|
|
{0x0119, 1280, 1024, 15},
|
|
{0x011a, 1280, 1024, 16},
|
|
{0x011b, 1280, 1024, 24},
|
|
{0xffff, 0, 0, 0}
|
|
};
|
|
|
|
|
|
/**********************************************************************
|
|
* INT10_FindMode
|
|
*/
|
|
static const INT10_MODE *INT10_FindMode( WORD mode )
|
|
{
|
|
const INT10_MODE *ptr = INT10_modelist;
|
|
|
|
/*
|
|
* Filter out flags.
|
|
*/
|
|
mode &= 0x17f;
|
|
|
|
while (ptr->Mode != 0xffff)
|
|
{
|
|
if (ptr->Mode == mode)
|
|
return ptr;
|
|
ptr++;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* INT10_FillControllerInformation
|
|
*
|
|
* Fill 256-byte (VBE1.x) or 512-byte buffer (VBE2.0+) with
|
|
* capabilities of the video controller.
|
|
*/
|
|
static void INT10_FillControllerInformation( BYTE *buffer )
|
|
{
|
|
/*
|
|
* FIXME: Move VESA info stuff here from dosmem.
|
|
*/
|
|
memcpy(buffer, &BIOS_EXTRA_PTR->vesa_info, sizeof(VESAINFO));
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* INT10_FillModeInformation
|
|
*
|
|
* Fill 256-byte buffer with extended information about display mode.
|
|
*
|
|
* Returns FALSE if mode is unknown and TRUE is mode is known
|
|
* even if it is not supported.
|
|
*/
|
|
static BOOL INT10_FillModeInformation( BYTE *buffer, WORD mode )
|
|
{
|
|
const INT10_MODE *ptr = INT10_FindMode( mode );
|
|
if (!ptr)
|
|
return FALSE;
|
|
|
|
/*
|
|
* 00 - WORD: mode attributes
|
|
* Bits:
|
|
* 0 - Mode supported by present hardware configuration.
|
|
* 1 - Optional information available. Must be =1 for VBE v1.2+.
|
|
* 2 - BIOS output supported.
|
|
* Int10 functions 01, 02, 06, 07, 09, 0a and 0e are supported.
|
|
* 3 - Set if color, clear if monochrome.
|
|
* 4 - Set if graphics mode, clear if text mode.
|
|
* 5 - Mode is not VGA-compatible if set.
|
|
* VGA-compatible modes support standard VGA I/O ports.
|
|
* 6 - Bank-switched (or windowed) mode is not supported if set.
|
|
* 7 - Linear framebuffer mode supported.
|
|
* 8 - Double scanning supported.
|
|
* 9 - Interlaced operation supported.
|
|
* 10 - Triple buffering supported.
|
|
* 11 - Stereoscopic display supported.
|
|
* 12 - Dual display start address supported.
|
|
* 13-15 - Reserved.
|
|
*/
|
|
{
|
|
WORD attr = 0x000a; /* color mode, optional info */
|
|
|
|
/*
|
|
* FIXME: Attribute handling is incomplete.
|
|
*/
|
|
|
|
/* Mode supported? FIXME: correct value */
|
|
attr |= 0x0001;
|
|
|
|
/* Graphical mode? */
|
|
if (ptr->Depth)
|
|
attr |= 0x0010;
|
|
|
|
/* Not VGA-compatible? */
|
|
if (mode > 0xff)
|
|
attr |= 0x0020;
|
|
|
|
*(WORD*)(buffer + 0) = attr;
|
|
}
|
|
|
|
/*
|
|
* 02 - BYTE[2]: window attributes, window A and window B
|
|
* Bits:
|
|
* 0 - Window exists.
|
|
* 1 - Window is readable.
|
|
* 2 - Window is writable.
|
|
* 3-7 - Reserved.
|
|
*/
|
|
buffer[2] = 0x07; /* window A exists, readable and writable */
|
|
buffer[3] = 0x00; /* window B not supported */
|
|
|
|
/* 04 - WORD: window granularity in KB */
|
|
*(WORD*)(buffer + 4) = 64;
|
|
|
|
/* 06 - WORD: window size in KB */
|
|
*(WORD*)(buffer + 6) = 64;
|
|
|
|
/* 08 - WORD[2]: start segments, window A and window B */
|
|
*(WORD*)(buffer + 8) = 0xa000; /* window A segment */
|
|
*(WORD*)(buffer + 10) = 0x0000; /* window B not supported */
|
|
|
|
/* 12 - DWORD: window positioning function */
|
|
*(DWORD*)(buffer + 12) = 0; /* not supported */
|
|
|
|
/* 16 - WORD: bytes per scan line */
|
|
*(WORD*)(buffer + 16) = 0; /* FIXME */
|
|
|
|
/* 18 - WORD: width in pixels (graphics) or characters (text) */
|
|
*(WORD*)(buffer + 18) = ptr->Width;
|
|
|
|
/* 20 - WORD: height in pixels (graphics) or characters (text) */
|
|
*(WORD*)(buffer + 20) = ptr->Height;
|
|
|
|
/* 22 - BYTE: width of character cell in pixels */
|
|
buffer[22] = 0; /* FIXME */
|
|
|
|
/* 23 - BYTE: height of character cell in pixels */
|
|
buffer[23] = 0; /* FIXME */
|
|
|
|
/* 24 - BYTE: number of memory planes */
|
|
buffer[24] = 1; /* FIXME */
|
|
|
|
/* 25 - BYTE: number of bits per pixel */
|
|
buffer[25] = ptr->Depth; /* FIXME: text modes? reserved bits? */
|
|
|
|
/* 26 - BYTE: number of banks */
|
|
buffer[26] = 1; /* FIXME */
|
|
|
|
/*
|
|
* 27 - BYTE: memory model type
|
|
* Values (hex):
|
|
* 00 - Text mode
|
|
* 01 - CGA graphics
|
|
* 02 - Hercules graphics
|
|
* 03 - Planar
|
|
* 04 - Packed pixel
|
|
* 05 - Non-chain 4, 256 color
|
|
* 06 - Direct color
|
|
* 07 - YUV
|
|
* 08-0F - Reserved for VESA.
|
|
* 10-FF - OEM memory models.
|
|
*/
|
|
if (!ptr->Depth)
|
|
buffer[27] = 0; /* text mode */
|
|
else
|
|
buffer[27] = 3; /* FIXME */
|
|
|
|
/* 28 - BYTE: size of bank in KB */
|
|
buffer[28] = 0; /* FIXME */
|
|
|
|
/* 29 - BYTE: number of image pages (less one) in video RAM */
|
|
buffer[29] = 0; /* FIXME */
|
|
|
|
/* 30 - BYTE: reserved (0x00 for VBE 1.0-2.0, 0x01 for VBE 3.0) */
|
|
buffer[30] = 0x01;
|
|
|
|
/*
|
|
* 31 - BYTE: red mask size
|
|
* Size of red color component in bits.
|
|
* Used only when memory model is direct color, otherwise set to zero.
|
|
*/
|
|
buffer[31] = 0; /* FIXME */
|
|
|
|
/*
|
|
* 32 - BYTE: red field position
|
|
* Bit position of the least significant bit of red color component.
|
|
* Used only when memory model is direct color, otherwise set to zero.
|
|
*/
|
|
buffer[32] = 0; /* FIXME */
|
|
|
|
/* 33 - BYTE: green mask size */
|
|
buffer[33] = 0; /* FIXME */
|
|
|
|
/* 34 - BYTE: green field position */
|
|
buffer[34] = 0; /* FIXME */
|
|
|
|
/* 35 - BYTE: blue mask size */
|
|
buffer[35] = 0; /* FIXME */
|
|
|
|
/* 36 - BYTE: blue field position */
|
|
buffer[36] = 0;
|
|
|
|
/* 37 - BYTE: reserved mask size */
|
|
buffer[37] = 0;
|
|
|
|
/* 38 - BYTE: reserved mask position */
|
|
buffer[38] = 0;
|
|
|
|
/*
|
|
* 39 - BYTE: direct color mode info
|
|
* Bits:
|
|
* 0 - Set if color ramp is programmable.
|
|
* 1 - Set if bytes in reserved field may be used by application.
|
|
*/
|
|
buffer[39] = 0; /* not supported */
|
|
|
|
/* 40 - DWORD: physical address of linear video buffer */
|
|
*(DWORD*)(buffer + 40) = 0; /* not supported */
|
|
|
|
/* 44 - DWORD: reserved, always zero */
|
|
*(DWORD*)(buffer + 44) = 0;
|
|
|
|
/* 48 - WORD: reserved, always zero */
|
|
*(WORD*)(buffer + 48) = 0;
|
|
|
|
/* 50 - WORD: bytes per scan line in linear modes */
|
|
*(WORD*)(buffer + 50) = *(WORD*)(buffer + 16);
|
|
|
|
/* 52 - BYTE: number of images (less one) for banked video modes */
|
|
buffer[52] = 0; /* FIXME */
|
|
|
|
/* 53 - BYTE: number of images (less one) for linear video modes */
|
|
buffer[53] = buffer[52];
|
|
|
|
/* 54 - BYTE: red mask size (linear modes) */
|
|
buffer[54] = buffer[31];
|
|
|
|
/* 55 - BYTE: red field position (linear modes) */
|
|
buffer[55] = buffer[32];
|
|
|
|
/* 56 - BYTE: green mask size (linear modes) */
|
|
buffer[56] = buffer[33];
|
|
|
|
/* 57 - BYTE: green field size (linear modes) */
|
|
buffer[57] = buffer[34];
|
|
|
|
/* 58 - BYTE: blue mask size (linear modes) */
|
|
buffer[58] = buffer[35];
|
|
|
|
/* 59 - BYTE: blue field position (linear modes) */
|
|
buffer[59] = buffer[36];
|
|
|
|
/* 60 - BYTE: reserved mask size (linear modes) */
|
|
buffer[60] = buffer[37];
|
|
|
|
/* 61 - BYTE: reserved mask position (linear modes) */
|
|
buffer[61] = buffer[38];
|
|
|
|
/* 62 - DWORD: maximum pixel clock for graphics video mode, in Hz */
|
|
*(DWORD*)(buffer + 62) = 0; /* FIXME */
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* INT10_FillStateInformation
|
|
*
|
|
* Fill 64-byte buffer with VGA state and functionality information.
|
|
*/
|
|
static void INT10_FillStateInformation( BYTE *buffer, BIOSDATA *data )
|
|
{
|
|
/*
|
|
* FIXME: Move VGA info stuff here from dosmem.
|
|
*/
|
|
memcpy( buffer, &BIOS_EXTRA_PTR->vid_state, sizeof(VIDEOSTATE) );
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* INT10_SetVideoMode
|
|
*
|
|
* Change current video mode to any VGA or VESA mode.
|
|
* Returns TRUE if mode is supported.
|
|
*
|
|
* Mode bitfields:
|
|
* 0-6: .. Mode number (combined with bit 8).
|
|
* 7: =0 Clear screen.
|
|
* =1 Preserve display memory on mode change (VGA modes).
|
|
* 8: =0 VGA mode.
|
|
* =1 VESA mode.
|
|
* 9: .. Reserved, must be zero.
|
|
* 10: .. Reserved, must be zero.
|
|
* 11: =0 Use default refresh rate.
|
|
* =1 Use user specified refresh rate.
|
|
* 12: .. Reserved, must be zero.
|
|
* 13: .. Reserved, must be zero.
|
|
* 14: =0 Use windowed frame buffer model.
|
|
* =1 Use linear frame buffer model.
|
|
* 15: =0 Clear screen.
|
|
* =1 Preserve display memory on mode change (VESA modes).
|
|
*/
|
|
static BOOL INT10_SetVideoMode( BIOSDATA *data, WORD mode )
|
|
{
|
|
const INT10_MODE *ptr = INT10_FindMode( mode );
|
|
|
|
if (!ptr)
|
|
return FALSE;
|
|
|
|
/*
|
|
* Linear framebuffer is not supported.
|
|
*/
|
|
if (mode & 0x4000)
|
|
return FALSE;
|
|
|
|
/*
|
|
* Note that we do not mask out flags here on purpose.
|
|
*
|
|
* FIXME: Store VESA mode somewhere.
|
|
*/
|
|
if (mode <= 0xff)
|
|
data->VideoMode = mode;
|
|
else
|
|
data->VideoMode = 0;
|
|
|
|
if (ptr->Depth == 0)
|
|
{
|
|
/* Text mode. */
|
|
TRACE( "Setting %s %dx%d text mode\n",
|
|
mode <= 0xff ? "VGA" : "VESA",
|
|
ptr->Width, ptr->Height );
|
|
/*
|
|
* FIXME: We should check here if alpha mode could be set.
|
|
*/
|
|
VGA_SetAlphaMode( ptr->Width, ptr->Height );
|
|
data->VideoColumns = ptr->Width;
|
|
}
|
|
else
|
|
{
|
|
/* Graphics mode. */
|
|
TRACE( "Setting %s %dx%dx%d graphics mode\n",
|
|
mode <= 0xff ? "VGA" : "VESA",
|
|
ptr->Width, ptr->Height, ptr->Depth );
|
|
if (VGA_SetMode( ptr->Width, ptr->Height, ptr->Depth ))
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* INT10_GetCursorPos
|
|
*/
|
|
static void INT10_GetCursorPos(BIOSDATA*data,unsigned page,unsigned*X,unsigned*Y)
|
|
{
|
|
*X = data->VideoCursorPos[page*2]; /* column */
|
|
*Y = data->VideoCursorPos[page*2+1]; /* row */
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* INT10_SetCursorPos
|
|
*/
|
|
static void INT10_SetCursorPos(BIOSDATA*data,unsigned page,unsigned X,unsigned Y)
|
|
{
|
|
data->VideoCursorPos[page*2] = X;
|
|
data->VideoCursorPos[page*2+1] = Y;
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* INT10_InitializeVideoMode
|
|
*
|
|
* The first time this function is called VGA emulation is set to the
|
|
* default text mode.
|
|
*/
|
|
static void INT10_InitializeVideoMode( BIOSDATA *data )
|
|
{
|
|
static BOOL already_initialized = FALSE;
|
|
unsigned width;
|
|
unsigned height;
|
|
|
|
if(already_initialized)
|
|
return;
|
|
already_initialized = TRUE;
|
|
|
|
VGA_InitAlphaMode(&width, &height);
|
|
|
|
/*
|
|
* FIXME: Add more mappings between initial size and
|
|
* text modes.
|
|
*/
|
|
if (width >= 80 && height >= 25)
|
|
INT10_SetVideoMode( data, 0x03 );
|
|
else
|
|
INT10_SetVideoMode( data, 0x01 );
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* INT10_HandleVESA
|
|
*
|
|
* Handler for VESA functions (int10 function 0x4f).
|
|
*/
|
|
static void INT10_HandleVESA( CONTEXT86 *context )
|
|
{
|
|
BIOSDATA *data = BIOS_DATA;
|
|
|
|
switch(AL_reg(context)) {
|
|
|
|
case 0x00: /* RETURN CONTROLLER INFORMATION */
|
|
TRACE( "VESA RETURN CONTROLLER INFORMATION\n" );
|
|
{
|
|
BYTE *ptr = CTX_SEG_OFF_TO_LIN(context,
|
|
context->SegEs,
|
|
context->Edi);
|
|
INT10_FillControllerInformation( ptr );
|
|
SET_AL( context, 0x4f );
|
|
SET_AH( context, 0x00 ); /* 0x00 = successful 0x01 = failed */
|
|
}
|
|
break;
|
|
|
|
case 0x01: /* RETURN MODE INFORMATION */
|
|
TRACE( "VESA RETURN MODE INFORMATION %04x\n", CX_reg(context) );
|
|
{
|
|
BYTE *ptr = CTX_SEG_OFF_TO_LIN(context,
|
|
context->SegEs,
|
|
context->Edi);
|
|
SET_AL( context, 0x4f );
|
|
if (INT10_FillModeInformation( ptr, CX_reg(context) ))
|
|
SET_AH( context, 0x00 ); /* status: success */
|
|
else
|
|
SET_AH( context, 0x01 ); /* status: failed */
|
|
}
|
|
break;
|
|
|
|
case 0x02: /* SET SuperVGA VIDEO MODE */
|
|
TRACE( "Set VESA video mode %04x\n", BX_reg(context) );
|
|
SET_AL( context, 0x4f ); /* function supported */
|
|
if (INT10_SetVideoMode( data, BX_reg(context) ))
|
|
SET_AH( context, 0x00 ); /* success */
|
|
else
|
|
SET_AH( context, 0x01 ); /* failed */
|
|
break;
|
|
|
|
case 0x03: /* VESA SuperVGA BIOS - GET CURRENT VIDEO MODE */
|
|
/*
|
|
* FIXME: This returns wrong value if current mode is VESA mode.
|
|
*/
|
|
SET_AL( context, 0x4f );
|
|
SET_AH( context, 0x00 ); /* should probly check if a vesa mode has ben set */
|
|
SET_BX( context, data->VideoMode );
|
|
break;
|
|
|
|
case 0x04: /* VESA SuperVGA BIOS - SAVE/RESTORE SuperVGA VIDEO STATE */
|
|
ERR("VESA SAVE/RESTORE Video State - Not Implemented\n");
|
|
/* AL_reg(context) = 0x4f; = supported so not set since not implemented */
|
|
/* maybe we should do this instead ? */
|
|
/* AH_reg(context = 0x01; not implemented so just fail */
|
|
break;
|
|
|
|
case 0x05: /* VESA SuperVGA BIOS - CPU VIDEO MEMORY CONTROL */
|
|
/*
|
|
* This subfunction supports only Window A (BL_reg == 0) and
|
|
* is assumes that window granularity is 64k.
|
|
*/
|
|
switch(BH_reg(context)) {
|
|
case 0x00: /* select video memory window */
|
|
SET_AL( context, 0x4f ); /* function supported */
|
|
if(BL_reg(context) == 0) {
|
|
VGA_SetWindowStart(DX_reg(context) * 64 * 1024);
|
|
SET_AH( context, 0x00 ); /* status: successful */
|
|
} else
|
|
SET_AH( context, 0x01 ); /* status: failed */
|
|
break;
|
|
case 0x01: /* get video memory window */
|
|
SET_AL( context, 0x4f ); /* function supported */
|
|
if(BL_reg(context) == 0) {
|
|
SET_DX( context, VGA_GetWindowStart() / 64 / 1024 );
|
|
SET_AH( context, 0x00 ); /* status: successful */
|
|
} else
|
|
SET_AH( context, 0x01 ); /* status: failed */
|
|
break;
|
|
default:
|
|
INT_BARF( context, 0x10 );
|
|
}
|
|
break;
|
|
|
|
case 0x06: /* VESA GET/SET LOGICAL SCAN LINE LENGTH */
|
|
ERR("VESA GET/SET LOGICAL SCAN LINE LENGTH - Not Implemented\n");
|
|
/* AL_reg(context) = 0x4f; = supported so not set since not implemented */
|
|
/* maybe we should do this instead ? */
|
|
/* AH_reg(context = 0x001; not implemented so just fail */
|
|
break;
|
|
|
|
case 0x07: /* GET/SET DISPLAY START */
|
|
ERR("VESA GET/SET DISPLAY START - Not Implemented\n");
|
|
/* AL_reg(context) = 0x4f; = supported so not set since not implemented */
|
|
/* maybe we should do this instead ? */
|
|
/* AH_reg(context = 0x001; not implemented so just fail */
|
|
break;
|
|
|
|
case 0x08: /* GET/SET DAC PALETTE CONTROL */
|
|
ERR("VESA GET/SET DAC PALETTE CONTROL- Not Implemented\n");
|
|
/* AL_reg(context) = 0x4f; = supported so not set since not implemented */
|
|
/* maybe we should do this instead ? */
|
|
/* AH_reg(context = 0x001; not implemented so just fail */
|
|
break;
|
|
|
|
case 0x09: /* SET PALETTE ENTRIES */
|
|
FIXME("VESA Set palette entries - not implemented\n");
|
|
break;
|
|
|
|
case 0x0a: /* GET PROTECTED-MODE CODE */
|
|
FIXME("VESA Get protected-mode code - not implemented\n");
|
|
break;
|
|
|
|
case 0x10: /* Display Power Management Extensions */
|
|
FIXME("VESA Display Power Management Extensions - not implemented\n");
|
|
break;
|
|
|
|
case 0xef: /* get video mode for hercules-compatibles */
|
|
/* There's no reason to really support this */
|
|
/* is there?....................(A.C.) */
|
|
TRACE("Just report the video not hercules compatible\n");
|
|
SET_DX( context, 0xffff );
|
|
break;
|
|
|
|
case 0xff: /* Turn VESA ON/OFF */
|
|
/* i dont know what to do */
|
|
break;
|
|
|
|
default:
|
|
FIXME("VESA Function (0x%x) - Not Supported\n", AH_reg(context));
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* DOSVM_Int10Handler
|
|
*
|
|
* Handler for int 10h (video).
|
|
*
|
|
* NOTE:
|
|
* Most INT 10 functions for text-mode, CGA, EGA, and VGA cards
|
|
* are present in this list. (SVGA and XGA are not) That is not
|
|
* to say that all these functions should be supported, but if
|
|
* anyone is brain-damaged enough to want to emulate one of these
|
|
* beasts then this should get you started.
|
|
*
|
|
* NOTE:
|
|
* Several common graphical extensions used by Microsoft hook
|
|
* off of here. I have *not* added them to this list (yet). They
|
|
* include:
|
|
*
|
|
* MSHERC.COM - More functionality for Hercules cards.
|
|
* EGA.SYS (also MOUSE.COM) - More for EGA cards.
|
|
*
|
|
* Yes, MS also added this support into their mouse driver. Don't
|
|
* ask me, I don't work for them.
|
|
*
|
|
* Joseph Pranevich - 9/98
|
|
*
|
|
* Jess Haas 2/99
|
|
* Added support for Vesa. It is not complete but is a start.
|
|
* NOTE: Im not sure if i did all this right or if eny of it works.
|
|
* Currently i dont have a program that uses Vesa that actually gets far
|
|
* enough without crashing to do vesa stuff.
|
|
*
|
|
* Added additional vga graphic support - 3/99
|
|
*/
|
|
void WINAPI DOSVM_Int10Handler( CONTEXT86 *context )
|
|
{
|
|
BIOSDATA *data = BIOS_DATA;
|
|
|
|
INT10_InitializeVideoMode( data );
|
|
|
|
switch(AH_reg(context)) {
|
|
|
|
case 0x00: /* SET VIDEO MODE */
|
|
TRACE( "Set VGA video mode %02x\n", AL_reg(context) );
|
|
if (!INT10_SetVideoMode( data, AL_reg(context) ))
|
|
FIXME( "Unsupported VGA video mode requested: %d\n",
|
|
AL_reg(context) );
|
|
break;
|
|
|
|
case 0x01: /* SET CURSOR SHAPE */
|
|
TRACE("Set Cursor Shape start %d end %d options %d\n",
|
|
CH_reg(context) & 0x1f,
|
|
CL_reg(context) & 0x1f,
|
|
CH_reg(context) & 0xe0);
|
|
data->VideoCursorType = CX_reg(context); /* direct copy */
|
|
VGA_SetCursorShape(CH_reg(context), CL_reg(context));
|
|
break;
|
|
|
|
case 0x02: /* SET CURSOR POSITION */
|
|
/* BH = Page Number */ /* Not supported */
|
|
/* DH = Row */ /* 0 is left */
|
|
/* DL = Column */ /* 0 is top */
|
|
INT10_SetCursorPos(data,BH_reg(context),DL_reg(context),DH_reg(context));
|
|
if (BH_reg(context))
|
|
{
|
|
FIXME("Set Cursor Position: Cannot set to page %d\n",
|
|
BH_reg(context));
|
|
}
|
|
else
|
|
{
|
|
VGA_SetCursorPos(DL_reg(context), DH_reg(context));
|
|
TRACE("Set Cursor Position: %d/%d\n", DL_reg(context),
|
|
DH_reg(context));
|
|
}
|
|
break;
|
|
|
|
case 0x03: /* GET CURSOR POSITION AND SIZE */
|
|
{
|
|
unsigned row, col;
|
|
|
|
TRACE("Get cursor position and size (page %d)\n", BH_reg(context));
|
|
SET_CX( context, data->VideoCursorType );
|
|
INT10_GetCursorPos(data,BH_reg(context),&col,&row);
|
|
SET_DH( context, row );
|
|
SET_DL( context, col );
|
|
TRACE("Cursor Position: %d/%d\n", DL_reg(context), DH_reg(context));
|
|
}
|
|
break;
|
|
|
|
case 0x04: /* READ LIGHT PEN POSITION */
|
|
FIXME("Read Light Pen Position - Not Supported\n");
|
|
SET_AH( context, 0x00 ); /* Not down */
|
|
break;
|
|
|
|
case 0x05: /* SELECT ACTIVE DISPLAY PAGE */
|
|
FIXME("Select Active Display Page (%d) - Not Supported\n", AL_reg(context));
|
|
data->VideoCurPage = AL_reg(context);
|
|
break;
|
|
|
|
case 0x06: /* SCROLL UP WINDOW */
|
|
/* AL = Lines to scroll */
|
|
/* BH = Attribute */
|
|
/* CH,CL = row, col upper-left */
|
|
/* DH,DL = row, col lower-right */
|
|
TRACE("Scroll Up Window %d\n", AL_reg(context));
|
|
|
|
if (AL_reg(context) == 0)
|
|
VGA_ClearText( CH_reg(context), CL_reg(context),
|
|
DH_reg(context), DL_reg(context),
|
|
BH_reg(context) );
|
|
else
|
|
VGA_ScrollUpText( CH_reg(context), CL_reg(context),
|
|
DH_reg(context), DL_reg(context),
|
|
AL_reg(context), BH_reg(context) );
|
|
break;
|
|
|
|
case 0x07: /* SCROLL DOWN WINDOW */
|
|
/* AL = Lines to scroll */
|
|
/* BH = Attribute */
|
|
/* CH,CL = row, col upper-left */
|
|
/* DH,DL = row, col lower-right */
|
|
TRACE("Scroll Down Window %d\n", AL_reg(context));
|
|
|
|
if (AL_reg(context) == 0)
|
|
VGA_ClearText( CH_reg(context), CL_reg(context),
|
|
DH_reg(context), DL_reg(context),
|
|
BH_reg(context) );
|
|
else
|
|
VGA_ScrollDownText( CH_reg(context), CL_reg(context),
|
|
DH_reg(context), DL_reg(context),
|
|
AL_reg(context), BH_reg(context) );
|
|
break;
|
|
|
|
case 0x08: /* READ CHARACTER AND ATTRIBUTE AT CURSOR POSITION */
|
|
{
|
|
if (BH_reg(context)) /* Write to different page */
|
|
{
|
|
FIXME("Read character and attribute at cursor position -"
|
|
" Can't read from non-0 page\n");
|
|
SET_AL( context, ' ' ); /* That page is blank */
|
|
SET_AH( context, 7 );
|
|
}
|
|
else
|
|
{
|
|
BYTE ascii, attr;
|
|
TRACE("Read Character and Attribute at Cursor Position\n");
|
|
VGA_GetCharacterAtCursor(&ascii, &attr);
|
|
SET_AL( context, ascii );
|
|
SET_AH( context, attr );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x09: /* WRITE CHARACTER AND ATTRIBUTE AT CURSOR POSITION */
|
|
case 0x0a: /* WRITE CHARACTER ONLY AT CURSOR POSITION */
|
|
/* AL = Character to display. */
|
|
/* BH = Page Number */ /* We can't write to non-0 pages, yet. */
|
|
/* BL = Attribute / Color */
|
|
/* CX = Times to Write Char */
|
|
/* Note here that the cursor is not advanced. */
|
|
{
|
|
unsigned row, col;
|
|
|
|
INT10_GetCursorPos(data,BH_reg(context),&col,&row);
|
|
VGA_WriteChars(col, row,
|
|
AL_reg(context),
|
|
(AH_reg(context) == 0x09) ? BL_reg(context) : -1,
|
|
CX_reg(context));
|
|
if (CX_reg(context) > 1)
|
|
TRACE("Write Character%s at Cursor Position (Rep. %d): %c\n",
|
|
(AH_reg(context) == 0x09) ? " and Attribute" : "",
|
|
CX_reg(context), AL_reg(context));
|
|
else
|
|
TRACE("Write Character%s at Cursor Position: %c\n",
|
|
(AH_reg(context) == 0x09) ? " and Attribute" : "",
|
|
AL_reg(context));
|
|
}
|
|
break;
|
|
|
|
case 0x0b:
|
|
switch BH_reg(context) {
|
|
case 0x00: /* SET BACKGROUND/BORDER COLOR */
|
|
/* In text modes, this sets only the border... */
|
|
/* According to the interrupt list and one of my books. */
|
|
/* Funny though that Beyond Zork seems to indicate that it
|
|
also sets up the default background attributes for clears
|
|
and scrolls... */
|
|
/* Bear in mind here that we do not want to change,
|
|
apparently, the foreground or attribute of the background
|
|
with this call, so we should check first to see what the
|
|
foreground already is... FIXME */
|
|
FIXME("Set Background/Border Color: %d/%d\n",
|
|
BH_reg(context), BL_reg(context));
|
|
break;
|
|
case 0x01: /* SET PALETTE */
|
|
FIXME("Set Palette - Not Supported\n");
|
|
break;
|
|
default:
|
|
FIXME("INT 10 AH = 0x0b BH = 0x%x - Unknown\n",
|
|
BH_reg(context));
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 0x0c: /* WRITE GRAPHICS PIXEL */
|
|
/* Not in graphics mode, can ignore w/o error */
|
|
FIXME("Write Graphics Pixel - Not Supported\n");
|
|
break;
|
|
|
|
case 0x0d: /* READ GRAPHICS PIXEL */
|
|
/* Not in graphics mode, can ignore w/o error */
|
|
FIXME("Read Graphics Pixel - Not Supported\n");
|
|
break;
|
|
|
|
case 0x0e: /* TELETYPE OUTPUT */
|
|
TRACE("Teletype Output\n");
|
|
DOSVM_PutChar(AL_reg(context));
|
|
break;
|
|
|
|
case 0x0f: /* GET CURRENT VIDEO MODE */
|
|
TRACE("Get current video mode: -> mode %d, columns %d\n", data->VideoMode, data->VideoColumns);
|
|
/* Note: This should not be a constant value. */
|
|
SET_AL( context, data->VideoMode );
|
|
SET_AH( context, data->VideoColumns );
|
|
SET_BH( context, 0 ); /* Display page 0 */
|
|
break;
|
|
|
|
case 0x10:
|
|
switch AL_reg(context) {
|
|
case 0x00: /* SET SINGLE PALETTE REGISTER - A.C. */
|
|
TRACE("Set Single Palette Register - Reg 0x0%x Value 0x0%x\n",
|
|
BL_reg(context),BH_reg(context));
|
|
/* BH is the value BL is the register */
|
|
VGA_SetColor16((int)BL_reg(context),(int)BH_reg(context));
|
|
break;
|
|
case 0x01: /* SET BORDER (OVERSCAN) */
|
|
/* Text terminals have no overscan */
|
|
/* I'm setting it anyway. - A.C. */
|
|
TRACE("Set Border (Overscan) - Ignored but set.\n");
|
|
VGA_SetColor16(16,(int)BH_reg(context));
|
|
break;
|
|
case 0x02: /* SET ALL PALETTE REGISTERS - A.C.*/
|
|
TRACE("Set all palette registers\n");
|
|
/* ES:DX points to a 17 byte table of colors */
|
|
/* No return data listed */
|
|
/* I'll have to update my table and the default palette */
|
|
VGA_Set16Palette(CTX_SEG_OFF_TO_LIN(context, context->SegEs, context->Edx));
|
|
break;
|
|
case 0x03: /* TOGGLE INTENSITY/BLINKING BIT */
|
|
FIXME("Toggle Intensity/Blinking Bit - Not Supported\n");
|
|
break;
|
|
case 0x07: /* GET INDIVIDUAL PALETTE REGISTER - A.C.*/
|
|
TRACE("Get Individual Palette Register 0x0%x\n",BL_reg(context));
|
|
/* BL is register to read [ 0-15 ] BH is return value */
|
|
SET_BH( context, VGA_GetColor16((int)BL_reg(context)) );
|
|
break;
|
|
case 0x08: /* READ OVERSCAN (BORDER COLOR) REGISTER - A.C. */
|
|
TRACE("Read Overscan (Border Color) Register \n");
|
|
SET_BH( context, VGA_GetColor16(16) );
|
|
break;
|
|
case 0x09: /* READ ALL PALETTE REGISTERS AND OVERSCAN REGISTER - A.C.*/
|
|
TRACE("Read All Palette Registers and Overscan Register \n");
|
|
/* ES:DX points to a 17 byte table where the results */
|
|
/* of this call should be stored. */
|
|
VGA_Get16Palette(CTX_SEG_OFF_TO_LIN(context, context->SegEs, context->Edx));
|
|
break;
|
|
case 0x10: /* SET INDIVIDUAL DAC REGISTER */
|
|
{
|
|
PALETTEENTRY paldat;
|
|
|
|
TRACE("Set Individual DAC register\n");
|
|
paldat.peRed = DH_reg(context);
|
|
paldat.peGreen = CH_reg(context);
|
|
paldat.peBlue = CL_reg(context);
|
|
paldat.peFlags = 0;
|
|
VGA_SetPalette(&paldat,BX_reg(context)&0xFF,1);
|
|
}
|
|
break;
|
|
case 0x12: /* SET BLOCK OF DAC REGISTERS */
|
|
{
|
|
int i;
|
|
PALETTEENTRY paldat;
|
|
BYTE *pt;
|
|
|
|
TRACE("Set Block of DAC registers\n");
|
|
pt = (BYTE*)CTX_SEG_OFF_TO_LIN(context,context->SegEs,context->Edx);
|
|
for (i=0;i<CX_reg(context);i++)
|
|
{
|
|
paldat.peRed = (*(pt+i*3+0)) << 2;
|
|
paldat.peGreen = (*(pt+i*3+1)) << 2;
|
|
paldat.peBlue = (*(pt+i*3+2)) << 2;
|
|
paldat.peFlags = 0;
|
|
VGA_SetPalette(&paldat,(BX_reg(context)+i)&0xFF,1);
|
|
}
|
|
}
|
|
break;
|
|
case 0x13: /* SELECT VIDEO DAC COLOR PAGE */
|
|
FIXME("Select video DAC color page - Not Supported\n");
|
|
break;
|
|
case 0x15: /* READ INDIVIDUAL DAC REGISTER */
|
|
FIXME("Read individual DAC register - Not Supported\n");
|
|
break;
|
|
case 0x17: /* READ BLOCK OF DAC REGISTERS */
|
|
FIXME("Read block of DAC registers - Not Supported\n");
|
|
break;
|
|
case 0x18: /* SET PEL MASK */
|
|
FIXME("Set PEL mask - Not Supported\n");
|
|
break;
|
|
case 0x19: /* READ PEL MASK */
|
|
FIXME("Read PEL mask - Not Supported\n");
|
|
break;
|
|
case 0x1a: /* GET VIDEO DAC COLOR PAGE STATE */
|
|
FIXME("Get video DAC color page state - Not Supported\n");
|
|
break;
|
|
case 0x1b: /* PERFORM GRAY-SCALE SUMMING */
|
|
FIXME("Perform Gray-scale summing - Not Supported\n");
|
|
break;
|
|
default:
|
|
FIXME("INT 10 AH = 0x10 AL = 0x%x - Unknown\n",
|
|
AL_reg(context));
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 0x11: /* TEXT MODE CHARGEN */
|
|
/* Note that second subfunction is *almost* identical. */
|
|
/* See INTERRUPT.A for details. */
|
|
switch AL_reg(context) {
|
|
case 0x00: /* LOAD USER SPECIFIED PATTERNS */
|
|
case 0x10:
|
|
FIXME("Load User Specified Patterns - Not Supported\n");
|
|
break;
|
|
case 0x01: /* LOAD ROM MONOCHROME PATTERNS */
|
|
case 0x11:
|
|
FIXME("Load ROM Monochrome Patterns - Not Supported\n");
|
|
break;
|
|
case 0x02: /* LOAD ROM 8x8 DOUBLE-DOT PATTERNS */
|
|
case 0x12:
|
|
FIXME(
|
|
"Load ROM 8x8 Double Dot Patterns - Not Supported\n");
|
|
break;
|
|
case 0x03: /* SET BLOCK SPECIFIER */
|
|
FIXME("Set Block Specifier - Not Supported\n");
|
|
break;
|
|
case 0x04: /* LOAD ROM 8x16 CHARACTER SET */
|
|
case 0x14:
|
|
FIXME("Load ROM 8x16 Character Set - Not Supported\n");
|
|
break;
|
|
case 0x20: /* SET USER 8x16 GRAPHICS CHARS */
|
|
FIXME("Set User 8x16 Graphics Chars - Not Supported\n");
|
|
break;
|
|
case 0x21: /* SET USER GRAPICS CHARACTERS */
|
|
FIXME("Set User Graphics Characters - Not Supported\n");
|
|
break;
|
|
case 0x22: /* SET ROM 8x14 GRAPHICS CHARS */
|
|
FIXME("Set ROM 8x14 Graphics Chars - Not Supported\n");
|
|
break;
|
|
case 0x23: /* SET ROM 8x8 DBL DOT CHARS */
|
|
FIXME(
|
|
"Set ROM 8x8 Dbl Dot Chars (Graphics) - Not Supported\n");
|
|
break;
|
|
case 0x24: /* LOAD 8x16 GRAPHIC CHARS */
|
|
FIXME("Load 8x16 Graphic Chars - Not Supported\n");
|
|
break;
|
|
case 0x30: /* GET FONT INFORMATION */
|
|
FIXME("Get Font Information - Not Supported\n");
|
|
break;
|
|
default:
|
|
FIXME("INT 10 AH = 0x11 AL = 0x%x - Unknown\n",
|
|
AL_reg(context));
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 0x12: /* ALTERNATE FUNCTION SELECT */
|
|
switch BL_reg(context) {
|
|
case 0x10: /* GET EGA INFO */
|
|
TRACE("EGA info requested\n");
|
|
SET_BH( context, 0x00 ); /* Color screen */
|
|
SET_BL( context, data->ModeOptions >> 5 ); /* EGA memory size */
|
|
SET_CX( context, data->FeatureBitsSwitches );
|
|
break;
|
|
case 0x20: /* ALTERNATE PRTSC */
|
|
FIXME("Install Alternate Print Screen - Not Supported\n");
|
|
break;
|
|
case 0x30: /* SELECT VERTICAL RESOULTION */
|
|
FIXME("Select vertical resolution - not supported\n");
|
|
break;
|
|
case 0x31: /* ENABLE/DISABLE DEFAULT PALETTE LOADING */
|
|
FIXME("Default palette loading - not supported\n");
|
|
data->VGASettings =
|
|
(data->VGASettings & 0xf7) |
|
|
((AL_reg(context) == 1) << 3);
|
|
break;
|
|
case 0x32: /* ENABLE/DISABLE VIDEO ADDRERSSING */
|
|
FIXME("Video Addressing - Not Supported\n");
|
|
break;
|
|
case 0x33: /* ENABLE/DISABLE GRAY SCALE SUMMING */
|
|
FIXME("Gray Scale Summing - Not Supported\n");
|
|
break;
|
|
case 0x34: /* ENABLE/DISABLE CURSOR EMULATION */
|
|
TRACE("Set cursor emulation to %d\n", AL_reg(context));
|
|
data->ModeOptions =
|
|
(data->ModeOptions & 0xfe)|(AL_reg(context) == 1);
|
|
break;
|
|
case 0x36: /* VIDEO ADDRESS CONTROL */
|
|
FIXME("Video Address Control - Not Supported\n");
|
|
break;
|
|
default:
|
|
FIXME("INT 10 AH = 0x11 AL = 0x%x - Unknown\n",
|
|
AL_reg(context));
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 0x13: /* WRITE STRING */
|
|
/* This one does not imply that string be at cursor. */
|
|
FIXME("Write String - Not Supported\n");
|
|
break;
|
|
|
|
case 0x1a:
|
|
switch AL_reg(context) {
|
|
case 0x00: /* GET DISPLAY COMBINATION CODE */
|
|
TRACE("Get Display Combination Code\n");
|
|
SET_AX( context, 0x001a );
|
|
SET_BL( context, 0x08 ); /* VGA w/ color analog display */
|
|
SET_BH( context, 0x00 ); /* No secondary hardware */
|
|
break;
|
|
case 0x01: /* SET DISPLAY COMBINATION CODE */
|
|
FIXME("Set Display Combination Code - Not Supported\n");
|
|
break;
|
|
default:
|
|
FIXME("INT 10 AH = 0x1a AL = 0x%x - Unknown\n",
|
|
AL_reg(context));
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case 0x1b: /* FUNCTIONALITY/STATE INFORMATION */
|
|
TRACE("Get functionality/state information\n");
|
|
if (BX_reg(context) == 0x0000)
|
|
{
|
|
BYTE *ptr = CTX_SEG_OFF_TO_LIN(context,
|
|
context->SegEs,
|
|
context->Edi);
|
|
SET_AL( context, 0x1b ); /* Function is supported */
|
|
INT10_FillStateInformation( ptr, data );
|
|
}
|
|
break;
|
|
|
|
case 0x1c: /* SAVE/RESTORE VIDEO STATE */
|
|
FIXME("Save/Restore Video State - Not Supported\n");
|
|
break;
|
|
|
|
case 0xef: /* get video mode for hercules-compatibles */
|
|
/* There's no reason to really support this */
|
|
/* is there?....................(A.C.) */
|
|
TRACE("Just report the video not hercules compatible\n");
|
|
SET_DX( context, 0xffff );
|
|
break;
|
|
|
|
case 0x4f: /* VESA */
|
|
INT10_HandleVESA(context);
|
|
break;
|
|
|
|
default:
|
|
FIXME("Unknown - 0x%x\n", AH_reg(context));
|
|
INT_BARF( context, 0x10 );
|
|
}
|
|
}
|
|
|
|
|
|
/**********************************************************************
|
|
* DOSVM_PutChar
|
|
*
|
|
* Write single character to VGA console at the current
|
|
* cursor position and updates the BIOS cursor position.
|
|
*/
|
|
void WINAPI DOSVM_PutChar( BYTE ascii )
|
|
{
|
|
BIOSDATA *data = BIOS_DATA;
|
|
unsigned xpos, ypos;
|
|
|
|
TRACE("char: 0x%02x(%c)\n", ascii, ascii);
|
|
|
|
INT10_InitializeVideoMode( data );
|
|
|
|
VGA_PutChar( ascii );
|
|
VGA_GetCursorPos( &xpos, &ypos );
|
|
INT10_SetCursorPos( data, 0, xpos, ypos );
|
|
}
|