|
|
|
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
|
|
file Copyright.txt or https://cmake.org/licensing for details. */
|
|
|
|
#include "cmScriptGenerator.h"
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
#include "cmStringAlgorithms.h"
|
|
|
|
#include "cmSystemTools.h"
|
|
|
|
|
|
|
|
cmScriptGenerator::cmScriptGenerator(std::string config_var,
|
|
|
|
std::vector<std::string> configurations)
|
|
|
|
: RuntimeConfigVariable(std::move(config_var))
|
|
|
|
, Configurations(std::move(configurations))
|
|
|
|
, ConfigurationTypes(nullptr)
|
|
|
|
, ActionsPerConfig(false)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
cmScriptGenerator::~cmScriptGenerator() = default;
|
|
|
|
|
|
|
|
void cmScriptGenerator::Generate(
|
|
|
|
std::ostream& os, const std::string& config,
|
|
|
|
std::vector<std::string> const& configurationTypes)
|
|
|
|
{
|
|
|
|
this->ConfigurationName = config;
|
|
|
|
this->ConfigurationTypes = &configurationTypes;
|
|
|
|
this->GenerateScript(os);
|
|
|
|
this->ConfigurationName.clear();
|
|
|
|
this->ConfigurationTypes = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cmScriptGeneratorEncodeConfig(const std::string& config,
|
|
|
|
std::string& result)
|
|
|
|
{
|
|
|
|
for (const char* c = config.c_str(); *c; ++c) {
|
|
|
|
if (*c >= 'a' && *c <= 'z') {
|
|
|
|
result += "[";
|
|
|
|
result += static_cast<char>(*c + 'A' - 'a');
|
|
|
|
result += *c;
|
|
|
|
result += "]";
|
|
|
|
} else if (*c >= 'A' && *c <= 'Z') {
|
|
|
|
result += "[";
|
|
|
|
result += *c;
|
|
|
|
result += static_cast<char>(*c + 'a' - 'A');
|
|
|
|
result += "]";
|
|
|
|
} else {
|
|
|
|
result += *c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string cmScriptGenerator::CreateConfigTest(const std::string& config)
|
|
|
|
{
|
|
|
|
std::string result =
|
|
|
|
cmStrCat("\"${", this->RuntimeConfigVariable, "}\" MATCHES \"^(");
|
|
|
|
if (!config.empty()) {
|
|
|
|
cmScriptGeneratorEncodeConfig(config, result);
|
|
|
|
}
|
|
|
|
result += ")$\"";
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string cmScriptGenerator::CreateConfigTest(
|
|
|
|
std::vector<std::string> const& configs)
|
|
|
|
{
|
|
|
|
std::string result =
|
|
|
|
cmStrCat("\"${", this->RuntimeConfigVariable, "}\" MATCHES \"^(");
|
|
|
|
const char* sep = "";
|
|
|
|
for (std::string const& config : configs) {
|
|
|
|
result += sep;
|
|
|
|
sep = "|";
|
|
|
|
cmScriptGeneratorEncodeConfig(config, result);
|
|
|
|
}
|
|
|
|
result += ")$\"";
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmScriptGenerator::GenerateScript(std::ostream& os)
|
|
|
|
{
|
|
|
|
// Track indentation.
|
|
|
|
Indent indent;
|
|
|
|
|
|
|
|
// Generate the script possibly with per-configuration code.
|
|
|
|
this->GenerateScriptConfigs(os, indent);
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmScriptGenerator::GenerateScriptConfigs(std::ostream& os, Indent indent)
|
|
|
|
{
|
|
|
|
if (this->ActionsPerConfig) {
|
|
|
|
this->GenerateScriptActionsPerConfig(os, indent);
|
|
|
|
} else {
|
|
|
|
this->GenerateScriptActionsOnce(os, indent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmScriptGenerator::GenerateScriptActions(std::ostream& os, Indent indent)
|
|
|
|
{
|
|
|
|
if (this->ActionsPerConfig) {
|
|
|
|
// This is reached for single-configuration build generators in a
|
|
|
|
// per-config script generator.
|
|
|
|
this->GenerateScriptForConfig(os, this->ConfigurationName, indent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmScriptGenerator::GenerateScriptForConfig(std::ostream& /*unused*/,
|
|
|
|
const std::string& /*unused*/,
|
|
|
|
Indent /*unused*/)
|
|
|
|
{
|
|
|
|
// No actions for this generator.
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cmScriptGenerator::GeneratesForConfig(const std::string& config)
|
|
|
|
{
|
|
|
|
// If this is not a configuration-specific rule then we install.
|
|
|
|
if (this->Configurations.empty()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is a configuration-specific rule. Check if the config
|
|
|
|
// matches this rule.
|
|
|
|
std::string config_upper = cmSystemTools::UpperCase(config);
|
|
|
|
return std::any_of(this->Configurations.begin(), this->Configurations.end(),
|
|
|
|
[&config_upper](std::string const& cfg) -> bool {
|
|
|
|
return cmSystemTools::UpperCase(cfg) == config_upper;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmScriptGenerator::GenerateScriptActionsOnce(std::ostream& os,
|
|
|
|
Indent indent)
|
|
|
|
{
|
|
|
|
if (this->Configurations.empty()) {
|
|
|
|
// This rule is for all configurations.
|
|
|
|
this->GenerateScriptActions(os, indent);
|
|
|
|
} else {
|
|
|
|
// Generate a per-configuration block.
|
|
|
|
std::string config_test = this->CreateConfigTest(this->Configurations);
|
|
|
|
os << indent << "if(" << config_test << ")\n";
|
|
|
|
this->GenerateScriptActions(os, indent.Next());
|
|
|
|
os << indent << "endif(" << config_test << ")\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmScriptGenerator::GenerateScriptActionsPerConfig(std::ostream& os,
|
|
|
|
Indent indent)
|
|
|
|
{
|
|
|
|
if (this->ConfigurationTypes->empty()) {
|
|
|
|
// In a single-configuration generator there is only one action
|
|
|
|
// and it applies if the runtime-requested configuration is among
|
|
|
|
// the rule's allowed configurations. The configuration built in
|
|
|
|
// the tree does not matter for this decision but will be used to
|
|
|
|
// generate proper target file names into the code.
|
|
|
|
this->GenerateScriptActionsOnce(os, indent);
|
|
|
|
} else {
|
|
|
|
// In a multi-configuration generator we produce a separate rule
|
|
|
|
// in a block for each configuration that is built. We restrict
|
|
|
|
// the list of configurations to those to which this rule applies.
|
|
|
|
bool first = true;
|
|
|
|
for (std::string const& cfgType : *this->ConfigurationTypes) {
|
|
|
|
if (this->GeneratesForConfig(cfgType)) {
|
|
|
|
// Generate a per-configuration block.
|
|
|
|
std::string config_test = this->CreateConfigTest(cfgType);
|
|
|
|
os << indent << (first ? "if(" : "elseif(") << config_test << ")\n";
|
|
|
|
this->GenerateScriptForConfig(os, cfgType, indent.Next());
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!first) {
|
|
|
|
if (this->NeedsScriptNoConfig()) {
|
|
|
|
os << indent << "else()\n";
|
|
|
|
this->GenerateScriptNoConfig(os, indent.Next());
|
|
|
|
}
|
|
|
|
os << indent << "endif()\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|