#!/usr/bin/perl -w # Copyright 2001 Patrik Stridvall use strict; BEGIN { $0 =~ m%^(.*?/?tools)/winapi/winapi_fixup$%; require "$1/winapi/setup.pm"; } use config qw( &file_type &files_filter &file_skip &files_skip &file_normalize &get_spec_files $current_dir $wine_dir $winapi_dir $winapi_check_dir ); use output qw($output); use winapi_fixup_options qw($options); use modules qw($modules); use winapi qw($win16api $win32api @winapis); use type; use util; use winapi_function; use winapi_parser; my @c_files = $options->c_files; @c_files = files_skip(@c_files); @c_files = files_filter("winelib", @c_files); my $progress_output; my $progress_current = 0; my $progress_max = scalar(@c_files); foreach my $file (@c_files) { my %insert_line; my %substitute_line; my %delete_line; my %spec_file; $progress_current++; if($options->progress) { $output->progress("$file: file $progress_current of $progress_max"); } my %documentation_line_used; my $create_function = sub { return 'winapi_function'->new; }; my $found_function = sub { my $function = shift; my $documentation_line = $function->documentation_line; my $documentation = $function->documentation; my $function_line = $function->function_line; my $linkage = $function->linkage; my $return_type = $function->return_type; my $calling_convention = $function->calling_convention; my $internal_name = $function->internal_name; my $statements = $function->statements; if($linkage eq "static" || ($linkage eq "extern" && !defined($statements)) || ($linkage eq "" && !defined($statements))) { return; } my @external_names = $function->external_names; if($#external_names < 0) { return; } if($documentation_line_used{$documentation_line}) { $documentation = undef; } $documentation_line_used{$documentation_line}++; $output->prefix_callback(sub { return $function->prefix; }); my @module_ordinal_entries = (); foreach my $entry2 ($function->get_all_module_ordinal) { (my $external_name2, my $module2, my $ordinal2) = @$entry2; if(($external_name2 eq "@" || ($win16api->is_module($module2) && !$win16api->is_function_stub_in_module($module2, $external_name2)) || ($win32api->is_module($module2) && !$win32api->is_function_stub_in_module($module2, $external_name2))) && $modules->is_allowed_module_in_file($module2, "$current_dir/$file")) { push @module_ordinal_entries, $entry2; } } my $spec_modified = 0; if($options->stub && defined($documentation)) { my $calling_convention16 = $function->calling_convention16; my $calling_convention32 = $function->calling_convention32; foreach my $winapi (@winapis) { my @entries = (); my $module = $winapi->function_internal_module($internal_name); my $ordinal = $winapi->function_internal_ordinal($internal_name); if($winapi->is_function_stub_in_module($module, $internal_name)) { my $external_name = $internal_name; if($winapi->name eq "win16") { $external_name =~ s/(?:_)?16([AW]?)$//; if(defined($1)) { $external_name .= $1; } } push @entries, [$external_name, $module, $ordinal]; } foreach (split(/\n/, $documentation)) { if(/^\s*\*\s*(\S+)\s*[\(\[]\s*(\w+)\s*\.\s*([^\s\)\]]*)\s*[\)\]].*?$/) { my $external_name = $1; my $module = lc($2); my $ordinal = $3; if($external_name ne "@" && $winapi->is_module($module) && $winapi->is_function_stub_in_module($module, $external_name) && $internal_name !~ /^\U$module\E_\Q$external_name\E$/) { push @entries, [$external_name, $module, $ordinal]; } } } foreach my $entry (@entries) { (my $external_name, my $module, my $ordinal) = @$entry; my $refargument_types = $function->argument_types; if(!defined($refargument_types)) { next; } my $abort = 0; my $n; my @argument_kinds = map { my $type = $_; my $kind; if($type ne "..." && !defined($kind = $winapi->translate_argument($type))) { $output->write("no translation defined: " . $type . "\n"); } # FIXME: Kludge if(defined($kind) && $kind eq "longlong") { $n += 2; ("long", "long"); } elsif(defined($kind)) { $n++; $kind; } elsif($type eq "...") { if($winapi->name eq "win16") { $calling_convention16 = "pascal"; # FIXME: Is this correct? } else { $calling_convention32 = "varargs"; } (); } else { $abort = 1; $n++; "undef"; } } @$refargument_types; my $substitute = {}; $substitute->{search} = "^\\s*$ordinal\\s+stub\\s+$external_name\\s*(?:#.*?)?\$"; if($winapi->name eq "win16") { $substitute->{replace} = "$ordinal $calling_convention16 $external_name(@argument_kinds) $internal_name"; } else { $substitute->{replace} = "$ordinal $calling_convention32 $external_name(@argument_kinds) $internal_name"; } if(!defined($spec_file{$module})) { $spec_file{$module} = []; } if(!$abort) { $spec_modified = 1; push @{$spec_file{$module}}, $substitute; } } } } my %found_external_names; foreach my $external_name (@external_names) { $found_external_names{$external_name} = {}; } my $documentation_modified = 0; if(!$spec_modified && (defined($documentation) && !$documentation_modified) && ($options->documentation_name || $options->documentation_ordinal || $options->documentation_missing)) { local $_; my $line3; my $search; my $replace; my $count = 0; my $line2 = $documentation_line - 1; foreach (split(/\n/, $documentation)) { $line2++; if(/^(\s*\*\s*(\S+)\s*)((?:\s*[\(\[]\s*\w+(?:\s*\.\s*[^\s\)\]]*\s*)?[\)\]])+)(.*?)$/) { my $part1 = $1; my $external_name = $2; my $part3 = $3; my $part4 = $4; $part4 =~ s/\s*$//; my @entries = (); while($part3 =~ s/^\s*([\(\[]\s*(\w+)(?:\s*\.\s*([^\s\)\]]*)\s*)?[\)\]])//) { push @entries, [$1, $2, $3]; } my $found = 0; foreach my $external_name2 (@external_names) { if($external_name eq $external_name2) { foreach my $entry (@entries) { (undef, my $module, undef) = @$entry; $found_external_names{$external_name2}{$module} = 1; } $found = 1; last; } } my $replaced = 0; my $replace2 = ""; foreach my $entry (@entries) { my $part12 = $part1; (my $part32, my $module, my $ordinal) = @$entry; foreach my $entry2 (@module_ordinal_entries) { (my $external_name2, my $module2, my $ordinal2) = @$entry2; if($options->documentation_name && lc($module) eq $module2 && $external_name ne $external_name2) { if(!$found && $part12 =~ s/\b\Q$external_name\E\b/$external_name2/) { $external_name = $external_name2; $replaced++; } } if($options->documentation_ordinal && $external_name eq $external_name2 && lc($module) eq $module2 && ($#entries > 0 || !defined($ordinal) || ($ordinal ne $ordinal2))) { if(defined($ordinal)) { if($part32 =~ s/\Q$module\E\s*.\s*\Q$ordinal\E/\U$module2\E.$ordinal2/ || $#entries > 0) { $replaced++; } } else { if($part32 =~ s/\Q$module\E/\U$module2\E.$ordinal2/ || $#entries > 0) { $replaced++; } } } } if($replace2) { $replace2 .= "\n"; } $replace2 .= "$part12$part32$part4"; } if($replaced > 0) { $line3 = $line2; $search = "^\Q$_\E\$"; $replace = $replace2; } $count++; } elsif(/^(\s*\*\s*)([^\s\(]+)(?:\(\))?\s*$/) { my $part1 = $1; my $external_name = $2; if($internal_name =~ /^(?:\S+_)?\Q$external_name\E(?:16)?$/) { foreach my $entry (@module_ordinal_entries) { (my $external_name2, my $module, my $ordinal) = @$entry; $line3 = $line2; $search = "^\Q$_\E\$"; $replace = "$part1$external_name2 (\U$module\E.$ordinal)"; } $count++; } } } if(defined($line3) && defined($search) && defined($replace)) { if($count > 1 || $#external_names >= 1) { $output->write("multiple entries (fixup not supported)\n"); # $output->write("s/$search/$replace/\n"); # $output->write("@external_names\n"); } else { $documentation_modified = 1; $substitute_line{$line3}{search} = $search; $substitute_line{$line3}{replace} = $replace; } } } if(!$spec_modified && !$documentation_modified && $options->documentation_missing && defined($documentation)) { my $part1; my $part2; my $part3; my $part4; my $line3 = 0; my $line2 = $documentation_line - 1; foreach (split(/\n/, $documentation)) { $line2++; if(/^(\s*\*\s*)(\S+\s*)([\(\[])\s*\w+\s*\.\s*[^\s\)\]]*\s*([\)\]]).*?$/) { $part1 = $1; $part2 = $2; $part3 = $3; $part4 = $4; $part2 =~ s/\S/ /g; $line3 = $line2 + 1; } } foreach my $entry2 (@module_ordinal_entries) { (my $external_name2, my $module2, my $ordinal2) = @$entry2; my $found = 0; foreach my $external_name (keys(%found_external_names)) { foreach my $module3 (keys(%{$found_external_names{$external_name}})) { if($external_name eq $external_name2 && uc($module2) eq $module3) { $found = 1; } } } # FIXME: Not 100% correct if(!$found && !$win16api->is_function_stub_in_module($module2, $internal_name) && !$win32api->is_function_stub_in_module($module2, $internal_name)) { if($line3 > 0) { $documentation_modified = 1; $part2 = $external_name2 . " " x (length($part2) - length($external_name2)); $insert_line{$line3} = "$part1$part2$part3\U$module2\E.$ordinal2$part4\n"; } else { $output->write("$external_name2 (\U$module2\E.$ordinal2) missing (fixup not supported)\n"); } } } } if(!$documentation_modified && defined($documentation) && $options->documentation_wrong) { my $line2 = $documentation_line - 1; foreach (split(/\n/, $documentation)) { $line2++; if(/^\s*\*\s*(\S+)\s*[\(\[]\s*(\w+)\s*\.\s*([^\s\)\]]*)\s*[\)\]].*?$/) { my $external_name = $1; my $module = $2; my $ordinal = $3; my $found = 0; foreach my $entry2 (@module_ordinal_entries) { (my $external_name2, my $module2, my $ordinal2) = @$entry2; if($external_name eq $external_name2 && lc($module) eq $module2 && $ordinal eq $ordinal2) { $found = 1; } } if(!$found) { if(1) { $documentation_modified = 1; $delete_line{$line2} = "^\Q$_\E\$"; } else { $output->write("$external_name (\U$module\E.$ordinal) wrong (fixup not supported)\n"); }; } } } } if(!$spec_modified && !$documentation_modified && !defined($documentation)) { my $insert = ""; foreach my $winapi (@winapis) { my $external_name = $winapi->function_external_name($internal_name); my $module = $winapi->function_internal_module($internal_name); my $ordinal = $winapi->function_internal_ordinal($internal_name); if(defined($external_name) && defined($module) && defined($ordinal)) { $insert .= " *\t\t$external_name (\U$module\E.$ordinal)\n"; } } if($insert) { $insert_line{$function_line} = "/" . "*" x 71 . "\n" . "$insert" . " */\n"; } } $output->prefix(""); }; my $create_type = sub { return 'type'->new; }; my $found_type = sub { my $type = shift; }; my $found_preprocessor = sub { my $directive = shift; my $argument = shift; }; &winapi_parser::parse_c_file($file, $create_function, $found_function, $create_type, $found_type, $found_preprocessor); my $editor = sub { local *IN = shift; local *OUT = shift; my $modified = 0; while() { chomp; my $line; $line = $insert_line{$.}; if(defined($line)) { if($options->modify) { $_ = "$line$_"; $modified = 1; } else { my $line2 = $line; chomp($line2); my @line2 = split(/\n/, $line2); if($#line2 > 0) { $output->write("$file: $.: insert: \\\n"); foreach my $line2 (@line2) { $output->write("'$line2'\n"); } } else { $output->write("$file: $.: insert: '$line2'\n"); } } } my $search = $substitute_line{$.}{search}; my $replace = $substitute_line{$.}{replace}; if(defined($search) && defined($replace)) { my $modified2 = 0; if(s/$search/$replace/) { if($options->modify) { $modified = 1; } $modified2 = 1; } if(!$options->modify || !$modified2) { my $search2; my $replace2; if(!$modified2) { $search2 = "unmatched search"; $replace2 = "unmatched replace"; } else { $search2 = "search"; $replace2 = "replace"; } $output->write("$file: $.: $search2 : '$search'\n"); my @replace2 = split(/\n/, $replace); if($#replace2 > 0) { $output->write("$file: $.: $replace2: \\\n"); foreach my $replace2 (@replace2) { $output->write("'$replace2'\n"); } } else { $output->write("$file: $.: $replace2: '$replace'\n"); } } } $line = $delete_line{$.}; if(defined($line)) { if(/$line/) { if($options->modify) { $modified = 1; next; } else { $output->write("$file: $.: delete: '$line'\n"); } } else { $output->write("$file: $.: unmatched delete: '$line'\n"); } } print OUT "$_\n"; } return $modified; }; my $n = 0; while(defined(each %insert_line)) { $n++; } while(defined(each %substitute_line)) { $n++; } while(defined(each %delete_line)) { $n++; } if($n > 0) { edit_file($file, $editor); } foreach my $module (sort(keys(%spec_file))) { my $file; foreach my $winapi (@winapis) { $file = ($winapi->module_file($module) || $file); } if(defined($file)) { $file = file_normalize($file); } my @substitutes = @{$spec_file{$module}}; my $editor = sub { local *IN = shift; local *OUT = shift; my $modified = 0; while() { chomp; my @substitutes2 = (); foreach my $substitute (@substitutes) { my $search = $substitute->{search}; my $replace = $substitute->{replace}; if(s/$search/$replace/) { if($options->modify) { $modified = 1; } else { $output->write("$file: search : '$search'\n"); $output->write("$file: replace: '$replace'\n"); } next; } else { push @substitutes2, $substitute; } } @substitutes = @substitutes2; print OUT "$_\n"; } return $modified; }; if(defined($file)) { edit_file($file, $editor); } else { $output->write("$module: doesn't have any spec file\n"); } if($#substitutes >= 0) { foreach my $substitute (@substitutes) { my $search = $substitute->{search}; my $replace = $substitute->{replace}; $output->write("$file: unmatched search : '$search'\n"); $output->write("$file: unmatched replace: '$replace'\n"); } } } }