#!/usr/bin/env perl # # Copyright (C) 2005 Apple Inc. # Copyright (C) 2006 Anders Carlsson # # This file is part of WebKit # # This library is free software; you can redistribute it and/or # modify it under the terms of the GNU Library General Public # License as published by the Free Software Foundation; either # version 2 of the License, or (at your option) any later version. # # This library is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU # Library General Public License for more details. # # You should have received a copy of the GNU Library General Public License # along with this library; see the file COPYING.LIB. If not, write to # the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, # Boston, MA 02110-1301, USA. # # This script is a temporary hack. # Files are generated in the source directory, when they really should go # to the DerivedSources directory. # This should also eventually be a build rule driven off of .idl files # however a build rule only solution is blocked by several radars: # use strict; use warnings; use FindBin; use lib '.', $FindBin::Bin; use English; use File::Path; use File::Basename; use Getopt::Long; use Text::ParseWords; use Cwd; use JSON::PP; use IDLParser; use CodeGenerator; my @idlDirectories; my $outputDirectory; my $outputHeadersDirectory; my $generator; my $defines; my $filename; my $prefix; my $preprocessor; my $writeDependencies; my $verbose; my $supplementalDependencyFile; my $additionalIdlFiles; my $idlAttributesFile; GetOptions('include=s@' => \@idlDirectories, 'outputDir=s' => \$outputDirectory, 'outputHeadersDir=s' => \$outputHeadersDirectory, 'generator=s' => \$generator, 'defines=s' => \$defines, 'filename=s' => \$filename, 'prefix=s' => \$prefix, 'preprocessor=s' => \$preprocessor, 'verbose' => \$verbose, 'write-dependencies' => \$writeDependencies, 'supplementalDependencyFile=s' => \$supplementalDependencyFile, 'additionalIdlFiles=s' => \$additionalIdlFiles, 'idlAttributesFile=s' => \$idlAttributesFile); die('Must specify input file.') unless @ARGV; die('Must specify generator') unless defined($generator); die('Must specify output directory.') unless defined($outputDirectory); die('Must specify IDL attributes file.') unless defined($idlAttributesFile); if (!$outputHeadersDirectory) { $outputHeadersDirectory = $outputDirectory; } generateBindings($_) for (@ARGV); sub generateBindings { my ($targetIdlFile) = @_; $targetIdlFile = Cwd::realpath($targetIdlFile); if ($verbose) { print "$generator: $targetIdlFile\n"; } my $targetInterfaceName = fileparse($targetIdlFile, ".idl"); my $idlFound = 0; my @supplementedIdlFiles; if ($supplementalDependencyFile) { # The format of a supplemental dependency file: # # DOMWindow.idl P.idl Q.idl R.idl # Document.idl S.idl # Event.idl # ... # # The above indicates that DOMWindow.idl is supplemented by P.idl, Q.idl and R.idl, # Document.idl is supplemented by S.idl, and Event.idl is supplemented by no IDLs. # The IDL that supplements another IDL (e.g. P.idl) never appears in the dependency file. open FH, "< $supplementalDependencyFile" or die "Cannot open $supplementalDependencyFile\n"; while (my $line = ) { my ($idlFile, @followingIdlFiles) = split(/\s+/, $line); if ($idlFile and fileparse($idlFile) eq fileparse($targetIdlFile)) { $idlFound = 1; @supplementedIdlFiles = sort @followingIdlFiles; } } close FH; # $additionalIdlFiles is list of IDL files which should not be included in # DerivedSources*.cpp (i.e. they are not described in the supplemental # dependency file) but should generate .h and .cpp files. if (!$idlFound and $additionalIdlFiles) { my @idlFiles = shellwords($additionalIdlFiles); $idlFound = grep { $_ and fileparse($_) eq fileparse($targetIdlFile) } @idlFiles; } if (!$idlFound) { my $codeGen = CodeGenerator->new(\@idlDirectories, $generator, $outputDirectory, $outputHeadersDirectory, $preprocessor, $writeDependencies, $verbose); # We generate empty .h and .cpp files just to tell build scripts that .h and .cpp files are created. generateEmptyHeaderAndCpp($codeGen->FileNamePrefix(), $targetInterfaceName, $outputHeadersDirectory, $outputDirectory); return; } } my $input; { local $INPUT_RECORD_SEPARATOR; open(JSON, "<", $idlAttributesFile) or die "Couldn't open $idlAttributesFile: $!"; $input = ; close(JSON); } my $jsonDecoder = JSON::PP->new->utf8; my $jsonHashRef = $jsonDecoder->decode($input); my $idlAttributes = $jsonHashRef->{attributes}; # Parse the target IDL file. my $targetParser = IDLParser->new(!$verbose); my $targetDocument = $targetParser->Parse($targetIdlFile, $defines, $preprocessor, $idlAttributes); foreach my $idlFile (@supplementedIdlFiles) { next if $idlFile eq $targetIdlFile; my $interfaceName = fileparse($idlFile, ".idl"); my $parser = IDLParser->new(!$verbose); my $document = $parser->Parse($idlFile, $defines, $preprocessor, $idlAttributes); foreach my $interface (@{$document->interfaces}) { if (!$interface->isPartial || $interface->type->name eq $targetInterfaceName) { my $targetDataNode; my @targetGlobalContexts; foreach my $interface (@{$targetDocument->interfaces}) { if ($interface->type->name eq $targetInterfaceName) { $targetDataNode = $interface; my $exposedAttribute = $targetDataNode->extendedAttributes->{"Exposed"} || "Window"; $exposedAttribute = substr($exposedAttribute, 1, -1) if substr($exposedAttribute, 0, 1) eq "("; @targetGlobalContexts = split(",", $exposedAttribute); last; } } die "Not found an interface ${targetInterfaceName} in ${targetInterfaceName}.idl." unless defined $targetDataNode; # Support for attributes of partial interfaces. foreach my $attribute (@{$interface->attributes}) { next unless shouldPropertyBeExposed($attribute, \@targetGlobalContexts); # Record that this attribute is implemented by $interfaceName. $attribute->extendedAttributes->{"ImplementedBy"} = $interfaceName if $interface->isPartial && !$attribute->extendedAttributes->{Reflect}; # Add interface-wide extended attributes to each attribute. foreach my $extendedAttributeName (keys %{$interface->extendedAttributes}) { $attribute->extendedAttributes->{$extendedAttributeName} = $interface->extendedAttributes->{$extendedAttributeName}; } push(@{$targetDataNode->attributes}, $attribute); } # Support for methods of partial interfaces. foreach my $operation (@{$interface->operations}) { next unless shouldPropertyBeExposed($operation, \@targetGlobalContexts); # Record that this method is implemented by $interfaceName. $operation->extendedAttributes->{"ImplementedBy"} = $interfaceName if $interface->isPartial; # Add interface-wide extended attributes to each method. foreach my $extendedAttributeName (keys %{$interface->extendedAttributes}) { $operation->extendedAttributes->{$extendedAttributeName} = $interface->extendedAttributes->{$extendedAttributeName}; } push(@{$targetDataNode->operations}, $operation); } # Support for constants of partial interfaces. foreach my $constant (@{$interface->constants}) { next unless shouldPropertyBeExposed($constant, \@targetGlobalContexts); # Record that this constant is implemented by $interfaceName. $constant->extendedAttributes->{"ImplementedBy"} = $interfaceName if $interface->isPartial; # Add interface-wide extended attributes to each constant. foreach my $extendedAttributeName (keys %{$interface->extendedAttributes}) { $constant->extendedAttributes->{$extendedAttributeName} = $interface->extendedAttributes->{$extendedAttributeName}; } push(@{$targetDataNode->constants}, $constant); } } else { die "$idlFile is not a supplemental dependency of $targetIdlFile. There maybe a bug in the supplemental dependency generator (preprocess-idls.pl).\n"; } } } # Generate desired output for the target IDL file. my $codeGen = CodeGenerator->new(\@idlDirectories, $generator, $outputDirectory, $outputHeadersDirectory, $preprocessor, $writeDependencies, $verbose, $targetIdlFile, $idlAttributes); $codeGen->ProcessDocument($targetDocument, $defines); } # Attributes / Operations / Constants of supplemental interfaces can have an [Exposed=XX] attribute which restricts # on which global contexts they should be exposed. sub shouldPropertyBeExposed { my ($context, $targetGlobalContexts) = @_; my $exposed = $context->extendedAttributes->{Exposed}; return 1 unless $exposed; $exposed = substr($exposed, 1, -1) if substr($exposed, 0, 1) eq "("; my @sourceGlobalContexts = split(",", $exposed); for my $targetGlobalContext (@$targetGlobalContexts) { return 1 if grep(/^$targetGlobalContext$/, @sourceGlobalContexts); } return 0; } sub generateEmptyHeaderAndCpp { my ($prefix, $targetInterfaceName, $outputHeadersDirectory, $outputDirectory) = @_; my $headerName = "${prefix}${targetInterfaceName}.h"; my $cppName = "${prefix}${targetInterfaceName}.cpp"; my $contents = "/* This file is generated just to tell build scripts that $headerName and $cppName are created for ${targetInterfaceName}.idl, and thus prevent the build scripts from trying to generate $headerName and $cppName at every build. This file must not be tried to compile. */ "; open FH, "> ${outputHeadersDirectory}/${headerName}" or die "Cannot open $headerName\n"; print FH $contents; close FH; open FH, "> ${outputDirectory}/${cppName}" or die "Cannot open $cppName\n"; print FH $contents; close FH; }