2011-11-11 09:00:49 +04:00
|
|
|
/*============================================================================
|
|
|
|
CMake - Cross Platform Makefile Generator
|
|
|
|
Copyright 2011 Peter Collingbourne <peter@pcc.me.uk>
|
|
|
|
Copyright 2011 Nicolas Despres <nicolas.despres@gmail.com>
|
|
|
|
|
|
|
|
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.
|
|
|
|
============================================================================*/
|
|
|
|
#include "cmLocalNinjaGenerator.h"
|
|
|
|
#include "cmCustomCommandGenerator.h"
|
|
|
|
#include "cmMakefile.h"
|
|
|
|
#include "cmGlobalNinjaGenerator.h"
|
|
|
|
#include "cmNinjaTargetGenerator.h"
|
|
|
|
#include "cmGeneratedFileStream.h"
|
|
|
|
#include "cmSourceFile.h"
|
|
|
|
#include "cmake.h"
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
cmLocalNinjaGenerator::cmLocalNinjaGenerator()
|
|
|
|
: cmLocalGenerator()
|
|
|
|
, ConfigName("")
|
|
|
|
, HomeRelativeOutputPath("")
|
|
|
|
{
|
2012-02-19 05:50:53 +04:00
|
|
|
#ifdef _WIN32
|
|
|
|
this->WindowsShell = true;
|
|
|
|
#endif
|
2012-02-20 01:38:10 +04:00
|
|
|
this->TargetImplib = "$TARGET_IMPLIB";
|
2011-11-11 09:00:49 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
// Virtual public methods.
|
|
|
|
|
|
|
|
cmLocalNinjaGenerator::~cmLocalNinjaGenerator()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmLocalNinjaGenerator::Generate()
|
|
|
|
{
|
|
|
|
this->SetConfigName();
|
|
|
|
|
|
|
|
this->WriteProcessedMakefile(this->GetBuildFileStream());
|
2012-06-10 22:20:29 +04:00
|
|
|
#ifdef NINJA_GEN_VERBOSE_FILES
|
2011-11-11 09:00:49 +04:00
|
|
|
this->WriteProcessedMakefile(this->GetRulesFileStream());
|
2012-06-10 22:20:29 +04:00
|
|
|
#endif
|
2011-11-11 09:00:49 +04:00
|
|
|
|
2013-10-18 14:59:47 +04:00
|
|
|
// We do that only once for the top CMakeLists.txt file.
|
|
|
|
if(this->isRootMakefile())
|
|
|
|
{
|
|
|
|
this->WriteBuildFileTop();
|
|
|
|
|
|
|
|
const std::string showIncludesPrefix = this->GetMakefile()
|
|
|
|
->GetSafeDefinition("CMAKE_CL_SHOWINCLUDES_PREFIX");
|
|
|
|
if (!showIncludesPrefix.empty())
|
|
|
|
{
|
|
|
|
cmGlobalNinjaGenerator::WriteComment(this->GetRulesFileStream(),
|
|
|
|
"localized /showIncludes string");
|
|
|
|
this->GetRulesFileStream()
|
|
|
|
<< "msvc_deps_prefix = " << showIncludesPrefix << "\n\n";
|
|
|
|
}
|
|
|
|
}
|
2011-11-11 09:00:49 +04:00
|
|
|
|
|
|
|
cmTargets& targets = this->GetMakefile()->GetTargets();
|
|
|
|
for(cmTargets::iterator t = targets.begin(); t != targets.end(); ++t)
|
|
|
|
{
|
|
|
|
cmNinjaTargetGenerator* tg = cmNinjaTargetGenerator::New(&t->second);
|
|
|
|
if(tg)
|
|
|
|
{
|
|
|
|
tg->Generate();
|
|
|
|
// Add the target to "all" if required.
|
|
|
|
if (!this->GetGlobalNinjaGenerator()->IsExcluded(
|
2012-07-18 13:27:49 +04:00
|
|
|
this->GetGlobalNinjaGenerator()->GetLocalGenerators()[0],
|
2011-11-11 09:00:49 +04:00
|
|
|
t->second))
|
|
|
|
this->GetGlobalNinjaGenerator()->AddDependencyToAll(&t->second);
|
|
|
|
delete tg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this->WriteCustomCommandBuildStatements();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Implemented in:
|
|
|
|
// cmLocalUnixMakefileGenerator3.
|
|
|
|
// Used in:
|
|
|
|
// Source/cmMakefile.cxx
|
|
|
|
// Source/cmGlobalGenerator.cxx
|
|
|
|
void cmLocalNinjaGenerator::Configure()
|
|
|
|
{
|
|
|
|
// Compute the path to use when referencing the current output
|
|
|
|
// directory from the top output directory.
|
|
|
|
this->HomeRelativeOutputPath =
|
|
|
|
this->Convert(this->Makefile->GetStartOutputDirectory(), HOME_OUTPUT);
|
|
|
|
if(this->HomeRelativeOutputPath == ".")
|
|
|
|
{
|
|
|
|
this->HomeRelativeOutputPath = "";
|
|
|
|
}
|
|
|
|
this->cmLocalGenerator::Configure();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Picked up from cmLocalUnixMakefileGenerator3. Refactor it.
|
|
|
|
std::string cmLocalNinjaGenerator
|
|
|
|
::GetTargetDirectory(cmTarget const& target) const
|
|
|
|
{
|
|
|
|
std::string dir = cmake::GetCMakeFilesDirectoryPostSlash();
|
|
|
|
dir += target.GetName();
|
|
|
|
#if defined(__VMS)
|
|
|
|
dir += "_dir";
|
|
|
|
#else
|
|
|
|
dir += ".dir";
|
|
|
|
#endif
|
|
|
|
return dir;
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
// Non-virtual public methods.
|
|
|
|
|
|
|
|
const cmGlobalNinjaGenerator*
|
|
|
|
cmLocalNinjaGenerator::GetGlobalNinjaGenerator() const
|
|
|
|
{
|
|
|
|
return
|
|
|
|
static_cast<const cmGlobalNinjaGenerator*>(this->GetGlobalGenerator());
|
|
|
|
}
|
|
|
|
|
|
|
|
cmGlobalNinjaGenerator* cmLocalNinjaGenerator::GetGlobalNinjaGenerator()
|
|
|
|
{
|
|
|
|
return static_cast<cmGlobalNinjaGenerator*>(this->GetGlobalGenerator());
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
// Virtual protected methods.
|
|
|
|
|
|
|
|
std::string
|
|
|
|
cmLocalNinjaGenerator::ConvertToLinkReference(std::string const& lib)
|
|
|
|
{
|
|
|
|
return this->Convert(lib.c_str(), HOME_OUTPUT, SHELL);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string
|
|
|
|
cmLocalNinjaGenerator::ConvertToIncludeReference(std::string const& path)
|
|
|
|
{
|
|
|
|
return this->Convert(path.c_str(), HOME_OUTPUT, SHELL);
|
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
|
|
// Private methods.
|
|
|
|
|
|
|
|
cmGeneratedFileStream& cmLocalNinjaGenerator::GetBuildFileStream() const
|
|
|
|
{
|
|
|
|
return *this->GetGlobalNinjaGenerator()->GetBuildFileStream();
|
|
|
|
}
|
|
|
|
|
|
|
|
cmGeneratedFileStream& cmLocalNinjaGenerator::GetRulesFileStream() const
|
|
|
|
{
|
|
|
|
return *this->GetGlobalNinjaGenerator()->GetRulesFileStream();
|
|
|
|
}
|
|
|
|
|
|
|
|
const cmake* cmLocalNinjaGenerator::GetCMakeInstance() const
|
|
|
|
{
|
|
|
|
return this->GetGlobalGenerator()->GetCMakeInstance();
|
|
|
|
}
|
|
|
|
|
|
|
|
cmake* cmLocalNinjaGenerator::GetCMakeInstance()
|
|
|
|
{
|
|
|
|
return this->GetGlobalGenerator()->GetCMakeInstance();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool cmLocalNinjaGenerator::isRootMakefile() const
|
|
|
|
{
|
|
|
|
return (strcmp(this->Makefile->GetCurrentDirectory(),
|
|
|
|
this->GetCMakeInstance()->GetHomeDirectory()) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmLocalNinjaGenerator::WriteBuildFileTop()
|
|
|
|
{
|
|
|
|
// For the build file.
|
|
|
|
this->WriteProjectHeader(this->GetBuildFileStream());
|
|
|
|
this->WriteNinjaFilesInclusion(this->GetBuildFileStream());
|
|
|
|
|
|
|
|
// For the rule file.
|
|
|
|
this->WriteProjectHeader(this->GetRulesFileStream());
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmLocalNinjaGenerator::WriteProjectHeader(std::ostream& os)
|
|
|
|
{
|
|
|
|
cmGlobalNinjaGenerator::WriteDivider(os);
|
|
|
|
os
|
|
|
|
<< "# Project: " << this->GetMakefile()->GetProjectName() << std::endl
|
|
|
|
<< "# Configuration: " << this->ConfigName << std::endl
|
|
|
|
;
|
|
|
|
cmGlobalNinjaGenerator::WriteDivider(os);
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmLocalNinjaGenerator::WriteNinjaFilesInclusion(std::ostream& os)
|
|
|
|
{
|
|
|
|
cmGlobalNinjaGenerator::WriteDivider(os);
|
|
|
|
os
|
|
|
|
<< "# Include auxiliary files.\n"
|
|
|
|
<< "\n"
|
|
|
|
;
|
|
|
|
cmGlobalNinjaGenerator::WriteInclude(os,
|
|
|
|
cmGlobalNinjaGenerator::NINJA_RULES_FILE,
|
|
|
|
"Include rules file.");
|
|
|
|
os << "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmLocalNinjaGenerator::SetConfigName()
|
|
|
|
{
|
|
|
|
// Store the configuration name that will be generated.
|
|
|
|
if(const char* config =
|
|
|
|
this->GetMakefile()->GetDefinition("CMAKE_BUILD_TYPE"))
|
|
|
|
{
|
|
|
|
// Use the build type given by the user.
|
|
|
|
this->ConfigName = config;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// No configuration type given.
|
|
|
|
this->ConfigName = "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmLocalNinjaGenerator::WriteProcessedMakefile(std::ostream& os)
|
|
|
|
{
|
|
|
|
cmGlobalNinjaGenerator::WriteDivider(os);
|
|
|
|
os
|
|
|
|
<< "# Write statements declared in CMakeLists.txt:" << std::endl
|
|
|
|
<< "# " << this->Makefile->GetCurrentListFile() << std::endl
|
|
|
|
;
|
|
|
|
if(this->isRootMakefile())
|
|
|
|
os << "# Which is the root file." << std::endl;
|
|
|
|
cmGlobalNinjaGenerator::WriteDivider(os);
|
|
|
|
os << std::endl;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string cmLocalNinjaGenerator::ConvertToNinjaPath(const char *path)
|
|
|
|
{
|
2012-02-19 05:50:53 +04:00
|
|
|
std::string convPath = this->Convert(path, cmLocalGenerator::HOME_OUTPUT);
|
|
|
|
#ifdef _WIN32
|
|
|
|
cmSystemTools::ReplaceString(convPath, "/", "\\");
|
|
|
|
#endif
|
|
|
|
return convPath;
|
2011-11-11 09:00:49 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cmLocalNinjaGenerator
|
|
|
|
::AppendTargetOutputs(cmTarget* target, cmNinjaDeps& outputs)
|
|
|
|
{
|
|
|
|
this->GetGlobalNinjaGenerator()->AppendTargetOutputs(target, outputs);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
cmLocalNinjaGenerator
|
|
|
|
::AppendTargetDepends(cmTarget* target, cmNinjaDeps& outputs)
|
|
|
|
{
|
|
|
|
this->GetGlobalNinjaGenerator()->AppendTargetDepends(target, outputs);
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmLocalNinjaGenerator::AppendCustomCommandDeps(const cmCustomCommand *cc,
|
|
|
|
cmNinjaDeps &ninjaDeps)
|
|
|
|
{
|
|
|
|
const std::vector<std::string> &deps = cc->GetDepends();
|
|
|
|
for (std::vector<std::string>::const_iterator i = deps.begin();
|
|
|
|
i != deps.end(); ++i) {
|
|
|
|
std::string dep;
|
|
|
|
if (this->GetRealDependency(i->c_str(), this->GetConfigName(), dep))
|
|
|
|
ninjaDeps.push_back(ConvertToNinjaPath(dep.c_str()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string cmLocalNinjaGenerator::BuildCommandLine(
|
|
|
|
const std::vector<std::string> &cmdLines)
|
|
|
|
{
|
|
|
|
// If we have no commands but we need to build a command anyway, use ":".
|
|
|
|
// This happens when building a POST_BUILD value for link targets that
|
|
|
|
// don't use POST_BUILD.
|
|
|
|
if (cmdLines.empty())
|
2012-02-18 11:07:11 +04:00
|
|
|
#ifdef _WIN32
|
2012-07-02 02:01:41 +04:00
|
|
|
return "cd .";
|
2012-02-18 11:07:11 +04:00
|
|
|
#else
|
2011-11-11 09:00:49 +04:00
|
|
|
return ":";
|
2012-02-18 11:07:11 +04:00
|
|
|
#endif
|
2011-11-11 09:00:49 +04:00
|
|
|
|
2012-06-06 23:14:40 +04:00
|
|
|
cmOStringStream cmd;
|
2011-11-11 09:00:49 +04:00
|
|
|
for (std::vector<std::string>::const_iterator li = cmdLines.begin();
|
|
|
|
li != cmdLines.end(); ++li) {
|
2012-06-06 00:39:42 +04:00
|
|
|
if (li != cmdLines.begin()) {
|
2011-11-11 09:00:49 +04:00
|
|
|
cmd << " && ";
|
2012-06-06 00:39:42 +04:00
|
|
|
#ifdef _WIN32
|
|
|
|
} else if (cmdLines.size() > 1) {
|
|
|
|
cmd << "cmd.exe /c ";
|
|
|
|
#endif
|
|
|
|
}
|
2011-11-11 09:00:49 +04:00
|
|
|
cmd << *li;
|
|
|
|
}
|
|
|
|
return cmd.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmLocalNinjaGenerator::AppendCustomCommandLines(const cmCustomCommand *cc,
|
|
|
|
std::vector<std::string> &cmdLines)
|
|
|
|
{
|
|
|
|
cmCustomCommandGenerator ccg(*cc, this->GetConfigName(), this->Makefile);
|
|
|
|
if (ccg.GetNumberOfCommands() > 0) {
|
2012-02-19 06:46:37 +04:00
|
|
|
const char* wd = cc->GetWorkingDirectory();
|
|
|
|
if (!wd)
|
|
|
|
wd = this->GetMakefile()->GetStartOutputDirectory();
|
|
|
|
|
2012-06-06 23:14:40 +04:00
|
|
|
cmOStringStream cdCmd;
|
2013-04-18 21:57:39 +04:00
|
|
|
#ifdef _WIN32
|
|
|
|
std::string cdStr = "cd /D ";
|
|
|
|
#else
|
|
|
|
std::string cdStr = "cd ";
|
|
|
|
#endif
|
|
|
|
cdCmd << cdStr << this->ConvertToOutputFormat(wd, SHELL);
|
2011-11-11 09:00:49 +04:00
|
|
|
cmdLines.push_back(cdCmd.str());
|
|
|
|
}
|
2013-10-30 01:56:23 +04:00
|
|
|
|
|
|
|
std::string launcher = this->MakeCustomLauncher(*cc);
|
|
|
|
|
2011-11-11 09:00:49 +04:00
|
|
|
for (unsigned i = 0; i != ccg.GetNumberOfCommands(); ++i) {
|
2013-10-30 01:56:23 +04:00
|
|
|
cmdLines.push_back(launcher +
|
|
|
|
this->ConvertToOutputFormat(ccg.GetCommand(i).c_str(), SHELL));
|
|
|
|
|
2011-11-11 09:00:49 +04:00
|
|
|
std::string& cmd = cmdLines.back();
|
|
|
|
ccg.AppendArguments(i, cmd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2012-03-13 17:17:46 +04:00
|
|
|
cmLocalNinjaGenerator::WriteCustomCommandBuildStatement(
|
|
|
|
cmCustomCommand const *cc, const cmNinjaDeps& orderOnlyDeps)
|
2011-11-11 09:00:49 +04:00
|
|
|
{
|
|
|
|
if (this->GetGlobalNinjaGenerator()->SeenCustomCommand(cc))
|
|
|
|
return;
|
|
|
|
|
|
|
|
const std::vector<std::string> &outputs = cc->GetOutputs();
|
|
|
|
cmNinjaDeps ninjaOutputs(outputs.size()), ninjaDeps;
|
|
|
|
|
|
|
|
std::transform(outputs.begin(), outputs.end(),
|
|
|
|
ninjaOutputs.begin(), MapToNinjaPath());
|
|
|
|
this->AppendCustomCommandDeps(cc, ninjaDeps);
|
|
|
|
|
|
|
|
for (cmNinjaDeps::iterator i = ninjaOutputs.begin(); i != ninjaOutputs.end();
|
|
|
|
++i)
|
|
|
|
this->GetGlobalNinjaGenerator()->SeenCustomCommandOutput(*i);
|
|
|
|
|
|
|
|
std::vector<std::string> cmdLines;
|
|
|
|
this->AppendCustomCommandLines(cc, cmdLines);
|
|
|
|
|
|
|
|
if (cmdLines.empty()) {
|
2013-06-07 22:25:33 +04:00
|
|
|
this->GetGlobalNinjaGenerator()->WritePhonyBuild(
|
|
|
|
this->GetBuildFileStream(),
|
|
|
|
"Phony custom command for " +
|
|
|
|
ninjaOutputs[0],
|
|
|
|
ninjaOutputs,
|
|
|
|
ninjaDeps,
|
|
|
|
cmNinjaDeps(),
|
|
|
|
orderOnlyDeps,
|
|
|
|
cmNinjaVars());
|
2011-11-11 09:00:49 +04:00
|
|
|
} else {
|
|
|
|
this->GetGlobalNinjaGenerator()->WriteCustomCommandBuild(
|
|
|
|
this->BuildCommandLine(cmdLines),
|
|
|
|
this->ConstructComment(*cc),
|
|
|
|
"Custom command for " + ninjaOutputs[0],
|
|
|
|
ninjaOutputs,
|
|
|
|
ninjaDeps,
|
|
|
|
orderOnlyDeps);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-13 17:17:46 +04:00
|
|
|
void cmLocalNinjaGenerator::AddCustomCommandTarget(cmCustomCommand const* cc,
|
2011-11-11 09:00:49 +04:00
|
|
|
cmTarget* target)
|
|
|
|
{
|
|
|
|
this->CustomCommandTargets[cc].insert(target);
|
|
|
|
}
|
|
|
|
|
|
|
|
void cmLocalNinjaGenerator::WriteCustomCommandBuildStatements()
|
|
|
|
{
|
|
|
|
for (CustomCommandTargetMap::iterator i = this->CustomCommandTargets.begin();
|
|
|
|
i != this->CustomCommandTargets.end(); ++i) {
|
|
|
|
// A custom command may appear on multiple targets. However, some build
|
|
|
|
// systems exist where the target dependencies on some of the targets are
|
|
|
|
// overspecified, leading to a dependency cycle. If we assume all target
|
|
|
|
// dependencies are a superset of the true target dependencies for this
|
|
|
|
// custom command, we can take the set intersection of all target
|
|
|
|
// dependencies to obtain a correct dependency list.
|
|
|
|
//
|
|
|
|
// FIXME: This won't work in certain obscure scenarios involving indirect
|
|
|
|
// dependencies.
|
|
|
|
std::set<cmTarget*>::iterator j = i->second.begin();
|
|
|
|
assert(j != i->second.end());
|
|
|
|
std::vector<std::string> ccTargetDeps;
|
|
|
|
this->AppendTargetDepends(*j, ccTargetDeps);
|
|
|
|
std::sort(ccTargetDeps.begin(), ccTargetDeps.end());
|
|
|
|
++j;
|
|
|
|
|
|
|
|
for (; j != i->second.end(); ++j) {
|
|
|
|
std::vector<std::string> jDeps, depsIntersection;
|
|
|
|
this->AppendTargetDepends(*j, jDeps);
|
|
|
|
std::sort(jDeps.begin(), jDeps.end());
|
|
|
|
std::set_intersection(ccTargetDeps.begin(), ccTargetDeps.end(),
|
|
|
|
jDeps.begin(), jDeps.end(),
|
|
|
|
std::back_inserter(depsIntersection));
|
|
|
|
ccTargetDeps = depsIntersection;
|
|
|
|
}
|
|
|
|
|
|
|
|
this->WriteCustomCommandBuildStatement(i->first, ccTargetDeps);
|
|
|
|
}
|
|
|
|
}
|
2013-10-30 01:56:23 +04:00
|
|
|
|
|
|
|
std::string cmLocalNinjaGenerator::MakeCustomLauncher(
|
|
|
|
const cmCustomCommand& cc)
|
|
|
|
{
|
|
|
|
const char* property = "RULE_LAUNCH_CUSTOM";
|
|
|
|
const char* property_value = this->Makefile->GetProperty(property);
|
|
|
|
|
|
|
|
if(!property_value || !*property_value)
|
|
|
|
{
|
|
|
|
return std::string();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Expand rules in the empty string. It may insert the launcher and
|
|
|
|
// perform replacements.
|
|
|
|
RuleVariables vars;
|
|
|
|
vars.RuleLauncher = property;
|
|
|
|
std::string output;
|
|
|
|
const std::vector<std::string>& outputs = cc.GetOutputs();
|
|
|
|
if(!outputs.empty())
|
|
|
|
{
|
|
|
|
RelativeRoot relative_root =
|
|
|
|
cc.GetWorkingDirectory() ? NONE : START_OUTPUT;
|
|
|
|
|
|
|
|
output = this->Convert(outputs[0].c_str(), relative_root, SHELL);
|
|
|
|
}
|
|
|
|
vars.Output = output.c_str();
|
|
|
|
|
|
|
|
std::string launcher;
|
|
|
|
this->ExpandRuleVariables(launcher, vars);
|
|
|
|
if(!launcher.empty())
|
|
|
|
{
|
|
|
|
launcher += " ";
|
|
|
|
}
|
|
|
|
|
|
|
|
return launcher;
|
|
|
|
}
|