36547f4eff
The main reasons for doing this are: 1. _qpa.h end up in the master QtGui include file. QtGui is meant for userland applications. qpa code is neither binary nor source compatible. Inadvertant use of QPA api makes the user code binary-incompatible. 2. syncqt creates forwarding headers for non-private header files. This gives people the impression that this is public API. As discussed on the mailing list, even though QPA api is internal and subject to change, it needs to treated differently from private headers since they will be used by in-qtbase and out-of-qtbase plugins. This commit does the following: 1. The _qpa in QPA header files is dropped. 2. syncqt now treats any file with qplatform prefix as a special file and moves it to qpa/ directory. The recommended way of using QPA API in plugins is: #include <qpa/qplatformfoo.h>. This allows the user include QPA API from multiple modules (for example, qplatformfoo might be in QtPrintSupport) 3. The user needs to explicitly add QT += <module>-private to get access to the qpa api. 4. Creates compat headers for the olden style qplatformfoo_qpa.h and QPlatformFoo includes. This commit does not change the cpp filenames. This requires a more careful merging of existing non qpa cpp files and existing cpp files on a case by case basis. This can be done at anytime. The following files are not renamed as part of this changed but will be fixed as part of a future change: src/gui/kernel/qgenericpluginfactory_qpa.h src/gui/kernel/qgenericplugin_qpa.h src/gui/kernel/qwindowsysteminterface_qpa.h files were renamed using for x in `find . -name "qplatform*_qpa.h"`; do git mv $x "${x/_qpa.h/.h}"; done for x in `find . -name "qplatform*_qpa_p.h"`; do git mv $x "${x/_qpa_p.h/_p.h}"; done includes were renamed using script for file in `find . -name "*.h" -or -name "*.cpp" -or -name "*.mm"`; do sed -i -e 's,.*#.*include.*<\(Qt.*/\)\?\(QPlatform.*\)>,#include <qpa/\L\2.h>,g' \ -e 's,.*#.*include.*"\(Qt.*/\)\?\(QPlatform.*\)",#include <qpa/\L\2.h>,g' \ -e 's,.*#.*include.* "\(qplatform.*\)_qpa.h",#include <qpa/\L\1.h>,g' \ -e 's,.*#.*include.*"\(qplatform.*\)_qpa_p.h",#include <qpa/\L\1_p.h>,g' \ -e 's,.*#.*include.*<\(Qt.*/\|Qt.*/private/\|private/\)\?\(qplatform.*\)_qpa\(.*\)>,#include <qpa/\2\3>,g' \ -e 's,.*#.*include.*"\(Qt.*/\|Qt.*/private/\|private/\)\?\(qplatform.*\)_qpa\(.*\)",#include <qpa/\2\3>,g' \ $file done Change-Id: I04a350314a45746e3911f54b3b21ad03315afb67 Reviewed-by: Morten Johan Sørvig <morten.sorvig@nokia.com> Reviewed-by: Samuel Rødal <samuel.rodal@nokia.com> Reviewed-by: Friedemann Kleint <Friedemann.Kleint@nokia.com> Reviewed-by: Sean Harmer <sean.harmer@kdab.com> Reviewed-by: Lars Knoll <lars.knoll@nokia.com> Reviewed-by: Gunnar Sletta <gunnar.sletta@nokia.com>
1436 lines
59 KiB
Perl
Executable File
1436 lines
59 KiB
Perl
Executable File
#!/usr/bin/perl
|
|
#############################################################################
|
|
##
|
|
## Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies).
|
|
## Contact: http://www.qt-project.org/
|
|
##
|
|
## This file is part of the build configuration tools of the Qt Toolkit.
|
|
##
|
|
## $QT_BEGIN_LICENSE:LGPL$
|
|
## GNU Lesser General Public License Usage
|
|
## This file may be used under the terms of the GNU Lesser General Public
|
|
## License version 2.1 as published by the Free Software Foundation and
|
|
## appearing in the file LICENSE.LGPL included in the packaging of this
|
|
## file. Please review the following information to ensure the GNU Lesser
|
|
## General Public License version 2.1 requirements will be met:
|
|
## http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
##
|
|
## In addition, as a special exception, Nokia gives you certain additional
|
|
## rights. These rights are described in the Nokia Qt LGPL Exception
|
|
## version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
##
|
|
## GNU General Public License Usage
|
|
## Alternatively, this file may be used under the terms of the GNU General
|
|
## Public License version 3.0 as published by the Free Software Foundation
|
|
## and appearing in the file LICENSE.GPL included in the packaging of this
|
|
## file. Please review the following information to ensure the GNU General
|
|
## Public License version 3.0 requirements will be met:
|
|
## http://www.gnu.org/copyleft/gpl.html.
|
|
##
|
|
## Other Usage
|
|
## Alternatively, this file may be used in accordance with the terms and
|
|
## conditions contained in a signed written agreement between you and Nokia.
|
|
##
|
|
##
|
|
##
|
|
##
|
|
##
|
|
##
|
|
## $QT_END_LICENSE$
|
|
##
|
|
#############################################################################
|
|
|
|
#
|
|
# Synchronizes Qt header files - internal development tool.
|
|
#
|
|
|
|
# use packages -------------------------------------------------------
|
|
use File::Basename;
|
|
use File::Path;
|
|
use Cwd;
|
|
use Cwd 'abs_path';
|
|
use Config;
|
|
use strict;
|
|
use warnings;
|
|
use English qw(-no_match_vars );
|
|
|
|
# set output basedir to be where ever syncqt is run from
|
|
our $out_basedir = getcwd();
|
|
$out_basedir =~ s=\\=/=g;
|
|
our $basedir;
|
|
our $quoted_basedir;
|
|
|
|
|
|
# try to figure out where QtBase is located
|
|
# normally the script location should be enough, if not fall back to
|
|
# QTDIR environment variable. If that doesn't work, later ask the
|
|
# user to use the -qtdir option explicitly.
|
|
my $qtbasedir = $ENV{"QTDIR"};
|
|
$qtbasedir = dirname(dirname($0)) if (!$qtbasedir);
|
|
$qtbasedir =~ s=\\=/=g if (defined $qtbasedir);
|
|
|
|
# will be defined based on the modules sync.profile
|
|
our (%modules, %moduleheaders, @allmoduleheadersprivate, %classnames, %mastercontent, %modulepris, %explicitheaders, %deprecatedheaders);
|
|
|
|
# global variables (modified by options)
|
|
my $isunix = 0;
|
|
my $module = 0;
|
|
my $showonly = 0;
|
|
my $verbose_level = 1;
|
|
my $remove_stale = 1;
|
|
my $force_win = 0;
|
|
my $force_relative = 0;
|
|
my $check_includes = 0;
|
|
my $copy_headers = 0;
|
|
my $create_uic_class_map = 0;
|
|
my $create_private_headers = 1;
|
|
my $module_fwd = "";
|
|
my $cache_module_fwd = 0;
|
|
my $developer_build = 0;
|
|
my $no_module_version_header = 0;
|
|
my $makefile_generator = "";
|
|
my @modules_to_sync ;
|
|
$force_relative = 1 if ( -d "/System/Library/Frameworks" );
|
|
|
|
|
|
# functions ----------------------------------------------------------
|
|
|
|
######################################################################
|
|
# Syntax: showUsage()
|
|
# Params: -none-
|
|
#
|
|
# Purpose: Show the usage of the script.
|
|
# Returns: -none-
|
|
######################################################################
|
|
sub showUsage
|
|
{
|
|
print "$0 usage:\n";
|
|
print " <module directory> Specifies which module to sync header files for (required for shadow builds!)\n\n";
|
|
|
|
print " -copy Copy headers instead of include-fwd(default: " . ($copy_headers ? "yes" : "no") . ")\n";
|
|
print " -remove-stale Removes stale headers (default: " . ($remove_stale ? "yes" : "no") . ")\n";
|
|
print " -relative Force relative symlinks (default: " . ($force_relative ? "yes" : "no") . ")\n";
|
|
print " -windows Force platform to Windows (default: " . ($force_win ? "yes" : "no") . ")\n";
|
|
print " -showonly Show action but not perform (default: " . ($showonly ? "yes" : "no") . ")\n";
|
|
print " -outdir <PATH> Specify output directory for sync (default: $out_basedir)\n";
|
|
print " -qtdir <PATH> Set the path to QtBase (detected: " . (defined $qtbasedir ? $qtbasedir : "-none-") . ")\n";
|
|
print " -quiet Only report problems, not activity (same as -verbose 0)\n";
|
|
print " -v, -verbose <level> Sets the verbosity level (max. 4) (default: $verbose_level)\n";
|
|
print " The short form increases the level by +1\n";
|
|
print " -separate-module <NAME>:<PROFILEDIR>:<HEADERDIR>\n";
|
|
print " Create headers for <NAME> with original headers in\n";
|
|
print " <HEADERDIR> relative to <PROFILEDIR> \n";
|
|
print " -private Force copy private headers (default: " . ($create_private_headers ? "yes" : "no") . ")\n";
|
|
print " -module-fwd <PATH> Create fwd includes for module pri files in the given\n";
|
|
print " path (default: none)\n";
|
|
print " -cache-module-fwd Create a .qmake.cache file to cache the location of the\n";
|
|
print " fwd includes\n";
|
|
print " -developer-build Point libraries and binaries to a common directory for\n";
|
|
print " easy development\n";
|
|
print " -no-module-version-header\n";
|
|
print " Don't create module version header file\n";
|
|
print " -generator <PATH> Specify the makefile generator setting (e.g. 'UNIX')\n";
|
|
print " -help This help\n";
|
|
exit 0;
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: checkUnix()
|
|
# Params: -none-
|
|
#
|
|
# Purpose: Check if script runs on a Unix system or not. Cygwin
|
|
# systems are _not_ detected as Unix systems.
|
|
# Returns: 1 if a unix system, else 0.
|
|
######################################################################
|
|
sub checkUnix {
|
|
my ($r) = 0;
|
|
if ( $force_win != 0) {
|
|
return 0;
|
|
} elsif ( -f "/bin/uname" ) {
|
|
$r = 1;
|
|
(-f "\\bin\\uname") && ($r = 0);
|
|
} elsif ( -f "/usr/bin/uname" ) {
|
|
$r = 1;
|
|
(-f "\\usr\\bin\\uname") && ($r = 0);
|
|
}
|
|
if($r) {
|
|
$_ = $Config{'osname'};
|
|
$r = 0 if( /(ms)|(cyg)win/i );
|
|
}
|
|
return $r;
|
|
}
|
|
|
|
sub checkRelative {
|
|
my ($dir) = @_;
|
|
return 0 if($dir =~ /^\//);
|
|
return 0 if(!checkUnix() && $dir =~ /[a-zA-Z]:[\/\\]/);
|
|
return 1;
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: shouldMasterInclude(iheader)
|
|
# Params: iheader, string, filename to verify inclusion
|
|
#
|
|
# Purpose: Determines if header should be in the master include file.
|
|
# Returns: 0 if file contains "#pragma qt_no_master_include" or not
|
|
# able to open, else 1.
|
|
######################################################################
|
|
sub shouldMasterInclude {
|
|
my ($iheader) = @_;
|
|
return 0 if(basename($iheader) =~ /_/);
|
|
return 0 if(basename($iheader) =~ /qconfig/);
|
|
if(open(F, "<$iheader")) {
|
|
while(<F>) {
|
|
chomp;
|
|
return 0 if(/^\#pragma qt_no_master_include$/);
|
|
}
|
|
close(F);
|
|
} else {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: classNames(iheader)
|
|
# Params: iheader, string, filename to parse for classname "symlinks"
|
|
#
|
|
# Purpose: Scans through iheader to find all classnames that should be
|
|
# synced into library's include structure.
|
|
# Returns: List of all class names in a file.
|
|
######################################################################
|
|
sub classNames {
|
|
my @ret;
|
|
my ($iheader) = @_;
|
|
|
|
my $classname = $classnames{basename($iheader)};
|
|
push @ret, $classname if ($classname);
|
|
|
|
my $parsable = "";
|
|
if(open(F, "<$iheader")) {
|
|
while(<F>) {
|
|
my $line = $_;
|
|
chomp $line;
|
|
chop $line if ($line =~ /\r$/);
|
|
if($line =~ /^\#/) {
|
|
if($line =~ /\\$/) {
|
|
while($line = <F>) {
|
|
chomp $line;
|
|
last unless($line =~ /\\$/);
|
|
}
|
|
}
|
|
return @ret if($line =~ m/^#pragma qt_sync_stop_processing/);
|
|
push(@ret, $1) if($line =~ m/^#pragma qt_class\(([^)]*)\)[\r\n]*$/);
|
|
$line = 0;
|
|
}
|
|
if($line) {
|
|
$line =~ s,//.*$,,; #remove c++ comments
|
|
$line .= ";" if($line =~ m/^Q_[A-Z_]*\(.*\)[\r\n]*$/); #qt macro
|
|
$line .= ";" if($line =~ m/^QT_(BEGIN|END)_HEADER[\r\n]*$/); #qt macro
|
|
$line .= ";" if($line =~ m/^QT_(BEGIN|END)_NAMESPACE(_[A-Z]+)*[\r\n]*$/); #qt macro
|
|
$line .= ";" if($line =~ m/^QT_MODULE\(.*\)[\r\n]*$/); # QT_MODULE macro
|
|
$parsable .= " " . $line;
|
|
}
|
|
}
|
|
close(F);
|
|
}
|
|
|
|
my $last_definition = 0;
|
|
my @namespaces;
|
|
for(my $i = 0; $i < length($parsable); $i++) {
|
|
my $definition = 0;
|
|
my $character = substr($parsable, $i, 1);
|
|
if($character eq "/" && substr($parsable, $i+1, 1) eq "*") { #I parse like this for greedy reasons
|
|
for($i+=2; $i < length($parsable); $i++) {
|
|
my $end = substr($parsable, $i, 2);
|
|
if($end eq "*/") {
|
|
$last_definition = $i+2;
|
|
$i++;
|
|
last;
|
|
}
|
|
}
|
|
} elsif($character eq "{") {
|
|
my $brace_depth = 1;
|
|
my $block_start = $i + 1;
|
|
BLOCK: for($i+=1; $i < length($parsable); $i++) {
|
|
my $ignore = substr($parsable, $i, 1);
|
|
if($ignore eq "{") {
|
|
$brace_depth++;
|
|
} elsif($ignore eq "}") {
|
|
$brace_depth--;
|
|
unless($brace_depth) {
|
|
for(my $i2 = $i+1; $i2 < length($parsable); $i2++) {
|
|
my $end = substr($parsable, $i2, 1);
|
|
if($end eq ";" || $end ne " ") {
|
|
$definition = substr($parsable, $last_definition, $block_start - $last_definition) . "}";
|
|
$i = $i2 if($end eq ";");
|
|
$last_definition = $i + 1;
|
|
last BLOCK;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} elsif($character eq ";") {
|
|
$definition = substr($parsable, $last_definition, $i - $last_definition + 1);
|
|
$last_definition = $i + 1;
|
|
} elsif($character eq "}") {
|
|
# a naked } must be a namespace ending
|
|
# if it's not a namespace, it's eaten by the loop above
|
|
pop @namespaces;
|
|
$last_definition = $i + 1;
|
|
}
|
|
|
|
if (substr($parsable, $last_definition, $i - $last_definition + 1) =~ m/ namespace ([^ ]*) /
|
|
&& substr($parsable, $i+1, 1) eq "{") {
|
|
push @namespaces, $1;
|
|
|
|
# Eat the opening { so that the condensing loop above doesn't see it
|
|
$i++;
|
|
$last_definition = $i + 1;
|
|
}
|
|
|
|
if($definition) {
|
|
$definition =~ s=[\n\r]==g;
|
|
my @symbols;
|
|
if($definition =~ m/^ *typedef *.*\(\*([^\)]*)\)\(.*\);$/) {
|
|
push @symbols, $1;
|
|
} elsif($definition =~ m/^ *typedef +(.*) +([^ ]*);$/) {
|
|
push @symbols, $2;
|
|
} elsif($definition =~ m/^ *(template *<.*> *)?(class|struct) +([^ ]* +)?([^<\s]+) ?(<[^>]*> ?)?\s*((,|:)\s*(public|protected|private) *.*)? *\{\}$/) {
|
|
push @symbols, $4;
|
|
} elsif($definition =~ m/^ *Q_DECLARE_.*ITERATOR\((.*)\);$/) {
|
|
push @symbols, "Q" . $1 . "Iterator";
|
|
push @symbols, "QMutable" . $1 . "Iterator";
|
|
}
|
|
|
|
our $publicclassregexp;
|
|
foreach my $symbol (@symbols) {
|
|
$symbol = (join("::", @namespaces) . "::" . $symbol) if (scalar @namespaces);
|
|
|
|
if ($symbol =~ /^Q[^:]*$/ # no-namespace, starting with Q
|
|
|| $symbol =~ /^Phonon::/) { # or in the Phonon namespace
|
|
push @ret, $symbol;
|
|
} elsif (defined($publicclassregexp)) {
|
|
push @ret, $symbol if ($symbol =~ $publicclassregexp);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return @ret;
|
|
}
|
|
|
|
sub make_path {
|
|
my ($dir, $lib, $be_verbose) = @_;
|
|
unless(-e $dir) {
|
|
mkpath $dir;
|
|
$dir = "<outbase>" . substr($dir, length($out_basedir)) if ($be_verbose < 3);
|
|
print "$lib: mkpath $dir\n" if ($be_verbose > 1);
|
|
}
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: syncHeader(header, iheader, copy, timestamp)
|
|
# Params: header, string, filename to create "symlink" for
|
|
# iheader, string, destination name of symlink
|
|
# copy, forces header to be a copy of iheader
|
|
# timestamp, the requested modification time if copying
|
|
#
|
|
# Purpose: Syncronizes header to iheader
|
|
# Returns: 1 if successful, else 0.
|
|
######################################################################
|
|
sub syncHeader {
|
|
my ($lib, $header, $iheader, $copy, $ts) = @_;
|
|
$iheader =~ s=\\=/=g;
|
|
$header =~ s=\\=/=g;
|
|
return copyFile($lib, $iheader, $header) if($copy);
|
|
|
|
unless(-e $header) {
|
|
my $header_dir = dirname($header);
|
|
make_path($header_dir, $lib, $verbose_level);
|
|
|
|
#write it
|
|
my $iheader_out = fixPaths($iheader, $header_dir);
|
|
open(HEADER, ">$header") || die "Could not open $header for writing: $!\n";
|
|
print HEADER "#include \"$iheader_out\"\n";
|
|
close HEADER;
|
|
if(defined($ts)) {
|
|
utime(time, $ts, $header) or die "$iheader, $header";
|
|
}
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: fixPaths(file, dir)
|
|
# Params: file, string, filepath to be made relative to dir
|
|
# dir, string, dirpath for point of origin
|
|
#
|
|
# Purpose: file is made relative (if possible) of dir.
|
|
# Returns: String with the above applied conversion.
|
|
######################################################################
|
|
sub fixPaths {
|
|
my ($file, $dir) = @_;
|
|
$file =~ s=\\=/=g;
|
|
$dir =~ s=\\=/=g;
|
|
|
|
#setup
|
|
my $ret = $file;
|
|
$ret =~ s,/cygdrive/([a-zA-Z])/,$1:/,g;
|
|
my $file_dir = dirname($file);
|
|
if($file_dir eq ".") {
|
|
$file_dir = getcwd();
|
|
$file_dir =~ s=\\=/=g;
|
|
}
|
|
$file_dir =~ s,/cygdrive/([a-zA-Z])/,$1:/,g;
|
|
if($dir eq ".") {
|
|
$dir = getcwd();
|
|
$dir =~ s=\\=/=g;
|
|
}
|
|
$dir =~ s,/cygdrive/([a-zA-Z])/,$1:/,g;
|
|
return basename($file) if($file_dir eq $dir);
|
|
|
|
#guts
|
|
while ($file_dir =~ s,/[^/]+/\.\./,/,) {}
|
|
while ($dir =~ s,/[^/]+/\.\./,/,) {}
|
|
my $match_dir = 0;
|
|
for(my $i = 1; $i < length($file_dir); $i++) {
|
|
my $slash = index($file_dir, "/", $i);
|
|
last if($slash == -1);
|
|
my $tmp = substr($file_dir, 0, $slash);
|
|
last unless($dir =~ m,^\Q$tmp\E/,);
|
|
$match_dir = $tmp;
|
|
$i = $slash;
|
|
}
|
|
if($match_dir) {
|
|
my $after = substr($dir, length($match_dir));
|
|
my $count = ($after =~ tr,/,,);
|
|
my $dots = "";
|
|
for(my $i = 0; $i < $count; $i++) {
|
|
$dots .= "../";
|
|
}
|
|
$ret =~ s,^\Q$match_dir\E,$dots,;
|
|
}
|
|
$ret =~ s,/+,/,g;
|
|
return $ret;
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: fileContents(filename)
|
|
# Params: filename, string, filename of file to return contents
|
|
#
|
|
# Purpose: Get the contents of a file.
|
|
# Returns: String with contents of the file, or empty string if file
|
|
# doens't exist.
|
|
# Warning: Dies if it does exist but script cannot get read access.
|
|
######################################################################
|
|
sub fileContents {
|
|
my ($filename) = @_;
|
|
my $filecontents = "";
|
|
if (-e $filename) {
|
|
open(I, "< $filename") || die "Could not open $filename for reading, read block?";
|
|
local $/;
|
|
binmode I;
|
|
$filecontents = <I>;
|
|
close I;
|
|
}
|
|
return $filecontents;
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: fileCompare(file1, file2)
|
|
# Params: file1, string, filename of first file
|
|
# file2, string, filename of second file
|
|
#
|
|
# Purpose: Determines if files are equal, and which one is newer.
|
|
# Returns: 0 if files are equal no matter the timestamp, -1 if file1
|
|
# is newer, 1 if file2 is newer.
|
|
######################################################################
|
|
sub fileCompare {
|
|
my ($file1, $file2) = @_;
|
|
my $file1contents = fileContents($file1);
|
|
my $file2contents = fileContents($file2);
|
|
if (! -e $file1) { return 1; }
|
|
if (! -e $file2) { return -1; }
|
|
return $file1contents ne $file2contents ? (stat($file2))[9] <=> (stat($file1))[9] : 0;
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: copyFile(file, ifile)
|
|
# Params: file, string, filename to create duplicate for
|
|
# ifile, string, destination name of duplicate
|
|
#
|
|
# Purpose: Keeps files in sync so changes in the newer file will be
|
|
# written to the other.
|
|
# Returns: 1 if files were synced, else 0.
|
|
# Warning: Dies if script cannot get write access.
|
|
######################################################################
|
|
sub copyFile
|
|
{
|
|
my ($lib, $file,$ifile, $copy,$knowdiff,$filecontents,$ifilecontents) = @_;
|
|
# Bi-directional synchronization
|
|
open( I, "< " . $file ) || die "Could not open $file for reading";
|
|
local $/;
|
|
binmode I;
|
|
$filecontents = <I>;
|
|
close I;
|
|
if ( open(I, "< " . $ifile) ) {
|
|
local $/;
|
|
binmode I;
|
|
$ifilecontents = <I>;
|
|
close I;
|
|
$copy = fileCompare($file, $ifile);
|
|
$knowdiff = 0,
|
|
} else {
|
|
$copy = -1;
|
|
$knowdiff = 1;
|
|
}
|
|
|
|
if ( $knowdiff || ($filecontents ne $ifilecontents) ) {
|
|
if ( $copy > 0 ) {
|
|
my $file_dir = dirname($file);
|
|
make_path($file_dir, $lib, $verbose_level);
|
|
open(O, "> " . $file) || die "Could not open $file for writing (no write permission?)";
|
|
local $/;
|
|
binmode O;
|
|
print O $ifilecontents;
|
|
close O;
|
|
utime time, (stat($ifile))[9], $file;
|
|
return 1;
|
|
} elsif ( $copy < 0 ) {
|
|
my $ifile_dir = dirname($ifile);
|
|
make_path($ifile_dir, $lib, $verbose_level);
|
|
open(O, "> " . $ifile) || die "Could not open $ifile for writing (no write permission?)";
|
|
local $/;
|
|
binmode O;
|
|
print O $filecontents;
|
|
close O;
|
|
utime time, (stat($file))[9], $ifile;
|
|
return 1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: symlinkFile(file, ifile)
|
|
# Params: file, string, filename to create "symlink" for
|
|
# ifile, string, destination name of symlink
|
|
#
|
|
# Purpose: File is symlinked to ifile (or copied if filesystem doesn't
|
|
# support symlink).
|
|
# Returns: 1 on success, else 0.
|
|
######################################################################
|
|
sub symlinkFile
|
|
{
|
|
my ($lib, $file, $ifile) = @_;
|
|
|
|
if ($isunix) {
|
|
print "$lib: symlink created for $file " if ($verbose_level);
|
|
if ( $force_relative && ($ifile =~ /^$quoted_basedir/)) {
|
|
my $t = getcwd();
|
|
my $c = -1;
|
|
my $p = "../";
|
|
$t =~ s-^$quoted_basedir/--;
|
|
$p .= "../" while( ($c = index( $t, "/", $c + 1)) != -1 );
|
|
$file =~ s-^$quoted_basedir/-$p-;
|
|
print " ($file)\n" if($verbose_level);
|
|
}
|
|
print "\n" if($verbose_level);
|
|
return symlink($file, $ifile);
|
|
}
|
|
return copyFile($lib, $file, $ifile);
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: findFiles(dir, match, descend)
|
|
# Params: dir, string, directory to search for name
|
|
# match, string, regular expression to match in dir
|
|
# descend, integer, 0 = non-recursive search
|
|
# 1 = recurse search into subdirectories
|
|
#
|
|
# Purpose: Finds files matching a regular expression.
|
|
# Returns: List of matching files.
|
|
#
|
|
# Examples:
|
|
# findFiles("/usr","\.cpp$",1) - finds .cpp files in /usr and below
|
|
# findFiles("/tmp","^#",0) - finds #* files in /tmp
|
|
######################################################################
|
|
sub findFiles {
|
|
my ($dir,$match,$descend) = @_;
|
|
my ($file,$p,@files);
|
|
local(*D);
|
|
$dir =~ s=\\=/=g;
|
|
($dir eq "") && ($dir = ".");
|
|
if ( opendir(D,$dir) ) {
|
|
if ( $dir eq "." ) {
|
|
$dir = "";
|
|
} else {
|
|
($dir =~ /\/$/) || ($dir .= "/");
|
|
}
|
|
foreach $file ( sort readdir(D) ) {
|
|
next if ( $file =~ /^\.\.?$/ );
|
|
$p = $file;
|
|
($file =~ /$match/) && (push @files, $p);
|
|
if ( $descend && -d $p && ! -l $p ) {
|
|
push @files, &findFiles($p,$match,$descend);
|
|
}
|
|
}
|
|
closedir(D);
|
|
}
|
|
return @files;
|
|
}
|
|
|
|
######################################################################
|
|
# Syntax: loadSyncProfile()
|
|
#
|
|
# Purpose: Locates the sync.profile.
|
|
# Returns: Hashmap of module name -> directory.
|
|
######################################################################
|
|
sub loadSyncProfile {
|
|
my ($srcbase, $outbase) = @_;
|
|
if ($verbose_level) {
|
|
print("<srcbase> = $$srcbase \n");
|
|
print("<outbase> = $$outbase \n");
|
|
}
|
|
|
|
my $syncprofile = "$$srcbase/sync.profile";
|
|
my $result;
|
|
unless ($result = do "$syncprofile") {
|
|
die "syncqt couldn't parse $syncprofile: $@" if $@;
|
|
die "syncqt couldn't execute $syncprofile: $!" unless defined $result;
|
|
}
|
|
return $result;
|
|
}
|
|
|
|
sub locateSyncProfile
|
|
{
|
|
my ($directory) = @_;
|
|
$directory = abs_path($directory);
|
|
while (1) {
|
|
my $file = $directory."/sync.profile";
|
|
return $file if (-e $file);
|
|
my $odir = $directory;
|
|
$directory = dirname($directory);
|
|
return undef if ($directory eq $odir);
|
|
}
|
|
}
|
|
|
|
# check if this is an in-source build, and if so use that as the basedir too
|
|
$basedir = locateSyncProfile($out_basedir);
|
|
if ($basedir) {
|
|
$basedir = dirname($basedir) ;
|
|
$basedir =~ s=\\=/=g;
|
|
$quoted_basedir = "\Q$basedir";
|
|
}
|
|
|
|
# --------------------------------------------------------------------
|
|
# "main" function
|
|
# --------------------------------------------------------------------
|
|
|
|
while ( @ARGV ) {
|
|
my $var = 0;
|
|
my $val = 0;
|
|
|
|
#parse
|
|
my $arg = shift @ARGV;
|
|
if ($arg eq "-h" || $arg eq "-help" || $arg eq "-?" || $arg eq "?") {
|
|
$var = "show_help";
|
|
$val = "yes";
|
|
} elsif($arg eq "-copy") {
|
|
$var = "copy";
|
|
$val = "yes";
|
|
} elsif($arg eq "-o" || $arg eq "-outdir") {
|
|
$var = "output";
|
|
$val = shift @ARGV;
|
|
} elsif($arg eq "-showonly" || $arg eq "-remove-stale" || $arg eq "-windows" ||
|
|
$arg eq "-relative" || $arg eq "-check-includes") {
|
|
$var = substr($arg, 1);
|
|
$val = "yes";
|
|
} elsif($arg eq "-module-fwd") {
|
|
$var = "module_fwd";
|
|
$val = shift @ARGV;
|
|
} elsif($arg eq "-cache-module-fwd") {
|
|
$var = "cache_module_fwd";
|
|
$val = "yes";
|
|
} elsif($arg eq "-developer-build") {
|
|
$var = "developer_build";
|
|
$val = "yes";
|
|
} elsif($arg eq "-no-module-version-header") {
|
|
$var = "no_module_version_header";
|
|
$val = "yes";
|
|
} elsif($arg =~ /^-no-(.*)$/) {
|
|
$var = $1;
|
|
$val = "no";
|
|
#these are for commandline compat
|
|
} elsif($arg eq "-inc") {
|
|
$var = "output";
|
|
$val = shift @ARGV;
|
|
} elsif($arg eq "-module") {
|
|
$var = "module";
|
|
$val = shift @ARGV;
|
|
} elsif($arg eq "-separate-module") {
|
|
$var = "separate-module";
|
|
$val = shift @ARGV;
|
|
} elsif($arg eq "-show") {
|
|
$var = "showonly";
|
|
$val = "yes";
|
|
} elsif($arg eq "-quiet") {
|
|
$var = "verbose";
|
|
$val = "0";
|
|
} elsif($arg eq "-v") {
|
|
$var = "verbose";
|
|
$val = "yes";
|
|
} elsif($arg eq "-verbose") {
|
|
$var = "verbose";
|
|
$val = shift @ARGV;
|
|
} elsif($arg eq "-private") {
|
|
$var = "create_private_headers";
|
|
$val = "yes";
|
|
} elsif($arg eq "-qtdir") {
|
|
$var = "qtdir";
|
|
$val = shift @ARGV;
|
|
} elsif($arg eq "-generator") {
|
|
$var = "makefile_generator";
|
|
$val = shift @ARGV;
|
|
} elsif($arg =~/^-/) {
|
|
print "Unknown option: $arg\n\n" if(!$var);
|
|
showUsage();
|
|
} else {
|
|
$basedir = locateSyncProfile($arg);
|
|
die "Could not find a sync.profile for '$arg'\n" if (!$basedir);
|
|
$basedir = dirname($basedir);
|
|
$basedir =~ s=\\=/=g;
|
|
$quoted_basedir = "\Q$basedir";
|
|
$var = "ignore";
|
|
}
|
|
|
|
#do something
|
|
if(!$var || $var eq "show_help") {
|
|
print "Unknown option: $arg\n\n" if(!$var);
|
|
showUsage();
|
|
} elsif ($var eq "copy") {
|
|
if($val eq "yes") {
|
|
$copy_headers++;
|
|
} elsif($showonly) {
|
|
$copy_headers--;
|
|
}
|
|
} elsif ($var eq "showonly") {
|
|
if($val eq "yes") {
|
|
$showonly++;
|
|
} elsif($showonly) {
|
|
$showonly--;
|
|
}
|
|
} elsif ($var eq "verbose") {
|
|
if($val eq "yes") {
|
|
$verbose_level++;
|
|
} elsif($val eq "no" && $verbose_level) {
|
|
$verbose_level--;
|
|
} else {
|
|
$verbose_level = int($val);
|
|
}
|
|
} elsif ($var eq "check-includes") {
|
|
if($val eq "yes") {
|
|
$check_includes++;
|
|
} elsif($check_includes) {
|
|
$check_includes--;
|
|
}
|
|
} elsif ($var eq "remove-stale") {
|
|
if($val eq "yes") {
|
|
$remove_stale++;
|
|
} elsif($remove_stale) {
|
|
$remove_stale--;
|
|
}
|
|
} elsif ($var eq "windows") {
|
|
if($val eq "yes") {
|
|
$force_win++;
|
|
} elsif($force_win) {
|
|
$force_win--;
|
|
}
|
|
} elsif ($var eq "relative") {
|
|
if($val eq "yes") {
|
|
$force_relative++;
|
|
} elsif($force_relative) {
|
|
$force_relative--;
|
|
}
|
|
} elsif ($var eq "module") {
|
|
print "module :$val:\n" if($verbose_level);
|
|
die "No such module: $val" unless(defined $modules{$val});
|
|
push @modules_to_sync, $val;
|
|
} elsif ($var eq "separate-module") {
|
|
my ($module, $prodir, $headerdir) = split(/:/, $val);
|
|
$modules{$module} = $prodir;
|
|
push @modules_to_sync, $module;
|
|
$moduleheaders{$module} = $headerdir;
|
|
$create_uic_class_map = 0;
|
|
} elsif ($var eq "qtdir") {
|
|
if($val) {
|
|
$qtbasedir = $val;
|
|
$qtbasedir =~ s=\\=/=g;
|
|
} else {
|
|
die "The -qtdir option requires an argument";
|
|
}
|
|
} elsif ($var eq "module_fwd") {
|
|
$module_fwd = $val;
|
|
} elsif ($var eq "cache_module_fwd") {
|
|
$cache_module_fwd = 1;
|
|
} elsif ($var eq "developer_build") {
|
|
$developer_build = 1;
|
|
} elsif ($var eq "makefile_generator") {
|
|
$makefile_generator = $val;
|
|
} elsif ($var eq "no_module_version_header") {
|
|
$no_module_version_header = 1;
|
|
} elsif ($var eq "output") {
|
|
my $outdir = $val;
|
|
if(checkRelative($outdir)) {
|
|
$out_basedir = getcwd();
|
|
chomp $out_basedir;
|
|
$out_basedir .= "/" . $outdir;
|
|
} else {
|
|
$out_basedir = $outdir;
|
|
}
|
|
# \ -> /
|
|
$out_basedir =~ s=\\=/=g;
|
|
}
|
|
}
|
|
|
|
die "Cannot automatically detect/use provided path to QtBase's build directory!\n" .
|
|
"QTDIR detected/provided: " . (defined $qtbasedir ? $qtbasedir : "-none-") . "\n" .
|
|
"Please use the -qtdir option to provide the correct path.\nsyncqt failed"
|
|
if (!$qtbasedir || !-d "$qtbasedir/mkspecs");
|
|
|
|
# if we have no $basedir we cannot be sure which sources you want, so die
|
|
die "Could not find any sync.profile for your module!\nPass <module directory> to syncqt to sync your header files.\nsyncqt failed" if (!$basedir);
|
|
|
|
my $class_lib_map_contents = "";
|
|
our @ignore_headers = ();
|
|
our @ignore_for_master_contents = ();
|
|
our @ignore_for_include_check = ();
|
|
our @ignore_for_qt_begin_header_check = ();
|
|
our @ignore_for_qt_begin_namespace_check = ();
|
|
our @ignore_for_qt_module_check = ();
|
|
my %colliding_headers = ();
|
|
my %inject_headers = ( "$basedir/src/corelib/global" => ( "qconfig.h" ) ); # all from build dir
|
|
|
|
# load the module's sync.profile here, before we can
|
|
loadSyncProfile(\$basedir, \$out_basedir);
|
|
|
|
@modules_to_sync = keys(%modules) if($#modules_to_sync == -1);
|
|
|
|
my %allmoduleheadersprivate = map { $_ => 1 } @allmoduleheadersprivate;
|
|
|
|
$isunix = checkUnix; #cache checkUnix
|
|
|
|
foreach my $lib (@modules_to_sync) {
|
|
#iteration info
|
|
my $dir = $modules{$lib};
|
|
my $module_version = "";
|
|
my $module_major_version = 0;
|
|
my $module_minor_version = 0;
|
|
my $module_patch_version = 0;
|
|
|
|
if (-e "$modulepris{$lib}") {
|
|
my $content = fileContents($modulepris{$lib});
|
|
my @version_rows = grep(/QT\..*\.VERSION/, split('\n', $content));
|
|
if(@version_rows) {
|
|
# We only pick the first one, since each module need a separate .pri file
|
|
$module_version = $version_rows[0];
|
|
chomp $module_version;
|
|
$module_version =~ s/^\s*QT\..*\.VERSION\s*=\s*([^#]+).*$/$1/;
|
|
$module_version =~ s/\s+$//;
|
|
my @versions = split(/\./, $module_version);
|
|
$module_major_version = int($versions[0]);
|
|
chomp $module_major_version;
|
|
$module_minor_version = int($versions[1]);
|
|
chomp $module_minor_version;
|
|
$module_patch_version = int($versions[2]);
|
|
chomp $module_patch_version;
|
|
}
|
|
}
|
|
print "$lib: WARNING: Module\'s pri missing QT.<module>.VERSION variable! Private headers not versioned!\n" if (!$module_version);
|
|
|
|
my $pathtoheaders = "";
|
|
$pathtoheaders = $moduleheaders{$lib} if ($moduleheaders{$lib});
|
|
|
|
my $allheadersprivate = 0;
|
|
$allheadersprivate = 1 if $allmoduleheadersprivate{$lib};
|
|
|
|
#information used after the syncing
|
|
my $pri_install_classes = "";
|
|
my $pri_install_files = "";
|
|
my $pri_install_pfiles = "";
|
|
|
|
my $libcapitals = $lib;
|
|
$libcapitals =~ y/a-z/A-Z/;
|
|
my $master_contents = "#ifndef QT_".$libcapitals."_MODULE_H\n#define QT_".$libcapitals."_MODULE_H\n";
|
|
|
|
#get dependencies
|
|
if(-e "$dir/" . basename($dir) . ".pro") {
|
|
if(open(F, "<$dir/" . basename($dir) . ".pro")) {
|
|
while(my $line = <F>) {
|
|
chomp $line;
|
|
if($line =~ /^ *QT *\+?= *([^\r\n]*)/) {
|
|
foreach(split(/ /, $1)) {
|
|
$_ =~ s/-private$//;
|
|
my $content = $mastercontent{$_};
|
|
$master_contents .= $content if ($content);
|
|
}
|
|
}
|
|
}
|
|
close(F);
|
|
}
|
|
}
|
|
|
|
#remove the old files
|
|
if($remove_stale) {
|
|
my @subdirs = ("$out_basedir/include/$lib");
|
|
foreach my $subdir (@subdirs) {
|
|
if (opendir DIR, $subdir) {
|
|
foreach my $t (sort readdir(DIR)) {
|
|
my $file = "$subdir/$t";
|
|
if(-d $file) {
|
|
push @subdirs, $file unless($t eq "." || $t eq "..");
|
|
} else {
|
|
my @files = ($file);
|
|
#push @files, "$out_basedir/include/Qt/$t" if(-e "$out_basedir/include/Qt/$t");
|
|
foreach my $file (@files) {
|
|
my $remove_file = 0;
|
|
if(open(F, "<$file")) {
|
|
while(my $line = <F>) {
|
|
chomp $line;
|
|
if($line =~ /^\#include \"([^\"]*)\"$/) {
|
|
my $include = $1;
|
|
$include = $subdir . "/" . $include unless(substr($include, 0, 1) eq "/");
|
|
$remove_file = 1 unless(-e $include);
|
|
} else {
|
|
$remove_file = 0;
|
|
last;
|
|
}
|
|
}
|
|
close(F);
|
|
unlink $file if($remove_file);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
closedir DIR;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
# create the version header files for each module
|
|
unless ($no_module_version_header) {
|
|
my $modulepri = $modulepris{$lib};
|
|
if (-e $modulepri) {
|
|
my $modulepriname = basename($modulepri);
|
|
# FIXME: this creates a file in the source location for shadow-builds
|
|
my $moduleversionheader = "$modules{$lib}/" . lc($lib) . "version.h";
|
|
my $modulehexstring = sprintf("0x%02X%02X%02X", $module_major_version, $module_minor_version, $module_patch_version);
|
|
open MODULE_VERSION_HEADER_FILE, ">$moduleversionheader" or die "Can't open $moduleversionheader for writing";
|
|
print MODULE_VERSION_HEADER_FILE "/* This file was generated by syncqt with the info from sync.profile. */\n";
|
|
print MODULE_VERSION_HEADER_FILE "#ifndef QT_". uc($lib) . "_VERSION_H\n";
|
|
print MODULE_VERSION_HEADER_FILE "#define QT_". uc($lib) . "_VERSION_H\n";
|
|
print MODULE_VERSION_HEADER_FILE "\n";
|
|
print MODULE_VERSION_HEADER_FILE "#define " .uc($lib) . "_VERSION_STR \"" . $module_version . "\"\n";
|
|
print MODULE_VERSION_HEADER_FILE "\n";
|
|
print MODULE_VERSION_HEADER_FILE "#define " .uc($lib) . "_VERSION $modulehexstring\n", ;
|
|
print MODULE_VERSION_HEADER_FILE "\n";
|
|
print MODULE_VERSION_HEADER_FILE "#endif // QT_". uc($lib) . "_VERSION_H\n";
|
|
close MODULE_VERSION_HEADER_FILE;
|
|
$moduleversionheader = "<srcbase>" . substr($moduleversionheader, length($basedir)) if ($verbose_level < 2);
|
|
print "$lib: created version header $moduleversionheader\n" if($verbose_level);
|
|
} elsif ($modulepri) {
|
|
print "$lib: WARNING: Module\'s pri file '$modulepri' not found.\n$lib: Skipped creating module version header.\n";
|
|
}
|
|
}
|
|
|
|
#create the new ones
|
|
foreach my $current_dir (split(/;/, $dir)) {
|
|
my @headers_paths = split(/;/, $pathtoheaders);
|
|
if (@headers_paths) {
|
|
@headers_paths = map { "$current_dir/$_" } @headers_paths;
|
|
} else {
|
|
push @headers_paths, $current_dir;
|
|
}
|
|
|
|
foreach my $headers_dir (@headers_paths) {
|
|
#calc subdirs
|
|
my @subdirs = ($headers_dir);
|
|
foreach my $subdir (@subdirs) {
|
|
if ($subdir =~ /\/doc$/) {
|
|
next;
|
|
}
|
|
opendir DIR, $subdir or next;
|
|
foreach my $t (sort readdir(DIR)) {
|
|
push @subdirs, "$subdir/$t" if(-d "$subdir/$t" && !($t eq ".") &&
|
|
!($t eq "..") && !($t eq ".obj") &&
|
|
!($t eq ".moc") && !($t eq ".rcc") &&
|
|
!($t eq ".uic") && !($t eq "build"));
|
|
}
|
|
closedir DIR;
|
|
}
|
|
|
|
#calc files and "copy" them
|
|
foreach my $subdir (@subdirs) {
|
|
my @headers = findFiles($subdir, "^[-a-z0-9_]*\\.h\$" , 0);
|
|
if (defined $inject_headers{$subdir}) {
|
|
foreach my $if ($inject_headers{$subdir}) {
|
|
@headers = grep(!/^\Q$if\E$/, @headers); #in case we configure'd previously
|
|
push @headers, "*".$if;
|
|
}
|
|
}
|
|
my $header_dirname = "";
|
|
foreach my $header (@headers) {
|
|
my $shadow = ($header =~ s/^\*//);
|
|
$header = 0 if($header =~ /^ui_.*.h/);
|
|
foreach (@ignore_headers) {
|
|
$header = 0 if($header eq $_);
|
|
}
|
|
if($header) {
|
|
my $header_copies = 0;
|
|
#figure out if it is a public header
|
|
my $public_header = $header;
|
|
my $qpa_header = 0;
|
|
if($public_header =~ /^qplatform/) {
|
|
$public_header = 0;
|
|
$qpa_header = 1;
|
|
} elsif($allheadersprivate || $public_header =~ /_p.h$/ || $public_header =~ /_pch.h$/) {
|
|
$public_header = 0;
|
|
} else {
|
|
foreach (@ignore_for_master_contents) {
|
|
$public_header = 0 if($header eq $_);
|
|
}
|
|
}
|
|
|
|
my $iheader = $subdir . "/" . $header;
|
|
$iheader =~ s/^\Q$basedir\E/$out_basedir/ if ($shadow);
|
|
my @classes = $public_header ? classNames($iheader) : ();
|
|
if($showonly) {
|
|
print "$header [$lib]\n";
|
|
foreach(@classes) {
|
|
print "SYMBOL: $_\n";
|
|
}
|
|
} else {
|
|
my $ts = (stat($iheader))[9];
|
|
#find out all the places it goes..
|
|
my @headers;
|
|
if ($public_header) {
|
|
@headers = ( "$out_basedir/include/$lib/$header" );
|
|
foreach my $full_class (@classes) {
|
|
my $header_base = basename($header);
|
|
# Strip namespaces:
|
|
my $class = $full_class;
|
|
$class =~ s/^.*:://;
|
|
# if ($class =~ m/::/) {
|
|
# class =~ s,::,/,g;
|
|
# }
|
|
|
|
if (defined $explicitheaders{$lib}{$class}) {
|
|
$header_copies++ if(syncHeader($lib, "$out_basedir/include/$lib/$class", "$out_basedir/include/$lib/$explicitheaders{$lib}{$class}", 0, $ts));
|
|
} else {
|
|
$class_lib_map_contents .= "QT_CLASS_LIB($full_class, $lib, $header_base)\n";
|
|
$header_copies++ if(syncHeader($lib, "$out_basedir/include/$lib/$class", "$out_basedir/include/$lib/$header", 0, $ts));
|
|
}
|
|
|
|
# KDE-Compat headers for Phonon
|
|
if ($lib eq "phonon") {
|
|
$header_copies++ if (syncHeader($lib, "$out_basedir/include/phonon_compat/Phonon/$class", "$out_basedir/include/$lib/$header", 0, $ts));
|
|
}
|
|
}
|
|
|
|
if ($explicitheaders{$lib}{basename($header)}) {
|
|
$header_copies++ if(syncHeader($lib, "$out_basedir/include/$lib/$explicitheaders{$lib}{basename($header)}", "$out_basedir/include/$lib/$header", 0, $ts));
|
|
}
|
|
|
|
} elsif ($create_private_headers && !$qpa_header) {
|
|
if ($module_version) {
|
|
@headers = ( "$out_basedir/include/$lib/$module_version/$lib/private/$header" );
|
|
} else {
|
|
@headers = ( "$out_basedir/include/$lib/private/$header" );
|
|
}
|
|
} elsif ($create_private_headers) {
|
|
if ($module_version) {
|
|
@headers = ( "$out_basedir/include/$lib/$module_version/$lib/qpa/$header" );
|
|
} else {
|
|
@headers = ( "$out_basedir/include/$lib/qpa/$header" );
|
|
}
|
|
}
|
|
|
|
foreach(@headers) { #sync them
|
|
$header_copies++ if(syncHeader($lib, $_, $iheader, $copy_headers && !$shadow, $ts));
|
|
}
|
|
|
|
if($public_header) {
|
|
#put it into the master file
|
|
$master_contents .= "#include \"$public_header\"\n" if(shouldMasterInclude($iheader));
|
|
|
|
#deal with the install directives
|
|
if($public_header) {
|
|
my $pri_install_iheader = fixPaths($iheader, $current_dir);
|
|
foreach my $class (@classes) {
|
|
# Strip namespaces:
|
|
$class =~ s/^.*:://;
|
|
# if ($class =~ m/::/) {
|
|
# $class =~ s,::,/,g;
|
|
# }
|
|
my $class_header = fixPaths("$out_basedir/include/$lib/$class",
|
|
$current_dir) . " ";
|
|
$pri_install_classes .= $class_header
|
|
unless($pri_install_classes =~ $class_header);
|
|
}
|
|
if ($explicitheaders{$lib}{basename($iheader)}) {
|
|
my $compat_header = fixPaths("$out_basedir/include/$lib/$explicitheaders{$lib}{basename($iheader)}", $current_dir) . " ";
|
|
$pri_install_files .= $compat_header unless($pri_install_files =~ $compat_header);
|
|
}
|
|
$pri_install_files.= "$pri_install_iheader ";;
|
|
}
|
|
}
|
|
else {
|
|
my $pri_install_iheader = fixPaths($iheader, $current_dir);
|
|
$pri_install_pfiles.= "$pri_install_iheader ";;
|
|
}
|
|
}
|
|
|
|
if ($verbose_level && $header_copies) {
|
|
my $new_header_dirname = dirname($iheader);
|
|
$new_header_dirname = "<srcbase>" . substr($new_header_dirname, length($basedir)) if ($new_header_dirname && $verbose_level < 2);
|
|
my $header_base = basename($iheader);
|
|
if ($verbose_level < 3) {
|
|
my $line_prefix = ",";
|
|
if ($new_header_dirname ne $header_dirname) {
|
|
$line_prefix = "$lib: created fwd-include header(s) for $new_header_dirname/ {";
|
|
$line_prefix = " }\n".$line_prefix if ($header_dirname);
|
|
$header_dirname = $new_header_dirname;
|
|
} else {
|
|
$line_prefix = ",";
|
|
}
|
|
print "$line_prefix $header_base ($header_copies)";
|
|
} else { # $verbose_level >= 3
|
|
$iheader = "<srcbase>" . substr($iheader, length($basedir)) if ($verbose_level == 3);
|
|
print "$lib: created $header_copies fwd-include headers for $iheader\n";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
print " }\n" if ($header_dirname && $verbose_level > 0 && $verbose_level < 3);
|
|
}
|
|
}
|
|
}
|
|
|
|
# close the master include:
|
|
$master_contents .= "#endif\n";
|
|
|
|
unless($showonly) {
|
|
# create deprecated headers
|
|
my $first = 1;
|
|
while (my ($header, $include) = each %{$deprecatedheaders{$lib}}) {
|
|
my $public_header = 0;
|
|
$public_header = 1 unless ($allheadersprivate || ($header =~ /_p\.h$/));
|
|
next unless ($public_header || $create_private_headers);
|
|
|
|
my $header_path = "$out_basedir/include/$lib/";
|
|
unless ($public_header) {
|
|
if ($module_version) {
|
|
$header_path .= "$module_version/$lib/private/";
|
|
} else {
|
|
$header_path .= "private/";
|
|
}
|
|
}
|
|
$header_path .= "$header";
|
|
|
|
unless (-e $header_path) {
|
|
my $guard = "DEPRECATED_HEADER_" . $lib . "_" . $header;
|
|
$guard =~ s/([^a-zA-Z0-9_])/_/g;
|
|
|
|
my $header_dir = dirname($header_path);
|
|
make_path($header_dir, $lib, $verbose_level);
|
|
|
|
open(HEADER, ">$header_path") || die "Could not open $header_path for writing: $!\n";
|
|
print HEADER "#ifndef $guard\n";
|
|
print HEADER "#define $guard\n";
|
|
my $warning = "Header <$lib/";
|
|
$warning .= "private/" unless ($public_header);
|
|
$warning .= "$header> is deprecated. Please include <$include> instead.";
|
|
print HEADER "#if defined(__GNUC__)\n";
|
|
print HEADER "# warning $warning\n";
|
|
print HEADER "#elif defined(_MSC_VER)\n";
|
|
print HEADER "# pragma message (\"$warning\")\n";
|
|
print HEADER "#endif\n";
|
|
print HEADER "#include <$include>\n";
|
|
if ($public_header) {
|
|
print HEADER "#if 0\n";
|
|
print HEADER "#pragma qt_no_master_include\n";
|
|
print HEADER "#endif\n";
|
|
}
|
|
print HEADER "#endif\n";
|
|
close HEADER;
|
|
|
|
if ($verbose_level < 3) {
|
|
my $line_prefix = ",";
|
|
$line_prefix = "$lib: created deprecated header(s) {" if ($first);
|
|
print "$line_prefix $header";
|
|
} else {
|
|
print "$lib: created deprecated header $header => $include\n";
|
|
}
|
|
|
|
my $addendum = fixPaths($header_path, $dir) . " ";
|
|
if ($public_header) {
|
|
$pri_install_files .= $addendum;
|
|
} else {
|
|
$pri_install_pfiles .= $addendum;
|
|
}
|
|
$first = 0;
|
|
}
|
|
}
|
|
if ($verbose_level < 3) {
|
|
print " }\n" unless ($first);
|
|
}
|
|
|
|
my @master_includes;
|
|
push @master_includes, "$out_basedir/include/$lib/$lib";
|
|
push @master_includes, "$out_basedir/include/phonon_compat/Phonon/Phonon" if ($lib eq "phonon");
|
|
foreach my $master_include (@master_includes) {
|
|
#generate the "master" include file
|
|
my @tmp = split(/;/,$modules{$lib});
|
|
$pri_install_files .= fixPaths($master_include, $tmp[0]) . " "; #get the master file installed too
|
|
if($master_include && -e $master_include) {
|
|
open MASTERINCLUDE, "<$master_include";
|
|
local $/;
|
|
binmode MASTERINCLUDE;
|
|
my $oldmaster = <MASTERINCLUDE>;
|
|
close MASTERINCLUDE;
|
|
$oldmaster =~ s/\r//g; # remove \r's , so comparison is ok on all platforms
|
|
$master_include = 0 if($oldmaster eq $master_contents);
|
|
}
|
|
if($master_include && $master_contents) {
|
|
my $master_dir = dirname($master_include);
|
|
make_path($master_dir, $lib, $verbose_level);
|
|
open MASTERINCLUDE, ">$master_include";
|
|
print MASTERINCLUDE $master_contents;
|
|
close MASTERINCLUDE;
|
|
print "$lib: created header (master) file\n" if($verbose_level);
|
|
}
|
|
}
|
|
|
|
#handle the headers.pri for each module
|
|
my $headers_pri_contents = "";
|
|
$headers_pri_contents .= "SYNCQT.HEADER_FILES = $pri_install_files\n";
|
|
$headers_pri_contents .= "SYNCQT.HEADER_CLASSES = $pri_install_classes\n";
|
|
$headers_pri_contents .= "SYNCQT.PRIVATE_HEADER_FILES = $pri_install_pfiles\n";
|
|
my $headers_pri_file = "$out_basedir/include/$lib/headers.pri";
|
|
if(-e $headers_pri_file) {
|
|
open HEADERS_PRI_FILE, "<$headers_pri_file";
|
|
local $/;
|
|
binmode HEADERS_PRI_FILE;
|
|
my $old_headers_pri_contents = <HEADERS_PRI_FILE>;
|
|
close HEADERS_PRI_FILE;
|
|
$old_headers_pri_contents =~ s/\r//g; # remove \r's , so comparison is ok on all platforms
|
|
$headers_pri_file = 0 if($old_headers_pri_contents eq $headers_pri_contents);
|
|
}
|
|
if($headers_pri_file && $master_contents) {
|
|
my $headers_pri_dir = dirname($headers_pri_file);
|
|
make_path($headers_pri_dir, $lib, $verbose_level);
|
|
open HEADERS_PRI_FILE, ">$headers_pri_file";
|
|
print HEADERS_PRI_FILE $headers_pri_contents;
|
|
close HEADERS_PRI_FILE;
|
|
print "$lib: created headers.pri file\n" if($verbose_level);
|
|
}
|
|
|
|
# create forwarding module pri in qtbase/mkspecs/modules
|
|
if ($module_fwd) {
|
|
my $modulepri = $modulepris{$lib};
|
|
if (-e $modulepri) {
|
|
my $modulepriname = basename($modulepri);
|
|
make_path($module_fwd, $lib, $verbose_level);
|
|
my $moduleprifwd = "$module_fwd/$modulepriname";
|
|
my $mod_base = $basedir;
|
|
my $mod_component_base = $developer_build ? $qtbasedir : $out_basedir;
|
|
open MODULE_PRI_FILE, ">$moduleprifwd" or die("Could not open $moduleprifwd for writing");
|
|
print MODULE_PRI_FILE "QT_MODULE_BASE = $mod_base\n";
|
|
print MODULE_PRI_FILE "QT_MODULE_BIN_BASE = $mod_component_base/bin\n";
|
|
print MODULE_PRI_FILE "QT_MODULE_INCLUDE_BASE = $out_basedir/include\n";
|
|
print MODULE_PRI_FILE "QT_MODULE_IMPORT_BASE = $mod_component_base/imports\n";
|
|
print MODULE_PRI_FILE "QT_MODULE_LIB_BASE = $mod_component_base/lib\n";
|
|
print MODULE_PRI_FILE "QT_MODULE_PLUGIN_BASE = $mod_component_base/plugins\n";
|
|
print MODULE_PRI_FILE "include($modulepri)\n";
|
|
close MODULE_PRI_FILE;
|
|
utime(time, (stat($modulepri))[9], $moduleprifwd);
|
|
if ($cache_module_fwd) {
|
|
my $cacheStatement = "QMAKE_EXTRA_MODULE_FORWARDS = \"$module_fwd\"";
|
|
my $cacheFile = "$out_basedir/.qmake.cache";
|
|
my $existingQmakeCache = fileContents($cacheFile);
|
|
# Skip if it's already there.
|
|
if ($existingQmakeCache !~ $cacheStatement) {
|
|
open QMAKE_CACHE_FILE, ">>$cacheFile" or die("Could not open $cacheFile for writing");
|
|
print QMAKE_CACHE_FILE "$cacheStatement\n";
|
|
close(QMAKE_CACHE_FILE);
|
|
}
|
|
}
|
|
} elsif ($modulepri) {
|
|
print "$lib: WARNING: Module\'s pri file '$modulepri' not found.\n$lib: Skipped creating forwarding pri.\n";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
unless($showonly || !$create_uic_class_map) {
|
|
my $class_lib_map = "$out_basedir/src/tools/uic/qclass_lib_map.h";
|
|
if(-e $class_lib_map) {
|
|
open CLASS_LIB_MAP, "<$class_lib_map";
|
|
local $/;
|
|
binmode CLASS_LIB_MAP;
|
|
my $old_class_lib_map_contents = <CLASS_LIB_MAP>;
|
|
close CLASS_LIB_MAP;
|
|
$old_class_lib_map_contents =~ s/\r//g; # remove \r's , so comparison is ok on all platforms
|
|
$class_lib_map = 0 if($old_class_lib_map_contents eq $class_lib_map_contents);
|
|
}
|
|
if($class_lib_map) {
|
|
my $class_lib_map_dir = dirname($class_lib_map);
|
|
make_path($class_lib_map_dir, "<outdir>", $verbose_level);
|
|
open CLASS_LIB_MAP, ">$class_lib_map";
|
|
print CLASS_LIB_MAP $class_lib_map_contents;
|
|
close CLASS_LIB_MAP;
|
|
}
|
|
}
|
|
|
|
if($check_includes) {
|
|
for my $lib (keys(%modules)) {
|
|
#calc subdirs
|
|
my @subdirs = ($modules{$lib});
|
|
foreach my $subdir (@subdirs) {
|
|
opendir DIR, $subdir or die "Huh, directory ".$subdir." cannot be opened.";
|
|
foreach my $t (sort readdir(DIR)) {
|
|
push @subdirs, "$subdir/$t" if(-d "$subdir/$t" && !($t eq ".") &&
|
|
!($t eq "..") && !($t eq ".obj") &&
|
|
!($t eq ".moc") && !($t eq ".rcc") &&
|
|
!($t eq ".uic") && !($t eq "build"));
|
|
}
|
|
closedir DIR;
|
|
}
|
|
|
|
foreach my $subdir (@subdirs) {
|
|
my $header_skip_qt_module_test = 0;
|
|
foreach(@ignore_for_qt_module_check) {
|
|
foreach (split(/;/, $_)) {
|
|
$header_skip_qt_module_test = 1 if ($subdir =~ /^$_/);
|
|
}
|
|
}
|
|
my @headers = findFiles($subdir, "^[-a-z0-9_]*\\.h\$" , 0);
|
|
foreach my $header (@headers) {
|
|
my $header_skip_qt_begin_header_test = 0;
|
|
my $header_skip_qt_begin_namespace_test = 0;
|
|
$header = 0 if($header =~ /^ui_.*.h/);
|
|
foreach (@ignore_headers) {
|
|
$header = 0 if($header eq $_);
|
|
}
|
|
if($header) {
|
|
my $public_header = $header;
|
|
if($public_header =~ /_p.h$/ || $public_header =~ /_pch.h$/) {
|
|
$public_header = 0;
|
|
} elsif ($public_header =~ /^qplatform/) {
|
|
$public_header = 0;
|
|
} else {
|
|
foreach (@ignore_for_master_contents) {
|
|
$public_header = 0 if($header eq $_);
|
|
}
|
|
if($public_header) {
|
|
foreach (@ignore_for_include_check) {
|
|
$public_header = 0 if($header eq $_);
|
|
}
|
|
foreach(@ignore_for_qt_begin_header_check) {
|
|
$header_skip_qt_begin_header_test = 1 if ($header eq $_);
|
|
}
|
|
foreach(@ignore_for_qt_begin_namespace_check) {
|
|
$header_skip_qt_begin_namespace_test = 1 if ($header eq $_);
|
|
}
|
|
}
|
|
}
|
|
|
|
my $iheader = $subdir . "/" . $header;
|
|
if($public_header) {
|
|
if(open(F, "<$iheader")) {
|
|
my $qt_begin_header_found = 0;
|
|
my $qt_end_header_found = 0;
|
|
my $qt_begin_namespace_found = 0;
|
|
my $qt_end_namespace_found = 0;
|
|
my $line;
|
|
while($line = <F>) {
|
|
chomp $line;
|
|
my $output_line = 1;
|
|
if($line =~ /^ *\# *pragma (qt_no_included_check|qt_sync_stop_processing)/) {
|
|
last;
|
|
} elsif($line =~ /^ *\# *include/) {
|
|
my $include = $line;
|
|
if($line =~ /<.*>/) {
|
|
$include =~ s,.*<(.*)>.*,$1,;
|
|
} elsif($line =~ /".*"/) {
|
|
$include =~ s,.*"(.*)".*,$1,;
|
|
} else {
|
|
$include = 0;
|
|
}
|
|
if($include) {
|
|
for my $trylib (keys(%modules)) {
|
|
if(-e "$out_basedir/include/$trylib/$include") {
|
|
print "$lib: WARNING: $iheader includes $include when it should include $trylib/$include\n";
|
|
}
|
|
}
|
|
}
|
|
} elsif ($header_skip_qt_begin_header_test == 0 and $line =~ /^QT_BEGIN_HEADER\s*$/) {
|
|
$qt_begin_header_found = 1;
|
|
} elsif ($header_skip_qt_begin_header_test == 0 and $line =~ /^QT_END_HEADER\s*$/) {
|
|
$qt_end_header_found = 1;
|
|
} elsif ($header_skip_qt_begin_namespace_test == 0 and $line =~ /^QT_BEGIN_NAMESPACE\s*$/) {
|
|
$qt_begin_namespace_found = 1;
|
|
} elsif ($header_skip_qt_begin_namespace_test == 0 and $line =~ /^QT_END_NAMESPACE\s*$/) {
|
|
$qt_end_namespace_found = 1;
|
|
}
|
|
}
|
|
if ($header_skip_qt_begin_header_test == 0) {
|
|
if ($qt_begin_header_found == 0) {
|
|
print "$lib: WARNING: $iheader does not include QT_BEGIN_HEADER\n";
|
|
}
|
|
|
|
if ($qt_begin_header_found && $qt_end_header_found == 0) {
|
|
print "$lib: WARNING: $iheader has QT_BEGIN_HEADER but no QT_END_HEADER\n";
|
|
}
|
|
}
|
|
|
|
if ($header_skip_qt_begin_namespace_test == 0) {
|
|
if ($qt_begin_namespace_found == 0) {
|
|
print "$lib: WARNING: $iheader does not include QT_BEGIN_NAMESPACE\n";
|
|
}
|
|
|
|
if ($qt_begin_namespace_found && $qt_end_namespace_found == 0) {
|
|
print "$lib: WARNING: $iheader has QT_BEGIN_NAMESPACE but no QT_END_NAMESPACE\n";
|
|
}
|
|
}
|
|
|
|
close(F);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# Do configure tests now (pass some things along)
|
|
# fatal tests have a non zero return
|
|
# If the generator is not set (e.g. someone invoking syncqt as part of configure etc, then don't run tests either)
|
|
unless ($showonly || $makefile_generator eq '') {
|
|
my $configtests = dirname($0)."/qtmodule-configtests";
|
|
if (! -f $configtests) {
|
|
$configtests = $qtbasedir."/bin/qtmodule-configtests";
|
|
}
|
|
if (! -f $configtests) {
|
|
warn "Unable to locate qtmodule-configtests script - config tests disabled.\n";
|
|
} else {
|
|
if (system($EXECUTABLE_NAME, $configtests, $basedir, $out_basedir, $qtbasedir, $makefile_generator)) {
|
|
die "$configtests exited with status $?";
|
|
}
|
|
}
|
|
}
|
|
|
|
exit 0;
|