You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
881 lines
34 KiB
881 lines
34 KiB
/****************************************************************************
|
|
**
|
|
** Implementation of UnixMakefileGenerator class.
|
|
**
|
|
** Copyright (C) 1992-2008 Trolltech ASA. All rights reserved.
|
|
**
|
|
** This file is part of qmake.
|
|
**
|
|
** This file may be used under the terms of the GNU General
|
|
** Public License versions 2.0 or 3.0 as published by the Free
|
|
** Software Foundation and appearing in the files LICENSE.GPL2
|
|
** and LICENSE.GPL3 included in the packaging of this file.
|
|
** Alternatively you may (at your option) use any later version
|
|
** of the GNU General Public License if such license has been
|
|
** publicly approved by Trolltech ASA (or its successors, if any)
|
|
** and the KDE Free Qt Foundation.
|
|
**
|
|
** Please review the following information to ensure GNU General
|
|
** Public Licensing requirements will be met:
|
|
** http://trolltech.com/products/qt/licenses/licensing/opensource/.
|
|
** If you are unsure which license is appropriate for your use, please
|
|
** review the following information:
|
|
** http://trolltech.com/products/qt/licenses/licensing/licensingoverview
|
|
** or contact the sales department at sales@trolltech.com.
|
|
**
|
|
** This file may be used under the terms of the Q Public License as
|
|
** defined by Trolltech ASA and appearing in the file LICENSE.QPL
|
|
** included in the packaging of this file. Licensees holding valid Qt
|
|
** Commercial licenses may use this file in accordance with the Qt
|
|
** Commercial License Agreement provided with the Software.
|
|
**
|
|
** This file is provided "AS IS" with NO WARRANTY OF ANY KIND,
|
|
** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR
|
|
** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted
|
|
** herein.
|
|
**
|
|
**********************************************************************/
|
|
|
|
#include "unixmake.h"
|
|
#include "option.h"
|
|
#include <qregexp.h>
|
|
#include <qfile.h>
|
|
#include <qdict.h>
|
|
#include <qdir.h>
|
|
#include <time.h>
|
|
|
|
|
|
void
|
|
UnixMakefileGenerator::init()
|
|
{
|
|
if(init_flag)
|
|
return;
|
|
init_flag = TRUE;
|
|
|
|
if(!project->isEmpty("QMAKE_FAILED_REQUIREMENTS")) /* no point */
|
|
return;
|
|
|
|
QStringList &configs = project->variables()["CONFIG"];
|
|
/* this should probably not be here, but I'm using it to wrap the .t files */
|
|
if(project->first("TEMPLATE") == "app")
|
|
project->variables()["QMAKE_APP_FLAG"].append("1");
|
|
else if(project->first("TEMPLATE") == "lib")
|
|
project->variables()["QMAKE_LIB_FLAG"].append("1");
|
|
else if(project->first("TEMPLATE") == "subdirs") {
|
|
MakefileGenerator::init();
|
|
if(project->isEmpty("MAKEFILE"))
|
|
project->variables()["MAKEFILE"].append("Makefile");
|
|
if(project->isEmpty("QMAKE"))
|
|
project->variables()["QMAKE"].append("qmake");
|
|
if(project->variables()["QMAKE_INTERNAL_QMAKE_DEPS"].findIndex("qmake_all") == -1)
|
|
project->variables()["QMAKE_INTERNAL_QMAKE_DEPS"].append("qmake_all");
|
|
return; /* subdirs is done */
|
|
}
|
|
|
|
if( project->isEmpty("QMAKE_EXTENSION_SHLIB") ) {
|
|
if ( project->isEmpty("QMAKE_CYGWIN_SHLIB") ) {
|
|
project->variables()["QMAKE_EXTENSION_SHLIB"].append( "so" );
|
|
} else {
|
|
project->variables()["QMAKE_EXTENSION_SHLIB"].append( "dll" );
|
|
}
|
|
}
|
|
if( project->isEmpty("QMAKE_CFLAGS_PRECOMPILE"))
|
|
project->variables()["QMAKE_CFLAGS_PRECOMPILE"].append("-x c-header -c");
|
|
if( project->isEmpty("QMAKE_CXXFLAGS_PRECOMPILE"))
|
|
project->variables()["QMAKE_CXXFLAGS_PRECOMPILE"].append("-x c++-header -c");
|
|
if( project->isEmpty("QMAKE_CFLAGS_USE_PRECOMPILE"))
|
|
project->variables()["QMAKE_CFLAGS_USE_PRECOMPILE"].append("-include");
|
|
if( project->isEmpty("QMAKE_EXTENSION_PLUGIN") )
|
|
project->variables()["QMAKE_EXTENSION_PLUGIN"].append(project->first("QMAKE_EXTENSION_SHLIB"));
|
|
if( project->isEmpty("QMAKE_COPY_FILE") )
|
|
project->variables()["QMAKE_COPY_FILE"].append( "$(COPY)" );
|
|
if( project->isEmpty("QMAKE_COPY_DIR") )
|
|
project->variables()["QMAKE_COPY_DIR"].append( "$(COPY) -R" );
|
|
if( project->isEmpty("QMAKE_INSTALL_FILE") )
|
|
project->variables()["QMAKE_INSTALL_FILE"].append( "$(COPY_FILE)" );
|
|
if( project->isEmpty("QMAKE_INSTALL_DIR") )
|
|
project->variables()["QMAKE_INSTALL_DIR"].append( "$(COPY_DIR)" );
|
|
if( project->isEmpty("QMAKE_LIBTOOL") )
|
|
project->variables()["QMAKE_LIBTOOL"].append( "libtool --silent" );
|
|
//If the TARGET looks like a path split it into DESTDIR and the resulting TARGET
|
|
if(!project->isEmpty("TARGET")) {
|
|
QString targ = project->first("TARGET");
|
|
int slsh = QMAX(targ.findRev('/'), targ.findRev(Option::dir_sep));
|
|
if(slsh != -1) {
|
|
if(project->isEmpty("DESTDIR"))
|
|
project->values("DESTDIR").append("");
|
|
else if(project->first("DESTDIR").right(1) != Option::dir_sep)
|
|
project->variables()["DESTDIR"] = project->first("DESTDIR") + Option::dir_sep;
|
|
project->variables()["DESTDIR"] = project->first("DESTDIR") + targ.left(slsh+1);
|
|
project->variables()["TARGET"] = targ.mid(slsh+1);
|
|
}
|
|
}
|
|
|
|
project->variables()["QMAKE_ORIG_TARGET"] = project->variables()["TARGET"];
|
|
project->variables()["QMAKE_ORIG_DESTDIR"] = project->variables()["DESTDIR"];
|
|
|
|
bool is_qt = (project->first("TARGET") == "qt" || project->first("TARGET") == "qte" ||
|
|
project->first("TARGET") == "qt-mt" || project->first("TARGET") == "qte-mt");
|
|
bool extern_libs = !project->isEmpty("QMAKE_APP_FLAG") ||
|
|
(!project->isEmpty("QMAKE_LIB_FLAG") &&
|
|
project->isActiveConfig("dll")) || is_qt;
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["LIBS"];
|
|
if ( (!project->isEmpty("QMAKE_LIB_FLAG") && !project->isActiveConfig("staticlib") ) ||
|
|
(project->isActiveConfig("qt") && project->isActiveConfig( "plugin" ) )) {
|
|
if(configs.findIndex("dll") == -1) configs.append("dll");
|
|
} else if ( !project->isEmpty("QMAKE_APP_FLAG") || project->isActiveConfig("dll") ) {
|
|
configs.remove("staticlib");
|
|
}
|
|
if ( project->isActiveConfig("warn_off") ) {
|
|
project->variables()["QMAKE_CFLAGS"] += project->variables()["QMAKE_CFLAGS_WARN_OFF"];
|
|
project->variables()["QMAKE_CXXFLAGS"] += project->variables()["QMAKE_CXXFLAGS_WARN_OFF"];
|
|
} else if ( project->isActiveConfig("warn_on") ) {
|
|
project->variables()["QMAKE_CFLAGS"] += project->variables()["QMAKE_CFLAGS_WARN_ON"];
|
|
project->variables()["QMAKE_CXXFLAGS"] += project->variables()["QMAKE_CXXFLAGS_WARN_ON"];
|
|
}
|
|
if ( project->isActiveConfig("debug") ) {
|
|
project->variables()["QMAKE_CFLAGS"] += project->variables()["QMAKE_CFLAGS_DEBUG"];
|
|
project->variables()["QMAKE_CXXFLAGS"] += project->variables()["QMAKE_CXXFLAGS_DEBUG"];
|
|
project->variables()["QMAKE_LFLAGS"] += project->variables()["QMAKE_LFLAGS_DEBUG"];
|
|
} else {
|
|
project->variables()["QMAKE_CFLAGS"] += project->variables()["QMAKE_CFLAGS_RELEASE"];
|
|
project->variables()["QMAKE_CXXFLAGS"] += project->variables()["QMAKE_CXXFLAGS_RELEASE"];
|
|
project->variables()["QMAKE_LFLAGS"] += project->variables()["QMAKE_LFLAGS_RELEASE"];
|
|
}
|
|
if(!project->isEmpty("QMAKE_INCREMENTAL"))
|
|
project->variables()["QMAKE_LFLAGS"] += project->variables()["QMAKE_LFLAGS_INCREMENTAL"];
|
|
else if(!project->isEmpty("QMAKE_LFLAGS_PREBIND") &&
|
|
!project->variables()["QMAKE_LIB_FLAG"].isEmpty() &&
|
|
project->isActiveConfig("dll"))
|
|
project->variables()["QMAKE_LFLAGS"] += project->variables()["QMAKE_LFLAGS_PREBIND"];
|
|
if(!project->isEmpty("QMAKE_INCDIR"))
|
|
project->variables()["INCLUDEPATH"] += project->variables()["QMAKE_INCDIR"];
|
|
if(!project->isEmpty("QMAKE_LIBDIR")) {
|
|
if ( !project->isEmpty("QMAKE_RPATH") )
|
|
project->variables()["QMAKE_LFLAGS"] += varGlue("QMAKE_LIBDIR", " " + var("QMAKE_RPATH"),
|
|
" " + var("QMAKE_RPATH"), "");
|
|
project->variables()["QMAKE_LIBDIR_FLAGS"] += varGlue( "QMAKE_LIBDIR", "-L", " -L", "" );
|
|
}
|
|
if ( project->isActiveConfig("qtopia") ) {
|
|
if(configs.findIndex("qtopialib") == -1)
|
|
configs.append("qtopialib");
|
|
if(configs.findIndex("qtopiainc") == -1)
|
|
configs.append("qtopiainc");
|
|
}
|
|
if ( project->isActiveConfig("qtopiainc") )
|
|
project->variables()["INCLUDEPATH"] += project->variables()["QMAKE_INCDIR_QTOPIA"];
|
|
if ( project->isActiveConfig("qtopialib") ) {
|
|
if(!project->isEmpty("QMAKE_LIBDIR_QTOPIA"))
|
|
project->variables()["QMAKE_LIBDIR_FLAGS"] += varGlue("QMAKE_LIBDIR_QTOPIA", "-L", " -L", "");
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["QMAKE_LIBS_QTOPIA"];
|
|
}
|
|
if ( project->isActiveConfig("qt") ) {
|
|
if ( project->isActiveConfig("accessibility" ) )
|
|
project->variables()[is_qt ? "PRL_EXPORT_DEFINES" : "DEFINES"].append("QT_ACCESSIBILITY_SUPPORT");
|
|
if ( project->isActiveConfig("tablet") )
|
|
project->variables()[is_qt ? "PRL_EXPORT_DEFINES" : "DEFINES"].append("QT_TABLET_SUPPORT");
|
|
if(configs.findIndex("moc")) configs.append("moc");
|
|
project->variables()["INCLUDEPATH"] += project->variables()["QMAKE_INCDIR_QT"];
|
|
if ( !project->isActiveConfig("debug") )
|
|
project->variables()[is_qt ? "PRL_EXPORT_DEFINES" : "DEFINES"].append("QT_NO_DEBUG");
|
|
if ( !is_qt ) {
|
|
if ( !project->isEmpty("QMAKE_RPATH") ) {
|
|
if ( !project->isEmpty("QMAKE_RTLDIR_QT") )
|
|
project->variables()["QMAKE_LFLAGS"] += varGlue("QMAKE_RTLDIR_QT", " " + var("QMAKE_RPATH"),
|
|
" " + var("QMAKE_RPATH"), "");
|
|
else if ( !project->isEmpty("QMAKE_LIBDIR_QT") )
|
|
project->variables()["QMAKE_LFLAGS"] += varGlue("QMAKE_LIBDIR_QT", " " + var("QMAKE_RPATH"),
|
|
" " + var("QMAKE_RPATH"), "");
|
|
}
|
|
if ( !project->isEmpty("QMAKE_LIBDIR_QT") )
|
|
project->variables()["QMAKE_LIBDIR_FLAGS"] += varGlue("QMAKE_LIBDIR_QT", "-L", " -L", "");
|
|
if ( project->isActiveConfig("thread") && !project->isEmpty("QMAKE_LIBS_QT_THREAD") )
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["QMAKE_LIBS_QT_THREAD"];
|
|
else
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["QMAKE_LIBS_QT"];
|
|
}
|
|
}
|
|
if ( project->isActiveConfig("opengl") && !project->isActiveConfig("dlopen_opengl")) {
|
|
project->variables()["INCLUDEPATH"] += project->variables()["QMAKE_INCDIR_OPENGL"];
|
|
if(!project->isEmpty("QMAKE_LIBDIR_OPENGL"))
|
|
project->variables()["QMAKE_LIBDIR_FLAGS"] += varGlue("QMAKE_LIBDIR_OPENGL", "-L", " -L", "");
|
|
if ( is_qt )
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["QMAKE_LIBS_OPENGL_QT"];
|
|
else
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["QMAKE_LIBS_OPENGL"];
|
|
}
|
|
if ( extern_libs && (project->isActiveConfig("qt") || project->isActiveConfig("opengl")) ) {
|
|
if(configs.findIndex("x11lib") == -1)
|
|
configs.append("x11lib");
|
|
if ( project->isActiveConfig("opengl") && configs.findIndex("x11inc") == -1 )
|
|
configs.append("x11inc");
|
|
}
|
|
if ( project->isActiveConfig("x11") ) {
|
|
if(configs.findIndex("x11lib") == -1)
|
|
configs.append("x11lib");
|
|
if(configs.findIndex("x11inc") == -1)
|
|
configs.append("x11inc");
|
|
}
|
|
if ( project->isActiveConfig("x11inc") )
|
|
project->variables()["INCLUDEPATH"] += project->variables()["QMAKE_INCDIR_X11"];
|
|
if ( project->isActiveConfig("x11lib") ) {
|
|
if(!project->isEmpty("QMAKE_LIBDIR_X11"))
|
|
project->variables()["QMAKE_LIBDIR_FLAGS"] += varGlue("QMAKE_LIBDIR_X11", "-L", " -L", "");
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["QMAKE_LIBS_X11"];
|
|
}
|
|
if ( project->isActiveConfig("x11sm") )
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["QMAKE_LIBS_X11SM"];
|
|
if ( project->isActiveConfig("dylib") )
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["QMAKE_LIBS_DYNLOAD"];
|
|
if ( project->isActiveConfig("thread") ) {
|
|
if(project->isActiveConfig("qt"))
|
|
project->variables()[is_qt ? "PRL_EXPORT_DEFINES" : "DEFINES"].append("QT_THREAD_SUPPORT");
|
|
if ( !project->isEmpty("QMAKE_CFLAGS_THREAD")) {
|
|
project->variables()["QMAKE_CFLAGS"] += project->variables()["QMAKE_CFLAGS_THREAD"];
|
|
project->variables()["PRL_EXPORT_CFLAGS"] += project->variables()["QMAKE_CFLAGS_THREAD"];
|
|
}
|
|
if( !project->isEmpty("QMAKE_CXXFLAGS_THREAD")) {
|
|
project->variables()["QMAKE_CXXFLAGS"] += project->variables()["QMAKE_CXXFLAGS_THREAD"];
|
|
project->variables()["PRL_EXPORT_CXXFLAGS"] += project->variables()["QMAKE_CXXFLAGS_THREAD"];
|
|
}
|
|
project->variables()["INCLUDEPATH"] += project->variables()["QMAKE_INCDIR_THREAD"];
|
|
project->variables()["QMAKE_LIBS"] += project->variables()["QMAKE_LIBS_THREAD"];
|
|
if(!project->isEmpty("QMAKE_LFLAGS_THREAD"))
|
|
project->variables()["QMAKE_LFLAGS"] += project->variables()["QMAKE_LFLAGS_THREAD"];
|
|
}
|
|
if ( project->isActiveConfig("moc") )
|
|
setMocAware(TRUE);
|
|
QString compile_flag = var("QMAKE_COMPILE_FLAG");
|
|
if(compile_flag.isEmpty())
|
|
compile_flag = "-c";
|
|
if(doPrecompiledHeaders() && !project->isEmpty("PRECOMPILED_HEADER")) {
|
|
QString prefix_flags = project->first("QMAKE_CFLAGS_PREFIX_INCLUDE");
|
|
if(prefix_flags.isEmpty())
|
|
prefix_flags = "-include";
|
|
compile_flag += " " + prefix_flags + " " + project->first("QMAKE_ORIG_TARGET");
|
|
}
|
|
if(!project->isEmpty("ALLMOC_HEADER")) {
|
|
initOutPaths(); // Need to fix outdirs since we do this before init() (because we could add to SOURCES et al)
|
|
QString allmoc = fileFixify(project->first("MOC_DIR") + "/allmoc.cpp", QDir::currentDirPath(), Option::output_dir);
|
|
project->variables()["SOURCES"].prepend(allmoc);
|
|
project->variables()["HEADERS_ORIG"] = project->variables()["HEADERS"];
|
|
project->variables()["HEADERS"].clear();
|
|
}
|
|
if ( project->isEmpty("QMAKE_RUN_CC") )
|
|
project->variables()["QMAKE_RUN_CC"].append("$(CC) " + compile_flag + " $(CFLAGS) $(INCPATH) -o $obj $src");
|
|
if ( project->isEmpty("QMAKE_RUN_CC_IMP") )
|
|
project->variables()["QMAKE_RUN_CC_IMP"].append("$(CC) " + compile_flag + " $(CFLAGS) $(INCPATH) -o $@ $<");
|
|
if ( project->isEmpty("QMAKE_RUN_CXX") )
|
|
project->variables()["QMAKE_RUN_CXX"].append("$(CXX) " + compile_flag + " $(CXXFLAGS) $(INCPATH) -o $obj $src");
|
|
if ( project->isEmpty("QMAKE_RUN_CXX_IMP") )
|
|
project->variables()["QMAKE_RUN_CXX_IMP"].append("$(CXX) " + compile_flag + " $(CXXFLAGS) $(INCPATH) -o $@ $<");
|
|
project->variables()["QMAKE_FILETAGS"] += QStringList::split("HEADERS SOURCES TARGET DESTDIR", " ");
|
|
if( project->isActiveConfig("GNUmake") && !project->isEmpty("QMAKE_CFLAGS_DEPS"))
|
|
include_deps = TRUE; //do not generate deps
|
|
if(project->isActiveConfig("compile_libtool"))
|
|
Option::obj_ext = ".lo"; //override the .o
|
|
|
|
MakefileGenerator::init();
|
|
if ( project->isActiveConfig("resource_fork") && !project->isActiveConfig("console")) {
|
|
if(!project->isEmpty("QMAKE_APP_FLAG")) {
|
|
if(project->isEmpty("DESTDIR"))
|
|
project->values("DESTDIR").append("");
|
|
project->variables()["DESTDIR"].first() += project->variables()["TARGET"].first() +
|
|
".app/Contents/MacOS/";
|
|
project->variables()["QMAKE_PKGINFO"].append(project->first("DESTDIR") + "../PkgInfo");
|
|
project->variables()["ALL_DEPS"] += project->first("QMAKE_PKGINFO");
|
|
|
|
QString plist = fileFixify(project->first("QMAKE_INFO_PLIST"));
|
|
if(plist.isEmpty())
|
|
plist = specdir() + QDir::separator() + "Info.plist." + project->first("TEMPLATE");
|
|
if(QFile::exists(Option::fixPathToLocalOS(plist))) {
|
|
if(project->isEmpty("QMAKE_INFO_PLIST"))
|
|
project->variables()["QMAKE_INFO_PLIST"].append(plist);
|
|
project->variables()["QMAKE_INFO_PLIST_OUT"].append(project->first("DESTDIR") +
|
|
"../Info.plist");
|
|
project->variables()["ALL_DEPS"] += project->first("QMAKE_INFO_PLIST_OUT");
|
|
if(!project->isEmpty("RC_FILE"))
|
|
project->variables()["ALL_DEPS"] += project->first("DESTDIR") +
|
|
"../Resources/application.icns";
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!project->isEmpty("QMAKE_INTERNAL_INCLUDED_FILES"))
|
|
project->variables()["DISTFILES"] += project->variables()["QMAKE_INTERNAL_INCLUDED_FILES"];
|
|
project->variables()["DISTFILES"] += Option::mkfile::project_files;
|
|
|
|
init2();
|
|
project->variables()["QMAKE_INTERNAL_PRL_LIBS"] << "QMAKE_LIBDIR_FLAGS" << "QMAKE_LIBS";
|
|
if(!project->isEmpty("QMAKE_MAX_FILES_PER_AR")) {
|
|
bool ok;
|
|
int max_files = project->first("QMAKE_MAX_FILES_PER_AR").toInt(&ok);
|
|
QStringList ar_sublibs, objs = project->variables()["OBJECTS"] + project->variables()["OBJMOC"];
|
|
if(ok && max_files > 5 && max_files < (int)objs.count()) {
|
|
int obj_cnt = 0, lib_cnt = 0;
|
|
QString lib;
|
|
for(QStringList::Iterator objit = objs.begin(); objit != objs.end(); ++objit) {
|
|
if((++obj_cnt) >= max_files) {
|
|
if(lib_cnt) {
|
|
lib.sprintf("lib%s-tmp%d.a", project->first("QMAKE_ORIG_TARGET").latin1(), lib_cnt);
|
|
ar_sublibs << lib;
|
|
obj_cnt = 0;
|
|
}
|
|
lib_cnt++;
|
|
}
|
|
}
|
|
}
|
|
if(!ar_sublibs.isEmpty()) {
|
|
project->variables()["QMAKE_AR_SUBLIBS"] = ar_sublibs;
|
|
project->variables()["QMAKE_INTERNAL_PRL_LIBS"] << "QMAKE_AR_SUBLIBS";
|
|
}
|
|
}
|
|
|
|
if(project->isActiveConfig("compile_libtool")) {
|
|
const QString libtoolify[] = { "QMAKE_RUN_CC", "QMAKE_RUN_CC_IMP",
|
|
"QMAKE_RUN_CXX", "QMAKE_RUN_CXX_IMP",
|
|
"QMAKE_LINK_THREAD", "QMAKE_LINK", "QMAKE_AR_CMD", "QMAKE_LINK_SHLIB_CMD",
|
|
QString::null };
|
|
for(int i = 0; !libtoolify[i].isNull(); i++) {
|
|
QStringList &l = project->variables()[libtoolify[i]];
|
|
if(!l.isEmpty()) {
|
|
QString libtool_flags, comp_flags;
|
|
if(libtoolify[i].startsWith("QMAKE_LINK") || libtoolify[i] == "QMAKE_AR_CMD") {
|
|
libtool_flags += " --mode=link";
|
|
if(project->isActiveConfig("staticlib")) {
|
|
libtool_flags += " -static";
|
|
} else {
|
|
if(!project->isEmpty("QMAKE_LIB_FLAG")) {
|
|
int maj = project->first("VER_MAJ").toInt();
|
|
int min = project->first("VER_MIN").toInt();
|
|
int pat = project->first("VER_PAT").toInt();
|
|
comp_flags += " -version-info " + QString::number(10*maj + min) +
|
|
":" + QString::number(pat) + ":0";
|
|
if(libtoolify[i] != "QMAKE_AR_CMD") {
|
|
QString rpath = Option::output_dir;
|
|
if(!project->isEmpty("DESTDIR")) {
|
|
rpath = project->first("DESTDIR");
|
|
if(QDir::isRelativePath(rpath))
|
|
rpath.prepend(Option::output_dir + Option::dir_sep);
|
|
}
|
|
comp_flags += " -rpath " + Option::fixPathToTargetOS(rpath, FALSE);
|
|
}
|
|
}
|
|
}
|
|
if(project->isActiveConfig("plugin"))
|
|
libtool_flags += " -module";
|
|
} else {
|
|
libtool_flags += " --mode=compile";
|
|
}
|
|
l.first().prepend("$(LIBTOOL)" + libtool_flags + " ");
|
|
if(!comp_flags.isEmpty())
|
|
l.first() += comp_flags;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
QStringList
|
|
UnixMakefileGenerator::combineSetLFlags(const QStringList &list1, const QStringList &list2)
|
|
{
|
|
if(project->isActiveConfig("no_smart_library_merge"))
|
|
return list1 + list2;
|
|
|
|
QStringList ret;
|
|
for(int i = 0; i < 2; i++) {
|
|
const QStringList *lst = i ? &list2 : &list1;
|
|
for(QStringList::ConstIterator it = lst->begin(); it != lst->end(); ++it) {
|
|
if((*it).startsWith("-")) {
|
|
if((*it).startsWith("-L")) {
|
|
if(ret.findIndex((*it)) == -1)
|
|
ret.append((*it));
|
|
} else if((*it).startsWith("-l")) {
|
|
while(1) {
|
|
QStringList::Iterator idx = ret.find((*it));
|
|
if(idx == ret.end())
|
|
break;
|
|
ret.remove(idx);
|
|
}
|
|
ret.append((*it));
|
|
} else if(project->isActiveConfig("macx") && (*it).startsWith("-framework")) {
|
|
int as_one = TRUE;
|
|
QString framework_in;
|
|
if((*it).length() > 11) {
|
|
framework_in = (*it).mid(11);
|
|
} else {
|
|
if(it != lst->end()) {
|
|
++it;
|
|
as_one = FALSE;
|
|
framework_in = (*it);
|
|
}
|
|
}
|
|
if(!framework_in.isEmpty()) {
|
|
for(QStringList::Iterator outit = ret.begin(); outit != ret.end(); ++outit) {
|
|
if((*outit).startsWith("-framework")) {
|
|
int found = 0;
|
|
if((*outit).length() > 11) {
|
|
if(framework_in == (*outit).mid(11))
|
|
found = 1;
|
|
} else {
|
|
if(it != lst->end()) {
|
|
++outit;
|
|
if(framework_in == (*outit)) {
|
|
--outit;
|
|
found = 2;
|
|
}
|
|
}
|
|
}
|
|
for(int i = 0; i < found; i++)
|
|
outit = ret.remove(outit);
|
|
}
|
|
}
|
|
if(as_one) {
|
|
ret.append("-framework " + framework_in);
|
|
} else {
|
|
ret.append("-framework");
|
|
ret.append(framework_in);
|
|
}
|
|
}
|
|
} else {
|
|
#if 1
|
|
while(1) {
|
|
QStringList::Iterator idx = ret.find((*it));
|
|
if(idx == ret.end())
|
|
break;
|
|
ret.remove(idx);
|
|
}
|
|
#endif
|
|
ret.append((*it));
|
|
}
|
|
} else /*if(QFile::exists((*it)))*/ {
|
|
while(1) {
|
|
QStringList::Iterator idx = ret.find((*it));
|
|
if(idx == ret.end())
|
|
break;
|
|
ret.remove(idx);
|
|
}
|
|
ret.append((*it));
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
UnixMakefileGenerator::processPrlVariable(const QString &var, const QStringList &l)
|
|
{
|
|
if(var == "QMAKE_PRL_LIBS")
|
|
project->variables()["QMAKE_CURRENT_PRL_LIBS"] = combineSetLFlags(project->variables()["QMAKE_CURRENT_PRL_LIBS"] +
|
|
project->variables()["QMAKE_LIBS"], l);
|
|
else
|
|
MakefileGenerator::processPrlVariable(var, l);
|
|
}
|
|
|
|
QString
|
|
UnixMakefileGenerator::findDependency(const QString &dep)
|
|
{
|
|
QStringList::Iterator it;
|
|
{
|
|
QStringList &qut = project->variables()["QMAKE_EXTRA_UNIX_TARGETS"];
|
|
for(it = qut.begin(); it != qut.end(); ++it) {
|
|
QString targ = var((*it) + ".target");
|
|
if(targ.isEmpty())
|
|
targ = (*it);
|
|
if(targ.endsWith(dep))
|
|
return targ;
|
|
}
|
|
}
|
|
{
|
|
QStringList &quc = project->variables()["QMAKE_EXTRA_UNIX_COMPILERS"];
|
|
for(it = quc.begin(); it != quc.end(); ++it) {
|
|
QString tmp_out = project->variables()[(*it) + ".output"].first();
|
|
QString tmp_cmd = project->variables()[(*it) + ".commands"].join(" ");
|
|
if(tmp_out.isEmpty() || tmp_cmd.isEmpty())
|
|
continue;
|
|
QStringList &tmp = project->variables()[(*it) + ".input"];
|
|
for(QStringList::Iterator it2 = tmp.begin(); it2 != tmp.end(); ++it2) {
|
|
QStringList &inputs = project->variables()[(*it2)];
|
|
for(QStringList::Iterator input = inputs.begin(); input != inputs.end(); ++input) {
|
|
QString out = tmp_out;
|
|
QFileInfo fi(Option::fixPathToLocalOS((*input)));
|
|
out.replace("${QMAKE_FILE_BASE}", fi.baseName());
|
|
out.replace("${QMAKE_FILE_NAME}", fi.filePath());
|
|
if(out.endsWith(dep))
|
|
return out;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return MakefileGenerator::findDependency(dep);
|
|
}
|
|
|
|
QStringList
|
|
&UnixMakefileGenerator::findDependencies(const QString &file)
|
|
{
|
|
QStringList &ret = MakefileGenerator::findDependencies(file);
|
|
// Note: The QMAKE_IMAGE_COLLECTION file have all images
|
|
// as dependency, so don't add precompiled header then
|
|
if(doPrecompiledHeaders() && !project->isEmpty("PRECOMPILED_HEADER")
|
|
&& file != project->first("QMAKE_IMAGE_COLLECTION")) {
|
|
QString header_prefix = project->first("QMAKE_ORIG_TARGET") + ".gch" + Option::dir_sep;
|
|
header_prefix += project->first("QMAKE_PRECOMP_PREFIX");
|
|
if(file.endsWith(".c")) {
|
|
QString precomp_h = header_prefix + "c";
|
|
if(!ret.contains(precomp_h))
|
|
ret += precomp_h;
|
|
} else {
|
|
for(QStringList::Iterator it = Option::cpp_ext.begin(); it != Option::cpp_ext.end(); ++it) {
|
|
if(file.endsWith(*it)) {
|
|
QString precomp_h = header_prefix + "c++";
|
|
if(!ret.contains(precomp_h))
|
|
ret += precomp_h;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
bool
|
|
UnixMakefileGenerator::findLibraries()
|
|
{
|
|
QPtrList<MakefileDependDir> libdirs;
|
|
libdirs.setAutoDelete(TRUE);
|
|
const QString lflags[] = { "QMAKE_LIBDIR_FLAGS", "QMAKE_LIBS", QString::null };
|
|
for(int i = 0; !lflags[i].isNull(); i++) {
|
|
QStringList &l = project->variables()[lflags[i]];
|
|
for(QStringList::Iterator it = l.begin(); it != l.end(); ++it) {
|
|
QString stub, dir, extn, opt = (*it).stripWhiteSpace();
|
|
if(opt.startsWith("-")) {
|
|
if(opt.startsWith("-L")) {
|
|
QString r = opt.right(opt.length() - 2), l = r;
|
|
fixEnvVariables(l);
|
|
libdirs.append(new MakefileDependDir(r.replace("\"",""),
|
|
l.replace("\"","")));
|
|
} else if(opt.startsWith("-l")) {
|
|
stub = opt.mid(2);
|
|
} else if(project->isActiveConfig("macx") && opt.startsWith("-framework")) {
|
|
if(opt.length() > 11) {
|
|
opt = opt.mid(11);
|
|
} else {
|
|
++it;
|
|
opt = (*it);
|
|
}
|
|
extn = "";
|
|
dir = "/System/Library/Frameworks/" + opt + ".framework/";
|
|
stub = opt;
|
|
}
|
|
} else {
|
|
extn = dir = "";
|
|
stub = opt;
|
|
int slsh = opt.findRev(Option::dir_sep);
|
|
if(slsh != -1) {
|
|
dir = opt.left(slsh);
|
|
stub = opt.mid(slsh+1);
|
|
}
|
|
QRegExp stub_reg("^.*lib(" + stub + "[^./=]*)\\.(.*)$");
|
|
if(stub_reg.exactMatch(stub)) {
|
|
stub = stub_reg.cap(1);
|
|
extn = stub_reg.cap(2);
|
|
}
|
|
}
|
|
if(!stub.isEmpty()) {
|
|
const QString modifs[] = { "", "-mt", QString::null };
|
|
for(int modif = 0; !modifs[modif].isNull(); modif++) {
|
|
bool found = FALSE;
|
|
QStringList extens;
|
|
if(!extn.isNull())
|
|
extens << extn;
|
|
else
|
|
extens << project->variables()["QMAKE_EXTENSION_SHLIB"].first() << "a";
|
|
for(QStringList::Iterator extit = extens.begin(); extit != extens.end(); ++extit) {
|
|
if(dir.isNull()) {
|
|
QString lib_stub;
|
|
for(MakefileDependDir *mdd = libdirs.first(); mdd; mdd = libdirs.next() ) {
|
|
if(QFile::exists(mdd->local_dir + Option::dir_sep + "lib" + stub +
|
|
modifs[modif] + "." + (*extit))) {
|
|
lib_stub = stub + modifs[modif];
|
|
break;
|
|
}
|
|
}
|
|
if(!lib_stub.isNull()) {
|
|
(*it) = "-l" + lib_stub;
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
} else {
|
|
if(QFile::exists("lib" + stub + modifs[modif] + "." + (*extit))) {
|
|
(*it) = "lib" + stub + modifs[modif] + "." + (*extit);
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(!found && project->isActiveConfig("compile_libtool")) {
|
|
for(MakefileDependDir *mdd = libdirs.first(); mdd; mdd = libdirs.next() ) {
|
|
if(QFile::exists(mdd->local_dir + Option::dir_sep + "lib" + stub + modifs[modif] + Option::libtool_ext)) {
|
|
(*it) = mdd->real_dir + Option::dir_sep + "lib" + stub + modifs[modif] + Option::libtool_ext;
|
|
found = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(found)
|
|
break;
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
QString linkLib(const QString &file, const QString &libName) {
|
|
QString ret;
|
|
QRegExp reg("^.*lib(" + libName + "[^./=]*).*$");
|
|
if(reg.exactMatch(file))
|
|
ret = "-l" + reg.cap(1);
|
|
return ret;
|
|
}
|
|
|
|
void
|
|
UnixMakefileGenerator::processPrlFiles()
|
|
{
|
|
QDict<void> processed;
|
|
QPtrList<MakefileDependDir> libdirs;
|
|
libdirs.setAutoDelete(TRUE);
|
|
const QString lflags[] = { "QMAKE_LIBDIR_FLAGS", "QMAKE_LIBS", QString::null };
|
|
for(int i = 0; !lflags[i].isNull(); i++) {
|
|
for(bool ret = FALSE; TRUE; ret = FALSE) {
|
|
QStringList l_out;
|
|
QStringList &l = project->variables()[lflags[i]];
|
|
for(QStringList::Iterator it = l.begin(); it != l.end(); ++it) {
|
|
project->variables()["QMAKE_CURRENT_PRL_LIBS"].clear();
|
|
QString opt = (*it).stripWhiteSpace();
|
|
if(opt.startsWith("-")) {
|
|
if(opt.startsWith("-L")) {
|
|
QString r = opt.right(opt.length() - 2), l = r;
|
|
fixEnvVariables(l);
|
|
libdirs.append(new MakefileDependDir(r.replace("\"",""),
|
|
l.replace("\"","")));
|
|
} else if(opt.startsWith("-l") && !processed[opt]) {
|
|
QString lib = opt.right(opt.length() - 2);
|
|
for(MakefileDependDir *mdd = libdirs.first(); mdd; mdd = libdirs.next() ) {
|
|
if(!project->isActiveConfig("compile_libtool")) { //give them the .libs..
|
|
QString la = mdd->local_dir + Option::dir_sep + "lib" + lib + Option::libtool_ext;
|
|
if(QFile::exists(la) && QFile::exists(mdd->local_dir + Option::dir_sep + ".libs")) {
|
|
l_out.append("-L" + mdd->real_dir + Option::dir_sep + ".libs");
|
|
libdirs.append(new MakefileDependDir(mdd->real_dir + Option::dir_sep + ".libs",
|
|
mdd->local_dir + Option::dir_sep + ".libs"));
|
|
}
|
|
}
|
|
|
|
QString prl = mdd->local_dir + Option::dir_sep + "lib" + lib;
|
|
if(processPrlFile(prl)) {
|
|
if(prl.startsWith(mdd->local_dir))
|
|
prl.replace(0, mdd->local_dir.length(), mdd->real_dir);
|
|
opt = linkLib(prl, lib);
|
|
processed.insert(opt, (void*)1);
|
|
ret = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
} else if(project->isActiveConfig("macx") && opt.startsWith("-framework")) {
|
|
if(opt.length() > 11) {
|
|
opt = opt.mid(11);
|
|
} else {
|
|
++it;
|
|
opt = (*it);
|
|
}
|
|
QString prl = "/System/Library/Frameworks/" + opt +
|
|
".framework/" + opt;
|
|
if(processPrlFile(prl))
|
|
ret = TRUE;
|
|
l_out.append("-framework");
|
|
}
|
|
if(!opt.isEmpty())
|
|
l_out.append(opt);
|
|
l_out = combineSetLFlags(l_out, project->variables()["QMAKE_CURRENT_PRL_LIBS"]);
|
|
} else {
|
|
QString lib = opt;
|
|
if(!processed[lib] && processPrlFile(lib)) {
|
|
processed.insert(lib, (void*)1);
|
|
ret = TRUE;
|
|
}
|
|
#if 0
|
|
if(ret)
|
|
opt = linkLib(lib, "");
|
|
#endif
|
|
if(!opt.isEmpty())
|
|
l_out.append(opt);
|
|
l_out = combineSetLFlags(l_out, project->variables()["QMAKE_CURRENT_PRL_LIBS"]);
|
|
}
|
|
}
|
|
if(ret && l != l_out)
|
|
l = l_out;
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
QString
|
|
UnixMakefileGenerator::defaultInstall(const QString &t)
|
|
{
|
|
if(t != "target" || project->first("TEMPLATE") == "subdirs")
|
|
return QString();
|
|
|
|
bool resource = FALSE;
|
|
const QString root = "$(INSTALL_ROOT)";
|
|
QStringList &uninst = project->variables()[t + ".uninstall"];
|
|
QString ret, destdir=project->first("DESTDIR");
|
|
QString targetdir = Option::fixPathToTargetOS(project->first("target.path"), FALSE);
|
|
if(!destdir.isEmpty() && destdir.right(1) != Option::dir_sep)
|
|
destdir += Option::dir_sep;
|
|
targetdir = fileFixify(targetdir);
|
|
if(targetdir.right(1) != Option::dir_sep)
|
|
targetdir += Option::dir_sep;
|
|
|
|
QStringList links;
|
|
QString target="$(TARGET)";
|
|
if(project->first("TEMPLATE") == "app") {
|
|
target = "$(QMAKE_TARGET)";
|
|
if(project->isActiveConfig("resource_fork") && !project->isActiveConfig("console")) {
|
|
destdir += "../../../";
|
|
target += ".app";
|
|
resource = TRUE;
|
|
}
|
|
} else if(project->first("TEMPLATE") == "lib") {
|
|
if(project->isActiveConfig("create_prl") && !project->isActiveConfig("no_install_prl") &&
|
|
!project->isEmpty("QMAKE_INTERNAL_PRL_FILE")) {
|
|
QString dst_prl = project->first("QMAKE_INTERNAL_PRL_FILE");
|
|
int slsh = dst_prl.findRev('/');
|
|
if(slsh != -1)
|
|
dst_prl = dst_prl.right(dst_prl.length() - slsh - 1);
|
|
dst_prl = root + targetdir + dst_prl;
|
|
ret += "-$(INSTALL_FILE) \"" + project->first("QMAKE_INTERNAL_PRL_FILE") + "\" \"" + dst_prl + "\"";
|
|
if(!uninst.isEmpty())
|
|
uninst.append("\n\t");
|
|
uninst.append("-$(DEL_FILE) \"" + dst_prl + "\"");
|
|
}
|
|
if(project->isActiveConfig("create_libtool") && !project->isActiveConfig("compile_libtool")) {
|
|
QString src_lt = var("QMAKE_ORIG_TARGET");
|
|
int slsh = src_lt.findRev(Option::dir_sep);
|
|
if(slsh != -1)
|
|
src_lt = src_lt.right(src_lt.length() - slsh);
|
|
int dot = src_lt.find('.');
|
|
if(dot != -1)
|
|
src_lt = src_lt.left(dot);
|
|
src_lt += Option::libtool_ext;
|
|
src_lt.prepend("lib");
|
|
QString dst_lt = root + targetdir + src_lt;
|
|
if(!project->isEmpty("DESTDIR")) {
|
|
src_lt.prepend(var("DESTDIR"));
|
|
src_lt = Option::fixPathToLocalOS(fileFixify(src_lt,
|
|
QDir::currentDirPath(), Option::output_dir));
|
|
}
|
|
if(!ret.isEmpty())
|
|
ret += "\n\t";
|
|
ret += "-$(INSTALL_FILE) \"" + src_lt + "\" \"" + dst_lt + "\"";
|
|
if(!uninst.isEmpty())
|
|
uninst.append("\n\t");
|
|
uninst.append("-$(DEL_FILE) \"" + dst_lt + "\"");
|
|
}
|
|
if(project->isActiveConfig("create_pc")) {
|
|
QString src_pc = var("QMAKE_ORIG_TARGET");
|
|
int slsh = src_pc.findRev(Option::dir_sep);
|
|
if(slsh != -1)
|
|
src_pc = src_pc.right(src_pc.length() - slsh);
|
|
int dot = src_pc.find('.');
|
|
if(dot != -1)
|
|
src_pc = src_pc.left(dot);
|
|
src_pc += ".pc";
|
|
QString d = root + targetdir + "pkgconfig" + Option::dir_sep;
|
|
QString dst_pc = d + src_pc;
|
|
if(!project->isEmpty("DESTDIR")) {
|
|
src_pc.prepend(var("DESTDIR"));
|
|
src_pc = Option::fixPathToLocalOS(fileFixify(src_pc,
|
|
QDir::currentDirPath(), Option::output_dir));
|
|
}
|
|
if(!ret.isEmpty())
|
|
ret += "\n\t";
|
|
ret += mkdir_p_asstring(d) + "\n\t";
|
|
ret += "-$(INSTALL_FILE) \"" + src_pc + "\" \"" + dst_pc + "\"";
|
|
if(!uninst.isEmpty())
|
|
uninst.append("\n\t");
|
|
uninst.append("-$(DEL_FILE) \"" + dst_pc + "\"");
|
|
}
|
|
if ( project->isEmpty("QMAKE_CYGWIN_SHLIB") && project->isEmpty("QMAKE_OPENBSD_SHLIBS") ) {
|
|
if ( !project->isActiveConfig("staticlib") && !project->isActiveConfig("plugin") ) {
|
|
if ( project->isEmpty("QMAKE_HPUX_SHLIB") ) {
|
|
links << "$(TARGET0)" << "$(TARGET1)" << "$(TARGET2)";
|
|
} else {
|
|
links << "$(TARGET0)";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!resource && project->isActiveConfig("compile_libtool")) {
|
|
QString src_targ = target;
|
|
if(src_targ == "$(TARGET)")
|
|
src_targ = "$(TARGETL)";
|
|
QString dst_dir = fileFixify(targetdir);
|
|
if(QDir::isRelativePath(dst_dir))
|
|
dst_dir = Option::fixPathToTargetOS(Option::output_dir + Option::dir_sep + dst_dir);
|
|
ret = "-$(LIBTOOL) --mode=install cp \"" + src_targ + "\" \"" + root + dst_dir + "\"";
|
|
uninst.append("-$(LIBTOOL) --mode=uninstall \"" + src_targ + "\"");
|
|
} else {
|
|
QString src_targ = target;
|
|
if(!destdir.isEmpty())
|
|
src_targ = Option::fixPathToTargetOS(destdir + target, FALSE);
|
|
QString dst_targ = root + targetdir + target;
|
|
if(!ret.isEmpty())
|
|
ret += "\n\t";
|
|
if(resource)
|
|
ret += "$(DEL_FILE) -r \"" + dst_targ + "\"" + "\n\t";
|
|
if(!ret.isEmpty())
|
|
ret += "\n\t";
|
|
ret += QString(resource ? "-$(INSTALL_DIR)" : "-$(INSTALL_FILE)") + " \"" +
|
|
src_targ + "\" \"" + dst_targ + "\"";
|
|
if(!project->isActiveConfig("debug") && !project->isEmpty("QMAKE_STRIP") &&
|
|
(project->first("TEMPLATE") != "lib" || !project->isActiveConfig("staticlib"))) {
|
|
ret += "\n\t-" + var("QMAKE_STRIP");
|
|
if(project->first("TEMPLATE") == "lib" && !project->isEmpty("QMAKE_STRIPFLAGS_LIB"))
|
|
ret += " " + var("QMAKE_STRIPFLAGS_LIB");
|
|
else if(project->first("TEMPLATE") == "app" && !project->isEmpty("QMAKE_STRIPFLAGS_APP"))
|
|
ret += " " + var("QMAKE_STRIPFLAGS_APP");
|
|
if(resource)
|
|
ret = " \"" + dst_targ + "/Contents/MacOS/$(QMAKE_TARGET)\"";
|
|
else
|
|
ret += " \"" + dst_targ + "\"";
|
|
}
|
|
if(!uninst.isEmpty())
|
|
uninst.append("\n\t");
|
|
if(resource)
|
|
uninst.append("-$(DEL_FILE) -r \"" + dst_targ + "\"");
|
|
else
|
|
uninst.append("-$(DEL_FILE) \"" + dst_targ + "\"");
|
|
if(!links.isEmpty()) {
|
|
for(QStringList::Iterator it = links.begin(); it != links.end(); it++) {
|
|
if(Option::target_mode == Option::TARG_WIN_MODE ||
|
|
Option::target_mode == Option::TARG_MAC9_MODE) {
|
|
} else if(Option::target_mode == Option::TARG_UNIX_MODE ||
|
|
Option::target_mode == Option::TARG_MACX_MODE) {
|
|
QString link = Option::fixPathToTargetOS(destdir + (*it), FALSE);
|
|
int lslash = link.findRev(Option::dir_sep);
|
|
if(lslash != -1)
|
|
link = link.right(link.length() - (lslash + 1));
|
|
QString dst_link = root + targetdir + link;
|
|
ret += "\n\t-$(SYMLINK) \"$(TARGET)\" \"" + dst_link + "\"";
|
|
if(!uninst.isEmpty())
|
|
uninst.append("\n\t");
|
|
uninst.append("-$(DEL_FILE) \"" + dst_link + "\"");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|