In commit v3.2.0-rc1~272^2~2 (Makefile: Fix rebuild with multiple custom command outputs, 2014-12-05) we changed the generated makefile pattern for multiple outputs from out1: depends... commands... out2: out1 to out1 out2: depends... commands... This was based on the incorrect assumption that make tools would treat this as a combined output rule and run the command(s) exactly once for them. It turns out that instead this new pattern is equivalent to out1: depends... commands... out2: depends... commands... so the commands may be run more than once. Some documents suggest using a "dedicated witness" stamp file: stamp: depends... rm -f stamp touch stamp.tmp commands... mv stamp.tmp stamp out1 out2: stamp However, if the commands fail the error message will refer to the stamp instead of any of the real outputs, which may be confusing to readers. Also, this approach seems to have the same behavior of the original approach that motiviated the above commit: multiple invocations are needed to bring consumers of the outputs up to date. Instead we can return to the original approach but add an explicit touch to each extra output rule: out1: depends... commands... out2: out1 touch -c out2 This causes make tools to recognize that all outputs have changed and therefore to execute any commands that consume them.
386 lines
14 KiB
C++
386 lines
14 KiB
C++
/*============================================================================
|
|
CMake - Cross Platform Makefile Generator
|
|
Copyright 2000-2009 Kitware, Inc., Insight Software Consortium
|
|
|
|
Distributed under the OSI-approved BSD License (the "License");
|
|
see accompanying file Copyright.txt for details.
|
|
|
|
This software is distributed WITHOUT ANY WARRANTY; without even the
|
|
implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
See the License for more information.
|
|
============================================================================*/
|
|
#ifndef cmLocalUnixMakefileGenerator3_h
|
|
#define cmLocalUnixMakefileGenerator3_h
|
|
|
|
#include "cmLocalGenerator.h"
|
|
|
|
// for cmDepends::DependencyVector
|
|
#include "cmDepends.h"
|
|
|
|
class cmCustomCommand;
|
|
class cmCustomCommandGenerator;
|
|
class cmDependInformation;
|
|
class cmDepends;
|
|
class cmMakefileTargetGenerator;
|
|
class cmTarget;
|
|
class cmSourceFile;
|
|
|
|
/** \class cmLocalUnixMakefileGenerator3
|
|
* \brief Write a LocalUnix makefiles.
|
|
*
|
|
* cmLocalUnixMakefileGenerator3 produces a LocalUnix makefile from its
|
|
* member Makefile.
|
|
*/
|
|
class cmLocalUnixMakefileGenerator3 : public cmLocalGenerator
|
|
{
|
|
public:
|
|
cmLocalUnixMakefileGenerator3();
|
|
virtual ~cmLocalUnixMakefileGenerator3();
|
|
|
|
/**
|
|
* Process the CMakeLists files for this directory to fill in the
|
|
* Makefile ivar
|
|
*/
|
|
virtual void Configure();
|
|
|
|
/**
|
|
* Generate the makefile for this directory.
|
|
*/
|
|
virtual void Generate();
|
|
|
|
|
|
// this returns the relative path between the HomeOutputDirectory and this
|
|
// local generators StartOutputDirectory
|
|
const std::string &GetHomeRelativeOutputPath();
|
|
|
|
// Write out a make rule
|
|
void WriteMakeRule(std::ostream& os,
|
|
const char* comment,
|
|
const std::string& target,
|
|
const std::vector<std::string>& depends,
|
|
const std::vector<std::string>& commands,
|
|
bool symbolic,
|
|
bool in_help = false);
|
|
|
|
// write the main variables used by the makefiles
|
|
void WriteMakeVariables(std::ostream& makefileStream);
|
|
|
|
/**
|
|
* If true, then explicitly pass MAKEFLAGS on the make all target for makes
|
|
* that do not use environment variables.
|
|
*
|
|
*/
|
|
void SetPassMakeflags(bool s){this->PassMakeflags = s;}
|
|
bool GetPassMakeflags() { return this->PassMakeflags; }
|
|
|
|
/**
|
|
* Set the flag used to keep the make program silent.
|
|
*/
|
|
void SetMakeSilentFlag(const std::string& s) { this->MakeSilentFlag = s; }
|
|
std::string &GetMakeSilentFlag() { return this->MakeSilentFlag; }
|
|
|
|
/**
|
|
* Set to true if the shell being used is the windows shell.
|
|
* This controls if statements in the makefile and the SHELL variable.
|
|
* The default is false.
|
|
*/
|
|
void SetWindowsShell(bool v) {this->WindowsShell = v;}
|
|
|
|
/**
|
|
* Set to true if the make tool being used is Watcom WMake.
|
|
*/
|
|
void SetWatcomWMake(bool v) {this->WatcomWMake = v;}
|
|
|
|
/**
|
|
* Set to true if the make tool being used is MinGW Make.
|
|
*/
|
|
void SetMinGWMake(bool v) {this->MinGWMake = v;}
|
|
|
|
/**
|
|
* Set to true if the make tool being used is NMake.
|
|
*/
|
|
void SetNMake(bool v) {this->NMake = v;}
|
|
|
|
/**
|
|
* Set to true if the shell being used is the MSYS shell.
|
|
* This controls if statements in the makefile and the SHELL variable.
|
|
* The default is false.
|
|
*/
|
|
void SetMSYSShell(bool v) {this->MSYSShell = v;}
|
|
|
|
/**
|
|
* If set to true, then NULL is set to nil for non Windows_NT.
|
|
* This uses make syntax used by nmake and borland.
|
|
* The default is false.
|
|
*/
|
|
void SetDefineWindowsNULL(bool v) {this->DefineWindowsNULL = v;}
|
|
|
|
/**
|
|
* If set to true, cd dir && command is used to
|
|
* run commands in a different directory.
|
|
*/
|
|
void SetUnixCD(bool v) {this->UnixCD = v;}
|
|
|
|
/**
|
|
* Set the string used to include one makefile into another default
|
|
* is include.
|
|
*/
|
|
void SetIncludeDirective(const std::string& s)
|
|
{ this->IncludeDirective = s; }
|
|
const std::string& GetIncludeDirective() { return this->IncludeDirective; }
|
|
|
|
/**
|
|
* Set max makefile variable size, default is 0 which means unlimited.
|
|
*/
|
|
void SetMakefileVariableSize(int s) { this->MakefileVariableSize = s; }
|
|
|
|
/**
|
|
* If ignore lib prefix is true, then do not strip lib from the name
|
|
* of a library.
|
|
*/
|
|
void SetIgnoreLibPrefix(bool s) { this->IgnoreLibPrefix = s; }
|
|
|
|
/**
|
|
* Set whether passing a make target on a command line requires an
|
|
* extra level of escapes.
|
|
*/
|
|
void SetMakeCommandEscapeTargetTwice(bool b)
|
|
{ this->MakeCommandEscapeTargetTwice = b; }
|
|
|
|
/**
|
|
* Set whether the Borland curly brace command line hack should be
|
|
* applied.
|
|
*/
|
|
void SetBorlandMakeCurlyHack(bool b)
|
|
{ this->BorlandMakeCurlyHack = b; }
|
|
|
|
// used in writing out Cmake files such as WriteDirectoryInformation
|
|
static void WriteCMakeArgument(std::ostream& os, const char* s);
|
|
|
|
/** creates the common disclaimer text at the top of each makefile */
|
|
void WriteDisclaimer(std::ostream& os);
|
|
|
|
// write a comment line #====... in the stream
|
|
void WriteDivider(std::ostream& os);
|
|
|
|
/** used to create a recursive make call */
|
|
std::string GetRecursiveMakeCall(const char *makefile,
|
|
const std::string& tgt);
|
|
|
|
// append flags to a string
|
|
virtual void AppendFlags(std::string& flags, const std::string& newFlags);
|
|
virtual void AppendFlags(std::string& flags, const char* newFlags);
|
|
|
|
// append an echo command
|
|
enum EchoColor { EchoNormal, EchoDepend, EchoBuild, EchoLink,
|
|
EchoGenerate, EchoGlobal };
|
|
void AppendEcho(std::vector<std::string>& commands, const char* text,
|
|
EchoColor color = EchoNormal);
|
|
|
|
/** Get whether the makefile is to have color. */
|
|
bool GetColorMakefile() const { return this->ColorMakefile; }
|
|
|
|
virtual std::string GetTargetDirectory(cmTarget const& target) const;
|
|
|
|
// create a command that cds to the start dir then runs the commands
|
|
void CreateCDCommand(std::vector<std::string>& commands,
|
|
const char *targetDir,
|
|
cmLocalGenerator::RelativeRoot returnDir);
|
|
|
|
static std::string ConvertToQuotedOutputPath(const char* p,
|
|
bool useWatcomQuote);
|
|
|
|
std::string CreateMakeVariable(const std::string& sin,
|
|
const std::string& s2in);
|
|
|
|
/** Called from command-line hook to bring dependencies up to date
|
|
for a target. */
|
|
virtual bool UpdateDependencies(const char* tgtInfo,
|
|
bool verbose, bool color);
|
|
|
|
/** Called from command-line hook to clear dependencies. */
|
|
virtual void ClearDependencies(cmMakefile* mf, bool verbose);
|
|
|
|
/** write some extra rules such as make test etc */
|
|
void WriteSpecialTargetsTop(std::ostream& makefileStream);
|
|
void WriteSpecialTargetsBottom(std::ostream& makefileStream);
|
|
|
|
std::string GetRelativeTargetDirectory(cmTarget const& target);
|
|
|
|
// File pairs for implicit dependency scanning. The key of the map
|
|
// is the depender and the value is the explicit dependee.
|
|
struct ImplicitDependFileMap:
|
|
public std::map<std::string, cmDepends::DependencyVector> {};
|
|
struct ImplicitDependLanguageMap:
|
|
public std::map<std::string, ImplicitDependFileMap> {};
|
|
struct ImplicitDependTargetMap:
|
|
public std::map<std::string, ImplicitDependLanguageMap> {};
|
|
ImplicitDependLanguageMap const& GetImplicitDepends(cmTarget const& tgt);
|
|
|
|
void AddImplicitDepends(cmTarget const& tgt, const std::string& lang,
|
|
const char* obj, const char* src);
|
|
|
|
void AppendGlobalTargetDepends(std::vector<std::string>& depends,
|
|
cmTarget& target);
|
|
|
|
// write the target rules for the local Makefile into the stream
|
|
void WriteLocalAllRules(std::ostream& ruleFileStream);
|
|
|
|
std::vector<std::string> const& GetLocalHelp() { return this->LocalHelp; }
|
|
|
|
/** Get whether to create rules to generate preprocessed and
|
|
assembly sources. This could be converted to a variable lookup
|
|
later. */
|
|
bool GetCreatePreprocessedSourceRules()
|
|
{
|
|
return !this->SkipPreprocessedSourceRules;
|
|
}
|
|
bool GetCreateAssemblySourceRules()
|
|
{
|
|
return !this->SkipAssemblySourceRules;
|
|
}
|
|
|
|
// Fill the vector with the target names for the object files,
|
|
// preprocessed files and assembly files. Currently only used by the
|
|
// Eclipse generator.
|
|
void GetIndividualFileTargets(std::vector<std::string>& targets);
|
|
|
|
protected:
|
|
void WriteLocalMakefile();
|
|
|
|
|
|
// write the target rules for the local Makefile into the stream
|
|
void WriteLocalMakefileTargets(std::ostream& ruleFileStream,
|
|
std::set<std::string> &emitted);
|
|
|
|
// this method Writes the Directory information files
|
|
void WriteDirectoryInformationFile();
|
|
|
|
|
|
// write the depend info
|
|
void WriteDependLanguageInfo(std::ostream& cmakefileStream, cmTarget &tgt);
|
|
|
|
// write the local help rule
|
|
void WriteHelpRule(std::ostream& ruleFileStream);
|
|
|
|
// this converts a file name that is relative to the StartOuputDirectory
|
|
// into a full path
|
|
std::string ConvertToFullPath(const std::string& localPath);
|
|
|
|
|
|
void WriteConvenienceRule(std::ostream& ruleFileStream,
|
|
const std::string& realTarget,
|
|
const std::string& helpTarget);
|
|
|
|
void WriteTargetDependRule(std::ostream& ruleFileStream,
|
|
cmTarget& target);
|
|
void WriteTargetCleanRule(std::ostream& ruleFileStream,
|
|
cmTarget& target,
|
|
const std::vector<std::string>& files);
|
|
void WriteTargetRequiresRule(std::ostream& ruleFileStream,
|
|
cmTarget& target,
|
|
const std::vector<std::string>& objects);
|
|
|
|
void AppendRuleDepend(std::vector<std::string>& depends,
|
|
const char* ruleFileName);
|
|
void AppendRuleDepends(std::vector<std::string>& depends,
|
|
std::vector<std::string> const& ruleFiles);
|
|
void AppendCustomDepends(std::vector<std::string>& depends,
|
|
const std::vector<cmCustomCommand>& ccs);
|
|
void AppendCustomDepend(std::vector<std::string>& depends,
|
|
cmCustomCommandGenerator const& cc);
|
|
void AppendCustomCommands(std::vector<std::string>& commands,
|
|
const std::vector<cmCustomCommand>& ccs,
|
|
cmTarget* target,
|
|
cmLocalGenerator::RelativeRoot relative =
|
|
cmLocalGenerator::HOME_OUTPUT);
|
|
void AppendCustomCommand(std::vector<std::string>& commands,
|
|
cmCustomCommandGenerator const& ccg,
|
|
cmTarget* target,
|
|
bool echo_comment=false,
|
|
cmLocalGenerator::RelativeRoot relative =
|
|
cmLocalGenerator::HOME_OUTPUT,
|
|
std::ostream* content = 0);
|
|
void AppendCleanCommand(std::vector<std::string>& commands,
|
|
const std::vector<std::string>& files,
|
|
cmTarget& target, const char* filename =0);
|
|
|
|
// Helper methods for dependeny updates.
|
|
bool ScanDependencies(const char* targetDir,
|
|
std::map<std::string, cmDepends::DependencyVector>& validDeps);
|
|
void CheckMultipleOutputs(bool verbose);
|
|
|
|
private:
|
|
std::string ConvertShellCommand(std::string const& cmd, RelativeRoot root);
|
|
std::string MakeLauncher(cmCustomCommandGenerator const& ccg,
|
|
cmTarget* target, RelativeRoot relative);
|
|
|
|
virtual void ComputeObjectFilenames(
|
|
std::map<cmSourceFile const*, std::string>& mapping,
|
|
cmGeneratorTarget const* gt = 0);
|
|
|
|
friend class cmMakefileTargetGenerator;
|
|
friend class cmMakefileExecutableTargetGenerator;
|
|
friend class cmMakefileLibraryTargetGenerator;
|
|
friend class cmMakefileUtilityTargetGenerator;
|
|
friend class cmGlobalUnixMakefileGenerator3;
|
|
|
|
ImplicitDependTargetMap ImplicitDepends;
|
|
|
|
//==========================================================================
|
|
// Configuration settings.
|
|
int MakefileVariableSize;
|
|
std::string IncludeDirective;
|
|
std::string MakeSilentFlag;
|
|
std::string ConfigurationName;
|
|
bool DefineWindowsNULL;
|
|
bool UnixCD;
|
|
bool PassMakeflags;
|
|
bool MakeCommandEscapeTargetTwice;
|
|
bool BorlandMakeCurlyHack;
|
|
//==========================================================================
|
|
|
|
std::string HomeRelativeOutputPath;
|
|
|
|
/* Copy the setting of CMAKE_COLOR_MAKEFILE from the makefile at the
|
|
beginning of generation to avoid many duplicate lookups. */
|
|
bool ColorMakefile;
|
|
|
|
/* Copy the setting of CMAKE_SKIP_PREPROCESSED_SOURCE_RULES and
|
|
CMAKE_SKIP_ASSEMBLY_SOURCE_RULES at the beginning of generation to
|
|
avoid many duplicate lookups. */
|
|
bool SkipPreprocessedSourceRules;
|
|
bool SkipAssemblySourceRules;
|
|
|
|
struct LocalObjectEntry
|
|
{
|
|
cmTarget* Target;
|
|
std::string Language;
|
|
LocalObjectEntry(): Target(0), Language() {}
|
|
LocalObjectEntry(cmTarget* t, const std::string& lang):
|
|
Target(t), Language(lang) {}
|
|
};
|
|
struct LocalObjectInfo: public std::vector<LocalObjectEntry>
|
|
{
|
|
bool HasSourceExtension;
|
|
bool HasPreprocessRule;
|
|
bool HasAssembleRule;
|
|
LocalObjectInfo():HasSourceExtension(false), HasPreprocessRule(false),
|
|
HasAssembleRule(false) {}
|
|
};
|
|
void GetLocalObjectFiles(
|
|
std::map<std::string, LocalObjectInfo> &localObjectFiles);
|
|
|
|
void WriteObjectConvenienceRule(std::ostream& ruleFileStream,
|
|
const char* comment, const char* output,
|
|
LocalObjectInfo const& info);
|
|
|
|
std::vector<std::string> LocalHelp;
|
|
|
|
/* does the work for each target */
|
|
std::map<std::string, std::string> MakeVariableMap;
|
|
std::map<std::string, std::string> ShortMakeVariableMap;
|
|
};
|
|
|
|
#endif
|