Commit bda50740 authored by Patrik Stridvall's avatar Patrik Stridvall Committed by Alexandre Julliard

Several additions and bug fixes.

parent ff338b27
...@@ -23,6 +23,9 @@ package _output; ...@@ -23,6 +23,9 @@ package _output;
use strict; use strict;
my $stdout_isatty = -t STDOUT;
my $stderr_isatty = -t STDERR;
sub new { sub new {
my $proto = shift; my $proto = shift;
my $class = ref($proto) || $proto; my $class = ref($proto) || $proto;
...@@ -51,7 +54,7 @@ sub show_progress { ...@@ -51,7 +54,7 @@ sub show_progress {
$$progress_count++; $$progress_count++;
if($$progress_count > 0 && $$progress) { if($$progress_count > 0 && $$progress && $stderr_isatty) {
print STDERR $$progress; print STDERR $$progress;
$$last_progress = $$progress; $$last_progress = $$progress;
} }
...@@ -65,7 +68,7 @@ sub hide_progress { ...@@ -65,7 +68,7 @@ sub hide_progress {
$$progress_count--; $$progress_count--;
if($$last_progress) { if($$last_progress && $stderr_isatty) {
my $message; my $message;
for (1..length($$last_progress)) { for (1..length($$last_progress)) {
$message .= " "; $message .= " ";
...@@ -124,9 +127,9 @@ sub write { ...@@ -124,9 +127,9 @@ sub write {
my $prefix = \${$self->{PREFIX}}; my $prefix = \${$self->{PREFIX}};
$self->hide_progress; $self->hide_progress if $stdout_isatty;
print $$prefix . $message; print $$prefix . $message;
$self->show_progress; $self->show_progress if $stdout_isatty;
} }
1; 1;
...@@ -14,6 +14,7 @@ use config qw( ...@@ -14,6 +14,7 @@ use config qw(
$current_dir $wine_dir $winapi_dir $winapi_check_dir $current_dir $wine_dir $winapi_dir $winapi_check_dir
); );
use modules; use modules;
use nativeapi;
use output; use output;
use options; use options;
use winapi; use winapi;
...@@ -89,6 +90,9 @@ if($wine_dir eq ".") { ...@@ -89,6 +90,9 @@ if($wine_dir eq ".") {
'winapi'->read_spec_files($modules, $wine_dir, $current_dir, \@spec_files, $win16api, $win32api); 'winapi'->read_spec_files($modules, $wine_dir, $current_dir, \@spec_files, $win16api, $win32api);
} }
my $nativeapi = 'nativeapi'->new($options, $output, "$winapi_check_dir/nativeapi.dat",
"$wine_dir/configure.in", "$wine_dir/include/config.h.in");
my %specifications; my %specifications;
sub documentation_specifications { sub documentation_specifications {
......
...@@ -106,7 +106,7 @@ foreach my $file (@c_files) { ...@@ -106,7 +106,7 @@ foreach my $file (@c_files) {
my @argument_documentations = @{$function->argument_documentations}; my @argument_documentations = @{$function->argument_documentations};
my $statements = $function->statements; my $statements = $function->statements;
if($linkage eq "static" || $linkage eq "extern" || !defined($statements)) { if($linkage eq "static" || ($linkage eq "" && !defined($statements))) {
return; return;
} }
...@@ -137,10 +137,10 @@ foreach my $file (@c_files) { ...@@ -137,10 +137,10 @@ foreach my $file (@c_files) {
foreach my $winapi (@winapis) { foreach my $winapi (@winapis) {
my @entries = (); my @entries = ();
if($winapi->function_stub($internal_name)) { my $module = $winapi->function_internal_module($internal_name);
my $module = $winapi->function_internal_module($internal_name); my $ordinal = $winapi->function_internal_ordinal($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; my $external_name = $internal_name;
if($winapi->name eq "win16") { if($winapi->name eq "win16") {
$external_name =~ s/(?:_)?16([AW]?)$//; $external_name =~ s/(?:_)?16([AW]?)$//;
...@@ -159,7 +159,7 @@ foreach my $file (@c_files) { ...@@ -159,7 +159,7 @@ foreach my $file (@c_files) {
if($external_name ne "@" && if($external_name ne "@" &&
$winapi->is_module($module) && $winapi->is_module($module) &&
$winapi->function_stub($external_name) && $winapi->is_function_stub_in_module($module, $external_name) &&
$internal_name !~ /^\U$module\E_\Q$external_name\E$/) $internal_name !~ /^\U$module\E_\Q$external_name\E$/)
{ {
push @entries, [$external_name, $module, $ordinal]; push @entries, [$external_name, $module, $ordinal];
...@@ -381,8 +381,8 @@ foreach my $file (@c_files) { ...@@ -381,8 +381,8 @@ foreach my $file (@c_files) {
} }
# FIXME: Not 100% correct # FIXME: Not 100% correct
if(!$found && if(!$found &&
!$win16api->function_stub($internal_name) && !$win16api->is_function_stub_in_module($module2, $internal_name) &&
!$win32api->function_stub($internal_name)) !$win32api->is_function_stub_in_module($module2, $internal_name))
{ {
if($line3 > 0) { if($line3 > 0) {
$part2 = $external_name2 . " " x (length($part2) - length($external_name2)); $part2 = $external_name2 . " " x (length($part2) - length($external_name2));
......
...@@ -346,7 +346,6 @@ dlls/user ...@@ -346,7 +346,6 @@ dlls/user
controls controls
dlls/kernel dlls/kernel
dlls/user dlls/user
memory
misc misc
windows windows
......
...@@ -19,7 +19,8 @@ sub new { ...@@ -19,7 +19,8 @@ sub new {
my $options = \${$self->{OPTIONS}}; my $options = \${$self->{OPTIONS}};
my $output = \${$self->{OUTPUT}}; my $output = \${$self->{OUTPUT}};
my $spec_files = \%{$self->{SPEC_FILES}}; my $dir2spec_file = \%{$self->{DIR2SPEC_FILE}};
my $spec_file2dir = \%{$self->{SPEC_FILE2DIR}};
my $spec_file2module = \%{$self->{SPEC_FILE2MODULE}}; my $spec_file2module = \%{$self->{SPEC_FILE2MODULE}};
$$options = shift; $$options = shift;
...@@ -71,12 +72,13 @@ sub new { ...@@ -71,12 +72,13 @@ sub new {
} else { } else {
$all_spec_files{"$wine_dir/$spec_file"}--; $all_spec_files{"$wine_dir/$spec_file"}--;
} }
$$spec_files{""}{$spec_file}++; # FIXME: Kludge $$dir2spec_file{""}{$spec_file}++; # FIXME: Kludge
next; next;
} else { } else {
$allowed_dir = $1; $allowed_dir = $1;
$$spec_file2dir{$spec_file}{$allowed_dir}++;
} }
$$spec_files{$allowed_dir}{$spec_file}++; $$dir2spec_file{$allowed_dir}{$spec_file}++;
if(!-d "$wine_dir/$allowed_dir") { if(!-d "$wine_dir/$allowed_dir") {
$$output->write("modules.dat: $spec_file: directory ($allowed_dir) doesn't exist or is no directory\n"); $$output->write("modules.dat: $spec_file: directory ($allowed_dir) doesn't exist or is no directory\n");
...@@ -95,6 +97,14 @@ sub new { ...@@ -95,6 +97,14 @@ sub new {
return $self; return $self;
} }
sub all_modules {
my $self = shift;
my $module2spec_file = \%{$self->{MODULE2SPEC_FILE}};
return sort(keys(%$module2spec_file));
}
sub spec_file_module { sub spec_file_module {
my $self = shift; my $self = shift;
...@@ -106,14 +116,14 @@ sub spec_file_module { ...@@ -106,14 +116,14 @@ sub spec_file_module {
my $module = shift; my $module = shift;
$$spec_file2module{$spec_file}{$module}++; $$spec_file2module{$spec_file} = $module;
$$module2spec_file{$module}{$spec_file}++; $$module2spec_file{$module} = $spec_file;
} }
sub is_allowed_module_in_file { sub is_allowed_module_in_file {
my $self = shift; my $self = shift;
my $spec_files = \%{$self->{SPEC_FILES}}; my $dir2spec_file = \%{$self->{DIR2SPEC_FILE}};
my $spec_file2module = \%{$self->{SPEC_FILE2MODULE}}; my $spec_file2module = \%{$self->{SPEC_FILE2MODULE}};
my $module = shift; my $module = shift;
...@@ -123,8 +133,8 @@ sub is_allowed_module_in_file { ...@@ -123,8 +133,8 @@ sub is_allowed_module_in_file {
my $dir = $file; my $dir = $file;
$dir =~ s/\/[^\/]*$//; $dir =~ s/\/[^\/]*$//;
foreach my $spec_file (sort(keys(%{$$spec_files{$dir}}))) { foreach my $spec_file (sort(keys(%{$$dir2spec_file{$dir}}))) {
if($$spec_file2module{$spec_file}{$module}) { if($$spec_file2module{$spec_file} eq $module) {
return 1; return 1;
} }
} }
...@@ -135,7 +145,7 @@ sub is_allowed_module_in_file { ...@@ -135,7 +145,7 @@ sub is_allowed_module_in_file {
sub allowed_modules_in_file { sub allowed_modules_in_file {
my $self = shift; my $self = shift;
my $spec_files = \%{$self->{SPEC_FILES}}; my $dir2spec_file = \%{$self->{DIR2SPEC_FILE}};
my $spec_file2module = \%{$self->{SPEC_FILE2MODULE}}; my $spec_file2module = \%{$self->{SPEC_FILE2MODULE}};
my $file = shift; my $file = shift;
...@@ -145,21 +155,33 @@ sub allowed_modules_in_file { ...@@ -145,21 +155,33 @@ sub allowed_modules_in_file {
$dir =~ s/\/[^\/]*$//; $dir =~ s/\/[^\/]*$//;
my %allowed_modules = (); my %allowed_modules = ();
foreach my $spec_file (sort(keys(%{$$spec_files{$dir}}))) { foreach my $spec_file (sort(keys(%{$$dir2spec_file{$dir}}))) {
foreach my $module (sort(keys(%{$$spec_file2module{$spec_file}}))) { my $module = $$spec_file2module{$spec_file};
$allowed_modules{$module}++; $allowed_modules{$module}++;
}
} }
return join(" & ", sort(keys(%allowed_modules))); return join(" & ", sort(keys(%allowed_modules)));
} }
sub allowed_dirs_for_module {
my $self = shift;
my $module2spec_file = \%{$self->{MODULE2SPEC_FILE}};
my $spec_file2dir = \%{$self->{SPEC_FILE2DIR}};
my $module = shift;
my $spec_file = $$module2spec_file{$module};
return sort(keys(%{$$spec_file2dir{$spec_file}}));
}
sub allowed_spec_files { sub allowed_spec_files {
my $self = shift; my $self = shift;
my $options = \${$self->{OPTIONS}}; my $options = \${$self->{OPTIONS}};
my $output = \${$self->{OUTPUT}}; my $output = \${$self->{OUTPUT}};
my $spec_files = \%{$self->{SPEC_FILES}}; my $dir2spec_file = \%{$self->{DIR2SPEC_FILE}};
my $wine_dir = shift; my $wine_dir = shift;
my $current_dir = shift; my $current_dir = shift;
...@@ -179,7 +201,7 @@ sub allowed_spec_files { ...@@ -179,7 +201,7 @@ sub allowed_spec_files {
my %allowed_spec_files = (); my %allowed_spec_files = ();
foreach my $dir (sort(@dirs)) { foreach my $dir (sort(@dirs)) {
foreach my $spec_file (sort(keys(%{$$spec_files{$dir}}))) { foreach my $spec_file (sort(keys(%{$$dir2spec_file{$dir}}))) {
$allowed_spec_files{$spec_file}++; $allowed_spec_files{$spec_file}++;
} }
} }
...@@ -202,18 +224,17 @@ sub global_report { ...@@ -202,18 +224,17 @@ sub global_report {
my $self = shift; my $self = shift;
my $output = \${$self->{OUTPUT}}; my $output = \${$self->{OUTPUT}};
my $spec_files = \%{$self->{SPEC_FILES}}; my $dir2spec_file = \%{$self->{DIR2SPEC_FILE}};
my $spec_file2module = \%{$self->{SPEC_FILE2MODULE}}; my $spec_file2module = \%{$self->{SPEC_FILE2MODULE}};
my $used_module_dirs = \%{$self->{USED_MODULE_DIRS}}; my $used_module_dirs = \%{$self->{USED_MODULE_DIRS}};
my @messages; my @messages;
foreach my $dir (sort(keys(%$spec_files))) { foreach my $dir (sort(keys(%$dir2spec_file))) {
if($dir eq "") { next; } if($dir eq "") { next; }
foreach my $spec_file (sort(keys(%{$$spec_files{$dir}}))) { foreach my $spec_file (sort(keys(%{$$dir2spec_file{$dir}}))) {
foreach my $module (sort(keys(%{$$spec_file2module{$spec_file}}))) { my $module = $$spec_file2module{$spec_file};
if(!$$used_module_dirs{$module}{$dir}) { if(!$$used_module_dirs{$module}{$dir}) {
push @messages, "modules.dat: $spec_file: directory ($dir) is not used\n"; push @messages, "modules.dat: $spec_file: directory ($dir) is not used\n";
}
} }
} }
} }
......
...@@ -2,6 +2,15 @@ package nativeapi; ...@@ -2,6 +2,15 @@ package nativeapi;
use strict; use strict;
use vars qw($VERSION @ISA @EXPORT @EXPORT_OK);
require Exporter;
@ISA = qw(Exporter);
@EXPORT = qw();
@EXPORT_OK = qw($nativeapi);
use vars qw($nativeapi);
sub new { sub new {
my $proto = shift; my $proto = shift;
my $class = ref($proto) || $proto; my $class = ref($proto) || $proto;
...@@ -32,10 +41,10 @@ sub new { ...@@ -32,10 +41,10 @@ sub new {
open(IN, "< $api_file"); open(IN, "< $api_file");
local $/ = "\n"; local $/ = "\n";
while(<IN>) { while(<IN>) {
s/^\s*?(.*?)\s*$/$1/; # remove whitespace at begin and end of line s/^\s*(.*?)\s*$/$1/; # remove whitespace at begin and end of line
s/^(.*?)\s*#.*$/$1/; # remove comments s/^(.*?)\s*#.*$/$1/; # remove comments
/^$/ && next; # skip empty lines /^$/ && next; # skip empty lines
$$functions{$_}++; $$functions{$_}++;
} }
close(IN); close(IN);
...@@ -110,6 +119,8 @@ sub new { ...@@ -110,6 +119,8 @@ sub new {
} }
close(IN); close(IN);
$nativeapi = $self;
return $self; return $self;
} }
......
...@@ -2,6 +2,9 @@ package output; ...@@ -2,6 +2,9 @@ package output;
use strict; use strict;
my $stdout_isatty = -t STDOUT;
my $stderr_isatty = -t STDERR;
sub new { sub new {
my $proto = shift; my $proto = shift;
my $class = ref($proto) || $proto; my $class = ref($proto) || $proto;
...@@ -30,7 +33,7 @@ sub show_progress { ...@@ -30,7 +33,7 @@ sub show_progress {
$$progress_count++; $$progress_count++;
if($$progress_count > 0 && $$progress) { if($$progress_count > 0 && $$progress && $stderr_isatty) {
print STDERR $$progress; print STDERR $$progress;
$$last_progress = $$progress; $$last_progress = $$progress;
} }
...@@ -44,7 +47,7 @@ sub hide_progress { ...@@ -44,7 +47,7 @@ sub hide_progress {
$$progress_count--; $$progress_count--;
if($$last_progress) { if($$last_progress && $stderr_isatty) {
my $message; my $message;
for (1..length($$last_progress)) { for (1..length($$last_progress)) {
$message .= " "; $message .= " ";
...@@ -103,9 +106,9 @@ sub write { ...@@ -103,9 +106,9 @@ sub write {
my $prefix = \${$self->{PREFIX}}; my $prefix = \${$self->{PREFIX}};
$self->hide_progress; $self->hide_progress if $stdout_isatty;
print $$prefix . $message; print $$prefix . $message;
$self->show_progress; $self->show_progress if $stdout_isatty;
} }
1; 1;
...@@ -15,6 +15,7 @@ WCHAR ...@@ -15,6 +15,7 @@ WCHAR
int int
long long
size_t size_t
time_t
unsigned int unsigned int
unsigned long unsigned long
...@@ -55,6 +56,7 @@ bad_typeid * ...@@ -55,6 +56,7 @@ bad_typeid *
char * char *
char ** char **
char *** char ***
double *
exception * exception *
int * int *
jmp_buf jmp_buf
...@@ -64,7 +66,9 @@ struct _stat * ...@@ -64,7 +66,9 @@ struct _stat *
struct _timeb * struct _timeb *
struct _utimbuf * struct _utimbuf *
struct _wfinddata_t * struct _wfinddata_t *
struct tm *
terminate_function terminate_function
time_t *
type_info * type_info *
unexpected_function unexpected_function
unsigned char * unsigned char *
......
...@@ -689,6 +689,30 @@ sub all_internal_functions_in_module { ...@@ -689,6 +689,30 @@ sub all_internal_functions_in_module {
return sort(@names); return sort(@names);
} }
sub all_external_functions {
my $self = shift;
my $function_internal_name = \%{$self->{FUNCTION_INTERNAL_NAME}};
return sort(keys(%$function_internal_name));
}
sub all_external_functions_in_module {
my $self = shift;
my $function_internal_name = \%{$self->{FUNCTION_INTERNAL_NAME}};
my $function_external_module = \%{$self->{FUNCTION_EXTERNAL_MODULE}};
my $module = shift;
my @names;
foreach my $name (keys(%$function_internal_name)) {
if($$function_external_module{$name} eq $module) {
push @names, $name;
}
}
return sort(@names);
}
sub all_functions_stub { sub all_functions_stub {
my $self = shift; my $self = shift;
my $function_stub = \%{$self->{FUNCTION_STUB}}; my $function_stub = \%{$self->{FUNCTION_STUB}};
......
...@@ -238,7 +238,9 @@ foreach my $file (@c_files) { ...@@ -238,7 +238,9 @@ foreach my $file (@c_files) {
$win16api->found_type($argument) if $options->win16; $win16api->found_type($argument) if $options->win16;
$win32api->found_type($argument) if $options->win32; $win32api->found_type($argument) if $options->win32;
} }
}
if($options->declared) {
$win16api->found_internal_function($internal_name) if $options->win16; $win16api->found_internal_function($internal_name) if $options->win16;
$win32api->found_internal_function($internal_name) if $options->win32; $win32api->found_internal_function($internal_name) if $options->win32;
} }
...@@ -355,7 +357,7 @@ foreach my $file (@c_files) { ...@@ -355,7 +357,7 @@ foreach my $file (@c_files) {
if($options->local && $options->documentation && if($options->local && $options->documentation &&
(defined($module16) || defined($module32)) && (defined($module16) || defined($module32)) &&
$linkage ne "extern" && defined($statements)) $linkage ne "static" && ($linkage ne "" || defined($statements)))
{ {
winapi_documentation::check_documentation $options, $output, $win16api, $win32api, $modules, $function; winapi_documentation::check_documentation $options, $output, $win16api, $win32api, $modules, $function;
} }
...@@ -488,6 +490,51 @@ foreach my $file (@c_files) { ...@@ -488,6 +490,51 @@ foreach my $file (@c_files) {
$output->hide_progress; $output->hide_progress;
if($options->declared) {
my %dirs;
foreach my $file (@c_files) {
my $dir = $file;
$dir =~ s%/?[^/]*$%%;
if($dir) {
if($current_dir ne ".") {
$dir = "$current_dir/$dir";
}
} else {
$dir = "$current_dir";
}
$dirs{$dir}++;
}
foreach my $module ($modules->all_modules) {
my $incomplete = 0;
foreach my $module_dir ($modules->allowed_dirs_for_module($module)) {
my $found = 0;
foreach my $dir (sort(keys(%dirs))) {
if($module_dir eq $dir) {
$found = 1;
last;
}
}
if(!$found) {
$incomplete = 1;
}
}
if(!$incomplete) {
if($options->declared) {
foreach my $winapi (@winapis) {
if(!$winapi->is_module($module)) { next; }
foreach my $internal_name ($winapi->all_internal_functions_in_module($module)) {
if(!$winapi->internal_function_found($internal_name)) {
$output->write("*.c: $module: $internal_name: " .
"function declared but not implemented or declared external\n");
}
}
}
}
}
}
}
if($options->global) { if($options->global) {
winapi_documentation::report_documentation $options, $output; winapi_documentation::report_documentation $options, $output;
......
...@@ -3,6 +3,7 @@ package winapi_documentation; ...@@ -3,6 +3,7 @@ package winapi_documentation;
use strict; use strict;
use config qw($current_dir $wine_dir); use config qw($current_dir $wine_dir);
use nativeapi qw($nativeapi);
my %comment_width; my %comment_width;
my %comment_indent; my %comment_indent;
......
...@@ -68,7 +68,7 @@ sub _external_names { ...@@ -68,7 +68,7 @@ sub _external_names {
} }
sub external_name16 { my $self = shift; return $self->_external_name($win16api, @_); } sub external_name16 { my $self = shift; return $self->_external_name($win16api, @_); }
sub external_name32{ my $self = shift; return $self->_external_name($win32api, @_); } sub external_name32 { my $self = shift; return $self->_external_name($win32api, @_); }
sub external_names16 { my $self = shift; return $self->_external_names($win16api, @_); } sub external_names16 { my $self = shift; return $self->_external_names($win16api, @_); }
sub external_names32 { my $self = shift; return $self->_external_names($win32api, @_); } sub external_names32 { my $self = shift; return $self->_external_names($win32api, @_); }
......
...@@ -19,16 +19,6 @@ sub check { ...@@ -19,16 +19,6 @@ sub check {
} }
} }
if($options->declared) {
foreach my $internal_name ($winapi->all_internal_functions) {
if(!$winapi->internal_function_found($internal_name) && !$nativeapi->is_function($internal_name)) {
my $module = $winapi->function_internal_module($internal_name);
$output->write("*.c: $module: $internal_name: ");
$output->write("function declared but not implemented: " . $winapi->function_internal_arguments($internal_name) . "\n");
}
}
}
if($options->argument && $options->argument_forbidden) { if($options->argument && $options->argument_forbidden) {
my $not_used = $winapi->types_not_used; my $not_used = $winapi->types_not_used;
......
...@@ -56,6 +56,10 @@ sub parse_c_file { ...@@ -56,6 +56,10 @@ sub parse_c_file {
my $function_end = sub { my $function_end = sub {
my $function = 'winapi_function'->new; my $function = 'winapi_function'->new;
if(!defined($documentation_line)) {
$documentation_line = 0;
}
$function->file($file); $function->file($file);
$function->debug_channels([@$debug_channels]); $function->debug_channels([@$debug_channels]);
$function->documentation($documentation); $function->documentation($documentation);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment