summaryrefslogtreecommitdiffstats
path: root/Source/cmGlobalXCodeGenerator.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'Source/cmGlobalXCodeGenerator.cxx')
-rw-r--r--Source/cmGlobalXCodeGenerator.cxx318
1 files changed, 232 insertions, 86 deletions
diff --git a/Source/cmGlobalXCodeGenerator.cxx b/Source/cmGlobalXCodeGenerator.cxx
index c4e765c..7b6099e 100644
--- a/Source/cmGlobalXCodeGenerator.cxx
+++ b/Source/cmGlobalXCodeGenerator.cxx
@@ -24,9 +24,6 @@
//TODO
-// custom commands
-// custom targets
-// ALL_BUILD
// RUN_TESTS
// add a group for Sources Headers, and other cmake group stuff
@@ -195,23 +192,49 @@ void cmGlobalXCodeGenerator::Generate()
std::map<cmStdString, std::vector<cmLocalGenerator*> >::iterator it;
for(it = m_ProjectMap.begin(); it!= m_ProjectMap.end(); ++it)
{
+ cmLocalGenerator* root = it->second[0];
+ cmMakefile* mf = root->GetMakefile();
// add a ALL_BUILD target to the first makefile of each project
- it->second[0]->GetMakefile()->
- AddUtilityCommand("ALL_BUILD", "echo",
+ mf->AddUtilityCommand("ALL_BUILD", "echo",
"\"Build all projects\"",false,srcs);
- cmTarget* allbuild =
- it->second[0]->GetMakefile()->FindTarget("ALL_BUILD");
+ cmTarget* allbuild = mf->FindTarget("ALL_BUILD");
+
+ std::string dir = root->ConvertToRelativeOutputPath(
+ mf->GetCurrentOutputDirectory());
+ m_CurrentXCodeHackMakefile = dir;
+ m_CurrentXCodeHackMakefile += "/XCODE_DEPEND_HELPER.make";
+ std::string makecommand = "make -C ";
+ makecommand += dir;
+ makecommand += " -f ";
+ makecommand += m_CurrentXCodeHackMakefile;
+ mf->AddUtilityCommand("XCODE_DEPEND_HELPER", makecommand.c_str(),
+ "",
+ false,srcs);
+ cmTarget* xcodedephack = mf->FindTarget("XCODE_DEPEND_HELPER");
// now make the allbuild depend on all the non-utility targets
// in the project
for(std::vector<cmLocalGenerator*>::iterator i = it->second.begin();
i != it->second.end(); ++i)
{
cmLocalGenerator* lg = *i;
+ if(this->IsExcluded(root, *i))
+ {
+ continue;
+ }
cmTargets& tgts = lg->GetMakefile()->GetTargets();
for(cmTargets::iterator l = tgts.begin(); l != tgts.end(); l++)
{
cmTarget& target = l->second;
- if(target.GetType() < cmTarget::UTILITY)
+ // make all exe, shared libs and modules depend
+ // on the XCODE_DEPEND_HELPER target
+ if((target.GetType() == cmTarget::EXECUTABLE ||
+ target.GetType() == cmTarget::SHARED_LIBRARY ||
+ target.GetType() == cmTarget::MODULE_LIBRARY))
+ {
+ target.AddUtility(xcodedephack->GetName());
+ }
+
+ if(target.IsInAll())
{
allbuild->AddUtility(target.GetName());
}
@@ -291,11 +314,14 @@ cmGlobalXCodeGenerator::CreateXCodeSourceFile(cmLocalGenerator* lg,
sourcecode += ext;
sourcecode += ext;
}
+ else if(sf->GetSourceExtension() == "mm")
+ {
+ sourcecode += ".cpp.objcpp";
+ }
else if(strcmp(lang, "C") == 0)
{
sourcecode += ".c.c";
}
- // default to c++
else
{
sourcecode += ".cpp.cpp";
@@ -331,6 +357,15 @@ cmGlobalXCodeGenerator::CreateXCodeTargets(cmLocalGenerator* gen,
}
m_DoneAllBuild = true;
}
+ // make sure ALL_BUILD is only done once
+ if(l->first == "XCODE_DEPEND_HELPER")
+ {
+ if(m_DoneXCodeHack)
+ {
+ continue;
+ }
+ m_DoneXCodeHack = true;
+ }
if(cmtarget.GetType() == cmTarget::UTILITY ||
cmtarget.GetType() == cmTarget::INSTALL_FILES ||
cmtarget.GetType() == cmTarget::INSTALL_PROGRAMS)
@@ -542,6 +577,8 @@ cmGlobalXCodeGenerator::AddCommandsToBuildPhase(cmXCodeObject* buildphase,
// have all depend on all outputs
makefileStream << "all: ";
+ std::map<const cmCustomCommand*, cmStdString> tname;
+ int count = 0;
for(std::vector<cmCustomCommand>::const_iterator i = commands.begin();
i != commands.end(); ++i)
{
@@ -555,7 +592,9 @@ cmGlobalXCodeGenerator::AddCommandsToBuildPhase(cmXCodeObject* buildphase,
}
else
{
- makefileStream << "\\\n\t" << target.GetName();
+ char c = '1' + count++;
+ tname[&cc] = std::string(target.GetName()) + c;
+ makefileStream << "\\\n\t" << tname[&cc];
}
}
}
@@ -577,9 +616,7 @@ cmGlobalXCodeGenerator::AddCommandsToBuildPhase(cmXCodeObject* buildphase,
}
else
{
- // if no outputs then use the target name as this must
- // be a utility target
- makefileStream << target.GetName() << ": ";
+ makefileStream << tname[&cc] << ": ";
}
for(std::vector<std::string>::const_iterator d = cc.GetDepends().begin();
d != cc.GetDepends().end(); ++d)
@@ -918,6 +955,23 @@ cmXCodeObject* cmGlobalXCodeGenerator::FindXCodeTarget(cmTarget* t)
void cmGlobalXCodeGenerator::AddDependTarget(cmXCodeObject* target,
cmXCodeObject* dependTarget)
{
+ // make sure a target does not depend on itself
+ if(target == dependTarget)
+ {
+ return;
+ }
+ // now avoid circular references if dependTarget already
+ // depends on target then skip it. Circular references crashes
+ // xcode
+ cmXCodeObject* dependTargetDepends = dependTarget->GetObject("dependencies");
+ if(dependTargetDepends)
+ {
+ if(dependTargetDepends->HasObject(target->GetPBXTargetDependency()))
+ {
+ return;
+ }
+ }
+
cmXCodeObject* targetdep = dependTarget->GetPBXTargetDependency();
if(!targetdep)
{
@@ -939,7 +993,7 @@ void cmGlobalXCodeGenerator::AddDependTarget(cmXCodeObject* target,
this->CreateObjectReference(container));
dependTarget->SetPBXTargetDependency(targetdep);
}
-
+
cmXCodeObject* depends = target->GetObject("dependencies");
if(!depends)
{
@@ -947,46 +1001,28 @@ void cmGlobalXCodeGenerator::AddDependTarget(cmXCodeObject* target,
}
else
{
- depends->AddObject(targetdep);
+ depends->AddUniqueObject(targetdep);
}
}
-void cmGlobalXCodeGenerator::AddLinkTarget(cmXCodeObject* target ,
- cmXCodeObject* dependTarget )
-{
- cmXCodeObject* ref = dependTarget->GetObject("productReference");
-
- cmXCodeObject* buildfile = this->CreateObject(cmXCodeObject::PBXBuildFile);
- buildfile->AddAttribute("fileRef", ref);
- cmXCodeObject* settings =
- this->CreateObject(cmXCodeObject::ATTRIBUTE_GROUP);
- buildfile->AddAttribute("settings", settings);
-
- cmXCodeObject* buildPhases = target->GetObject("buildPhases");
- cmXCodeObject* frameworkBuildPhase =
- buildPhases->GetObject(cmXCodeObject::PBXFrameworksBuildPhase);
- cmXCodeObject* files = frameworkBuildPhase->GetObject("files");
- files->AddObject(buildfile);
-}
void cmGlobalXCodeGenerator::AddLinkLibrary(cmXCodeObject* target,
- const char* library)
+ const char* library,
+ cmTarget* dtarget)
{
// if the library is a full path then create a file reference
// and build file and add them to the PBXFrameworksBuildPhase
// for the target
+ std::string file = library;
if(cmSystemTools::FileIsFullPath(library))
{
- std::string libPath = library;
- cmXCodeObject* fileRef =
- this->CreateObject(cmXCodeObject::PBXFileReference);
+ target->AddDependLibrary(library);
+ std::string dir;
+ cmSystemTools::SplitProgramPath(library, dir, file);
// add the library path to the search path for the target
cmXCodeObject* bset = target->GetObject("buildSettings");
if(bset)
{
- std::string dir;
- std::string file;
- cmSystemTools::SplitProgramPath(library, dir, file);
cmXCodeObject* spath = bset->GetObject("LIBRARY_SEARCH_PATHS");
if(spath)
{
@@ -1006,58 +1042,67 @@ void cmGlobalXCodeGenerator::AddLinkLibrary(cmXCodeObject* target,
this->CreateString(libs.c_str()));
}
}
-
- if(libPath[libPath.size()-1] == 'a')
- {
- fileRef->AddAttribute("lastKnownFileType",
- this->CreateString("archive.ar"));
- }
- else
+ cmsys::RegularExpression libname("^lib([^/]*)(\\.a|\\.dylib).*");
+ if(libname.find(file))
{
- fileRef->AddAttribute("lastKnownFileType",
- this->CreateString("compiled.mach-o.dylib"));
+ file = libname.match(1);
}
- fileRef->AddAttribute("name",
- this->CreateString(
- cmSystemTools::GetFilenameName(libPath).c_str()));
-
- fileRef->AddAttribute("path", this->CreateString(libPath.c_str()));
- fileRef->AddAttribute("refType", this->CreateString("0"));
- fileRef->AddAttribute("sourceTree", this->CreateString("<absolute>"));
- cmXCodeObject* buildfile = this->CreateObject(cmXCodeObject::PBXBuildFile);
- buildfile->AddAttribute("fileRef", this->CreateObjectReference(fileRef));
- cmXCodeObject* settings =
- this->CreateObject(cmXCodeObject::ATTRIBUTE_GROUP);
- buildfile->AddAttribute("settings", settings);
- // get the framework build phase from the target
- cmXCodeObject* buildPhases = target->GetObject("buildPhases");
- cmXCodeObject* frameworkBuildPhase =
- buildPhases->GetObject(cmXCodeObject::PBXFrameworksBuildPhase);
- cmXCodeObject* files = frameworkBuildPhase->GetObject("files");
- files->AddObject(buildfile);
- m_ExternalGroupChildren->AddObject(fileRef);
}
else
{
- // if the library is not a full path then add it with a -l flag
- // to the settings of the target
- cmXCodeObject* settings = target->GetObject("buildSettings");
- cmXCodeObject* ldflags = settings->GetObject("OTHER_LDFLAGS");
- std::string link = ldflags->GetString();
- cmSystemTools::ReplaceString(link, "\"", "");
- cmsys::RegularExpression reg("^([ \t]*\\-[lWRB])|([ \t]*\\-framework)|(\\${)|([ \t]*\\-pthread)|([ \t]*`)");
- // if the library is not already in the form required by the compiler
- // add a -l infront of the name
- link += " ";
- if(!reg.find(library))
+ if(dtarget)
{
- link += "-l";
+ target->AddDependLibrary(this->GetTargetFullPath(dtarget).c_str());
}
- link += library;
- ldflags->SetString(link.c_str());
}
+
+ // if the library is not a full path then add it with a -l flag
+ // to the settings of the target
+ cmXCodeObject* settings = target->GetObject("buildSettings");
+ cmXCodeObject* ldflags = settings->GetObject("OTHER_LDFLAGS");
+ std::string link = ldflags->GetString();
+ cmSystemTools::ReplaceString(link, "\"", "");
+ cmsys::RegularExpression reg("^([ \t]*\\-[lWRB])|([ \t]*\\-framework)|(\\${)|([ \t]*\\-pthread)|([ \t]*`)");
+ // if the library is not already in the form required by the compiler
+ // add a -l infront of the name
+ link += " ";
+ if(!reg.find(file))
+ {
+ link += "-l";
+ }
+ link += file;
+ ldflags->SetString(link.c_str());
}
+
+std::string cmGlobalXCodeGenerator::GetTargetFullPath(cmTarget* target)
+{
+ std::string libPath;
+ cmXCodeObject* xtarget = this->FindXCodeTarget(target);
+ cmXCodeObject* bset = xtarget->GetObject("buildSettings");
+ cmXCodeObject* spath = bset->GetObject("SYMROOT");
+ libPath = spath->GetString();
+ libPath = libPath.substr(1, libPath.size()-2);
+ if(target->GetType() == cmTarget::STATIC_LIBRARY)
+ {
+ libPath += "lib";
+ libPath += target->GetName();
+ libPath += ".a";
+ }
+ else if(target->GetType() == cmTarget::SHARED_LIBRARY)
+ {
+ libPath += "lib";
+ libPath += target->GetName();
+ libPath += ".dylib";
+ }
+ else
+ {
+ libPath += target->GetName();
+ }
+ return libPath;
+}
+
+
void cmGlobalXCodeGenerator::AddDependAndLinkInformation(cmXCodeObject* target)
{
cmTarget* cmtarget = target->GetcmTarget();
@@ -1079,7 +1124,7 @@ void cmGlobalXCodeGenerator::AddDependAndLinkInformation(cmXCodeObject* target)
this->AddDependTarget(target, dptarget);
if(cmtarget->GetType() != cmTarget::STATIC_LIBRARY)
{
- this->AddLinkTarget(target, dptarget);
+ this->AddLinkLibrary(target, t->GetName(), t);
}
}
else
@@ -1104,7 +1149,11 @@ void cmGlobalXCodeGenerator::AddDependAndLinkInformation(cmXCodeObject* target)
}
else
{
- std::cerr << "Error External Util???: " << i->c_str() << "\n";
+ std::cerr << "Error Utility: " << i->c_str() << "\n";
+ std::cerr << "Is on the target " << cmtarget->GetName() << "\n";
+ std::cerr << "But it has no xcode target created yet??\n";
+ std::cerr << "Current project is "
+ << m_CurrentMakefile->GetProjectName() << "\n";
}
}
}
@@ -1113,7 +1162,7 @@ void cmGlobalXCodeGenerator::AddDependAndLinkInformation(cmXCodeObject* target)
// add this to build settings of target SYMROOT = /tmp/;
//----------------------------------------------------------------------------
-void cmGlobalXCodeGenerator::CreateXCodeObjects(cmLocalGenerator* ,
+void cmGlobalXCodeGenerator::CreateXCodeObjects(cmLocalGenerator* root,
std::vector<cmLocalGenerator*>&
generators
)
@@ -1187,17 +1236,30 @@ void cmGlobalXCodeGenerator::CreateXCodeObjects(cmLocalGenerator* ,
this->CreateString("0"));
std::vector<cmXCodeObject*> targets;
m_DoneAllBuild = false;
+ m_DoneXCodeHack = false;
for(std::vector<cmLocalGenerator*>::iterator i = generators.begin();
i != generators.end(); ++i)
{
- this->CreateXCodeTargets(*i, targets);
+ if(!this->IsExcluded(root, *i))
+ {
+ this->CreateXCodeTargets(*i, targets);
+ }
}
- cmXCodeObject* allTargets = this->CreateObject(cmXCodeObject::OBJECT_LIST);
+ // loop over all targets and add link and depend info
for(std::vector<cmXCodeObject*>::iterator i = targets.begin();
i != targets.end(); ++i)
{
cmXCodeObject* t = *i;
this->AddDependAndLinkInformation(t);
+ }
+ // now create xcode depend hack makefile
+ this->CreateXCodeDependHackTarget(targets);
+ // now add all targets to the root object
+ cmXCodeObject* allTargets = this->CreateObject(cmXCodeObject::OBJECT_LIST);
+ for(std::vector<cmXCodeObject*>::iterator i = targets.begin();
+ i != targets.end(); ++i)
+ {
+ cmXCodeObject* t = *i;
allTargets->AddObject(t);
cmXCodeObject* productRef = t->GetObject("productReference");
if(productRef)
@@ -1208,6 +1270,89 @@ void cmGlobalXCodeGenerator::CreateXCodeObjects(cmLocalGenerator* ,
m_RootObject->AddAttribute("targets", allTargets);
}
+
+//----------------------------------------------------------------------------
+void
+cmGlobalXCodeGenerator::CreateXCodeDependHackTarget(
+ std::vector<cmXCodeObject*>& targets)
+{
+ cmGeneratedFileStream makefileStream(m_CurrentXCodeHackMakefile.c_str());
+ if(!makefileStream)
+ {
+ cmSystemTools::Error("Could not create",
+ m_CurrentXCodeHackMakefile.c_str());
+ return;
+ }
+
+ // one more pass for external depend information not handled
+ // correctly by xcode
+ makefileStream << "# DO NOT EDIT\n";
+ makefileStream << "# This makefile makes sure all linkable targets are \n";
+ makefileStream
+ << "# up-to-date with anything they link to,avoiding a bug in XCode 1.5\n";
+ makefileStream << "all: ";
+ for(std::vector<cmXCodeObject*>::iterator i = targets.begin();
+ i != targets.end(); ++i)
+ {
+ cmXCodeObject* target = *i;
+ cmTarget* t =target->GetcmTarget();
+ if(t->GetType() == cmTarget::EXECUTABLE ||
+ t->GetType() == cmTarget::SHARED_LIBRARY ||
+ t->GetType() == cmTarget::MODULE_LIBRARY)
+ {
+ makefileStream << "\\\n\t"
+ << this->GetTargetFullPath(target->GetcmTarget());
+ }
+ }
+ makefileStream << "\n\n";
+ makefileStream
+ << "# For each target create a dummy rule "
+ "so the target does not have to exist\n";
+ std::set<cmStdString> emitted;
+ for(std::vector<cmXCodeObject*>::iterator i = targets.begin();
+ i != targets.end(); ++i)
+ {
+ cmXCodeObject* target = *i;
+ std::vector<cmStdString> const& deplibs = target->GetDependLibraries();
+ for(std::vector<cmStdString>::const_iterator d = deplibs.begin();
+ d != deplibs.end(); ++d)
+ {
+ if(emitted.insert(*d).second)
+ {
+ makefileStream << *d << ":\n";
+ }
+ }
+ }
+ makefileStream << "\n\n";
+ makefileStream <<
+ "# Each linkable target depends on everything it links to.\n";
+ makefileStream
+ << "#And the target is removed if it is older than what it linkes to\n";
+
+ for(std::vector<cmXCodeObject*>::iterator i = targets.begin();
+ i != targets.end(); ++i)
+ {
+ cmXCodeObject* target = *i;
+ cmTarget* t =target->GetcmTarget();
+ if(t->GetType() == cmTarget::EXECUTABLE ||
+ t->GetType() == cmTarget::SHARED_LIBRARY ||
+ t->GetType() == cmTarget::MODULE_LIBRARY)
+ {
+ std::vector<cmStdString> const& deplibs = target->GetDependLibraries();
+ std::string tfull = this->GetTargetFullPath(target->GetcmTarget());
+ makefileStream << tfull << ": ";
+ for(std::vector<cmStdString>::const_iterator d = deplibs.begin();
+ d != deplibs.end(); ++d)
+ {
+ makefileStream << "\\\n\t" << *d;
+ }
+ makefileStream << "\n";
+ makefileStream << "\t/bin/rm -f " << tfull << "\n";
+ makefileStream << "\n\n";
+ }
+ }
+}
+
//----------------------------------------------------------------------------
void
cmGlobalXCodeGenerator::OutputXCodeProject(cmLocalGenerator* root,
@@ -1218,6 +1363,7 @@ cmGlobalXCodeGenerator::OutputXCodeProject(cmLocalGenerator* root,
{
return;
}
+
this->CreateXCodeObjects(root,
generators);
std::string xcodeDir = root->GetMakefile()->GetStartOutputDirectory();