msvcmaker: Remove a reference to libs/unicode in msvcmaker.
[wine] / tools / winapi / msvcmaker
1 #! /usr/bin/perl -w
2
3 # Copyright 2002 Patrik Stridvall
4
5 use strict;
6
7 BEGIN {
8     $0 =~ m%^(.*?/?tools)/winapi/msvcmaker$%;
9     require "$1/winapi/setup.pm";
10 }
11
12 use setup qw($current_dir $wine_dir);
13 use lib $setup::winapi_dir;
14 use config qw(get_spec_files get_makefile_in_files);
15 use output qw($output);
16 use util qw(replace_file);
17
18 use msvcmaker_options qw($options);
19
20 if($options->progress) {
21     $output->enable_progress;
22 } else {
23     $output->disable_progress;
24 }
25
26 ########################################################################
27 # main
28
29 my @spec_files = get_spec_files("winelib");
30 my @makefile_in_files = get_makefile_in_files("winelib");
31
32 my $wine = 1;
33
34 my $output_prefix_dir = "Output";
35 my $no_release = 1;
36
37 my %modules;
38
39 sub read_spec_file($) {
40     my $spec_file = shift;
41
42     my $module = $spec_file;
43     $module =~ s%^.*?([^/]+)\.spec$%$1%;
44     $module .= ".dll" if $module !~ /\./;
45
46     my $type = "win32";
47
48     open(IN, "< $wine_dir/$spec_file") || die "Error: Can't open $wine_dir/$spec_file: $!\n";
49
50     my $header = 1;
51     my $lookahead = 0;
52     while($lookahead || defined($_ = <IN>)) {
53         $lookahead = 0;
54
55         s/^\s*?(.*?)\s*$/$1/; # remove whitespace at beginning and end of line
56         s/^(.*?)\s*#.*$/$1/;  # remove comments
57         /^$/ && next;         # skip empty lines
58
59         if($header)  {
60             if(/^(?:\d+|@)/) {
61                 $header = 0;
62                 $lookahead = 1;
63             }
64             next;
65         }
66
67         if(/^(\d+|@)\s+pascal(?:16)?/) {
68             $type = "win16";
69             last;
70         }
71     }
72     close(IN);
73
74     # FIXME: Kludge
75     if($module =~ /^(?:(?:imm|ole2conv|ole2prox|ole2thk|rasapi16|msacm|windebug)\.dll|comm\.drv)$/) {
76         $type = "win16";
77     }
78
79     if($type eq "win32") {
80         $modules{$module}{module} = $module;
81         $modules{$module}{type} = $type;
82         $modules{$module}{spec_file} = $spec_file;
83     }
84 }
85
86 if ($options->wine || $options->winetest) {
87     foreach my $spec_file (@spec_files) {
88         read_spec_file($spec_file);
89     }
90 }
91
92 my @gdi32_dirs = qw(dlls/gdi/enhmfdrv dlls/gdi/mfdrv);
93
94 push @makefile_in_files, "libs/wine/Makefile.in";
95 push @makefile_in_files, "tools/winebuild/Makefile.in";
96
97 sub filter_files($$) {
98     my $files = shift;
99     my $filter = shift;
100
101     my $filtered_files = [];
102     my $rest_of_files = [];
103     foreach my $file (@$files) {
104         if($file =~ /$filter/) {
105             $file =~ s%.*?([^/]+)$%./$1%; # FIXME: Kludge
106             push @$filtered_files, $file;
107         } else {
108             push @$rest_of_files, $file;
109         }
110     }
111     return ($rest_of_files, $filtered_files);
112 }
113
114 my %wine_test_dsp_files;
115
116 MAKEFILE_IN: foreach my $makefile_in_file (@makefile_in_files) {
117     open(IN, "< $wine_dir/$makefile_in_file") || die "Error: Can't open $wine_dir/$makefile_in_file: $!\n";
118
119     my $topobjdir;
120     my $module;
121     my $testdll;
122     my @imports;
123
124     my %vars;
125
126     my $again = 0;
127     my $lookahead = 0;
128
129     if($makefile_in_file eq "loader/Makefile.in") {
130         next;
131     }
132
133     while($again || defined(my $line = <IN>)) {
134         if(!$again) {
135             chomp $line;
136             if($lookahead) {
137                 $lookahead = 0;
138                 $_ .= "\n" . $line;
139             } else {
140                 $_ = $line;
141             }
142         } else {
143             $again = 0;
144         }
145
146         s/^\s*?(.*?)\s*$/$1/; # remove whitespace at beginning and end of line
147         s/^(.*?)\s*#.*$/$1/;  # remove comments
148         /^$/ && next;         # skip empty lines
149
150         if(s/\\$/ /s) {
151             $lookahead = 1;
152             next;
153         }
154
155         if(/^MODULE\s*=\s*([\w\.]+)$/) {
156             $module = $1;
157
158             if($module eq "none") {
159                 if($makefile_in_file eq "tools/winebuild/Makefile.in") {
160                     $module = "winebuild.exe";
161                 } else {
162                     next MAKEFILE_IN;
163                 }
164             }
165         } elsif(/^LIBRARY\s*=\s*(\S+)\s*$/) {
166             $module = $1  . "\.lib";
167         } elsif(/^TOPOBJDIR\s*=\s*(\S+)\s*$/) {
168             $topobjdir = $1;
169         } elsif (/^TESTDLL\s*=\s*(\S+)\s*$/) {
170             $testdll = $1;
171         } elsif (/^IMPORTS\s*=\s*/) {
172             push @imports, grep !/^ntdll$/, split /\s+/s, $';
173         } elsif (/^EXTRALIBS\s*=\s*/) {
174             push @imports, map { /^-l(dxerr8|dxerr9|dxguid|strmiids|uuid)$/ ? $1 : () } split /\s+/s, $';
175         } elsif (/^CTESTS\s*=\s*/) {
176             my @files = split /\s+/s, $';
177
178             my $dir = $makefile_in_file;
179             $dir =~ s/\/Makefile\.in$//;
180
181             my $dsp_file = $testdll;
182             $dsp_file =~ s/\.(dll|drv)$/_test.dsp/;
183             $dsp_file = "$dir/$dsp_file";
184
185             $wine_test_dsp_files{$dsp_file}{files} = [@files, "testlist.c"];
186             $wine_test_dsp_files{$dsp_file}{imports} = [@imports];
187         } elsif(/^(\w+)\s*=\s*/) {
188             my $var = $1;
189             my @files = split /\s+/s, $';
190
191             @files = map {
192                 if(/^\$\((\w+):\%=(.*?)\%(.*?)\)$/) {
193                     my @list = @{$vars{$1}};
194                     my $prefix = $2;
195                     my $suffix = $3;
196                     foreach my $item (@list) {
197                         $item = "$prefix$item$suffix";
198                     }
199                     @list;
200                 } elsif(/^\$\(TOPOBJDIR\)(.*?)$/) {
201                     "$topobjdir$1";
202                 } elsif(/^\$/) {
203                     print STDERR "unknown variable '$_'\n" if 0;
204                     ();
205                 } else {
206                     $_;
207                 }
208             } @files;
209
210             $vars{$var} = \@files;
211         }
212     }
213
214     close(IN);
215
216     next if !$module;
217
218     my $c_srcs = [];
219     my $source_files = [];
220     if(exists($vars{C_SRCS})) {
221         $c_srcs = [sort(@{$vars{C_SRCS}})];
222         $source_files = [sort(@{$vars{C_SRCS}})];
223     }
224
225     my $header_files = [];
226     if(exists($vars{H_SRCS})) {
227         $header_files = [sort(@{$vars{H_SRCS}})];
228     }
229
230     my $resource_files = [];
231     if(exists($vars{RC_SRCS})) {
232         $resource_files = [sort(@{$vars{RC_SRCS}})];
233     }
234
235     my $project = $module;
236     $project =~ s/\.(?:dll|exe|lib)$//;
237     $project =~ y/./_/;
238
239     my $type;
240     if($module =~ /\.exe$/) {
241         $type = "exe";
242     } elsif($module =~ /\.lib$/) {
243         $type = "lib";
244     } else {
245         $type = "dll";
246     }
247
248     my $dsp_file = $makefile_in_file;
249     $dsp_file =~ s/Makefile.in$/$project.dsp/;
250
251     if($module eq "gdi32.dll") {
252         foreach my $dir (@gdi32_dirs) {
253             my $dir2 = $dir;
254             $dir2 =~ s%^.*?/([^/]+)$%$1%;
255
256             my $module = "gdi32_$dir2.lib";
257             $module =~ s%/%_%g;
258
259             my $project = "gdi32_$dir2";
260             $project =~ s%/%_%g;
261
262             my $type = "lib";
263             my $dsp_file = "$dir/$project.dsp";
264
265             ($source_files, my $local_source_files) = filter_files($source_files, "$dir2/");
266             ($header_files, my $local_header_files) = filter_files($header_files, "$dir2/");
267             ($resource_files, my $local_resource_files) = filter_files($resource_files, "$dir2/");
268
269             $modules{$module}{wine} = 1;
270             $modules{$module}{winetest} = 0;
271             $modules{$module}{project} = $project;
272             $modules{$module}{type} = $type;
273             $modules{$module}{dsp_file} = $dsp_file;
274             $modules{$module}{c_srcs} = $c_srcs;
275             $modules{$module}{source_files} = $local_source_files;
276             $modules{$module}{header_files} = $local_header_files;
277             $modules{$module}{resource_files} = $local_resource_files;
278             $modules{$module}{imports} = [];
279         }
280     }
281
282     $modules{$module}{wine} = 1;
283     $modules{$module}{winetest} = 0;
284     $modules{$module}{project} = $project;
285     $modules{$module}{type} = $type;
286     $modules{$module}{dsp_file} = $dsp_file;
287     $modules{$module}{c_srcs} = $c_srcs;
288     $modules{$module}{source_files} = $source_files;
289     $modules{$module}{header_files} = $header_files;
290     $modules{$module}{resource_files} = $resource_files;
291     $modules{$module}{imports} = [];
292 }
293
294 $wine_test_dsp_files{"wineruntests.dsp"}{files} = ["runtests.c"];
295 $wine_test_dsp_files{"wineruntests.dsp"}{imports} = [];
296
297 $wine_test_dsp_files{"winetest.dsp"}{files} = [
298   'include/wine/exception.h',
299   'include/wine/test.h',
300   'include/wine/unicode.h',
301   'winetest.c'
302 ];
303 $wine_test_dsp_files{"winetest.dsp"}{imports} = [];
304
305 my %runtests = ();
306
307 foreach my $dsp_file (keys(%wine_test_dsp_files)) {
308     my $project = $dsp_file;
309     $project =~ s%^(?:.*?/)?([^/]+)\.dsp$%$1%;
310
311     my @files = @{$wine_test_dsp_files{$dsp_file}{files}};
312     my @imports = @{$wine_test_dsp_files{$dsp_file}{imports}};
313
314     my $type;
315     my $c_srcs = [];
316     my $source_files = [];
317     my $header_files = [];
318     my $resource_files = [];
319
320     my @tests = ();
321
322     if ($project eq "winetest") {
323         $type = "lib";
324         $c_srcs = [@files];
325         $source_files = [@files];
326         $header_files = [];
327         $resource_files = [];
328     } elsif ($project eq "wineruntests") {
329         $type = "exe";
330         $c_srcs = [@files];
331         $source_files = [@files];
332         $header_files = [];
333         $resource_files = [];
334     } else {
335         $type = "exe";
336         $c_srcs = [@files];
337         $source_files = [@files];
338         $header_files = [];
339         $resource_files = [];
340         
341         @tests = map {
342             if (/^testlist\.c$/) {
343                 ();
344             } else {
345                 s/\.c$//;
346                 $_;
347             }
348         } @files;
349
350         $runtests{$dsp_file} = [@tests];
351     }
352     my $module = "$project.$type";
353
354     $modules{$module}{wine} = 0;
355     $modules{$module}{winetest} = 1;
356
357     $modules{$module}{project} = $project;
358     $modules{$module}{type} = $type;
359     $modules{$module}{dsp_file} = $dsp_file;
360     $modules{$module}{c_srcs} = $c_srcs;
361     $modules{$module}{source_files} = $source_files;
362     $modules{$module}{header_files} = $header_files;
363     $modules{$module}{resource_files} = $resource_files;
364     $modules{$module}{imports} = [@imports];
365
366     $modules{$module}{tests} = [@tests];
367 }
368
369 foreach my $module (sort(keys(%modules))) {
370     if($module =~ /^(?:ttydrv.dll|x11drv.dll)$/) {
371         delete $modules{$module};
372     }
373 }
374
375 my @modules = ();
376 foreach my $module (sort(keys(%modules))) {
377     if (($options->wine && $modules{$module}{wine}) ||
378         ($options->winetest && $modules{$module}{winetest}))
379     {
380         push @modules, $module;
381     }
382 }
383
384 my $progress_output;
385 my $progress_current = 0;
386 my $progress_max = scalar(@modules);
387
388 foreach my $module (@modules) {
389     my $dsp_file = $modules{$module}{dsp_file};
390     replace_file("$wine_dir/$dsp_file", \&_generate_dsp, $module);
391 }
392
393 sub _generate_dsp($$) {
394     local *OUT = shift;
395
396     my $module = shift;
397
398     my $dsp_file = $modules{$module}{dsp_file};
399     my $project = $modules{$module}{project};
400     my @imports = @{$modules{$module}{imports}};
401
402     my $lib = ($modules{$module}{type} eq "lib");
403     my $dll = ($modules{$module}{type} eq "dll");
404     my $exe = ($modules{$module}{type} eq "exe");
405
406     my $console = $exe; # FIXME: Not always correct
407
408     my $msvc_wine_dir = do {
409         my @parts = split(m%/%, $dsp_file);
410         if($#parts == 1) {
411             "..";
412         } elsif($#parts == 2) {
413             "..\\..";
414         } else {
415             "..\\..\\..";
416         }
417     };
418     my $wine_include_dir = "$msvc_wine_dir\\include";
419
420     $progress_current++;
421     $output->progress("$dsp_file (file $progress_current of $progress_max)");
422
423     my @c_srcs = @{$modules{$module}{c_srcs}};
424     my @source_files = @{$modules{$module}{source_files}};
425     my @header_files = @{$modules{$module}{header_files}};
426     my @resource_files = @{$modules{$module}{resource_files}};
427
428     if ($project !~ /^wine(?:_unicode|build|runtests|test)?$/ &&
429         $project !~ /^(?:gdi32)_.+?$/ &&
430         $project !~ /_test$/)
431     {
432         push @source_files, "$project.spec";
433         # push @source_files, "$project.spec.c";
434         @source_files = sort(@source_files);
435     }
436
437     my $no_cpp = 1;
438     my $no_msvc_headers = 1;
439     if ($project =~ /^wine(?:runtests|test)$/ || $project =~ /_test$/) {
440         $no_msvc_headers = 0;
441     }
442
443     my @cfgs;
444
445     push @cfgs, "$project - Win32";
446
447     if (!$no_cpp) {
448         my @_cfgs;
449         foreach my $cfg (@cfgs) {
450             push @_cfgs, "$cfg C";
451             push @_cfgs, "$cfg C++";
452         }
453         @cfgs = @_cfgs;
454     }
455
456     if (!$no_release) {
457         my @_cfgs;
458         foreach my $cfg (@cfgs) {
459             push @_cfgs, "$cfg Debug";
460             push @_cfgs, "$cfg Release";
461         }
462         @cfgs = @_cfgs;
463     }
464
465     if (!$no_msvc_headers) {
466         my @_cfgs;
467         foreach my $cfg (@cfgs) {
468             push @_cfgs, "$cfg MSVC Headers";
469             push @_cfgs, "$cfg Wine Headers";
470         }
471         @cfgs = @_cfgs;
472     }
473
474     my $default_cfg = $cfgs[$#cfgs];
475
476     print OUT "# Microsoft Developer Studio Project File - Name=\"$project\" - Package Owner=<4>\r\n";
477     print OUT "# Microsoft Developer Studio Generated Build File, Format Version 6.00\r\n";
478     print OUT "# ** DO NOT EDIT **\r\n";
479     print OUT "\r\n";
480
481     if ($lib) {
482         print OUT "# TARGTYPE \"Win32 (x86) Static Library\" 0x0104\r\n";
483     } elsif ($dll) {
484         print OUT "# TARGTYPE \"Win32 (x86) Dynamic-Link Library\" 0x0102\r\n";
485     } else {
486         print OUT "# TARGTYPE \"Win32 (x86) Console Application\" 0x0103\r\n";
487     }
488     print OUT "\r\n";
489
490     print OUT "CFG=$default_cfg\r\n";
491     print OUT "!MESSAGE This is not a valid makefile. To build this project using NMAKE,\r\n";
492     print OUT "!MESSAGE use the Export Makefile command and run\r\n";
493     print OUT "!MESSAGE \r\n";
494     print OUT "!MESSAGE NMAKE /f \"$project.mak\".\r\n";
495     print OUT "!MESSAGE \r\n";
496     print OUT "!MESSAGE You can specify a configuration when running NMAKE\r\n";
497     print OUT "!MESSAGE by defining the macro CFG on the command line. For example:\r\n";
498     print OUT "!MESSAGE \r\n";
499     print OUT "!MESSAGE NMAKE /f \"$project.mak\" CFG=\"$default_cfg\"\r\n";
500     print OUT "!MESSAGE \r\n";
501     print OUT "!MESSAGE Possible choices for configuration are:\r\n";
502     print OUT "!MESSAGE \r\n";
503     foreach my $cfg (@cfgs) {
504         if ($lib) {
505             print OUT "!MESSAGE \"$cfg\" (based on \"Win32 (x86) Static Library\")\r\n";
506         } elsif ($dll) {
507             print OUT "!MESSAGE \"$cfg\" (based on \"Win32 (x86) Dynamic-Link Library\")\r\n";
508         } else {
509             print OUT "!MESSAGE \"$cfg\" (based on \"Win32 (x86) Console Application\")\r\n";
510         }
511     }
512     print OUT "!MESSAGE \r\n";
513     print OUT "\r\n";
514
515     print OUT "# Begin Project\r\n";
516     print OUT "# PROP AllowPerConfigDependencies 0\r\n";
517     print OUT "# PROP Scc_ProjName \"\"\r\n";
518     print OUT "# PROP Scc_LocalPath \"\"\r\n";
519     print OUT "CPP=cl.exe\r\n";
520     print OUT "MTL=midl.exe\r\n" if !$lib && !$exe;
521     print OUT "RSC=rc.exe\r\n";
522
523     my $n = 0;
524
525     my $output_dir;
526     foreach my $cfg (@cfgs) {
527         if($#cfgs == 0) {
528             # Nothing
529         } elsif($n == 0) {
530             print OUT "!IF  \"\$(CFG)\" == \"$cfg\"\r\n";
531             print OUT "\r\n";
532         } else {
533             print OUT "\r\n";
534             print OUT "!ELSEIF  \"\$(CFG)\" == \"$cfg\"\r\n";
535             print OUT "\r\n";
536         }
537
538         my $debug = ($cfg !~ /Release/);
539         my $msvc_headers = ($cfg =~ /MSVC Headers/);
540
541         print OUT "# PROP BASE Use_MFC 0\r\n";
542
543         if($debug) {
544             print OUT "# PROP BASE Use_Debug_Libraries 1\r\n";
545         } else {
546             print OUT "# PROP BASE Use_Debug_Libraries 0\r\n";
547         }
548
549         $output_dir = $cfg;
550         $output_dir =~ s/^$project - //;
551         $output_dir =~ s/ /_/g;
552         $output_dir =~ s/C\+\+/Cxx/g;
553         if($output_prefix_dir) {
554             $output_dir = "$output_prefix_dir\\$output_dir";
555         }
556
557         print OUT "# PROP BASE Output_Dir \"$output_dir\"\r\n";
558         print OUT "# PROP BASE Intermediate_Dir \"$output_dir\"\r\n";
559
560         print OUT "# PROP BASE Target_Dir \"\"\r\n";
561
562         print OUT "# PROP Use_MFC 0\r\n";
563         if($debug) {
564             print OUT "# PROP Use_Debug_Libraries 1\r\n";
565         } else {
566             print OUT "# PROP Use_Debug_Libraries 0\r\n";
567         }
568         print OUT "# PROP Output_Dir \"$output_dir\"\r\n";
569         print OUT "# PROP Intermediate_Dir \"$output_dir\"\r\n";
570
571         print OUT "# PROP Ignore_Export_Lib 0\r\n" if $dll;
572         print OUT "# PROP Target_Dir \"\"\r\n";
573
574         my @defines;
575         if($debug) {
576             if($lib || $exe) {
577                 print OUT "# ADD BASE CPP /nologo /W3 /Gm /GX /Zi /Od";
578                 @defines = (qw(WINVER=0x0501 _WIN32_WINNT=0x0501 _WIN32_IE=0x0600 WIN32 _DEBUG _MBCS _LIB));
579             } else {
580                 print OUT "# ADD BASE CPP /nologo /MTd /W3 /Gm /GX /Zi /Od";
581                 @defines = (qw(WINVER=0x0501 _WIN32_WINNT=0x0501 _WIN32_IE=0x0600 WIN32 _DEBUG _WINDOWS _MBCS _USRDLL), ("\U${project}\E_EXPORTS"));
582             }
583         } else {
584             if($lib || $exe) {
585                 print OUT "# ADD BASE CPP /nologo /W3 /GX /O2";
586                 @defines = (qw(WINVER=0x0501 _WIN32_WINNT=0x0501 _WIN32_IE=0x0600 WIN32 NDEBUG _MBCS _LIB));
587             } else {
588                 print OUT "# ADD BASE CPP /nologo /MT /W3 /GX /O2";
589                 @defines = (qw(WINVER=0x0501 _WIN32_WINNT=0x0501 _WIN32_IE=0x0600 WIN32 NDEBUG _WINDOWS _MBCS _USRDLL), ("\U${project}\E_EXPORTS"));
590             }
591         }
592
593         foreach my $define (@defines) {
594             print OUT " /D \"$define\"";
595         }
596         print OUT " /YX" if $lib || $exe;
597         print OUT " /FD";
598         print OUT " /GZ" if $debug;
599         print OUT " " if $debug && ($lib || $exe);
600         print OUT " /c";
601         print OUT "\r\n";
602
603         my @defines2;
604         if($debug) {
605             if($lib) {
606                 print OUT "# ADD CPP /nologo /MTd /W3 /Gm /GX /Zi /Od";
607                 @defines2 = qw(WINVER=0x0501 _WIN32_WINNT=0x0501 _WIN32_IE=0x0600 WIN32 _DEBUG _WINDOWS _MBCS _LIB);
608             } else {
609                 print OUT "# ADD CPP /nologo /MTd /W3 /Gm /GX /Zi /Od";
610                 @defines2 = qw(WINVER=0x0501 _WIN32_WINNT=0x0501 _WIN32_IE=0x0600 _DEBUG WIN32 _WINDOWS _MBCS _USRDLL);
611             }
612         } else {
613             if($lib) {
614                 print OUT "# ADD CPP /nologo /MT /W3 /GX /O2";
615                 @defines2 = qw(WINVER=0x0501 _WIN32_WINNT=0x0501 _WIN32_IE=0x0600 WIN32 NDEBUG _WINDOWS _MBCS _LIB);
616             } else {
617                 print OUT "# ADD CPP /nologo /MT /W3 /GX /O2";
618                 @defines2 = qw(WINVER=0x0501 _WIN32_WINNT=0x0501 _WIN32_IE=0x0600 NDEBUG WIN32 _WINDOWS _MBCS _USRDLL);
619             }
620         }
621
622         my @includes = ();
623         if($wine) {
624             push @defines2, "_\U${project}\E_";
625             push @defines2, qw(__WINESRC__) if $project !~ /^(?:wine(?:build|test)|.*?_test)$/;
626             if ($msvc_headers) {
627                 push @defines2, qw(__WINE_USE_NATIVE_HEADERS);
628             }
629             my $output_dir2 = $output_dir;
630             $output_dir =~ s/\\/\\\\/g;
631             push @defines2, "__WINETEST_OUTPUT_DIR=\\\"$output_dir\\\"";
632             push @defines2, qw(__i386__ _X86_);
633
634             if($project =~ /^gdi32_(?:enhmfdrv|mfdrv)$/) {
635                 push @includes, "..";
636             }
637
638             if ($project =~ /_test$/) {
639                 push @includes, "$msvc_wine_dir\\$output_dir";
640             }
641
642             if (!$msvc_headers || $project eq "winetest") {
643                 push @includes, $wine_include_dir;
644             }
645         }
646
647         if($wine) {
648             foreach my $include (@includes) {
649                 if ($include !~ /[\\\"]/) {
650                     print OUT " /I \"$include\"";
651                 } else {
652                     print OUT " /I $include";
653                 }
654             }
655         }
656
657         foreach my $define (@defines2) {
658             if ($define !~ /[\\\"]/) {
659                 print OUT " /D \"$define\"";
660             } else {
661                 print OUT " /D $define";
662             }
663         }
664         print OUT " /D inline=__inline" if $wine;
665         print OUT " /D \"__STDC__\"" if 0 && $wine;
666
667         print OUT " /YX" if $lib;
668         print OUT " /FR" if !$lib;
669         print OUT " /FD";
670         print OUT " /GZ" if $debug;
671         print OUT " " if $debug && $lib;
672         print OUT " /c";
673         print OUT " /TP" if !$no_cpp;
674         print OUT "\r\n";
675
676         if($debug) {
677             print OUT "# SUBTRACT CPP /X /YX\r\n" if $dll;
678             print OUT "# ADD BASE MTL /nologo /D \"_DEBUG\" /mktyplib203 /win32\r\n" if $dll;
679             print OUT "# ADD MTL /nologo /D \"_DEBUG\" /mktyplib203 /win32\r\n" if $dll;
680             print OUT "# ADD BASE RSC /l 0x41d /d \"_DEBUG\"\r\n";
681             print OUT "# ADD RSC /l 0x41d";
682             if($wine) {
683                 foreach my $include (@includes) {
684                     print OUT " /i \"$include\"";
685                 }
686             }
687             print OUT " /d \"_DEBUG\"\r\n";
688         } else {
689             print OUT "# SUBTRACT CPP /YX\r\n" if $dll;
690             print OUT "# ADD BASE MTL /nologo /D \"NDEBUG\" /mktyplib203 /win32\r\n" if $dll;
691             print OUT "# ADD MTL /nologo /D \"NDEBUG\" /mktyplib203 /win32\r\n" if $dll;
692             print OUT "# ADD BASE RSC /l 0x41d /d \"NDEBUG\"\r\n";
693             print OUT "# ADD RSC /l 0x41d";
694             if($wine) {
695                 foreach my $include (@includes) {
696                     print OUT " /i \"$include\"";
697                 }
698             }
699             print OUT "/d \"NDEBUG\"\r\n";
700         }
701         print OUT "BSC32=bscmake.exe\r\n";
702         print OUT "# ADD BASE BSC32 /nologo\r\n";
703         print OUT "# ADD BSC32 /nologo\r\n";
704
705         if($exe || $dll) {
706             print OUT "LINK32=link.exe\r\n";
707             print OUT "# ADD BASE LINK32 ";
708             my @libraries = qw(kernel32.lib user32.lib gdi32.lib winspool.lib
709                                comdlg32.lib advapi32.lib shell32.lib ole32.lib
710                                oleaut32.lib uuid.lib odbc32.lib odbccp32.lib);
711             foreach my $library (@libraries) {
712                 print OUT "$library ";
713             }
714             print OUT " /nologo";
715             print OUT " /dll" if $dll;
716             print OUT " /subsystem:console" if $console;
717             print OUT " /debug" if $debug;
718             print OUT " /machine:I386";
719             print OUT " /pdbtype:sept" if $debug;
720             print OUT "\r\n";
721
722             print OUT "# ADD LINK32";
723             print OUT " /nologo";
724             print OUT " libcmt.lib" if $project =~ /^ntdll$/; # FIXME: Kludge
725             foreach my $import (@imports) {
726                 print OUT " $import.lib" if ($import ne "msvcrt");
727             }
728             print OUT " /dll" if $dll;
729             print OUT " /subsystem:console" if $console;
730             print OUT " /debug" if $debug;
731             print OUT " /machine:I386";
732             print OUT " /nodefaultlib" if $project =~ /^ntdll$/; # FIXME: Kludge
733             print OUT " /def:\"$project.def\"" if $dll;
734             print OUT " /pdbtype:sept" if $debug;
735             print OUT "\r\n";
736         } else {
737             print OUT "LIB32=link.exe -lib\r\n";
738             print OUT "# ADD BASE LIB32 /nologo\r\n";
739             print OUT "# ADD LIB32 /nologo\r\n";
740         }
741
742         $n++;
743     }
744
745     if($#cfgs != 0) {
746         print OUT "\r\n";
747         print OUT "!ENDIF \r\n";
748         print OUT "\r\n";
749     }
750
751     if ($project eq "winebuild") {
752         print OUT "# Begin Special Build Tool\r\n";
753         print OUT "SOURCE=\"\$(InputPath)\"\r\n";
754         print OUT "PostBuild_Desc=Copying wine.lib and wine_unicode.lib ...\r\n";
755         print OUT "PostBuild_Cmds=";
756         print OUT "copy ..\\..\\libs\\wine\\$output_dir\\wine.lib \$(OutDir)\t";
757         print OUT "copy ..\\..\\libs\\unicode\\$output_dir\\wine_unicode.lib \$(OutDir)\r\n";
758         print OUT "# End Special Build Tool\r\n";
759     }
760     print OUT "# Begin Target\r\n";
761     print OUT "\r\n";
762     foreach my $cfg (@cfgs) {
763         print OUT "# Name \"$cfg\"\r\n";
764     }
765
766     print OUT "# Begin Group \"Source Files\"\r\n";
767     print OUT "\r\n";
768     print OUT "# PROP Default_Filter \"cpp;c;cxx;rc;def;r;odl;idl;hpj;bat\"\r\n";
769
770     foreach my $source_file (@source_files) {
771         $source_file =~ s%/%\\%g;
772         if($source_file !~ /^\./) {
773             $source_file = ".\\$source_file";
774         }
775
776         if($source_file =~ /^(.*?)\.spec$/) {
777             my $basename = $1;
778
779             $basename = "$basename.dll" if $basename !~ /\..{1,3}$/;
780             my $dbg_c_file = "$basename.dbg.c";
781
782             print OUT "# Begin Source File\r\n";
783             print OUT "\r\n";
784             print OUT "SOURCE=$dbg_c_file\r\n";
785             print OUT "# End Source File\r\n";
786         }
787
788         print OUT "# Begin Source File\r\n";
789         print OUT "\r\n";
790
791         print OUT "SOURCE=$source_file\r\n";
792
793         if($source_file =~ /^(.*?)\.spec$/) {
794             my $basename = $1;
795
796             my $spec_file = $source_file;
797             my $def_file = "$basename.def";
798
799             $basename = "$basename.dll" if $basename !~ /\..{1,3}$/;
800             my $dbg_file = "$basename.dbg";
801             my $dbg_c_file = "$basename.dbg.c";
802
803             my $srcdir = "."; # FIXME: Is this really always correct?
804
805             print OUT "# Begin Custom Build\r\n";
806             print OUT "InputPath=$spec_file\r\n";
807             print OUT "\r\n";
808             print OUT "BuildCmds= \\\r\n";
809             print OUT "\t..\\..\\tools\\winebuild\\$output_dir\\winebuild.exe --def $spec_file > $def_file \\\r\n";
810             
811             if($project =~ /^ntdll$/) {
812                 my $n = 0;
813                 foreach my $c_src (@c_srcs) {
814                     if($n++ > 0)  {
815                         print OUT "\techo $c_src >> $dbg_file \\\r\n";
816                     } else {
817                         print OUT "\techo $c_src > $dbg_file \\\r\n";
818                     }
819                 }
820                 print OUT "\t..\\..\\tools\\winebuild\\$output_dir\\winebuild.exe";
821                 print OUT " -o $dbg_c_file --debug -C$srcdir $dbg_file \\\r\n";
822             } else {
823                 my $c_srcs = join(" ", grep(/\.c$/, @c_srcs));
824
825                 print OUT "\t..\\..\\tools\\winebuild\\$output_dir\\winebuild.exe";
826                 print OUT " -o $dbg_c_file --debug -C$srcdir $c_srcs \\\r\n";
827             }
828
829             print OUT "\t\r\n";
830             print OUT "\r\n";
831             print OUT "\"$def_file\" : \$(SOURCE) \"\$(INTDIR)\" \"\$(OUTDIR)\"\r\n";
832             print OUT "   \$(BuildCmds)\r\n";
833             print OUT "\r\n";
834             print OUT "\"$dbg_c_file\" : \$(SOURCE) \"\$(INTDIR)\" \"\$(OUTDIR)\"\r\n";
835             print OUT "   \$(BuildCmds)\r\n";
836             print OUT "# End Custom Build\r\n";
837         } elsif($source_file =~ /([^\\]*?\.h)$/) {
838             my $h_file = $1;
839
840             foreach my $cfg (@cfgs) {
841                 if($#cfgs == 0) {
842                     # Nothing
843                 } elsif($n == 0) {
844                     print OUT "!IF  \"\$(CFG)\" == \"$cfg\"\r\n";
845                     print OUT "\r\n";
846                 } else {
847                     print OUT "\r\n";
848                     print OUT "!ELSEIF  \"\$(CFG)\" == \"$cfg\"\r\n";
849                     print OUT "\r\n";
850                 }
851
852                 $output_dir = $cfg;
853                 $output_dir =~ s/^$project - //;
854                 $output_dir =~ s/ /_/g;
855                 $output_dir =~ s/C\+\+/Cxx/g;
856                 if($output_prefix_dir) {
857                     $output_dir = "$output_prefix_dir\\$output_dir";
858                 }
859
860                 print OUT "# Begin Custom Build\r\n";
861                 print OUT "OutDir=$output_dir\r\n";
862                 print OUT "InputPath=$source_file\r\n";
863                 print OUT "\r\n";
864                 print OUT "\"\$(OutDir)\\wine\\$h_file\" : \$(SOURCE) \"\$(INTDIR)\" \"\$(OUTDIR)\"\r\n";
865                 print OUT "\tcopy \"\$(InputPath)\" \"\$(OutDir)\\wine\"\r\n";
866                 print OUT "\r\n";
867                 print OUT "# End Custom Build\r\n";
868             }
869
870             if($#cfgs != 0) {
871                 print OUT "\r\n";
872                 print OUT "!ENDIF \r\n";
873                 print OUT "\r\n";
874             }
875         }
876
877         print OUT "# End Source File\r\n";
878     }
879     print OUT "# End Group\r\n";
880
881     print OUT "# Begin Group \"Header Files\"\r\n";
882     print OUT "\r\n";
883     print OUT "# PROP Default_Filter \"h;hpp;hxx;hm;inl\"\r\n";
884     foreach my $header_file (@header_files) {
885         print OUT "# Begin Source File\r\n";
886         print OUT "\r\n";
887         print OUT "SOURCE=.\\$header_file\r\n";
888         print OUT "# End Source File\r\n";
889     }
890     print OUT "# End Group\r\n";
891
892
893
894     print OUT "# Begin Group \"Resource Files\"\r\n";
895     print OUT "\r\n";
896     print OUT "# PROP Default_Filter \"ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe\"\r\n";
897     foreach my $resource_file (@resource_files) {
898         print OUT "# Begin Source File\r\n";
899         print OUT "\r\n";
900         print OUT "SOURCE=.\\$resource_file\r\n";
901         print OUT "# End Source File\r\n";
902     }
903     print OUT "# End Group\r\n";
904
905     print OUT "# End Target\r\n";
906     print OUT "# End Project\r\n";
907
908     close(OUT);
909 }
910
911 sub _generate_dsw_header($) {
912     local *OUT = shift;
913
914     print OUT "Microsoft Developer Studio Workspace File, Format Version 6.00\r\n";
915     print OUT "# WARNING: DO NOT EDIT OR DELETE THIS WORKSPACE FILE!\r\n";
916     print OUT "\r\n";
917 }
918
919 sub _generate_dsw_project($$$$) {
920     local *OUT = shift;
921
922     my $project = shift;
923     my $dsp_file = shift;
924     my @dependencies = @{(shift)};
925
926     $dsp_file = "./$dsp_file";
927     $dsp_file =~ y%/%\\%;
928     
929     @dependencies = sort(@dependencies);
930
931     print OUT "###############################################################################\r\n";
932     print OUT "\r\n";
933     print OUT "Project: \"$project\"=$dsp_file - Package Owner=<4>\r\n";
934     print OUT "\r\n";
935     print OUT "Package=<5>\r\n";
936     print OUT "{{{\r\n";
937     print OUT "}}}\r\n";
938     print OUT "\r\n";
939     print OUT "Package=<4>\r\n";
940     print OUT "{{{\r\n";
941     foreach my $dependency (@dependencies) {
942         print OUT "    Begin Project Dependency\r\n";
943         print OUT "    Project_Dep_Name $dependency\r\n";
944         print OUT "    End Project Dependency\r\n";
945     }
946     print OUT "}}}\r\n";
947     print OUT "\r\n";
948 }
949
950 sub _generate_dsw_footer($) {
951     local *OUT = shift;
952
953     print OUT "###############################################################################\r\n";
954     print OUT "\r\n";
955     print OUT "Global:\r\n";
956     print OUT "\r\n";
957     print OUT "Package=<5>\r\n";
958     print OUT "{{{\r\n";
959     print OUT "}}}\r\n";
960     print OUT "\r\n";
961     print OUT "Package=<3>\r\n";
962     print OUT "{{{\r\n";
963     print OUT "}}}\r\n";
964     print OUT "\r\n";
965     print OUT "###############################################################################\r\n";
966     print OUT "\r\n";
967 }
968
969 if ($options->wine) {
970     my $dsw_file = "wine.dsw";
971     $output->progress("$dsw_file");
972     replace_file("$wine_dir/$dsw_file", \&_generate_wine_dsw);
973 }
974
975 sub _generate_wine_dsw($) {
976     local *OUT = shift;
977
978     _generate_dsw_header(\*OUT);
979     foreach my $module (sort(keys(%modules))) {
980         next if $module =~ /(?:winetest\.lib|wineruntests\.exe|_test\.exe)$/;
981
982         my $project = $modules{$module}{project};
983         my $dsp_file = $modules{$module}{dsp_file};
984
985         my @dependencies;
986         if($project =~ /^wine(?:_unicode)?$/) {
987             @dependencies = ();
988         } elsif($project =~ /^winebuild$/) {
989             @dependencies = ("wine", "wine_unicode");
990         } elsif($project =~ /^(?:gdi32)_.+?$/) {
991             @dependencies = ();
992         } else {
993             @dependencies = ("wine", "wine_unicode", "winebuild");
994         }
995
996         if($project =~ /^gdi32$/) {
997             foreach my $dir (@gdi32_dirs) {
998                 my $dir2 = $dir;
999                 $dir2 =~ s%^.*?/([^/]+)$%$1%;
1000
1001                 my $module = "gdi32_$dir2";
1002                 $module =~ s%/%_%g;
1003                 push @dependencies, $module;
1004             }
1005         }
1006
1007         _generate_dsw_project(\*OUT, $project, $dsp_file, \@dependencies);
1008     }
1009     _generate_dsw_footer(\*OUT);
1010
1011     return 1;
1012 }
1013
1014 if ($options->winetest) {
1015     my $dsw_file = "winetest.dsw";
1016     $output->progress("$dsw_file");
1017     replace_file("$wine_dir/$dsw_file", \&_generate_winetest_dsw);
1018 }
1019
1020 sub _generate_winetest_dsw($) {
1021     local *OUT = shift;
1022
1023     _generate_dsw_header(\*OUT);
1024
1025     my @runtests_dependencies = ();
1026     foreach my $module (sort(keys(%modules))) {
1027         next if $module !~ /(?:winetest\.lib|wineruntests\.exe|_test\.exe)$/;
1028         next if $module eq "wineruntests";
1029
1030         my $project = $modules{$module}{project};
1031
1032         push @runtests_dependencies, $project;
1033     }
1034
1035     foreach my $module (sort(keys(%modules))) {
1036         next if $module !~ /(?:winetest\.lib|wineruntests\.exe|_test\.exe)$/;
1037
1038         my $project = $modules{$module}{project};
1039         my $dsp_file = $modules{$module}{dsp_file};
1040
1041         my @dependencies;
1042         if($project =~ /^winetest$/) {
1043             @dependencies = ();
1044         } elsif($project =~ /^wineruntests$/) {
1045             @dependencies = @runtests_dependencies;
1046         } else {
1047             @dependencies = ("winetest");
1048         }
1049
1050         _generate_dsw_project(\*OUT, $project, $dsp_file, \@dependencies);
1051     }
1052
1053     _generate_dsw_footer(\*OUT);
1054 }
1055
1056 if ($options->winetest) {
1057     foreach my $module (sort(keys(%modules))) {
1058         next if $module !~ /_test\.exe$/;
1059
1060         my $project = $modules{$module}{project};
1061         my $dsp_file = $modules{$module}{dsp_file};
1062         my @tests = @{$modules{$module}{tests}};
1063
1064         my $testlist_c = $dsp_file;
1065         $testlist_c =~ s%[^/]*\.dsp$%testlist.c%;
1066
1067         replace_file("$wine_dir/$testlist_c", \&_generate_testlist_c, \@tests);
1068     }
1069 }
1070
1071 # ***** Keep in sync with tools/make_ctests *****
1072 sub _generate_testlist_c($$) {
1073     local *OUT = shift;
1074
1075     my @tests = @{(shift)};
1076
1077     print OUT "/* Automatically generated file; DO NOT EDIT!! */\n";
1078     print OUT "\n";
1079     print OUT "/* stdarg.h is needed for Winelib */\n";
1080     print OUT "#include <stdarg.h>\n";
1081     print OUT "#include <stdio.h>\n";
1082     print OUT "#include <stdlib.h>\n";
1083     print OUT "#include \"windef.h\"\n";
1084     print OUT "#include \"winbase.h\"\n";
1085     print OUT "\n";
1086     print OUT "#define STANDALONE\n";
1087     print OUT "#include \"wine/test.h\"\n";
1088     print OUT "\n";
1089     foreach my $test (@tests) {
1090         print OUT "extern void func_$test(void);\n";
1091     }
1092     print OUT "\n";
1093     print OUT "const struct test winetest_testlist[] =\n";
1094     print OUT "{\n";
1095     foreach my $test (@tests) {
1096         print OUT "    { \"$test\", func_$test },\n";
1097     }
1098     print OUT "    { 0, 0 }\n";
1099     print OUT "};\n";
1100 }
1101
1102 if ($options->winetest) {
1103     replace_file("$wine_dir/runtests.c", \&_generate_runtests_c);
1104 }
1105
1106 sub _generate_runtests_c($) {
1107     local *OUT = shift;
1108
1109     print OUT "/* Automatically generated file; DO NOT EDIT!! */\n";
1110
1111     print OUT "\n";
1112     print OUT "#include <stdio.h>\n";
1113     print OUT "#include <stdlib.h>\n";
1114     print OUT "\n";
1115
1116     print OUT "int main(int argc, char *argv[])\n";
1117     print OUT "{\n";
1118     print OUT "    char output_dir[] = __WINETEST_OUTPUT_DIR;\n";
1119     print OUT "    char command[4096];\n";
1120     print OUT "\n";
1121     foreach my $dsp_file (keys(%runtests)) {
1122         my @tests =  @{$runtests{$dsp_file}};
1123
1124         my $project = $dsp_file;
1125         $project =~ s%^(.*?)/?([^/]+)\.dsp$%$2%;
1126         my $dir = $1;
1127         $dir =~ s%/%\\\\%g; 
1128
1129         foreach my $test (@tests) {
1130             print OUT "    sprintf(command, \"$dir\\\\%s\\\\$project.exe $test\", output_dir);\n";
1131             print OUT "    system(command);\n";
1132             print OUT "\n";
1133         }
1134     }
1135     print OUT "    return 0;\n";
1136     print OUT "}\n";
1137 }
1138
1139 if ($options->winetest) {
1140     replace_file("$wine_dir/winetest.c", \&_generate_winetest_c);
1141 }
1142
1143 sub _generate_winetest_c($) {
1144     local *OUT = shift;
1145
1146     print OUT "/* Automatically generated file; DO NOT EDIT!! */\n\n";
1147
1148     print OUT "/* Force the linker to generate a .lib file */\n";
1149     print OUT "void __wine_dummy_lib_function()\n{\n}\n\n";
1150 }
1151
1152 if ($options->wine) {
1153     my $config_h = "include/config.h";
1154
1155     $output->progress("$config_h");
1156
1157     replace_file("$wine_dir/$config_h", \&_generate_config_h);
1158 }
1159
1160 sub _generate_config_h($) {
1161     local *OUT = shift;
1162
1163     print OUT "#define __WINE_CONFIG_H\n";
1164     print OUT "\n";
1165
1166     my @headers = qw(direct.h fcntl.h io.h string.h process.h);
1167     foreach my $header (@headers) {
1168         $header =~ y/\.\//__/;
1169         print OUT "#define HAVE_\U$header\E\n";
1170         print OUT "\n";
1171     }
1172
1173     my @functions = qw(
1174         _alldiv _allmul _allrem _aulldiv _aullrem
1175         _access _chdir _close _lseek _mkdir _open _pclose _popen _read _rmdir _write _stat
1176         _snprintf _spawnvp _stricmp _strnicmp _vsnprintf _wcsicmp
1177         ecvt fcvt gcvt
1178         memmove
1179         strerror
1180         wcslen
1181     );
1182     foreach my $function (@functions) {
1183         print OUT "#define HAVE_\U$function\E 1\n";
1184         print OUT "\n";
1185     }
1186
1187     if(0) {
1188         print OUT "#define NEED_STDCALL_DECORATION 1\n";
1189         print OUT "\n";
1190     }
1191
1192     print OUT "#define X_DISPLAY_MISSING 1\n";
1193     print OUT "\n";
1194
1195     print OUT "/* Define to a macro to generate an assembly function directive */\n";
1196     print OUT "#define __ASM_FUNC(name) \"\"\n";
1197     print OUT "\n";
1198
1199     print OUT "/* Define to a macro to generate an assembly name from a C symbol */\n";
1200     print OUT "#define __ASM_NAME(name) name\n";
1201     print OUT "\n";
1202
1203     print OUT "/* Define to the assembler keyword used to specify a word value */\n";
1204     print OUT "#define __ASM_SHORT \".short\"\n";
1205     print OUT "\n";
1206
1207     print OUT "/* Define to the assembler keyword used to specify an ASCII string */\n";
1208     print OUT "#define __ASM_STRING \".string\"\n";
1209     print OUT "\n";
1210
1211     print OUT "/* Define to the address where bug reports for this package should be sent. */\n";
1212     print OUT "#define PACKAGE_BUGREPORT \"\"\n";
1213     print OUT "\n";
1214
1215     print OUT "/* Define to the full name of this package. */\n";
1216     print OUT "#define PACKAGE_NAME \"Wine\"\n";
1217     print OUT "\n";
1218
1219     print OUT "/* Define to the full name and version of this package. */\n";
1220     print OUT "#define PACKAGE_STRING \"Wine YYYYMMDD\"\n";
1221     print OUT "\n";
1222
1223     print OUT "/* Define to the one symbol short name of this package. */\n";
1224     print OUT "#define PACKAGE_TARNAME \"wine\"\n";
1225     print OUT "\n";
1226
1227     print OUT "/* Define to the version of this package. */\n";
1228     print OUT "#define PACKAGE_VERSION \"YYYYMMDD\"\n";
1229     print OUT "\n";
1230
1231     close(OUT);
1232 }