From 7adb808244669555135f87a728c983d5e382454e Mon Sep 17 00:00:00 2001 From: Brad King Date: Wed, 23 Jan 2008 16:21:49 -0500 Subject: ENH: Teach find_library to avoid returning library paths in system directories that may be converted to architecture-specific directories by the compiler when it invokes the linker. --- Source/cmFindLibraryCommand.cxx | 54 ++++++++++++++++++++++++++++++++++++++++- Source/cmFindLibraryCommand.h | 1 + 2 files changed, 54 insertions(+), 1 deletion(-) diff --git a/Source/cmFindLibraryCommand.cxx b/Source/cmFindLibraryCommand.cxx index f1bce09..2938e2e 100644 --- a/Source/cmFindLibraryCommand.cxx +++ b/Source/cmFindLibraryCommand.cxx @@ -52,6 +52,17 @@ cmFindLibraryCommand::cmFindLibraryCommand() "When a full path to a framework is used as a library, " "CMake will use a -framework A, and a -F to " "link the framework to the target. "; + this->GenericDocumentation += + "\n" + "Some platforms define implicit library directories such as " + "/lib and /usr/lib that are automatically searched by the linker. " + "If this command finds a library in one of these directories " + "it will report only the name of the library file and not the path. " + "When the name is used to link the library CMake will generate a " + "link line that asks the linker to search for it. This allows " + "the system linker to automatically adjust the implicit directory " + "set based on the current architecture." + ; } // cmFindLibraryCommand @@ -107,7 +118,8 @@ bool cmFindLibraryCommand { library = this->FindLibrary(i->c_str()); if(library != "") - { + { + library = this->FixForImplicitLocations(library); this->Makefile->AddCacheDefinition(this->VariableName.c_str(), library.c_str(), this->VariableDocumentation.c_str(), @@ -293,3 +305,43 @@ std::string cmFindLibraryCommand::FindLibrary(const char* name) // Couldn't find the library. return ""; } + +//---------------------------------------------------------------------------- +std::string +cmFindLibraryCommand::FixForImplicitLocations(std::string const& lib) +{ + // Get implicit link directories for the platform. + const char* implicitLinks = + (this->Makefile->GetDefinition + ("CMAKE_PLATFORM_IMPLICIT_LINK_DIRECTORIES")); + if(!implicitLinks) + { + // There are no implicit link directories. No fix is needed. + return lib; + } + std::vector implicitLinkVec; + cmSystemTools::ExpandListArgument(implicitLinks, implicitLinkVec); + + // Get the path containing the library. + std::string libDir = cmSystemTools::GetFilenamePath(lib); + + // Many system linkers support multiple architectures by + // automatically selecting the implicit linker search path for the + // current architecture. If the library appears in an implicit link + // directory then just report the file name without the directory + // portion. This will allow the system linker to locate the proper + // library for the architecture at link time. + for(std::vector::const_iterator i = implicitLinkVec.begin(); + i != implicitLinkVec.end(); ++i) + { + if(*i == libDir) + { + // The library appears in an implicit link directory. Report + // only the file name. + return cmSystemTools::GetFilenameName(lib); + } + } + + // No implicit link directory matched. No fix is needed. + return lib; +} diff --git a/Source/cmFindLibraryCommand.h b/Source/cmFindLibraryCommand.h index 233f766..e29dae2 100644 --- a/Source/cmFindLibraryCommand.h +++ b/Source/cmFindLibraryCommand.h @@ -69,6 +69,7 @@ protected: void AddArchitecturePaths(const char* suffix); void AddLib64Paths(); std::string FindLibrary(const char* name); + std::string FixForImplicitLocations(std::string const& lib); }; -- cgit v0.12