/* ----------------------------------------------------------------------- *
*
* Copyright 1994-2008 H. Peter Anvin - All Rights Reserved
* Copyright 2013 Intel Corporation
*
* 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, Inc., 53 Temple Place Ste 330,
* Boston MA 02111-1307, USA; either version 2 of the License, or
* (at your option) any later version; incorporated herein by reference.
*
* ----------------------------------------------------------------------- */
/*
*
* font.c
*
* VGA font handling code
*
*/
#include <syslinux/firmware.h>
#include <syslinux/video.h>
#include <sys/io.h>
#include <stdio.h>
#include <fs.h>
#include "bios.h"
#include "graphics.h"
#include "core.h"
__export uint8_t UserFont = 0; /* Using a user-specified font */
__export __lowmem char fontbuf[8192];
uint16_t GXPixCols = 1; /* Graphics mode pixel columns */
uint16_t GXPixRows = 1; /* Graphics mode pixel rows */
/*
* loadfont: Load a .psf font file and install it onto the VGA console
* (if we're not on a VGA screen then ignore.)
*/
__export void loadfont(const char *filename)
{
struct psfheader {
uint16_t magic;
uint8_t mode;
uint8_t height;
} hdr;
FILE *f;
f = fopen(filename, "r");
if (!f)
return;
/* Read header */
if (_fread(&hdr, sizeof hdr, f) != sizeof hdr)
goto fail;
/* Magic number */
if (hdr.magic != 0x0436)
goto fail;
/* File mode: font modes 0-5 supported */
if (hdr.mode > 5)
goto fail;
/* VGA minimum/maximum */
if (hdr.height < 2 || hdr.height > 32)
goto fail;
/* Load the actual font into the font buffer. */
memset(fontbuf, 0, 256*32);
if (_fread(fontbuf, 256*hdr.height, f) != 256*hdr.height)
goto fail;
/* Loaded OK */
VGAFontSize = hdr.height;
UserFont = 1; /* Set font flag */
use_font();
fail:
fclose(f);
}
/*
* use_font:
* This routine activates whatever font happens to be in the
* vgafontbuf, and updates the bios_adjust_screen data.
* Must be called with CS = DS
*/
void use_font(void)
{
com32sys_t ireg, oreg;
uint8_t bytes = VGAFontSize;
/* Nonstandard mode? */
if (UsingVGA & ~0x3)
syslinux_force_text_mode();
memset(&ireg, 0, sizeof(ireg));
ireg.es = SEG(fontbuf);
ireg.ebp.w[0] = OFFS(fontbuf); /* ES:BP -> font */
/* Are we using a user-specified font? */
if (UserFont & 0x1) {
/* Are we in graphics mode? */
if (UsingVGA & 0x1) {
uint8_t rows;
rows = GXPixRows / bytes;
VidRows = rows - 1;
/* Set user character table */
ireg.eax.w[0] = 0x1121;
ireg.ebx.b[0] = 0;
ireg.ecx.b[0] = bytes; /* bytes/character */
ireg.edx.b[0] = rows;
__intcall(0x10, &ireg, &oreg);
/* 8 pixels/character */
VidCols = ((GXPixCols >> 3) - 1);
/* No need to call bios_adjust_screen */
return;
} else {
ireg.eax.w[0] = 0x1110; /* Load into VGA RAM */
ireg.ebx.b[0] = 0;
ireg.ebx.b[1] = bytes; /* bytes/character */
ireg.ecx.w[0] = 256;
ireg.edx.w[0] = 0;
__intcall(0x10, &ireg, &oreg);
memset(&ireg, 0, sizeof(ireg));
ireg.ebx.b[0] = 0;
ireg.eax.w[0] = 0x1103; /* Select page 0 */
__intcall(0x10, &ireg, NULL);
}
}
bios_adjust_screen();
}
/*
* bios_adjust_screen: Set the internal variables associated with the screen size.
* This is a subroutine in case we're loading a custom font.
*/
void bios_adjust_screen(void)
{
com32sys_t ireg, oreg;
volatile uint8_t *vidrows = (volatile uint8_t *)BIOS_vidrows;
uint8_t rows, cols;
memset(&ireg, 0, sizeof(ireg));
rows = *vidrows;
if (!rows) {
/*
* No vidrows in BIOS, assume 25.
* (Remember: vidrows == rows-1)
*/
rows = 24;
}
VidRows = rows;
ireg.eax.b[1] = 0x0f; /* Read video state */
__intcall(0x10, &ireg, &oreg);
cols = oreg.eax.b[1];
VidCols = --cols; /* Store count-1 (same as rows) */
}
void adjust_screen(void)
{
if (firmware->adjust_screen)
firmware->adjust_screen();
}
void pm_adjust_screen(com32sys_t *regs __unused)
{
adjust_screen();
}
void pm_userfont(com32sys_t *regs)
{
regs->es = SEG(fontbuf);
regs->ebx.w[0] = OFFS(fontbuf);
}