usp10: Glyph class tables are just generic class tables.
[wine] / dlls / msvcrt / data.c
1 /*
2  * msvcrt.dll dll data items
3  *
4  * Copyright 2000 Jon Griffiths
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  */
20
21 #include "config.h"
22 #include "wine/port.h"
23
24 #include <math.h>
25 #include "msvcrt.h"
26 #include "wine/library.h"
27 #include "wine/unicode.h"
28 #include "wine/debug.h"
29
30 WINE_DEFAULT_DEBUG_CHANNEL(msvcrt);
31
32 int MSVCRT___argc = 0;
33 static int argc_expand;
34 static int wargc_expand;
35 unsigned int MSVCRT_basemajor = 0;/* FIXME: */
36 unsigned int MSVCRT_baseminor = 0;/* FIXME: */
37 unsigned int MSVCRT_baseversion = 0; /* FIXME: */
38 unsigned int MSVCRT__commode = 0;
39 unsigned int MSVCRT__fmode = 0;
40 unsigned int MSVCRT_osmajor = 0;/* FIXME: */
41 unsigned int MSVCRT_osminor = 0;/* FIXME: */
42 unsigned int MSVCRT_osmode = 0;/* FIXME: */
43 unsigned int MSVCRT__osver = 0;
44 unsigned int MSVCRT__osplatform = 0;
45 unsigned int MSVCRT_osversion = 0; /* FIXME: */
46 unsigned int MSVCRT__winmajor = 0;
47 unsigned int MSVCRT__winminor = 0;
48 unsigned int MSVCRT__winver = 0;
49 unsigned int MSVCRT___setlc_active = 0;
50 unsigned int MSVCRT___unguarded_readlc_active = 0;
51 double MSVCRT__HUGE = 0;
52 char **MSVCRT___argv = NULL;
53 static char **argv_expand;
54 MSVCRT_wchar_t **MSVCRT___wargv = NULL;
55 static MSVCRT_wchar_t **wargv_expand;
56 char *MSVCRT__acmdln = NULL;
57 MSVCRT_wchar_t *MSVCRT__wcmdln = NULL;
58 char **MSVCRT__environ = NULL;
59 MSVCRT_wchar_t **MSVCRT__wenviron = NULL;
60 char **MSVCRT___initenv = NULL;
61 MSVCRT_wchar_t **MSVCRT___winitenv = NULL;
62 int MSVCRT_app_type = 0;
63 char* MSVCRT__pgmptr = NULL;
64 WCHAR* MSVCRT__wpgmptr = NULL;
65
66 /* Get a snapshot of the current environment
67  * and construct the __p__environ array
68  *
69  * The pointer returned from GetEnvironmentStrings may get invalid when
70  * some other module cause a reallocation of the env-variable block
71  *
72  * blk is an array of pointers to environment strings, ending with a NULL
73  * and after that the actual copy of the environment strings, ending in a \0
74  */
75 char ** msvcrt_SnapshotOfEnvironmentA(char **blk)
76 {
77   char* environ_strings = GetEnvironmentStringsA();
78   int count = 1, len = 1, i = 0; /* keep space for the trailing NULLS */
79   char *ptr;
80
81   for (ptr = environ_strings; *ptr; ptr += strlen(ptr) + 1)
82   {
83     count++;
84     len += strlen(ptr) + 1;
85   }
86   if (blk)
87       blk = HeapReAlloc( GetProcessHeap(), 0, blk, count* sizeof(char*) + len );
88   else
89     blk = HeapAlloc(GetProcessHeap(), 0, count* sizeof(char*) + len );
90
91   if (blk)
92     {
93       if (count)
94         {
95           memcpy(&blk[count],environ_strings,len);
96           for (ptr = (char*) &blk[count]; *ptr; ptr += strlen(ptr) + 1)
97             {
98               blk[i++] = ptr;
99             }
100         }
101       blk[i] = NULL;
102     }
103   FreeEnvironmentStringsA(environ_strings);
104   return blk;
105 }
106
107 MSVCRT_wchar_t ** msvcrt_SnapshotOfEnvironmentW(MSVCRT_wchar_t **wblk)
108 {
109   MSVCRT_wchar_t* wenviron_strings = GetEnvironmentStringsW();
110   int count = 1, len = 1, i = 0; /* keep space for the trailing NULLS */
111   MSVCRT_wchar_t *wptr;
112
113   for (wptr = wenviron_strings; *wptr; wptr += strlenW(wptr) + 1)
114   {
115     count++;
116     len += strlenW(wptr) + 1;
117   }
118   if (wblk)
119       wblk = HeapReAlloc( GetProcessHeap(), 0, wblk, count* sizeof(MSVCRT_wchar_t*) + len * sizeof(MSVCRT_wchar_t));
120   else
121     wblk = HeapAlloc(GetProcessHeap(), 0, count* sizeof(MSVCRT_wchar_t*) + len * sizeof(MSVCRT_wchar_t));
122   if (wblk)
123     {
124       if (count)
125         {
126           memcpy(&wblk[count],wenviron_strings,len * sizeof(MSVCRT_wchar_t));
127           for (wptr = (MSVCRT_wchar_t*)&wblk[count]; *wptr; wptr += strlenW(wptr) + 1)
128             {
129               wblk[i++] = wptr;
130             }
131         }
132       wblk[i] = NULL;
133     }
134   FreeEnvironmentStringsW(wenviron_strings);
135   return wblk;
136 }
137
138 typedef void (CDECL *_INITTERMFUN)(void);
139 typedef int (CDECL *_INITTERM_E_FN)(void);
140
141 /***********************************************************************
142  *              __p___argc (MSVCRT.@)
143  */
144 int* CDECL __p___argc(void) { return &MSVCRT___argc; }
145
146 /***********************************************************************
147  *              __p__commode (MSVCRT.@)
148  */
149 unsigned int* CDECL __p__commode(void) { return &MSVCRT__commode; }
150
151
152 /***********************************************************************
153  *              __p__pgmptr (MSVCRT.@)
154  */
155 char** CDECL __p__pgmptr(void) { return &MSVCRT__pgmptr; }
156
157 /***********************************************************************
158  *              __p__wpgmptr (MSVCRT.@)
159  */
160 WCHAR** CDECL __p__wpgmptr(void) { return &MSVCRT__wpgmptr; }
161
162 /***********************************************************************
163  *              _get_pgmptr (MSVCRT.@)
164  */
165 int CDECL _get_pgmptr(char** p)
166 {
167   if (!MSVCRT_CHECK_PMT(p)) return MSVCRT_EINVAL;
168
169   *p = MSVCRT__pgmptr;
170   return 0;
171 }
172
173 /***********************************************************************
174  *              _get_wpgmptr (MSVCRT.@)
175  */
176 int CDECL _get_wpgmptr(WCHAR** p)
177 {
178   if (!MSVCRT_CHECK_PMT(p)) return MSVCRT_EINVAL;
179   *p = MSVCRT__wpgmptr;
180   return 0;
181 }
182
183 /***********************************************************************
184  *              __p__fmode (MSVCRT.@)
185  */
186 unsigned int* CDECL __p__fmode(void) { return &MSVCRT__fmode; }
187
188 /***********************************************************************
189  *              __p__osver (MSVCRT.@)
190  */
191 unsigned int* CDECL __p__osver(void) { return &MSVCRT__osver; }
192
193 /***********************************************************************
194  *              __p__winmajor (MSVCRT.@)
195  */
196 unsigned int* CDECL __p__winmajor(void) { return &MSVCRT__winmajor; }
197
198 /***********************************************************************
199  *              __p__winminor (MSVCRT.@)
200  */
201 unsigned int* CDECL __p__winminor(void) { return &MSVCRT__winminor; }
202
203 /***********************************************************************
204  *              __p__winver (MSVCRT.@)
205  */
206 unsigned int* CDECL __p__winver(void) { return &MSVCRT__winver; }
207
208 /*********************************************************************
209  *              __p__acmdln (MSVCRT.@)
210  */
211 char** CDECL __p__acmdln(void) { return &MSVCRT__acmdln; }
212
213 /*********************************************************************
214  *              __p__wcmdln (MSVCRT.@)
215  */
216 MSVCRT_wchar_t** CDECL __p__wcmdln(void) { return &MSVCRT__wcmdln; }
217
218 /*********************************************************************
219  *              __p___argv (MSVCRT.@)
220  */
221 char*** CDECL __p___argv(void) { return &MSVCRT___argv; }
222
223 /*********************************************************************
224  *              __p___wargv (MSVCRT.@)
225  */
226 MSVCRT_wchar_t*** CDECL __p___wargv(void) { return &MSVCRT___wargv; }
227
228 /*********************************************************************
229  *              __p__environ (MSVCRT.@)
230  */
231 char*** CDECL MSVCRT___p__environ(void)
232 {
233   return &MSVCRT__environ;
234 }
235
236 /*********************************************************************
237  *              __p__wenviron (MSVCRT.@)
238  */
239 MSVCRT_wchar_t*** CDECL MSVCRT___p__wenviron(void)
240 {
241   return &MSVCRT__wenviron;
242 }
243
244 /*********************************************************************
245  *              __p___initenv (MSVCRT.@)
246  */
247 char*** CDECL __p___initenv(void) { return &MSVCRT___initenv; }
248
249 /*********************************************************************
250  *              __p___winitenv (MSVCRT.@)
251  */
252 MSVCRT_wchar_t*** CDECL __p___winitenv(void) { return &MSVCRT___winitenv; }
253
254 /*********************************************************************
255  *              _get_osplatform (MSVCRT.@)
256  */
257 int CDECL MSVCRT__get_osplatform(int *pValue)
258 {
259     if (!MSVCRT_CHECK_PMT(pValue != NULL)) return MSVCRT_EINVAL;
260     *pValue = MSVCRT__osplatform;
261     return 0;
262 }
263
264 /* INTERNAL: Create a wide string from an ascii string */
265 MSVCRT_wchar_t *msvcrt_wstrdupa(const char *str)
266 {
267   const unsigned int len = strlen(str) + 1 ;
268   MSVCRT_wchar_t *wstr = MSVCRT_malloc(len* sizeof (MSVCRT_wchar_t));
269   if (!wstr)
270     return NULL;
271    MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED,str,len,wstr,len);
272   return wstr;
273 }
274
275 /*********************************************************************
276  *              ___unguarded_readlc_active_add_func (MSVCRT.@)
277  */
278 unsigned int * CDECL MSVCRT____unguarded_readlc_active_add_func(void)
279 {
280   return &MSVCRT___unguarded_readlc_active;
281 }
282
283 /*********************************************************************
284  *              ___setlc_active_func (MSVCRT.@)
285  */
286 unsigned int CDECL MSVCRT____setlc_active_func(void)
287 {
288   return MSVCRT___setlc_active;
289 }
290
291 /* INTERNAL: Since we can't rely on Winelib startup code calling w/getmainargs,
292  * we initialise data values during DLL loading. When called by a native
293  * program we simply return the data we've already initialised. This also means
294  * you can call multiple times without leaking
295  */
296 void msvcrt_init_args(void)
297 {
298   OSVERSIONINFOW osvi;
299
300   MSVCRT__acmdln = MSVCRT__strdup( GetCommandLineA() );
301   MSVCRT__wcmdln = msvcrt_wstrdupa(MSVCRT__acmdln);
302   MSVCRT___argc = __wine_main_argc;
303   MSVCRT___argv = __wine_main_argv;
304   MSVCRT___wargv = __wine_main_wargv;
305
306   TRACE("got %s, wide = %s argc=%d\n", debugstr_a(MSVCRT__acmdln),
307         debugstr_w(MSVCRT__wcmdln),MSVCRT___argc);
308
309   osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFOW);
310   GetVersionExW( &osvi );
311   MSVCRT__winver     = (osvi.dwMajorVersion << 8) | osvi.dwMinorVersion;
312   MSVCRT__winmajor   = osvi.dwMajorVersion;
313   MSVCRT__winminor   = osvi.dwMinorVersion;
314   MSVCRT__osver      = osvi.dwBuildNumber;
315   MSVCRT__osplatform = osvi.dwPlatformId;
316   MSVCRT_osversion   = MSVCRT__winver;
317   MSVCRT_osmajor     = MSVCRT__winmajor;
318   MSVCRT_osminor     = MSVCRT__winminor;
319   MSVCRT_baseversion = MSVCRT__osver;
320   MSVCRT_baseminor   = MSVCRT_baseversion & 0xFF;
321   MSVCRT_basemajor   = (MSVCRT_baseversion >> 8) & 0xFF;
322   TRACE( "winver %08x winmajor %08x winminor %08x osver%08x baseversion %08x basemajor %08x baseminor %08x\n",
323           MSVCRT__winver, MSVCRT__winmajor, MSVCRT__winminor, MSVCRT__osver, MSVCRT_baseversion,
324           MSVCRT_basemajor, MSVCRT_baseminor);
325   TRACE( "osversion %08x osmajor %08x osminor %08x\n", MSVCRT_osversion, MSVCRT_osmajor, MSVCRT_osminor);
326
327   MSVCRT__HUGE = HUGE_VAL;
328   MSVCRT___setlc_active = 0;
329   MSVCRT___unguarded_readlc_active = 0;
330   MSVCRT__fmode = MSVCRT__O_TEXT;
331
332   MSVCRT__environ = msvcrt_SnapshotOfEnvironmentA(NULL);
333   MSVCRT___initenv = msvcrt_SnapshotOfEnvironmentA(NULL);
334   MSVCRT___winitenv = msvcrt_SnapshotOfEnvironmentW(NULL);
335
336   MSVCRT__pgmptr = HeapAlloc(GetProcessHeap(), 0, MAX_PATH);
337   if (MSVCRT__pgmptr)
338   {
339     if (!GetModuleFileNameA(0, MSVCRT__pgmptr, MAX_PATH))
340       MSVCRT__pgmptr[0] = '\0';
341     else
342       MSVCRT__pgmptr[MAX_PATH - 1] = '\0';
343   }
344
345   MSVCRT__wpgmptr = HeapAlloc(GetProcessHeap(), 0, MAX_PATH * sizeof(WCHAR));
346   if (MSVCRT__wpgmptr)
347   {
348     if (!GetModuleFileNameW(0, MSVCRT__wpgmptr, MAX_PATH))
349       MSVCRT__wpgmptr[0] = '\0';
350     else
351       MSVCRT__wpgmptr[MAX_PATH - 1] = '\0';
352   }
353 }
354
355 /* INTERNAL: free memory used by args */
356 void msvcrt_free_args(void)
357 {
358   /* FIXME: more things to free */
359   HeapFree(GetProcessHeap(), 0, MSVCRT___initenv);
360   HeapFree(GetProcessHeap(), 0, MSVCRT___winitenv);
361   HeapFree(GetProcessHeap(), 0, MSVCRT__environ);
362   HeapFree(GetProcessHeap(), 0, MSVCRT__wenviron);
363   HeapFree(GetProcessHeap(), 0, MSVCRT__pgmptr);
364   HeapFree(GetProcessHeap(), 0, MSVCRT__wpgmptr);
365   HeapFree(GetProcessHeap(), 0, argv_expand);
366   HeapFree(GetProcessHeap(), 0, wargv_expand);
367 }
368
369 static int build_expanded_argv(int *argc, char **argv)
370 {
371     int i, size=0, args_no=0, path_len;
372     BOOL is_expandable;
373     HANDLE h;
374
375     args_no = 0;
376     for(i=0; i<__wine_main_argc; i++) {
377         WIN32_FIND_DATAA data;
378         int len = 0;
379
380         is_expandable = FALSE;
381         for(path_len = strlen(__wine_main_argv[i])-1; path_len>=0; path_len--) {
382             if(__wine_main_argv[i][path_len]=='*' || __wine_main_argv[i][path_len]=='?')
383                 is_expandable = TRUE;
384             else if(__wine_main_argv[i][path_len]=='\\' || __wine_main_argv[i][path_len]=='/')
385                 break;
386         }
387         path_len++;
388
389         if(is_expandable)
390             h = FindFirstFileA(__wine_main_argv[i], &data);
391         else
392             h = INVALID_HANDLE_VALUE;
393
394         if(h != INVALID_HANDLE_VALUE) {
395             do {
396                 if(data.cFileName[0]=='.' && (data.cFileName[1]=='\0' ||
397                             (data.cFileName[1]=='.' && data.cFileName[2]=='\0')))
398                     continue;
399
400                 len = strlen(data.cFileName)+1;
401                 if(argv) {
402                     argv[args_no] = (char*)(argv+*argc)+size;
403                     memcpy(argv[args_no], __wine_main_argv[i], path_len*sizeof(char));
404                     memcpy(argv[args_no]+path_len, data.cFileName, len*sizeof(char));
405                 }
406                 args_no++;
407                 size += len+path_len;
408             }while(FindNextFileA(h, &data));
409             CloseHandle(h);
410         }
411
412         if(!len) {
413             len = strlen(__wine_main_argv[i])+1;
414             if(argv) {
415                 argv[args_no] = (char*)(argv+*argc)+size;
416                 memcpy(argv[args_no], __wine_main_argv[i], len*sizeof(char));
417             }
418             args_no++;
419             size += len;
420         }
421     }
422
423     size += args_no*sizeof(char*);
424     *argc = args_no;
425     return size;
426 }
427
428 /*********************************************************************
429  *              __getmainargs (MSVCRT.@)
430  */
431 void CDECL __getmainargs(int *argc, char** *argv, char** *envp,
432                          int expand_wildcards, int *new_mode)
433 {
434     TRACE("(%p,%p,%p,%d,%p).\n", argc, argv, envp, expand_wildcards, new_mode);
435
436     if (expand_wildcards) {
437         HeapFree(GetProcessHeap(), 0, argv_expand);
438         argv_expand = NULL;
439
440         argv_expand = HeapAlloc(GetProcessHeap(), 0,
441                 build_expanded_argv(&argc_expand, NULL));
442         if (argv_expand) {
443             build_expanded_argv(&argc_expand, argv_expand);
444
445             MSVCRT___argc = argc_expand;
446             MSVCRT___argv = argv_expand;
447         }else {
448             expand_wildcards = 0;
449         }
450     }
451     if (!expand_wildcards) {
452         MSVCRT___argc = __wine_main_argc;
453         MSVCRT___argv = __wine_main_argv;
454     }
455
456     *argc = MSVCRT___argc;
457     *argv = MSVCRT___argv;
458     *envp = MSVCRT___initenv;
459
460     if (new_mode)
461         MSVCRT__set_new_mode( *new_mode );
462 }
463
464 static int build_expanded_wargv(int *argc, MSVCRT_wchar_t **argv)
465 {
466     int i, size=0, args_no=0, path_len;
467     BOOL is_expandable;
468     HANDLE h;
469
470     args_no = 0;
471     for(i=0; i<__wine_main_argc; i++) {
472         WIN32_FIND_DATAW data;
473         int len = 0;
474
475         is_expandable = FALSE;
476         for(path_len = strlenW(__wine_main_wargv[i])-1; path_len>=0; path_len--) {
477             if(__wine_main_wargv[i][path_len]=='*' || __wine_main_wargv[i][path_len]=='?')
478                 is_expandable = TRUE;
479             else if(__wine_main_wargv[i][path_len]=='\\' || __wine_main_wargv[i][path_len]=='/')
480                 break;
481         }
482         path_len++;
483
484         if(is_expandable)
485             h = FindFirstFileW(__wine_main_wargv[i], &data);
486         else
487             h = INVALID_HANDLE_VALUE;
488
489         if(h != INVALID_HANDLE_VALUE) {
490             do {
491                 if(data.cFileName[0]=='.' && (data.cFileName[1]=='\0' ||
492                             (data.cFileName[1]=='.' && data.cFileName[2]=='\0')))
493                     continue;
494
495                 len = strlenW(data.cFileName)+1;
496                 if(argv) {
497                     argv[args_no] = (MSVCRT_wchar_t*)(argv+*argc)+size;
498                     memcpy(argv[args_no], __wine_main_wargv[i], path_len*sizeof(MSVCRT_wchar_t));
499                     memcpy(argv[args_no]+path_len, data.cFileName, len*sizeof(MSVCRT_wchar_t));
500                 }
501                 args_no++;
502                 size += len+path_len;
503             }while(FindNextFileW(h, &data));
504             CloseHandle(h);
505         }
506
507         if(!len) {
508             len = strlenW(__wine_main_wargv[i])+1;
509             if(argv) {
510                 argv[args_no] = (MSVCRT_wchar_t*)(argv+*argc)+size;
511                 memcpy(argv[args_no], __wine_main_wargv[i], len*sizeof(MSVCRT_wchar_t));
512             }
513             args_no++;
514             size += len;
515         }
516     }
517
518     size *= sizeof(MSVCRT_wchar_t);
519     size += args_no*sizeof(MSVCRT_wchar_t*);
520     *argc = args_no;
521     return size;
522 }
523
524 /*********************************************************************
525  *              __wgetmainargs (MSVCRT.@)
526  */
527 void CDECL __wgetmainargs(int *argc, MSVCRT_wchar_t** *wargv, MSVCRT_wchar_t** *wenvp,
528                           int expand_wildcards, int *new_mode)
529 {
530     TRACE("(%p,%p,%p,%d,%p).\n", argc, wargv, wenvp, expand_wildcards, new_mode);
531
532     if (expand_wildcards) {
533         HeapFree(GetProcessHeap(), 0, wargv_expand);
534         wargv_expand = NULL;
535
536         wargv_expand = HeapAlloc(GetProcessHeap(), 0,
537                 build_expanded_wargv(&wargc_expand, NULL));
538         if (wargv_expand) {
539             build_expanded_wargv(&wargc_expand, wargv_expand);
540
541             MSVCRT___argc = wargc_expand;
542             MSVCRT___wargv = wargv_expand;
543         }else {
544             expand_wildcards = 0;
545         }
546     }
547     if (!expand_wildcards) {
548         MSVCRT___argc = __wine_main_argc;
549         MSVCRT___wargv = __wine_main_wargv;
550     }
551
552     /* Initialize the _wenviron array if it's not already created. */
553     if (!MSVCRT__wenviron)
554         MSVCRT__wenviron = msvcrt_SnapshotOfEnvironmentW(NULL);
555     *argc = MSVCRT___argc;
556     *wargv = MSVCRT___wargv;
557     *wenvp = MSVCRT___winitenv;
558     if (new_mode)
559         MSVCRT__set_new_mode( *new_mode );
560 }
561
562 /*********************************************************************
563  *              _initterm (MSVCRT.@)
564  */
565 void CDECL _initterm(_INITTERMFUN *start,_INITTERMFUN *end)
566 {
567   _INITTERMFUN* current = start;
568
569   TRACE("(%p,%p)\n",start,end);
570   while (current<end)
571   {
572     if (*current)
573     {
574       TRACE("Call init function %p\n",*current);
575       (**current)();
576       TRACE("returned\n");
577     }
578     current++;
579   }
580 }
581
582 /*********************************************************************
583  *  _initterm_e (MSVCRT.@)
584  *
585  * call an array of application initialization functions and report the return value
586  */
587 int CDECL _initterm_e(_INITTERM_E_FN *table, _INITTERM_E_FN *end)
588 {
589     int res = 0;
590
591     TRACE("(%p, %p)\n", table, end);
592
593     while (!res && table < end) {
594         if (*table) {
595             TRACE("calling %p\n", **table);
596             res = (**table)();
597             if (res)
598                 TRACE("function %p failed: 0x%x\n", *table, res);
599
600         }
601         table++;
602     }
603     return res;
604 }
605
606 /*********************************************************************
607  *              __set_app_type (MSVCRT.@)
608  */
609 void CDECL MSVCRT___set_app_type(int app_type)
610 {
611   TRACE("(%d) %s application\n", app_type, app_type == 2 ? "Gui" : "Console");
612   MSVCRT_app_type = app_type;
613 }