blob: aec6153eac9462cf1e42b52fb3c3152790ecd0fd [file] [log] [blame]
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001/*
2 * Profile functions
3 *
4 * Copyright 1993 Miguel de Icaza
5 * Copyright 1996 Alexandre Julliard
6 */
7
8#include <ctype.h>
Alexandre Julliard73be8d12000-12-06 20:25:11 +00009#include <fcntl.h>
Alexandre Julliard02ed4c21996-03-02 19:34:10 +000010#include <stdlib.h>
Alexandre Julliard7e56f681996-01-31 19:02:28 +000011#include <string.h>
Jeremy Whited3e22d92000-02-10 19:03:02 +000012#include <stdio.h>
Alexandre Julliard829fe321998-07-26 14:27:39 +000013#include <sys/stat.h>
Alexandre Julliard647876e2000-01-25 01:35:01 +000014#include <sys/types.h>
15#include <pwd.h>
16#include <unistd.h>
Alexandre Julliard829fe321998-07-26 14:27:39 +000017
Alexandre Julliard24a62ab2000-11-28 22:40:56 +000018#include "windef.h"
Marcus Meissner317af321999-02-17 13:51:06 +000019#include "winbase.h"
Alexandre Julliard24a62ab2000-11-28 22:40:56 +000020#include "winnls.h"
Marcus Meissner9b4fcf71999-12-04 04:19:55 +000021#include "winerror.h"
Marcus Meissner317af321999-02-17 13:51:06 +000022#include "wine/winbase16.h"
Alexandre Julliard00377a72000-02-19 20:50:00 +000023#include "winreg.h"
Alexandre Julliardc6c09441997-01-12 18:32:19 +000024#include "file.h"
Alexandre Julliard7ebe1a41996-12-22 18:27:48 +000025#include "heap.h"
Alexandre Julliard359f497e1999-07-04 16:02:24 +000026#include "debugtools.h"
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000027#include "options.h"
Alexandre Julliard00377a72000-02-19 20:50:00 +000028#include "server.h"
Alexandre Julliard7e56f681996-01-31 19:02:28 +000029
Jeremy Whited3e22d92000-02-10 19:03:02 +000030DEFAULT_DEBUG_CHANNEL(profile);
Patrik Stridvallb4b9fae1999-04-19 14:56:29 +000031
Alexandre Julliard7e56f681996-01-31 19:02:28 +000032typedef struct tagPROFILEKEY
33{
34 char *name;
35 char *value;
36 struct tagPROFILEKEY *next;
37} PROFILEKEY;
38
39typedef struct tagPROFILESECTION
40{
41 char *name;
42 struct tagPROFILEKEY *key;
43 struct tagPROFILESECTION *next;
44} PROFILESECTION;
45
46
47typedef struct
48{
Eric Poueched155751999-03-25 13:24:08 +000049 BOOL changed;
Alexandre Julliard7e56f681996-01-31 19:02:28 +000050 PROFILESECTION *section;
51 char *dos_name;
Alexandre Julliardc6c09441997-01-12 18:32:19 +000052 char *unix_name;
53 char *filename;
Alexandre Julliard829fe321998-07-26 14:27:39 +000054 time_t mtime;
Alexandre Julliard7e56f681996-01-31 19:02:28 +000055} PROFILE;
56
57
Alexandre Julliard829fe321998-07-26 14:27:39 +000058#define N_CACHED_PROFILES 10
59
60/* Cached profile files */
61static PROFILE *MRUProfile[N_CACHED_PROFILES]={NULL};
62
63#define CurProfile (MRUProfile[0])
Alexandre Julliard7e56f681996-01-31 19:02:28 +000064
Alexandre Julliard00377a72000-02-19 20:50:00 +000065/* wine.ini config file registry root */
66static HKEY wine_profile_key;
Alexandre Julliard7e56f681996-01-31 19:02:28 +000067
68#define PROFILE_MAX_LINE_LEN 1024
69
70/* Wine profile name in $HOME directory; must begin with slash */
71static const char PROFILE_WineIniName[] = "/.winerc";
72
Alexandre Julliardd30dfd21998-09-27 18:28:36 +000073/* Wine profile: the profile file being used */
74static char PROFILE_WineIniUsed[MAX_PATHNAME_LEN] = "";
75
Alexandre Julliard7e56f681996-01-31 19:02:28 +000076/* Check for comments in profile */
77#define IS_ENTRY_COMMENT(str) ((str)[0] == ';')
78
Alexandre Julliard00377a72000-02-19 20:50:00 +000079static const WCHAR wininiW[] = { 'w','i','n','.','i','n','i',0 };
Alexandre Julliardda0cfb31996-12-01 17:17:47 +000080
Alexandre Julliard227a0892000-04-08 21:06:06 +000081static CRITICAL_SECTION PROFILE_CritSect = CRITICAL_SECTION_INIT;
Eric Poueched155751999-03-25 13:24:08 +000082
Andreas Mohr17b1f462000-01-30 21:15:14 +000083static const char hex[16] = "0123456789ABCDEF";
Alexandre Julliard647876e2000-01-25 01:35:01 +000084
85/***********************************************************************
Alexandre Julliard7e56f681996-01-31 19:02:28 +000086 * PROFILE_CopyEntry
87 *
88 * Copy the content of an entry into a buffer, removing quotes, and possibly
89 * translating environment variables.
90 */
91static void PROFILE_CopyEntry( char *buffer, const char *value, int len,
92 int handle_env )
93{
94 char quote = '\0';
95 const char *p;
96
Dmitry Timoshkovfdc78a32000-12-19 19:37:03 +000097 if(!buffer) return;
98
Alexandre Julliard7e56f681996-01-31 19:02:28 +000099 if ((*value == '\'') || (*value == '\"'))
100 {
101 if (value[1] && (value[strlen(value)-1] == *value)) quote = *value++;
102 }
103
104 if (!handle_env)
105 {
Alexandre Julliarda3960291999-02-26 11:11:13 +0000106 lstrcpynA( buffer, value, len );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000107 if (quote && (len >= strlen(value))) buffer[strlen(buffer)-1] = '\0';
108 return;
109 }
110
111 for (p = value; (*p && (len > 1)); *buffer++ = *p++, len-- )
112 {
113 if ((*p == '$') && (p[1] == '{'))
114 {
115 char env_val[1024];
116 const char *env_p;
117 const char *p2 = strchr( p, '}' );
118 if (!p2) continue; /* ignore it */
Francois Gouget6d77d3a2000-03-25 21:44:35 +0000119 lstrcpynA(env_val, p + 2, min( sizeof(env_val), (int)(p2-p)-1 ));
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000120 if ((env_p = getenv( env_val )) != NULL)
121 {
Alexandre Julliarda3960291999-02-26 11:11:13 +0000122 lstrcpynA( buffer, env_p, len );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000123 buffer += strlen( buffer );
124 len -= strlen( buffer );
125 }
126 p = p2 + 1;
127 }
128 }
Eric Pouechf54c95f1999-10-31 17:32:57 +0000129 if (quote && (len > 1)) buffer--;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000130 *buffer = '\0';
131}
132
133
134/***********************************************************************
135 * PROFILE_Save
136 *
137 * Save a profile tree to a file.
138 */
139static void PROFILE_Save( FILE *file, PROFILESECTION *section )
140{
141 PROFILEKEY *key;
142
143 for ( ; section; section = section->next)
144 {
Alexandre Julliard23946ad1997-06-16 17:43:53 +0000145 if (section->name) fprintf( file, "\r\n[%s]\r\n", section->name );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000146 for (key = section->key; key; key = key->next)
147 {
148 fprintf( file, "%s", key->name );
149 if (key->value) fprintf( file, "=%s", key->value );
Alexandre Julliard7d654eb1996-02-25 11:36:22 +0000150 fprintf( file, "\r\n" );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000151 }
152 }
153}
154
155
156/***********************************************************************
157 * PROFILE_Free
158 *
159 * Free a profile tree.
160 */
161static void PROFILE_Free( PROFILESECTION *section )
162{
163 PROFILESECTION *next_section;
164 PROFILEKEY *key, *next_key;
165
166 for ( ; section; section = next_section)
167 {
Alexandre Julliard90476d62000-02-16 22:47:24 +0000168 if (section->name) HeapFree( GetProcessHeap(), 0, section->name );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000169 for (key = section->key; key; key = next_key)
170 {
171 next_key = key->next;
Alexandre Julliard90476d62000-02-16 22:47:24 +0000172 if (key->name) HeapFree( GetProcessHeap(), 0, key->name );
173 if (key->value) HeapFree( GetProcessHeap(), 0, key->value );
174 HeapFree( GetProcessHeap(), 0, key );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000175 }
176 next_section = section->next;
Alexandre Julliard90476d62000-02-16 22:47:24 +0000177 HeapFree( GetProcessHeap(), 0, section );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000178 }
179}
180
Andreas Mohr17b1f462000-01-30 21:15:14 +0000181static inline int PROFILE_isspace(char c)
182{
Alexandre Julliardf90efa91998-06-14 15:24:15 +0000183 if (isspace(c)) return 1;
184 if (c=='\r' || c==0x1a) return 1;
185 /* CR and ^Z (DOS EOF) are spaces too (found on CD-ROMs) */
186 return 0;
187}
188
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000189
190/***********************************************************************
191 * PROFILE_Load
192 *
193 * Load a profile tree from a file.
194 */
195static PROFILESECTION *PROFILE_Load( FILE *file )
196{
197 char buffer[PROFILE_MAX_LINE_LEN];
198 char *p, *p2;
199 int line = 0;
200 PROFILESECTION *section, *first_section;
Alexandre Julliard642d3131998-07-12 19:29:36 +0000201 PROFILESECTION **next_section;
202 PROFILEKEY *key, *prev_key, **next_key;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000203
Dimitrie O. Paunabdbced2000-04-29 14:20:28 +0000204 first_section = HeapAlloc( GetProcessHeap(), 0, sizeof(*section) );
205 if(first_section == NULL) return NULL;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000206 first_section->name = NULL;
207 first_section->key = NULL;
208 first_section->next = NULL;
Alexandre Julliard642d3131998-07-12 19:29:36 +0000209 next_section = &first_section->next;
210 next_key = &first_section->key;
211 prev_key = NULL;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000212
213 while (fgets( buffer, PROFILE_MAX_LINE_LEN, file ))
214 {
215 line++;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000216 p = buffer;
Alexandre Julliardf90efa91998-06-14 15:24:15 +0000217 while (*p && PROFILE_isspace(*p)) p++;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000218 if (*p == '[') /* section start */
219 {
220 if (!(p2 = strrchr( p, ']' )))
221 {
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000222 WARN("Invalid section header at line %d: '%s'\n",
Alexandre Julliardc7c217b1998-04-13 12:21:30 +0000223 line, p );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000224 }
225 else
226 {
227 *p2 = '\0';
228 p++;
Dimitrie O. Paunabdbced2000-04-29 14:20:28 +0000229 section = HeapAlloc( GetProcessHeap(), 0, sizeof(*section) );
230 if(section == NULL) break;
Alexandre Julliard90476d62000-02-16 22:47:24 +0000231 section->name = HEAP_strdupA( GetProcessHeap(), 0, p );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000232 section->key = NULL;
233 section->next = NULL;
Alexandre Julliard642d3131998-07-12 19:29:36 +0000234 *next_section = section;
235 next_section = &section->next;
236 next_key = &section->key;
237 prev_key = NULL;
Alexandre Julliard829fe321998-07-26 14:27:39 +0000238
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000239 TRACE("New section: '%s'\n",section->name);
Alexandre Julliard829fe321998-07-26 14:27:39 +0000240
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000241 continue;
242 }
243 }
Alexandre Julliard829fe321998-07-26 14:27:39 +0000244
245 p2=p+strlen(p) - 1;
246 while ((p2 > p) && ((*p2 == '\n') || PROFILE_isspace(*p2))) *p2--='\0';
247
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000248 if ((p2 = strchr( p, '=' )) != NULL)
249 {
250 char *p3 = p2 - 1;
Alexandre Julliardf90efa91998-06-14 15:24:15 +0000251 while ((p3 > p) && PROFILE_isspace(*p3)) *p3-- = '\0';
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000252 *p2++ = '\0';
Alexandre Julliardf90efa91998-06-14 15:24:15 +0000253 while (*p2 && PROFILE_isspace(*p2)) p2++;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000254 }
Alexandre Julliard642d3131998-07-12 19:29:36 +0000255
256 if(*p || !prev_key || *prev_key->name)
Dimitrie O. Paunabdbced2000-04-29 14:20:28 +0000257 {
258 key = HeapAlloc( GetProcessHeap(), 0, sizeof(*key) );
259 if(key == NULL) break;
Alexandre Julliard90476d62000-02-16 22:47:24 +0000260 key->name = HEAP_strdupA( GetProcessHeap(), 0, p );
261 key->value = p2 ? HEAP_strdupA( GetProcessHeap(), 0, p2 ) : NULL;
Alexandre Julliard642d3131998-07-12 19:29:36 +0000262 key->next = NULL;
263 *next_key = key;
264 next_key = &key->next;
265 prev_key = key;
266
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000267 TRACE("New key: name='%s', value='%s'\n",key->name,key->value?key->value:"(none)");
Dimitrie O. Paunabdbced2000-04-29 14:20:28 +0000268 }
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000269 }
270 return first_section;
271}
272
Alexandre Julliard73be8d12000-12-06 20:25:11 +0000273/* convert the .winerc file to the new format */
274static int convert_config( FILE *in, const char *output_name )
275{
276 char buffer[PROFILE_MAX_LINE_LEN];
277 char *p, *p2;
278 FILE *out;
279
280 /* create the output file, only if it doesn't exist already */
281 int fd = open( output_name, O_WRONLY|O_CREAT|O_EXCL, 0666 );
282 if (fd == -1) return 0;
283
284 out = fdopen( fd, "w" );
285 fprintf( out, "WINE REGISTRY Version 2\n" );
286 fprintf( out, ";; All keys relative to \\\\Machine\\\\Software\\\\Wine\\\\Wine\\\\Config\n\n" );
287 while (fgets( buffer, PROFILE_MAX_LINE_LEN, in ))
288 {
289 if (buffer[strlen(buffer)-1] == '\n') buffer[strlen(buffer)-1] = 0;
290 p = buffer;
291 while (*p && PROFILE_isspace(*p)) p++;
292 if (*p == '[') /* section start */
293 {
294 if ((p2 = strrchr( p, ']' )))
295 {
296 *p2 = '\0';
297 p++;
298 fprintf( out, "[%s]\n", p );
299 }
300 continue;
301 }
302
303 if (*p == ';' || *p == '#')
304 {
305 fprintf( out, "%s\n", p );
306 continue;
307 }
308
309 p2=p+strlen(p) - 1;
310 while ((p2 > p) && ((*p2 == '\n') || PROFILE_isspace(*p2))) *p2--='\0';
311
312 if ((p2 = strchr( p, '=' )) != NULL)
313 {
314 char *p3 = p2 - 1;
315 while ((p3 > p) && PROFILE_isspace(*p3)) *p3-- = '\0';
316 *p2++ = '\0';
317 while (*p2 && PROFILE_isspace(*p2)) p2++;
318 }
319
320 if (!*p)
321 {
322 fprintf( out, "\n" );
323 continue;
324 }
325 fputc( '"', out );
326 while (*p)
327 {
328 if (*p == '\\') fputc( '\\', out );
329 fputc( *p, out );
330 p++;
331 }
332 fprintf( out, "\" = \"" );
Alexandre Julliard550ba5b2001-01-05 22:23:37 +0000333 if (p2)
Alexandre Julliard73be8d12000-12-06 20:25:11 +0000334 {
Alexandre Julliard550ba5b2001-01-05 22:23:37 +0000335 while (*p2)
336 {
337 if (*p2 == '\\') fputc( '\\', out );
338 fputc( *p2, out );
339 p2++;
340 }
Alexandre Julliard73be8d12000-12-06 20:25:11 +0000341 }
342 fprintf( out, "\"\n" );
343 }
344 fclose( out );
345 return 1;
346}
347
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000348
349/***********************************************************************
Alexandre Julliard00377a72000-02-19 20:50:00 +0000350 * PROFILE_RegistryLoad
351 *
352 * Load a profile tree from a file into a registry key.
353 */
354static DWORD PROFILE_RegistryLoad( HKEY root, FILE *file )
355{
356 HKEY hkey = 0;
357 DWORD err = 0;
358 char buffer[PROFILE_MAX_LINE_LEN];
359 char *p, *p2;
360 int line = 0;
361
362 while (fgets( buffer, PROFILE_MAX_LINE_LEN, file ))
363 {
364 line++;
365 p = buffer;
366 while (*p && PROFILE_isspace(*p)) p++;
367 if (*p == '[') /* section start */
368 {
369 if (!(p2 = strrchr( p, ']' )))
370 {
371 WARN("Invalid section header at line %d: '%s'\n",
372 line, p );
373 }
374 else
375 {
376 *p2 = '\0';
377 p++;
378 if (hkey) RegCloseKey( hkey );
379 if ((err = RegCreateKeyExA( root, p, 0, NULL, REG_OPTION_VOLATILE,
380 KEY_ALL_ACCESS, NULL, &hkey, NULL ))) return err;
381 TRACE("New section: '%s'\n",p);
382 continue;
383 }
384 }
385
386 p2=p+strlen(p) - 1;
387 while ((p2 > p) && ((*p2 == '\n') || PROFILE_isspace(*p2))) *p2--='\0';
388
389 if ((p2 = strchr( p, '=' )) != NULL)
390 {
391 char *p3 = p2 - 1;
392 while ((p3 > p) && PROFILE_isspace(*p3)) *p3-- = '\0';
393 *p2++ = '\0';
394 while (*p2 && PROFILE_isspace(*p2)) p2++;
395 }
396
397 if (*p && hkey && !IS_ENTRY_COMMENT(p))
398 {
399 if (!p2) p2 = "";
400 if ((err = RegSetValueExA( hkey, p, 0, REG_SZ, p2, strlen(p2)+1 )))
401 {
402 RegCloseKey( hkey );
403 return err;
404 }
405 TRACE("New key: name='%s', value='%s'\n",p,p2);
406 }
407 }
408 if (hkey) RegCloseKey( hkey );
409 return 0;
410}
411
412
413/***********************************************************************
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000414 * PROFILE_DeleteSection
415 *
416 * Delete a section from a profile tree.
417 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000418static BOOL PROFILE_DeleteSection( PROFILESECTION **section, LPCSTR name )
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000419{
420 while (*section)
421 {
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +0000422 if ((*section)->name && !strcasecmp( (*section)->name, name ))
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000423 {
424 PROFILESECTION *to_del = *section;
425 *section = to_del->next;
426 to_del->next = NULL;
427 PROFILE_Free( to_del );
428 return TRUE;
429 }
430 section = &(*section)->next;
431 }
432 return FALSE;
433}
434
435
436/***********************************************************************
437 * PROFILE_DeleteKey
438 *
439 * Delete a key from a profile tree.
440 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000441static BOOL PROFILE_DeleteKey( PROFILESECTION **section,
Eric Poueched155751999-03-25 13:24:08 +0000442 LPCSTR section_name, LPCSTR key_name )
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000443{
444 while (*section)
445 {
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +0000446 if ((*section)->name && !strcasecmp( (*section)->name, section_name ))
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000447 {
448 PROFILEKEY **key = &(*section)->key;
449 while (*key)
450 {
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +0000451 if (!strcasecmp( (*key)->name, key_name ))
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000452 {
453 PROFILEKEY *to_del = *key;
454 *key = to_del->next;
Alexandre Julliard90476d62000-02-16 22:47:24 +0000455 if (to_del->name) HeapFree( GetProcessHeap(), 0, to_del->name );
456 if (to_del->value) HeapFree( GetProcessHeap(), 0, to_del->value);
457 HeapFree( GetProcessHeap(), 0, to_del );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000458 return TRUE;
459 }
460 key = &(*key)->next;
461 }
462 }
463 section = &(*section)->next;
464 }
465 return FALSE;
466}
467
468
469/***********************************************************************
Uwe Bonnesd88fbb72000-06-20 20:48:46 +0000470 * PROFILE_DeleteAllKeys
471 *
472 * Delete all keys from a profile tree.
473 */
474void PROFILE_DeleteAllKeys( LPCSTR section_name)
475{
476 PROFILESECTION **section= &CurProfile->section;
477 while (*section)
478 {
479 if ((*section)->name && !strcasecmp( (*section)->name, section_name ))
480 {
481 PROFILEKEY **key = &(*section)->key;
482 while (*key)
483 {
484 PROFILEKEY *to_del = *key;
485 *key = to_del->next;
486 if (to_del->name) HeapFree( GetProcessHeap(), 0, to_del->name );
487 if (to_del->value) HeapFree( GetProcessHeap(), 0, to_del->value);
488 HeapFree( GetProcessHeap(), 0, to_del );
489 CurProfile->changed =TRUE;
490 }
491 }
492 section = &(*section)->next;
493 }
494}
495
496
497/***********************************************************************
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000498 * PROFILE_Find
499 *
500 * Find a key in a profile tree, optionally creating it.
501 */
502static PROFILEKEY *PROFILE_Find( PROFILESECTION **section,
503 const char *section_name,
504 const char *key_name, int create )
505{
Andreas Mohr17b1f462000-01-30 21:15:14 +0000506 const char *p;
507 int seclen, keylen;
508
509 while (PROFILE_isspace(*section_name)) section_name++;
510 p = section_name + strlen(section_name) - 1;
511 while ((p > section_name) && PROFILE_isspace(*p)) p--;
512 seclen = p - section_name + 1;
513
514 while (PROFILE_isspace(*key_name)) key_name++;
515 p = key_name + strlen(key_name) - 1;
516 while ((p > key_name) && PROFILE_isspace(*p)) p--;
517 keylen = p - key_name + 1;
518
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000519 while (*section)
520 {
Andreas Mohr17b1f462000-01-30 21:15:14 +0000521 if ( ((*section)->name)
522 && (!(strncasecmp( (*section)->name, section_name, seclen )))
523 && (((*section)->name)[seclen] == '\0') )
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000524 {
525 PROFILEKEY **key = &(*section)->key;
526 while (*key)
527 {
Andreas Mohr17b1f462000-01-30 21:15:14 +0000528 if ( (!(strncasecmp( (*key)->name, key_name, keylen )))
529 && (((*key)->name)[keylen] == '\0') )
530 return *key;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000531 key = &(*key)->next;
532 }
533 if (!create) return NULL;
Dimitrie O. Paunabdbced2000-04-29 14:20:28 +0000534 *key = HeapAlloc( GetProcessHeap(), 0, sizeof(PROFILEKEY) );
535 if(*key == NULL) return NULL;
Alexandre Julliard90476d62000-02-16 22:47:24 +0000536 (*key)->name = HEAP_strdupA( GetProcessHeap(), 0, key_name );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000537 (*key)->value = NULL;
538 (*key)->next = NULL;
539 return *key;
540 }
541 section = &(*section)->next;
542 }
543 if (!create) return NULL;
Dimitrie O. Paunabdbced2000-04-29 14:20:28 +0000544 *section = HeapAlloc( GetProcessHeap(), 0, sizeof(PROFILESECTION) );
545 if(*section == NULL) return NULL;
Alexandre Julliard90476d62000-02-16 22:47:24 +0000546 (*section)->name = HEAP_strdupA( GetProcessHeap(), 0, section_name );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000547 (*section)->next = NULL;
Dimitrie O. Paunabdbced2000-04-29 14:20:28 +0000548 (*section)->key = HeapAlloc( GetProcessHeap(), 0, sizeof(PROFILEKEY) );
549 if((*section)->key == NULL)
550 {
551 HeapFree(GetProcessHeap(), 0, *section);
552 return NULL;
553 }
Alexandre Julliard90476d62000-02-16 22:47:24 +0000554 (*section)->key->name = HEAP_strdupA( GetProcessHeap(), 0, key_name );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000555 (*section)->key->value = NULL;
556 (*section)->key->next = NULL;
557 return (*section)->key;
558}
559
560
561/***********************************************************************
562 * PROFILE_FlushFile
563 *
564 * Flush the current profile to disk if changed.
565 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000566static BOOL PROFILE_FlushFile(void)
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000567{
568 char *p, buffer[MAX_PATHNAME_LEN];
569 const char *unix_name;
570 FILE *file = NULL;
Alexandre Julliard829fe321998-07-26 14:27:39 +0000571 struct stat buf;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000572
Alexandre Julliard829fe321998-07-26 14:27:39 +0000573 if(!CurProfile)
574 {
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000575 WARN("No current profile!\n");
Alexandre Julliard829fe321998-07-26 14:27:39 +0000576 return FALSE;
577 }
578
579 if (!CurProfile->changed || !CurProfile->dos_name) return TRUE;
580 if (!(unix_name = CurProfile->unix_name) || !(file = fopen(unix_name, "w")))
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000581 {
582 /* Try to create it in $HOME/.wine */
583 /* FIXME: this will need a more general solution */
Alexandre Julliardde1d5ad2000-04-06 20:36:17 +0000584 strcpy( buffer, get_config_dir() );
Alexandre Julliard647876e2000-01-25 01:35:01 +0000585 p = buffer + strlen(buffer);
586 *p++ = '/';
587 strcpy( p, strrchr( CurProfile->dos_name, '\\' ) + 1 );
Alexandre Julliarddcd247e2000-08-14 17:39:15 +0000588 _strlwr( p );
Alexandre Julliard647876e2000-01-25 01:35:01 +0000589 file = fopen( buffer, "w" );
590 unix_name = buffer;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000591 }
592
593 if (!file)
594 {
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000595 WARN("could not save profile file %s\n", CurProfile->dos_name);
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000596 return FALSE;
597 }
598
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000599 TRACE("Saving '%s' into '%s'\n", CurProfile->dos_name, unix_name );
Alexandre Julliard829fe321998-07-26 14:27:39 +0000600 PROFILE_Save( file, CurProfile->section );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000601 fclose( file );
Alexandre Julliard829fe321998-07-26 14:27:39 +0000602 CurProfile->changed = FALSE;
603 if(!stat(unix_name,&buf))
604 CurProfile->mtime=buf.st_mtime;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000605 return TRUE;
606}
607
608
609/***********************************************************************
Alexandre Julliarde61b7921999-07-31 19:24:11 +0000610 * PROFILE_ReleaseFile
611 *
612 * Flush the current profile to disk and remove it from the cache.
613 */
614static void PROFILE_ReleaseFile(void)
615{
616 PROFILE_FlushFile();
617 PROFILE_Free( CurProfile->section );
Alexandre Julliard90476d62000-02-16 22:47:24 +0000618 if (CurProfile->dos_name) HeapFree( GetProcessHeap(), 0, CurProfile->dos_name );
619 if (CurProfile->unix_name) HeapFree( GetProcessHeap(), 0, CurProfile->unix_name );
620 if (CurProfile->filename) HeapFree( GetProcessHeap(), 0, CurProfile->filename );
Alexandre Julliarde61b7921999-07-31 19:24:11 +0000621 CurProfile->changed = FALSE;
622 CurProfile->section = NULL;
623 CurProfile->dos_name = NULL;
624 CurProfile->unix_name = NULL;
625 CurProfile->filename = NULL;
626 CurProfile->mtime = 0;
627}
628
629
630/***********************************************************************
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000631 * PROFILE_Open
632 *
633 * Open a profile file, checking the cached file first.
634 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000635static BOOL PROFILE_Open( LPCSTR filename )
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000636{
Alexandre Julliardc6c09441997-01-12 18:32:19 +0000637 DOS_FULL_NAME full_name;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000638 char buffer[MAX_PATHNAME_LEN];
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000639 char *newdos_name, *p;
640 FILE *file = NULL;
Alexandre Julliard829fe321998-07-26 14:27:39 +0000641 int i,j;
642 struct stat buf;
643 PROFILE *tempProfile;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000644
Alexandre Julliard829fe321998-07-26 14:27:39 +0000645 /* First time around */
646
647 if(!CurProfile)
648 for(i=0;i<N_CACHED_PROFILES;i++)
649 {
Dimitrie O. Paunabdbced2000-04-29 14:20:28 +0000650 MRUProfile[i]=HeapAlloc( GetProcessHeap(), 0, sizeof(PROFILE) );
651 if(MRUProfile[i] == NULL) break;
Alexandre Julliard829fe321998-07-26 14:27:39 +0000652 MRUProfile[i]->changed=FALSE;
653 MRUProfile[i]->section=NULL;
654 MRUProfile[i]->dos_name=NULL;
655 MRUProfile[i]->unix_name=NULL;
656 MRUProfile[i]->filename=NULL;
657 MRUProfile[i]->mtime=0;
658 }
659
660 /* Check for a match */
Alexandre Julliardc6c09441997-01-12 18:32:19 +0000661
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000662 if (strchr( filename, '/' ) || strchr( filename, '\\' ) ||
663 strchr( filename, ':' ))
664 {
Alexandre Julliardc6c09441997-01-12 18:32:19 +0000665 if (!DOSFS_GetFullName( filename, FALSE, &full_name )) return FALSE;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000666 }
667 else
668 {
Alexandre Julliarda3960291999-02-26 11:11:13 +0000669 GetWindowsDirectoryA( buffer, sizeof(buffer) );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000670 strcat( buffer, "\\" );
671 strcat( buffer, filename );
Alexandre Julliardc6c09441997-01-12 18:32:19 +0000672 if (!DOSFS_GetFullName( buffer, FALSE, &full_name )) return FALSE;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000673 }
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000674
Alexandre Julliard829fe321998-07-26 14:27:39 +0000675 for(i=0;i<N_CACHED_PROFILES;i++)
676 {
677 if ((MRUProfile[i]->filename && !strcmp( filename, MRUProfile[i]->filename )) ||
678 (MRUProfile[i]->dos_name && !strcmp( full_name.short_name, MRUProfile[i]->dos_name )))
679 {
680 if(i)
681 {
682 PROFILE_FlushFile();
683 tempProfile=MRUProfile[i];
684 for(j=i;j>0;j--)
685 MRUProfile[j]=MRUProfile[j-1];
686 CurProfile=tempProfile;
687 }
688 if(!stat(CurProfile->unix_name,&buf) && CurProfile->mtime==buf.st_mtime)
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000689 TRACE("(%s): already opened (mru=%d)\n",
Alexandre Julliard829fe321998-07-26 14:27:39 +0000690 filename, i );
Alexandre Julliard767e6f61998-08-09 12:47:43 +0000691 else
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000692 TRACE("(%s): already opened, needs refreshing (mru=%d)\n",
Alexandre Julliard767e6f61998-08-09 12:47:43 +0000693 filename, i );
694 return TRUE;
Alexandre Julliard829fe321998-07-26 14:27:39 +0000695 }
696 }
697
Andreas Mohr87bff281999-11-21 02:03:03 +0000698 /* Flush the old current profile */
699 PROFILE_FlushFile();
Alexandre Julliard829fe321998-07-26 14:27:39 +0000700
Andreas Mohr87bff281999-11-21 02:03:03 +0000701 /* Make the oldest profile the current one only in order to get rid of it */
Alexandre Julliard829fe321998-07-26 14:27:39 +0000702 if(i==N_CACHED_PROFILES)
703 {
704 tempProfile=MRUProfile[N_CACHED_PROFILES-1];
705 for(i=N_CACHED_PROFILES-1;i>0;i--)
706 MRUProfile[i]=MRUProfile[i-1];
707 CurProfile=tempProfile;
708 }
Alexandre Julliarde61b7921999-07-31 19:24:11 +0000709 if(CurProfile->filename) PROFILE_ReleaseFile();
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000710
Andreas Mohr87bff281999-11-21 02:03:03 +0000711 /* OK, now that CurProfile is definitely free we assign it our new file */
Alexandre Julliard90476d62000-02-16 22:47:24 +0000712 newdos_name = HEAP_strdupA( GetProcessHeap(), 0, full_name.short_name );
Alexandre Julliard829fe321998-07-26 14:27:39 +0000713 CurProfile->dos_name = newdos_name;
Alexandre Julliard90476d62000-02-16 22:47:24 +0000714 CurProfile->filename = HEAP_strdupA( GetProcessHeap(), 0, filename );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000715
716 /* Try to open the profile file, first in $HOME/.wine */
717
718 /* FIXME: this will need a more general solution */
Alexandre Julliardde1d5ad2000-04-06 20:36:17 +0000719 strcpy( buffer, get_config_dir() );
Alexandre Julliard647876e2000-01-25 01:35:01 +0000720 p = buffer + strlen(buffer);
721 *p++ = '/';
722 strcpy( p, strrchr( newdos_name, '\\' ) + 1 );
Alexandre Julliarddcd247e2000-08-14 17:39:15 +0000723 _strlwr( p );
Alexandre Julliard647876e2000-01-25 01:35:01 +0000724 if ((file = fopen( buffer, "r" )))
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000725 {
Alexandre Julliard647876e2000-01-25 01:35:01 +0000726 TRACE("(%s): found it in %s\n",
727 filename, buffer );
Alexandre Julliard90476d62000-02-16 22:47:24 +0000728 CurProfile->unix_name = HEAP_strdupA( GetProcessHeap(), 0, buffer );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000729 }
730
Alexandre Julliardc6c09441997-01-12 18:32:19 +0000731 if (!file)
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000732 {
Alexandre Julliard90476d62000-02-16 22:47:24 +0000733 CurProfile->unix_name = HEAP_strdupA( GetProcessHeap(), 0,
Alexandre Julliardc6c09441997-01-12 18:32:19 +0000734 full_name.long_name );
735 if ((file = fopen( full_name.long_name, "r" )))
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000736 TRACE("(%s): found it in %s\n",
Alexandre Julliardc6c09441997-01-12 18:32:19 +0000737 filename, full_name.long_name );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000738 }
739
740 if (file)
741 {
Alexandre Julliard829fe321998-07-26 14:27:39 +0000742 CurProfile->section = PROFILE_Load( file );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000743 fclose( file );
Alexandre Julliard829fe321998-07-26 14:27:39 +0000744 if(!stat(CurProfile->unix_name,&buf))
745 CurProfile->mtime=buf.st_mtime;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000746 }
747 else
Alexandre Julliardd1ce8b21996-09-02 16:46:30 +0000748 {
749 /* Does not exist yet, we will create it in PROFILE_FlushFile */
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000750 WARN("profile file %s not found\n", newdos_name );
Alexandre Julliardd1ce8b21996-09-02 16:46:30 +0000751 }
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000752 return TRUE;
753}
754
755
756/***********************************************************************
757 * PROFILE_GetSection
758 *
Alexandre Julliarda845b881998-06-01 10:44:35 +0000759 * Returns all keys of a section.
Alexandre Julliard642d3131998-07-12 19:29:36 +0000760 * If return_values is TRUE, also include the corresponding values.
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000761 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000762static INT PROFILE_GetSection( PROFILESECTION *section, LPCSTR section_name,
Eric Poueched155751999-03-25 13:24:08 +0000763 LPSTR buffer, UINT len, BOOL handle_env,
764 BOOL return_values )
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000765{
766 PROFILEKEY *key;
Dmitry Timoshkovfdc78a32000-12-19 19:37:03 +0000767
768 if(!buffer) return 0;
769
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000770 while (section)
771 {
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +0000772 if (section->name && !strcasecmp( section->name, section_name ))
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000773 {
Alexandre Julliarda3960291999-02-26 11:11:13 +0000774 UINT oldlen = len;
Alexandre Julliard9416aba1999-01-26 17:29:49 +0000775 for (key = section->key; key; key = key->next)
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000776 {
777 if (len <= 2) break;
Alexandre Julliard9416aba1999-01-26 17:29:49 +0000778 if (!*key->name) continue; /* Skip empty lines */
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000779 if (IS_ENTRY_COMMENT(key->name)) continue; /* Skip comments */
780 PROFILE_CopyEntry( buffer, key->name, len - 1, handle_env );
Alexandre Julliard0c126c71996-02-18 18:44:41 +0000781 len -= strlen(buffer) + 1;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000782 buffer += strlen(buffer) + 1;
Alexandre Julliarda845b881998-06-01 10:44:35 +0000783 if (return_values && key->value) {
784 buffer[-1] = '=';
785 PROFILE_CopyEntry ( buffer,
786 key->value, len - 1, handle_env );
787 len -= strlen(buffer) + 1;
788 buffer += strlen(buffer) + 1;
789 }
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000790 }
791 *buffer = '\0';
Marcus Meissneracfae0c1999-05-08 18:29:10 +0000792 if (len <= 1)
Alexandre Julliard46ea8b31998-05-03 19:01:20 +0000793 /*If either lpszSection or lpszKey is NULL and the supplied
794 destination buffer is too small to hold all the strings,
795 the last string is truncated and followed by two null characters.
796 In this case, the return value is equal to cchReturnBuffer
797 minus two. */
798 {
799 buffer[-1] = '\0';
800 return oldlen - 2;
801 }
Alexandre Julliard829fe321998-07-26 14:27:39 +0000802 return oldlen - len;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000803 }
804 section = section->next;
805 }
806 buffer[0] = buffer[1] = '\0';
Alexandre Julliarddadf78f1998-05-17 17:13:43 +0000807 return 0;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000808}
809
810
Andreas Mohradfeec91999-05-22 16:04:57 +0000811static INT PROFILE_GetSectionNames( LPSTR buffer, UINT len )
812{
813 LPSTR buf = buffer;
814 WORD l, cursize = 0;
815 PROFILESECTION *section;
816
Dmitry Timoshkovfdc78a32000-12-19 19:37:03 +0000817 if(!buffer) return 0;
818
Andreas Mohradfeec91999-05-22 16:04:57 +0000819 for (section = CurProfile->section; section; section = section->next)
820 if (section->name) {
821 l = strlen(section->name);
822 cursize += l+1;
823 if (cursize > len+1)
824 return len-2;
825
826 strcpy(buf, section->name);
827 buf += l+1;
828 }
829
830 *buf=0;
831 buf++;
832 return buf-buffer;
833}
834
835
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000836/***********************************************************************
837 * PROFILE_GetString
838 *
839 * Get a profile string.
Andreas Mohr3ac104f2000-09-29 21:04:44 +0000840 *
841 * Tests with GetPrivateProfileString16, W95a,
842 * with filled buffer ("****...") and section "set1" and key_name "1" valid:
843 * section key_name def_val res buffer
844 * "set1" "1" "x" 43 [data]
845 * "set1" "1 " "x" 43 [data] (!)
846 * "set1" " 1 "' "x" 43 [data] (!)
847 * "set1" "" "x" 1 "x"
848 * "set1" "" "x " 1 "x" (!)
849 * "set1" "" " x " 3 " x" (!)
850 * "set1" NULL "x" 6 "1\02\03\0\0"
851 * "set1" "" "x" 1 "x"
852 * NULL "1" "x" 0 "" (!)
853 * "" "1" "x" 1 "x"
854 * NULL NULL "" 0 ""
855 *
856 *
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000857 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000858static INT PROFILE_GetString( LPCSTR section, LPCSTR key_name,
Eric Poueched155751999-03-25 13:24:08 +0000859 LPCSTR def_val, LPSTR buffer, UINT len )
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000860{
861 PROFILEKEY *key = NULL;
862
Dmitry Timoshkovfdc78a32000-12-19 19:37:03 +0000863 if(!buffer) return 0;
864
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000865 if (!def_val) def_val = "";
Alexandre Julliard77b99181997-09-14 17:17:23 +0000866 if (key_name && key_name[0])
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000867 {
Alexandre Julliard829fe321998-07-26 14:27:39 +0000868 key = PROFILE_Find( &CurProfile->section, section, key_name, FALSE );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000869 PROFILE_CopyEntry( buffer, (key && key->value) ? key->value : def_val,
870 len, FALSE );
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000871 TRACE("('%s','%s','%s'): returning '%s'\n",
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000872 section, key_name, def_val, buffer );
873 return strlen( buffer );
874 }
Uwe Bonnesb2ccbb22000-03-04 19:03:15 +0000875 if (key_name && !(key_name[0]))
Andreas Mohr3ac104f2000-09-29 21:04:44 +0000876 /* Win95 returns 0 on keyname "". Tested with Likse32 bon 000227 */
Uwe Bonnesb2ccbb22000-03-04 19:03:15 +0000877 return 0;
Andreas Mohradfeec91999-05-22 16:04:57 +0000878 if (section && section[0])
879 return PROFILE_GetSection(CurProfile->section, section, buffer, len,
Alexandre Julliarda845b881998-06-01 10:44:35 +0000880 FALSE, FALSE);
Andreas Mohr3ac104f2000-09-29 21:04:44 +0000881 buffer[0] = '\0';
882 return 0;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000883}
884
885
886/***********************************************************************
887 * PROFILE_SetString
888 *
889 * Set a profile string.
890 */
Alexandre Julliarda3960291999-02-26 11:11:13 +0000891static BOOL PROFILE_SetString( LPCSTR section_name, LPCSTR key_name,
Eric Poueched155751999-03-25 13:24:08 +0000892 LPCSTR value )
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000893{
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000894 if (!key_name) /* Delete a whole section */
895 {
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000896 TRACE("('%s')\n", section_name);
Alexandre Julliard829fe321998-07-26 14:27:39 +0000897 CurProfile->changed |= PROFILE_DeleteSection( &CurProfile->section,
898 section_name );
Alexandre Julliard7e6ae4b1996-12-08 19:25:27 +0000899 return TRUE; /* Even if PROFILE_DeleteSection() has failed,
900 this is not an error on application's level.*/
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000901 }
902 else if (!value) /* Delete a key */
903 {
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000904 TRACE("('%s','%s')\n",
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000905 section_name, key_name );
Alexandre Julliard829fe321998-07-26 14:27:39 +0000906 CurProfile->changed |= PROFILE_DeleteKey( &CurProfile->section,
907 section_name, key_name );
Alexandre Julliard7e6ae4b1996-12-08 19:25:27 +0000908 return TRUE; /* same error handling as above */
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000909 }
910 else /* Set the key value */
911 {
Alexandre Julliard829fe321998-07-26 14:27:39 +0000912 PROFILEKEY *key = PROFILE_Find( &CurProfile->section, section_name,
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000913 key_name, TRUE );
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000914 TRACE("('%s','%s','%s'): \n",
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000915 section_name, key_name, value );
Alexandre Julliard9ea19e51997-01-01 17:29:55 +0000916 if (!key) return FALSE;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000917 if (key->value)
918 {
Marcus Meissnera6ae5552000-06-13 01:06:22 +0000919 /* strip the leading spaces. We can safely strip \n\r and
920 * friends too, they should not happen here anyway. */
921 while (PROFILE_isspace(*value)) value++;
922
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000923 if (!strcmp( key->value, value ))
924 {
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000925 TRACE(" no change needed\n" );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000926 return TRUE; /* No change needed */
927 }
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000928 TRACE(" replacing '%s'\n", key->value );
Alexandre Julliard90476d62000-02-16 22:47:24 +0000929 HeapFree( GetProcessHeap(), 0, key->value );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000930 }
Alexandre Julliard359f497e1999-07-04 16:02:24 +0000931 else TRACE(" creating key\n" );
Alexandre Julliard90476d62000-02-16 22:47:24 +0000932 key->value = HEAP_strdupA( GetProcessHeap(), 0, value );
Alexandre Julliard829fe321998-07-26 14:27:39 +0000933 CurProfile->changed = TRUE;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000934 }
Alexandre Julliard7e6ae4b1996-12-08 19:25:27 +0000935 return TRUE;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000936}
937
938
939/***********************************************************************
940 * PROFILE_GetWineIniString
941 *
942 * Get a config string from the wine.ini file.
943 */
944int PROFILE_GetWineIniString( const char *section, const char *key_name,
945 const char *def, char *buffer, int len )
946{
Alexandre Julliard00377a72000-02-19 20:50:00 +0000947 char tmp[PROFILE_MAX_LINE_LEN];
948 HKEY hkey;
949 DWORD err;
Eric Poueched155751999-03-25 13:24:08 +0000950
Alexandre Julliard00377a72000-02-19 20:50:00 +0000951 if (!(err = RegOpenKeyA( wine_profile_key, section, &hkey )))
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000952 {
Alexandre Julliard00377a72000-02-19 20:50:00 +0000953 DWORD type;
954 DWORD count = sizeof(tmp);
955 err = RegQueryValueExA( hkey, key_name, 0, &type, tmp, &count );
956 RegCloseKey( hkey );
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000957 }
Alexandre Julliard00377a72000-02-19 20:50:00 +0000958 PROFILE_CopyEntry( buffer, err ? def : tmp, len, TRUE );
959 TRACE( "('%s','%s','%s'): returning '%s'\n", section, key_name, def, buffer );
960 return strlen(buffer);
961}
Eric Poueched155751999-03-25 13:24:08 +0000962
Alexandre Julliard00377a72000-02-19 20:50:00 +0000963
964/***********************************************************************
965 * PROFILE_EnumWineIniString
966 *
967 * Get a config string from the wine.ini file.
968 */
969BOOL PROFILE_EnumWineIniString( const char *section, int index,
970 char *name, int name_len, char *buffer, int len )
971{
972 char tmp[PROFILE_MAX_LINE_LEN];
973 HKEY hkey;
974 DWORD err, type;
975 DWORD count = sizeof(tmp);
976
977 if (RegOpenKeyA( wine_profile_key, section, &hkey )) return FALSE;
978 err = RegEnumValueA( hkey, index, name, (DWORD*)&name_len, NULL, &type, tmp, &count );
979 RegCloseKey( hkey );
980 if (!err)
981 {
982 PROFILE_CopyEntry( buffer, tmp, len, TRUE );
983 TRACE( "('%s',%d): returning '%s'='%s'\n", section, index, name, buffer );
984 }
985 return !err;
Alexandre Julliard7e56f681996-01-31 19:02:28 +0000986}
987
988
989/***********************************************************************
Alexandre Julliard18f92e71996-07-17 20:02:21 +0000990 * PROFILE_GetWineIniInt
991 *
992 * Get a config integer from the wine.ini file.
993 */
994int PROFILE_GetWineIniInt( const char *section, const char *key_name, int def )
995{
996 char buffer[20];
997 char *p;
998 long result;
999
Alexandre Julliard00377a72000-02-19 20:50:00 +00001000 PROFILE_GetWineIniString( section, key_name, "", buffer, sizeof(buffer) );
1001 if (!buffer[0]) return def;
1002 result = strtol( buffer, &p, 0 );
1003 return (p == buffer) ? 0 /* No digits at all */ : (int)result;
Alexandre Julliardd37eb361997-07-20 16:23:21 +00001004}
1005
1006
1007/******************************************************************************
1008 *
1009 * int PROFILE_GetWineIniBool(
1010 * char const *section,
1011 * char const *key_name,
1012 * int def )
1013 *
1014 * Reads a boolean value from the wine.ini file. This function attempts to
1015 * be user-friendly by accepting 'n', 'N' (no), 'f', 'F' (false), or '0'
1016 * (zero) for false, 'y', 'Y' (yes), 't', 'T' (true), or '1' (one) for
1017 * true. Anything else results in the return of the default value.
1018 *
1019 * This function uses 1 to indicate true, and 0 for false. You can check
1020 * for existence by setting def to something other than 0 or 1 and
1021 * examining the return value.
1022 */
1023int PROFILE_GetWineIniBool(
1024 char const *section,
1025 char const *key_name,
1026 int def )
1027{
1028 char key_value[2];
1029 int retval;
1030
1031 PROFILE_GetWineIniString(section, key_name, "~", key_value, 2);
1032
1033 switch(key_value[0]) {
1034 case 'n':
1035 case 'N':
1036 case 'f':
1037 case 'F':
1038 case '0':
1039 retval = 0;
1040 break;
1041
1042 case 'y':
1043 case 'Y':
1044 case 't':
1045 case 'T':
1046 case '1':
1047 retval = 1;
1048 break;
1049
1050 default:
1051 retval = def;
1052 }
1053
Alexandre Julliard359f497e1999-07-04 16:02:24 +00001054 TRACE("(\"%s\", \"%s\", %s), "
Alexandre Julliardd37eb361997-07-20 16:23:21 +00001055 "[%c], ret %s.\n", section, key_name,
1056 def ? "TRUE" : "FALSE", key_value[0],
1057 retval ? "TRUE" : "FALSE");
1058
1059 return retval;
1060}
1061
1062
Alexandre Julliard18f92e71996-07-17 20:02:21 +00001063/***********************************************************************
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001064 * PROFILE_LoadWineIni
1065 *
Chris Morgan4a3d5082001-01-10 22:57:34 +00001066 * Load the old .winerc file.
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001067 */
1068int PROFILE_LoadWineIni(void)
1069{
1070 char buffer[MAX_PATHNAME_LEN];
1071 const char *p;
1072 FILE *f;
Eric Pouech0521a542000-02-20 13:39:42 +00001073 HKEY hKeySW;
Alexandre Julliard6c8d9172000-08-26 04:40:07 +00001074 DWORD disp;
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001075
Alexandre Julliard86ff8c02000-04-13 16:10:20 +00001076 /* make sure HKLM\\Software\\Wine\\Wine exists as non-volatile key */
1077 if (RegCreateKeyA( HKEY_LOCAL_MACHINE, "Software\\Wine\\Wine", &hKeySW ))
Eric Pouech0521a542000-02-20 13:39:42 +00001078 {
1079 ERR("Cannot create config registry key\n" );
1080 return 0;
1081 }
1082 RegCloseKey( hKeySW );
Alexandre Julliard00377a72000-02-19 20:50:00 +00001083 if (RegCreateKeyExA( HKEY_LOCAL_MACHINE, "Software\\Wine\\Wine\\Config", 0, NULL,
Alexandre Julliard6c8d9172000-08-26 04:40:07 +00001084 REG_OPTION_VOLATILE, KEY_ALL_ACCESS, NULL, &wine_profile_key, &disp ))
Alexandre Julliard00377a72000-02-19 20:50:00 +00001085 {
1086 ERR("Cannot create config registry key\n" );
1087 return 0;
1088 }
Eric Poueched155751999-03-25 13:24:08 +00001089
Alexandre Julliard00377a72000-02-19 20:50:00 +00001090 if (!CLIENT_IsBootThread()) return 1; /* already loaded */
1091
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001092 if ((p = getenv( "HOME" )) != NULL)
1093 {
Alexandre Julliarda3960291999-02-26 11:11:13 +00001094 lstrcpynA(buffer, p, MAX_PATHNAME_LEN - sizeof(PROFILE_WineIniName));
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001095 strcat( buffer, PROFILE_WineIniName );
1096 if ((f = fopen( buffer, "r" )) != NULL)
1097 {
Francois Gougetbaa9bf91999-12-27 05:24:06 +00001098 lstrcpynA(PROFILE_WineIniUsed,buffer,MAX_PATHNAME_LEN);
Alexandre Julliard00377a72000-02-19 20:50:00 +00001099 goto found;
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001100 }
1101 }
Alexandre Julliard359f497e1999-07-04 16:02:24 +00001102 else WARN("could not get $HOME value for config file.\n" );
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001103
Alexandre Julliard6c8d9172000-08-26 04:40:07 +00001104 if (disp == REG_OPENED_EXISTING_KEY) return 1; /* loaded by the server */
1105
Chris Morgan4a3d5082001-01-10 22:57:34 +00001106 MESSAGE( "Can't open configuration file %s/config\n",get_config_dir() );
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001107 return 0;
Alexandre Julliard00377a72000-02-19 20:50:00 +00001108
1109 found:
Alexandre Julliard6c8d9172000-08-26 04:40:07 +00001110
1111 if (disp == REG_OPENED_EXISTING_KEY)
1112 {
Chris Morgan4a3d5082001-01-10 22:57:34 +00001113 MESSAGE( "Warning: configuration loaded by the server from '%s/config',\n"
1114 " file '%s' was ignored.\n", get_config_dir(), PROFILE_WineIniUsed );
Alexandre Julliard6c8d9172000-08-26 04:40:07 +00001115 fclose( f );
1116 return 1;
1117 }
1118
Alexandre Julliard73be8d12000-12-06 20:25:11 +00001119 /* convert to the new format */
1120 sprintf( buffer, "%s/config", get_config_dir() );
1121 if (convert_config( f, buffer ))
1122 {
1123 MESSAGE( "The '%s' configuration file has been converted\n"
1124 "to the new format and saved as '%s'.\n", PROFILE_WineIniUsed, buffer );
1125 MESSAGE( "You should verify that the contents of the new file are correct,\n"
1126 "and then remove the old one and restart Wine.\n" );
1127 ExitProcess(0);
1128 }
1129
Alexandre Julliard00377a72000-02-19 20:50:00 +00001130 PROFILE_RegistryLoad( wine_profile_key, f );
1131 fclose( f );
1132 return 1;
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001133}
1134
1135
Alexandre Julliard23946ad1997-06-16 17:43:53 +00001136/***********************************************************************
Alexandre Julliardd30dfd21998-09-27 18:28:36 +00001137 * PROFILE_UsageWineIni
1138 *
1139 * Explain the wine.ini file to those who don't read documentation.
1140 * Keep below one screenful in length so that error messages above are
1141 * noticed.
1142 */
1143void PROFILE_UsageWineIni(void)
1144{
Alexandre Julliard359f497e1999-07-04 16:02:24 +00001145 MESSAGE("Perhaps you have not properly edited or created "
Alexandre Julliardd30dfd21998-09-27 18:28:36 +00001146 "your Wine configuration file.\n");
Gerard Patel151023d2001-01-22 19:27:06 +00001147 MESSAGE("This is '%s/config'\n", get_config_dir());
Alexandre Julliardd30dfd21998-09-27 18:28:36 +00001148 /* RTFM, so to say */
1149}
1150
Alexandre Julliardd30dfd21998-09-27 18:28:36 +00001151/***********************************************************************
Alexandre Julliard23946ad1997-06-16 17:43:53 +00001152 * PROFILE_GetStringItem
1153 *
1154 * Convenience function that turns a string 'xxx, yyy, zzz' into
1155 * the 'xxx\0 yyy, zzz' and returns a pointer to the 'yyy, zzz'.
1156 */
1157char* PROFILE_GetStringItem( char* start )
1158{
1159 char* lpchX, *lpch;
1160
1161 for (lpchX = start, lpch = NULL; *lpchX != '\0'; lpchX++ )
1162 {
Alexandre Julliard23946ad1997-06-16 17:43:53 +00001163 if( *lpchX == ',' )
1164 {
1165 if( lpch ) *lpch = '\0'; else *lpchX = '\0';
1166 while( *(++lpchX) )
Alexandre Julliardf90efa91998-06-14 15:24:15 +00001167 if( !PROFILE_isspace(*lpchX) ) return lpchX;
Alexandre Julliard23946ad1997-06-16 17:43:53 +00001168 }
Alexandre Julliardf90efa91998-06-14 15:24:15 +00001169 else if( PROFILE_isspace( *lpchX ) && !lpch ) lpch = lpchX;
Alexandre Julliard670cdc41997-08-24 16:00:30 +00001170 else lpch = NULL;
Alexandre Julliard23946ad1997-06-16 17:43:53 +00001171 }
1172 if( lpch ) *lpch = '\0';
1173 return NULL;
1174}
1175
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001176/********************* API functions **********************************/
1177
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001178/***********************************************************************
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001179 * GetProfileInt16 (KERNEL.57)
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001180 */
Alexandre Julliard670cdc41997-08-24 16:00:30 +00001181UINT16 WINAPI GetProfileInt16( LPCSTR section, LPCSTR entry, INT16 def_val )
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001182{
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001183 return GetPrivateProfileInt16( section, entry, def_val, "win.ini" );
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001184}
1185
1186
1187/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001188 * GetProfileIntA (KERNEL32.264)
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001189 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001190UINT WINAPI GetProfileIntA( LPCSTR section, LPCSTR entry, INT def_val )
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001191{
Alexandre Julliarda3960291999-02-26 11:11:13 +00001192 return GetPrivateProfileIntA( section, entry, def_val, "win.ini" );
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001193}
1194
1195/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001196 * GetProfileIntW (KERNEL32.264)
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001197 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001198UINT WINAPI GetProfileIntW( LPCWSTR section, LPCWSTR entry, INT def_val )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001199{
Alexandre Julliarda3960291999-02-26 11:11:13 +00001200 return GetPrivateProfileIntW( section, entry, def_val, wininiW );
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001201}
1202
Andreas Mohr3ac104f2000-09-29 21:04:44 +00001203/*
1204 * undoc_feature means:
1205 * return section names string list if both section and entry are NULL.
1206 */
1207static int PROFILE_GetPrivateProfileString( LPCSTR section, LPCSTR entry,
1208 LPCSTR def_val, LPSTR buffer,
1209 UINT16 len, LPCSTR filename,
1210 BOOL undoc_feature )
1211{
1212 int ret;
1213 LPSTR pDefVal = NULL;
1214
1215 if (!filename)
1216 filename = "win.ini";
1217
1218 /* strip any trailing ' ' of def_val. */
1219 if (def_val)
1220 {
1221 LPSTR p = (LPSTR)&def_val[strlen(def_val)]; /* even "" works ! */
1222
1223 while (p > def_val)
1224 {
1225 p--;
1226 if ((*p) != ' ')
1227 break;
1228 }
1229 if (*p == ' ') /* ouch, contained trailing ' ' */
1230 {
1231 int len = (int)p - (int)def_val;
1232 pDefVal = HeapAlloc(GetProcessHeap(), 0, len + 1);
1233 strncpy(pDefVal, def_val, len);
1234 pDefVal[len] = '\0';
1235 }
1236 }
1237 if (!pDefVal)
1238 pDefVal = (LPSTR)def_val;
1239
1240 EnterCriticalSection( &PROFILE_CritSect );
1241
1242 if (PROFILE_Open( filename )) {
1243 if ((undoc_feature) && (section == NULL) && (entry == NULL))
1244 /* undocumented; both section and entry are NULL */
1245 ret = PROFILE_GetSectionNames(buffer, len);
1246 else
1247 ret = PROFILE_GetString( section, entry, pDefVal, buffer, len );
1248 } else {
1249 lstrcpynA( buffer, pDefVal, len );
1250 ret = strlen( buffer );
1251 }
1252
1253 LeaveCriticalSection( &PROFILE_CritSect );
1254
1255 if (pDefVal != def_val) /* allocated */
1256 HeapFree(GetProcessHeap(), 0, pDefVal);
1257
1258 return ret;
1259}
1260
1261/***********************************************************************
1262 * GetPrivateProfileString16 (KERNEL.128)
1263 */
1264INT16 WINAPI GetPrivateProfileString16( LPCSTR section, LPCSTR entry,
1265 LPCSTR def_val, LPSTR buffer,
1266 UINT16 len, LPCSTR filename )
1267{
1268 return PROFILE_GetPrivateProfileString( section, entry, def_val,
1269 buffer, len, filename, FALSE );
1270}
1271
1272/***********************************************************************
1273 * GetPrivateProfileStringA (KERNEL32.255)
1274 */
1275INT WINAPI GetPrivateProfileStringA( LPCSTR section, LPCSTR entry,
1276 LPCSTR def_val, LPSTR buffer,
1277 UINT len, LPCSTR filename )
1278{
1279 return PROFILE_GetPrivateProfileString( section, entry, def_val,
1280 buffer, len, filename, TRUE );
1281}
1282
1283/***********************************************************************
1284 * GetPrivateProfileStringW (KERNEL32.256)
1285 */
1286INT WINAPI GetPrivateProfileStringW( LPCWSTR section, LPCWSTR entry,
1287 LPCWSTR def_val, LPWSTR buffer,
1288 UINT len, LPCWSTR filename )
1289{
1290 LPSTR sectionA = HEAP_strdupWtoA( GetProcessHeap(), 0, section );
1291 LPSTR entryA = HEAP_strdupWtoA( GetProcessHeap(), 0, entry );
1292 LPSTR filenameA = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
1293 LPSTR def_valA = HEAP_strdupWtoA( GetProcessHeap(), 0, def_val );
1294 LPSTR bufferA = HeapAlloc( GetProcessHeap(), 0, len );
1295 INT ret = GetPrivateProfileStringA( sectionA, entryA, def_valA,
1296 bufferA, len, filenameA );
Alexandre Julliard24a62ab2000-11-28 22:40:56 +00001297 if (len > 0 && !MultiByteToWideChar( CP_ACP, 0, bufferA, -1, buffer, len ))
1298 buffer[len-1] = 0;
Andreas Mohr3ac104f2000-09-29 21:04:44 +00001299 HeapFree( GetProcessHeap(), 0, sectionA );
1300 HeapFree( GetProcessHeap(), 0, entryA );
1301 HeapFree( GetProcessHeap(), 0, filenameA );
1302 HeapFree( GetProcessHeap(), 0, def_valA );
1303 HeapFree( GetProcessHeap(), 0, bufferA);
1304 return ret;
1305}
1306
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001307/***********************************************************************
1308 * GetProfileString16 (KERNEL.58)
1309 */
Alexandre Julliard670cdc41997-08-24 16:00:30 +00001310INT16 WINAPI GetProfileString16( LPCSTR section, LPCSTR entry, LPCSTR def_val,
Alexandre Julliard642d3131998-07-12 19:29:36 +00001311 LPSTR buffer, UINT16 len )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001312{
Andreas Mohr3ac104f2000-09-29 21:04:44 +00001313 return PROFILE_GetPrivateProfileString( section, entry, def_val,
1314 buffer, len, "win.ini", FALSE );
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001315}
1316
1317/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001318 * GetProfileStringA (KERNEL32.268)
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001319 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001320INT WINAPI GetProfileStringA( LPCSTR section, LPCSTR entry, LPCSTR def_val,
Eric Poueched155751999-03-25 13:24:08 +00001321 LPSTR buffer, UINT len )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001322{
Andreas Mohr3ac104f2000-09-29 21:04:44 +00001323 return PROFILE_GetPrivateProfileString( section, entry, def_val,
1324 buffer, len, "win.ini", TRUE );
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001325}
1326
1327/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001328 * GetProfileStringW (KERNEL32.269)
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001329 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001330INT WINAPI GetProfileStringW( LPCWSTR section, LPCWSTR entry,
Eric Poueched155751999-03-25 13:24:08 +00001331 LPCWSTR def_val, LPWSTR buffer, UINT len )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001332{
Alexandre Julliarda3960291999-02-26 11:11:13 +00001333 return GetPrivateProfileStringW( section, entry, def_val,
Eric Poueched155751999-03-25 13:24:08 +00001334 buffer, len, wininiW );
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001335}
1336
Alexandre Julliard77b99181997-09-14 17:17:23 +00001337/***********************************************************************
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001338 * WriteProfileString16 (KERNEL.59)
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001339 */
Alexandre Julliard670cdc41997-08-24 16:00:30 +00001340BOOL16 WINAPI WriteProfileString16( LPCSTR section, LPCSTR entry,
1341 LPCSTR string )
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001342{
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001343 return WritePrivateProfileString16( section, entry, string, "win.ini" );
1344}
1345
1346/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001347 * WriteProfileStringA (KERNEL32.587)
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001348 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001349BOOL WINAPI WriteProfileStringA( LPCSTR section, LPCSTR entry,
Eric Poueched155751999-03-25 13:24:08 +00001350 LPCSTR string )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001351{
Alexandre Julliarda3960291999-02-26 11:11:13 +00001352 return WritePrivateProfileStringA( section, entry, string, "win.ini" );
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001353}
1354
1355/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001356 * WriteProfileStringW (KERNEL32.588)
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001357 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001358BOOL WINAPI WriteProfileStringW( LPCWSTR section, LPCWSTR entry,
Alexandre Julliard670cdc41997-08-24 16:00:30 +00001359 LPCWSTR string )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001360{
Alexandre Julliarda3960291999-02-26 11:11:13 +00001361 return WritePrivateProfileStringW( section, entry, string, wininiW );
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001362}
1363
1364
1365/***********************************************************************
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001366 * GetPrivateProfileInt16 (KERNEL.127)
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001367 */
Alexandre Julliard670cdc41997-08-24 16:00:30 +00001368UINT16 WINAPI GetPrivateProfileInt16( LPCSTR section, LPCSTR entry,
1369 INT16 def_val, LPCSTR filename )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001370{
Alexandre Julliarda3960291999-02-26 11:11:13 +00001371 long result=(long)GetPrivateProfileIntA(section,entry,def_val,filename);
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001372
1373 if (result > 65535) return 65535;
1374 if (result >= 0) return (UINT16)result;
1375 if (result < -32768) return -32768;
1376 return (UINT16)(INT16)result;
1377}
1378
1379/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001380 * GetPrivateProfileIntA (KERNEL32.251)
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001381 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001382UINT WINAPI GetPrivateProfileIntA( LPCSTR section, LPCSTR entry,
Eric Poueched155751999-03-25 13:24:08 +00001383 INT def_val, LPCSTR filename )
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001384{
1385 char buffer[20];
1386 char *p;
1387 long result;
1388
Andreas Mohr3ac104f2000-09-29 21:04:44 +00001389 PROFILE_GetPrivateProfileString( section, entry, "",
1390 buffer, sizeof(buffer), filename, FALSE );
Alexandre Julliarda3960291999-02-26 11:11:13 +00001391 if (!buffer[0]) return (UINT)def_val;
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001392 result = strtol( buffer, &p, 0 );
1393 if (p == buffer) return 0; /* No digits at all */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001394 return (UINT)result;
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001395}
1396
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001397/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001398 * GetPrivateProfileIntW (KERNEL32.252)
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001399 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001400UINT WINAPI GetPrivateProfileIntW( LPCWSTR section, LPCWSTR entry,
Eric Poueched155751999-03-25 13:24:08 +00001401 INT def_val, LPCWSTR filename )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001402{
Alexandre Julliard7ebe1a41996-12-22 18:27:48 +00001403 LPSTR sectionA = HEAP_strdupWtoA( GetProcessHeap(), 0, section );
1404 LPSTR entryA = HEAP_strdupWtoA( GetProcessHeap(), 0, entry );
1405 LPSTR filenameA = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
Alexandre Julliarda3960291999-02-26 11:11:13 +00001406 UINT res = GetPrivateProfileIntA(sectionA, entryA, def_val, filenameA);
Alexandre Julliard7ebe1a41996-12-22 18:27:48 +00001407 HeapFree( GetProcessHeap(), 0, sectionA );
1408 HeapFree( GetProcessHeap(), 0, filenameA );
1409 HeapFree( GetProcessHeap(), 0, entryA );
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001410 return res;
1411}
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001412
1413/***********************************************************************
Ulrich Weigand8228bd61999-04-22 09:56:09 +00001414 * GetPrivateProfileSection16 (KERNEL.418)
1415 */
1416INT16 WINAPI GetPrivateProfileSection16( LPCSTR section, LPSTR buffer,
1417 UINT16 len, LPCSTR filename )
1418{
1419 return GetPrivateProfileSectionA( section, buffer, len, filename );
1420}
1421
1422/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001423 * GetPrivateProfileSectionA (KERNEL32.255)
Alexandre Julliard77b99181997-09-14 17:17:23 +00001424 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001425INT WINAPI GetPrivateProfileSectionA( LPCSTR section, LPSTR buffer,
Eric Poueched155751999-03-25 13:24:08 +00001426 DWORD len, LPCSTR filename )
Alexandre Julliard77b99181997-09-14 17:17:23 +00001427{
Eric Poueched155751999-03-25 13:24:08 +00001428 int ret = 0;
1429
1430 EnterCriticalSection( &PROFILE_CritSect );
1431
Alexandre Julliard77b99181997-09-14 17:17:23 +00001432 if (PROFILE_Open( filename ))
Eric Poueched155751999-03-25 13:24:08 +00001433 ret = PROFILE_GetSection(CurProfile->section, section, buffer, len,
1434 FALSE, TRUE);
1435
1436 LeaveCriticalSection( &PROFILE_CritSect );
Alexandre Julliarda845b881998-06-01 10:44:35 +00001437
Marcus Meissner264360f1999-05-08 10:02:05 +00001438 return ret;
Alexandre Julliard77b99181997-09-14 17:17:23 +00001439}
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001440
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001441/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001442 * GetPrivateProfileSectionW (KERNEL32.256)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001443 */
1444
Alexandre Julliarda3960291999-02-26 11:11:13 +00001445INT WINAPI GetPrivateProfileSectionW (LPCWSTR section, LPWSTR buffer,
Eric Poueched155751999-03-25 13:24:08 +00001446 DWORD len, LPCWSTR filename )
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001447
1448{
1449 LPSTR sectionA = HEAP_strdupWtoA( GetProcessHeap(), 0, section );
1450 LPSTR filenameA = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
1451 LPSTR bufferA = HeapAlloc( GetProcessHeap(), 0, len );
Alexandre Julliarda3960291999-02-26 11:11:13 +00001452 INT ret = GetPrivateProfileSectionA( sectionA, bufferA, len,
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001453 filenameA );
Marcus Meissner264360f1999-05-08 10:02:05 +00001454 MultiByteToWideChar(CP_ACP,0,bufferA,ret,buffer,len);
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001455 HeapFree( GetProcessHeap(), 0, sectionA );
1456 HeapFree( GetProcessHeap(), 0, filenameA );
1457 HeapFree( GetProcessHeap(), 0, bufferA);
1458 return ret;
1459}
1460
1461/***********************************************************************
Ulrich Weigand8228bd61999-04-22 09:56:09 +00001462 * GetProfileSection16 (KERNEL.419)
1463 */
1464INT16 WINAPI GetProfileSection16( LPCSTR section, LPSTR buffer, UINT16 len )
1465{
1466 return GetPrivateProfileSection16( section, buffer, len, "win.ini" );
1467}
1468
1469/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001470 * GetProfileSectionA (KERNEL32.268)
Ulrich Weigand8228bd61999-04-22 09:56:09 +00001471 */
1472INT WINAPI GetProfileSectionA( LPCSTR section, LPSTR buffer, DWORD len )
1473{
1474 return GetPrivateProfileSectionA( section, buffer, len, "win.ini" );
1475}
1476
1477/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001478 * GetProfileSectionW (KERNEL32)
Ulrich Weigand8228bd61999-04-22 09:56:09 +00001479 */
1480INT WINAPI GetProfileSectionW( LPCWSTR section, LPWSTR buffer, DWORD len )
1481{
Ulrich Weigand8228bd61999-04-22 09:56:09 +00001482 return GetPrivateProfileSectionW( section, buffer, len, wininiW );
1483}
1484
1485
1486/***********************************************************************
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001487 * WritePrivateProfileString16 (KERNEL.129)
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001488 */
Alexandre Julliard670cdc41997-08-24 16:00:30 +00001489BOOL16 WINAPI WritePrivateProfileString16( LPCSTR section, LPCSTR entry,
1490 LPCSTR string, LPCSTR filename )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001491{
Alexandre Julliarda3960291999-02-26 11:11:13 +00001492 return WritePrivateProfileStringA(section,entry,string,filename);
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001493}
1494
1495/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001496 * WritePrivateProfileStringA (KERNEL32.582)
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001497 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001498BOOL WINAPI WritePrivateProfileStringA( LPCSTR section, LPCSTR entry,
Eric Poueched155751999-03-25 13:24:08 +00001499 LPCSTR string, LPCSTR filename )
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001500{
Alexandre Julliarde61b7921999-07-31 19:24:11 +00001501 BOOL ret = FALSE;
Eric Poueched155751999-03-25 13:24:08 +00001502
1503 EnterCriticalSection( &PROFILE_CritSect );
1504
Alexandre Julliarde61b7921999-07-31 19:24:11 +00001505 if (PROFILE_Open( filename ))
1506 {
1507 if (!section && !entry && !string)
1508 PROFILE_ReleaseFile(); /* always return FALSE in this case */
1509 else
1510 ret = PROFILE_SetString( section, entry, string );
Eric Poueched155751999-03-25 13:24:08 +00001511 }
1512
1513 LeaveCriticalSection( &PROFILE_CritSect );
Eric Poueched155751999-03-25 13:24:08 +00001514 return ret;
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001515}
1516
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001517/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001518 * WritePrivateProfileStringW (KERNEL32.583)
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001519 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001520BOOL WINAPI WritePrivateProfileStringW( LPCWSTR section, LPCWSTR entry,
Eric Poueched155751999-03-25 13:24:08 +00001521 LPCWSTR string, LPCWSTR filename )
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001522{
Alexandre Julliard7ebe1a41996-12-22 18:27:48 +00001523 LPSTR sectionA = HEAP_strdupWtoA( GetProcessHeap(), 0, section );
1524 LPSTR entryA = HEAP_strdupWtoA( GetProcessHeap(), 0, entry );
1525 LPSTR stringA = HEAP_strdupWtoA( GetProcessHeap(), 0, string );
1526 LPSTR filenameA = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
Alexandre Julliarda3960291999-02-26 11:11:13 +00001527 BOOL res = WritePrivateProfileStringA( sectionA, entryA,
Eric Poueched155751999-03-25 13:24:08 +00001528 stringA, filenameA );
Alexandre Julliard7ebe1a41996-12-22 18:27:48 +00001529 HeapFree( GetProcessHeap(), 0, sectionA );
1530 HeapFree( GetProcessHeap(), 0, entryA );
1531 HeapFree( GetProcessHeap(), 0, stringA );
1532 HeapFree( GetProcessHeap(), 0, filenameA );
Alexandre Julliardda0cfb31996-12-01 17:17:47 +00001533 return res;
1534}
1535
Alexandre Julliard44ed71f1997-12-21 19:17:50 +00001536/***********************************************************************
Ulrich Weigand8228bd61999-04-22 09:56:09 +00001537 * WritePrivateProfileSection16 (KERNEL.416)
1538 */
1539BOOL16 WINAPI WritePrivateProfileSection16( LPCSTR section,
1540 LPCSTR string, LPCSTR filename )
1541{
1542 return WritePrivateProfileSectionA( section, string, filename );
1543}
1544
1545/***********************************************************************
Rein Klazes7458be02000-01-05 01:42:51 +00001546 * WritePrivateProfileSectionA (KERNEL32)
Alexandre Julliard44ed71f1997-12-21 19:17:50 +00001547 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001548BOOL WINAPI WritePrivateProfileSectionA( LPCSTR section,
Eric Poueched155751999-03-25 13:24:08 +00001549 LPCSTR string, LPCSTR filename )
Alexandre Julliard44ed71f1997-12-21 19:17:50 +00001550{
Rein Klazes7458be02000-01-05 01:42:51 +00001551 BOOL ret = FALSE;
1552 LPSTR p ;
Alexandre Julliard44ed71f1997-12-21 19:17:50 +00001553
Rein Klazes7458be02000-01-05 01:42:51 +00001554 EnterCriticalSection( &PROFILE_CritSect );
1555
1556 if (PROFILE_Open( filename )) {
Marcus Meissner25e74032000-11-26 22:36:19 +00001557 if (!section && !string)
Rein Klazes7458be02000-01-05 01:42:51 +00001558 PROFILE_ReleaseFile(); /* always return FALSE in this case */
Uwe Bonnesd88fbb72000-06-20 20:48:46 +00001559 else if (!string) /* delete the named section*/
1560 ret = PROFILE_SetString(section,NULL,NULL);
Rein Klazes7458be02000-01-05 01:42:51 +00001561 else {
Uwe Bonnesd88fbb72000-06-20 20:48:46 +00001562 PROFILE_DeleteAllKeys(section);
1563 ret = TRUE;
1564 while(*string) {
1565 LPSTR buf=HEAP_strdupA( GetProcessHeap(), 0, string );
Rein Klazes7458be02000-01-05 01:42:51 +00001566 if((p=strchr( buf, '='))){
1567 *p='\0';
1568 ret = PROFILE_SetString( section, buf, p+1 );
1569
1570 }
1571 HeapFree( GetProcessHeap(), 0, buf );
1572 string += strlen(string)+1;
1573 }
1574
1575 }
Alexandre Julliard44ed71f1997-12-21 19:17:50 +00001576 }
Rein Klazes7458be02000-01-05 01:42:51 +00001577
1578 LeaveCriticalSection( &PROFILE_CritSect );
1579 return ret;
Alexandre Julliard44ed71f1997-12-21 19:17:50 +00001580}
1581
Alexandre Julliarda845b881998-06-01 10:44:35 +00001582/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001583 * WritePrivateProfileSectionW (KERNEL32)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001584 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001585BOOL WINAPI WritePrivateProfileSectionW( LPCWSTR section,
Eric Poueched155751999-03-25 13:24:08 +00001586 LPCWSTR string, LPCWSTR filename)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001587
1588{
1589 LPSTR sectionA = HEAP_strdupWtoA( GetProcessHeap(), 0, section );
1590 LPSTR stringA = HEAP_strdupWtoA( GetProcessHeap(), 0, string );
1591 LPSTR filenameA = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
Alexandre Julliarda3960291999-02-26 11:11:13 +00001592 BOOL res = WritePrivateProfileSectionA( sectionA, stringA, filenameA );
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001593 HeapFree( GetProcessHeap(), 0, sectionA );
1594 HeapFree( GetProcessHeap(), 0, stringA );
1595 HeapFree( GetProcessHeap(), 0, filenameA );
1596 return res;
1597}
1598
Ulrich Weigand8228bd61999-04-22 09:56:09 +00001599/***********************************************************************
1600 * WriteProfileSection16 (KERNEL.417)
1601 */
1602BOOL16 WINAPI WriteProfileSection16( LPCSTR section, LPCSTR keys_n_values)
1603{
1604 return WritePrivateProfileSection16( section, keys_n_values, "win.ini");
1605}
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001606
1607/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001608 * WriteProfileSectionA (KERNEL32.747)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001609 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001610BOOL WINAPI WriteProfileSectionA( LPCSTR section, LPCSTR keys_n_values)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001611
1612{
Alexandre Julliarda3960291999-02-26 11:11:13 +00001613 return WritePrivateProfileSectionA( section, keys_n_values, "win.ini");
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001614}
1615
1616/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001617 * WriteProfileSectionW (KERNEL32.748)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001618 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001619BOOL WINAPI WriteProfileSectionW( LPCWSTR section, LPCWSTR keys_n_values)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001620{
Alexandre Julliarda3960291999-02-26 11:11:13 +00001621 return (WritePrivateProfileSectionW (section,keys_n_values, wininiW));
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001622}
1623
1624/***********************************************************************
Alexandre Julliarda845b881998-06-01 10:44:35 +00001625 * GetPrivateProfileSectionNames16 (KERNEL.143)
1626 */
1627WORD WINAPI GetPrivateProfileSectionNames16( LPSTR buffer, WORD size,
1628 LPCSTR filename )
1629{
Andreas Mohradfeec91999-05-22 16:04:57 +00001630 WORD ret = 0;
Eric Poueched155751999-03-25 13:24:08 +00001631
1632 EnterCriticalSection( &PROFILE_CritSect );
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001633
Andreas Mohradfeec91999-05-22 16:04:57 +00001634 if (PROFILE_Open( filename ))
1635 ret = PROFILE_GetSectionNames(buffer, size);
Eric Poueched155751999-03-25 13:24:08 +00001636
1637 LeaveCriticalSection( &PROFILE_CritSect );
1638
1639 return ret;
Alexandre Julliarda845b881998-06-01 10:44:35 +00001640}
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001641
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001642
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001643/***********************************************************************
1644 * GetProfileSectionNames16 (KERNEL.142)
1645 */
1646WORD WINAPI GetProfileSectionNames16( LPSTR buffer, WORD size)
1647
1648{
Eric Poueched155751999-03-25 13:24:08 +00001649 return (GetPrivateProfileSectionNames16 (buffer,size,"win.ini"));
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001650}
1651
1652
1653/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001654 * GetPrivateProfileSectionNamesA (KERNEL32.365)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001655 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001656DWORD WINAPI GetPrivateProfileSectionNamesA( LPSTR buffer, DWORD size,
Eric Poueched155751999-03-25 13:24:08 +00001657 LPCSTR filename)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001658
1659{
1660 return (GetPrivateProfileSectionNames16 (buffer,size,filename));
1661}
1662
1663
1664/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001665 * GetPrivateProfileSectionNamesW (KERNEL32.366)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001666 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001667DWORD WINAPI GetPrivateProfileSectionNamesW( LPWSTR buffer, DWORD size,
Eric Poueched155751999-03-25 13:24:08 +00001668 LPCWSTR filename)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001669
1670{
1671 LPSTR filenameA = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
1672 LPSTR bufferA = HeapAlloc( GetProcessHeap(), 0, size);
1673
Alexandre Julliarda3960291999-02-26 11:11:13 +00001674 INT ret = GetPrivateProfileSectionNames16 (bufferA, size, filenameA);
Alexandre Julliard24a62ab2000-11-28 22:40:56 +00001675 if (size > 0 && !MultiByteToWideChar( CP_ACP, 0, bufferA, -1, buffer, size ))
1676 buffer[size-1] = 0;
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001677 HeapFree( GetProcessHeap(), 0, bufferA);
1678 HeapFree( GetProcessHeap(), 0, filenameA );
1679
1680 return ret;
1681}
1682
Ulrich Weigand8228bd61999-04-22 09:56:09 +00001683/***********************************************************************
1684 * GetPrivateProfileStruct16 (KERNEL.407)
1685 */
1686BOOL16 WINAPI GetPrivateProfileStruct16(LPCSTR section, LPCSTR key,
1687 LPVOID buf, UINT16 len, LPCSTR filename)
1688{
1689 return GetPrivateProfileStructA( section, key, buf, len, filename );
1690}
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001691
1692/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001693 * GetPrivateProfileStructA (KERNEL32.370)
Andreas Mohr17b1f462000-01-30 21:15:14 +00001694 *
1695 * Should match Win95's behaviour pretty much
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001696 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001697BOOL WINAPI GetPrivateProfileStructA (LPCSTR section, LPCSTR key,
Eric Poueched155751999-03-25 13:24:08 +00001698 LPVOID buf, UINT len, LPCSTR filename)
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001699{
Eric Poueched155751999-03-25 13:24:08 +00001700 BOOL ret = FALSE;
1701
1702 EnterCriticalSection( &PROFILE_CritSect );
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001703
1704 if (PROFILE_Open( filename )) {
Eric Poueched155751999-03-25 13:24:08 +00001705 PROFILEKEY *k = PROFILE_Find ( &CurProfile->section, section, key, FALSE);
1706 if (k) {
Andreas Mohr17b1f462000-01-30 21:15:14 +00001707 TRACE("value (at %p): '%s'\n", k->value, k->value);
1708 if (((strlen(k->value) - 2) / 2) == len)
1709 {
1710 LPSTR end, p;
1711 BOOL valid = TRUE;
1712 CHAR c;
1713 DWORD chksum = 0;
1714
1715 end = k->value + strlen(k->value); /* -> '\0' */
1716 /* check for invalid chars in ASCII coded hex string */
1717 for (p=k->value; p < end; p++)
1718 {
1719 if (!isxdigit(*p))
1720 {
1721 WARN("invalid char '%c' in file '%s'->'[%s]'->'%s' !\n",
1722 *p, filename, section, key);
1723 valid = FALSE;
1724 break;
1725 }
1726 }
1727 if (valid)
1728 {
1729 BOOL highnibble = TRUE;
1730 BYTE b = 0, val;
1731 LPBYTE binbuf = (LPBYTE)buf;
1732
1733 end -= 2; /* don't include checksum in output data */
1734 /* translate ASCII hex format into binary data */
1735 for (p=k->value; p < end; p++)
1736 {
1737 c = toupper(*p);
1738 val = (c > '9') ?
1739 (c - 'A' + 10) : (c - '0');
1740
1741 if (highnibble)
1742 b = val << 4;
1743 else
1744 {
1745 b += val;
1746 *binbuf++ = b; /* feed binary data into output */
1747 chksum += b; /* calculate checksum */
1748 }
1749 highnibble ^= 1; /* toggle */
1750 }
1751 /* retrieve stored checksum value */
1752 c = toupper(*p++);
1753 b = ( (c > '9') ? (c - 'A' + 10) : (c - '0') ) << 4;
1754 c = toupper(*p);
1755 b += (c > '9') ? (c - 'A' + 10) : (c - '0');
1756 if (b == (chksum & 0xff)) /* checksums match ? */
1757 ret = TRUE;
1758 }
1759 }
Eric Poueched155751999-03-25 13:24:08 +00001760 }
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001761 }
Eric Poueched155751999-03-25 13:24:08 +00001762 LeaveCriticalSection( &PROFILE_CritSect );
1763
Andreas Mohr17b1f462000-01-30 21:15:14 +00001764 return ret;
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001765}
1766
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001767/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001768 * GetPrivateProfileStructW (KERNEL32.543)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001769 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001770BOOL WINAPI GetPrivateProfileStructW (LPCWSTR section, LPCWSTR key,
Eric Poueched155751999-03-25 13:24:08 +00001771 LPVOID buffer, UINT len, LPCWSTR filename)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001772{
1773 LPSTR sectionA = HEAP_strdupWtoA( GetProcessHeap(), 0, section );
1774 LPSTR keyA = HEAP_strdupWtoA( GetProcessHeap(), 0, key);
1775 LPSTR filenameA = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
1776 LPSTR bufferA = HeapAlloc( GetProcessHeap(), 0, len );
1777
Alexandre Julliarda3960291999-02-26 11:11:13 +00001778 INT ret = GetPrivateProfileStructA( sectionA, keyA, bufferA,
Eric Poueched155751999-03-25 13:24:08 +00001779 len, filenameA );
Alexandre Julliard24a62ab2000-11-28 22:40:56 +00001780 if (len > 0 && !MultiByteToWideChar( CP_ACP, 0, bufferA, -1, buffer, len ))
1781 ((LPWSTR)buffer)[len-1] = 0;
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001782 HeapFree( GetProcessHeap(), 0, bufferA);
1783 HeapFree( GetProcessHeap(), 0, sectionA );
1784 HeapFree( GetProcessHeap(), 0, keyA );
1785 HeapFree( GetProcessHeap(), 0, filenameA );
1786
1787 return ret;
1788}
1789
1790
1791
Ulrich Weigand8228bd61999-04-22 09:56:09 +00001792/***********************************************************************
1793 * WritePrivateProfileStruct16 (KERNEL.406)
1794 */
1795BOOL16 WINAPI WritePrivateProfileStruct16 (LPCSTR section, LPCSTR key,
1796 LPVOID buf, UINT16 bufsize, LPCSTR filename)
1797{
1798 return WritePrivateProfileStructA( section, key, buf, bufsize, filename );
1799}
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001800
1801/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001802 * WritePrivateProfileStructA (KERNEL32.744)
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001803 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001804BOOL WINAPI WritePrivateProfileStructA (LPCSTR section, LPCSTR key,
Alexandre Julliarde61b7921999-07-31 19:24:11 +00001805 LPVOID buf, UINT bufsize, LPCSTR filename)
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001806{
Alexandre Julliarde61b7921999-07-31 19:24:11 +00001807 BOOL ret = FALSE;
Andreas Mohr17b1f462000-01-30 21:15:14 +00001808 LPBYTE binbuf;
1809 LPSTR outstring, p;
1810 DWORD sum = 0;
Alexandre Julliarde61b7921999-07-31 19:24:11 +00001811
1812 if (!section && !key && !buf) /* flush the cache */
1813 return WritePrivateProfileStringA( NULL, NULL, NULL, filename );
Eric Poueched155751999-03-25 13:24:08 +00001814
Andreas Mohr17b1f462000-01-30 21:15:14 +00001815 /* allocate string buffer for hex chars + checksum hex char + '\0' */
Alexandre Julliard90476d62000-02-16 22:47:24 +00001816 outstring = HeapAlloc( GetProcessHeap(), 0, bufsize*2 + 2 + 1);
Andreas Mohr17b1f462000-01-30 21:15:14 +00001817 p = outstring;
1818 for (binbuf = (LPBYTE)buf; binbuf < (LPBYTE)buf+bufsize; binbuf++) {
1819 *p++ = hex[*binbuf >> 4];
1820 *p++ = hex[*binbuf & 0xf];
1821 sum += *binbuf;
1822 }
1823 /* checksum is sum & 0xff */
1824 *p++ = hex[(sum & 0xf0) >> 4];
1825 *p++ = hex[sum & 0xf];
1826 *p++ = '\0';
1827
Eric Poueched155751999-03-25 13:24:08 +00001828 EnterCriticalSection( &PROFILE_CritSect );
1829
Alexandre Julliarde61b7921999-07-31 19:24:11 +00001830 if (PROFILE_Open( filename ))
Andreas Mohr17b1f462000-01-30 21:15:14 +00001831 ret = PROFILE_SetString( section, key, outstring );
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001832
Eric Poueched155751999-03-25 13:24:08 +00001833 LeaveCriticalSection( &PROFILE_CritSect );
1834
Alexandre Julliard90476d62000-02-16 22:47:24 +00001835 HeapFree( GetProcessHeap(), 0, outstring );
Andreas Mohr17b1f462000-01-30 21:15:14 +00001836
Eric Poueched155751999-03-25 13:24:08 +00001837 return ret;
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001838}
1839
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001840/***********************************************************************
Patrik Stridvall2d6457c2000-03-28 20:22:59 +00001841 * WritePrivateProfileStructW (KERNEL32.544)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001842 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001843BOOL WINAPI WritePrivateProfileStructW (LPCWSTR section, LPCWSTR key,
Eric Poueched155751999-03-25 13:24:08 +00001844 LPVOID buf, UINT bufsize, LPCWSTR filename)
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001845{
1846 LPSTR sectionA = HEAP_strdupWtoA( GetProcessHeap(), 0, section );
1847 LPSTR keyA = HEAP_strdupWtoA( GetProcessHeap(), 0, key);
1848 LPSTR filenameA = HEAP_strdupWtoA( GetProcessHeap(), 0, filename );
Alexandre Julliarda3960291999-02-26 11:11:13 +00001849 INT ret = WritePrivateProfileStructA( sectionA, keyA, buf, bufsize,
Eric Poueched155751999-03-25 13:24:08 +00001850 filenameA );
Alexandre Julliarda0d77311998-09-13 16:32:00 +00001851 HeapFree( GetProcessHeap(), 0, sectionA );
1852 HeapFree( GetProcessHeap(), 0, keyA );
1853 HeapFree( GetProcessHeap(), 0, filenameA );
1854
1855 return ret;
1856}
1857
Alexandre Julliardebfc0fe1998-06-28 18:40:26 +00001858
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001859/***********************************************************************
1860 * WriteOutProfiles (KERNEL.315)
1861 */
Alexandre Julliarda3960291999-02-26 11:11:13 +00001862void WINAPI WriteOutProfiles16(void)
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001863{
Eric Poueched155751999-03-25 13:24:08 +00001864 EnterCriticalSection( &PROFILE_CritSect );
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001865 PROFILE_FlushFile();
Eric Poueched155751999-03-25 13:24:08 +00001866 LeaveCriticalSection( &PROFILE_CritSect );
Alexandre Julliard7e56f681996-01-31 19:02:28 +00001867}
Marcus Meissner9b4fcf71999-12-04 04:19:55 +00001868
1869/***********************************************************************
1870 * CloseProfileUserMapping (KERNEL.138)
1871 */
1872BOOL WINAPI CloseProfileUserMapping(void) {
1873 FIXME("(), stub!\n");
1874 SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
1875 return FALSE;
1876}