diff options
Diffstat (limited to 'com32/lib/sys/vesa/initvesa.c')
-rw-r--r-- | com32/lib/sys/vesa/initvesa.c | 149 |
1 files changed, 149 insertions, 0 deletions
diff --git a/com32/lib/sys/vesa/initvesa.c b/com32/lib/sys/vesa/initvesa.c new file mode 100644 index 00000000..43b34985 --- /dev/null +++ b/com32/lib/sys/vesa/initvesa.c @@ -0,0 +1,149 @@ +/* ----------------------------------------------------------------------- * + * + * Copyright 1999-2006 H. Peter Anvin - All Rights Reserved + * + * Permission is hereby granted, free of charge, to any person + * obtaining a copy of this software and associated documentation + * files (the "Software"), to deal in the Software without + * restriction, including without limitation the rights to use, + * copy, modify, merge, publish, distribute, sublicense, and/or + * sell copies of the Software, and to permit persons to whom + * the Software is furnished to do so, subject to the following + * conditions: + * + * The above copyright notice and this permission notice shall + * be included in all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES + * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT + * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, + * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR + * OTHER DEALINGS IN THE SOFTWARE. + * + * ----------------------------------------------------------------------- */ + +/* + * initvesa.c + * + * Query the VESA BIOS and select a 640x480x32 mode with local mapping + * support, if one exists. + */ + +#include <inttypes.h> +#include <com32.h> +#include <string.h> +#include "vesa.h" +#include "video.h" + +struct vesa_general_info __vesa_general_info; +struct vesa_mode_info __vesa_mode_info; + +uint8_t __vesacon_graphics_font[FONT_MAX_CHARS][FONT_MAX_HEIGHT]; +uint32_t __vesacon_background[VIDEO_Y_SIZE][VIDEO_X_SIZE]; +uint32_t __vesacon_shadowfb[VIDEO_Y_SIZE][VIDEO_X_SIZE]; + +static void unpack_font(uint8_t *dst, uint8_t *src, int height) +{ + int i; + + for (i = 0; i < FONT_MAX_CHARS; i++) { + memcpy(dst, src, height); + memset(dst+height, 0, 32-height); + + dst += 32; + src += height; + } +} + +static int vesacon_set_mode(void) +{ + com32sys_t rm; + uint8_t *rom_font; + uint16_t mode, *mode_ptr; + struct vesa_general_info *gi; + struct vesa_mode_info *mi; + + /* Allocate space in the bounce buffer for these structures */ + gi = &((struct vesa_info *)__com32.cs_bounce)->gi; + mi = &((struct vesa_info *)__com32.cs_bounce)->mi; + + memset(&rm, 0, sizeof rm); + + gi->signature = VBE2_MAGIC; /* Get VBE2 extended data */ + rm.eax.w[0] = 0x4F00; /* Get SVGA general information */ + rm.edi.w[0] = OFFS(gi); + rm.es = SEG(gi); + __intcall(0x10, &rm, &rm); + + if ( rm.eax.w[0] != 0x004F ) + return 1; /* Function call failed */ + if ( gi->signature != VESA_MAGIC ) + return 2; /* No magic */ + if ( gi->version < 0x0200 ) { + return 3; /* VESA 2.0 not supported */ + } + + /* Search for a 640x480 32-bit linear frame buffer mode */ + mode_ptr = CVT_PTR(gi->video_mode_ptr); + + for(;;) { + if ((mode = *mode_ptr++) == 0xFFFF) + return 4; /* No mode found */ + + rm.eax.w[0] = 0x4F01; /* Get SVGA mode information */ + rm.ecx.w[0] = mode; + rm.edi.w[0] = OFFS(mi); + rm.es = SEG(mi); + __intcall(0x10, &rm, &rm); + + /* Must be a supported mode */ + if ( rm.eax.w[0] != 0x004f ) + continue; + /* Must be an LFB color graphics mode supported by the hardware */ + if ( (mi->mode_attr & 0x0099) != 0x0099 ) + continue; + /* Must be 640x480, 32 bpp */ + if ( mi->h_res != VIDEO_X_SIZE || mi->v_res != VIDEO_Y_SIZE || + mi->bpp != 32 ) + continue; + /* Must either be a packed-pixel mode or a direct color mode + (depending on VESA version ) */ + if ( mi->memory_layout != 4 && /* Packed pixel */ + (mi->memory_layout != 6 || mi->rpos != 24 || + mi->gpos != 16 || mi->bpos != 0) ) + continue; + + /* Hey, looks like we found something we can live with */ + break; + } + + /* Download the BIOS-provided font */ + rm.eax.w[0] = 0x1130; /* Get Font Information */ + rm.ebx.w[0] = 0x0600; /* Get 8x16 ROM font */ + __intcall(0x10, &rm, &rm); + rom_font = MK_PTR(rm.es, rm.ebp.w[0]); + unpack_font(graphics_font, rom_font, 16); + + /* Now set video mode */ + rm.eax.w[0] = 0x4F02; /* Set SVGA video mode */ + rm.ebx.w[0] = mode | 0xC000; /* Don't clear video RAM, use linear fb */ + __intcall(0x10, &rm, &rm); + if ( rm.eax.w[0] != 0x004F ) { + rm.eax.w[0] = 0x0003; /* Set regular text mode */ + __intcall(0x10, &rm, NULL); + return 9; /* Failed to set mode */ + } + + /* Copy established state out of the bounce buffer */ + memcpy(&__vesa_info, __com32.cs_bounce, sizeof __vesa_info); + + return 0; +} + +int __vesacon_init(void) +{ + return vesacon_set_mode(); +} |