518 lines
16 KiB
C++
518 lines
16 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 "cmLocalGenerator.h"
|
|
#include "cmGlobalGenerator.h"
|
|
#include "cmake.h"
|
|
#include "cmMakefile.h"
|
|
#include "cmGeneratedFileStream.h"
|
|
|
|
cmLocalGenerator::cmLocalGenerator()
|
|
{
|
|
m_Makefile = new cmMakefile;
|
|
m_Makefile->SetLocalGenerator(this);
|
|
m_ExcludeFromAll = false;
|
|
m_Parent = 0;
|
|
}
|
|
|
|
cmLocalGenerator::~cmLocalGenerator()
|
|
{
|
|
delete m_Makefile;
|
|
}
|
|
|
|
void cmLocalGenerator::Configure()
|
|
{
|
|
// set the PROJECT_SOURCE_DIR and PROJECT_BIN_DIR to default values
|
|
// just in case the project does not include a PROJECT command
|
|
m_Makefile->AddDefinition("PROJECT_BINARY_DIR",
|
|
m_Makefile->GetHomeOutputDirectory());
|
|
m_Makefile->AddDefinition("PROJECT_SOURCE_DIR",
|
|
m_Makefile->GetHomeDirectory());
|
|
|
|
// find & read the list file
|
|
std::string currentStart = m_Makefile->GetStartDirectory();
|
|
currentStart += "/CMakeLists.txt";
|
|
m_Makefile->ReadListFile(currentStart.c_str());
|
|
}
|
|
|
|
void cmLocalGenerator::SetGlobalGenerator(cmGlobalGenerator *gg)
|
|
{
|
|
m_GlobalGenerator = gg;
|
|
|
|
// setup the home directories
|
|
m_Makefile->SetHomeDirectory(
|
|
gg->GetCMakeInstance()->GetHomeDirectory());
|
|
m_Makefile->SetHomeOutputDirectory(
|
|
gg->GetCMakeInstance()->GetHomeOutputDirectory());
|
|
|
|
}
|
|
|
|
|
|
void cmLocalGenerator::ConfigureFinalPass()
|
|
{
|
|
m_Makefile->ConfigureFinalPass();
|
|
}
|
|
|
|
void cmLocalGenerator::GenerateInstallRules()
|
|
{
|
|
const cmTargets &tgts = m_Makefile->GetTargets();
|
|
const char* prefix
|
|
= m_Makefile->GetDefinition("CMAKE_INSTALL_PREFIX");
|
|
if (!prefix)
|
|
{
|
|
prefix = "/usr/local";
|
|
}
|
|
|
|
std::string file = m_Makefile->GetStartOutputDirectory();
|
|
std::string homedir = m_Makefile->GetHomeOutputDirectory();
|
|
std::string currdir = m_Makefile->GetCurrentOutputDirectory();
|
|
cmSystemTools::ConvertToUnixSlashes(file);
|
|
cmSystemTools::ConvertToUnixSlashes(homedir);
|
|
cmSystemTools::ConvertToUnixSlashes(currdir);
|
|
int toplevel_install = 0;
|
|
if ( currdir == homedir )
|
|
{
|
|
toplevel_install = 1;
|
|
}
|
|
file += "/cmake_install.cmake";
|
|
cmGeneratedFileStream tempFile(file.c_str());
|
|
std::ostream& fout = tempFile.GetStream();
|
|
|
|
fout << "# Install script for directory: " << m_Makefile->GetCurrentDirectory()
|
|
<< std::endl << std::endl;
|
|
|
|
const char* cmakeDebugPosfix = m_Makefile->GetDefinition("CMAKE_DEBUG_POSTFIX");
|
|
if ( cmakeDebugPosfix )
|
|
{
|
|
fout << "SET(CMAKE_DEBUG_POSTFIX \"" << cmakeDebugPosfix << "\")"
|
|
<< std::endl << std::endl;
|
|
}
|
|
|
|
std::string libOutPath = "";
|
|
if (m_Makefile->GetDefinition("LIBRARY_OUTPUT_PATH"))
|
|
{
|
|
libOutPath = m_Makefile->GetDefinition("LIBRARY_OUTPUT_PATH");
|
|
if(libOutPath.size())
|
|
{
|
|
if(libOutPath[libOutPath.size() -1] != '/')
|
|
{
|
|
libOutPath += "/";
|
|
}
|
|
}
|
|
}
|
|
|
|
std::string exeOutPath = "";
|
|
if (m_Makefile->GetDefinition("EXECUTABLE_OUTPUT_PATH"))
|
|
{
|
|
exeOutPath =
|
|
m_Makefile->GetDefinition("EXECUTABLE_OUTPUT_PATH");
|
|
if(exeOutPath.size())
|
|
{
|
|
if(exeOutPath[exeOutPath.size() -1] != '/')
|
|
{
|
|
exeOutPath += "/";
|
|
}
|
|
}
|
|
}
|
|
if ( libOutPath.size() == 0 )
|
|
{
|
|
// LIBRARY_OUTPUT_PATH not defined
|
|
libOutPath = currdir + "/";
|
|
}
|
|
if ( exeOutPath.size() == 0 )
|
|
{
|
|
// EXECUTABLE_OUTPUT_PATH not defined
|
|
exeOutPath = currdir + "/";
|
|
}
|
|
|
|
std::string destination;
|
|
for(cmTargets::const_iterator l = tgts.begin();
|
|
l != tgts.end(); l++)
|
|
{
|
|
const char* preinstall = l->second.GetProperty("PRE_INSTALL_SCRIPT");
|
|
const char* postinstall = l->second.GetProperty("POST_INSTALL_SCRIPT");
|
|
if ( preinstall )
|
|
{
|
|
fout << "INCLUDE(\"" << preinstall << "\")" << std::endl;
|
|
}
|
|
if (l->second.GetInstallPath() != "")
|
|
{
|
|
destination = prefix + l->second.GetInstallPath();
|
|
cmSystemTools::ConvertToUnixSlashes(destination);
|
|
const char* dest = destination.c_str();
|
|
int type = l->second.GetType();
|
|
|
|
|
|
std::string fname;
|
|
const char* files;
|
|
// now install the target
|
|
switch (type)
|
|
{
|
|
case cmTarget::STATIC_LIBRARY:
|
|
case cmTarget::MODULE_LIBRARY:
|
|
fname = libOutPath;
|
|
fname += this->GetFullTargetName(l->first.c_str(), l->second);
|
|
files = fname.c_str();
|
|
this->AddInstallRule(fout, dest, type, files);
|
|
break;
|
|
case cmTarget::SHARED_LIBRARY:
|
|
{
|
|
// Special code to handle DLL
|
|
fname = libOutPath;
|
|
fname += this->GetFullTargetName(l->first.c_str(), l->second);
|
|
std::string ext = cmSystemTools::GetFilenameExtension(fname);
|
|
ext = cmSystemTools::LowerCase(ext);
|
|
if ( ext == ".dll" )
|
|
{
|
|
std::string libname = libOutPath;
|
|
libname += cmSystemTools::GetFilenameWithoutExtension(fname);
|
|
libname += ".lib";
|
|
files = libname.c_str();
|
|
this->AddInstallRule(fout, dest, cmTarget::STATIC_LIBRARY, files, true);
|
|
std::string dlldest = prefix + l->second.GetRuntimeInstallPath();
|
|
files = fname.c_str();
|
|
this->AddInstallRule(fout, dlldest.c_str(), type, files);
|
|
}
|
|
else
|
|
{
|
|
files = fname.c_str();
|
|
std::string properties;
|
|
const char* lib_version = l->second.GetProperty("VERSION");
|
|
const char* lib_soversion = l->second.GetProperty("SOVERSION");
|
|
if(!m_Makefile->GetDefinition("CMAKE_SHARED_LIBRARY_SONAME_C_FLAG"))
|
|
{
|
|
// Versioning is supported only for shared libraries and modules,
|
|
// and then only when the platform supports an soname flag.
|
|
lib_version = 0;
|
|
lib_soversion = 0;
|
|
}
|
|
if ( lib_version )
|
|
{
|
|
properties += " VERSION ";
|
|
properties += lib_version;
|
|
}
|
|
if ( lib_soversion )
|
|
{
|
|
properties += " SOVERSION ";
|
|
properties += lib_soversion;
|
|
}
|
|
this->AddInstallRule(fout, dest, type, files, false, properties.c_str());
|
|
}
|
|
}
|
|
break;
|
|
case cmTarget::EXECUTABLE:
|
|
fname = exeOutPath;
|
|
fname += this->GetFullTargetName(l->first.c_str(), l->second);
|
|
files = fname.c_str();
|
|
this->AddInstallRule(fout, dest, type, files);
|
|
break;
|
|
case cmTarget::INSTALL_FILES:
|
|
{
|
|
std::string sourcePath = m_Makefile->GetCurrentDirectory();
|
|
std::string binaryPath = m_Makefile->GetCurrentOutputDirectory();
|
|
sourcePath += "/";
|
|
binaryPath += "/";
|
|
const std::vector<std::string> &sf = l->second.GetSourceLists();
|
|
std::vector<std::string>::const_iterator i;
|
|
for (i = sf.begin(); i != sf.end(); ++i)
|
|
{
|
|
std::string f = *i;
|
|
if(f.substr(0, sourcePath.length()) == sourcePath)
|
|
{
|
|
f = f.substr(sourcePath.length());
|
|
}
|
|
else if(f.substr(0, binaryPath.length()) == binaryPath)
|
|
{
|
|
f = f.substr(binaryPath.length());
|
|
}
|
|
|
|
files = i->c_str();
|
|
this->AddInstallRule(fout, dest, type, files);
|
|
}
|
|
}
|
|
break;
|
|
case cmTarget::INSTALL_PROGRAMS:
|
|
{
|
|
std::string sourcePath = m_Makefile->GetCurrentDirectory();
|
|
std::string binaryPath = m_Makefile->GetCurrentOutputDirectory();
|
|
sourcePath += "/";
|
|
binaryPath += "/";
|
|
const std::vector<std::string> &sf = l->second.GetSourceLists();
|
|
std::vector<std::string>::const_iterator i;
|
|
for (i = sf.begin(); i != sf.end(); ++i)
|
|
{
|
|
std::string f = *i;
|
|
if(f.substr(0, sourcePath.length()) == sourcePath)
|
|
{
|
|
f = f.substr(sourcePath.length());
|
|
}
|
|
else if(f.substr(0, binaryPath.length()) == binaryPath)
|
|
{
|
|
f = f.substr(binaryPath.length());
|
|
}
|
|
files = i->c_str();
|
|
this->AddInstallRule(fout, dest, type, files);
|
|
}
|
|
}
|
|
break;
|
|
case cmTarget::UTILITY:
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
if ( postinstall )
|
|
{
|
|
fout << "INCLUDE(\"" << postinstall << "\")" << std::endl;
|
|
}
|
|
}
|
|
cmMakefile* mf = this->GetMakefile();
|
|
if ( !mf->GetSubDirectories().empty() )
|
|
{
|
|
const std::vector<std::pair<cmStdString, bool> >& subdirs = mf->GetSubDirectories();
|
|
std::vector<std::pair<cmStdString, bool> >::const_iterator i = subdirs.begin();
|
|
for(; i != subdirs.end(); ++i)
|
|
{
|
|
std::string odir = mf->GetCurrentOutputDirectory();
|
|
odir += "/" + (*i).first;
|
|
cmSystemTools::ConvertToUnixSlashes(odir);
|
|
fout << "INCLUDE(\"" << odir.c_str()
|
|
<< "/cmake_install.cmake\")" << std::endl;
|
|
}
|
|
fout << std::endl;;
|
|
}
|
|
if ( toplevel_install )
|
|
{
|
|
fout << "FOREACH(file ${CMAKE_INSTALL_MANIFEST_FILES})" << std::endl
|
|
<< " FILE(APPEND \"" << homedir.c_str() << "/install_manifest.txt\" "
|
|
<< "\"${file}\\n\")" << std::endl
|
|
<< "ENDFOREACH(file)" << std::endl;
|
|
}
|
|
}
|
|
|
|
void cmLocalGenerator::AddInstallRule(std::ostream& fout, const char* dest,
|
|
int type, const char* files, bool optional /* = false */, const char* properties /* = 0 */)
|
|
{
|
|
std::string sfiles = files;
|
|
std::string destination = dest;
|
|
std::string stype;
|
|
switch ( type )
|
|
{
|
|
case cmTarget::INSTALL_PROGRAMS: stype = "PROGRAM"; break;
|
|
case cmTarget::EXECUTABLE: stype = "EXECUTABLE"; break;
|
|
case cmTarget::STATIC_LIBRARY: stype = "STATIC_LIBRARY"; break;
|
|
case cmTarget::SHARED_LIBRARY: stype = "SHARED_LIBRARY"; break;
|
|
case cmTarget::MODULE_LIBRARY: stype = "MODULE"; break;
|
|
case cmTarget::INSTALL_FILES:
|
|
default: stype = "FILE"; break;
|
|
}
|
|
std::string fname = cmSystemTools::GetFilenameName(sfiles.c_str());
|
|
fout
|
|
<< "MESSAGE(STATUS \"Installing " << destination.c_str()
|
|
<< "/" << fname.c_str() << "\")\n"
|
|
<< "FILE(INSTALL DESTINATION \"" << destination.c_str()
|
|
<< "\" TYPE " << stype.c_str() << (optional?" OPTIONAL":"") ;
|
|
if ( properties && *properties )
|
|
{
|
|
fout << " PROPERTIES" << properties;
|
|
}
|
|
fout
|
|
<< " FILES \"" << sfiles.c_str() << "\")\n";
|
|
}
|
|
|
|
std::string cmLocalGenerator::GetFullTargetName(const char* n,
|
|
const cmTarget& t)
|
|
{
|
|
const char* targetPrefix = t.GetProperty("PREFIX");
|
|
const char* targetSuffix = t.GetProperty("SUFFIX");
|
|
const char* prefixVar = 0;
|
|
const char* suffixVar = 0;
|
|
switch(t.GetType())
|
|
{
|
|
case cmTarget::STATIC_LIBRARY:
|
|
prefixVar = "CMAKE_STATIC_LIBRARY_PREFIX";
|
|
suffixVar = "CMAKE_STATIC_LIBRARY_SUFFIX";
|
|
break;
|
|
case cmTarget::SHARED_LIBRARY:
|
|
prefixVar = "CMAKE_SHARED_LIBRARY_PREFIX";
|
|
suffixVar = "CMAKE_SHARED_LIBRARY_SUFFIX";
|
|
break;
|
|
case cmTarget::MODULE_LIBRARY:
|
|
prefixVar = "CMAKE_SHARED_MODULE_PREFIX";
|
|
suffixVar = "CMAKE_SHARED_MODULE_SUFFIX";
|
|
break;
|
|
case cmTarget::EXECUTABLE:
|
|
targetSuffix = cmSystemTools::GetExecutableExtension();
|
|
case cmTarget::UTILITY:
|
|
case cmTarget::INSTALL_FILES:
|
|
case cmTarget::INSTALL_PROGRAMS:
|
|
break;
|
|
}
|
|
// if there is no prefix on the target use the cmake definition
|
|
if(!targetPrefix && prefixVar)
|
|
{
|
|
targetPrefix = m_Makefile->GetSafeDefinition(prefixVar);
|
|
}
|
|
// if there is no suffix on the target use the cmake definition
|
|
if(!targetSuffix && suffixVar)
|
|
{
|
|
targetSuffix = m_Makefile->GetSafeDefinition(suffixVar);
|
|
}
|
|
std::string name = targetPrefix?targetPrefix:"";
|
|
name += n;
|
|
name += targetSuffix?targetSuffix:"";
|
|
return name;
|
|
}
|
|
|
|
|
|
std::string cmLocalGenerator::ConvertToRelativeOutputPath(const char* p)
|
|
{
|
|
if ( !m_Makefile->IsOn("CMAKE_USE_RELATIVE_PATHS") )
|
|
{
|
|
return cmSystemTools::ConvertToOutputPath(p);
|
|
}
|
|
// do not use relative paths for network build trees
|
|
// the network paths do not work
|
|
const char* outputDirectory = m_Makefile->GetHomeOutputDirectory();
|
|
if ( outputDirectory && *outputDirectory && *(outputDirectory+1) &&
|
|
outputDirectory[0] == '/' && outputDirectory[1] == '/' )
|
|
{
|
|
return cmSystemTools::ConvertToOutputPath(p);
|
|
}
|
|
|
|
// The first time this is called, initialize all
|
|
// the path ivars that are used. This can not
|
|
// be moved to the constructor because all the paths are not set yet.
|
|
if(m_CurrentOutputDirectory.size() == 0)
|
|
{
|
|
m_CurrentOutputDirectory = m_Makefile->GetCurrentOutputDirectory();
|
|
m_HomeOutputDirectory = m_Makefile->GetHomeOutputDirectory();
|
|
m_HomeDirectory = m_Makefile->GetHomeDirectory();
|
|
if(m_RelativePathToSourceDir.size() == 0)
|
|
{
|
|
m_RelativePathToSourceDir = cmSystemTools::RelativePath(
|
|
m_CurrentOutputDirectory.c_str(),
|
|
m_HomeDirectory.c_str());
|
|
std::string path = m_CurrentOutputDirectory;
|
|
cmSystemTools::ReplaceString(path, m_HomeOutputDirectory.c_str(), "");
|
|
unsigned i;
|
|
m_RelativePathToBinaryDir = "";
|
|
for(i =0; i < path.size(); ++i)
|
|
{
|
|
if(path[i] == '/')
|
|
{
|
|
m_RelativePathToBinaryDir += "../";
|
|
}
|
|
}
|
|
}
|
|
m_HomeOutputDirectoryNoSlash = m_HomeOutputDirectory;
|
|
m_HomeOutputDirectory += "/";
|
|
m_CurrentOutputDirectory += "/";
|
|
}
|
|
|
|
// Do the work of converting to a relative path
|
|
std::string pathIn = p;
|
|
if(pathIn.find('/') == pathIn.npos)
|
|
{
|
|
return pathIn;
|
|
}
|
|
|
|
if(pathIn.size() && pathIn[0] == '\"')
|
|
{
|
|
pathIn = pathIn.substr(1, pathIn.size()-2);
|
|
}
|
|
|
|
std::string ret = pathIn;
|
|
if(m_CurrentOutputDirectory.size() <= ret.size())
|
|
{
|
|
std::string sub = ret.substr(0, m_CurrentOutputDirectory.size());
|
|
if(
|
|
#if defined(_WIN32) || defined(__APPLE__)
|
|
cmSystemTools::LowerCase(sub) ==
|
|
cmSystemTools::LowerCase(m_CurrentOutputDirectory)
|
|
#else
|
|
sub == m_CurrentOutputDirectory
|
|
#endif
|
|
)
|
|
{
|
|
ret = ret.substr(m_CurrentOutputDirectory.size(), ret.npos);
|
|
}
|
|
}
|
|
if(m_HomeDirectory.size() <= ret.size())
|
|
{
|
|
std::string sub = ret.substr(0, m_HomeDirectory.size());
|
|
if(
|
|
#if defined(_WIN32) || defined(__APPLE__)
|
|
cmSystemTools::LowerCase(sub) ==
|
|
cmSystemTools::LowerCase(m_HomeDirectory)
|
|
#else
|
|
sub == m_HomeDirectory
|
|
#endif
|
|
)
|
|
{
|
|
ret = m_RelativePathToSourceDir + ret.substr(m_HomeDirectory.size(), ret.npos);
|
|
}
|
|
}
|
|
if(m_HomeOutputDirectory.size() <= ret.size())
|
|
{
|
|
std::string sub = ret.substr(0, m_HomeOutputDirectory.size());
|
|
if(
|
|
#if defined(_WIN32) || defined(__APPLE__)
|
|
cmSystemTools::LowerCase(sub) ==
|
|
cmSystemTools::LowerCase(m_HomeOutputDirectory)
|
|
#else
|
|
sub == m_HomeOutputDirectory
|
|
#endif
|
|
)
|
|
{
|
|
ret = m_RelativePathToBinaryDir + ret.substr(m_HomeOutputDirectory.size(), ret.npos);
|
|
}
|
|
}
|
|
|
|
std::string relpath = m_RelativePathToBinaryDir;
|
|
if(relpath.size())
|
|
{
|
|
relpath.erase(relpath.size()-1, 1);
|
|
}
|
|
else
|
|
{
|
|
relpath = ".";
|
|
}
|
|
if(
|
|
#if defined(_WIN32) || defined(__APPLE__)
|
|
cmSystemTools::LowerCase(ret) ==
|
|
cmSystemTools::LowerCase(m_HomeOutputDirectoryNoSlash)
|
|
#else
|
|
ret == m_HomeOutputDirectoryNoSlash
|
|
#endif
|
|
)
|
|
{
|
|
ret = relpath;
|
|
}
|
|
|
|
// Relative paths should always start in a '.', so add a './' if
|
|
// necessary.
|
|
if(ret.size()
|
|
&& ret[0] != '\"' && ret[0] != '/' && ret[0] != '.' && ret[0] != '$')
|
|
{
|
|
if(ret.size() > 1 && ret[1] != ':')
|
|
{
|
|
ret = std::string("./") + ret;
|
|
}
|
|
}
|
|
ret = cmSystemTools::ConvertToOutputPath(ret.c_str());
|
|
return ret;
|
|
}
|