cmFindFileCommand.cxx 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. /*=========================================================================
  2. Program: CMake - Cross-Platform Makefile Generator
  3. Module: $RCSfile$
  4. Language: C++
  5. Date: $Date$
  6. Version: $Revision$
  7. Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved.
  8. See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details.
  9. This software is distributed WITHOUT ANY WARRANTY; without even
  10. the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
  11. PURPOSE. See the above copyright notices for more information.
  12. =========================================================================*/
  13. #include "cmFindFileCommand.h"
  14. #include "cmCacheManager.h"
  15. #include "cmGlob.h"
  16. #include <stdlib.h>
  17. // cmFindFileCommand
  18. bool cmFindFileCommand::InitialPass(std::vector<std::string> const& argsIn)
  19. {
  20. if(argsIn.size() < 2)
  21. {
  22. this->SetError("called with incorrect number of arguments");
  23. return false;
  24. }
  25. std::string helpString = "Where can the ";
  26. helpString += argsIn[1] + " file be found";
  27. size_t size = argsIn.size();
  28. std::vector<std::string> args;
  29. for(unsigned int j = 0; j < size; ++j)
  30. {
  31. if(argsIn[j] != "DOC")
  32. {
  33. args.push_back(argsIn[j]);
  34. }
  35. else
  36. {
  37. if(j+1 < size)
  38. {
  39. helpString = argsIn[j+1];
  40. }
  41. break;
  42. }
  43. }
  44. std::vector<std::string>::const_iterator i = args.begin();
  45. // Use the first argument as the name of something to be defined
  46. const char* define = (*i).c_str();
  47. i++; // move iterator to next arg
  48. // Now check and see if the value has been stored in the cache
  49. // already, if so use that value and don't look for the program
  50. const char* cacheValue
  51. = m_Makefile->GetDefinition(define);
  52. if(cacheValue && !cmSystemTools::IsNOTFOUND(cacheValue))
  53. {
  54. return true;
  55. }
  56. // if it is not in the cache, then search the system path
  57. std::vector<std::string> path;
  58. // add any user specified paths
  59. for (unsigned int j = 2; j < args.size(); j++)
  60. {
  61. // Glob the entry in case of wildcards.
  62. cmSystemTools::GlobDirs(args[j].c_str(), path);
  63. }
  64. // add the standard path
  65. cmSystemTools::GetPath(path);
  66. for(unsigned int k=0; k < path.size(); k++)
  67. {
  68. std::string tryPath = path[k];
  69. tryPath += "/";
  70. tryPath += *i;
  71. if(cmSystemTools::FileExists(tryPath.c_str()))
  72. {
  73. // Save the value in the cache
  74. m_Makefile->AddCacheDefinition(define,
  75. tryPath.c_str(),
  76. helpString.c_str(),
  77. cmCacheManager::FILEPATH);
  78. return true;
  79. }
  80. }
  81. #if defined (__APPLE__)
  82. cmStdString fpath = this->FindHeaderInFrameworks(args[0].c_str(), args[1].c_str());
  83. if(fpath.size())
  84. {
  85. m_Makefile->AddCacheDefinition(args[0].c_str(),
  86. fpath.c_str(),
  87. helpString.c_str(),
  88. cmCacheManager::FILEPATH);
  89. return true;
  90. }
  91. #endif
  92. std::string s = args[0] + "-NOTFOUND";
  93. m_Makefile->AddCacheDefinition(args[0].c_str(),
  94. s.c_str(),
  95. helpString.c_str(),
  96. cmCacheManager::FILEPATH);
  97. return true;
  98. }
  99. cmStdString cmFindFileCommand::FindHeaderInFrameworks(const char* defineVar,
  100. const char* file)
  101. {
  102. #ifndef __APPLE__
  103. return cmStdString("");
  104. #else
  105. cmStdString fileName = file;
  106. cmStdString frameWorkName;
  107. cmStdString::size_type pos = fileName.find("/");
  108. std::cerr << "ff " << fileName << " " << pos << "\n";
  109. if(pos != fileName.npos)
  110. {
  111. // remove the name from the slash;
  112. fileName = fileName.substr(pos+1);
  113. frameWorkName = file;
  114. frameWorkName = frameWorkName.substr(0, frameWorkName.size()-fileName.size()-1);
  115. // if the framework has a path in it then just use the filename
  116. std::cerr << fileName << " " << frameWorkName << "\n";
  117. if(frameWorkName.find("/") != frameWorkName.npos)
  118. {
  119. fileName = file;
  120. frameWorkName = "";
  121. }
  122. }
  123. std::vector<cmStdString> path;
  124. path.push_back("~/Library/Frameworks");
  125. path.push_back("/Library/Frameworks");
  126. path.push_back("/System/Library/Frameworks");
  127. path.push_back("/Network/Library/Frameworks");
  128. for( std::vector<cmStdString>::iterator i = path.begin();
  129. i != path.end(); ++i)
  130. {
  131. if(frameWorkName.size())
  132. {
  133. std::string fpath = *i;
  134. fpath += "/";
  135. fpath += frameWorkName;
  136. fpath += ".framework";
  137. std::string path = fpath;
  138. path += "/Headers/";
  139. path += fileName;
  140. std::cerr << "try " << path << "\n";
  141. if(cmSystemTools::FileExists(path.c_str()))
  142. {
  143. return fpath;
  144. }
  145. }
  146. cmStdString glob = *i;
  147. glob += "/*/Headers/";
  148. glob += file;
  149. cmGlob globIt;
  150. globIt.FindFiles(glob);
  151. std::vector<std::string> files = globIt.GetFiles();
  152. if(files.size())
  153. {
  154. cmStdString fheader = cmSystemTools::CollapseFullPath(files[0].c_str());
  155. fheader = cmSystemTools::GetFilenamePath(fheader);
  156. return fheader;
  157. }
  158. }
  159. return cmStdString("");
  160. #endif
  161. }