503 lines
13 KiB
C
503 lines
13 KiB
C
/*
|
|
* DOS CONFIG.SYS parser
|
|
*
|
|
* Copyright 1998 Andreas Mohr
|
|
*
|
|
* 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 "wine/port.h"
|
|
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#ifdef HAVE_UNISTD_H
|
|
# include <unistd.h>
|
|
#endif
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
|
|
#include "windef.h"
|
|
#include "winbase.h"
|
|
#include "winreg.h"
|
|
|
|
#include "file.h"
|
|
#include "dosexe.h"
|
|
|
|
#include "wine/debug.h"
|
|
#include "wine/unicode.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(profile);
|
|
|
|
|
|
static int DOSCONF_Device(char **confline);
|
|
static int DOSCONF_Dos(char **confline);
|
|
static int DOSCONF_Fcbs(char **confline);
|
|
static int DOSCONF_Break(char **confline);
|
|
static int DOSCONF_Files(char **confline);
|
|
static int DOSCONF_Install(char **confline);
|
|
static int DOSCONF_Lastdrive(char **confline);
|
|
static int DOSCONF_Menu(char **confline);
|
|
static int DOSCONF_Include(char **confline);
|
|
static int DOSCONF_Country(char **confline);
|
|
static int DOSCONF_Numlock(char **confline);
|
|
static int DOSCONF_Switches(char **confline);
|
|
static int DOSCONF_Shell(char **confline);
|
|
static int DOSCONF_Stacks(char **confline);
|
|
static int DOSCONF_Buffers(char **confline);
|
|
static void DOSCONF_Parse(char *menuname);
|
|
|
|
static DOSCONF DOSCONF_config =
|
|
{
|
|
'E', /* lastdrive */
|
|
0, /* brk_flag */
|
|
8, /* files */
|
|
9, /* stacks_nr */
|
|
256, /* stacks_sz */
|
|
15, /* buf */
|
|
1, /* buf2 */
|
|
4, /* fcbs */
|
|
0, /* flags */
|
|
NULL, /* shell */
|
|
NULL /* country */
|
|
};
|
|
|
|
static BOOL DOSCONF_loaded = FALSE;
|
|
|
|
typedef struct {
|
|
const char *tag_name;
|
|
int (*tag_handler)(char **p);
|
|
} TAG_ENTRY;
|
|
|
|
|
|
/*
|
|
* see
|
|
* http://egeria.cm.cf.ac.uk/User/P.L.Poulain/project/internal/allinter.htm
|
|
* or
|
|
* http://www.csulb.edu/~murdock/dosindex.html
|
|
*/
|
|
|
|
static const TAG_ENTRY DOSCONF_tag_entries[] =
|
|
{
|
|
{ ";", NULL },
|
|
{ "REM ", NULL },
|
|
{ "DEVICE", DOSCONF_Device },
|
|
{ "[", DOSCONF_Menu },
|
|
{ "SUBMENU", NULL },
|
|
{ "MENUDEFAULT", DOSCONF_Menu },
|
|
{ "INCLUDE", DOSCONF_Include },
|
|
{ "INSTALL", DOSCONF_Install },
|
|
{ "DOS", DOSCONF_Dos },
|
|
{ "FCBS", DOSCONF_Fcbs },
|
|
{ "BREAK", DOSCONF_Break },
|
|
{ "FILES", DOSCONF_Files },
|
|
{ "SHELL", DOSCONF_Shell },
|
|
{ "STACKS", DOSCONF_Stacks },
|
|
{ "BUFFERS", DOSCONF_Buffers },
|
|
{ "COUNTRY", DOSCONF_Country },
|
|
{ "NUMLOCK", DOSCONF_Numlock },
|
|
{ "SWITCHES", DOSCONF_Switches },
|
|
{ "LASTDRIVE", DOSCONF_Lastdrive }
|
|
};
|
|
|
|
static FILE *DOSCONF_fd = NULL;
|
|
|
|
static char *DOSCONF_menu_default = NULL;
|
|
static int DOSCONF_menu_in_listing = 0; /* we are in the [menu] section */
|
|
static int DOSCONF_menu_skip = 0; /* the current menu gets skipped */
|
|
|
|
static void DOSCONF_skip(char **pconfline)
|
|
{
|
|
char *p;
|
|
|
|
p = *pconfline;
|
|
while ( (*p == ' ') || (*p == '\t') ) p++;
|
|
*pconfline = p;
|
|
}
|
|
|
|
static int DOSCONF_JumpToEntry(char **pconfline, char separator)
|
|
{
|
|
char *p;
|
|
|
|
p = *pconfline;
|
|
while ( (*p != separator) && (*p != '\0') ) p++;
|
|
|
|
if (*p != separator)
|
|
return 0;
|
|
else
|
|
p++;
|
|
|
|
while ( (*p == ' ') || (*p == '\t') ) p++;
|
|
*pconfline = p;
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Device(char **confline)
|
|
{
|
|
int loadhigh = 0;
|
|
|
|
*confline += 6; /* strlen("DEVICE") */
|
|
if (!(strncasecmp(*confline, "HIGH", 4)))
|
|
{
|
|
loadhigh = 1;
|
|
*confline += 4;
|
|
/* FIXME: get DEVICEHIGH parameters if avail ? */
|
|
}
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
TRACE("Loading device '%s'\n", *confline);
|
|
#if 0
|
|
DOSMOD_LoadDevice(*confline, loadhigh);
|
|
#endif
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Dos(char **confline)
|
|
{
|
|
*confline += 3; /* strlen("DOS") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
while (**confline != '\0')
|
|
{
|
|
if (!(strncasecmp(*confline, "HIGH", 4)))
|
|
{
|
|
DOSCONF_config.flags |= DOSCONF_MEM_HIGH;
|
|
*confline += 4;
|
|
}
|
|
else if (!(strncasecmp(*confline, "UMB", 3)))
|
|
{
|
|
DOSCONF_config.flags |= DOSCONF_MEM_UMB;
|
|
*confline += 3;
|
|
}
|
|
else
|
|
{
|
|
(*confline)++;
|
|
}
|
|
|
|
DOSCONF_JumpToEntry(confline, ',');
|
|
}
|
|
TRACE( "DOSCONF_Dos: HIGH is %d, UMB is %d\n",
|
|
(DOSCONF_config.flags & DOSCONF_MEM_HIGH) != 0,
|
|
(DOSCONF_config.flags & DOSCONF_MEM_UMB) != 0 );
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Fcbs(char **confline)
|
|
{
|
|
*confline += 4; /* strlen("FCBS") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
DOSCONF_config.fcbs = atoi(*confline);
|
|
if (DOSCONF_config.fcbs > 255)
|
|
{
|
|
WARN( "The FCBS value in the config.sys file is too high! Setting to 255.\n" );
|
|
DOSCONF_config.fcbs = 255;
|
|
}
|
|
TRACE( "DOSCONF_Fcbs returning %d\n", DOSCONF_config.fcbs );
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Break(char **confline)
|
|
{
|
|
*confline += 5; /* strlen("BREAK") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
if (!(strcasecmp(*confline, "ON")))
|
|
DOSCONF_config.brk_flag = 1;
|
|
TRACE( "BREAK is %d\n", DOSCONF_config.brk_flag );
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Files(char **confline)
|
|
{
|
|
*confline += 5; /* strlen("FILES") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
DOSCONF_config.files = atoi(*confline);
|
|
if (DOSCONF_config.files > 255)
|
|
{
|
|
WARN( "The FILES value in the config.sys file is too high! Setting to 255.\n" );
|
|
DOSCONF_config.files = 255;
|
|
}
|
|
if (DOSCONF_config.files < 8)
|
|
{
|
|
WARN( "The FILES value in the config.sys file is too low! Setting to 8.\n" );
|
|
DOSCONF_config.files = 8;
|
|
}
|
|
TRACE( "DOSCONF_Files returning %d\n", DOSCONF_config.files );
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Install(char **confline)
|
|
{
|
|
#if 0
|
|
int loadhigh = 0;
|
|
#endif
|
|
|
|
*confline += 7; /* strlen("INSTALL") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
TRACE( "Installing '%s'\n", *confline );
|
|
#if 0
|
|
DOSMOD_Install(*confline, loadhigh);
|
|
#endif
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Lastdrive(char **confline)
|
|
{
|
|
*confline += 9; /* strlen("LASTDRIVE") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
DOSCONF_config.lastdrive = toupper(**confline);
|
|
TRACE( "Lastdrive %c\n", DOSCONF_config.lastdrive );
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Country(char **confline)
|
|
{
|
|
*confline += 7; /* strlen("COUNTRY") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
TRACE( "Country '%s'\n", *confline );
|
|
if (DOSCONF_config.country == NULL)
|
|
DOSCONF_config.country = malloc(strlen(*confline) + 1);
|
|
strcpy(DOSCONF_config.country, *confline);
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Numlock(char **confline)
|
|
{
|
|
*confline += 7; /* strlen("NUMLOCK") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
if (!(strcasecmp(*confline, "ON")))
|
|
DOSCONF_config.flags |= DOSCONF_NUMLOCK;
|
|
TRACE( "NUMLOCK is %d\n",
|
|
(DOSCONF_config.flags & DOSCONF_NUMLOCK) != 0 );
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Switches(char **confline)
|
|
{
|
|
char *p;
|
|
|
|
*confline += 8; /* strlen("SWITCHES") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
p = strtok(*confline, "/");
|
|
do
|
|
{
|
|
if ( toupper(*p) == 'K')
|
|
DOSCONF_config.flags |= DOSCONF_KEYB_CONV;
|
|
}
|
|
while ((p = strtok(NULL, "/")));
|
|
TRACE( "'Force conventional keyboard' is %d\n",
|
|
(DOSCONF_config.flags & DOSCONF_KEYB_CONV) != 0 );
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Shell(char **confline)
|
|
{
|
|
*confline += 5; /* strlen("SHELL") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
TRACE( "Shell '%s'\n", *confline );
|
|
if (DOSCONF_config.shell == NULL)
|
|
DOSCONF_config.shell = malloc(strlen(*confline) + 1);
|
|
strcpy(DOSCONF_config.shell, *confline);
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Stacks(char **confline)
|
|
{
|
|
|
|
*confline += 6; /* strlen("STACKS") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
DOSCONF_config.stacks_nr = atoi(strtok(*confline, ","));
|
|
DOSCONF_config.stacks_sz = atoi((strtok(NULL, ",")));
|
|
TRACE( "%d stacks of size %d\n",
|
|
DOSCONF_config.stacks_nr, DOSCONF_config.stacks_sz );
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Buffers(char **confline)
|
|
{
|
|
char *p;
|
|
|
|
*confline += 7; /* strlen("BUFFERS") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
p = strtok(*confline, ",");
|
|
DOSCONF_config.buf = atoi(p);
|
|
if ((p = strtok(NULL, ",")))
|
|
DOSCONF_config.buf2 = atoi(p);
|
|
TRACE( "%d primary buffers, %d secondary buffers\n",
|
|
DOSCONF_config.buf, DOSCONF_config.buf2 );
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Menu(char **confline)
|
|
{
|
|
if (!(strncasecmp(*confline, "[MENU]", 6)))
|
|
{
|
|
DOSCONF_menu_in_listing = 1;
|
|
}
|
|
else if ((!(strncasecmp(*confline, "[COMMON]", 8)))
|
|
|| (!(strncasecmp(*confline, "[WINE]", 6))))
|
|
{
|
|
DOSCONF_menu_skip = 0;
|
|
}
|
|
else if (**confline == '[')
|
|
{
|
|
(*confline)++;
|
|
if ((DOSCONF_menu_default)
|
|
&& (!(strncasecmp(*confline, DOSCONF_menu_default,
|
|
strlen(DOSCONF_menu_default)))))
|
|
{
|
|
free(DOSCONF_menu_default);
|
|
DOSCONF_menu_default = NULL;
|
|
DOSCONF_menu_skip = 0;
|
|
}
|
|
else
|
|
DOSCONF_menu_skip = 1;
|
|
DOSCONF_menu_in_listing = 0;
|
|
}
|
|
else if (!(strncasecmp(*confline, "menudefault", 11))
|
|
&& (DOSCONF_menu_in_listing))
|
|
{
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
*confline = strtok(*confline, ",");
|
|
DOSCONF_menu_default = malloc(strlen(*confline) + 1);
|
|
strcpy(DOSCONF_menu_default, *confline);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int DOSCONF_Include(char **confline)
|
|
{
|
|
fpos_t oldpos;
|
|
char *temp;
|
|
|
|
*confline += 7; /* strlen("INCLUDE") */
|
|
if (!(DOSCONF_JumpToEntry(confline, '='))) return 0;
|
|
fgetpos(DOSCONF_fd, &oldpos);
|
|
fseek(DOSCONF_fd, 0, SEEK_SET);
|
|
TRACE( "Including menu '%s'\n", *confline );
|
|
temp = malloc(strlen(*confline) + 1);
|
|
strcpy(temp, *confline);
|
|
DOSCONF_Parse(temp);
|
|
free(temp);
|
|
fsetpos(DOSCONF_fd, &oldpos);
|
|
return 1;
|
|
}
|
|
|
|
static void DOSCONF_Parse(char *menuname)
|
|
{
|
|
char confline[256];
|
|
char *p, *trail;
|
|
int i;
|
|
|
|
if (menuname != NULL) /* we need to jump to a certain sub menu */
|
|
{
|
|
while (fgets(confline, 255, DOSCONF_fd))
|
|
{
|
|
p = confline;
|
|
DOSCONF_skip(&p);
|
|
if (*p == '[')
|
|
{
|
|
p++;
|
|
if (!(trail = strrchr(p, ']')))
|
|
return;
|
|
if (!(strncasecmp(p, menuname, (int)trail - (int)p)))
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
while (fgets(confline, 255, DOSCONF_fd))
|
|
{
|
|
p = confline;
|
|
DOSCONF_skip(&p);
|
|
|
|
if ((menuname) && (*p == '['))
|
|
/*
|
|
* we were handling a specific sub menu,
|
|
* but now next menu begins
|
|
*/
|
|
break;
|
|
|
|
if ((trail = strrchr(confline, '\n')))
|
|
*trail = '\0';
|
|
if ((trail = strrchr(confline, '\r')))
|
|
*trail = '\0';
|
|
if (!(DOSCONF_menu_skip))
|
|
{
|
|
for (i = 0; i < sizeof(DOSCONF_tag_entries) / sizeof(TAG_ENTRY);
|
|
i++)
|
|
if (!(strncasecmp(p, DOSCONF_tag_entries[i].tag_name,
|
|
strlen(DOSCONF_tag_entries[i].tag_name))))
|
|
{
|
|
TRACE( "tag '%s'\n", DOSCONF_tag_entries[i].tag_name );
|
|
if (DOSCONF_tag_entries[i].tag_handler != NULL)
|
|
DOSCONF_tag_entries[i].tag_handler(&p);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* the current menu gets skipped */
|
|
DOSCONF_Menu(&p);
|
|
}
|
|
}
|
|
}
|
|
|
|
DOSCONF *DOSCONF_GetConfig(void)
|
|
{
|
|
HKEY hkey;
|
|
CHAR filename[MAX_PATH];
|
|
|
|
if (DOSCONF_loaded)
|
|
return &DOSCONF_config;
|
|
|
|
/* default value */
|
|
strcpy( filename, "*" );
|
|
|
|
if (!RegOpenKeyA(HKEY_LOCAL_MACHINE,
|
|
"Software\\Wine\\Wine\\Config\\wine",
|
|
&hkey))
|
|
{
|
|
DWORD type;
|
|
DWORD count = sizeof(filename);
|
|
|
|
RegQueryValueExA(hkey, "config.sys", 0, &type, filename, &count);
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
if (strcmp(filename, "*") && *filename != '\0')
|
|
{
|
|
CHAR fullname[MAX_PATH];
|
|
|
|
if (wine_get_unix_file_name(filename, fullname, sizeof(fullname)))
|
|
DOSCONF_fd = fopen(fullname, "r");
|
|
|
|
if (DOSCONF_fd)
|
|
{
|
|
DOSCONF_Parse(NULL);
|
|
fclose(DOSCONF_fd);
|
|
DOSCONF_fd = NULL;
|
|
}
|
|
else
|
|
{
|
|
WARN( "Couldn't open config.sys file given as %s in"
|
|
" configuration file, section [wine]!\n",
|
|
filename );
|
|
}
|
|
}
|
|
|
|
DOSCONF_loaded = TRUE;
|
|
return &DOSCONF_config;
|
|
}
|