1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306
|
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
file LICENSE.rst or https://cmake.org/licensing for details. */
#pragma once
#include "cmConfigure.h" // IWYU pragma: keep
#include <iosfwd>
#include <map>
#include <set>
#include <string>
#include <utility>
#include <vector>
#include "cmDepends.h"
#include "cmGeneratorOptions.h"
#include "cmLocalCommonGenerator.h"
class cmCustomCommand;
class cmCustomCommandGenerator;
class cmGeneratorTarget;
class cmGlobalGenerator;
class cmMakefile;
class cmSourceFile;
/** \class cmLocalUnixMakefileGenerator3
* \brief Write a LocalUnix makefiles.
*
* cmLocalUnixMakefileGenerator3 produces a LocalUnix makefile from its
* member Makefile.
*/
class cmLocalUnixMakefileGenerator3 : public cmLocalCommonGenerator
{
public:
cmLocalUnixMakefileGenerator3(cmGlobalGenerator* gg, cmMakefile* mf);
~cmLocalUnixMakefileGenerator3() override;
std::string const& GetConfigName() const;
void ComputeHomeRelativeOutputPath() override;
/**
* Generate the makefile for this directory.
*/
void Generate() override;
// this returns the relative path between the HomeOutputDirectory and this
// local generators StartOutputDirectory
std::string const& GetHomeRelativeOutputPath();
/**
* Convert a file path to a Makefile target or dependency with
* escaping and quoting suitable for the generator's make tool.
*/
std::string ConvertToMakefilePath(std::string const& path) const;
// Write out a make rule
void WriteMakeRule(std::ostream& os, char const* comment,
std::string const& target,
std::vector<std::string> const& depends,
std::vector<std::string> const& commands, bool symbolic,
bool in_help = false);
// write the main variables used by the makefiles
void WriteMakeVariables(std::ostream& makefileStream);
/**
* Set max makefile variable size, default is 0 which means unlimited.
*/
void SetMakefileVariableSize(int s) { this->MakefileVariableSize = 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, std::string const& 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(std::string const& makefile,
std::string const& tgt);
// append flags to a string
void AppendFlags(std::string& flags,
std::string const& newFlags) const override;
using cmLocalCommonGenerator::AppendFlags;
// append an echo command
enum EchoColor
{
EchoNormal,
EchoDepend,
EchoBuild,
EchoLink,
EchoGenerate,
EchoGlobal
};
struct EchoProgress
{
std::string Dir;
std::string Arg;
};
void AppendEcho(std::vector<std::string>& commands, std::string const& text,
EchoColor color = EchoNormal, EchoProgress const* = nullptr);
/** Get whether the makefile is to have color. */
bool GetColorMakefile() const { return this->ColorMakefile; }
// create a command that cds to the start dir then runs the commands
void CreateCDCommand(std::vector<std::string>& commands,
std::string const& targetDir,
std::string const& relDir);
static std::string ConvertToQuotedOutputPath(std::string const& p,
bool useWatcomQuote);
std::string CreateMakeVariable(std::string const& sin,
std::string const& s2in);
/** Called from command-line hook to bring dependencies up to date
for a target. */
bool UpdateDependencies(std::string const& tgtInfo, bool verbose,
bool color) override;
/** Called from command-line hook to clear dependencies. */
void ClearDependencies(cmMakefile* mf, bool verbose) override;
/** write some extra rules such as make test etc */
void WriteSpecialTargetsTop(std::ostream& makefileStream);
void WriteSpecialTargetsBottom(std::ostream& makefileStream);
std::string GetRelativeTargetDirectory(
cmGeneratorTarget const* target) const;
// File pairs for implicit dependency scanning. The key of the map
// is the depender and the value is the explicit dependee.
using ImplicitDependFileMap = cmDepends::DependencyMap;
using ImplicitDependLanguageMap =
std::map<std::string, ImplicitDependFileMap>;
using ImplicitDependScannerMap =
std::map<cmDependencyScannerKind, ImplicitDependLanguageMap>;
using ImplicitDependTargetMap =
std::map<std::string, ImplicitDependScannerMap>;
ImplicitDependLanguageMap const& GetImplicitDepends(
cmGeneratorTarget const* tgt,
cmDependencyScannerKind scanner = cmDependencyScannerKind::CMake);
void AddImplicitDepends(
cmGeneratorTarget const* tgt, std::string const& lang,
std::string const& obj, std::string const& src,
cmDependencyScannerKind scanner = cmDependencyScannerKind::CMake);
// 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() const
{
return !this->SkipPreprocessedSourceRules;
}
bool GetCreateAssemblySourceRules() const
{
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);
std::string GetLinkDependencyFile(cmGeneratorTarget* target,
std::string const& config) const override;
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,
cmGeneratorTarget* tgt);
// this converts a file name that is relative to the StartOutputDirectory
// into a full path
std::string ConvertToFullPath(std::string const& localPath);
void WriteConvenienceRule(std::ostream& ruleFileStream,
std::string const& realTarget,
std::string const& helpTarget);
void AppendRuleDepend(std::vector<std::string>& depends,
char const* ruleFileName);
void AppendRuleDepends(std::vector<std::string>& depends,
std::vector<std::string> const& ruleFiles);
void AppendCustomDepends(std::vector<std::string>& depends,
std::vector<cmCustomCommand> const& ccs);
void AppendCustomDepend(std::vector<std::string>& depends,
cmCustomCommandGenerator const& cc);
void AppendCustomCommands(std::vector<std::string>& commands,
std::vector<cmCustomCommand> const& ccs,
cmGeneratorTarget* target,
std::string const& relative);
void AppendCustomCommand(std::vector<std::string>& commands,
cmCustomCommandGenerator const& ccg,
cmGeneratorTarget* target,
std::string const& relative,
bool echo_comment = false,
std::ostream* content = nullptr);
void AppendCleanCommand(std::vector<std::string>& commands,
std::set<std::string> const& files,
cmGeneratorTarget* target,
char const* filename = nullptr);
void AppendDirectoryCleanCommand(std::vector<std::string>& commands);
// Helper methods for dependency updates.
bool ScanDependencies(std::string const& targetDir,
std::string const& dependFile,
std::string const& internalDependFile,
cmDepends::DependencyMap& validDeps);
void CheckMultipleOutputs(bool verbose);
private:
std::string MaybeConvertWatcomShellCommand(std::string const& cmd);
friend class cmMakefileTargetGenerator;
friend class cmMakefileExecutableTargetGenerator;
friend class cmMakefileLibraryTargetGenerator;
friend class cmMakefileUtilityTargetGenerator;
friend class cmGlobalUnixMakefileGenerator3;
ImplicitDependTargetMap ImplicitDepends;
std::string HomeRelativeOutputPath;
struct LocalObjectEntry
{
cmGeneratorTarget* Target = nullptr;
std::string Language;
LocalObjectEntry() = default;
LocalObjectEntry(cmGeneratorTarget* t, std::string lang)
: Target(t)
, Language(std::move(lang))
{
}
};
struct LocalObjectInfo : public std::vector<LocalObjectEntry>
{
bool HasSourceExtension = false;
bool HasPreprocessRule = false;
bool HasAssembleRule = false;
};
void GetLocalObjectFiles(
std::map<std::string, LocalObjectInfo>& localObjectFiles);
void WriteObjectConvenienceRule(std::ostream& ruleFileStream,
char const* comment,
std::string const& 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;
int MakefileVariableSize;
bool MakeCommandEscapeTargetTwice;
bool BorlandMakeCurlyHack;
bool ColorMakefile;
bool SkipPreprocessedSourceRules;
bool SkipAssemblySourceRules;
std::set<cmSourceFile const*>& GetCommandsVisited(
cmGeneratorTarget const* target)
{
return this->CommandsVisited[target];
}
std::map<cmGeneratorTarget const*, std::set<cmSourceFile const*>>
CommandsVisited;
};
|