mirror of
https://github.com/reactos/reactos.git
synced 2024-12-28 18:15:11 +00:00
1322 lines
41 KiB
C
1322 lines
41 KiB
C
/*
|
|
* PROJECT: ReactOS Build Tools [Keyboard Layout Compiler]
|
|
* LICENSE: BSD - See COPYING.BSD in the top level directory
|
|
* FILE: tools/kbdtool/output.c
|
|
* PURPOSE: Output Logic (Source Builder)
|
|
* PROGRAMMERS: ReactOS Foundation
|
|
*/
|
|
|
|
/* INCLUDES *******************************************************************/
|
|
|
|
#include "kbdtool.h"
|
|
|
|
/* GLOBALS ********************************************************************/
|
|
|
|
ULONG gStringIdForDescriptions = 1000;
|
|
ULONG gStringIdForLanguageNames = 1100;
|
|
ULONG gStringIdForLocaleName = 1200;
|
|
time_t Clock;
|
|
struct tm *Now;
|
|
CHAR gCharName[32];
|
|
|
|
/* FUNCTIONS ******************************************************************/
|
|
|
|
PCHAR
|
|
WChName(IN ULONG Char,
|
|
IN BOOLEAN AddZero)
|
|
{
|
|
PCHAR p;
|
|
|
|
/* Check for invalid character */
|
|
if (Char == -1)
|
|
{
|
|
/* No name */
|
|
strcpy(gCharName, "WCH_NONE");
|
|
}
|
|
else if ((Char > 31) && (Char < 128))
|
|
{
|
|
/* Use our global buffer */
|
|
p = gCharName;
|
|
|
|
/* Add the first quote unless this was a zero-string */
|
|
if (!AddZero) *p++ = '\'';
|
|
|
|
/* Now replace any other quote or comment character with a slash */
|
|
if ((Char == '\"') || (Char == '\'') || (Char == '\\')) *p++ = '\\';
|
|
|
|
/* Now plug in the character */
|
|
*p++ = Char;
|
|
|
|
/* And add the terminating quote, unless this was a zero-string */
|
|
if (!AddZero) *p++ = '\'';
|
|
|
|
/* Terminate the buffer */
|
|
*p = '\0';
|
|
}
|
|
else
|
|
{
|
|
/* Handle special cases */
|
|
if (Char == '\b')
|
|
{
|
|
/* Bell */
|
|
strcpy(gCharName, "'\\b'");
|
|
}
|
|
else if (Char == '\n')
|
|
{
|
|
/* New line */
|
|
strcpy(gCharName, "'\\n'");
|
|
}
|
|
else if (Char == '\r')
|
|
{
|
|
/* Return */
|
|
strcpy(gCharName, "'\\r'");
|
|
}
|
|
else if (!AddZero)
|
|
{
|
|
/* Char value, in hex */
|
|
sprintf(gCharName, "0x%04x", Char);
|
|
}
|
|
else
|
|
{
|
|
/* Char value, C-style */
|
|
sprintf(gCharName, "\\x%04x", Char);
|
|
}
|
|
}
|
|
|
|
/* Return the name */
|
|
return gCharName;
|
|
}
|
|
|
|
VOID
|
|
PrintNameTable(FILE* FileHandle,
|
|
PKEYNAME KeyName,
|
|
BOOL DeadKey)
|
|
{
|
|
CHAR CharBuffer[255];
|
|
PKEYNAME NextName;
|
|
PCHAR Name, Buffer;
|
|
ULONG i;
|
|
|
|
/* Loop all key names */
|
|
while (KeyName)
|
|
{
|
|
/* Go to the next key name */
|
|
NextName = KeyName->Next;
|
|
|
|
/* Remember the name and our buffer address */
|
|
Name = KeyName->Name;
|
|
Buffer = CharBuffer;
|
|
|
|
/* Check if it's an IDS name */
|
|
if (strncmp(Name, "IDS_", 4))
|
|
{
|
|
/* No, so start parsing it. First, handle initial quote */
|
|
if (*Name != '\"') *Buffer++ = '\"';
|
|
|
|
/* Next, parse the name */
|
|
while (*Name)
|
|
{
|
|
/* Check if this is a C-style hex string */
|
|
if ((*Name != '\\') || ((*(Name + 1) != 'x') && (*(Name + 1) != 'X')))
|
|
{
|
|
/* It's not, so just copy straight into our buffer */
|
|
*Buffer++ = *Name++;
|
|
}
|
|
else
|
|
{
|
|
/* Continue scanning as long as this is a C-style hex string */
|
|
while ((*Name == '\\') && ((*(Name + 1) == 'x') || (*(Name + 1) == 'X')))
|
|
{
|
|
/* Copy 6 characters */
|
|
for (i = 0; (*Name) && (i < 6); i++) *Buffer++ = *Name++;
|
|
}
|
|
|
|
/* Check if we still have something at the end */
|
|
if (*Name)
|
|
{
|
|
/* Terminate our buffer */
|
|
*Buffer++ = '\"';
|
|
*Buffer++ = ' ';
|
|
*Buffer++ = 'L';
|
|
*Buffer++ = '\"';
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Check for terminating quote */
|
|
if (*(Buffer - 1) != '\"') *Buffer++ = '\"';
|
|
|
|
/* Terminate the buffer */
|
|
*Buffer++ = '\0';
|
|
}
|
|
else
|
|
{
|
|
/* Not yet supported */
|
|
printf("IDS Entries not yet handled!\n");
|
|
exit(1);
|
|
}
|
|
|
|
/* Is this a dead key? */
|
|
if (DeadKey)
|
|
{
|
|
/* Not yet handled */
|
|
printf("Dead keys not yet handled\n");
|
|
exit(1);
|
|
}
|
|
else
|
|
{
|
|
/* Print the entry */
|
|
fprintf(FileHandle, " 0x%02x, L%s,\n", KeyName->Code, CharBuffer);
|
|
}
|
|
|
|
/* Cleanup allocation */
|
|
free(KeyName->Name);
|
|
free(KeyName);
|
|
|
|
/* Move on */
|
|
KeyName = NextName;
|
|
}
|
|
|
|
/* Is this a dead key? */
|
|
if (DeadKey)
|
|
{
|
|
/* Not yet handled */
|
|
printf("Dead keys not yet handled\n");
|
|
exit(1);
|
|
}
|
|
else
|
|
{
|
|
/* Terminate the table */
|
|
fprintf(FileHandle, " 0 , NULL\n");
|
|
}
|
|
}
|
|
|
|
BOOLEAN
|
|
kbd_h(IN PLAYOUT Layout)
|
|
{
|
|
CHAR OutputFile[13];
|
|
FILE *FileHandle;
|
|
ULONG i;
|
|
CHAR UndefChar;
|
|
USHORT SubCode;
|
|
|
|
/* Build the keyboard name */
|
|
strcpy(OutputFile, gKBDName);
|
|
strcat(OutputFile, ".H");
|
|
|
|
/* Open it */
|
|
FileHandle = fopen(OutputFile, "wt");
|
|
if (!FileHandle)
|
|
{
|
|
/* Fail */
|
|
printf(" %12s : can't open for write.\n", OutputFile);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Print the module header */
|
|
fprintf(FileHandle,
|
|
"/****************************** Module Header ******************************\\\n"
|
|
"* Module Name: %s\n*\n* keyboard layout header\n"
|
|
"*\n"
|
|
"* Copyright (c) 2009, ReactOS Foundation\n"
|
|
"*\n"
|
|
"* Various defines for use by keyboard input code.\n*\n* History:\n"
|
|
"*\n"
|
|
"* created by KBDTOOL v%d.%02d %s*\n"
|
|
"\\***************************************************************************/\n\n",
|
|
OutputFile,
|
|
gVersion,
|
|
gSubVersion,
|
|
asctime(Now));
|
|
|
|
/* Print out the includes and defines */
|
|
fprintf(FileHandle,
|
|
"/*\n"
|
|
" * kbd type should be controlled by cl command-line argument\n"
|
|
" *\\n"
|
|
"#define KBD_TYPE 4\n\n"
|
|
"/*\n"
|
|
"* Include the basis of all keyboard table values\n"
|
|
"*/\n"
|
|
"#include \"kbd.h\"\n");
|
|
|
|
/* Now print out the virtual key conversion table */
|
|
fprintf(FileHandle,
|
|
"/***************************************************************************\\\n"
|
|
"* The table below defines the virtual keys for various keyboard types where\n"
|
|
"* the keyboard differ from the US keyboard.\n"
|
|
"*\n"
|
|
"* _EQ() : all keyboard types have the same virtual key for this scancode\n"
|
|
"* _NE() : different virtual keys for this scancode, depending on kbd type\n"
|
|
"*\n"
|
|
"* +------+ +----------+----------+----------+----------+----------+----------+\n"
|
|
"* | Scan | | kbd | kbd | kbd | kbd | kbd | kbd |\n"
|
|
"* | code | | type 1 | type 2 | type 3 | type 4 | type 5 | type 6 |\n"
|
|
"\\****+-------+_+----------+----------+----------+----------+----------+----------+*/\n\n");
|
|
|
|
/* Loop all keys */
|
|
for (i = 0; i < 110; i++)
|
|
{
|
|
/* Check if we processed this key */
|
|
if (Layout->Entry[i].Processed)
|
|
{
|
|
/* Check if it redefined a virtual key */
|
|
if (Layout->Entry[i].VirtualKey != Layout->Entry[i].OriginalVirtualKey)
|
|
{
|
|
/* Do we have a subcode? */
|
|
SubCode = Layout->Entry[i].ScanCode & 0xFF00;
|
|
if (SubCode)
|
|
{
|
|
/* Which kind is it? */
|
|
if (SubCode == 0xE000)
|
|
{
|
|
/* Extended 0 */
|
|
UndefChar = 'X';
|
|
}
|
|
else
|
|
{
|
|
/* Illegal */
|
|
if (SubCode != 0xE100)
|
|
{
|
|
/* Unrecognized */
|
|
printf("Weird scancode value %04x: expected xx, E0xx, or E1xx\n", SubCode);
|
|
exit(1);
|
|
}
|
|
|
|
/* Extended 1 */
|
|
UndefChar = 'Y';
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Normal key */
|
|
UndefChar = 'T';
|
|
}
|
|
|
|
/* Print out the virtual key redefinition */
|
|
fprintf(FileHandle,
|
|
"#undef %c%02X\n#define %c%02X _EQ(%43s%23s\n",
|
|
UndefChar,
|
|
Layout->Entry[i].ScanCode,
|
|
UndefChar,
|
|
Layout->Entry[i].ScanCode,
|
|
getVKName(Layout->Entry[i].VirtualKey, 0),
|
|
")");
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Cleanup and close */
|
|
fprintf(FileHandle,"\n");
|
|
fclose(FileHandle);
|
|
|
|
/* We made it */
|
|
return TRUE;
|
|
}
|
|
|
|
BOOLEAN
|
|
kbd_rc(IN PKEYNAME DescriptionData,
|
|
IN PKEYNAME LanguageData)
|
|
{
|
|
CHAR OutputFile[13];
|
|
CHAR InternalName[13];
|
|
CHAR TimeBuffer[5];
|
|
FILE *FileHandle;
|
|
ULONG Length;
|
|
PCHAR p;
|
|
PKEYNAME NextDescription, NextLanguage;
|
|
|
|
/* Build the keyboard name and internal name */
|
|
strcpy(OutputFile, gKBDName);
|
|
strcat(OutputFile, ".RC");
|
|
strcpy(InternalName, gKBDName);
|
|
for (p = InternalName; *p; p++) *p = tolower(*p);
|
|
|
|
/* Open it */
|
|
FileHandle = fopen(OutputFile, "wb");
|
|
if (!FileHandle)
|
|
{
|
|
/* Fail */
|
|
printf(" %12s : can't open for write.\n", OutputFile);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Check if we have copyright */
|
|
Length = strlen(gCopyright);
|
|
if (!Length)
|
|
{
|
|
/* Set time string */
|
|
strftime(TimeBuffer, 5, "%Y", Now);
|
|
|
|
/* Add copyright character */
|
|
strcpy(gCopyright, "(C)");
|
|
|
|
/* Add the year */
|
|
strcat(gCopyright, TimeBuffer);
|
|
|
|
/* Add blank company */
|
|
strcat(gCopyright, " ");
|
|
}
|
|
|
|
/* Write the resource file header */
|
|
fprintf(FileHandle,
|
|
"#include \"winver.h\"\r\n"
|
|
"1 VERSIONINFO\r\n"
|
|
" FILEVERSION 1,0,%d,%d\r\n"
|
|
" PRODUCTVERSION 1,0,%d,%d\r\n"
|
|
" FILEFLAGSMASK 0x3fL\r\n"
|
|
" FILEFLAGS 0x0L\r\n"
|
|
"FILEOS 0x40004L\r\n"
|
|
" FILETYPE VFT_DLL\r\n"
|
|
" FILESUBTYPE VFT2_DRV_KEYBOARD\r\n"
|
|
"BEGIN\r\n"
|
|
" BLOCK \"StringFileInfo\"\r\n"
|
|
" BEGIN\r\n"
|
|
" BLOCK \"000004B0\"\r\n"
|
|
" BEGIN\r\n"
|
|
" VALUE \"CompanyName\", \"%s\\0\"\r\n"
|
|
" VALUE \"FileDescription\", \"%s Keyboard Layout\\0\"\r\n"
|
|
" VALUE \"FileVersion\", \"1, 0, %d, %d\\0\"\r\n",
|
|
gVersion,
|
|
gSubVersion,
|
|
gVersion,
|
|
gSubVersion,
|
|
gCompany,
|
|
gDescription,
|
|
gVersion,
|
|
gSubVersion);
|
|
|
|
/* Continue writing it */
|
|
fprintf(FileHandle,
|
|
" VALUE \"InternalName\", \"%s (%d.%d)\\0\"\r\n"
|
|
" VALUE \"ProductName\",\"%s\\0\"\r\n"
|
|
" VALUE \"Release Information\",\"%s\\0\"\r\n"
|
|
" VALUE \"LegalCopyright\", \"%s\\0\"\r\n"
|
|
" VALUE \"OriginalFilename\",\"%s\\0\"\r\n"
|
|
" VALUE \"ProductVersion\", \"1, 0, %d, %d\\0\"\r\n"
|
|
" END\r\n"
|
|
" END\r\n"
|
|
" BLOCK \"VarFileInfo\"\r\n"
|
|
" BEGIN\r\n"
|
|
" VALUE \"Translation\", 0x0000, 0x04B0\r\n"
|
|
" END\r\n"
|
|
"END\r\n",
|
|
InternalName,
|
|
gVersion,
|
|
gSubVersion,
|
|
"Created by ReactOS KbdTool",
|
|
"Created by ReactOS KbdTool",
|
|
gCopyright,
|
|
InternalName,
|
|
gVersion,
|
|
gSubVersion);
|
|
|
|
/* Now check if we have a locale name */
|
|
Length = strlen(gLocaleName);
|
|
if (Length)
|
|
{
|
|
/* Write the stringtable header */
|
|
fprintf(FileHandle,
|
|
"\r\nSTRINGTABLE DISCARDABLE\r\nLANGUAGE %d, %d\r\n",
|
|
9,
|
|
1);
|
|
fprintf(FileHandle, "BEGIN\r\n");
|
|
|
|
/* Language or locale? */
|
|
if (strchr(gLocaleName, '\"'))
|
|
{
|
|
/* Write the language name */
|
|
fprintf(FileHandle, " %d %s\r\n", gStringIdForLanguageNames, gLocaleName);
|
|
}
|
|
else
|
|
{
|
|
/* Write the locale name */
|
|
fprintf(FileHandle, " %d \"%s\"\r\n", gStringIdForLocaleName, gLocaleName);
|
|
}
|
|
|
|
/* Terminate the entry */
|
|
fprintf(FileHandle, "END\r\n\r\n");
|
|
}
|
|
|
|
/* Check for description information */
|
|
while (DescriptionData)
|
|
{
|
|
/* Remember the next pointer */
|
|
NextDescription = DescriptionData->Next;
|
|
|
|
/* Write the header */
|
|
fprintf(FileHandle,
|
|
"\r\nSTRINGTABLE DISCARDABLE\r\nLANGUAGE %d, %d\r\n",
|
|
DescriptionData->Code & 0x3FF,
|
|
DescriptionData->Code >> 10);
|
|
fprintf(FileHandle, "BEGIN\r\n");
|
|
|
|
/* Quoted string or not? */
|
|
if (strchr(DescriptionData->Name, '\"'))
|
|
{
|
|
/* Write the description name */
|
|
fprintf(FileHandle, " %d %s\r\n", gStringIdForDescriptions, DescriptionData->Name);
|
|
}
|
|
else
|
|
{
|
|
/* Write the description name */
|
|
fprintf(FileHandle, " %d \"%s\"\r\n", gStringIdForDescriptions, DescriptionData->Name);
|
|
}
|
|
|
|
/* Terminate the entry */
|
|
fprintf(FileHandle, "END\r\n\r\n");
|
|
|
|
/* Free the allocation */
|
|
free(DescriptionData->Name);
|
|
free(DescriptionData);
|
|
|
|
/* Move to the next entry */
|
|
DescriptionData = NextDescription;
|
|
}
|
|
|
|
/* Check for language information */
|
|
while (LanguageData)
|
|
{
|
|
/* Remember the next pointer */
|
|
NextLanguage = LanguageData->Next;
|
|
|
|
/* Write the header */
|
|
fprintf(FileHandle,
|
|
"\r\nSTRINGTABLE DISCARDABLE\r\nLANGUAGE %d, %d\r\n",
|
|
LanguageData->Code & 0x3FF,
|
|
LanguageData->Code >> 10);
|
|
fprintf(FileHandle, "BEGIN\r\n");
|
|
|
|
/* Quoted string or not? */
|
|
if (strchr(LanguageData->Name, '\"'))
|
|
{
|
|
/* Write the language name */
|
|
fprintf(FileHandle, " %d %s\r\n", gStringIdForLanguageNames, LanguageData->Name);
|
|
}
|
|
else
|
|
{
|
|
/* Write the language name */
|
|
fprintf(FileHandle, " %d \"%s\"\r\n", gStringIdForLanguageNames, LanguageData->Name);
|
|
}
|
|
|
|
/* Terminate the entry */
|
|
fprintf(FileHandle, "END\r\n\r\n");
|
|
|
|
/* Free the allocation */
|
|
free(LanguageData->Name);
|
|
free(LanguageData);
|
|
|
|
/* Move to the next entry */
|
|
LanguageData = NextLanguage;
|
|
}
|
|
|
|
/* We're done! */
|
|
fclose(FileHandle);
|
|
return TRUE;
|
|
}
|
|
|
|
BOOLEAN
|
|
kbd_def(VOID)
|
|
{
|
|
CHAR OutputFile[13];
|
|
FILE *FileHandle;
|
|
|
|
/* Build the keyboard name and internal name */
|
|
strcpy(OutputFile, gKBDName);
|
|
strcat(OutputFile, ".DEF");
|
|
|
|
/* Open it */
|
|
FileHandle = fopen(OutputFile, "wt");
|
|
if (!FileHandle)
|
|
{
|
|
/* Fail */
|
|
printf(" %12s : can't open for write.\n", OutputFile);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Write the file exports */
|
|
fprintf(FileHandle,
|
|
"LIBRARY %s\n\n"
|
|
"EXPORTS\n"
|
|
" KbdLayerDescriptor @1\n",
|
|
gKBDName);
|
|
|
|
/* Clean up */
|
|
fclose(FileHandle);
|
|
return TRUE;
|
|
}
|
|
|
|
BOOLEAN
|
|
kbd_c(IN ULONG StateCount,
|
|
IN PULONG ShiftStates,
|
|
IN PVOID AttributeData,
|
|
IN PLAYOUT Layout,
|
|
IN PVOID DeadKeyData,
|
|
IN PVOID LigatureData,
|
|
IN PKEYNAME KeyNameData,
|
|
IN PKEYNAME KeyNameExtData,
|
|
IN PKEYNAME KeyNameDeadData)
|
|
{
|
|
CHAR OutputFile[13];
|
|
CHAR KeyNameBuffer[50];
|
|
CHAR LineBuffer[256];
|
|
BOOLEAN NeedPlus;
|
|
FILE *FileHandle;
|
|
ULONG States[8];
|
|
ULONG i, j, k;
|
|
ULONG HighestState;
|
|
PVKNAME Entry;
|
|
PCHAR p;
|
|
|
|
/* Build the keyboard name and internal name */
|
|
strcpy(OutputFile, gKBDName);
|
|
strcat(OutputFile, ".C");
|
|
|
|
/* Open it */
|
|
FileHandle = fopen(OutputFile, "wt");
|
|
if (!FileHandle)
|
|
{
|
|
/* Fail */
|
|
printf(" %12s : can't open for write.\n", OutputFile);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Print the header */
|
|
fprintf(FileHandle,
|
|
"/***************************************************************************\\\n"
|
|
"* Module Name: %s\n*\n* keyboard layout\n"
|
|
"*\n"
|
|
"* Copyright (c) 2009, ReactOS Foundation\n"
|
|
"*\n"
|
|
"* History:\n"
|
|
"* KBDTOOL v%d.%02d - Created %s"
|
|
"\\***************************************************************************/\n\n",
|
|
OutputFile,
|
|
gVersion,
|
|
gSubVersion,
|
|
asctime(Now));
|
|
|
|
/* What kind of driver is this? */
|
|
if (FallbackDriver)
|
|
{
|
|
/* Fallback only */
|
|
fprintf(FileHandle, "#include \"precomp.h\"\n");
|
|
}
|
|
else
|
|
{
|
|
/* Add the includes */
|
|
fprintf(FileHandle,
|
|
"#include <windows.h>\n"
|
|
"#include \"kbd.h\"\n"
|
|
"#include \"%s.h\"\n\n",
|
|
gKBDName);
|
|
}
|
|
|
|
/* What kind of driver is this? */
|
|
if (FallbackDriver)
|
|
{
|
|
/* Only one section */
|
|
fprintf(FileHandle,
|
|
"#pragma data_seg(\"%s\")\n#define ALLOC_SECTION_LDATA\n\n",
|
|
".kbdfallback");
|
|
}
|
|
else
|
|
{
|
|
/* Section and attributes depend on architecture */
|
|
fprintf(FileHandle,
|
|
"#if defined(_M_IA64)\n"
|
|
"#pragma section(\"%s\")\n"
|
|
"#define ALLOC_SECTION_LDATA __declspec(allocate(\"%s\"))\n"
|
|
"#else\n"
|
|
"#pragma data_seg(\"%s\")\n"
|
|
"#define ALLOC_SECTION_LDATA\n"
|
|
"#endif\n\n",
|
|
".data",
|
|
".data",
|
|
".data");
|
|
}
|
|
|
|
/* Scan code to virtual key conversion table header */
|
|
fprintf(FileHandle,
|
|
"/***************************************************************************\\\n"
|
|
"* ausVK[] - Virtual Scan Code to Virtual Key conversion table\n"
|
|
"\\***************************************************************************/\n\n");
|
|
|
|
/* Table begin */
|
|
fprintf(FileHandle,
|
|
"static ALLOC_SECTION_LDATA USHORT ausVK[] = {\n"
|
|
" T00, T01, T02, T03, T04, T05, T06, T07,\n"
|
|
" T08, T09, T0A, T0B, T0C, T0D, T0E, T0F,\n"
|
|
" T10, T11, T12, T13, T14, T15, T16, T17,\n"
|
|
" T18, T19, T1A, T1B, T1C, T1D, T1E, T1F,\n"
|
|
" T20, T21, T22, T23, T24, T25, T26, T27,\n"
|
|
" T28, T29, T2A, T2B, T2C, T2D, T2E, T2F,\n"
|
|
" T30, T31, T32, T33, T34, T35,\n\n");
|
|
|
|
/* Table continue */
|
|
fprintf(FileHandle,
|
|
" /*\n"
|
|
" * Right-hand Shift key must have KBDEXT bit set.\n"
|
|
" */\n"
|
|
" T36 | KBDEXT,\n\n"
|
|
" T37 | KBDMULTIVK, // numpad_* + Shift/Alt -> SnapShot\n\n"
|
|
" T38, T39, T3A, T3B, T3C, T3D, T3E,\n"
|
|
" T3F, T40, T41, T42, T43, T44,\n\n");
|
|
|
|
/* Table continue */
|
|
fprintf(FileHandle,
|
|
" /*\n"
|
|
" * NumLock Key:\n"
|
|
" * KBDEXT - VK_NUMLOCK is an Extended key\n"
|
|
" * KBDMULTIVK - VK_NUMLOCK or VK_PAUSE (without or with CTRL)\n"
|
|
" */\n"
|
|
" T45 | KBDEXT | KBDMULTIVK,\n\n"
|
|
" T46 | KBDMULTIVK,\n\n");
|
|
|
|
/* Numpad table */
|
|
fprintf(FileHandle,
|
|
" /*\n"
|
|
" * Number Pad keys:\n"
|
|
" * KBDNUMPAD - digits 0-9 and decimal point.\n"
|
|
" * KBDSPECIAL - require special processing by Windows\n"
|
|
" */\n"
|
|
" T47 | KBDNUMPAD | KBDSPECIAL, // Numpad 7 (Home)\n"
|
|
" T48 | KBDNUMPAD | KBDSPECIAL, // Numpad 8 (Up),\n"
|
|
" T49 | KBDNUMPAD | KBDSPECIAL, // Numpad 9 (PgUp),\n"
|
|
" T4A,\n"
|
|
" T4B | KBDNUMPAD | KBDSPECIAL, // Numpad 4 (Left),\n"
|
|
" T4C | KBDNUMPAD | KBDSPECIAL, // Numpad 5 (Clear),\n"
|
|
" T4D | KBDNUMPAD | KBDSPECIAL, // Numpad 6 (Right),\n"
|
|
" T4E,\n"
|
|
" T4F | KBDNUMPAD | KBDSPECIAL, // Numpad 1 (End),\n"
|
|
" T50 | KBDNUMPAD | KBDSPECIAL, // Numpad 2 (Down),\n"
|
|
" T51 | KBDNUMPAD | KBDSPECIAL, // Numpad 3 (PgDn),\n"
|
|
" T52 | KBDNUMPAD | KBDSPECIAL, // Numpad 0 (Ins),\n"
|
|
" T53 | KBDNUMPAD | KBDSPECIAL, // Numpad . (Del),\n\n");
|
|
|
|
/* Table finish */
|
|
fprintf(FileHandle,
|
|
" T54, T55, T56, T57, T58, T59, T5A, T5B,\n"
|
|
" T5C, T5D, T5E, T5F, T60, T61, T62, T63,\n"
|
|
" T64, T65, T66, T67, T68, T69, T6A, T6B,\n"
|
|
" T6C, T6D, T6E, T6F, T70, T71, T72, T73,\n"
|
|
" T74, T75, T76, T77, T78, T79, T7A, T7B,\n"
|
|
" T7C, T7D, T7E\n\n"
|
|
"};\n\n");
|
|
|
|
/* Key name table header */
|
|
fprintf(FileHandle, "static ALLOC_SECTION_LDATA VSC_VK aE0VscToVk[] = {\n");
|
|
|
|
/* Loop 110-key table */
|
|
for (i = 0; i < 110; i++)
|
|
{
|
|
/* Check for non-extended keys */
|
|
if ((Layout->Entry[i].ScanCode & 0xFF00) == 0xE000)
|
|
{
|
|
/* Which are valid */
|
|
if (Layout->Entry[i].ScanCode != 0xFF)
|
|
{
|
|
/* And mapped */
|
|
if (Layout->Entry[i].VirtualKey != 0xFF)
|
|
{
|
|
/* Output them */
|
|
fprintf(FileHandle,
|
|
" { 0x%02X, X%02X | KBDEXT }, // %s\n",
|
|
Layout->Entry[i].ScanCode,
|
|
Layout->Entry[i].ScanCode,
|
|
Layout->Entry[i].Name);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Key name table finish */
|
|
fprintf(FileHandle, " { 0, 0 }\n};\n\n");
|
|
|
|
/* Extended key name table header */
|
|
fprintf(FileHandle, "static ALLOC_SECTION_LDATA VSC_VK aE1VscToVk[] = {\n");
|
|
|
|
/* Loop 110-key table */
|
|
for (i = 0; i < 110; i++)
|
|
{
|
|
/* Check for extended keys */
|
|
if ((Layout->Entry[i].ScanCode & 0xFF00) == 0xE100)
|
|
{
|
|
/* Which are valid */
|
|
if (Layout->Entry[i].ScanCode != 0xFF)
|
|
{
|
|
/* And mapped */
|
|
if (Layout->Entry[i].VirtualKey != 0xFF)
|
|
{
|
|
/* Output them */
|
|
fprintf(FileHandle,
|
|
" { 0x%02X, Y%02X | KBDEXT }, // %s\n",
|
|
Layout->Entry[i].ScanCode,
|
|
Layout->Entry[i].ScanCode,
|
|
Layout->Entry[i].Name);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Extended key name table finish */
|
|
fprintf(FileHandle,
|
|
" { 0x1D, Y1D }, // Pause\n"
|
|
" { 0 , 0 }\n};\n\n");
|
|
|
|
/* Modifier table description */
|
|
fprintf(FileHandle,
|
|
"/***************************************************************************\\\n"
|
|
"* aVkToBits[] - map Virtual Keys to Modifier Bits\n"
|
|
"*\n"
|
|
"* See kbd.h for a full description.\n"
|
|
"*\n"
|
|
"* The keyboard has only three shifter keys:\n"
|
|
"* SHIFT (L & R) affects alphabnumeric keys,\n"
|
|
"* CTRL (L & R) is used to generate control characters\n"
|
|
"* ALT (L & R) used for generating characters by number with numpad\n"
|
|
"\\***************************************************************************/\n");
|
|
|
|
/* Modifier table header */
|
|
fprintf(FileHandle, "static ALLOC_SECTION_LDATA VK_TO_BIT aVkToBits[] = {\n");
|
|
|
|
/* Loop modifier table */
|
|
i = 0;
|
|
Entry = &Modifiers[0];
|
|
while (Entry->VirtualKey)
|
|
{
|
|
/* Print out entry */
|
|
fprintf(FileHandle,
|
|
" { %-12s, %-12s },\n",
|
|
getVKName(Entry->VirtualKey, 1),
|
|
Entry->Name);
|
|
|
|
/* Move to the next one */
|
|
Entry = &Modifiers[++i];
|
|
}
|
|
|
|
/* Modifier table finish */
|
|
fprintf(FileHandle, " { 0, 0 }\n};\n\n");
|
|
|
|
/* Modifier conversion table description */
|
|
fprintf(FileHandle,
|
|
"/***************************************************************************\\\n"
|
|
"* aModification[] - map character modifier bits to modification number\n"
|
|
"*\n"
|
|
"* See kbd.h for a full description.\n"
|
|
"*\n"
|
|
"\\***************************************************************************/\n\n");
|
|
|
|
/* Zero out local state data */
|
|
for (i = 0; i < 8; i++) States[i] = -1;
|
|
|
|
/* Find the highest set state */
|
|
for (HighestState = 1, i = 0; (i < 8) && (ShiftStates[i] != -1); i++)
|
|
{
|
|
/* Save all state values */
|
|
States[ShiftStates[i]] = i;
|
|
if (ShiftStates[i] > HighestState) HighestState = ShiftStates[i];
|
|
}
|
|
|
|
/* Modifier conversion table header */
|
|
fprintf(FileHandle,
|
|
"static ALLOC_SECTION_LDATA MODIFIERS CharModifiers = {\n"
|
|
" &aVkToBits[0],\n"
|
|
" %d,\n"
|
|
" {\n"
|
|
" // Modification# // Keys Pressed\n"
|
|
" // ============= // =============\n",
|
|
HighestState);
|
|
|
|
/* Loop states */
|
|
for (i = 0; i <= HighestState; i++)
|
|
{
|
|
/* Check for invalid state */
|
|
if (States[i] == -1)
|
|
{
|
|
/* Invalid state header */
|
|
fprintf(FileHandle, " SHFT_INVALID, // ");
|
|
}
|
|
else
|
|
{
|
|
/* Is this the last one? */
|
|
if (i == HighestState)
|
|
{
|
|
/* Last state header */
|
|
fprintf(FileHandle, " %d // ", States[i]);
|
|
}
|
|
else
|
|
{
|
|
/* Normal state header */
|
|
fprintf(FileHandle, " %d, // ", States[i]);
|
|
}
|
|
|
|
/* State 1 or higher? */
|
|
if (i >= 1)
|
|
{
|
|
/* J is the loop variable, K is the double */
|
|
for (NeedPlus = 0, j = 0, k = 1; (1u << j) <= i; j++, k = (1 << j))
|
|
{
|
|
/* Do we need to add a plus? */
|
|
if (NeedPlus)
|
|
{
|
|
/* Add it */
|
|
fprintf(FileHandle, "+");
|
|
NeedPlus = FALSE;
|
|
}
|
|
|
|
/* Check if it's time to add a modifier */
|
|
if (i & k)
|
|
{
|
|
/* Get the key state name and copy it into our buffer */
|
|
strcpy(KeyNameBuffer, getVKName(Modifiers[j].VirtualKey, 1));
|
|
|
|
/* Go go the 4th char (past the "KBD") and lower name */
|
|
for (p = &KeyNameBuffer[4]; *p; p++) *p = tolower(*p);
|
|
|
|
/* Print it */
|
|
fprintf(FileHandle, "%s", &KeyNameBuffer[3]);
|
|
|
|
/* We'll need a plus sign next */
|
|
NeedPlus = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Terminate the entry */
|
|
fprintf(FileHandle, "\n");
|
|
}
|
|
}
|
|
|
|
|
|
/* Modifier conversion table end */
|
|
fprintf(FileHandle," }\n" "};\n\n");
|
|
|
|
/* Shift state translation table description */
|
|
fprintf(FileHandle,
|
|
"/***************************************************************************\\\n"
|
|
"*\n"
|
|
"* aVkToWch2[] - Virtual Key to WCHAR translation for 2 shift states\n"
|
|
"* aVkToWch3[] - Virtual Key to WCHAR translation for 3 shift states\n"
|
|
"* aVkToWch4[] - Virtual Key to WCHAR translation for 4 shift states\n");
|
|
|
|
/* Check if there's exta shift states */
|
|
for (i = 5; i < HighestState; i++)
|
|
{
|
|
/* Print out extra information */
|
|
fprintf(FileHandle,
|
|
"* aVkToWch%d[] - Virtual Key to WCHAR translation for %d shift states\n",
|
|
i,
|
|
i);
|
|
}
|
|
|
|
/* Shift state translation table description continue */
|
|
fprintf(FileHandle,
|
|
"*\n"
|
|
"* Table attributes: Unordered Scan, null-terminated\n"
|
|
"*\n"
|
|
"* Search this table for an entry with a matching Virtual Key to find the\n"
|
|
"* corresponding unshifted and shifted WCHAR characters.\n"
|
|
"*\n"
|
|
"* Special values for VirtualKey (column 1)\n"
|
|
"* 0xff - dead chars for the previous entry\n"
|
|
"* 0 - terminate the list\n"
|
|
"*\n"
|
|
"* Special values for Attributes (column 2)\n"
|
|
"* CAPLOK bit - CAPS-LOCK affect this key like SHIFT\n"
|
|
"*\n"
|
|
"* Special values for wch[*] (column 3 & 4)\n"
|
|
"* WCH_NONE - No character\n"
|
|
"* WCH_DEAD - Dead Key (diaresis) or invalid (US keyboard has none)\n"
|
|
"* WCH_LGTR - Ligature (generates multiple characters)\n"
|
|
"*\n"
|
|
"\\***************************************************************************/\n\n");
|
|
|
|
/* Loop all the states */
|
|
for (i = 2; i <= StateCount; i++)
|
|
{
|
|
/* Check if this something else than state 2 */
|
|
if (i != 2)
|
|
{
|
|
/* Loop all the scan codes */
|
|
for (j = 0; j < 110; j++)
|
|
{
|
|
/* Check if this is the state for the entry */
|
|
if (i == Layout->Entry[j].StateCount) break;
|
|
}
|
|
}
|
|
|
|
/* Print the table header */
|
|
fprintf(FileHandle,
|
|
"static ALLOC_SECTION_LDATA VK_TO_WCHARS%d aVkToWch%d[] = {\n"
|
|
"// | | Shift |",
|
|
i,
|
|
i);
|
|
|
|
/* Print the correct state label */
|
|
for (k = 2; k < i; k++) fprintf(FileHandle, "%-9.9s|",
|
|
StateLabel[ShiftStates[k]]);
|
|
|
|
/* Print the next separator */
|
|
fprintf(FileHandle, "\n// |=========|=========|");
|
|
|
|
/* Check for extra states and print their separators too */
|
|
for (k = 2; k < i; k++) fprintf(FileHandle, "=========|");
|
|
|
|
/* Finalize the separator header */
|
|
fprintf(FileHandle, "\n");
|
|
|
|
/* Loop all the scan codes */
|
|
for (j = 0; j < 110; j++)
|
|
{
|
|
/* Check if this is the state for the entry */
|
|
if (i != Layout->Entry[j].StateCount) continue;
|
|
|
|
/* Print out the entry for this key */
|
|
fprintf(FileHandle,
|
|
" {%-13s,%-7s",
|
|
getVKName(Layout->Entry[j].VirtualKey, 1),
|
|
CapState[Layout->Entry[j].Cap]);
|
|
|
|
/* Initialize the buffer for this line */
|
|
*LineBuffer = '\0';
|
|
|
|
/* Loop states */
|
|
for (k = 0; k < i; k++)
|
|
{
|
|
/* Check for dead key data */
|
|
if (DeadKeyData)
|
|
{
|
|
/* Not yet supported */
|
|
printf("Dead key data not supported!\n");
|
|
exit(1);
|
|
}
|
|
|
|
/* Check if it's a ligature key */
|
|
if (Layout->Entry[j].LigatureCharData[k])
|
|
{
|
|
/* Not yet supported */
|
|
printf("Ligature key data not supported!\n");
|
|
exit(1);
|
|
}
|
|
|
|
/* Print out the WCH_ name */
|
|
fprintf(FileHandle,
|
|
",%-9s",
|
|
WChName(Layout->Entry[j].CharData[k], 0));
|
|
|
|
/* If we have something on the line buffer by now, add WCH_NONE */
|
|
if (*LineBuffer != '\0') strcpy(LineBuffer, "WCH_NONE ");
|
|
}
|
|
|
|
/* Finish the line */
|
|
fprintf(FileHandle, "},\n");
|
|
|
|
/* Do we have any data at all? */
|
|
if (*LineBuffer != '\0')
|
|
{
|
|
/* Print it, we're done */
|
|
fprintf(FileHandle, "%s},\n", LineBuffer);
|
|
continue;
|
|
}
|
|
|
|
/* Otherwise, we're done, unless this requires SGCAP data */
|
|
if (Layout->Entry[j].Cap != 2) continue;
|
|
|
|
/* Not yet supported */
|
|
printf("SGCAP not yet supported!\n");
|
|
exit(1);
|
|
}
|
|
|
|
/* Did we only have two states? */
|
|
if (i == 2)
|
|
{
|
|
/* Print out the built-in table */
|
|
fprintf(FileHandle,
|
|
" {VK_TAB ,0 ,'\\t' ,'\\t' },\n"
|
|
" {VK_ADD ,0 ,'+' ,'+' },\n"
|
|
" {VK_DIVIDE ,0 ,'/' ,'/' },\n"
|
|
" {VK_MULTIPLY ,0 ,'*' ,'*' },\n"
|
|
" {VK_SUBTRACT ,0 ,'-' ,'-' },\n");
|
|
}
|
|
|
|
/* Terminate the table */
|
|
fprintf(FileHandle, " {0 ,0 ");
|
|
for (k = 0; k < i; k++) fprintf(FileHandle, ",0 ");
|
|
|
|
/* Terminate the structure */
|
|
fprintf(FileHandle, "}\n" "};\n\n");
|
|
}
|
|
|
|
/* Numpad translation table */
|
|
fprintf(FileHandle,
|
|
"// Put this last so that VkKeyScan interprets number characters\n"
|
|
"// as coming from the main section of the kbd (aVkToWch2 and\n"
|
|
"// aVkToWch5) before considering the numpad (aVkToWch1).\n\n"
|
|
"static ALLOC_SECTION_LDATA VK_TO_WCHARS1 aVkToWch1[] = {\n"
|
|
" { VK_NUMPAD0 , 0 , '0' },\n"
|
|
" { VK_NUMPAD1 , 0 , '1' },\n"
|
|
" { VK_NUMPAD2 , 0 , '2' },\n"
|
|
" { VK_NUMPAD3 , 0 , '3' },\n"
|
|
" { VK_NUMPAD4 , 0 , '4' },\n"
|
|
" { VK_NUMPAD5 , 0 , '5' },\n"
|
|
" { VK_NUMPAD6 , 0 , '6' },\n"
|
|
" { VK_NUMPAD7 , 0 , '7' },\n"
|
|
" { VK_NUMPAD8 , 0 , '8' },\n"
|
|
" { VK_NUMPAD9 , 0 , '9' },\n"
|
|
" { 0 , 0 , '\\0' }\n"
|
|
"};\n\n");
|
|
|
|
/* Translation tables header */
|
|
fprintf(FileHandle,"static ALLOC_SECTION_LDATA VK_TO_WCHAR_TABLE aVkToWcharTable[] = {\n");
|
|
|
|
/* Loop states higher than 3 */
|
|
for (i = 3; i <= StateCount; i++)
|
|
{
|
|
/* Print out the extra tables */
|
|
fprintf(FileHandle,
|
|
" { (PVK_TO_WCHARS1)aVkToWch%d, %d, sizeof(aVkToWch%d[0]) },\n",
|
|
i,
|
|
i,
|
|
i);
|
|
}
|
|
|
|
/* Array of translation tables */
|
|
fprintf(FileHandle,
|
|
" { (PVK_TO_WCHARS1)aVkToWch2, 2, sizeof(aVkToWch2[0]) },\n"
|
|
" { (PVK_TO_WCHARS1)aVkToWch1, 1, sizeof(aVkToWch1[0]) },\n"
|
|
" { NULL, 0, 0 },\n"
|
|
"};\n\n");
|
|
|
|
/* Scan code to key name conversion table description */
|
|
fprintf(FileHandle,
|
|
"/***************************************************************************\\\n"
|
|
"* aKeyNames[], aKeyNamesExt[] - Virtual Scancode to Key Name tables\n"
|
|
"*\n"
|
|
"* Table attributes: Ordered Scan (by scancode), null-terminated\n"
|
|
"*\n"
|
|
"* Only the names of Extended, NumPad, Dead and Non-Printable keys are here.\n"
|
|
"* (Keys producing printable characters are named by that character)\n"
|
|
"\\***************************************************************************/\n\n");
|
|
|
|
/* Check for key name data */
|
|
if (KeyNameData)
|
|
{
|
|
/* Table header */
|
|
fprintf(FileHandle, "static ALLOC_SECTION_LDATA VSC_LPWSTR aKeyNames[] = {\n");
|
|
|
|
/* Print table */
|
|
PrintNameTable(FileHandle, KeyNameData, FALSE);
|
|
|
|
/* Table end */
|
|
fprintf(FileHandle, "};\n\n");
|
|
}
|
|
|
|
/* Check for extended key name data */
|
|
if (KeyNameExtData)
|
|
{
|
|
/* Table header */
|
|
fprintf(FileHandle, "static ALLOC_SECTION_LDATA VSC_LPWSTR aKeyNamesExt[] = {\n");
|
|
|
|
/* Print table */
|
|
PrintNameTable(FileHandle, KeyNameExtData, FALSE);
|
|
|
|
/* Table end */
|
|
fprintf(FileHandle, "};\n\n");
|
|
}
|
|
|
|
/* Check for dead key name data */
|
|
if (KeyNameDeadData)
|
|
{
|
|
/* Not yet supported */
|
|
printf("Dead key name data not supported!\n");
|
|
exit(1);
|
|
}
|
|
|
|
/* Check for dead key data */
|
|
if (DeadKeyData)
|
|
{
|
|
/* Not yet supported */
|
|
printf("Dead key data not supported!\n");
|
|
exit(1);
|
|
}
|
|
|
|
/* Check for ligature data */
|
|
if (LigatureData)
|
|
{
|
|
/* Not yet supported */
|
|
printf("Ligature key data not supported!\n");
|
|
exit(1);
|
|
}
|
|
|
|
/* Main keyboard table descriptor type */
|
|
fprintf(FileHandle, "static ");
|
|
|
|
/* FIXME? */
|
|
|
|
/* Main keyboard table descriptor header */
|
|
fprintf(FileHandle,
|
|
"ALLOC_SECTION_LDATA KBDTABLES KbdTables%s = {\n"
|
|
" /*\n"
|
|
" * Modifier keys\n"
|
|
" */\n"
|
|
" &CharModifiers,\n\n"
|
|
" /*\n"
|
|
" * Characters tables\n"
|
|
" */\n"
|
|
" aVkToWcharTable,\n\n"
|
|
" /*\n"
|
|
" * Diacritics\n"
|
|
" */\n",
|
|
FallbackDriver ? "Fallback" : "" );
|
|
|
|
/* Descriptor dead key data section */
|
|
if (DeadKeyData)
|
|
{
|
|
fprintf(FileHandle, " aDeadKey,\n\n");
|
|
}
|
|
else
|
|
{
|
|
fprintf(FileHandle, " NULL,\n\n");
|
|
}
|
|
|
|
/* Descriptor key name comment */
|
|
fprintf(FileHandle,
|
|
" /*\n"
|
|
" * Names of Keys\n"
|
|
" */\n");
|
|
|
|
/* Descriptor key name section */
|
|
if (KeyNameData)
|
|
{
|
|
fprintf(FileHandle, " aKeyNames,\n");
|
|
}
|
|
else
|
|
{
|
|
fprintf(FileHandle, " NULL,\n");
|
|
}
|
|
|
|
/* Descriptor extended key name section */
|
|
if (KeyNameExtData)
|
|
{
|
|
fprintf(FileHandle, " aKeyNamesExt,\n");
|
|
}
|
|
else
|
|
{
|
|
fprintf(FileHandle, " NULL,\n");
|
|
}
|
|
|
|
/* Descriptor dead key name section */
|
|
if ((DeadKeyData) && (KeyNameDeadData))
|
|
{
|
|
fprintf(FileHandle, " aKeyNamesDead,\n\n");
|
|
}
|
|
else
|
|
{
|
|
fprintf(FileHandle, " NULL,\n\n");
|
|
}
|
|
|
|
/* Descriptor conversion table section */
|
|
fprintf(FileHandle,
|
|
" /*\n"
|
|
" * Scan codes to Virtual Keys\n"
|
|
" */\n"
|
|
" ausVK,\n"
|
|
" sizeof(ausVK) / sizeof(ausVK[0]),\n"
|
|
" aE0VscToVk,\n"
|
|
" aE1VscToVk,\n\n"
|
|
" /*\n"
|
|
" * Locale-specific special processing\n"
|
|
" */\n");
|
|
|
|
/* FIXME: AttributeData and KLLF_ALTGR stuff */
|
|
|
|
/* Descriptor locale-specific section */
|
|
fprintf(FileHandle, " MAKELONG(%s, KBD_VERSION),\n\n", "0"); /* FIXME */
|
|
|
|
/* Descriptor ligature data comment */
|
|
fprintf(FileHandle, " /*\n * Ligatures\n */\n %d,\n", 0); /* FIXME */
|
|
|
|
/* Descriptor ligature data section */
|
|
if (!LigatureData)
|
|
{
|
|
fprintf(FileHandle, " 0,\n");
|
|
fprintf(FileHandle, " NULL\n");
|
|
}
|
|
else
|
|
{
|
|
fprintf(FileHandle, " sizeof(aLigature[0]),\n");
|
|
fprintf(FileHandle, " (PLIGATURE1)aLigature\n");
|
|
}
|
|
|
|
/* Descriptor finish */
|
|
fprintf(FileHandle, "};\n\n");
|
|
|
|
/* Keyboard layout callback function */
|
|
if (!FallbackDriver) fprintf(FileHandle,
|
|
"PKBDTABLES KbdLayerDescriptor(VOID)\n"
|
|
"{\n"
|
|
" return &KbdTables;\n"
|
|
"}\n");
|
|
|
|
/* Clean up */
|
|
fclose(FileHandle);
|
|
return TRUE;
|
|
}
|
|
|
|
ULONG
|
|
DoOutput(IN ULONG StateCount,
|
|
IN PULONG ShiftStates,
|
|
IN PKEYNAME DescriptionData,
|
|
IN PKEYNAME LanguageData,
|
|
IN PVOID AttributeData,
|
|
IN PVOID DeadKeyData,
|
|
IN PVOID LigatureData,
|
|
IN PKEYNAME KeyNameData,
|
|
IN PKEYNAME KeyNameExtData,
|
|
IN PKEYNAME KeyNameDeadData)
|
|
{
|
|
ULONG FailureCode = 0;
|
|
|
|
/* Take the time */
|
|
time(&Clock);
|
|
Now = localtime(&Clock);
|
|
|
|
/* Check if this just a fallback driver*/
|
|
if (!FallbackDriver)
|
|
{
|
|
/* It's not, create header file */
|
|
if (!kbd_h(&g_Layout)) FailureCode = 1;
|
|
|
|
/* Create the resource file */
|
|
if (!kbd_rc(DescriptionData, LanguageData)) FailureCode = 2;
|
|
}
|
|
|
|
/* Create the C file */
|
|
if (!kbd_c(StateCount,
|
|
ShiftStates,
|
|
AttributeData,
|
|
&g_Layout,
|
|
DeadKeyData,
|
|
LigatureData,
|
|
KeyNameData,
|
|
KeyNameExtData,
|
|
KeyNameDeadData))
|
|
{
|
|
/* Failed in C file generation */
|
|
FailureCode = 3;
|
|
}
|
|
|
|
/* Check if this just a fallback driver*/
|
|
if (!FallbackDriver)
|
|
{
|
|
/* Generate the definition file */
|
|
if (!kbd_def()) FailureCode = 4;
|
|
}
|
|
|
|
/* Done */
|
|
return FailureCode;
|
|
}
|
|
|
|
|
|
/* EOF */
|