/*============================================================================ CMake - Cross Platform Makefile Generator Copyright 2004-2009 Kitware, Inc. Copyright 2004 Alexander Neundorf (neundorf@kde.org) Distributed under the OSI-approved BSD License (the "License"); see accompanying file Copyright.txt for details. This software is distributed WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the License for more information. ============================================================================*/ #include "cmGlobalKdevelopGenerator.h" #include "cmGlobalUnixMakefileGenerator3.h" #include "cmLocalUnixMakefileGenerator3.h" #include "cmMakefile.h" #include "cmake.h" #include "cmSourceFile.h" #include "cmGeneratedFileStream.h" #include "cmSystemTools.h" #include #include //---------------------------------------------------------------------------- void cmGlobalKdevelopGenerator ::GetDocumentation(cmDocumentationEntry& entry, const char*) const { entry.Name = this->GetName(); entry.Brief = "Generates KDevelop 3 project files."; entry.Full = "Project files for KDevelop 3 will be created in the top directory " "and in every subdirectory which features a CMakeLists.txt file " "containing a PROJECT() call. " "If you change the settings using KDevelop cmake will try its best " "to keep your changes when regenerating the project files. " "Additionally a hierarchy of UNIX makefiles is generated into the " "build tree. Any " "standard UNIX-style make program can build the project through the " "default make target. A \"make install\" target is also provided."; } cmGlobalKdevelopGenerator::cmGlobalKdevelopGenerator() :cmExternalMakefileProjectGenerator() { this->SupportedGlobalGenerators.push_back("Unix Makefiles"); #ifdef CMAKE_USE_NINJA this->SupportedGlobalGenerators.push_back("Ninja"); #endif } void cmGlobalKdevelopGenerator::Generate() { // for each sub project in the project create // a kdevelop project for (std::map >::const_iterator it = this->GlobalGenerator->GetProjectMap().begin(); it!= this->GlobalGenerator->GetProjectMap().end(); ++it) { cmMakefile* mf = it->second[0]->GetMakefile(); std::string outputDir=mf->GetStartOutputDirectory(); std::string projectDir=mf->GetHomeDirectory(); std::string projectName=mf->GetProjectName(); std::string cmakeFilePattern("CMakeLists.txt;*.cmake;"); std::string fileToOpen; const std::vector& lgs= it->second; // create the project.kdevelop.filelist file if(!this->CreateFilelistFile(lgs, outputDir, projectDir, projectName, cmakeFilePattern, fileToOpen)) { cmSystemTools::Error("Can not create filelist file"); return; } //try to find the name of an executable so we have something to //run from kdevelop for now just pick the first executable found std::string executable; for (std::vector::const_iterator lg=lgs.begin(); lg!=lgs.end(); lg++) { cmMakefile* makefile=(*lg)->GetMakefile(); cmTargets& targets=makefile->GetTargets(); for (cmTargets::iterator ti = targets.begin(); ti != targets.end(); ti++) { if (ti->second.GetType()==cmTarget::EXECUTABLE) { executable = ti->second.GetProperty("LOCATION"); break; } } if (!executable.empty()) { break; } } // now create a project file this->CreateProjectFile(outputDir, projectDir, projectName, executable, cmakeFilePattern, fileToOpen); } } bool cmGlobalKdevelopGenerator ::CreateFilelistFile(const std::vector& lgs, const std::string& outputDir, const std::string& projectDirIn, const std::string& projectname, std::string& cmakeFilePattern, std::string& fileToOpen) { std::string projectDir = projectDirIn + "/"; std::string filename = outputDir+ "/" + projectname +".kdevelop.filelist"; std::set files; std::string tmp; for (std::vector::const_iterator it=lgs.begin(); it!=lgs.end(); it++) { cmMakefile* makefile=(*it)->GetMakefile(); const std::vector& listFiles=makefile->GetListFiles(); for (std::vector::const_iterator lt=listFiles.begin(); lt!=listFiles.end(); lt++) { tmp=*lt; cmSystemTools::ReplaceString(tmp, projectDir.c_str(), ""); // make sure the file is part of this source tree if ((tmp[0]!='/') && (strstr(tmp.c_str(), cmake::GetCMakeFilesDirectoryPostSlash())==0)) { files.insert(tmp); tmp=cmSystemTools::GetFilenameName(tmp); //add all files which dont match the default // */CMakeLists.txt;*cmake; to the file pattern if ((tmp!="CMakeLists.txt") && (strstr(tmp.c_str(), ".cmake")==0)) { cmakeFilePattern+=tmp+";"; } } } //get all sources cmTargets& targets=makefile->GetTargets(); for (cmTargets::iterator ti = targets.begin(); ti != targets.end(); ti++) { const std::vector& sources=ti->second.GetSourceFiles(); for (std::vector::const_iterator si=sources.begin(); si!=sources.end(); si++) { tmp=(*si)->GetFullPath(); std::string headerBasename=cmSystemTools::GetFilenamePath(tmp); headerBasename+="/"; headerBasename+=cmSystemTools::GetFilenameWithoutExtension(tmp); cmSystemTools::ReplaceString(tmp, projectDir.c_str(), ""); if ((tmp[0]!='/') && (strstr(tmp.c_str(), cmake::GetCMakeFilesDirectoryPostSlash())==0) && (cmSystemTools::GetFilenameExtension(tmp)!=".moc")) { files.insert(tmp); // check if there's a matching header around for(std::vector::const_iterator ext = makefile->GetHeaderExtensions().begin(); ext != makefile->GetHeaderExtensions().end(); ++ext) { std::string hname=headerBasename; hname += "."; hname += *ext; if(cmSystemTools::FileExists(hname.c_str())) { cmSystemTools::ReplaceString(hname, projectDir.c_str(), ""); files.insert(hname); break; } } } } for (std::vector::const_iterator lt=listFiles.begin(); lt!=listFiles.end(); lt++) { tmp=*lt; cmSystemTools::ReplaceString(tmp, projectDir.c_str(), ""); if ((tmp[0]!='/') && (strstr(tmp.c_str(), cmake::GetCMakeFilesDirectoryPostSlash())==0)) { files.insert(tmp.c_str()); } } } } //check if the output file already exists and read it //insert all files which exist into the set of files std::ifstream oldFilelist(filename.c_str()); if (oldFilelist) { while (cmSystemTools::GetLineFromStream(oldFilelist, tmp)) { if (tmp[0]=='/') { continue; } std::string completePath=projectDir+tmp; if (cmSystemTools::FileExists(completePath.c_str())) { files.insert(tmp); } } oldFilelist.close(); } //now write the new filename cmGeneratedFileStream fout(filename.c_str()); if(!fout) { return false; } fileToOpen=""; for (std::set::const_iterator it=files.begin(); it!=files.end(); it++) { // get the full path to the file tmp=cmSystemTools::CollapseFullPath(it->c_str(), projectDir.c_str()); // just select the first source file if (fileToOpen.empty()) { std::string ext = cmSystemTools::GetFilenameExtension(tmp); if ((ext==".c") || (ext==".cc") || (ext==".cpp") || (ext==".cxx") || (ext==".C") || (ext==".h") || (ext==".hpp")) { fileToOpen=tmp; } } // make it relative to the project dir cmSystemTools::ReplaceString(tmp, projectDir.c_str(), ""); // only put relative paths if (tmp.size() && tmp[0] != '/') { fout << tmp.c_str() <<"\n"; } } return true; } /* create the project file, if it already exists, merge it with the existing one, otherwise create a new one */ void cmGlobalKdevelopGenerator ::CreateProjectFile(const std::string& outputDir, const std::string& projectDir, const std::string& projectname, const std::string& executable, const std::string& cmakeFilePattern, const std::string& fileToOpen) { this->Blacklist.clear(); std::string filename=outputDir+"/"; filename+=projectname+".kdevelop"; std::string sessionFilename=outputDir+"/"; sessionFilename+=projectname+".kdevses"; if (cmSystemTools::FileExists(filename.c_str())) { this->MergeProjectFiles(outputDir, projectDir, filename, executable, cmakeFilePattern, fileToOpen, sessionFilename); } else { // add all subdirectories which are cmake build directories to the // kdevelop blacklist so they are not monitored for added or removed files // since this is handled by adding files to the cmake files cmsys::Directory d; if (d.Load(projectDir.c_str())) { size_t numf = d.GetNumberOfFiles(); for (unsigned int i = 0; i < numf; i++) { std::string nextFile = d.GetFile(i); if ((nextFile!=".") && (nextFile!="..")) { std::string tmp = projectDir; tmp += "/"; tmp += nextFile; if (cmSystemTools::FileIsDirectory(tmp.c_str())) { tmp += "/CMakeCache.txt"; if ((nextFile == "CMakeFiles") || (cmSystemTools::FileExists(tmp.c_str()))) { this->Blacklist.push_back(nextFile); } } } } } this->CreateNewProjectFile(outputDir, projectDir, filename, executable, cmakeFilePattern, fileToOpen, sessionFilename); } } void cmGlobalKdevelopGenerator ::MergeProjectFiles(const std::string& outputDir, const std::string& projectDir, const std::string& filename, const std::string& executable, const std::string& cmakeFilePattern, const std::string& fileToOpen, const std::string& sessionFilename) { std::ifstream oldProjectFile(filename.c_str()); if (!oldProjectFile) { this->CreateNewProjectFile(outputDir, projectDir, filename, executable, cmakeFilePattern, fileToOpen, sessionFilename); return; } /* Read the existing project file (line by line), copy all lines into the new project file, except the ones which can be reliably set from contents of the CMakeLists.txt */ std::string tmp; std::vector lines; while (cmSystemTools::GetLineFromStream(oldProjectFile, tmp)) { lines.push_back(tmp); } oldProjectFile.close(); cmGeneratedFileStream fout(filename.c_str()); if(!fout) { return; } for (std::vector::const_iterator it=lines.begin(); it!=lines.end(); it++) { const char* line=(*it).c_str(); // skip these tags as they are always replaced if ((strstr(line, "")!=0) || (strstr(line, "")!=0) || (strstr(line, "")!=0) || (strstr(line, "")!=0) || (strstr(line, "")!=0) || (strstr(line, "")!=0)) { continue; } // output the line from the file if it is not one of the above tags fout<<*it<<"\n"; // if this is the tag output the stuff that goes in the // general tag if (strstr(line, "")) { fout<< " KDevCustomProject\n"; fout<< " " <\n"; //this one is important fout<<" true\n"; //and this one } // inside kdevcustomproject the must be put if (strstr(line, "")) { fout<<" "<\n"; } // buildtool and builddir go inside if (strstr(line, "")) { fout<<" make\n"; fout<<" "<\n"; } } } void cmGlobalKdevelopGenerator ::CreateNewProjectFile(const std::string& outputDir, const std::string& projectDir, const std::string& filename, const std::string& executable, const std::string& cmakeFilePattern, const std::string& fileToOpen, const std::string& sessionFilename) { cmGeneratedFileStream fout(filename.c_str()); if(!fout) { return; } // check for a version control system bool hasSvn = cmSystemTools::FileExists((projectDir + "/.svn").c_str()); bool hasCvs = cmSystemTools::FileExists((projectDir + "/CVS").c_str()); bool enableCxx = (this->GlobalGenerator->GetLanguageEnabled("C") || this->GlobalGenerator->GetLanguageEnabled("CXX")); bool enableFortran = this->GlobalGenerator->GetLanguageEnabled("Fortran"); std::string primaryLanguage = "C++"; if (enableFortran && !enableCxx) { primaryLanguage="Fortran77"; } fout<<"\n" "\n" " \n" " \n" " \n" " $VERSION$\n" " KDevCustomProject\n" " " << primaryLanguage << "\n" " \n" " " << projectDir.c_str() << "\n"; //this one is important fout<<" true\n"; //and this one // setup additional languages fout<<" \n"; if (enableFortran && enableCxx) { fout<<" Fortran\n"; } if (enableCxx) { fout<<" C\n"; } fout<<" \n"; if (hasSvn) { fout << " kdevsubversion\n"; } else if (hasCvs) { fout << " kdevcvsservice\n"; } fout<<" \n" " \n" " " << outputDir.c_str() << "\n" " \n" " " << executable.c_str() << "\n" " custom\n" " "<\n" " \n" " false\n" " true\n" " \n" " \n" " \n" " make\n"; //this one is important fout<<" "<\n"; //and this one fout<<" \n" " \n" " false\n" " 1\n" " false\n" " " << this->GlobalGenerator->GetLocalGenerators()[0]-> GetMakefile()->GetRequiredDefinition("CMAKE_BUILD_TOOL") << " \n" " default\n" " \n" " \n" " \n" " \n" " \n" " \n" " \n"; fout<<" \n"; for(std::vector::const_iterator dirIt=this->Blacklist.begin(); dirIt != this->Blacklist.end(); ++dirIt) { fout<<" " << dirIt->c_str() << "\n"; } fout<<" \n"; fout<<" \n" " \n" " \n" " \n" " \n" " \n" " \n" " \n" " \n" " \n" " \n" " html/\n" " html/\n" " \n" " \n" " \n" " \n" " \n" " \n" " \n"; if (enableCxx) { fout<<" \n" " \n" " .h\n" " .cpp\n" " \n" " \n" " \n" " \n" " true\n" " true\n" " true\n" " false\n" " true\n" " true\n" " true\n" " 250\n" " 400\n" " 250\n" " \n" " \n" " \n"; } if (enableFortran) { fout<<" \n" " \n" " false\n" " false\n" " false\n" " false\n" " false\n" " false\n" " false\n" " false\n" " false\n" " false\n" " \n" " \n" " \n" " \n" " \n" " \n" " \n" " \n"; } // set up file groups. maybe this can be used with the CMake SOURCE_GROUP() // command fout<<" \n" " \n" " \n"; if (enableCxx) { fout<<" \n" " \n" " \n"; } if (enableFortran) { fout<<" \n"; } fout<<" \n" " true\n" " \n" " \n" " *.o,*.lo,CVS,*~,cmake*\n" " true\n" " \n" " \n" "\n"; if (sessionFilename.empty()) { return; } // and a session file, so that kdevelop opens a file if it opens the // project the first time cmGeneratedFileStream devses(sessionFilename.c_str()); if(!devses) { return; } devses<<"\n" "\n" "\n" " \n" " \n" " \n" " \n" " \n" "\n"; }