CMake/Source/cmMakefile.cxx

1022 lines
29 KiB
C++
Raw Normal View History

2001-01-11 22:55:47 +03:00
/*=========================================================================
Program: Insight Segmentation & Registration Toolkit
Module: $RCSfile$
Language: C++
Date: $Date$
Version: $Revision$
2001-04-27 16:01:17 +04:00
Copyright (c) 2001 Insight Consortium
All rights reserved.
2001-01-11 22:55:47 +03:00
2001-04-27 16:01:17 +04:00
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
2001-01-11 22:55:47 +03:00
2001-04-27 16:01:17 +04:00
* Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
* The name of the Insight Consortium, nor the names of any consortium members,
nor of any contributors, may be used to endorse or promote products derived
from this software without specific prior written permission.
* Modified source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER AND CONTRIBUTORS ``AS IS''
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE FOR
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
2001-01-11 22:55:47 +03:00
=========================================================================*/
#include "cmMakefile.h"
2001-01-18 19:20:24 +03:00
#include "cmCommand.h"
#include "cmStandardIncludes.h"
2001-04-25 00:49:12 +04:00
#include "cmSourceFile.h"
#include "cmDirectory.h"
#include "cmSystemTools.h"
#include "cmMakefileGenerator.h"
#include "cmCommands.h"
#include "cmCacheManager.h"
2001-04-20 01:39:03 +04:00
#include "cmFunctionBlocker.h"
// default is not to be building executables
cmMakefile::cmMakefile()
{
// Setup the default include file regular expression.
// Should be changed to something like "\\.(h|hh|hpp|hxx)$" or "^.*$"
m_IncludeFileRegularExpression = "^itk|^vtk|^vnl|^vcl|^f2c";
m_DefineFlags = " ";
m_MakefileGenerator = 0;
this->AddSourceGroup("", "^.*$");
this->AddSourceGroup("Source Files", "\\.(cpp|C|c|cxx|rc|def|r|odl|idl|hpj|bat)$");
this->AddSourceGroup("Header Files", "\\.(h|hh|hpp|hxx|hm|inl)$");
2001-01-18 19:20:24 +03:00
this->AddDefaultCommands();
this->AddDefaultDefinitions();
2001-05-02 00:34:53 +04:00
cmCacheManager::GetInstance()->DefineCache(this);
}
2001-01-18 19:20:24 +03:00
void cmMakefile::AddDefaultCommands()
{
std::list<cmCommand*> commands;
GetPredefinedCommands(commands);
for(std::list<cmCommand*>::iterator i = commands.begin();
i != commands.end(); ++i)
{
this->AddCommand(*i);
}
2001-05-08 02:11:16 +04:00
#if defined(_WIN32) || defined(__CYGWIN__)
this->AddDefinition("WIN32", "1");
#else
this->AddDefinition("UNIX", "1");
#endif
2001-01-18 19:20:24 +03:00
// Cygwin is more like unix so enable the unix commands
#if defined(__CYGWIN__)
this->AddDefinition("UNIX", "1");
#endif
2001-05-23 19:53:37 +04:00
// always creat an empty install target
cmTarget target;
target.SetType(cmTarget::INSTALL);
target.SetInAll(false);
m_Targets.insert(cmTargets::value_type("INSTALL",target));
}
cmMakefile::~cmMakefile()
{
for(unsigned int i=0; i < m_UsedCommands.size(); i++)
{
2001-01-18 19:20:24 +03:00
delete m_UsedCommands[i];
}
2001-01-18 19:20:24 +03:00
for(RegisteredCommandsMap::iterator j = m_Commands.begin();
j != m_Commands.end(); ++j)
{
delete (*j).second;
}
for(DataMap::const_iterator d = m_DataMap.begin();
d != m_DataMap.end(); ++d)
{
if(d->second)
{
delete d->second;
}
}
2001-06-04 18:18:03 +04:00
std::set<cmFunctionBlocker *>::const_iterator pos;
for (pos = m_FunctionBlockers.begin();
pos != m_FunctionBlockers.end(); ++pos)
{
cmFunctionBlocker* b = *pos;
m_FunctionBlockers.erase(*pos);
delete b;
}
delete m_MakefileGenerator;
}
2001-04-20 01:39:03 +04:00
void cmMakefile::PrintStringVector(const char* s, const std::vector<std::string>& v) const
{
std::cout << s << ": ( \n";
2001-04-20 01:39:03 +04:00
for(std::vector<std::string>::const_iterator i = v.begin();
i != v.end(); ++i)
{
std::cout << (*i).c_str() << " ";
}
std::cout << " )\n";
}
// call print on all the classes in the makefile
2001-04-20 01:39:03 +04:00
void cmMakefile::Print() const
{
// print the class lists
std::cout << "classes:\n";
2001-04-25 00:49:12 +04:00
for(SourceMap::const_iterator l = m_Sources.begin();
l != m_Sources.end(); l++)
{
std::cout << " Class list named: " << l->first << std::endl;
2001-04-25 00:49:12 +04:00
for(std::vector<cmSourceFile>::const_iterator i = l->second.begin();
i != l->second.end(); i++)
{
i->Print();
}
}
std::cout << " m_Targets: ";
for (cmTargets::const_iterator l = m_Targets.begin();
l != m_Targets.end(); l++)
{
std::cout << l->first << std::endl;
}
2001-02-15 21:30:13 +03:00
std::cout << " m_CurrentOutputDirectory; " <<
m_CurrentOutputDirectory.c_str() << std::endl;
std::cout << " m_StartOutputDirectory; " <<
m_StartOutputDirectory.c_str() << std::endl;
std::cout << " m_HomeOutputDirectory; " <<
m_HomeOutputDirectory.c_str() << std::endl;
std::cout << " m_cmCurrentDirectory; " <<
m_cmCurrentDirectory.c_str() << std::endl;
2001-02-15 21:30:13 +03:00
std::cout << " m_cmStartDirectory; " <<
m_cmStartDirectory.c_str() << std::endl;
std::cout << " m_cmHomeDirectory; " <<
m_cmHomeDirectory.c_str() << std::endl;
std::cout << " m_ProjectName; " << m_ProjectName.c_str() << std::endl;
this->PrintStringVector("m_SubDirectories ", m_SubDirectories);
this->PrintStringVector("m_MakeVerbatim ", m_MakeVerbatim);
this->PrintStringVector("m_IncludeDirectories;", m_IncludeDirectories);
this->PrintStringVector("m_LinkDirectories", m_LinkDirectories);
this->PrintStringVector("m_Utilities", m_Utilities);
this->PrintStringVector("m_UtilityDirectories", m_UtilityDirectories);
for( std::vector<cmSourceGroup>::const_iterator i = m_SourceGroups.begin();
i != m_SourceGroups.end(); ++i)
{
i->Print();
}
}
// Parse the given CMakeLists.txt file into a list of classes.
2001-02-15 21:30:13 +03:00
// Reads in current CMakeLists file and all parent CMakeLists files
// executing all inherited commands in the parents
2001-04-30 18:52:58 +04:00
//
// if external is non-zero, this means that we have branched to grab some
// commands from a remote list-file (that is, the equivalent of a
// #include has been called). We DO NOT look at the parents of this
// list-file, and for all other purposes, the name of this list-file
// is "filename" and not "external".
bool cmMakefile::ReadListFile(const char* filename, const char* external)
{
2001-04-30 18:52:58 +04:00
// keep track of the current file being read
2001-05-03 16:52:32 +04:00
if (filename)
{
2001-06-04 18:18:03 +04:00
if(m_cmCurrentListFile != filename)
{
m_cmCurrentListFile = filename;
}
2001-05-03 16:52:32 +04:00
}
2001-04-30 18:52:58 +04:00
// if this is not a remote makefile
// (if it were, this would be called from the "filename" call,
// rather than the "external" call)
if (!external)
{
2001-04-30 18:52:58 +04:00
// is there a parent CMakeLists file that does not go beyond the
// Home directory? if so recurse and read in that List file
std::string parentList = this->GetParentListFileName(filename);
if (parentList != "")
{
// save the current directory
std::string srcdir = m_cmCurrentDirectory;
std::string bindir = m_CurrentOutputDirectory;
// compute the new current directories
std::string::size_type pos = m_cmCurrentDirectory.rfind('/');
if(pos != std::string::npos)
{
m_cmCurrentDirectory = m_cmCurrentDirectory.substr(0, pos);
}
pos = m_CurrentOutputDirectory.rfind('/');
if(pos != std::string::npos)
{
m_CurrentOutputDirectory = m_CurrentOutputDirectory.substr(0, pos);
}
this->ReadListFile(parentList.c_str());
// restore the current directory
m_cmCurrentDirectory = srcdir;
m_CurrentOutputDirectory = bindir;
}
}
2001-02-15 21:30:13 +03:00
// are we at the start CMakeLists file or are we processing a parent
// lists file
2001-04-30 18:52:58 +04:00
//
// this might, or might not be true, irrespective if we are
// off looking at an external makefile.
2001-02-15 21:30:13 +03:00
bool inheriting = (m_cmCurrentDirectory != m_cmStartDirectory);
// Now read the input file
2001-04-30 18:52:58 +04:00
const char *filenametoread= filename;
if( external)
2001-05-03 16:52:32 +04:00
{
2001-04-30 18:52:58 +04:00
filenametoread= external;
2001-05-03 16:52:32 +04:00
}
2001-04-30 18:52:58 +04:00
std::ifstream fin(filenametoread);
if(!fin)
{
2001-04-30 18:52:58 +04:00
cmSystemTools::Error("error can not open file ", filenametoread);
return false;
}
std::string name;
std::vector<std::string> arguments;
while ( fin )
{
// add this list file to the list of dependencies
m_ListFiles.push_back( filenametoread);
2001-04-30 18:52:58 +04:00
2001-04-20 01:39:03 +04:00
if(cmSystemTools::ParseFunction(fin, name, arguments) &&
!this->IsFunctionBlocked(name.c_str(),arguments))
{
RegisteredCommandsMap::iterator pos = m_Commands.find(name);
if(pos != m_Commands.end())
{
cmCommand* rm = (*pos).second;
cmCommand* usedCommand = rm->Clone();
usedCommand->SetMakefile(this);
bool keepCommand = false;
if(usedCommand->GetEnabled())
{
// if not running in inherit mode or
// if the command is inherited then Invoke it.
if(!inheriting || usedCommand->IsInherited())
{
if(!usedCommand->Invoke(arguments))
{
cmSystemTools::Error(usedCommand->GetName(),
": Error : \n",
usedCommand->GetError(),
m_cmCurrentDirectory.c_str());
}
else
{
// use the command
keepCommand = true;
m_UsedCommands.push_back(usedCommand);
}
}
}
// if the Cloned command was not used
// then delete it
if(!keepCommand)
{
delete usedCommand;
}
}
else
{
cmSystemTools::Error("unknown CMake command ", name.c_str(), filename);
}
}
}
2001-05-04 16:46:05 +04:00
// send scope ended to and funciton blockers
if (filename)
{
// loop over all function blockers to see if any block this command
std::set<cmFunctionBlocker *>::const_iterator pos;
for (pos = m_FunctionBlockers.begin();
pos != m_FunctionBlockers.end(); ++pos)
{
(*pos)->ScopeEnded(*this);
}
}
return true;
}
2001-04-25 00:49:12 +04:00
cmSourceFile *cmMakefile::GetSource(const char *srclist, const char *cname)
{
2001-04-25 00:49:12 +04:00
SourceMap::iterator sl = m_Sources.find(srclist);
// find the src list
2001-04-25 00:49:12 +04:00
if (sl == m_Sources.end())
{
return 0;
}
// find the class
2001-04-25 00:49:12 +04:00
for (std::vector<cmSourceFile>::iterator i = sl->second.begin();
i != sl->second.end(); ++i)
{
2001-04-25 00:49:12 +04:00
if (i->GetSourceName() == cname)
{
return &(*i);
}
}
return 0;
}
2001-01-18 19:20:24 +03:00
void cmMakefile::AddCommand(cmCommand* wg)
{
std::string name = wg->GetName();
2001-01-18 19:20:24 +03:00
m_Commands.insert( RegisteredCommandsMap::value_type(name, wg));
}
// Set the make file
void cmMakefile::SetMakefileGenerator(cmMakefileGenerator* mf)
{
2001-06-04 18:18:03 +04:00
if(mf == m_MakefileGenerator)
{
return;
}
delete m_MakefileGenerator;
m_MakefileGenerator = mf;
2001-05-21 17:32:11 +04:00
mf->SetMakefile(this);
}
// Generate the output file
void cmMakefile::GenerateMakefile()
{
// do all the variable expansions here
this->ExpandVariables();
2001-01-18 19:20:24 +03:00
// give all the commands a chance to do something
// after the file has been parsed before generation
2001-01-18 19:20:24 +03:00
for(std::vector<cmCommand*>::iterator i = m_UsedCommands.begin();
i != m_UsedCommands.end(); ++i)
{
(*i)->FinalPass();
}
// merge libraries
for (cmTargets::iterator l = m_Targets.begin();
l != m_Targets.end(); l++)
{
2001-05-16 23:15:21 +04:00
l->second.GenerateSourceFilesFromSourceLists(*this);
l->second.MergeLibraries(m_LinkLibraries);
}
// now do the generation
m_MakefileGenerator->GenerateMakefile();
// generate any testing files
this->GenerateTestfile();
}
void cmMakefile::GenerateTestfile()
{
if (m_Tests.empty() && this->GetSubDirectories().empty())
{
return;
}
// Create a full path filename for output Testfile
std::string fname;
fname = this->GetCurrentOutputDirectory();
fname += "/";
fname += "Testfile";
// Open the output Testfile
std::ofstream fout(fname.c_str());
if (!fout)
{
cmSystemTools::Error("Error Writing ", fname.c_str());
return;
}
fout << "# CMake generated Testfile for " << std::endl
<< "#\tSource directory: "
<< this->GetCurrentDirectory()
<< std::endl
<< "#\tBuild directory: " << this->GetCurrentOutputDirectory()
<< std::endl
<< "# " << std::endl
<< "# This file replicates the SUBDIRS() and ADD_TEST() commands from the source"
<< std::endl
<< "# tree CMakeLists.txt file, skipping any SUBDIRS() or ADD_TEST() commands"
<< std::endl
<< "# that are excluded by CMake control structures, i.e. IF() commands."
<< std::endl
<< "#"
<< std::endl << std::endl;
// write out the subdirs for the current directory
if (!this->GetSubDirectories().empty())
{
fout << "SUBDIRS(";
const std::vector<std::string>& subdirs = this->GetSubDirectories();
std::vector<std::string>::const_iterator i = subdirs.begin();
fout << (*i).c_str();
++i;
for(; i != subdirs.end(); ++i)
{
fout << " " << (*i).c_str();
}
fout << ")" << std::endl << std::endl;;
}
// write out each test
std::vector<std::vector<std::string> >::iterator testIt;
std::vector<std::string>::iterator it;
// for each test
for (testIt = m_Tests.begin(); testIt != m_Tests.end(); ++testIt)
{
if (!(*testIt).empty())
{
// for each arg in the test
fout << "ADD_TEST(";
it = (*testIt).begin();
fout << (*it).c_str();
++it;
for (; it != (*testIt).end(); ++it)
{
fout << " " << (*it).c_str();
}
fout << ")" << std::endl;
}
}
fout << std::endl;
2001-06-06 15:47:56 +04:00
fout.close();
}
2001-04-25 00:49:12 +04:00
void cmMakefile::AddSource(cmSourceFile& cmfile, const char *srclist)
{
2001-04-25 00:49:12 +04:00
m_Sources[srclist].push_back(cmfile);
}
void cmMakefile::AddCustomCommand(const char* source,
const char* command,
const std::vector<std::string>& depends,
const std::vector<std::string>& outputs,
const char *target)
{
// find the target,
if (m_Targets.find(target) != m_Targets.end())
{
cmCustomCommand cc(source,command,depends,outputs);
m_Targets[target].GetCustomCommands().push_back(cc);
}
}
2001-01-18 19:20:24 +03:00
void cmMakefile::AddCustomCommand(const char* source,
const char* command,
const std::vector<std::string>& depends,
const char* output,
const char *target)
{
std::vector<std::string> outputs;
outputs.push_back(output);
this->AddCustomCommand(source, command, depends, outputs, target);
}
void cmMakefile::AddDefineFlag(const char* flag)
{
m_DefineFlags += " ";
m_DefineFlags += flag;
}
void cmMakefile::AddUtility(const char* util)
{
m_Utilities.push_back(util);
}
void cmMakefile::AddUtilityDirectory(const char* dir)
{
m_UtilityDirectories.push_back(dir);
}
void cmMakefile::AddLinkLibrary(const char* lib, cmTarget::LinkLibraryType llt)
2001-04-27 00:22:53 +04:00
{
m_LinkLibraries.push_back(
std::pair<std::string, cmTarget::LinkLibraryType>(lib,llt));
}
void cmMakefile::AddLinkLibraryForTarget(const char *target,
const char* lib,
cmTarget::LinkLibraryType llt)
{
if (m_Targets.find(target) != m_Targets.end())
{
m_Targets[target].GetLinkLibraries().
push_back(
std::pair<std::string, cmTarget::LinkLibraryType>(lib,llt));
}
2001-04-27 00:22:53 +04:00
}
void cmMakefile::AddLinkLibrary(const char* lib)
{
this->AddLinkLibrary(lib,cmTarget::GENERAL);
}
void cmMakefile::AddLinkDirectory(const char* dir)
{
m_LinkDirectories.push_back(dir);
}
void cmMakefile::AddSubDirectory(const char* sub)
{
m_SubDirectories.push_back(sub);
}
void cmMakefile::AddIncludeDirectory(const char* inc)
{
m_IncludeDirectories.push_back(inc);
}
void cmMakefile::AddDefinition(const char* name, const char* value)
{
m_Definitions.erase( DefinitionMap::key_type(name));
m_Definitions.insert(DefinitionMap::value_type(name, value));
}
2001-04-24 20:40:37 +04:00
void cmMakefile::AddDefinition(const char* name, bool value)
{
if(value)
{
m_Definitions.erase( DefinitionMap::key_type(name));
2001-04-24 20:40:37 +04:00
m_Definitions.insert(DefinitionMap::value_type(name, "ON"));
}
else
{
m_Definitions.erase( DefinitionMap::key_type(name));
2001-04-24 20:40:37 +04:00
m_Definitions.insert(DefinitionMap::value_type(name, "OFF"));
}
}
void cmMakefile::AddTest(const std::vector<std::string> &args)
{
m_Tests.push_back(args);
}
void cmMakefile::SetProjectName(const char* p)
{
m_ProjectName = p;
}
void cmMakefile::AddLibrary(const char* lname, const std::vector<std::string> &srcs)
{
cmTarget target;
target.SetType(cmTarget::LIBRARY);
2001-05-05 01:00:22 +04:00
target.SetInAll(true);
target.GetSourceLists() = srcs;
m_Targets.insert(cmTargets::value_type(lname,target));
// Add an entry into the cache
cmCacheManager::GetInstance()->
AddCacheEntry(lname,
this->GetCurrentOutputDirectory(),
"Path to a library", cmCacheManager::INTERNAL);
}
void cmMakefile::AddExecutable(const char *exeName,
const std::vector<std::string> &srcs)
{
this->AddExecutable(exeName,srcs,false);
}
void cmMakefile::AddExecutable(const char *exeName,
const std::vector<std::string> &srcs,
bool win32)
{
cmTarget target;
if (win32)
{
target.SetType(cmTarget::WIN32_EXECUTABLE);
}
else
{
target.SetType(cmTarget::EXECUTABLE);
}
2001-05-05 01:00:22 +04:00
target.SetInAll(true);
target.GetSourceLists() = srcs;
m_Targets.insert(cmTargets::value_type(exeName,target));
// Add an entry into the cache
cmCacheManager::GetInstance()->
AddCacheEntry(exeName,
this->GetCurrentOutputDirectory(),
"Path to an executable", cmCacheManager::INTERNAL);
}
void cmMakefile::AddUtilityCommand(const char* utilityName,
const char* command,
bool all)
{
std::vector<std::string> empty;
this->AddUtilityCommand(utilityName,command,all,
empty,empty);
}
void cmMakefile::AddUtilityCommand(const char* utilityName,
const char* command,
bool all,
const std::vector<std::string> &dep,
const std::vector<std::string> &out)
{
cmTarget target;
target.SetType(cmTarget::UTILITY);
target.SetInAll(all);
cmCustomCommand cc(utilityName, command, dep, out);
target.GetCustomCommands().push_back(cc);
m_Targets.insert(cmTargets::value_type(utilityName,target));
}
void cmMakefile::AddSourceGroup(const char* name, const char* regex)
{
// First see if the group exists. If so, replace its regular expression.
for(std::vector<cmSourceGroup>::iterator sg = m_SourceGroups.begin();
sg != m_SourceGroups.end(); ++sg)
{
std::string sgName = sg->GetName();
if(sgName == name)
{
// We only want to set the regular expression. If there are already
// source files in the group, we don't want to remove them.
sg->SetGroupRegex(regex);
return;
}
}
// The group doesn't exist. Add it.
m_SourceGroups.push_back(cmSourceGroup(name, regex));
}
void cmMakefile::AddExtraDirectory(const char* dir)
{
m_AuxSourceDirectories.push_back(dir);
}
2001-02-15 21:30:13 +03:00
// return the file name for the parent CMakeLists file to the
// one passed in. Zero is returned if the CMakeLists file is the
// one in the home directory or if for some reason a parent cmake lists
// file cannot be found.
std::string cmMakefile::GetParentListFileName(const char *currentFileName)
{
2001-02-15 21:30:13 +03:00
// extract the directory name
std::string parentFile;
std::string listsDir = currentFileName;
std::string::size_type pos = listsDir.rfind('/');
// if we could not find the directory return 0
if(pos == std::string::npos)
{
2001-02-15 21:30:13 +03:00
return parentFile;
}
2001-02-15 21:30:13 +03:00
listsDir = listsDir.substr(0, pos);
// if we are in the home directory then stop, return 0
if(m_cmHomeDirectory == listsDir)
{
2001-02-15 21:30:13 +03:00
return parentFile;
}
2001-02-15 21:30:13 +03:00
// is there a parent directory we can check
pos = listsDir.rfind('/');
// if we could not find the directory return 0
if(pos == std::string::npos)
{
return parentFile;
}
listsDir = listsDir.substr(0, pos);
// is there a CMakeLists.txt file in the parent directory ?
parentFile = listsDir;
parentFile += "/CMakeLists.txt";
if(!cmSystemTools::FileExists(parentFile.c_str()))
{
2001-02-15 21:30:13 +03:00
parentFile = "";
return parentFile;
}
2001-02-15 21:30:13 +03:00
return parentFile;
}
// expance CMAKE_BINARY_DIR and CMAKE_SOURCE_DIR in the
// include and library directories.
void cmMakefile::ExpandVariables()
{
// Now expand varibles in the include and link strings
std::vector<std::string>::iterator j, begin, end;
begin = m_IncludeDirectories.begin();
end = m_IncludeDirectories.end();
for(j = begin; j != end; ++j)
{
this->ExpandVariablesInString(*j);
}
begin = m_LinkDirectories.begin();
end = m_LinkDirectories.end();
for(j = begin; j != end; ++j)
{
this->ExpandVariablesInString(*j);
}
cmTarget::LinkLibraries::iterator j2, end2;
2001-04-27 00:22:53 +04:00
j2 = m_LinkLibraries.begin();
end2 = m_LinkLibraries.end();
for(; j2 != end2; ++j2)
{
2001-04-27 00:22:53 +04:00
this->ExpandVariablesInString(j2->first);
}
}
const char* cmMakefile::GetDefinition(const char* name)
{
DefinitionMap::iterator pos = m_Definitions.find(name);
if(pos != m_Definitions.end())
{
return (*pos).second.c_str();
}
return 0;
}
int cmMakefile::DumpDocumentationToFile(const char *fileName)
{
// Open the supplied filename
std::ofstream f;
f.open(fileName, std::ios::out);
if ( f.fail() )
{
return 0;
}
2001-01-18 19:20:24 +03:00
// Loop over all registered commands and print out documentation
const char *name;
const char *terse;
const char *full;
2001-01-18 19:20:24 +03:00
for(RegisteredCommandsMap::iterator j = m_Commands.begin();
j != m_Commands.end(); ++j)
{
name = (*j).second->GetName();
2001-01-12 22:35:15 +03:00
terse = (*j).second->GetTerseDocumentation();
full = (*j).second->GetFullDocumentation();
f << name << " - " << terse << std::endl
2001-01-12 20:49:49 +03:00
<< "Usage: " << full << std::endl << std::endl;
}
return 1;
}
void cmMakefile::ExpandVariablesInString(std::string& source) const
{
for(DefinitionMap::const_iterator i = m_Definitions.begin();
i != m_Definitions.end(); ++i)
{
std::string variable = "${";
variable += (*i).first;
variable += "}";
cmSystemTools::ReplaceString(source, variable.c_str(),
(*i).second.c_str());
variable = "@";
variable += (*i).first;
variable += "@";
cmSystemTools::ReplaceString(source, variable.c_str(),
(*i).second.c_str());
}
}
void cmMakefile::RemoveVariablesInString(std::string& source) const
{
cmRegularExpression var("(\\${[A-Za-z_0-9]*})");
cmRegularExpression var2("(@[A-Za-z_0-9]*@)");
while (var.find(source))
{
source.erase(var.start(),var.end() - var.start());
}
while (var2.find(source))
{
source.erase(var2.start(),var2.end() - var2.start());
}
}
2001-02-23 03:24:43 +03:00
// recursive function to create a vector of cmMakefile objects
// This is done by reading the sub directory CMakeLists.txt files,
// then calling this function with the new cmMakefile object
void
cmMakefile::FindSubDirectoryCMakeListsFiles(std::vector<cmMakefile*>&
makefiles)
{
// loop over all the sub directories of this makefile
const std::vector<std::string>& subdirs = this->GetSubDirectories();
for(std::vector<std::string>::const_iterator i = subdirs.begin();
i != subdirs.end(); ++i)
{
std::string subdir = *i;
// Create a path to the list file in the sub directory
std::string listFile = this->GetCurrentDirectory();
listFile += "/";
listFile += subdir;
listFile += "/CMakeLists.txt";
// if there is a CMakeLists.txt file read it
if(!cmSystemTools::FileExists(listFile.c_str()))
{
cmSystemTools::Error("CMakeLists.txt file missing from sub directory:",
listFile.c_str());
}
else
{
cmMakefile* mf = new cmMakefile;
makefiles.push_back(mf);
// initialize new makefile
mf->SetHomeOutputDirectory(this->GetHomeOutputDirectory());
mf->SetHomeDirectory(this->GetHomeDirectory());
// add the subdir to the start output directory
std::string outdir = this->GetStartOutputDirectory();
outdir += "/";
outdir += subdir;
mf->SetStartOutputDirectory(outdir.c_str());
// add the subdir to the start source directory
std::string currentDir = this->GetStartDirectory();
currentDir += "/";
currentDir += subdir;
mf->SetStartDirectory(currentDir.c_str());
// Parse the CMakeLists.txt file
currentDir += "/CMakeLists.txt";
mf->MakeStartDirectoriesCurrent();
mf->ReadListFile(currentDir.c_str());
// recurse into nextDir
mf->FindSubDirectoryCMakeListsFiles(makefiles);
}
}
}
/**
* Add the default definitions to the makefile. These values must not
* be dependent on anything that isn't known when this cmMakefile instance
* is constructed.
*/
void cmMakefile::AddDefaultDefinitions()
{
#if defined(_WIN32) && !defined(__CYGWIN__)
this->AddDefinition("CMAKE_CFG_OUTDIR","$(OUTDIR)");
#else
this->AddDefinition("CMAKE_CFG_OUTDIR",".");
#endif
}
/**
* Find a source group whose regular expression matches the filename
* part of the given source name. Search backward through the list of
* source groups, and take the first matching group found. This way
* non-inherited SOURCE_GROUP commands will have precedence over
* inherited ones.
*/
cmSourceGroup&
cmMakefile::FindSourceGroup(const char* source,
std::vector<cmSourceGroup> &groups)
{
std::string file = source;
std::string::size_type pos = file.rfind('/');
if(pos != std::string::npos)
{
file = file.substr(pos, file.length()-pos);
}
for(std::vector<cmSourceGroup>::reverse_iterator sg = groups.rbegin();
sg != groups.rend(); ++sg)
{
if(sg->Matches(file.c_str()))
{
return *sg;
}
}
// Shouldn't get here, but just in case, return the default group.
return groups.front();
}
2001-04-20 01:39:03 +04:00
bool cmMakefile::IsFunctionBlocked(const char *name,
std::vector<std::string> &args) const
{
// loop over all function blockers to see if any block this command
std::set<cmFunctionBlocker *>::const_iterator pos;
for (pos = m_FunctionBlockers.begin();
pos != m_FunctionBlockers.end(); ++pos)
{
if ((*pos)->IsFunctionBlocked(name, args, *this))
{
return true;
}
}
return false;
}
void cmMakefile::RemoveFunctionBlocker(const char *name,
2001-06-04 18:18:03 +04:00
const std::vector<std::string> &args)
2001-04-20 01:39:03 +04:00
{
// loop over all function blockers to see if any block this command
std::set<cmFunctionBlocker *>::const_iterator pos;
for (pos = m_FunctionBlockers.begin();
pos != m_FunctionBlockers.end(); ++pos)
{
if ((*pos)->ShouldRemove(name, args, *this))
{
2001-06-04 18:18:03 +04:00
cmFunctionBlocker* b = *pos;
2001-04-20 01:39:03 +04:00
m_FunctionBlockers.erase(*pos);
2001-06-04 18:18:03 +04:00
delete b;
2001-04-20 01:39:03 +04:00
return;
}
}
return;
}
void cmMakefile::SetHomeDirectory(const char* dir)
{
m_cmHomeDirectory = dir;
cmSystemTools::ConvertToUnixSlashes(m_cmHomeDirectory);
this->AddDefinition("CMAKE_SOURCE_DIR", this->GetHomeDirectory());
}
void cmMakefile::SetHomeOutputDirectory(const char* lib)
{
m_HomeOutputDirectory = lib;
cmSystemTools::ConvertToUnixSlashes(m_HomeOutputDirectory);
this->AddDefinition("CMAKE_BINARY_DIR", this->GetHomeOutputDirectory());
}
/**
* Register the given cmData instance with its own name.
*/
void cmMakefile::RegisterData(cmData* data)
{
std::string name = data->GetName();
DataMap::const_iterator d = m_DataMap.find(name);
if((d != m_DataMap.end()) && (d->second != NULL) && (d->second != data))
{
delete d->second;
}
m_DataMap[name] = data;
}
/**
* Register the given cmData instance with the given name. This can be used
* to register a NULL pointer.
*/
void cmMakefile::RegisterData(const char* name, cmData* data)
{
DataMap::const_iterator d = m_DataMap.find(name);
if((d != m_DataMap.end()) && (d->second != NULL) && (d->second != data))
{
delete d->second;
}
m_DataMap[name] = data;
}
/**
* Lookup a cmData instance previously registered with the given name. If
* the instance cannot be found, return NULL.
*/
cmData* cmMakefile::LookupData(const char* name) const
{
DataMap::const_iterator d = m_DataMap.find(name);
if(d != m_DataMap.end())
{
return d->second;
}
else
{
return NULL;
}
}