Release 8.6.
[wine.git] / tools / make_specfiles
blob3c4c40544b8e8ce1d767d8ec85f7e3afb8f2db11
1 #!/usr/bin/perl -w
3 # Update spec files across dlls that share an implementation
5 # Copyright 2011 Alexandre Julliard
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.
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.
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
22 use strict;
24 my %funcs;
25 my $group_head;
27 my @dll_groups =
30 "msvcrt",
31 "msvcirt",
32 "msvcrt40",
33 "msvcrt20",
36 "msvcrt",
37 "msvcp90",
38 "msvcp100",
39 "msvcp110",
40 "msvcp120",
41 "msvcp140",
42 "msvcp71",
43 "msvcp80",
44 "msvcp70",
45 "msvcp60",
48 "msvcr120",
49 "msvcr120_app",
50 "concrt140",
53 "ucrtbase",
54 "vcruntime140",
57 "msvcp120",
58 "msvcp120_app",
61 "msvcp140",
62 "msvcp_win",
65 "d3d10",
66 "d3d10_1",
69 "d3dx10_43",
70 "d3dx10_42",
71 "d3dx10_41",
72 "d3dx10_40",
73 "d3dx10_39",
74 "d3dx10_38",
75 "d3dx10_37",
76 "d3dx10_36",
77 "d3dx10_35",
78 "d3dx10_34",
79 "d3dx10_33",
82 "xinput1_3",
83 "xinput1_4",
84 "xinput1_2",
85 "xinput1_1",
86 "xinput9_1_0",
89 "vcomp",
90 "vcomp140",
91 "vcomp120",
92 "vcomp110",
93 "vcomp100",
94 "vcomp90",
97 "advapi32",
98 "sechost",
101 "netapi32",
102 "srvcli",
105 "ole32",
106 "iprop",
109 "secur32",
110 "security",
111 "sspicli",
114 "gdi32",
115 "usp10"
118 "bthprops.cpl",
119 "irprops.cpl",
122 "sfc_os",
123 "sfc",
126 "bcrypt",
127 "ncrypt",
128 "cng.sys",
131 "ntoskrnl.exe",
132 "hal",
135 "mscoree",
136 "mscorwks",
139 "sppc",
140 "slc",
144 my $update_flags = 0;
145 my $show_duplicates = 0;
147 foreach my $arg (@ARGV)
149 if ($arg eq "-f") { $update_flags = 1; }
150 elsif ($arg eq "-d") { $show_duplicates = 1; }
153 # update a file if changed
154 sub update_file($$)
156 my $file = shift;
157 my $new = shift;
159 open FILE, ">$file.new" or die "cannot create $file.new";
160 print FILE $new;
161 close FILE;
162 rename "$file.new", "$file";
163 print "$file updated\n";
166 # parse a spec file line
167 sub parse_line($$$)
169 my ($name, $line, $str) = @_;
171 if ($str =~ /^\s*(\@|\d+)\s+(stdcall|cdecl|varargs|thiscall|stub|extern)\s+((?:-\S+\s+)*)([A-Za-z0-9_\@\$?]+)(?:\s*(\([^)]*\)))?(?:\s+([A-Za-z0-9_\@\$?.]+))?(\s*\#.*)?/)
173 return ( "ordinal" => $1, "callconv" => $2, "flags" => $3, "name" => $4, "args" => $5 || "",
174 "target" => $6 || $4, "comment" => $7, "spec" => $name );
176 return () if $str =~ /^\s*$/;
177 return () if $str =~ /^\s*\#/;
178 printf STDERR "$name.spec:$line: error: Unrecognized line $_\n";
181 sub read_spec_file($)
183 my $name = shift;
184 my $file = "dlls/$name/$name.spec";
185 my %stubs;
186 open SPEC, "<$file" or die "cannot open $file";
187 while (<SPEC>)
189 chomp;
190 my %descr = parse_line( $name, $., $_ );
191 next unless %descr;
193 my $func = $descr{name};
194 if (defined $funcs{$func})
196 my %update = %{$funcs{$func}};
197 next if $update{ordinal} ne $descr{ordinal} or $update{callconv} ne $descr{callconv} or $update{args} ne $descr{args};
199 my $arch = $1 if $update{flags} =~ /-arch=(\S+)/;
200 my $new_arch = $1 if $descr{flags} =~ /-arch=(\S+)/;
201 next if !defined $arch or !defined $new_arch;
203 if (($arch eq "win32" and $new_arch eq "win64") or ($arch eq "win64" and $new_arch eq "win32"))
205 $funcs{$func}{flags} =~ s/-arch=\S+\s+//;
206 next;
209 $funcs{$func}{flags} =~ s/-arch=$arch/-arch=$arch,$new_arch/;
210 next;
212 next if $func eq "@";
213 $funcs{$func} = \%descr;
215 close SPEC;
218 sub update_spec_file($)
220 my $name = shift;
221 my $file = "dlls/$name/$name.spec";
222 my %stubs;
223 my ($old, $new);
225 open SPEC, "<$file" or die "cannot open $file";
226 while (<SPEC>)
228 $old .= $_;
229 chomp;
231 my $commented_out = 0;
232 my %descr = parse_line( $name, $., $_ );
233 if (!%descr)
235 # check for commented out exports
236 if (/^\s*\#\s*((?:\@|\d+)\s+)?((?:extern|stub|stdcall|cdecl|varargs|thiscall)\s+.*)/)
238 $commented_out = 1;
239 %descr = parse_line( $name, $., ($1 || "\@ ") . $2 );
242 goto done unless %descr;
244 my $func = $descr{name};
245 if (!defined $funcs{$func})
247 $funcs{$func} = \%descr unless $commented_out || $name =~ /-/;
248 goto done;
251 my %parent = %{$funcs{$func}};
252 goto done if $parent{spec} eq $descr{spec}; # the definition is in this spec file
253 goto done if $descr{comment} && $descr{comment} =~ /don't forward/;
254 if ($descr{callconv} ne "stub" && $descr{target} !~ /\./ && !$commented_out)
256 printf "%s:%u: note: %s already defined in %s\n", $file, $., $func, $parent{spec} if $show_duplicates;
257 goto done;
260 my $flags = $descr{flags};
261 if ($parent{callconv} ne "stub" || $update_flags)
263 $flags = $parent{flags};
264 $flags =~ s/-ordinal\s*// if $descr{ordinal} eq "@";
265 $flags =~ s/-noname\s*// if $descr{ordinal} eq "@";
266 $flags =~ s/-import\s*//;
267 if ($descr{flags} =~ /-private/) # preserve -private flag
269 $flags = "-private " . $flags unless $flags =~ /-private/;
273 if ($parent{callconv} ne "stub" || $parent{args})
275 my $callconv = $parent{callconv} ne "stub" ? $parent{callconv} :
276 $parent{spec} =~ /(msvc|ucrtbase)/ ? "cdecl" : "stdcall"; # hack
277 $_ = sprintf "$descr{ordinal} %s %s%s", $callconv, $flags, $func;
279 if ($parent{target} =~ /$group_head\./) # use the same forward as parent if possible
281 $_ .= sprintf "%s %s", $parent{args}, $parent{target};
283 else
285 $_ .= sprintf "%s %s.%s", $parent{args}, $parent{spec}, $func;
288 else
290 $_ = sprintf "$descr{ordinal} stub %s%s", $flags, $func;
292 $_ .= $descr{comment} || "";
294 done:
295 $new .= "$_\n";
297 close SPEC;
298 update_file( $file, $new ) if $old ne $new;
301 sub sync_spec_files(@)
303 %funcs = ();
304 $group_head = shift;
305 read_spec_file( $group_head );
306 foreach my $spec (@_) { update_spec_file($spec); }
309 foreach my $group (@dll_groups)
311 sync_spec_files( @{$group} );