summaryrefslogtreecommitdiffstats
path: root/Source/cmGlobalXCodeGenerator.cxx
diff options
context:
space:
mode:
authorBrad King <brad.king@kitware.com>2009-07-10 20:51:44 (GMT)
committerBrad King <brad.king@kitware.com>2009-07-10 20:51:44 (GMT)
commita232dbe8d23729e4aeeb3e27fe0a807903e82a62 (patch)
treecb0e83679ce19a02b87c5b1a4e2857cb9206cde4 /Source/cmGlobalXCodeGenerator.cxx
parentbb2228ea1a2596c7068620a094e2ded3b4e5c8df (diff)
downloadCMake-a232dbe8d23729e4aeeb3e27fe0a807903e82a62.zip
CMake-a232dbe8d23729e4aeeb3e27fe0a807903e82a62.tar.gz
CMake-a232dbe8d23729e4aeeb3e27fe0a807903e82a62.tar.bz2
BUG: Fix Xcode linker language
Xcode does not seem to support direct requests for using the linker for a particular language. It always infers the linker using the languages in the source files. When no user source files compile with target's linker language we add one to help Xcode pick the linker. A typical use case is when a C executable links to a C++ archive. The executable has no C++ source files but we need to use the C++ linker.
Diffstat (limited to 'Source/cmGlobalXCodeGenerator.cxx')
-rw-r--r--Source/cmGlobalXCodeGenerator.cxx57
1 files changed, 57 insertions, 0 deletions
diff --git a/Source/cmGlobalXCodeGenerator.cxx b/Source/cmGlobalXCodeGenerator.cxx
index 5f22946..353fa6e 100644
--- a/Source/cmGlobalXCodeGenerator.cxx
+++ b/Source/cmGlobalXCodeGenerator.cxx
@@ -239,6 +239,7 @@ void cmGlobalXCodeGenerator::Generate()
// add ALL_BUILD, INSTALL, etc
this->AddExtraTargets(root, it->second);
}
+ this->ForceLinkerLanguages();
this->cmGlobalGenerator::Generate();
for(it = this->ProjectMap.begin(); it!= this->ProjectMap.end(); ++it)
{
@@ -969,6 +970,62 @@ cmGlobalXCodeGenerator::CreateXCodeTargets(cmLocalGenerator* gen,
}
//----------------------------------------------------------------------------
+void cmGlobalXCodeGenerator::ForceLinkerLanguages()
+{
+ // This makes sure all targets link using the proper language.
+ for(std::map<cmStdString, cmTarget*>::const_iterator
+ ti = this->TotalTargets.begin(); ti != this->TotalTargets.end(); ++ti)
+ {
+ this->ForceLinkerLanguage(*ti->second);
+ }
+}
+
+//----------------------------------------------------------------------------
+void cmGlobalXCodeGenerator::ForceLinkerLanguage(cmTarget& cmtarget)
+{
+ // This matters only for targets that link.
+ if(cmtarget.GetType() != cmTarget::EXECUTABLE &&
+ cmtarget.GetType() != cmTarget::SHARED_LIBRARY &&
+ cmtarget.GetType() != cmTarget::MODULE_LIBRARY)
+ {
+ return;
+ }
+
+ const char* llang = cmtarget.GetLinkerLanguage("NOCONFIG");
+ if(!llang) { return; }
+
+ // If the language is compiled as a source trust Xcode to link with it.
+ cmTarget::LinkImplementation const* impl =
+ cmtarget.GetLinkImplementation("NOCONFIG");
+ for(std::vector<std::string>::const_iterator li = impl->Languages.begin();
+ li != impl->Languages.end(); ++li)
+ {
+ if(*li == llang) { return; }
+ }
+
+ // Add an empty source file to the target that compiles with the
+ // linker language. This should convince Xcode to choose the proper
+ // language.
+ cmMakefile* mf = cmtarget.GetMakefile();
+ std::string fname = mf->GetCurrentOutputDirectory();
+ fname += cmake::GetCMakeFilesDirectory();
+ fname += "/";
+ fname += cmtarget.GetName();
+ fname += "-CMakeForceLinker";
+ fname += ".";
+ fname += cmSystemTools::LowerCase(llang);
+ {
+ cmGeneratedFileStream fout(fname.c_str());
+ fout << "\n";
+ }
+ if(cmSourceFile* sf = mf->GetOrCreateSource(fname.c_str()))
+ {
+ sf->SetProperty("LANGUAGE", llang);
+ cmtarget.AddSourceFile(sf);
+ }
+}
+
+//----------------------------------------------------------------------------
bool cmGlobalXCodeGenerator::IsHeaderFile(cmSourceFile* sf)
{
const std::vector<std::string>& hdrExts =