qmake/generators/win32/msvc_dsp.cpp
changeset 0 1918ee327afb
child 4 3b1da2848fc7
equal deleted inserted replaced
-1:000000000000 0:1918ee327afb
       
     1 /****************************************************************************
       
     2 **
       
     3 ** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
       
     4 ** All rights reserved.
       
     5 ** Contact: Nokia Corporation (qt-info@nokia.com)
       
     6 **
       
     7 ** This file is part of the qmake application of the Qt Toolkit.
       
     8 **
       
     9 ** $QT_BEGIN_LICENSE:LGPL$
       
    10 ** No Commercial Usage
       
    11 ** This file contains pre-release code and may not be distributed.
       
    12 ** You may use this file in accordance with the terms and conditions
       
    13 ** contained in the Technology Preview License Agreement accompanying
       
    14 ** this package.
       
    15 **
       
    16 ** GNU Lesser General Public License Usage
       
    17 ** Alternatively, this file may be used under the terms of the GNU Lesser
       
    18 ** General Public License version 2.1 as published by the Free Software
       
    19 ** Foundation and appearing in the file LICENSE.LGPL included in the
       
    20 ** packaging of this file.  Please review the following information to
       
    21 ** ensure the GNU Lesser General Public License version 2.1 requirements
       
    22 ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
       
    23 **
       
    24 ** In addition, as a special exception, Nokia gives you certain additional
       
    25 ** rights.  These rights are described in the Nokia Qt LGPL Exception
       
    26 ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
       
    27 **
       
    28 ** If you have questions regarding the use of this file, please contact
       
    29 ** Nokia at qt-info@nokia.com.
       
    30 **
       
    31 **
       
    32 **
       
    33 **
       
    34 **
       
    35 **
       
    36 **
       
    37 **
       
    38 ** $QT_END_LICENSE$
       
    39 **
       
    40 ****************************************************************************/
       
    41 
       
    42 #include "msvc_dsp.h"
       
    43 #include "option.h"
       
    44 
       
    45 #include <qdir.h>
       
    46 #include <qset.h>
       
    47 
       
    48 #include <stdlib.h>
       
    49 
       
    50 QT_BEGIN_NAMESPACE
       
    51 
       
    52 DspMakefileGenerator::DspMakefileGenerator() : Win32MakefileGenerator(), init_flag(false)
       
    53 {
       
    54 }
       
    55 
       
    56 bool DspMakefileGenerator::writeMakefile(QTextStream &t)
       
    57 {
       
    58     if(!project->values("QMAKE_FAILED_REQUIREMENTS").isEmpty()) {
       
    59         /* for now just dump, I need to generated an empty dsp or something.. */
       
    60         fprintf(stderr, "Project file not generated because all requirements not met:\n\t%s\n",
       
    61                 var("QMAKE_FAILED_REQUIREMENTS").toLatin1().constData());
       
    62         return true;
       
    63     }
       
    64 
       
    65     // Generate workspace file
       
    66     if(project->first("TEMPLATE") == "vcsubdirs") {
       
    67         if (!project->isActiveConfig("build_pass")) {
       
    68             debug_msg(1, "Generator: MSVC: Writing workspave file");
       
    69             writeSubDirs(t);
       
    70         } else {
       
    71             debug_msg(1, "Generator: MSVC: Not writing workspace file for build_pass configs");
       
    72         }
       
    73         return true;
       
    74     } else if (project->first("TEMPLATE") == "vcapp" || project->first("TEMPLATE") == "vclib") {
       
    75         if(!project->isActiveConfig("build_pass"))
       
    76            return writeDspParts(t);
       
    77         return true;
       
    78     }
       
    79     return project->isActiveConfig("build_pass");
       
    80 }
       
    81 
       
    82 bool DspMakefileGenerator::hasBuiltinCompiler(const QString &filename) const
       
    83 {
       
    84     for (int i = 0; i < Option::cpp_ext.count(); ++i)
       
    85         if (filename.endsWith(Option::cpp_ext.at(i)))
       
    86             return true;
       
    87     for (int i = 0; i < Option::c_ext.count(); ++i)
       
    88         if (filename.endsWith(Option::c_ext.at(i)))
       
    89             return true;
       
    90     return false;
       
    91 }
       
    92 
       
    93 QString DspMakefileGenerator::replaceExtraCompilerVariables(const QString &var, const QStringList &in, const QStringList &out)
       
    94 {
       
    95     QString ret = MakefileGenerator::replaceExtraCompilerVariables(var, in, out);
       
    96     ret.replace("$(DEFINES)",  varGlue("PRL_EXPORT_DEFINES"," -D"," -D","") +
       
    97                 varGlue("DEFINES"," -D"," -D",""));
       
    98 
       
    99     QString incpath = this->var("MSVCDSP_INCPATH");
       
   100     incpath.replace("/I", "-I");
       
   101     ret.replace("$(INCPATH)", incpath);
       
   102     return ret;
       
   103 }
       
   104 
       
   105 
       
   106 // if config is part of a multibuild thenthe gule (this) has the correct MSVCDSP_PROJECT
       
   107 QString DspMakefileGenerator::configName(DspMakefileGenerator * config)
       
   108 {
       
   109     return var("MSVCDSP_PROJECT") + config->var("MSVCDSP_CONFIG_NAME");
       
   110 }
       
   111 
       
   112 bool DspMakefileGenerator::writeDspHeader(QTextStream &t)
       
   113 {
       
   114     DspMakefileGenerator * config = this;
       
   115     if (mergedProjects.count())
       
   116         config = mergedProjects.at(0);
       
   117 
       
   118     t << "# Microsoft Developer Studio Project File - Name=\"" << var("MSVCDSP_PROJECT") << "\" - Package Owner=<4>" << endl;
       
   119     t << "# Microsoft Developer Studio Generated Build File, Format Version 6.00" << endl;
       
   120     t << "# ** DO NOT EDIT **" << endl;
       
   121     t << endl;
       
   122     t << "# TARGTYPE \"Win32 (x86) " << var("MSVCDSP_TARGETTYPE") << "\" " << var("MSVCDSP_DSPTYPE") << endl;
       
   123     t << endl;
       
   124     t << "CFG=\"" << configName(config) << "\"" << endl;
       
   125     t << "!MESSAGE This is not a valid makefile. To build this project using NMAKE," << endl;
       
   126     t << "!MESSAGE use the Export Makefile command and run" << endl;
       
   127     t << "!MESSAGE " << endl;
       
   128     t << "!MESSAGE NMAKE /f " << escapeFilePath(var("TARGET")) << ".mak." << endl;
       
   129     t << "!MESSAGE " << endl;
       
   130     t << "!MESSAGE You can specify a configuration when running NMAKE" << endl;
       
   131     t << "!MESSAGE by defining the macro CFG on the command line. For example:" << endl;
       
   132     t << "!MESSAGE " << endl;
       
   133     t << "!MESSAGE NMAKE /f " << escapeFilePath(var("TARGET")) << ".mak CFG=\"" << configName(config) << "\"" << endl;
       
   134     t << "!MESSAGE " << endl;
       
   135     t << "!MESSAGE Possible choices for configuration are:" << endl;
       
   136     t << "!MESSAGE " << endl;
       
   137     if (mergedProjects.count()) {
       
   138         for (int i = 0; i < mergedProjects.count(); ++i) {
       
   139             DspMakefileGenerator * config = mergedProjects.at(i);
       
   140             t << "!MESSAGE \"" << configName(config) << "\" (based on \"Win32 (x86) " << config->var("MSVCDSP_TARGETTYPE") << "\")" << endl;
       
   141         }
       
   142     } else {
       
   143         t << "!MESSAGE \"" << configName(config) << "\" (based on \"Win32 (x86) " << config->var("MSVCDSP_TARGETTYPE") << "\")" << endl;
       
   144     }
       
   145     t << "!MESSAGE " << endl;
       
   146     t << endl;
       
   147     t << "# Begin Project" << endl;
       
   148     t << "# PROP AllowPerConfigDependencies 0" << endl;
       
   149     t << "# PROP Scc_ProjName \"\"" << endl;
       
   150     t << "# PROP Scc_LocalPath \"\"" << endl;
       
   151     t << "CPP=" << config->var("QMAKE_CC") << endl;
       
   152     t << "MTL=" << config->var("QMAKE_IDL") << endl;
       
   153     t << "RSC=" << config->var("QMAKE_RC") << endl;
       
   154     t << "BSC32=bscmake.exe" << endl;
       
   155 
       
   156     return true;
       
   157 }
       
   158 
       
   159 
       
   160 bool DspMakefileGenerator::writeDspParts(QTextStream &t)
       
   161 {
       
   162     //bool staticLibTarget = var("MSVCDSP_DSPTYPE") == "0x0104";
       
   163 
       
   164     writeDspHeader(t);
       
   165     writeDspConfig(t, this);
       
   166     t << endl;
       
   167     t << "# Begin Target" << endl;
       
   168     t << endl;
       
   169     t << "# Name \"" << configName(this) << "\"" << endl;
       
   170     t << endl;
       
   171 
       
   172 
       
   173     QStringList listNames = QString("SOURCES|DEF_FILE").split("|");
       
   174     QStringList allListNames = listNames;
       
   175     writeFileGroup(t, listNames, "Source Files", "cpp;c;cxx;rc;def;r;odl;idl;hpj;bat");
       
   176     listNames = QStringList("HEADERS");
       
   177     allListNames += listNames;
       
   178     writeFileGroup(t, QStringList("HEADERS"), "Header Files", "h;hpp;hxx;hm;inl");
       
   179     listNames = QString("FORMS|INTERFACES|FORMS3").split("|");
       
   180     allListNames += listNames;
       
   181     writeFileGroup(t, listNames, "Form Files", "ui");
       
   182     listNames = QStringList("IMAGES");
       
   183     allListNames += listNames;
       
   184     writeFileGroup(t, QStringList("IMAGES"), "Image Files", "");
       
   185     listNames = QString("RC_FILE|RESOURCES").split("|");
       
   186     allListNames += listNames;
       
   187     writeFileGroup(t, listNames, "Resources", "rc;qrc");
       
   188     listNames = QStringList("TRANSLATIONS");
       
   189     allListNames += listNames;
       
   190     writeFileGroup(t, listNames, "Translations", "ts;xlf");
       
   191     listNames = QStringList("LEXSOURCES");
       
   192     allListNames += listNames;
       
   193     writeFileGroup(t, listNames, "Lexables", "l");
       
   194     listNames = QStringList("YACCSOURCES");
       
   195     allListNames += listNames;
       
   196     writeFileGroup(t, listNames, "Yaccables", "y");
       
   197     listNames = QStringList("TYPELIBS");
       
   198     allListNames += listNames;
       
   199     writeFileGroup(t, listNames, "Type Libraries", "tlb;olb");
       
   200 
       
   201     if (!project->isEmpty("QMAKE_EXTRA_COMPILERS")) {
       
   202          const QStringList &quc = project->values("QMAKE_EXTRA_COMPILERS");
       
   203          for (QStringList::ConstIterator it = quc.begin(); it != quc.end(); ++it) {
       
   204             const QStringList &inputs = project->values((*it)+".input");
       
   205             for (QStringList::ConstIterator input = inputs.begin(); input != inputs.end(); ++input) {
       
   206                 if (!allListNames.contains((*input)) && *input != "UIC3_HEADERS")
       
   207                     writeFileGroup(t, QStringList((*input)), (*input) + " Files", "");
       
   208             }
       
   209         }
       
   210     }
       
   211 
       
   212     project->values("SWAPPED_BUILD_STEPS") = swappedBuildSteps.keys();
       
   213 
       
   214     writeFileGroup(t, QString("GENERATED_SOURCES|GENERATED_FILES|SWAPPED_BUILD_STEPS").split("|"), "Generated", "");
       
   215 
       
   216     t << "# End Target" << endl;
       
   217     t << "# End Project" << endl;
       
   218     return true;
       
   219 }
       
   220 
       
   221 void
       
   222 DspMakefileGenerator::init()
       
   223 {
       
   224     if(init_flag)
       
   225         return;
       
   226     QStringList::Iterator it;
       
   227     init_flag = true;
       
   228 
       
   229     platform = "Win32";
       
   230     if(!project->values("QMAKE_PLATFORM").isEmpty())
       
   231         platform = varGlue("QMAKE_PLATFORM", "", " ", "");
       
   232 
       
   233     // this should probably not be here, but I'm using it to wrap the .t files
       
   234     if(project->first("TEMPLATE") == "vcapp")
       
   235         project->values("QMAKE_APP_FLAG").append("1");
       
   236     else if(project->first("TEMPLATE") == "vclib")
       
   237         project->values("QMAKE_LIB_FLAG").append("1");
       
   238 
       
   239     if(project->values("QMAKESPEC").isEmpty())
       
   240         project->values("QMAKESPEC").append(qgetenv("QMAKESPEC"));
       
   241 
       
   242     project->values("QMAKE_LIBS") += escapeFilePaths(project->values("LIBS"));
       
   243     processVars();
       
   244 
       
   245     if(!project->values("VERSION").isEmpty()) {
       
   246         QString version = project->values("VERSION").first();
       
   247         int firstDot = version.indexOf(".");
       
   248         QString major = version.left(firstDot);
       
   249         QString minor = version.right(version.length() - firstDot - 1);
       
   250         minor.replace(".", "");
       
   251         project->values("MSVCDSP_LFLAGS").append("/VERSION:" + major + "." + minor);
       
   252     }
       
   253 
       
   254     QString msvcdsp_project;
       
   255     if(!project->isEmpty("TARGET")) {
       
   256         project->values("TARGET") = unescapeFilePaths(project->values("TARGET"));
       
   257         msvcdsp_project = project->first("TARGET");
       
   258     }
       
   259 
       
   260     MakefileGenerator::init();
       
   261 
       
   262     if(msvcdsp_project.isEmpty())
       
   263         msvcdsp_project = Option::output.fileName();
       
   264 
       
   265     msvcdsp_project = msvcdsp_project.right(msvcdsp_project.length() - msvcdsp_project.lastIndexOf("\\") - 1);
       
   266     int dotFind = msvcdsp_project.lastIndexOf(".");
       
   267     if(dotFind != -1)
       
   268         msvcdsp_project = msvcdsp_project.left(dotFind);
       
   269     msvcdsp_project.replace("-", "");
       
   270 
       
   271     project->values("MSVCDSP_PROJECT").append(msvcdsp_project);
       
   272 
       
   273     QStringList &proj = project->values("MSVCDSP_PROJECT");
       
   274 
       
   275     for(QStringList::Iterator it = proj.begin(); it != proj.end(); ++it)
       
   276         (*it).replace(QRegExp("\\.[a-zA-Z0-9_]*$"), "");
       
   277 
       
   278     if(!project->values("QMAKE_APP_FLAG").isEmpty()) {
       
   279         if(project->isActiveConfig("console")) {
       
   280             project->values("MSVCDSP_TARGETTYPE").append("Console Application");
       
   281             project->values("MSVCDSP_DSPTYPE").append("0x0103");
       
   282             project->values("MSVCDSP_DEFINES").append(" /D \"_CONSOLE\" ");
       
   283         } else {
       
   284             project->values("MSVCDSP_TARGETTYPE").append("Application");
       
   285             project->values("MSVCDSP_DSPTYPE").append("0x0101");
       
   286             project->values("MSVCDSP_DEFINES").append(" /D \"_WINDOWS\" ");
       
   287         }
       
   288     } else {
       
   289         if(project->isActiveConfig("dll")) {
       
   290             project->values("MSVCDSP_TARGETTYPE").append("Dynamic-Link Library");
       
   291             project->values("MSVCDSP_DSPTYPE").append("0x0102");
       
   292             project->values("MSVCDSP_DEFINES").append(" /D \"_USRDLL\" ");
       
   293         } else {
       
   294             project->values("MSVCDSP_TARGETTYPE").append("Static Library");
       
   295             project->values("MSVCDSP_DSPTYPE").append("0x0104");
       
   296             project->values("MSVCDSP_DEFINES").append(" /D \"_LIB\" ");
       
   297         }
       
   298     }
       
   299 
       
   300     project->values("MSVCDSP_LFLAGS") += project->values("QMAKE_LFLAGS");
       
   301 
       
   302     if(!project->values("QMAKE_LIBDIR").isEmpty())
       
   303         project->values("MSVCDSP_LFLAGS").append(valGlue(
       
   304                                                      escapeFilePaths(project->values("QMAKE_LIBDIR")),
       
   305                                                      "/LIBPATH:"," /LIBPATH:",""));
       
   306 
       
   307     project->values("MSVCDSP_DEFINES").append(varGlue("DEFINES","/D ","" " /D ",""));
       
   308     project->values("MSVCDSP_DEFINES").append(varGlue("PRL_EXPORT_DEFINES","/D ","" " /D ",""));
       
   309     project->values("MSVCDSP_DEFINES").append(" /D \"WIN32\" ");
       
   310 
       
   311     QStringList &libs = project->values("QMAKE_LIBS");
       
   312     for(QStringList::Iterator libit = libs.begin(); libit != libs.end(); ++libit) {
       
   313         project->values("MSVCDSP_LIBS").append(" " + escapeFilePath(*libit));
       
   314     }
       
   315 
       
   316     QStringList &incs = project->values("INCLUDEPATH");
       
   317     for(QStringList::Iterator incit = incs.begin(); incit != incs.end(); ++incit) {
       
   318         QString inc = (*incit);
       
   319         project->values("MSVCDSP_INCPATH").append("/I" + escapeFilePath(inc));
       
   320     }
       
   321     project->values("MSVCDSP_INCPATH").append("/I" + escapeFilePath(specdir()));
       
   322 
       
   323     QString dest;
       
   324     QString preLinkStep;
       
   325     QString postLinkStep;
       
   326     QString copyDllStep;
       
   327 
       
   328     if(!project->values("QMAKE_PRE_LINK").isEmpty())
       
   329         preLinkStep += var("QMAKE_PRE_LINK");
       
   330 
       
   331     if(!project->values("QMAKE_POST_LINK").isEmpty())
       
   332         postLinkStep += var("QMAKE_POST_LINK");
       
   333 
       
   334     // don't destroy the target, it is used by prl writer.
       
   335     if(!project->values("DESTDIR").isEmpty()) {
       
   336         dest = project->first("DESTDIR");
       
   337         project->values("DESTDIR").first() = dest;
       
   338         dest = project->values("TARGET").first() + project->first("TARGET_EXT");
       
   339         dest.prepend(project->first("DESTDIR"));
       
   340         Option::fixPathToTargetOS(dest);
       
   341         dest = escapeFilePath(dest);
       
   342 
       
   343         project->values("MSVCDSP_TARGET").append(
       
   344             QString("/out:") + dest);
       
   345         if(project->isActiveConfig("dll")) {
       
   346             QString imp = dest;
       
   347             imp.replace(".dll", ".lib");
       
   348             project->values("MSVCDSP_TARGET").append(QString(" /implib:") + escapeFilePath(imp));
       
   349         }
       
   350     }
       
   351 
       
   352     if(project->isActiveConfig("dll") && !project->values("DLLDESTDIR").isEmpty()) {
       
   353         QStringList dlldirs = project->values("DLLDESTDIR");
       
   354         if(dlldirs.count())
       
   355             copyDllStep += "\t";
       
   356         for(QStringList::Iterator dlldir = dlldirs.begin(); dlldir != dlldirs.end(); ++dlldir) {
       
   357             copyDllStep += "copy \"$(TargetPath)\" " + escapeFilePath(Option::fixPathToTargetOS(*dlldir)) + "\t";
       
   358         }
       
   359     }
       
   360 
       
   361     if(!preLinkStep.isEmpty()) {
       
   362         project->values("MSVCDSP_PRE_LINK").append(
       
   363             "# Begin Special Build Tool\n"
       
   364             "SOURCE=$(InputPath)\n"
       
   365             "PreLink_Desc=Post Build Step\n"
       
   366             "PreLink_Cmds=" + preLinkStep + "\n"
       
   367             "# End Special Build Tool\n");
       
   368     }
       
   369 
       
   370     if(!postLinkStep.isEmpty() || !copyDllStep.isEmpty()) {
       
   371         project->values("MSVCDSP_POST_LINK").append(
       
   372             "# Begin Special Build Tool\n"
       
   373             "SOURCE=$(InputPath)\n"
       
   374             "PostBuild_Desc=Post Build Step\n"
       
   375             "PostBuild_Cmds=" + postLinkStep + copyDllStep + "\n"
       
   376             "# End Special Build Tool\n");
       
   377     }
       
   378 
       
   379     QStringList &formList = project->values("FORMS");
       
   380     for(QStringList::ConstIterator hit = formList.begin(); hit != formList.end(); ++hit) {
       
   381         if(exists(*hit + ".h"))
       
   382             project->values("SOURCES").append(*hit + ".h");
       
   383     }
       
   384     QStringList &form3List = project->values("FORMS3");
       
   385     for(QStringList::ConstIterator hit = form3List.begin(); hit != form3List.end(); ++hit) {
       
   386         if(exists(*hit + ".h"))
       
   387             project->values("SOURCES").append(*hit + ".h");
       
   388     }
       
   389 
       
   390     project->values("QMAKE_INTERNAL_PRL_LIBS") << "MSVCDSP_LIBS";
       
   391 
       
   392     // Move some files around //### is this compat?
       
   393     if (!project->values("IMAGES").isEmpty()) {
       
   394         QString imageFactory(project->first("QMAKE_IMAGE_COLLECTION"));
       
   395         project->values("GENERATED_SOURCES") += imageFactory;
       
   396         project->values("SOURCES").removeAll(imageFactory);
       
   397     }
       
   398 
       
   399     // Setup PCH variables
       
   400     precompH = project->first("PRECOMPILED_HEADER");
       
   401     namePCH = fileInfo(precompH).fileName();
       
   402     usePCH = !precompH.isEmpty() && project->isActiveConfig("precompile_header");
       
   403     if (usePCH) {
       
   404         // Created files
       
   405         precompObj = var("PRECOMPILED_DIR") + project->first("TARGET") + "_pch" + Option::obj_ext;
       
   406         precompPch = var("PRECOMPILED_DIR") + project->first("TARGET") + "_pch.pch";
       
   407 
       
   408         // Add PRECOMPILED_HEADER to HEADERS
       
   409         if (!project->values("HEADERS").contains(precompH))
       
   410             project->values("HEADERS") += precompH;
       
   411         // Add precompile compiler options
       
   412         project->values("PRECOMPILED_FLAGS")  = QStringList("/Fp" + precompPch + " /Yu" + escapeFilePath(namePCH) + " /FI" + escapeFilePath(namePCH) + " ");
       
   413         // Return to variable pool
       
   414         project->values("PRECOMPILED_OBJECT") = QStringList(precompObj);
       
   415         project->values("PRECOMPILED_PCH")    = QStringList(precompPch);
       
   416     }
       
   417 
       
   418     QString buildName;
       
   419     if (!var("BUILD_NAME").isEmpty())
       
   420         buildName =  var("BUILD_NAME");
       
   421     else if (project->isActiveConfig("debug"))
       
   422         buildName = "Debug";
       
   423     else
       
   424         buildName = "Release";
       
   425 
       
   426     project->values("MSVCDSP_CONFIG_NAME") = QStringList(" - " + platform + " " + buildName);
       
   427 }
       
   428 
       
   429 void DspMakefileGenerator::processPrlVariable(const QString &var, const QStringList &l)
       
   430 {
       
   431     if(var == "QMAKE_PRL_DEFINES") {
       
   432         QStringList &out = project->values("MSVCDSP_DEFINES");
       
   433         for(QStringList::ConstIterator it = l.begin(); it != l.end(); ++it) {
       
   434             if(out.indexOf((*it)) == -1)
       
   435                 out.append((" /D \"" + *it + "\""));
       
   436         }
       
   437     } else {
       
   438         MakefileGenerator::processPrlVariable(var, l);
       
   439     }
       
   440 }
       
   441 
       
   442 bool DspMakefileGenerator::openOutput(QFile &file, const QString &build) const
       
   443 {
       
   444     QString outdir;
       
   445     if(!file.fileName().isEmpty()) {
       
   446         if(QDir::isRelativePath(file.fileName()))
       
   447             file.setFileName(Option::output_dir + "/" + file.fileName()); //pwd when qmake was run
       
   448         QFileInfo fi(fileInfo(file.fileName()));
       
   449         if(fi.isDir())
       
   450             outdir = file.fileName() + QDir::separator();
       
   451     }
       
   452 
       
   453     if(!outdir.isEmpty() || file.fileName().isEmpty()) {
       
   454         QString ext = project->first("DSP_EXTENSION");
       
   455         if(project->first("TEMPLATE") == "vcsubdirs") {
       
   456             if (!project->first("DSW_EXTENSION").isEmpty())
       
   457                 ext = project->first("DSW_EXTENSION");
       
   458             else
       
   459                 ext = ".dsw";
       
   460         }
       
   461         QString outputName = unescapeFilePath(project->first("QMAKE_DSP_PROJECT_NAME"));
       
   462         if (!project->first("MAKEFILE").isEmpty())
       
   463             outputName = unescapeFilePath(project->first("MAKEFILE"));
       
   464         if (outputName.isEmpty())
       
   465             outputName = unescapeFilePath(project->first("QMAKE_ORIG_TARGET"));
       
   466         file.setFileName(outdir + outputName + ext);
       
   467     }
       
   468 
       
   469     if(QDir::isRelativePath(file.fileName())) {
       
   470         QString ofile = Option::fixPathToLocalOS(file.fileName());
       
   471         int slashfind = ofile.lastIndexOf(Option::dir_sep);
       
   472         if(slashfind == -1) {
       
   473             ofile = ofile.replace(QRegExp("-"), "_");
       
   474         } else {
       
   475             int hypenfind = ofile.indexOf('-', slashfind);
       
   476             while (hypenfind != -1 && slashfind < hypenfind) {
       
   477                 ofile = ofile.replace(hypenfind, 1, "_");
       
   478                 hypenfind = ofile.indexOf('-', hypenfind + 1);
       
   479             }
       
   480         }
       
   481         file.setFileName(Option::fixPathToLocalOS(qmake_getpwd() + Option::dir_sep + ofile));
       
   482     }
       
   483     return Win32MakefileGenerator::openOutput(file, build);
       
   484 }
       
   485 
       
   486 bool DspMakefileGenerator::mergeBuildProject(MakefileGenerator *other)
       
   487 {
       
   488 
       
   489     mergedProjects.prepend(static_cast<DspMakefileGenerator*>(other));
       
   490     return true;
       
   491 }
       
   492 
       
   493 bool DspMakefileGenerator::writeProjectMakefile()
       
   494 {
       
   495     bool ret = true;
       
   496 
       
   497     QTextStream t(&Option::output);
       
   498     // Check if all requirements are fulfilled
       
   499     if(!project->values("QMAKE_FAILED_REQUIREMENTS").isEmpty()) {
       
   500         fprintf(stderr, "Project file not generated because all requirements not met:\n\t%s\n",
       
   501                 var("QMAKE_FAILED_REQUIREMENTS").toLatin1().constData());
       
   502         return true;
       
   503     }
       
   504 
       
   505     // Generate project file
       
   506     if(project->first("TEMPLATE") == "vcapp" ||
       
   507        project->first("TEMPLATE") == "vclib") {
       
   508         if (!mergedProjects.count()) {
       
   509             warn_msg(WarnLogic, "Generator: MSVC DSP: no single configuration created, cannot output project!");
       
   510             return false;
       
   511         }
       
   512         debug_msg(1, "Generator: MSVC 6: Writing project file");
       
   513 
       
   514         writeDspHeader(t);
       
   515         for (int i = 0; i < mergedProjects.count(); ++i) {
       
   516             DspMakefileGenerator* config = mergedProjects.at(i);
       
   517             t << endl;
       
   518             if (i == 0)
       
   519                 t << "!IF";
       
   520             else
       
   521                 t << "!ELSEIF";
       
   522             t << "  \"$(CFG)\" == \"" << configName(config) << "\"" << endl;
       
   523             t << endl;
       
   524             writeDspConfig(t, config);
       
   525         }
       
   526         t << endl;
       
   527         t << "!ENDIF " << endl;
       
   528         t << endl;
       
   529         t << "# Begin Target" << endl;
       
   530         t << endl;
       
   531         for (int i = 0; i < mergedProjects.count(); ++i)
       
   532             t << "# Name \"" << configName(mergedProjects.at(i)) << "\"" << endl;
       
   533         t << endl;
       
   534 
       
   535         QMap< QString, QSet<QString> > files;
       
   536 
       
   537         // merge source files
       
   538         for (int i = 0; i < mergedProjects.count(); ++i) {
       
   539 
       
   540             DspMakefileGenerator* config = mergedProjects.at(i);
       
   541 
       
   542             files["DEF_FILE"] += config->project->values("DEF_FILE").toSet();
       
   543             files["SOURCES"] += config->project->values("SOURCES").toSet();
       
   544             files["HEADERS"] += config->project->values("HEADERS").toSet();
       
   545             files["INTERFACES"] += config->project->values("INTERFACES").toSet();
       
   546             files["FORMS"] += config->project->values("FORMS").toSet();
       
   547             files["FORMS"] += config->project->values("FORMS3").toSet();
       
   548             files["IMAGES"] += config->project->values("IMAGES").toSet();
       
   549             files["RC_FILE"] += config->project->values("RC_FILE").toSet();
       
   550             files["RESOURCES"] += config->project->values("RESOURCES").toSet();
       
   551             files["TRANSLATIONS"] += config->project->values("TRANSLATIONS").toSet();
       
   552             files["LEXSOURCES"] += config->project->values("LEXSOURCES").toSet();
       
   553             files["YACCSOURCES"] += config->project->values("YACCSOURCES").toSet();
       
   554             files["TYPELIBS"] += config->project->values("TYPELIBS").toSet();
       
   555 
       
   556             if (!config->project->isEmpty("QMAKE_EXTRA_COMPILERS")) {
       
   557                 const QStringList &quc = config->project->values("QMAKE_EXTRA_COMPILERS");
       
   558                 for (QStringList::ConstIterator it = quc.begin(); it != quc.end(); ++it) {
       
   559                     const QStringList &inputs = project->values((*it)+".input");
       
   560                     for (QStringList::ConstIterator input = inputs.begin(); input != inputs.end(); ++input) {
       
   561                         if (*input != "UIC3_HEADERS")
       
   562                             files[(*input)] += config->project->values((*input)).toSet();
       
   563                     }
       
   564                 }
       
   565             }
       
   566         }
       
   567 
       
   568         QStringList keys = files.keys();
       
   569         for (int k = 0; k < keys.size(); ++k)
       
   570             project->values(keys.at(k)) = QList<QString>::fromSet(files[keys.at(k)]);
       
   571 
       
   572         QStringList listNames = QString("SOURCES|DEF_FILE").split("|");
       
   573         QStringList allListNames = listNames;
       
   574         writeFileGroup(t, listNames, "Source Files", "cpp;c;cxx;rc;def;r;odl;idl;hpj;bat");
       
   575         listNames = QStringList("HEADERS");
       
   576         allListNames += listNames;
       
   577         writeFileGroup(t, listNames, "Header Files", "h;hpp;hxx;hm;inl");
       
   578         listNames = QString("FORMS|INTERFACES|FORMS3").split("|");
       
   579         allListNames += listNames;
       
   580         writeFileGroup(t, listNames, "Form Files", "ui");
       
   581         listNames = QStringList("IMAGES");
       
   582         allListNames += listNames;
       
   583         writeFileGroup(t, listNames, "Image Files", "");
       
   584         listNames = QString("RC_FILE|RESOURCES").split("|");
       
   585         allListNames += listNames;
       
   586         writeFileGroup(t, listNames, "Resources", "rc;qrc");
       
   587         listNames = QStringList("TRANSLATIONS");
       
   588         allListNames += listNames;
       
   589         writeFileGroup(t, listNames, "Translations", "ts;xlf");
       
   590         listNames = QStringList("LEXSOURCES");
       
   591         allListNames += listNames;
       
   592         writeFileGroup(t, listNames, "Lexables", "l");
       
   593         listNames = QStringList("YACCSOURCES");
       
   594         allListNames += listNames;
       
   595         writeFileGroup(t, listNames, "Yaccables", "y");
       
   596         listNames = QStringList("TYPELIBS");
       
   597         allListNames += listNames;
       
   598         writeFileGroup(t, listNames, "Type Libraries", "tlb;olb");
       
   599 
       
   600         for (int l = 0; l < allListNames.size(); ++l)
       
   601             keys.removeAll(allListNames.at(l));
       
   602 
       
   603         for (int k = 0; k < keys.size(); ++k)
       
   604             writeFileGroup(t, QStringList(keys.at(k)), keys.at(k) + " Files", "");
       
   605 
       
   606         // done last as generated may have changed when creating build rules for the above
       
   607         for (int i = 0; i < mergedProjects.count(); ++i) {
       
   608 
       
   609             DspMakefileGenerator* config = mergedProjects.at(i);
       
   610 
       
   611             config->project->values("SWAPPED_BUILD_STEPS") = config->swappedBuildSteps.keys();
       
   612             files["SWAPPED_BUILD_STEPS"] +=  config->project->values("SWAPPED_BUILD_STEPS").toSet();
       
   613 
       
   614             files["GENERATED_SOURCES"] += config->project->values("GENERATED_SOURCES").toSet();
       
   615             files["GENERATED_FILES"] += config->project->values("GENERATED_FILES").toSet();
       
   616         }
       
   617 
       
   618         project->values("SWAPPED_BUILD_STEPS") = QList<QString>::fromSet(files["SWAPPED_BUILD_STEPS"]);
       
   619         project->values("GENERATED_SOURCES") = QList<QString>::fromSet(files["GENERATED_SOURCES"]);
       
   620         project->values("GENERATED_FILES") = QList<QString>::fromSet(files["GENERATED_FILES"]);
       
   621 
       
   622         writeFileGroup(t, QString("GENERATED_SOURCES|GENERATED_FILES|SWAPPED_BUILD_STEPS").split("|"), "Generated", "");
       
   623         t << endl;
       
   624         t << "# End Target" << endl;
       
   625         t << "# End Project" << endl;
       
   626     }else if(project->first("TEMPLATE") == "vcsubdirs") {
       
   627         ret = writeMakefile(t);
       
   628     }
       
   629 
       
   630     return ret;
       
   631 }
       
   632 
       
   633 const char _dswHeader60[]      = "Microsoft Developer Studio Workspace File, Format Version 6.00\n";
       
   634 const char _dswWarning[]       = "# WARNING: DO NOT EDIT OR DELETE THIS WORKSPACE FILE!\n";
       
   635 const char _dswDevider[]       = "###############################################################################\n";
       
   636 const char _dswProjectName[]   = "Project: \"%1\"=%2 - Package Owner=<4>\n"; // %1 = project name, %2 = project path
       
   637 const char _dswPackage5Start[] = "Package=<5>\n{{{\n";
       
   638 const char _dswPackage5Stop[]  = "}}}\n";
       
   639 const char _dswPackage4Start[] = "Package=<4>\n{{{\n";
       
   640 const char _dswPackage4Stop[]  = "}}}\n";
       
   641 const char _dswProjectDep[]    = "    Begin Project Dependency\n    Project_Dep_Name %1\n    End Project Dependency\n"; // %1 = project name
       
   642 const char _dswGlobal[]        = "Global:\n\nPackage=<5>\n{{{\n}}}\n\nPackage=<3>\n{{{\n}}}\n\n";
       
   643 
       
   644 
       
   645 struct WorkspaceDepend {
       
   646     QString dspProjectFile, orig_target, target;
       
   647     QStringList dependencies;
       
   648 };
       
   649 
       
   650 void DspMakefileGenerator::writeSubDirs(QTextStream &t)
       
   651 {
       
   652     // Output headers
       
   653     t << _dswHeader60;
       
   654     t << _dswWarning;
       
   655     t << endl;
       
   656 
       
   657     QHash<QString, WorkspaceDepend*> workspace_depends;
       
   658     QList<WorkspaceDepend*> workspace_cleanup;
       
   659     QStringList subdirs = project->values("SUBDIRS");
       
   660     QString oldpwd = qmake_getpwd();
       
   661 
       
   662     // Make sure that all temp projects are configured
       
   663     // for release so that the depends are created
       
   664     // without the debug <lib>dxxx.lib name mangling
       
   665     QStringList old_after_vars = Option::after_user_vars;
       
   666     Option::after_user_vars.append("CONFIG+=release");
       
   667 
       
   668     for(int i = 0; i < subdirs.size(); ++i) {
       
   669         QString tmp = subdirs.at(i);
       
   670         if(!project->isEmpty(tmp + ".file")) {
       
   671             if(!project->isEmpty(tmp + ".subdir"))
       
   672                 warn_msg(WarnLogic, "Cannot assign both file and subdir for subdir %s",
       
   673                 tmp.toLatin1().constData());
       
   674             tmp = project->first(tmp + ".file");
       
   675         } else if(!project->isEmpty(tmp + ".subdir")) {
       
   676             tmp = project->first(tmp + ".subdir");
       
   677         }
       
   678 
       
   679         QFileInfo fi(fileInfo(Option::fixPathToLocalOS(tmp, true)));
       
   680         if(fi.exists()) {
       
   681             if(fi.isDir()) {
       
   682                 QString profile = tmp;
       
   683                 if(!profile.endsWith(Option::dir_sep))
       
   684                     profile += Option::dir_sep;
       
   685                 profile += fi.baseName() + Option::pro_ext;
       
   686                 subdirs.append(profile);
       
   687             } else {
       
   688                 QMakeProject tmp_proj;
       
   689                 QString dir = fi.path(), fn = fi.fileName();
       
   690                 if(!dir.isEmpty()) {
       
   691                     if(!qmake_setpwd(dir))
       
   692                         fprintf(stderr, "Cannot find directory: %s\n", dir.toLatin1().constData());
       
   693                 }
       
   694                 if(tmp_proj.read(fn)) {
       
   695                     // Check if all requirements are fulfilled
       
   696                     if(!tmp_proj.variables()["QMAKE_FAILED_REQUIREMENTS"].isEmpty()) {
       
   697                         fprintf(stderr, "Project file(%s) not added to Workspace because all requirements not met:\n\t%s\n",
       
   698                             fn.toLatin1().constData(), tmp_proj.values("QMAKE_FAILED_REQUIREMENTS").join(" ").toLatin1().constData());
       
   699                         continue;
       
   700                     }
       
   701                     if(tmp_proj.first("TEMPLATE") == "vcsubdirs") {
       
   702                         QStringList tmp_proj_subdirs = tmp_proj.variables()["SUBDIRS"];
       
   703                         for(int x = 0; x < tmp_proj_subdirs.size(); ++x) {
       
   704                             QString tmpdir = tmp_proj_subdirs.at(x);
       
   705                             if(!tmp_proj.isEmpty(tmpdir + ".file")) {
       
   706                                 if(!tmp_proj.isEmpty(tmpdir + ".subdir"))
       
   707                                     warn_msg(WarnLogic, "Cannot assign both file and subdir for subdir %s",
       
   708                                     tmpdir.toLatin1().constData());
       
   709                                 tmpdir = tmp_proj.first(tmpdir + ".file");
       
   710                             } else if(!tmp_proj.isEmpty(tmpdir + ".subdir")) {
       
   711                                 tmpdir = tmp_proj.first(tmpdir + ".subdir");
       
   712                             }
       
   713                             subdirs += fileFixify(tmpdir);
       
   714                         }
       
   715                     } else if(tmp_proj.first("TEMPLATE") == "vcapp" || tmp_proj.first("TEMPLATE") == "vclib") {
       
   716                         // Initialize a 'fake' project to get the correct variables
       
   717                         // and to be able to extract all the dependencies
       
   718                         DspMakefileGenerator tmp_dsp;
       
   719                         tmp_dsp.setNoIO(true);
       
   720                         tmp_dsp.setProjectFile(&tmp_proj);
       
   721                         if(Option::debug_level) {
       
   722                             QMap<QString, QStringList> &vars = tmp_proj.variables();
       
   723                             for(QMap<QString, QStringList>::Iterator it = vars.begin();
       
   724                                 it != vars.end(); ++it) {
       
   725                                     if(it.key().left(1) != "." && !it.value().isEmpty())
       
   726                                         debug_msg(1, "%s: %s === %s", fn.toLatin1().constData(), it.key().toLatin1().constData(),
       
   727                                         it.value().join(" :: ").toLatin1().constData());
       
   728                             }
       
   729                         }
       
   730 
       
   731                         // We assume project filename is [QMAKE_ORIG_TARGET].vcproj
       
   732                         QString dsp = unescapeFilePath(tmp_dsp.project->first("MSVCDSP_PROJECT") + project->first("DSP_EXTENSION"));
       
   733 
       
   734                         // If file doesn't exsist, then maybe the users configuration
       
   735                         // doesn't allow it to be created. Skip to next...
       
   736                         if(!exists(qmake_getpwd() + Option::dir_sep + dsp)) {
       
   737                             warn_msg(WarnLogic, "Ignored (not found) '%s'", QString(qmake_getpwd() + Option::dir_sep + dsp).toLatin1().constData());
       
   738                             goto nextfile; // # Dirty!
       
   739                         }
       
   740 
       
   741                         WorkspaceDepend *newDep = new WorkspaceDepend;
       
   742                         newDep->dspProjectFile = fileFixify(dsp);
       
   743                         newDep->orig_target = unescapeFilePath(tmp_proj.first("QMAKE_ORIG_TARGET"));
       
   744                         newDep->target = tmp_proj.first("MSVCDSP_PROJECT").section(Option::dir_sep, -1) + tmp_proj.first("TARGET_EXT");
       
   745 
       
   746                         // We want to store it as the .lib name.
       
   747                         if(newDep->target.endsWith(".dll"))
       
   748                             newDep->target = newDep->target.left(newDep->target.length()-3) + "lib";
       
   749 
       
   750                         // All projects having mocable sourcefiles are dependent on moc.exe
       
   751                         if(tmp_proj.variables()["CONFIG"].contains("moc"))
       
   752                             newDep->dependencies << "moc.exe";
       
   753 
       
   754                         // All extra compilers which has valid input are considered dependencies
       
   755                         const QStringList &quc = tmp_proj.variables()["QMAKE_EXTRA_COMPILERS"];
       
   756                         for(QStringList::ConstIterator it = quc.constBegin(); it != quc.constEnd(); ++it) {
       
   757                             const QStringList &invar = tmp_proj.variables().value((*it) + ".input");
       
   758                             for(QStringList::ConstIterator iit = invar.constBegin(); iit != invar.constEnd(); ++iit) {
       
   759                                 const QStringList fileList = tmp_proj.variables().value(*iit);
       
   760                                 if (!fileList.isEmpty()) {
       
   761                                     QString dep = tmp_proj.first((*it) + ".commands").section('/', -1).section('\\', -1);
       
   762                                     if (!newDep->dependencies.contains(dep))
       
   763                                         newDep->dependencies << dep;
       
   764                                 }
       
   765                             }
       
   766                         }
       
   767 
       
   768                         // Add all unknown libs to the deps
       
   769                         QStringList where("QMAKE_LIBS");
       
   770                         if(!tmp_proj.isEmpty("QMAKE_INTERNAL_PRL_LIBS"))
       
   771                             where = tmp_proj.variables()["QMAKE_INTERNAL_PRL_LIBS"];
       
   772 
       
   773                         for(QStringList::iterator wit = where.begin();
       
   774                             wit != where.end(); ++wit) {
       
   775                                 QStringList &l = tmp_proj.variables()[(*wit)];
       
   776                                 for(QStringList::Iterator it = l.begin(); it != l.end(); ++it) {
       
   777                                     QString opt = (*it).trimmed();
       
   778                                     if(!opt.startsWith("/") &&   // Not a switch
       
   779                                         opt != newDep->target && // Not self
       
   780                                         opt != "opengl32.lib" && // We don't care about these libs
       
   781                                         opt != "glu32.lib" &&    // to make depgen alittle faster
       
   782                                         opt != "kernel32.lib" &&
       
   783                                         opt != "user32.lib" &&
       
   784                                         opt != "gdi32.lib" &&
       
   785                                         opt != "comdlg32.lib" &&
       
   786                                         opt != "advapi32.lib" &&
       
   787                                         opt != "shell32.lib" &&
       
   788                                         opt != "ole32.lib" &&
       
   789                                         opt != "oleaut32.lib" &&
       
   790                                         opt != "uuid.lib" &&
       
   791                                         opt != "imm32.lib" &&
       
   792                                         opt != "winmm.lib" &&
       
   793                                         opt != "wsock32.lib" &&
       
   794                                         opt != "ws2_32.lib" &&
       
   795                                         opt != "winspool.lib" &&
       
   796                                         opt != "delayimp.lib")
       
   797                                     {
       
   798                                         newDep->dependencies << opt.section(Option::dir_sep, -1);
       
   799                                     }
       
   800                                 }
       
   801                         }
       
   802                         workspace_cleanup.append(newDep);
       
   803                         workspace_depends.insert(newDep->target, newDep);
       
   804 
       
   805                         debug_msg(1, "Generator: MSVC: Added project (name:'%s'  path:'%s'  deps:'%s')",
       
   806                                   qPrintable(newDep->target) , qPrintable(newDep->dspProjectFile),
       
   807                                   qPrintable(newDep->dependencies.join(";")));
       
   808                     }
       
   809                 }
       
   810 nextfile:
       
   811                 qmake_setpwd(oldpwd);
       
   812             }
       
   813         }
       
   814     }
       
   815 
       
   816     // Restore previous after_user_var options
       
   817     Option::after_user_vars = old_after_vars;
       
   818 
       
   819     // Output all projects
       
   820     QString dswProjectName = QLatin1String(_dswProjectName);
       
   821     QString dswProjectDep  = QLatin1String(_dswProjectDep);
       
   822     for(QList<WorkspaceDepend*>::Iterator it = workspace_cleanup.begin(); it != workspace_cleanup.end(); ++it) {
       
   823         t << _dswDevider;
       
   824         t << endl;
       
   825         t << dswProjectName.arg((*it)->orig_target).arg((*it)->dspProjectFile);
       
   826         t << endl;
       
   827         t << _dswPackage5Start;
       
   828         t << _dswPackage5Stop;
       
   829         t << endl;
       
   830         t << _dswPackage4Start;
       
   831 
       
   832         // Output project dependencies
       
   833         for(QStringList::iterator dit = (*it)->dependencies.begin();  dit != (*it)->dependencies.end(); ++dit) {
       
   834             if(WorkspaceDepend *vc = workspace_depends[*dit])
       
   835                 t << dswProjectDep.arg(vc->orig_target);
       
   836         }
       
   837 
       
   838         t << _dswPackage4Stop;
       
   839     }
       
   840 
       
   841     // Output global part
       
   842     t << _dswDevider << endl;
       
   843     t << _dswGlobal;
       
   844     t << _dswDevider;
       
   845     t << endl << endl;
       
   846 }
       
   847 
       
   848 class FolderGroup
       
   849 {
       
   850 public:
       
   851     QString name;
       
   852     QString filter;
       
   853     QMap<QString, FolderGroup *> subFolders;
       
   854     QMap<QString, QString> files;
       
   855 
       
   856     void insertStructured(const QString &file, const QString &fileListName)
       
   857     {
       
   858         QStringList path = QFileInfo(file).path().split("/");
       
   859         if (!path.isEmpty() && path.at(0) == ".")
       
   860             path.takeAt(0);
       
   861         FolderGroup *currentFolder = this;
       
   862         for (int i = 0; i < path.size(); i++) {
       
   863             if (currentFolder->subFolders.contains(path.at(i))) {
       
   864                 currentFolder = currentFolder->subFolders.value(path.at(i));
       
   865             } else {
       
   866                 FolderGroup *newFolder = new FolderGroup;
       
   867                 newFolder->name = path.at(i);
       
   868                 currentFolder->subFolders.insert(path.at(i), newFolder);
       
   869                 currentFolder = newFolder;
       
   870             }
       
   871         }
       
   872         currentFolder->files.insert(file, fileListName);
       
   873     }
       
   874 
       
   875     void insertFlat(const QString &file, const QString &fileListName)
       
   876     {
       
   877         files.insert(file, fileListName);
       
   878     }
       
   879 
       
   880     ~FolderGroup()
       
   881     {
       
   882         qDeleteAll(subFolders.values());
       
   883     }
       
   884 };
       
   885 
       
   886 bool DspMakefileGenerator::writeFileGroup(QTextStream &t, const QStringList &listNames, const QString &group, const QString &filter)
       
   887 {
       
   888     FolderGroup root;
       
   889     root.name = group;
       
   890     root.filter = filter;
       
   891 
       
   892     for (int i = 0; i < listNames.count(); ++i) {
       
   893         QStringList list = project->values(listNames.at(i));
       
   894         for (int j = 0; j < list.count(); ++j) {
       
   895             const QString name = list.at(j);
       
   896             if (name.isEmpty())
       
   897                 continue;
       
   898             if (project->isActiveConfig("flat"))
       
   899                 root.insertFlat(name, listNames.at(i));
       
   900             else
       
   901                 root.insertStructured(name, listNames.at(i));
       
   902         }
       
   903     }
       
   904 
       
   905     if (root.files.isEmpty() && root.subFolders.isEmpty())
       
   906         return true;
       
   907 
       
   908     writeSubFileGroup(t, &root);
       
   909 
       
   910     return true;
       
   911 }
       
   912 
       
   913 void DspMakefileGenerator::writeSubFileGroup(QTextStream &t, FolderGroup *folder)
       
   914 {
       
   915     t << "# Begin Group \"" << folder->name << "\"" << endl;
       
   916     t << "# PROP Default_Filter \"" << folder->filter << "\"" << endl;
       
   917     QMap<QString, FolderGroup *>::const_iterator folderIt = folder->subFolders.begin();
       
   918     while (folderIt != folder->subFolders.end()) {
       
   919         writeSubFileGroup(t, folderIt.value());
       
   920         ++folderIt;
       
   921     }
       
   922     QMap<QString, QString>::const_iterator it = folder->files.begin();
       
   923     while (it != folder->files.end()) {
       
   924         t << "# Begin Source File" << endl;
       
   925         t << "SOURCE=" << escapeFilePath(it.key()) << endl;
       
   926         writeBuildstepForFile(t, it.key(), it.value());
       
   927         t << "# End Source File" << endl;
       
   928         t << endl;
       
   929         ++it;
       
   930     }
       
   931     t << "# End Group" << endl;
       
   932     t << endl;
       
   933 }
       
   934 
       
   935 bool DspMakefileGenerator::writeBuildstepForFile(QTextStream &t, const QString &file, const QString &listName)
       
   936 {
       
   937 
       
   938     if (!mergedProjects.count()) {
       
   939         t << writeBuildstepForFileForConfig(file, listName, this);
       
   940         return true;
       
   941     }
       
   942 
       
   943     //only add special build rules when needed
       
   944 
       
   945     QStringList specialBuilds;
       
   946     int i = 0;
       
   947     for (i = 0; i < mergedProjects.count(); ++i)
       
   948         specialBuilds += writeBuildstepForFileForConfig(file, listName, mergedProjects.at(i));
       
   949 
       
   950     // no special build just return
       
   951     if (specialBuilds.join("").isEmpty())
       
   952         return true;
       
   953 
       
   954     for (i = 0; i < mergedProjects.count(); ++i) {
       
   955         if (i == 0)
       
   956             t << "!IF";
       
   957         else
       
   958             t << "!ELSEIF";
       
   959         t << " \"$(CFG)\" == \"" << configName(mergedProjects.at(i)) << "\"" << endl;
       
   960         t << endl;
       
   961         t << specialBuilds.at(i);
       
   962         t << endl;
       
   963     }
       
   964 
       
   965     t << "!ENDIF" << endl;
       
   966 
       
   967     return true;
       
   968 }
       
   969 
       
   970 bool DspMakefileGenerator::writeDspConfig(QTextStream &t, DspMakefileGenerator *config)
       
   971 {
       
   972 
       
   973     bool isDebug = config->project->isActiveConfig("debug");
       
   974     bool staticLibTarget = config->var("MSVCDSP_DSPTYPE") == "0x0104";
       
   975 
       
   976     QString outDir = Option::fixPathToTargetOS(config->project->first("DESTDIR"));
       
   977     while (outDir.endsWith(Option::dir_sep))
       
   978         outDir.chop(1);
       
   979     outDir = config->escapeFilePath(outDir);
       
   980 
       
   981     QString intDir = config->project->first("OBJECTS_DIR");
       
   982     while (intDir.endsWith(Option::dir_sep))
       
   983         intDir.chop(1);
       
   984     intDir = config->escapeFilePath(intDir);
       
   985 
       
   986     t << "# PROP BASE Use_MFC 0" << endl;
       
   987     t << "# PROP BASE Use_Debug_Libraries " << (isDebug ? "1" : "0") << endl;
       
   988     t << "# PROP BASE Output_Dir " << outDir << endl;
       
   989     t << "# PROP BASE Intermediate_Dir " << intDir << endl;
       
   990     t << "# PROP BASE Target_Dir \"\"" << endl;
       
   991     t << "# PROP Use_MFC 0" << endl;
       
   992     t << "# PROP Use_Debug_Libraries " << (isDebug ? "1" : "0") << endl;
       
   993 
       
   994     t << "# PROP Output_Dir " << outDir << endl;
       
   995     t << "# PROP Intermediate_Dir " << intDir << endl;
       
   996     if (config->project->isActiveConfig("dll") || config->project->isActiveConfig("plugin"))
       
   997         t << "# PROP Ignore_Export_Lib 1" << endl;
       
   998     t << "# PROP Target_Dir \"\"" << endl;
       
   999     t << "# ADD CPP " << config->var("MSVCDSP_INCPATH") << " /c /FD " << config->var("QMAKE_CXXFLAGS") << " " << config->var("MSVCDSP_DEFINES") << " " << config->var("PRECOMPILED_FLAGS") << endl;
       
  1000     t << "# ADD MTL /nologo /mktyplib203 /win32 /D " << (isDebug ? "\"_DEBUG\"" : "\"NDEBUG\"") << endl;
       
  1001     t << "# ADD RSC /l 0x409 /d " << (isDebug ? "\"_DEBUG\"" : "\"NDEBUG\"") << endl;
       
  1002     t << "# ADD BSC32 /nologo" << endl;
       
  1003     if (staticLibTarget) {
       
  1004         t << "LIB32=" << config->var("QMAKE_LIB") << endl;
       
  1005         t << "# ADD LIB32 " << config->var("MSVCDSP_TARGET") << " " << config->var("PRECOMPILED_OBJECT") << endl;
       
  1006     } else {
       
  1007         t << "LINK32=" << config->var("QMAKE_LINK") << endl;
       
  1008         t << "# ADD LINK32 " << config->var("MSVCDSP_LFLAGS") << " " << config->var("MSVCDSP_LIBS") << " " << config->var("MSVCDSP_TARGET") << " " << config->var("PRECOMPILED_OBJECT") << endl;
       
  1009     }
       
  1010 
       
  1011     if (!config->project->values("MSVCDSP_PRE_LINK").isEmpty())
       
  1012         t << config->project->values("MSVCDSP_PRE_LINK").first();
       
  1013 
       
  1014     if (!config->project->values("MSVCDSP_POST_LINK").isEmpty())
       
  1015         t << config->project->values("MSVCDSP_POST_LINK").first();
       
  1016 
       
  1017     return true;
       
  1018 }
       
  1019 
       
  1020 QString DspMakefileGenerator::writeBuildstepForFileForConfig(const QString &file, const QString &listName, DspMakefileGenerator *config)
       
  1021 {
       
  1022     QString ret;
       
  1023     QTextStream t(&ret);
       
  1024 
       
  1025     // exclude from build
       
  1026     if (!config->project->values(listName).contains(file)) {
       
  1027         t << "# PROP Exclude_From_Build 1" << endl;
       
  1028         return ret;
       
  1029     }
       
  1030 
       
  1031     if (config->usePCH) {
       
  1032         bool c_file = false;
       
  1033         for (QStringList::Iterator it = Option::c_ext.begin(); it != Option::c_ext.end(); ++it) {
       
  1034             if (file.endsWith(*it)) {
       
  1035                 c_file = true;
       
  1036                 break;
       
  1037             }
       
  1038         }
       
  1039         if(c_file) {
       
  1040             t << "# SUBTRACT CPP /FI" << config->escapeFilePath(config->namePCH) << " /Yu" << config->escapeFilePath(config->namePCH) << " /Fp" << endl;
       
  1041             return ret;
       
  1042         } else if (config->precompH.endsWith(file)) {
       
  1043             // ### dependency list quickly becomes too long for VS to grok...
       
  1044             t << "USERDEP_" << file << "=" << config->valGlue(config->escapeFilePaths(config->findDependencies(config->precompH)), "", "\t", "") << endl;
       
  1045             t << endl;
       
  1046             t << "# Begin Custom Build - Creating precompiled header from " << file << "..." << endl;
       
  1047             t << "InputPath=.\\" << config->escapeFilePath(file) << endl << endl;
       
  1048             t << config->precompPch + ": $(SOURCE) \"$(IntDir)\" \"$(OUTDIR)\"" << endl;
       
  1049             t << "\t" << config->var("QMAKE_CC") << " /TP /W3 /FD /c /Yc /Fp" << config->precompPch << " /Fo" << config->precompObj << " /Fd\"$(IntDir)\\\\\" " << file << " ";
       
  1050             t << config->var("MSVCDSP_INCPATH") << " " << config->var("MSVCDSP_DEFINES") << " " << config->var("QMAKE_CXXFLAGS") << endl;
       
  1051             t << "# End Custom Build" << endl << endl;
       
  1052             return ret;
       
  1053         }
       
  1054     }
       
  1055 
       
  1056     QString fileBase = QFileInfo(file).completeBaseName();
       
  1057 
       
  1058     bool hasBuiltin = config->hasBuiltinCompiler(file);
       
  1059     BuildStep allSteps;
       
  1060 
       
  1061     if (!config->swappedBuildSteps.contains(file)) {
       
  1062         QStringList compilers = config->project->values("QMAKE_EXTRA_COMPILERS");
       
  1063         for (int i = 0; i < compilers.count(); ++i) {
       
  1064             QString compiler = compilers.at(i);
       
  1065             if (config->project->values(compiler + ".input").isEmpty())
       
  1066                 continue;
       
  1067             QString input = config->project->values(compiler + ".input").first();
       
  1068             QStringList inputList = config->project->values(input);
       
  1069             if (!inputList.contains(file))
       
  1070                 continue;
       
  1071 
       
  1072             QStringList compilerCommands = config->project->values(compiler + ".commands");
       
  1073             QStringList compilerOutput = config->project->values(compiler + ".output");
       
  1074             if (compilerCommands.isEmpty() || compilerOutput.isEmpty())
       
  1075                 continue;
       
  1076 
       
  1077             QStringList compilerName = config->project->values(compiler + ".name");
       
  1078             if (compilerName.isEmpty())
       
  1079                 compilerName << compiler;
       
  1080             QStringList compilerDepends = config->project->values(compiler + ".depends");
       
  1081             QString compilerDependsCommand = config->project->values(compiler + ".depend_command").join(" ");
       
  1082             if (!compilerDependsCommand.isEmpty()) {
       
  1083                 if(!config->canExecute(compilerDependsCommand))
       
  1084                     compilerDependsCommand = QString();
       
  1085             }
       
  1086             QStringList compilerConfig = config->project->values(compiler + ".CONFIG");
       
  1087 
       
  1088             if (!config->verifyExtraCompiler(compiler, file))
       
  1089                 continue;
       
  1090 
       
  1091             bool combineAll = compilerConfig.contains("combine");
       
  1092             if (combineAll && inputList.first() != file)
       
  1093                 continue;
       
  1094 
       
  1095             QString fileIn("$(InputPath)");
       
  1096 
       
  1097             if (combineAll && !inputList.isEmpty()) {
       
  1098                 fileIn = inputList.join(" ");
       
  1099                 compilerDepends += inputList;
       
  1100             }
       
  1101 
       
  1102             QString fileOut = compilerOutput.first();
       
  1103             QString fileOutBase = QFileInfo(fileOut).completeBaseName();
       
  1104             fileOut.replace("${QMAKE_FILE_IN}", fileIn);
       
  1105             fileOut.replace("${QMAKE_FILE_BASE}", fileBase);
       
  1106             fileOut.replace("${QMAKE_FILE_OUT_BASE}", fileOutBase);
       
  1107             fileOut.replace('/', '\\');
       
  1108 
       
  1109             BuildStep step;
       
  1110             for (int i2 = 0; i2 < compilerDepends.count(); ++i2) {
       
  1111                 QString dependency = compilerDepends.at(i2);
       
  1112                 dependency.replace("${QMAKE_FILE_IN}", fileIn);
       
  1113                 dependency.replace("${QMAKE_FILE_BASE}", fileBase);
       
  1114                 dependency.replace("${QMAKE_FILE_OUT_BASE}", fileOutBase);
       
  1115                 dependency.replace('/', '\\');
       
  1116                 if (!step.deps.contains(dependency, Qt::CaseInsensitive))
       
  1117                     step.deps << dependency;
       
  1118             }
       
  1119             // depends command
       
  1120             if (!compilerDependsCommand.isEmpty() && config->doDepends()) {
       
  1121                 char buff[256];
       
  1122                 QString dep_cmd = config->replaceExtraCompilerVariables(compilerDependsCommand, file,
       
  1123                     fileOut);
       
  1124                 dep_cmd = Option::fixPathToLocalOS(dep_cmd, true, false);
       
  1125                 if(config->canExecute(dep_cmd)) {
       
  1126                     if(FILE *proc = QT_POPEN(dep_cmd.toLatin1().constData(), "r")) {
       
  1127                         QString indeps;
       
  1128                         while(!feof(proc)) {
       
  1129                             int read_in = (int)fread(buff, 1, 255, proc);
       
  1130                             if(!read_in)
       
  1131                                 break;
       
  1132                             indeps += QByteArray(buff, read_in);
       
  1133                         }
       
  1134                         QT_PCLOSE(proc);
       
  1135                         if(!indeps.isEmpty())
       
  1136                             step.deps += config->fileFixify(indeps.replace('\n', ' ').simplified().split(' '));
       
  1137                     }
       
  1138                 }
       
  1139             }
       
  1140 
       
  1141 
       
  1142             QString mappedFile;
       
  1143             if (hasBuiltin) {
       
  1144                 mappedFile = fileOut;
       
  1145                 fileOut = fileIn;
       
  1146                 fileIn = file;
       
  1147             }
       
  1148 
       
  1149             step.buildStep += " \\\n\t";
       
  1150             QString command(compilerCommands.join(" "));
       
  1151             // Replace any newlines with proper line-continuance
       
  1152             command.replace("\n", " \\\n\t");
       
  1153             // Might be a macro, and not a valid filename, so the replaceExtraCompilerVariables() would eat it
       
  1154             command.replace("${QMAKE_FILE_IN}", config->escapeFilePath(fileIn));
       
  1155             command.replace("${QMAKE_FILE_BASE}", config->escapeFilePath(fileBase));
       
  1156             command.replace("${QMAKE_FILE_OUT_BASE}", config->escapeFilePath(fileOutBase));
       
  1157             command.replace("${QMAKE_FILE_OUT}", config->escapeFilePath(fileOut));
       
  1158 
       
  1159             command = config->replaceExtraCompilerVariables(command, fileIn, fileOut);
       
  1160 
       
  1161             step.buildName = compilerName.first();
       
  1162             step.buildStep += command;
       
  1163             step.buildOutputs += fileOut;
       
  1164 
       
  1165             if (hasBuiltin) {
       
  1166                 step.deps << fileIn;
       
  1167                 config->swappedBuildSteps[mappedFile] = step;
       
  1168             } else {
       
  1169                 allSteps << step;
       
  1170             }
       
  1171         }
       
  1172     } else {
       
  1173         allSteps << config->swappedBuildSteps.value(file);
       
  1174     }
       
  1175 
       
  1176     if (allSteps.buildStep.isEmpty())
       
  1177         return ret;
       
  1178 
       
  1179     int i;
       
  1180     QStringList dependencyList;
       
  1181     // remove dependencies that are also output
       
  1182     for (i = 0; i < 1; ++i) {
       
  1183         QStringList buildOutput(allSteps.buildOutputs.at(i));
       
  1184 
       
  1185         for (int i2 = 0; i2 < allSteps.deps.count(); ++i2) {
       
  1186             QString dependency = allSteps.deps.at(i2);
       
  1187             if (!buildOutput.contains(dependency) && !dependencyList.contains(dependency))
       
  1188                 dependencyList << dependency;
       
  1189         }
       
  1190     }
       
  1191     QString allDependencies = config->valGlue(dependencyList, "", "\t", "");
       
  1192     t << "USERDEP_" << file << "=" << allDependencies << endl;
       
  1193     t << "# PROP Ignore_Default_Tool 1" << endl;
       
  1194     t << "# Begin Custom Build - Running " << allSteps.buildName << " on " << file << endl;
       
  1195     t << "InputPath=" << file << endl;
       
  1196     t << "BuildCmds= " << allSteps.buildStep << endl;
       
  1197     for (i = 0; i < allSteps.buildOutputs.count(); ++i) {
       
  1198         t << config->escapeFilePath(allSteps.buildOutputs.at(i))
       
  1199           << " : $(SOURCE) $(INTDIR) $(OUTDIR)\n\t$(BuildCmds)\n";
       
  1200     }
       
  1201     t << endl;
       
  1202     t << "# End Custom Build" << endl;
       
  1203 
       
  1204     return ret;
       
  1205 }
       
  1206 
       
  1207 QT_END_NAMESPACE