Mon Dec 11 19:08:55 1995 Alexandre Julliard <julliard@sunsite.unc.edu> * [misc/lstr.c] Replaced wine_strncpy() by a 32-bit version of lstrcpyn(), since they do the same job. * [tools/build.c] Fixed __attribute__((stdcall)) to make it compile with gcc versions under 2.7. Doesn't mean it will run OK though... Sat Dec 09 13:22:58 1995 Cameron Heide <heide@ee.ualberta.ca> * [include/kernel32.h] [include/winerror.h] Added file attribute definitions and more error codes. * [win32/error.c] Added some rudimentary errno-to-Win32 error conversion code. * [win32/file.c] Added to GetFileInformationByHandle, filled in some known error codes, and switched to dprintf_win32. * [win32/time.c] Added GetLocalTime. Fri Dec 8 14:37:39 1995 Jim Peterson <jspeter@birch.ee.vt.edu> * [controls/combo.c] Converted functions of the type LONG _(HWND,WORD,LONG) to the type LRESULT _(HWND,WPARAM,LPARAM) where needed. * [include/libres.h] Restructured libres prototypes to closer match the windows API. * [include/windows.h] Changed several API prototypes' parameter types from 'short' to INT, which is #defined as short in the emulator, but is a normal int in WINELIB32. Also changed SEGPTR from DWORD to void* when WINELIB32. (This creates a lot of warnings at library-compile time, but less warnings at app-compile time. I'll remove the warnings soon.) * [loader/resource.c] Fixed parameter mismatch in call to LIBRES_FindResource(). Changed various implementations of the LIBRES_* API functions. * [loader/signal.c] Deleted local 'i' from win_fault(), since it was unused. * [objects/bitblt.c] Mirrored changes to include/windows.h mentioned above. * [toolkit/hello3.c] Changed LoadMenuIndirect() call to LoadMenu() to test the new resource registration technique. * [toolkit/libres.c] Removed definition of 'struct resource' and fixed bugs in the resource implementation. Implemented LIBRES_FindResource. * [windows/graphics.c] Mirrored changes to include/windows.h mentioned above. Thu Dec 7 23:15:56 1995 Martin von Loewis <loewis@informatik.hu-berlin.de> * [controls/edit.c] LOCAL_HeapExists: Changed parameter to HANDLE. For WineLib, return true * [controls/listbox.c] CreateListBoxStruct: Initialize HeapSel to 0 for WineLib * [include/listbox.h] change HeapSel from WORD to HANDLE * [include/resource.h][rc/winerc.c] struct ResourceTable: removed struct resource: moved to header file autoregister resources if supported by compiler * [memory/local.h] LOCAL_GetHeap: expect HANDLE rather than WORD * [toolkit/Makefile.in] Add ALLCFLAGS to make hello3 * [toolkit/heap.c] LocalFree, HEAP_Free: handle 0 parameter gracefully Wed Dec 06 15:34:23 1995 Greg Cooper <cooper@ima-inc.com> * [misc/winsocket.c] Fixed the msgsnd and msgrcv errors that winsock programs get. Wed Dec 06 12:47:23 MET 1995 Sven Verdoolaege <skimo@dns.ufsia.ac.be> * [if1632/kernel.spec] Fixed _hread and _hwrite return type * [if1632/relay32.c] [loader/pe_image.c] Hacked loading of PE-dll's in * [win32/advapi.c] Added stubs for RegCreateKeyEx, RegSetValueEx, RegQueryValueEx * [win32/file.c] Added stubs for OpenFileMapping, CreateFileMapping, MapViewOfFileEx * [win32/process.c] Added stubs for CreateMutexA, ReleaseMutex, CreateEventA, WaitForSingleObject, DuplicateHandle, GetCurrentProcess Mon Dec 04 13:06:37 1995 Bernd Schmidt <crux@pool.informatik.rwth-aachen.de> * [include/wine.h] [misc/lstr.c] Define wine_strncpy(). This function does not pad the buffer with zeroes like GNU strncpy(), which might break some Windows programs that pass bogus size arguments. * [loader/module.c]: GetModuleFileName(), [misc/commdlg.c]: GetFileTitle(), [misc/keyboard.c], [misc/lstr.c]: lstrcpyn(), [misc/ole2nls.c], [misc/profile.c], [multimedia/mcistring.c], [multimedia/mmsystem.c], [objects/font.c]: Use wine_strncpy() where strings are returned to Windows programs. * [objects/metafile.c] PlayMetafile(): Clear the handle table before using it. * [misc/shell.c] [misc/main.c] Rename SHELL_RegCheckForRoot() to SHELL_Init() and call it from main(). * [misc/profile.c] load(): Need to handle comments. * [toolkit/libres.c] Make it compile. * [windows/nonclient.c] Use MAKE_SEGPTR macro in two places where a user heap block used to be allocated instead. Sat Dec 02 16:43:43 1995 Ramon Garcia <ramon@ie3.clubs.etsit.upm.es> * [windows/winpos.c] In function SetWindowPos: do not redraw the parent of a window if the specified window is placed on the top. This avoids that ShowWindow(hwnd,1) hides hwnd instead of showing it. Sat Dec 02 11:00:00 1995 Alex Korobka <alex@phm30.pharm.sunysb.edu> * [windows/scroll.c] Now it can scroll children along with the client region of parent window. Tried to optimize update region calculation. * [windows/mdi.c] ScrollChildren function, more other features added. Basically a rewrite. * [windows/winpos.c] [windows/focus.c] Reimplemented window activation and focus handling. * [windows/nonclient.c] Added new flag WIN_NCACTIVATED. * [windows/message.c] [loader/task.c] Small changes (to maintain linked list of message queues). Wed Nov 29 15:51:48 1995 Daniel Schepler <daniel@shep13.wustl.edu> * [include/options.h] [misc/main.c] [windows/defwnd.c] [windows/event.c] [windows/nonclient.c] [windows/win.c] [Wine.man] Implemented a -managed option to replace the standard Windows frame of top-level windows with the window manager's decorations. If a top-level window makes its own frame, this will still show up, inside the window manager decorations (I believe ctl3dv2.dll would do this, although I can't test this).
436 lines
10 KiB
C
436 lines
10 KiB
C
/*
|
|
* Initialization-File Functions.
|
|
*
|
|
* Copyright (c) 1993 Miguel de Icaza
|
|
*
|
|
* 1/Dec o Corrected return values for Get*ProfileString
|
|
*
|
|
* o Now, if AppName == NULL in Get*ProfileString it returns a list
|
|
* of the KeyNames (as documented in the MS-SDK).
|
|
*
|
|
* o if KeyValue == NULL now clears the value in Get*ProfileString
|
|
*
|
|
* 20/Apr SL - I'm not sure where these definitions came from, but my SDK
|
|
* has a NULL KeyValue returning a list of KeyNames, and a NULL
|
|
* AppName undefined. I changed GetSetProfile to match. This makes
|
|
* PROGMAN.EXE do the right thing.
|
|
*/
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
|
|
#include "wine.h"
|
|
#include "windows.h"
|
|
#include "dos_fs.h"
|
|
#include "toolhelp.h"
|
|
#include "stddebug.h"
|
|
#include "debug.h"
|
|
#include "xmalloc.h"
|
|
|
|
#define STRSIZE 255
|
|
|
|
typedef struct TKeys {
|
|
char *KeyName;
|
|
char *Value;
|
|
struct TKeys *link;
|
|
} TKeys;
|
|
|
|
typedef struct TSecHeader {
|
|
char *AppName;
|
|
TKeys *Keys;
|
|
struct TSecHeader *link;
|
|
} TSecHeader;
|
|
|
|
typedef struct TProfile {
|
|
char *FileName;
|
|
char *FullName;
|
|
TSecHeader *Section;
|
|
struct TProfile *link;
|
|
int changed;
|
|
} TProfile;
|
|
|
|
TProfile *Current = 0;
|
|
TProfile *Base = 0;
|
|
|
|
static TSecHeader *is_loaded (char *FileName)
|
|
{
|
|
TProfile *p = Base;
|
|
|
|
while (p){
|
|
if (!strcasecmp (FileName, p->FileName)){
|
|
Current = p;
|
|
return p->Section;
|
|
}
|
|
p = p->link;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static char *GetIniFileName(char *name, char *dir)
|
|
{
|
|
char temp[256];
|
|
|
|
if (strchr(name, '/'))
|
|
return name;
|
|
|
|
if (strchr(name, '\\'))
|
|
return DOS_GetUnixFileName(name);
|
|
|
|
strcpy(temp, dir);
|
|
strcat(temp, "\\");
|
|
strcat(temp, name);
|
|
|
|
return DOS_GetUnixFileName(temp);
|
|
}
|
|
|
|
static TSecHeader *load (char *filename, char **pfullname)
|
|
{
|
|
FILE *f;
|
|
TSecHeader *SecHeader = 0;
|
|
char CharBuffer [STRSIZE];
|
|
char *bufptr;
|
|
char *lastnonspc;
|
|
int bufsize;
|
|
char *file;
|
|
int c;
|
|
char path[MAX_PATH+1];
|
|
BOOL firstbrace;
|
|
|
|
*pfullname = NULL;
|
|
|
|
/* Try the Windows directory */
|
|
|
|
GetWindowsDirectory(path, sizeof(path));
|
|
file = GetIniFileName(filename, path);
|
|
|
|
dprintf_profile(stddeb,"Load %s\n", file);
|
|
f = fopen(file, "r");
|
|
|
|
if (f == NULL) {
|
|
/* Try the path of the current executable */
|
|
|
|
if (GetCurrentTask())
|
|
{
|
|
char *p;
|
|
GetModuleFileName( GetCurrentTask(), path, MAX_PATH );
|
|
if ((p = strrchr( path, '\\' )))
|
|
{
|
|
p[1] = '\0';
|
|
file = GetIniFileName(filename, path);
|
|
f = fopen(file, "r");
|
|
}
|
|
}
|
|
}
|
|
if (f == NULL) {
|
|
fprintf(stderr, "profile.c: load() can't find file %s\n", filename);
|
|
/* FIXME: we ought to create it now (in which directory?) */
|
|
return NULL;
|
|
}
|
|
|
|
*pfullname = strdup(file);
|
|
dprintf_profile(stddeb,"Loading %s\n", file);
|
|
|
|
firstbrace = TRUE;
|
|
for(;;) {
|
|
c = fgetc(f);
|
|
if (c == EOF) goto finished;
|
|
|
|
if (isspace(c))
|
|
continue;
|
|
if (c == ';') {
|
|
do {
|
|
c = fgetc(f);
|
|
} while (!(c == EOF || c == '\n'));
|
|
if (c == EOF) goto finished;
|
|
}
|
|
if (c == '[') {
|
|
TSecHeader *temp = SecHeader;
|
|
|
|
SecHeader = (TSecHeader *) xmalloc (sizeof (TSecHeader));
|
|
SecHeader->link = temp;
|
|
SecHeader->Keys = NULL;
|
|
do {
|
|
c = fgetc(f);
|
|
if (c == EOF) goto bad_file;
|
|
} while (isspace(c));
|
|
bufptr = lastnonspc = CharBuffer;
|
|
bufsize = 0;
|
|
do {
|
|
if (c != ']') {
|
|
bufsize++;
|
|
*bufptr++ = c;
|
|
if (!isspace(c))
|
|
lastnonspc = bufptr;
|
|
} else
|
|
break;
|
|
c = fgetc(f);
|
|
if (c == EOF) goto bad_file;
|
|
} while(bufsize < STRSIZE-1);
|
|
*lastnonspc = 0;
|
|
if (!strlen(CharBuffer))
|
|
fprintf(stderr, "warning: empty section name in ini file\n");
|
|
SecHeader->AppName = strdup (CharBuffer);
|
|
dprintf_profile(stddeb,"%s: section %s\n", file, CharBuffer);
|
|
firstbrace = FALSE;
|
|
} else if (SecHeader) {
|
|
TKeys *temp = SecHeader->Keys;
|
|
BOOL skipspc;
|
|
|
|
if (firstbrace)
|
|
goto bad_file;
|
|
bufptr = lastnonspc = CharBuffer;
|
|
bufsize = 0;
|
|
do {
|
|
if (c != '=') {
|
|
bufsize++;
|
|
*bufptr++ = c;
|
|
if (!isspace(c))
|
|
lastnonspc = bufptr;
|
|
} else
|
|
break;
|
|
c = fgetc(f);
|
|
if (c == EOF) goto bad_file;
|
|
} while(bufsize < STRSIZE-1);
|
|
*lastnonspc = 0;
|
|
if (!strlen(CharBuffer))
|
|
fprintf(stderr, "warning: empty key name in ini file\n");
|
|
SecHeader->Keys = (TKeys *) xmalloc (sizeof (TKeys));
|
|
SecHeader->Keys->link = temp;
|
|
SecHeader->Keys->KeyName = strdup (CharBuffer);
|
|
|
|
dprintf_profile(stddeb,"%s: key %s\n", file, CharBuffer);
|
|
|
|
bufptr = lastnonspc = CharBuffer;
|
|
bufsize = 0;
|
|
skipspc = TRUE;
|
|
do {
|
|
c = fgetc(f);
|
|
if (c == EOF || c == '\n' || c == ';') break;
|
|
if (!isspace(c) || !skipspc) {
|
|
skipspc = FALSE;
|
|
bufsize++;
|
|
*bufptr++ = c;
|
|
if (!isspace(c))
|
|
lastnonspc = bufptr;
|
|
}
|
|
} while(bufsize < STRSIZE-1);
|
|
*lastnonspc = 0;
|
|
SecHeader->Keys->Value = strdup (CharBuffer);
|
|
dprintf_profile (stddeb, "[%s] (%s)=%s\n", SecHeader->AppName,
|
|
SecHeader->Keys->KeyName, SecHeader->Keys->Value);
|
|
if (c == ';') {
|
|
do {
|
|
c = fgetc(f);
|
|
} while (!(c == EOF || c == '\n'));
|
|
if (c == EOF)
|
|
goto finished;
|
|
}
|
|
}
|
|
|
|
}
|
|
bad_file:
|
|
fprintf(stderr, "warning: bad ini file\n");
|
|
finished:
|
|
return SecHeader;
|
|
}
|
|
|
|
static void new_key (TSecHeader *section, char *KeyName, char *Value)
|
|
{
|
|
TKeys *key;
|
|
|
|
key = (TKeys *) xmalloc (sizeof (TKeys));
|
|
key->KeyName = strdup (KeyName);
|
|
key->Value = strdup (Value);
|
|
key->link = section->Keys;
|
|
section->Keys = key;
|
|
}
|
|
|
|
static short GetSetProfile (int set, LPSTR AppName, LPSTR KeyName,
|
|
LPSTR Default, LPSTR ReturnedString, short Size,
|
|
LPSTR FileName)
|
|
|
|
{
|
|
TProfile *New;
|
|
TSecHeader *section;
|
|
TKeys *key;
|
|
|
|
/* Supposedly Default should NEVER be NULL. But sometimes it is. */
|
|
if (Default == NULL)
|
|
Default = "";
|
|
|
|
if (!(section = is_loaded (FileName))){
|
|
New = (TProfile *) xmalloc (sizeof (TProfile));
|
|
New->link = Base;
|
|
New->FileName = strdup (FileName);
|
|
New->Section = load (FileName, &New->FullName);
|
|
New->changed = FALSE;
|
|
Base = New;
|
|
section = New->Section;
|
|
Current = New;
|
|
}
|
|
|
|
/* Start search */
|
|
for (; section; section = section->link){
|
|
if (strcasecmp (section->AppName, AppName))
|
|
continue;
|
|
|
|
/* If no key value given, then list all the keys */
|
|
if ((!KeyName) && (!set)){
|
|
char *p = ReturnedString;
|
|
int left = Size - 2;
|
|
int slen;
|
|
|
|
dprintf_profile(stddeb,"GetSetProfile // KeyName == NULL, Enumeration !\n");
|
|
for (key = section->Keys; key; key = key->link){
|
|
if (left < 1) {
|
|
dprintf_profile(stddeb,"GetSetProfile // No more storage for enum !\n");
|
|
return Size - 2;
|
|
}
|
|
slen = MIN(strlen(key->KeyName) + 1, left);
|
|
lstrcpyn(p, key->KeyName, slen);
|
|
dprintf_profile(stddeb,"GetSetProfile // enum '%s' !\n", p);
|
|
left -= slen;
|
|
p += slen;
|
|
}
|
|
*p = '\0';
|
|
return Size - 2 - left;
|
|
}
|
|
for (key = section->Keys; key; key = key->link){
|
|
int slen;
|
|
if (strcasecmp (key->KeyName, KeyName))
|
|
continue;
|
|
if (set){
|
|
free (key->Value);
|
|
key->Value = strdup (Default ? Default : "");
|
|
Current->changed=TRUE;
|
|
return 1;
|
|
}
|
|
slen = MIN(strlen(key->Value)+1, Size);
|
|
lstrcpyn(ReturnedString, key->Value, slen);
|
|
dprintf_profile(stddeb,"GetSetProfile // Return ``%s''\n", ReturnedString);
|
|
return 1;
|
|
}
|
|
/* If Getting the information, then don't write the information
|
|
to the INI file, need to run a couple of tests with windog */
|
|
/* No key found */
|
|
if (set) {
|
|
new_key (section, KeyName, Default);
|
|
} else {
|
|
int slen = MIN(strlen(Default)+1, Size);
|
|
lstrcpyn(ReturnedString, Default, slen);
|
|
dprintf_profile(stddeb,"GetSetProfile // Key not found\n");
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/* Non existent section */
|
|
if (set){
|
|
section = (TSecHeader *) xmalloc (sizeof (TSecHeader));
|
|
section->AppName = strdup (AppName);
|
|
section->Keys = 0;
|
|
new_key (section, KeyName, Default);
|
|
section->link = Current->Section;
|
|
Current->Section = section;
|
|
Current->changed = TRUE;
|
|
} else {
|
|
int slen = MIN(strlen(Default)+1, Size);
|
|
lstrcpyn(ReturnedString, Default, slen);
|
|
dprintf_profile(stddeb,"GetSetProfile // Section not found\n");
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
short GetPrivateProfileString (LPSTR AppName, LPSTR KeyName,
|
|
LPSTR Default, LPSTR ReturnedString,
|
|
short Size, LPSTR FileName)
|
|
{
|
|
int v;
|
|
|
|
dprintf_profile(stddeb,"GetPrivateProfileString ('%s', '%s', '%s', %p, %d, %s\n",
|
|
AppName, KeyName, Default, ReturnedString, Size, FileName);
|
|
v = GetSetProfile (0,AppName,KeyName,Default,ReturnedString,Size,FileName);
|
|
if (AppName)
|
|
return strlen (ReturnedString);
|
|
else
|
|
return Size - v;
|
|
}
|
|
|
|
int GetProfileString (LPSTR AppName, LPSTR KeyName, LPSTR Default,
|
|
LPSTR ReturnedString, int Size)
|
|
{
|
|
return GetPrivateProfileString (AppName, KeyName, Default,
|
|
ReturnedString, Size, WIN_INI);
|
|
}
|
|
|
|
WORD GetPrivateProfileInt (LPSTR AppName, LPSTR KeyName, short Default,
|
|
LPSTR File)
|
|
{
|
|
static char IntBuf[10];
|
|
static char buf[10];
|
|
|
|
sprintf (buf, "%d", Default);
|
|
|
|
/* Check the exact semantic with the SDK */
|
|
GetPrivateProfileString (AppName, KeyName, buf, IntBuf, 10, File);
|
|
if (!strcasecmp (IntBuf, "true"))
|
|
return 1;
|
|
if (!strcasecmp (IntBuf, "yes"))
|
|
return 1;
|
|
return strtoul( IntBuf, NULL, 0 );
|
|
}
|
|
|
|
WORD GetProfileInt (LPSTR AppName, LPSTR KeyName, int Default)
|
|
{
|
|
return GetPrivateProfileInt (AppName, KeyName, Default, WIN_INI);
|
|
}
|
|
|
|
BOOL WritePrivateProfileString (LPSTR AppName, LPSTR KeyName, LPSTR String,
|
|
LPSTR FileName)
|
|
{
|
|
if (!AppName || !KeyName || !String) /* Flush file to disk */
|
|
return TRUE;
|
|
return GetSetProfile (1, AppName, KeyName, String, "", 0, FileName);
|
|
}
|
|
|
|
BOOL WriteProfileString (LPSTR AppName, LPSTR KeyName, LPSTR String)
|
|
{
|
|
return (WritePrivateProfileString (AppName, KeyName, String, WIN_INI));
|
|
}
|
|
|
|
static void dump_keys (FILE *profile, TKeys *p)
|
|
{
|
|
if (!p)
|
|
return;
|
|
dump_keys (profile, p->link);
|
|
fprintf (profile, "%s=%s\r\n", p->KeyName, p->Value);
|
|
}
|
|
|
|
static void dump_sections (FILE *profile, TSecHeader *p)
|
|
{
|
|
if (!p)
|
|
return;
|
|
dump_sections (profile, p->link);
|
|
fprintf (profile, "\r\n[%s]\r\n", p->AppName);
|
|
dump_keys (profile, p->Keys);
|
|
}
|
|
|
|
static void dump_profile (TProfile *p)
|
|
{
|
|
FILE *profile;
|
|
|
|
if (!p)
|
|
return;
|
|
dump_profile (p->link);
|
|
if(!p->changed)
|
|
return;
|
|
if (p->FullName && (profile = fopen (p->FullName, "w")) != NULL){
|
|
dump_sections (profile, p->Section);
|
|
fclose (profile);
|
|
}
|
|
}
|
|
|
|
void sync_profiles (void)
|
|
{
|
|
dump_profile (Base);
|
|
}
|