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
121 my $update_flags = 0;
122 my $show_duplicates = 0;
124 foreach my $arg (@ARGV)
126 if ($arg eq "-f") { $update_flags = 1; }
127 elsif ($arg eq "-d") { $show_duplicates = 1; }
133 my $ret = !(-f $file) || system "cmp $file $file.new >/dev/null";
140 #system "diff -u $file $file.new";
141 rename "$file.new", "$file";
142 print "$file updated\n";
147 # parse a spec file line
150 my ($name, $line, $_) = @_;
152 if (/^\s*(\@|\d+)\s+(stdcall|cdecl|varargs|thiscall|stub|extern)\s+((?:-\S+\s+)*)([A-Za-z0-9_\@\$?]+)(?:\s*(\([^)]*\)))?(?:\s+([A-Za-z0-9_\@\$?.]+))?(\s*\#.*)?/)
154 return ( "ordinal" => $1, "callconv" => $2, "flags" => $3, "name" => $4, "args" => $5 || "",
155 "target" => $6 || $4, "comment" => $7, "spec" => $name );
157 return () if /^\s*$/;
158 return () if /^\s*\#/;
159 printf STDERR "$name.spec:$line: error: Unrecognized line $_\n";
162 sub read_spec_file($)
165 my $file = "dlls/$name/$name.spec";
167 open SPEC, "<$file" or die "cannot open $file";
171 my %descr = parse_line( $name, $., $_ );
174 my $func = $descr{name};
175 next if defined $funcs{$func};
176 $funcs{$func} = \%descr;
181 sub update_spec_file($)
184 my $file = "dlls/$name/$name.spec";
187 open SPEC, "<$file" or die "cannot open $file";
188 open NEW, ">$file.new" or die "cannot create $file.new";
193 my $commented_out = 0;
194 my %descr = parse_line( $name, $., $_ );
197 # check for commented out exports
198 if (/^\s*\#\s*((?:\@|\d+)\s+)?((?:extern|stub|stdcall|cdecl|varargs|thiscall)\s+.*)/)
201 %descr = parse_line( $name, $., ($1 || "\@ ") . $2 );
204 goto done unless %descr;
206 my $func = $descr{name};
207 if (!defined $funcs{$func})
209 $funcs{$func} = \%descr unless $commented_out;
213 my %parent = %{$funcs{$func}};
214 goto done if $parent{spec} eq $descr{spec}; # the definition is in this spec file
215 goto done if $descr{comment} && $descr{comment} =~ /don't forward/;
216 if ($descr{callconv} ne "stub" && $descr{target} !~ /\./ && !$commented_out)
218 printf "%s:%u: note: %s already defined in %s\n", $file, $., $func, $parent{spec} if $show_duplicates;
222 my $flags = ($parent{callconv} ne "stub" || $update_flags) ? $parent{flags} : $descr{flags};
224 if ($parent{callconv} ne "stub" || $parent{args})
226 my $callconv = $parent{callconv} ne "stub" ? $parent{callconv} :
227 $parent{spec} =~ /msvc/ ? "cdecl" : "stdcall"; # hack
228 $_ = sprintf "$descr{ordinal} %s %s%s", $callconv, $flags, $func;
230 if ($parent{target} =~ /$group_head\./) # use the same forward as parent if possible
232 $_ .= sprintf "%s %s", $parent{args}, $parent{target};
236 $_ .= sprintf "%s %s.%s", $parent{args}, $parent{spec}, $func;
241 $_ = sprintf "$descr{ordinal} stub %s%s", $flags, $func;
243 $_ .= $descr{comment} || "";
250 update_file( $file );
253 sub sync_spec_files(@)
257 read_spec_file( $group_head );
258 foreach my $spec (@_) { update_spec_file($spec); }
261 foreach my $group (@dll_groups)
263 sync_spec_files( @{$group} );