mirror of
https://github.com/reactos/CMake.git
synced 2024-11-28 22:10:32 +00:00
576 lines
18 KiB
C++
576 lines
18 KiB
C++
/*=========================================================================
|
|
|
|
Program: CMake - Cross-Platform Makefile Generator
|
|
Module: $RCSfile$
|
|
Language: C++
|
|
Date: $Date$
|
|
Version: $Revision$
|
|
|
|
Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved.
|
|
See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details.
|
|
|
|
This software is distributed WITHOUT ANY WARRANTY; without even
|
|
the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
PURPOSE. See the above copyright notices for more information.
|
|
|
|
=========================================================================*/
|
|
#include "cmGlobalGenerator.h"
|
|
#include "cmLocalGenerator.h"
|
|
#include "cmake.h"
|
|
#include "cmMakefile.h"
|
|
#include <stdlib.h> // required for atof
|
|
|
|
#if defined(_WIN32) && !defined(__CYGWIN__)
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
int cmGlobalGenerator::s_TryCompileTimeout = 0;
|
|
|
|
cmGlobalGenerator::cmGlobalGenerator()
|
|
{
|
|
// by default use the native paths
|
|
m_ForceUnixPaths = false;
|
|
}
|
|
|
|
cmGlobalGenerator::~cmGlobalGenerator()
|
|
{
|
|
// Delete any existing cmLocalGenerators
|
|
unsigned int i;
|
|
for (i = 0; i < m_LocalGenerators.size(); ++i)
|
|
{
|
|
delete m_LocalGenerators[i];
|
|
}
|
|
m_LocalGenerators.clear();
|
|
}
|
|
|
|
|
|
void cmGlobalGenerator::EnableLanguage(const char* lang,
|
|
cmMakefile *mf)
|
|
{
|
|
if(m_FindMakeProgramFile.size() == 0)
|
|
{
|
|
cmSystemTools::Error(
|
|
"Generator implementation error, "
|
|
"all generators must specify m_FindMakeProgramFile");
|
|
}
|
|
std::string root = mf->GetDefinition("CMAKE_ROOT");
|
|
if(!mf->GetDefinition("CMAKE_MAKE_PROGRAM")
|
|
|| cmSystemTools::IsOff(mf->GetDefinition("CMAKE_MAKE_PROGRAM")))
|
|
{
|
|
std::string setMakeProgram = root;
|
|
setMakeProgram += "/Modules/";
|
|
setMakeProgram += m_FindMakeProgramFile;
|
|
mf->ReadListFile(0, setMakeProgram.c_str());
|
|
}
|
|
if(!mf->GetDefinition("CMAKE_MAKE_PROGRAM")
|
|
|| cmSystemTools::IsOff(mf->GetDefinition("CMAKE_MAKE_PROGRAM")))
|
|
{
|
|
cmOStringStream err;
|
|
err << "CMake was unable to find a build program corresponding to \""
|
|
<< this->GetName() << "\". CMAKE_MAKE_PROGRAM is not set. You "
|
|
<< "probably need to select a different build tool.";
|
|
cmSystemTools::Error(err.str().c_str());
|
|
cmSystemTools::SetFatalErrorOccured();
|
|
return;
|
|
}
|
|
std::string makeProgram = mf->GetDefinition("CMAKE_MAKE_PROGRAM");
|
|
// if there are spaces in the make program use short path
|
|
// but do not short path the actual program name, as
|
|
// this can cause trouble with VSExpress
|
|
if(makeProgram.find(' ') != makeProgram.npos)
|
|
{
|
|
std::string dir;
|
|
std::string file;
|
|
cmSystemTools::SplitProgramPath(makeProgram.c_str(),
|
|
dir, file);
|
|
std::string saveFile = file;
|
|
cmSystemTools::GetShortPath(makeProgram.c_str(), makeProgram);
|
|
cmSystemTools::SplitProgramPath(makeProgram.c_str(),
|
|
dir, file);
|
|
makeProgram = dir;
|
|
makeProgram += "/";
|
|
makeProgram += saveFile;
|
|
this->GetCMakeInstance()->AddCacheEntry("CMAKE_MAKE_PROGRAM", makeProgram.c_str(),
|
|
"make program",
|
|
cmCacheManager::FILEPATH);
|
|
}
|
|
|
|
bool isLocal = m_CMakeInstance->GetLocal();
|
|
// if no lang specified use CXX
|
|
if(!lang )
|
|
{
|
|
lang = "CXX";
|
|
}
|
|
std::string rootBin = mf->GetHomeOutputDirectory();
|
|
if(m_ConfiguredFilesPath.size())
|
|
{
|
|
rootBin = m_ConfiguredFilesPath;
|
|
}
|
|
bool needCBackwards = false;
|
|
bool needCXXBackwards = false;
|
|
|
|
if (!isLocal &&
|
|
!this->GetLanguageEnabled("C") && !this->GetLanguageEnabled("CXX") &&
|
|
!this->GetLanguageEnabled("JAVA"))
|
|
{
|
|
#if defined(_WIN32) && !defined(__CYGWIN__)
|
|
/* Windows version number data. */
|
|
OSVERSIONINFO osvi;
|
|
ZeroMemory(&osvi, sizeof(osvi));
|
|
osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
|
|
GetVersionEx (&osvi);
|
|
cmOStringStream windowsVersionString;
|
|
windowsVersionString << osvi.dwMajorVersion << "." << osvi.dwMinorVersion;
|
|
windowsVersionString.str();
|
|
mf->AddDefinition("CMAKE_SYSTEM_VERSION", windowsVersionString.str().c_str());
|
|
#endif
|
|
// Read the DetermineSystem file
|
|
std::string systemFile = root;
|
|
systemFile += "/Modules/CMakeDetermineSystem.cmake";
|
|
mf->ReadListFile(0, systemFile.c_str());
|
|
}
|
|
// check for a C compiler and configure it
|
|
if(!isLocal &&
|
|
!this->GetLanguageEnabled("C") &&
|
|
lang[0] == 'C')
|
|
{
|
|
if (m_CMakeInstance->GetIsInTryCompile())
|
|
{
|
|
cmSystemTools::Error("This should not have happen. "
|
|
"If you see this message, you are probably using a "
|
|
"broken CMakeLists.txt file or a problematic release of "
|
|
"CMake");
|
|
}
|
|
needCBackwards = true;
|
|
// read determine C compiler
|
|
std::string determineCFile = root;
|
|
determineCFile += "/Modules/CMakeDetermineCCompiler.cmake";
|
|
mf->ReadListFile(0,determineCFile.c_str());
|
|
this->SetLanguageEnabled("C");
|
|
// put CC in the environment in case user scripts want
|
|
// to run configure
|
|
if(mf->GetDefinition("CMAKE_C_COMPILER"))
|
|
{
|
|
std::string env = "CC=${CMAKE_C_COMPILER}";
|
|
mf->ExpandVariablesInString(env);
|
|
cmSystemTools::PutEnv(env.c_str());
|
|
}
|
|
}
|
|
|
|
// check for a CXX compiler and configure it
|
|
if(!isLocal &&
|
|
!this->GetLanguageEnabled("CXX") &&
|
|
strcmp(lang, "CXX") == 0)
|
|
{
|
|
needCXXBackwards = true;
|
|
std::string determineCFile = root;
|
|
determineCFile += "/Modules/CMakeDetermineCXXCompiler.cmake";
|
|
mf->ReadListFile(0,determineCFile.c_str());
|
|
this->SetLanguageEnabled("CXX");
|
|
// put CXX in the environment in case user scripts want
|
|
// to run configure
|
|
if(mf->GetDefinition("CMAKE_CXX_COMPILER"))
|
|
{
|
|
std::string env = "CXX=${CMAKE_CXX_COMPILER}";
|
|
mf->ExpandVariablesInString(env);
|
|
cmSystemTools::PutEnv(env.c_str());
|
|
}
|
|
}
|
|
// check for a Java compiler and configure it
|
|
if(!isLocal &&
|
|
!this->GetLanguageEnabled("JAVA") &&
|
|
strcmp(lang, "JAVA") == 0)
|
|
{
|
|
std::string determineCFile = root;
|
|
determineCFile += "/Modules/CMakeDetermineJavaCompiler.cmake";
|
|
mf->ReadListFile(0,determineCFile.c_str());
|
|
this->SetLanguageEnabled("JAVA");
|
|
}
|
|
|
|
std::string fpath = rootBin;
|
|
if(!mf->GetDefinition("CMAKE_SYSTEM_LOADED"))
|
|
{
|
|
fpath += "/CMakeSystem.cmake";
|
|
mf->ReadListFile(0,fpath.c_str());
|
|
}
|
|
// if C, then enable C
|
|
if(lang[0] == 'C' && !mf->GetDefinition("CMAKE_C_COMPILER_LOADED"))
|
|
{
|
|
fpath = rootBin;
|
|
fpath += "/CMakeCCompiler.cmake";
|
|
mf->ReadListFile(0,fpath.c_str());
|
|
this->SetLanguageEnabled("C");
|
|
}
|
|
if(strcmp(lang, "CXX") == 0 && !mf->GetDefinition("CMAKE_CXX_COMPILER_LOADED"))
|
|
{
|
|
fpath = rootBin;
|
|
fpath += "/CMakeCXXCompiler.cmake";
|
|
mf->ReadListFile(0,fpath.c_str());
|
|
this->SetLanguageEnabled("CXX");
|
|
|
|
}
|
|
if(strcmp(lang, "JAVA") == 0 && !mf->GetDefinition("CMAKE_JAVA_COMPILER_LOADED"))
|
|
{
|
|
fpath = rootBin;
|
|
fpath += "/CMakeJavaCompiler.cmake";
|
|
mf->ReadListFile(0,fpath.c_str());
|
|
this->SetLanguageEnabled("JAVA");
|
|
}
|
|
if ( lang[0] == 'C' && !mf->GetDefinition("CMAKE_SYSTEM_SPECIFIC_INFORMATION_LOADED"))
|
|
{
|
|
fpath = root;
|
|
fpath += "/Modules/CMakeSystemSpecificInformation.cmake";
|
|
mf->ReadListFile(0,fpath.c_str());
|
|
}
|
|
|
|
if(!isLocal)
|
|
{
|
|
// At this point we should have enough info for a try compile
|
|
// which is used in the backward stuff
|
|
if(needCBackwards)
|
|
{
|
|
if (!m_CMakeInstance->GetIsInTryCompile())
|
|
{
|
|
std::string ifpath = root + "/Modules/CMakeTestCCompiler.cmake";
|
|
mf->ReadListFile(0,ifpath.c_str());
|
|
// for old versions of CMake ListFiles
|
|
const char* versionValue
|
|
= mf->GetDefinition("CMAKE_BACKWARDS_COMPATIBILITY");
|
|
if (atof(versionValue) <= 1.4)
|
|
{
|
|
ifpath = root + "/Modules/CMakeBackwardCompatibilityC.cmake";
|
|
mf->ReadListFile(0,ifpath.c_str());
|
|
}
|
|
}
|
|
}
|
|
if(needCXXBackwards)
|
|
{
|
|
if (!m_CMakeInstance->GetIsInTryCompile())
|
|
{
|
|
std::string ifpath = root + "/Modules/CMakeTestCXXCompiler.cmake";
|
|
mf->ReadListFile(0,ifpath.c_str());
|
|
// for old versions of CMake ListFiles
|
|
const char* versionValue
|
|
= mf->GetDefinition("CMAKE_BACKWARDS_COMPATIBILITY");
|
|
if (atof(versionValue) <= 1.4)
|
|
{
|
|
std::string nfpath = root + "/Modules/CMakeBackwardCompatibilityCXX.cmake";
|
|
mf->ReadListFile(0,nfpath.c_str());
|
|
}
|
|
}
|
|
}
|
|
// if we are from the top, always define this
|
|
mf->AddDefinition("RUN_CONFIGURE", true);
|
|
}
|
|
}
|
|
|
|
|
|
void cmGlobalGenerator::SetLanguageEnabled(const char* l)
|
|
{
|
|
m_LanguageEnabled[l] = true;
|
|
}
|
|
|
|
bool cmGlobalGenerator::GetLanguageEnabled(const char* l)
|
|
{
|
|
return (m_LanguageEnabled.count(l) > 0);
|
|
}
|
|
|
|
void cmGlobalGenerator::ClearEnabledLanguages()
|
|
{
|
|
m_LanguageEnabled.clear();
|
|
}
|
|
|
|
void cmGlobalGenerator::Configure()
|
|
{
|
|
// Delete any existing cmLocalGenerators
|
|
unsigned int i;
|
|
for (i = 0; i < m_LocalGenerators.size(); ++i)
|
|
{
|
|
delete m_LocalGenerators[i];
|
|
}
|
|
m_LocalGenerators.clear();
|
|
|
|
// start with this directory
|
|
cmLocalGenerator *lg = this->CreateLocalGenerator();
|
|
m_LocalGenerators.push_back(lg);
|
|
|
|
// set the Start directories
|
|
lg->GetMakefile()->SetStartDirectory(m_CMakeInstance->GetStartDirectory());
|
|
lg->GetMakefile()->SetStartOutputDirectory(m_CMakeInstance->GetStartOutputDirectory());
|
|
lg->GetMakefile()->MakeStartDirectoriesCurrent();
|
|
|
|
// now do it
|
|
this->RecursiveConfigure(lg,0.0f,0.9f);
|
|
|
|
std::set<std::string> notFoundMap;
|
|
// after it is all done do a ConfigureFinalPass
|
|
cmCacheManager* manager = 0;
|
|
for (i = 0; i < m_LocalGenerators.size(); ++i)
|
|
{
|
|
manager = m_LocalGenerators[i]->GetMakefile()->GetCacheManager();
|
|
m_LocalGenerators[i]->ConfigureFinalPass();
|
|
cmTargets const& targets = m_LocalGenerators[i]->GetMakefile()->GetTargets();
|
|
for (cmTargets::const_iterator l = targets.begin();
|
|
l != targets.end(); l++)
|
|
{
|
|
cmTarget::LinkLibraries libs = l->second.GetLinkLibraries();
|
|
for(cmTarget::LinkLibraries::iterator lib = libs.begin();
|
|
lib != libs.end(); ++lib)
|
|
{
|
|
if(lib->first.size() > 9 &&
|
|
cmSystemTools::IsNOTFOUND(lib->first.c_str()))
|
|
{
|
|
std::string varName = lib->first.substr(0, lib->first.size()-9);
|
|
notFoundMap.insert(varName);
|
|
}
|
|
}
|
|
std::vector<std::string>& incs =
|
|
m_LocalGenerators[i]->GetMakefile()->GetIncludeDirectories();
|
|
|
|
for( std::vector<std::string>::iterator lib = incs.begin();
|
|
lib != incs.end(); ++lib)
|
|
{
|
|
if(lib->size() > 9 &&
|
|
cmSystemTools::IsNOTFOUND(lib->c_str()))
|
|
{
|
|
std::string varName = lib->substr(0, lib->size()-9);
|
|
notFoundMap.insert(varName);
|
|
}
|
|
}
|
|
m_CMakeInstance->UpdateProgress("Configuring",
|
|
0.9f+0.1f*(i+1.0f)/m_LocalGenerators.size());
|
|
m_LocalGenerators[i]->GetMakefile()->CheckInfiniteLoops();
|
|
}
|
|
}
|
|
|
|
if(notFoundMap.size())
|
|
{
|
|
std::string notFoundVars;
|
|
for(std::set<std::string>::iterator ii = notFoundMap.begin();
|
|
ii != notFoundMap.end(); ++ii)
|
|
{
|
|
notFoundVars += *ii;
|
|
if(manager)
|
|
{
|
|
cmCacheManager::CacheIterator it =
|
|
manager->GetCacheIterator(ii->c_str());
|
|
if(it.GetPropertyAsBool("ADVANCED"))
|
|
{
|
|
notFoundVars += " (ADVANCED)";
|
|
}
|
|
}
|
|
notFoundVars += "\n";
|
|
}
|
|
cmSystemTools::Error("This project requires some variables to be set,\n"
|
|
"and cmake can not find them.\n"
|
|
"Please set the following variables:\n",
|
|
notFoundVars.c_str());
|
|
}
|
|
if ( !m_CMakeInstance->GetScriptMode() )
|
|
{
|
|
m_CMakeInstance->UpdateProgress("Configuring done", -1);
|
|
}
|
|
}
|
|
|
|
// loop through the directories creating cmLocalGenerators and Configure()
|
|
void cmGlobalGenerator::RecursiveConfigure(cmLocalGenerator *lg,
|
|
float startProgress,
|
|
float endProgress)
|
|
{
|
|
// configure the current directory
|
|
lg->Configure();
|
|
|
|
// get all the subdirectories
|
|
std::vector<std::pair<cmStdString, bool> > subdirs = lg->GetMakefile()->GetSubDirectories();
|
|
float progressPiece = (endProgress - startProgress)/(1.0f+subdirs.size());
|
|
m_CMakeInstance->UpdateProgress("Configuring",
|
|
startProgress + progressPiece);
|
|
|
|
// for each subdir recurse
|
|
unsigned int i;
|
|
for (i = 0; i < subdirs.size(); ++i)
|
|
{
|
|
cmLocalGenerator *lg2 = this->CreateLocalGenerator();
|
|
lg2->SetParent(lg);
|
|
m_LocalGenerators.push_back(lg2);
|
|
|
|
// add the subdir to the start output directory
|
|
std::string outdir = lg->GetMakefile()->GetStartOutputDirectory();
|
|
outdir += "/";
|
|
outdir += subdirs[i].first;
|
|
lg2->GetMakefile()->SetStartOutputDirectory(outdir.c_str());
|
|
lg2->SetExcludeAll(!subdirs[i].second);
|
|
// add the subdir to the start source directory
|
|
std::string currentDir = lg->GetMakefile()->GetStartDirectory();
|
|
currentDir += "/";
|
|
currentDir += subdirs[i].first;
|
|
lg2->GetMakefile()->SetStartDirectory(currentDir.c_str());
|
|
lg2->GetMakefile()->MakeStartDirectoriesCurrent();
|
|
|
|
this->RecursiveConfigure(lg2,
|
|
startProgress + (i+1.0f)*progressPiece,
|
|
startProgress + (i+2.0f)*progressPiece);
|
|
}
|
|
}
|
|
|
|
void cmGlobalGenerator::Generate()
|
|
{
|
|
// For each existing cmLocalGenerator
|
|
unsigned int i;
|
|
for (i = 0; i < m_LocalGenerators.size(); ++i)
|
|
{
|
|
m_LocalGenerators[i]->Generate(true);
|
|
m_LocalGenerators[i]->GenerateInstallRules();
|
|
m_CMakeInstance->UpdateProgress("Generating",
|
|
(i+1.0f)/m_LocalGenerators.size());
|
|
}
|
|
m_CMakeInstance->UpdateProgress("Generating done", -1);
|
|
}
|
|
|
|
void cmGlobalGenerator::LocalGenerate()
|
|
{
|
|
// for this case we create one LocalGenerator
|
|
// configure it, and then Generate it
|
|
// start with this directory
|
|
cmLocalGenerator *lg = this->CreateLocalGenerator();
|
|
|
|
// set the Start directories
|
|
lg->GetMakefile()->SetStartDirectory(m_CMakeInstance->GetStartDirectory());
|
|
lg->GetMakefile()->SetStartOutputDirectory(m_CMakeInstance->GetStartOutputDirectory());
|
|
lg->GetMakefile()->MakeStartDirectoriesCurrent();
|
|
|
|
// now do trhe configure
|
|
lg->Configure();
|
|
lg->ConfigureFinalPass();
|
|
lg->Generate(false);
|
|
delete lg;
|
|
}
|
|
|
|
int cmGlobalGenerator::TryCompile(const char *, const char *bindir,
|
|
const char *, const char *target,
|
|
std::string *output)
|
|
{
|
|
// now build the test
|
|
std::string makeCommand =
|
|
m_CMakeInstance->GetCacheManager()->GetCacheValue("CMAKE_MAKE_PROGRAM");
|
|
if(makeCommand.size() == 0)
|
|
{
|
|
cmSystemTools::Error(
|
|
"Generator cannot find the appropriate make command.");
|
|
return 1;
|
|
}
|
|
makeCommand = cmSystemTools::ConvertToOutputPath(makeCommand.c_str());
|
|
|
|
/**
|
|
* Run an executable command and put the stdout in output.
|
|
*/
|
|
std::string cwd = cmSystemTools::GetCurrentWorkingDirectory();
|
|
cmSystemTools::ChangeDirectory(bindir);
|
|
|
|
// Since we have full control over the invocation of nmake, let us
|
|
// make it quiet.
|
|
if ( strcmp(this->GetName(), "NMake Makefiles") == 0 )
|
|
{
|
|
makeCommand += " /NOLOGO ";
|
|
}
|
|
|
|
// now build
|
|
if (target)
|
|
{
|
|
makeCommand += " ";
|
|
makeCommand += target;
|
|
#if defined(_WIN32) || defined(__CYGWIN__)
|
|
std::string tmp = target;
|
|
// if the target does not already end in . something
|
|
// then assume .exe
|
|
if(tmp.size() < 4 || tmp[tmp.size()-4] != '.')
|
|
{
|
|
makeCommand += ".exe";
|
|
}
|
|
#endif // WIN32
|
|
}
|
|
else
|
|
{
|
|
makeCommand += " all";
|
|
}
|
|
int retVal;
|
|
int timeout = cmGlobalGenerator::s_TryCompileTimeout;
|
|
bool hideconsole = cmSystemTools::GetRunCommandHideConsole();
|
|
cmSystemTools::SetRunCommandHideConsole(true);
|
|
if (!cmSystemTools::RunSingleCommand(makeCommand.c_str(), output,
|
|
&retVal, 0, false, timeout))
|
|
{
|
|
cmSystemTools::SetRunCommandHideConsole(hideconsole);
|
|
cmSystemTools::Error("Generator: execution of make failed.");
|
|
// return to the original directory
|
|
cmSystemTools::ChangeDirectory(cwd.c_str());
|
|
return 1;
|
|
}
|
|
cmSystemTools::SetRunCommandHideConsole(hideconsole);
|
|
|
|
// The SGI MipsPro 7.3 compiler does not return an error code when
|
|
// the source has a #error in it! This is a work-around for such
|
|
// compilers.
|
|
if((retVal == 0) && (output->find("#error") != std::string::npos))
|
|
{
|
|
retVal = 1;
|
|
}
|
|
|
|
cmSystemTools::ChangeDirectory(cwd.c_str());
|
|
return retVal;
|
|
}
|
|
|
|
cmLocalGenerator *cmGlobalGenerator::CreateLocalGenerator()
|
|
{
|
|
cmLocalGenerator *lg = new cmLocalGenerator;
|
|
lg->SetGlobalGenerator(this);
|
|
return lg;
|
|
}
|
|
|
|
void cmGlobalGenerator::EnableLanguagesFromGenerator(cmGlobalGenerator *gen )
|
|
{
|
|
this->SetConfiguredFilesPath(
|
|
gen->GetCMakeInstance()->GetHomeOutputDirectory());
|
|
const char* make =
|
|
gen->GetCMakeInstance()->GetCacheDefinition("CMAKE_MAKE_PROGRAM");
|
|
this->GetCMakeInstance()->AddCacheEntry("CMAKE_MAKE_PROGRAM", make,
|
|
"make program",
|
|
cmCacheManager::FILEPATH);
|
|
// if C, then enable C
|
|
if(gen->GetLanguageEnabled("C"))
|
|
{
|
|
this->SetLanguageEnabled("C");
|
|
}
|
|
|
|
// if CXX
|
|
if(gen->GetLanguageEnabled("CXX"))
|
|
{
|
|
this->SetLanguageEnabled("CXX");
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------------
|
|
void cmGlobalGenerator::GetDocumentation(cmDocumentationEntry& entry) const
|
|
{
|
|
entry.name = this->GetName();
|
|
entry.brief = "";
|
|
entry.full = "";
|
|
}
|
|
|
|
bool cmGlobalGenerator::IsExcluded(cmLocalGenerator* root,
|
|
cmLocalGenerator* gen)
|
|
{
|
|
cmLocalGenerator* cur = gen->GetParent();
|
|
while(cur && cur != root)
|
|
{
|
|
if(cur->GetExcludeAll())
|
|
{
|
|
return true;
|
|
}
|
|
cur = cur->GetParent();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|