netstat: Parse command line arguments.
[wine] / tools / make_specfiles
1 #!/usr/bin/perl -w
2 #
3 # Update spec files across dlls that share an implementation
4 #
5 # Copyright 2011 Alexandre Julliard
6 #
7 # This library is free software; you can redistribute it and/or
8 # modify it under the terms of the GNU Lesser General Public
9 # License as published by the Free Software Foundation; either
10 # version 2.1 of the License, or (at your option) any later version.
11 #
12 # This library is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15 # Lesser General Public License for more details.
16 #
17 # You should have received a copy of the GNU Lesser General Public
18 # License along with this library; if not, write to the Free Software
19 # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20 #
21
22 use strict;
23
24 my %funcs;
25 my $group_head;
26
27 my @dll_groups =
28 (
29  [
30   "msvcrt",
31   "msvcr100",
32   "msvcirt",
33   "msvcr90",
34   "msvcr80",
35   "msvcr71",
36   "msvcr70",
37   "msvcrt40",
38   "msvcrt20",
39   "msvcrtd",
40   "crtdll",
41  ],
42  [
43   "msvcrt",
44   "msvcp90",
45   "msvcp100",
46   "msvcp80",
47   "msvcp71",
48   "msvcp70",
49   "msvcp60",
50  ],
51  [
52   "d3dx9_36",
53   "d3dx9_43",
54   "d3dx9_42",
55   "d3dx9_41",
56   "d3dx9_40",
57   "d3dx9_39",
58   "d3dx9_38",
59   "d3dx9_37",
60   "d3dx9_35",
61   "d3dx9_34",
62   "d3dx9_33",
63   "d3dx9_32",
64   "d3dx9_31",
65   "d3dx9_30",
66   "d3dx9_29",
67   "d3dx9_28",
68   "d3dx9_27",
69   "d3dx9_26",
70   "d3dx9_25",
71   "d3dx9_24",
72  ],
73  [
74   "d3dx10_43",
75   "d3dx10_42",
76   "d3dx10_41",
77   "d3dx10_40",
78   "d3dx10_39",
79   "d3dx10_38",
80   "d3dx10_37",
81   "d3dx10_36",
82   "d3dx10_35",
83   "d3dx10_34",
84   "d3dx10_33",
85  ],
86  [
87   "d3dcompiler_43",
88   "d3dcompiler_42",
89   "d3dcompiler_41",
90   "d3dcompiler_40",
91   "d3dcompiler_39",
92   "d3dcompiler_38",
93   "d3dcompiler_37",
94   "d3dcompiler_36",
95   "d3dcompiler_35",
96   "d3dcompiler_34",
97   "d3dcompiler_33",
98  ],
99  [
100   "xinput1_3",
101   "xinput1_2",
102   "xinput1_1",
103   "xinput9_1_0",
104  ],
105  [
106   "setupapi",
107   "cfgmgr32",
108  ],
109  [
110   "vcomp",
111   "vcomp100",
112   "vcomp90",
113  ],
114  [
115   "atl100",
116   "atl",
117   "atl80",
118  ],
119  [
120   "advapi32",
121   "api-ms-win-downlevel-advapi32-l1-1-0",
122   "api-ms-win-downlevel-advapi32-l2-1-0",
123  ],
124  [
125   "kernel32",
126   "api-ms-win-downlevel-normaliz-l1-1-0",
127  ],
128  [
129   "ole32",
130   "api-ms-win-downlevel-ole32-l1-1-0",
131  ],
132  [
133   "shell32",
134   "api-ms-win-downlevel-shell32-l1-1-0",
135  ],
136  [
137   "shlwapi",
138   "api-ms-win-downlevel-shlwapi-l1-1-0",
139   "api-ms-win-downlevel-shlwapi-l2-1-0",
140  ],
141  [
142   "user32",
143   "api-ms-win-downlevel-user32-l1-1-0",
144  ],
145  [
146   "version",
147   "api-ms-win-downlevel-version-l1-1-0",
148  ],
149 );
150
151 my $update_flags = 0;
152 my $show_duplicates = 0;
153
154 foreach my $arg (@ARGV)
155 {
156     if ($arg eq "-f") { $update_flags = 1; }
157     elsif ($arg eq "-d") { $show_duplicates = 1; }
158 }
159
160 sub update_file($)
161 {
162     my $file = shift;
163     my $ret = !(-f $file) || system "cmp $file $file.new >/dev/null";
164     if (!$ret)
165     {
166         unlink "$file.new";
167     }
168     else
169     {
170         #system "diff -u $file $file.new";
171         rename "$file.new", "$file";
172         print "$file updated\n";
173     }
174     return $ret;
175 }
176
177 # parse a spec file line
178 sub parse_line($$$)
179 {
180     my ($name, $line, $_) = @_;
181
182     if (/^\s*(\@|\d+)\s+(stdcall|cdecl|varargs|thiscall|stub|extern)\s+((?:-\S+\s+)*)([A-Za-z0-9_\@\$?]+)(?:\s*(\([^)]*\)))?(?:\s+([A-Za-z0-9_\@\$?.]+))?(\s*\#.*)?/)
183     {
184         return ( "ordinal" => $1, "callconv" => $2, "flags" => $3, "name" => $4, "args" => $5 || "",
185                  "target" => $6 || $4, "comment" => $7, "spec" => $name );
186     }
187     return () if /^\s*$/;
188     return () if /^\s*\#/;
189     printf STDERR "$name.spec:$line: error: Unrecognized line $_\n";
190 }
191
192 sub read_spec_file($)
193 {
194     my $name = shift;
195     my $file = "dlls/$name/$name.spec";
196     my %stubs;
197     open SPEC, "<$file" or die "cannot open $file";
198     while (<SPEC>)
199     {
200         chomp;
201         my %descr = parse_line( $name, $., $_ );
202         next unless %descr;
203
204         my $func = $descr{name};
205         next if defined $funcs{$func};
206         $funcs{$func} = \%descr;
207     }
208     close SPEC;
209 }
210
211 sub update_spec_file($)
212 {
213     my $name = shift;
214     my $file = "dlls/$name/$name.spec";
215     my %stubs;
216
217     open SPEC, "<$file" or die "cannot open $file";
218     open NEW, ">$file.new" or die "cannot create $file.new";
219     while (<SPEC>)
220     {
221         chomp;
222
223         my $commented_out = 0;
224         my %descr = parse_line( $name, $., $_ );
225         if (!%descr)
226         {
227             # check for commented out exports
228             if (/^\s*\#\s*((?:\@|\d+)\s+)?((?:extern|stub|stdcall|cdecl|varargs|thiscall)\s+.*)/)
229             {
230                 $commented_out = 1;
231                 %descr = parse_line( $name, $., ($1 || "\@ ") . $2 );
232             }
233         }
234         goto done unless %descr;
235
236         my $func = $descr{name};
237         if (!defined $funcs{$func})
238         {
239             $funcs{$func} = \%descr unless $commented_out;
240             goto done;
241         }
242
243         my %parent = %{$funcs{$func}};
244         goto done if $parent{spec} eq $descr{spec};  # the definition is in this spec file
245         goto done if $descr{comment} && $descr{comment} =~ /don't forward/;
246         if ($descr{callconv} ne "stub" && $descr{target} !~ /\./ && !$commented_out)
247         {
248             printf "%s:%u: note: %s already defined in %s\n", $file, $., $func, $parent{spec} if $show_duplicates;
249             goto done;
250         }
251
252         my $flags = $descr{flags};
253         if ($parent{callconv} ne "stub" || $update_flags)
254         {
255             $flags = $parent{flags};
256             $flags =~ s/-ordinal\s*// if $descr{ordinal} eq "@";
257         }
258
259         if ($parent{callconv} ne "stub" || $parent{args})
260         {
261             my $callconv = $parent{callconv} ne "stub" ? $parent{callconv} :
262                            $parent{spec} =~ /msvc/ ? "cdecl" : "stdcall";  # hack
263             $_ = sprintf "$descr{ordinal} %s %s%s", $callconv, $flags, $func;
264
265             if ($parent{target} =~ /$group_head\./)  # use the same forward as parent if possible
266             {
267                 $_ .= sprintf "%s %s", $parent{args}, $parent{target};
268             }
269             else
270             {
271                 $_ .= sprintf "%s %s.%s", $parent{args}, $parent{spec}, $func;
272             }
273         }
274         else
275         {
276             $_ = sprintf "$descr{ordinal} stub %s%s", $flags, $func;
277         }
278         $_ .= $descr{comment} || "";
279
280       done:
281         print NEW "$_\n";
282     }
283     close SPEC;
284     close NEW;
285     update_file( $file );
286 }
287
288 sub sync_spec_files(@)
289 {
290     %funcs = ();
291     $group_head = shift;
292     read_spec_file( $group_head );
293     foreach my $spec (@_) { update_spec_file($spec); }
294 }
295
296 foreach my $group (@dll_groups)
297 {
298     sync_spec_files( @{$group} );
299 }