2008-01-19 14:51:12 +00:00
|
|
|
/*
|
2019-12-14 17:05:45 +00:00
|
|
|
* PROJECT: ReactOS Console Text-Mode Device Driver
|
|
|
|
* LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
|
|
|
|
* PURPOSE: Loading specific fonts into VGA.
|
|
|
|
* COPYRIGHT: Copyright 2008-2019 Aleksey Bragin (aleksey@reactos.org)
|
|
|
|
* Copyright 2008-2019 Colin Finck (mail@colinfinck.de)
|
|
|
|
* Copyright 2008-2019 Christoph von Wittich (christoph_vw@reactos.org)
|
|
|
|
*/
|
2008-01-19 14:51:12 +00:00
|
|
|
|
|
|
|
/* INCLUDES ***************************************************************/
|
|
|
|
|
|
|
|
#include "blue.h"
|
2019-12-14 17:22:49 +00:00
|
|
|
#include <ndk/rtlfuncs.h>
|
2014-02-03 10:33:35 +00:00
|
|
|
|
2008-01-19 14:51:12 +00:00
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
|
|
|
|
2019-12-14 17:05:45 +00:00
|
|
|
NTSTATUS ExtractFont(_In_ ULONG CodePage, _In_ PUCHAR FontBitField);
|
|
|
|
VOID OpenBitPlane(VOID);
|
|
|
|
VOID CloseBitPlane(VOID);
|
|
|
|
VOID LoadFont(_In_ PUCHAR Bitplane, _In_ PUCHAR FontBitfield);
|
2008-01-19 14:51:12 +00:00
|
|
|
|
|
|
|
/* FUNCTIONS ****************************************************************/
|
|
|
|
|
|
|
|
VOID
|
2018-07-01 08:13:18 +00:00
|
|
|
ScrLoadFontTable(
|
2019-12-14 17:05:45 +00:00
|
|
|
_In_ ULONG CodePage)
|
2008-01-19 14:51:12 +00:00
|
|
|
{
|
|
|
|
PHYSICAL_ADDRESS BaseAddress;
|
2008-11-25 17:11:42 +00:00
|
|
|
PUCHAR Bitplane;
|
2008-01-21 13:33:52 +00:00
|
|
|
PUCHAR FontBitfield = NULL;
|
2008-01-21 21:01:52 +00:00
|
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
2008-01-19 14:51:12 +00:00
|
|
|
|
2018-07-01 08:13:18 +00:00
|
|
|
FontBitfield = (PUCHAR)ExAllocatePoolWithTag(NonPagedPool, 2048, TAG_BLUE);
|
2018-07-01 08:24:22 +00:00
|
|
|
if (FontBitfield == NULL)
|
2008-01-21 13:33:52 +00:00
|
|
|
{
|
2018-07-01 08:24:22 +00:00
|
|
|
DPRINT1("ExAllocatePoolWithTag failed\n");
|
|
|
|
return;
|
|
|
|
}
|
2008-01-21 21:01:52 +00:00
|
|
|
|
2018-07-01 08:24:22 +00:00
|
|
|
/* open bit plane for font table access */
|
|
|
|
OpenBitPlane();
|
2008-02-05 19:31:05 +00:00
|
|
|
|
2018-07-01 08:24:22 +00:00
|
|
|
/* get pointer to video memory */
|
|
|
|
BaseAddress.QuadPart = BITPLANE_BASE;
|
|
|
|
Bitplane = (PUCHAR)MmMapIoSpace(BaseAddress, 0xFFFF, MmNonCached);
|
2008-01-21 21:01:52 +00:00
|
|
|
|
2018-07-01 08:24:22 +00:00
|
|
|
Status = ExtractFont(CodePage, FontBitfield);
|
|
|
|
if (NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
LoadFont(Bitplane, FontBitfield);
|
2008-01-21 13:33:52 +00:00
|
|
|
}
|
2018-07-01 08:24:22 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
DPRINT1("ExtractFont failed with Status 0x%lx\n", Status);
|
|
|
|
}
|
|
|
|
|
|
|
|
MmUnmapIoSpace(Bitplane, 0xFFFF);
|
2019-12-15 01:41:42 +00:00
|
|
|
ExFreePoolWithTag(FontBitfield, TAG_BLUE);
|
2018-07-01 08:24:22 +00:00
|
|
|
|
|
|
|
/* close bit plane */
|
2020-04-25 19:34:16 +00:00
|
|
|
CloseBitPlane();
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
ScrSetFont(
|
|
|
|
_In_ PUCHAR FontBitfield)
|
|
|
|
{
|
|
|
|
PHYSICAL_ADDRESS BaseAddress;
|
|
|
|
PUCHAR Bitplane;
|
|
|
|
|
|
|
|
/* open bit plane for font table access */
|
|
|
|
OpenBitPlane();
|
|
|
|
|
|
|
|
/* get pointer to video memory */
|
|
|
|
BaseAddress.QuadPart = BITPLANE_BASE;
|
|
|
|
Bitplane = (PUCHAR)MmMapIoSpace(BaseAddress, 0xFFFF, MmNonCached);
|
|
|
|
|
|
|
|
LoadFont(Bitplane, FontBitfield);
|
|
|
|
|
|
|
|
MmUnmapIoSpace(Bitplane, 0xFFFF);
|
|
|
|
|
|
|
|
/* close bit plane */
|
2018-07-01 08:24:22 +00:00
|
|
|
CloseBitPlane();
|
2008-01-19 14:51:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* PRIVATE FUNCTIONS *********************************************************/
|
|
|
|
|
2018-07-01 08:13:18 +00:00
|
|
|
NTSTATUS
|
|
|
|
ExtractFont(
|
2019-12-14 17:05:45 +00:00
|
|
|
_In_ ULONG CodePage,
|
|
|
|
_In_ PUCHAR FontBitField)
|
2008-01-21 21:01:52 +00:00
|
|
|
{
|
2008-02-05 19:31:05 +00:00
|
|
|
BOOLEAN bFoundFile = FALSE;
|
2008-01-21 21:01:52 +00:00
|
|
|
HANDLE Handle;
|
2008-02-05 19:31:05 +00:00
|
|
|
NTSTATUS Status;
|
|
|
|
CHAR FileName[20];
|
2008-01-21 21:01:52 +00:00
|
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
|
|
UNICODE_STRING LinkName;
|
|
|
|
UNICODE_STRING SourceName;
|
2008-02-05 19:31:05 +00:00
|
|
|
CFHEADER CabFileHeader;
|
|
|
|
CFFILE CabFile;
|
|
|
|
ULONG CabFileOffset = 0;
|
2008-01-21 21:01:52 +00:00
|
|
|
LARGE_INTEGER ByteOffset;
|
2018-07-01 08:13:18 +00:00
|
|
|
WCHAR SourceBuffer[MAX_PATH] = { L'\0' };
|
2011-11-12 12:23:15 +00:00
|
|
|
ULONG ReadCP;
|
2008-01-21 21:01:52 +00:00
|
|
|
|
2018-07-01 08:13:18 +00:00
|
|
|
if (KeGetCurrentIrql() != PASSIVE_LEVEL)
|
2008-02-05 19:31:05 +00:00
|
|
|
return STATUS_INVALID_DEVICE_STATE;
|
2008-01-21 21:01:52 +00:00
|
|
|
|
|
|
|
RtlInitUnicodeString(&LinkName,
|
|
|
|
L"\\SystemRoot");
|
|
|
|
|
|
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
|
|
&LinkName,
|
2015-09-19 21:10:11 +00:00
|
|
|
OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
|
2008-01-21 21:01:52 +00:00
|
|
|
NULL,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
Status = ZwOpenSymbolicLinkObject(&Handle,
|
|
|
|
SYMBOLIC_LINK_ALL_ACCESS,
|
|
|
|
&ObjectAttributes);
|
|
|
|
|
|
|
|
if (!NT_SUCCESS(Status))
|
2018-07-01 08:24:22 +00:00
|
|
|
{
|
|
|
|
DPRINT1("ZwOpenSymbolicLinkObject failed with Status 0x%lx\n", Status);
|
|
|
|
return Status;
|
|
|
|
}
|
2008-01-21 21:01:52 +00:00
|
|
|
|
|
|
|
SourceName.Length = 0;
|
2011-11-12 12:23:15 +00:00
|
|
|
SourceName.MaximumLength = MAX_PATH * sizeof(WCHAR);
|
2008-01-21 21:01:52 +00:00
|
|
|
SourceName.Buffer = SourceBuffer;
|
|
|
|
|
|
|
|
Status = ZwQuerySymbolicLinkObject(Handle,
|
2018-07-01 08:13:18 +00:00
|
|
|
&SourceName,
|
|
|
|
NULL);
|
2008-01-21 21:01:52 +00:00
|
|
|
ZwClose(Handle);
|
|
|
|
|
2018-07-01 08:24:22 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
DPRINT1("ZwQuerySymbolicLinkObject failed with Status 0x%lx\n", Status);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2008-02-05 19:31:05 +00:00
|
|
|
Status = RtlAppendUnicodeToString(&SourceName, L"\\vgafonts.cab");
|
2018-07-01 08:24:22 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
DPRINT1("RtlAppendUnicodeToString failed with Status 0x%lx\n", Status);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2018-07-01 08:13:18 +00:00
|
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
|
|
&SourceName,
|
2008-01-21 21:01:52 +00:00
|
|
|
OBJ_CASE_INSENSITIVE | OBJ_KERNEL_HANDLE,
|
2018-07-01 08:13:18 +00:00
|
|
|
NULL,
|
|
|
|
NULL);
|
2008-01-21 21:01:52 +00:00
|
|
|
|
|
|
|
Status = ZwCreateFile(&Handle,
|
|
|
|
GENERIC_READ,
|
2018-07-01 08:13:18 +00:00
|
|
|
&ObjectAttributes,
|
|
|
|
&IoStatusBlock,
|
|
|
|
NULL,
|
2008-01-21 21:01:52 +00:00
|
|
|
FILE_ATTRIBUTE_NORMAL,
|
|
|
|
0,
|
|
|
|
FILE_OPEN,
|
|
|
|
FILE_SYNCHRONOUS_IO_NONALERT,
|
2018-07-01 08:13:18 +00:00
|
|
|
NULL,
|
|
|
|
0);
|
2018-07-01 08:24:22 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
DPRINT1("Error: Cannot open vgafonts.cab (0x%lx)\n", Status);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2018-07-01 08:47:48 +00:00
|
|
|
ByteOffset.QuadPart = 0;
|
2018-07-01 08:24:22 +00:00
|
|
|
Status = ZwReadFile(Handle,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&IoStatusBlock,
|
|
|
|
&CabFileHeader,
|
|
|
|
sizeof(CabFileHeader),
|
|
|
|
&ByteOffset,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
DPRINT1("Error: Cannot read from file (0x%lx)\n", Status);
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CabFileHeader.Signature != CAB_SIGNATURE)
|
|
|
|
{
|
2018-07-01 08:47:48 +00:00
|
|
|
DPRINT1("Invalid CAB signature: 0x%lx!\n", CabFileHeader.Signature);
|
2018-07-01 08:24:22 +00:00
|
|
|
Status = STATUS_UNSUCCESSFUL;
|
|
|
|
goto Exit;
|
|
|
|
}
|
|
|
|
|
|
|
|
// We have a valid CAB file!
|
|
|
|
// Read the file table now and decrement the file count on every file. When it's zero, we read the complete table.
|
2018-07-01 08:47:48 +00:00
|
|
|
ByteOffset.QuadPart = CabFileHeader.FileTableOffset;
|
2018-07-01 08:24:22 +00:00
|
|
|
|
|
|
|
while (CabFileHeader.FileCount)
|
2008-01-22 09:42:37 +00:00
|
|
|
{
|
2018-07-01 08:13:18 +00:00
|
|
|
Status = ZwReadFile(Handle,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&IoStatusBlock,
|
2018-07-01 08:24:22 +00:00
|
|
|
&CabFile,
|
|
|
|
sizeof(CabFile),
|
2018-07-01 08:13:18 +00:00
|
|
|
&ByteOffset,
|
|
|
|
NULL);
|
2008-02-05 19:31:05 +00:00
|
|
|
|
2018-07-01 08:13:18 +00:00
|
|
|
if (NT_SUCCESS(Status))
|
2008-01-21 21:01:52 +00:00
|
|
|
{
|
2018-07-01 08:47:48 +00:00
|
|
|
ByteOffset.QuadPart += sizeof(CabFile);
|
2018-07-01 08:24:22 +00:00
|
|
|
|
|
|
|
// We assume here that the file name is max. 19 characters (+ 1 NULL character) long.
|
|
|
|
// This should be enough for our purpose.
|
|
|
|
Status = ZwReadFile(Handle,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&IoStatusBlock,
|
|
|
|
FileName,
|
|
|
|
sizeof(FileName),
|
|
|
|
&ByteOffset,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
if (NT_SUCCESS(Status))
|
2008-01-21 21:01:52 +00:00
|
|
|
{
|
2018-07-01 08:24:22 +00:00
|
|
|
if (!bFoundFile)
|
2008-01-21 21:01:52 +00:00
|
|
|
{
|
2018-07-01 08:24:22 +00:00
|
|
|
Status = RtlCharToInteger(FileName, 0, &ReadCP);
|
|
|
|
if (NT_SUCCESS(Status) && ReadCP == CodePage)
|
2008-01-22 09:42:37 +00:00
|
|
|
{
|
2018-07-01 08:24:22 +00:00
|
|
|
// We got the correct file.
|
|
|
|
// Save the offset and loop through the rest of the file table to find the position, where the actual data starts.
|
|
|
|
CabFileOffset = CabFile.FileOffset;
|
|
|
|
bFoundFile = TRUE;
|
2008-01-22 09:42:37 +00:00
|
|
|
}
|
2008-01-21 21:01:52 +00:00
|
|
|
}
|
2008-02-05 19:31:05 +00:00
|
|
|
|
2018-07-01 08:47:48 +00:00
|
|
|
ByteOffset.QuadPart += strlen(FileName) + 1;
|
2008-01-21 21:01:52 +00:00
|
|
|
}
|
|
|
|
}
|
2008-02-05 19:31:05 +00:00
|
|
|
|
2018-07-01 08:24:22 +00:00
|
|
|
CabFileHeader.FileCount--;
|
2008-01-21 21:01:52 +00:00
|
|
|
}
|
2018-07-01 08:24:22 +00:00
|
|
|
|
|
|
|
// 8 = Size of a CFFOLDER structure (see cabman). As we don't need the values of that structure, just increase the offset here.
|
2018-07-01 08:47:48 +00:00
|
|
|
ByteOffset.QuadPart += 8;
|
|
|
|
ByteOffset.QuadPart += CabFileOffset;
|
2018-07-01 08:24:22 +00:00
|
|
|
|
|
|
|
// ByteOffset now contains the offset of the actual data, so we can read the RAW font
|
|
|
|
Status = ZwReadFile(Handle,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
NULL,
|
|
|
|
&IoStatusBlock,
|
|
|
|
FontBitField,
|
|
|
|
2048,
|
|
|
|
&ByteOffset,
|
|
|
|
NULL);
|
|
|
|
if (!NT_SUCCESS(Status))
|
2008-01-21 21:01:52 +00:00
|
|
|
{
|
2018-07-01 08:24:22 +00:00
|
|
|
DPRINT1("ZwReadFile failed with Status 0x%lx\n", Status);
|
2008-01-21 21:01:52 +00:00
|
|
|
}
|
2018-07-01 08:24:22 +00:00
|
|
|
|
|
|
|
Exit:
|
|
|
|
|
|
|
|
ZwClose(Handle);
|
|
|
|
return Status;
|
2008-01-21 21:01:52 +00:00
|
|
|
}
|
|
|
|
|
2008-01-19 14:51:12 +00:00
|
|
|
/* Font-load specific funcs */
|
|
|
|
VOID
|
2015-09-04 00:21:03 +00:00
|
|
|
OpenBitPlane(VOID)
|
2008-01-19 14:51:12 +00:00
|
|
|
{
|
|
|
|
/* disable interrupts */
|
2008-01-19 16:36:54 +00:00
|
|
|
_disable();
|
2008-01-19 14:51:12 +00:00
|
|
|
|
|
|
|
/* sequence reg */
|
2018-07-01 08:13:18 +00:00
|
|
|
WRITE_PORT_UCHAR(SEQ_COMMAND, SEQ_RESET); WRITE_PORT_UCHAR(SEQ_DATA, 0x01);
|
|
|
|
WRITE_PORT_UCHAR(SEQ_COMMAND, SEQ_ENABLE_WRT_PLANE); WRITE_PORT_UCHAR(SEQ_DATA, 0x04);
|
|
|
|
WRITE_PORT_UCHAR(SEQ_COMMAND, SEQ_MEM_MODE); WRITE_PORT_UCHAR(SEQ_DATA, 0x07);
|
|
|
|
WRITE_PORT_UCHAR(SEQ_COMMAND, SEQ_RESET); WRITE_PORT_UCHAR(SEQ_DATA, 0x03);
|
2008-01-19 14:51:12 +00:00
|
|
|
|
|
|
|
/* graphic reg */
|
2018-07-01 08:13:18 +00:00
|
|
|
WRITE_PORT_UCHAR(GCT_COMMAND, GCT_READ_PLANE); WRITE_PORT_UCHAR(GCT_DATA, 0x02);
|
|
|
|
WRITE_PORT_UCHAR(GCT_COMMAND, GCT_RW_MODES); WRITE_PORT_UCHAR(GCT_DATA, 0x00);
|
|
|
|
WRITE_PORT_UCHAR(GCT_COMMAND, GCT_GRAPH_MODE); WRITE_PORT_UCHAR(GCT_DATA, 0x00);
|
2008-01-19 14:51:12 +00:00
|
|
|
|
|
|
|
/* enable interrupts */
|
2008-01-19 16:36:54 +00:00
|
|
|
_enable();
|
2008-01-19 14:51:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
2015-09-04 00:21:03 +00:00
|
|
|
CloseBitPlane(VOID)
|
2008-01-19 14:51:12 +00:00
|
|
|
{
|
|
|
|
/* disable interrupts */
|
2008-01-19 16:36:54 +00:00
|
|
|
_disable();
|
2008-01-19 14:51:12 +00:00
|
|
|
|
|
|
|
/* sequence reg */
|
2018-07-01 08:13:18 +00:00
|
|
|
WRITE_PORT_UCHAR(SEQ_COMMAND, SEQ_RESET); WRITE_PORT_UCHAR(SEQ_DATA, 0x01);
|
|
|
|
WRITE_PORT_UCHAR(SEQ_COMMAND, SEQ_ENABLE_WRT_PLANE); WRITE_PORT_UCHAR(SEQ_DATA, 0x03);
|
|
|
|
WRITE_PORT_UCHAR(SEQ_COMMAND, SEQ_MEM_MODE); WRITE_PORT_UCHAR(SEQ_DATA, 0x03);
|
|
|
|
WRITE_PORT_UCHAR(SEQ_COMMAND, SEQ_RESET); WRITE_PORT_UCHAR(SEQ_DATA, 0x03);
|
2008-01-19 14:51:12 +00:00
|
|
|
|
|
|
|
/* graphic reg */
|
2018-07-01 08:13:18 +00:00
|
|
|
WRITE_PORT_UCHAR(GCT_COMMAND, GCT_READ_PLANE); WRITE_PORT_UCHAR(GCT_DATA, 0x00);
|
|
|
|
WRITE_PORT_UCHAR(GCT_COMMAND, GCT_RW_MODES); WRITE_PORT_UCHAR(GCT_DATA, 0x10);
|
|
|
|
WRITE_PORT_UCHAR(GCT_COMMAND, GCT_GRAPH_MODE); WRITE_PORT_UCHAR(GCT_DATA, 0x0e);
|
2008-01-19 14:51:12 +00:00
|
|
|
|
|
|
|
/* enable interrupts */
|
2008-01-19 16:36:54 +00:00
|
|
|
_enable();
|
2008-01-19 14:51:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
2018-07-01 08:13:18 +00:00
|
|
|
LoadFont(
|
2019-12-14 17:05:45 +00:00
|
|
|
_In_ PUCHAR Bitplane,
|
|
|
|
_In_ PUCHAR FontBitfield)
|
2008-01-19 14:51:12 +00:00
|
|
|
{
|
2018-07-01 08:13:18 +00:00
|
|
|
UINT32 i, j;
|
2008-01-19 14:51:12 +00:00
|
|
|
|
2018-07-01 08:13:18 +00:00
|
|
|
for (i = 0; i < 256; i++)
|
2008-01-19 14:51:12 +00:00
|
|
|
{
|
2018-07-01 08:13:18 +00:00
|
|
|
for (j = 0; j < 8; j++)
|
2008-01-19 14:51:12 +00:00
|
|
|
{
|
2018-07-01 08:13:18 +00:00
|
|
|
*Bitplane = FontBitfield[i * 8 + j];
|
2008-01-19 14:51:12 +00:00
|
|
|
Bitplane++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// padding
|
2018-07-01 08:13:18 +00:00
|
|
|
for (j = 8; j < 32; j++)
|
2008-01-19 14:51:12 +00:00
|
|
|
{
|
|
|
|
*Bitplane = 0;
|
|
|
|
Bitplane++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|