| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473 |
- /*=========================================================================
- Program: CMake - Cross-Platform Makefile Generator
- Module: $RCSfile$
- Language: C++
- Date: $Date$
- Version: $Revision$
- Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved.
- See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details.
- This software is distributed WITHOUT ANY WARRANTY; without even
- the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
- PURPOSE. See the above copyright notices for more information.
- =========================================================================*/
- #include "cmVTKMakeInstantiatorCommand.h"
- #include "cmCacheManager.h"
- #include "cmGeneratedFileStream.h"
- bool
- cmVTKMakeInstantiatorCommand
- ::InitialPass(std::vector<std::string> const& argsIn)
- {
- if(argsIn.size() < 3)
- {
- this->SetError("called with incorrect number of arguments");
- return false;
- }
- std::vector<std::string> args;
- m_Makefile->ExpandSourceListArguments(argsIn, args, 2);
- std::string sourceListValue;
-
- m_ClassName = args[0];
-
- std::vector<cmStdString> inSourceLists;
- m_ExportMacro = "-";
- bool includesMode = false;
- bool oldVersion = true;
-
- // Find the path of the files to be generated.
- std::string filePath = m_Makefile->GetCurrentOutputDirectory();
- std::string headerPath = filePath;
-
- // Check whether to use the old or new form.
- if(m_Makefile->GetDefinition("VTK_USE_INSTANTIATOR_NEW"))
- {
- oldVersion = false;
- }
-
- for(unsigned int i=2;i < args.size();++i)
- {
- if(args[i] == "HEADER_LOCATION")
- {
- includesMode = false;
- if(++i < args.size())
- {
- headerPath = args[i];
- }
- else
- {
- this->SetError("HEADER_LOCATION option used without value.");
- return false;
- }
- }
- else if(args[i] == "EXPORT_MACRO")
- {
- includesMode = false;
- if(++i < args.size())
- {
- m_ExportMacro = args[i];
- }
- else
- {
- this->SetError("EXPORT_MACRO option used without value.");
- return false;
- }
- }
- else if(args[i] == "INCLUDES")
- {
- includesMode = true;
- }
- // If not an option, it must be another input source list name or
- // an include file.
- else
- {
- if(!includesMode)
- {
- inSourceLists.push_back(args[i]);
- }
- else
- {
- m_Includes.push_back(args[i]);
- }
- }
- }
-
- if(m_ExportMacro == "-")
- {
- this->SetError("No EXPORT_MACRO option given.");
- return false;
- }
-
- for(std::vector<cmStdString>::const_iterator s = inSourceLists.begin();
- s != inSourceLists.end(); ++s)
- {
- std::string srcName = cmSystemTools::GetFilenameWithoutExtension(*s);
- cmSourceFile *sf = m_Makefile->GetSource(s->c_str());
-
- // Wrap-excluded and abstract classes do not have a New() method.
- // vtkIndent and vtkTimeStamp are special cases and are not
- // vtkObject subclasses.
- if(
- (!sf || (!sf->GetPropertyAsBool("WRAP_EXCLUDE") &&
- !sf->GetPropertyAsBool("ABSTRACT"))) &&
- ((srcName != "vtkIndent") && (srcName != "vtkTimeStamp")))
- {
- m_Classes.push_back(srcName);
- }
- }
-
- // Generate the header with the class declaration.
- {
- std::string fileName = m_ClassName + ".h";
- std::string fullName = headerPath+"/"+fileName;
-
- // Generate the output file with copy-if-different.
- cmGeneratedFileStream fout(fullName.c_str());
-
- // Actually generate the code in the file.
- if(!oldVersion)
- {
- this->GenerateHeaderFile(fout.GetStream());
- }
- else
- {
- this->OldGenerateHeaderFile(fout.GetStream());
- }
- }
-
- // Generate the implementation file.
- {
- std::string fileName = m_ClassName + ".cxx";
- std::string fullName = filePath+"/"+fileName;
-
- // Generate the output file with copy-if-different.
- {
- cmGeneratedFileStream fout(fullName.c_str());
-
- // Actually generate the code in the file.
- if(!oldVersion)
- {
- this->GenerateImplementationFile(fout.GetStream());
- }
- else
- {
- this->OldGenerateImplementationFile(fout.GetStream());
- }
- }
-
- // Add the generated source file into the source list.
- cmSourceFile file;
- file.SetProperty("WRAP_EXCLUDE","1");
- file.SetProperty("ABSTRACT","0");
- file.SetName(fileName.c_str(), filePath.c_str(),
- m_Makefile->GetSourceExtensions(),
- m_Makefile->GetHeaderExtensions());
- m_Makefile->AddSource(file);
- sourceListValue += file.GetSourceName() + ".cxx";
- }
-
- if(oldVersion)
- {
- int groupSize = 10;
- size_t numClasses = m_Classes.size();
- size_t numFullBlocks = numClasses / groupSize;
- size_t lastBlockSize = numClasses % groupSize;
- size_t numBlocks = numFullBlocks + ((lastBlockSize>0)? 1:0);
-
- // Generate the files with the ::New() calls to each class. These
- // are done in groups to keep the translation unit size smaller.
- for(unsigned int block=0; block < numBlocks;++block)
- {
- std::string fileName = this->OldGenerateCreationFileName(block);
- std::string fullName = filePath+"/"+fileName;
-
- // Generate the output file with copy-if-different.
- {
- cmGeneratedFileStream fout(fullName.c_str());
-
- size_t thisBlockSize =
- (block < numFullBlocks)? groupSize:lastBlockSize;
-
- // Actually generate the code in the file.
- this->OldGenerateCreationFile(fout.GetStream(),
- block*groupSize,
- static_cast<int>(thisBlockSize));
- }
-
- // Add the generated source file into the source list.
- cmSourceFile file;
- file.SetProperty("WRAP_EXCLUDE","1");
- file.SetProperty("ABSTRACT","0");
- file.SetName(fileName.c_str(), filePath.c_str(),
- m_Makefile->GetSourceExtensions(),
- m_Makefile->GetHeaderExtensions());
- m_Makefile->AddSource(file);
- sourceListValue += ";";
- sourceListValue += file.GetSourceName() + ".cxx";
- }
- }
-
- m_Makefile->AddDefinition(args[1].c_str(), sourceListValue.c_str());
- return true;
- }
- // Generates the class header file with the definition of the class
- // and its initializer class.
- void
- cmVTKMakeInstantiatorCommand
- ::GenerateHeaderFile(std::ostream& os)
- {
- os <<
- "#ifndef __" << m_ClassName.c_str() << "_h\n"
- "#define __" << m_ClassName.c_str() << "_h\n"
- "\n"
- "#include \"vtkInstantiator.h\"\n";
- for(unsigned int i=0;i < m_Includes.size();++i)
- {
- os << "#include \"" << m_Includes[i].c_str() << "\"\n";
- }
-
- // Write the instantiator class definition.
- os <<
- "\n"
- "class " << m_ExportMacro.c_str() << " " << m_ClassName.c_str() << "\n"
- "{\n"
- "public:\n"
- " " << m_ClassName.c_str() << "();\n"
- " ~" << m_ClassName.c_str() << "();\n"
- "private:\n"
- " static void ClassInitialize();\n"
- " static void ClassFinalize();\n"
- " static unsigned int Count;\n"
- "};\n"
- "\n";
-
- // Write the initialization instance to make sure the creation
- // functions get registered when this generated header is included.
- os <<
- "static "
- << m_ClassName.c_str() << " "
- << m_ClassName.c_str() << "Initializer;\n"
- "\n"
- "#endif\n";
- }
- // Generates the file with the implementation of the class. All
- // methods except the actual object creation functions are generated
- // here.
- void
- cmVTKMakeInstantiatorCommand
- ::GenerateImplementationFile(std::ostream& os)
- {
- // Include the instantiator class header.
- os <<
- "#include \"" << m_ClassName.c_str() << ".h\"\n"
- "\n";
-
- // Write the extern declarations for all the creation functions.
- for(unsigned int i=0;i < m_Classes.size();++i)
- {
- os << "extern vtkObject* vtkInstantiator" << m_Classes[i].c_str() << "New();\n";
- }
-
- // Write the ClassInitialize method to register all the creation functions.
- os <<
- "\n"
- "void " << m_ClassName.c_str() << "::ClassInitialize()\n"
- "{\n";
-
- for(unsigned int i=0;i < m_Classes.size();++i)
- {
- os << " vtkInstantiator::RegisterInstantiator(\""
- << m_Classes[i].c_str() << "\", vtkInstantiator"
- << m_Classes[i].c_str() << "New);\n";
- }
-
- // Write the ClassFinalize method to unregister all the creation functions.
- os <<
- "}\n"
- "\n"
- "void " << m_ClassName.c_str() << "::ClassFinalize()\n"
- "{\n";
-
- for(unsigned int i=0;i < m_Classes.size();++i)
- {
- os << " vtkInstantiator::UnRegisterInstantiator(\""
- << m_Classes[i].c_str() << "\", vtkInstantiator"
- << m_Classes[i].c_str() << "New);\n";
- }
-
- // Write the constructor and destructor of the initializer class to
- // call the ClassInitialize and ClassFinalize methods at the right
- // time.
- os <<
- "}\n"
- "\n" <<
- m_ClassName.c_str() << "::" << m_ClassName.c_str() << "()\n"
- "{\n"
- " if(++" << m_ClassName.c_str() << "::Count == 1)\n"
- " { " << m_ClassName.c_str() << "::ClassInitialize(); }\n"
- "}\n"
- "\n" <<
- m_ClassName.c_str() << "::~" << m_ClassName.c_str() << "()\n"
- "{\n"
- " if(--" << m_ClassName.c_str() << "::Count == 0)\n"
- " { " << m_ClassName.c_str() << "::ClassFinalize(); }\n"
- "}\n"
- "\n"
- "// Number of translation units that include this class's header.\n"
- "// Purposely not initialized. Default is static initialization to 0.\n"
- "unsigned int " << m_ClassName.c_str() << "::Count;\n";
- }
- std::string
- cmVTKMakeInstantiatorCommand::OldGenerateCreationFileName(unsigned int block)
- {
- cmOStringStream nameStr;
- nameStr << m_ClassName.c_str() << block << ".cxx";
- std::string result = nameStr.str();
- return result;
- }
- // Generates a file that includes the headers of the classes it knows
- // how to create and provides functions which create the classes with
- // the New() method.
- void
- cmVTKMakeInstantiatorCommand
- ::OldGenerateCreationFile(std::ostream& os, unsigned int groupStart,
- unsigned int groupSize)
- {
- // Need to include header of generated class.
- os <<
- "#include \"" << m_ClassName.c_str() << ".h\"\n"
- "\n";
-
- // Include class files.
- for(unsigned int i=0;i < groupSize;++i)
- {
- os << "#include \"" << m_Classes[groupStart+i].c_str() << ".h\"\n";
- }
- os <<
- "\n";
- // Write the create function implementations.
- for(unsigned int i=0;i < groupSize;++i)
- {
- os << "vtkObject* " << m_ClassName.c_str() << "::Create_"
- << m_Classes[groupStart+i].c_str() << "() { return "
- << m_Classes[groupStart+i].c_str() << "::New(); }\n";
- }
- }
- // Generates the class header file with the definition of the class
- // and its initializer class.
- void
- cmVTKMakeInstantiatorCommand
- ::OldGenerateHeaderFile(std::ostream& os)
- {
- os <<
- "#ifndef __" << m_ClassName.c_str() << "_h\n"
- "#define __" << m_ClassName.c_str() << "_h\n"
- "\n"
- "#include \"vtkInstantiator.h\"\n";
- for(unsigned int i=0;i < m_Includes.size();++i)
- {
- os << "#include \"" << m_Includes[i].c_str() << "\"\n";
- }
- os <<
- "\n"
- "class " << m_ClassName.c_str() << "Initialize;\n"
- "\n"
- "class " << m_ExportMacro.c_str() << " " << m_ClassName.c_str() << "\n"
- "{\n"
- " friend class " << m_ClassName.c_str() << "Initialize;\n"
- "\n"
- " static void ClassInitialize();\n"
- " static void ClassFinalize();\n"
- "\n";
-
- for(unsigned int i=0;i < m_Classes.size();++i)
- {
- os << " static vtkObject* Create_" << m_Classes[i].c_str() << "();\n";
- }
-
- // Write the initializer class to make sure the creation functions
- // get registered when this generated header is included.
- os <<
- "};\n"
- "\n"
- "class " << m_ExportMacro.c_str() << " " << m_ClassName.c_str() << "Initialize\n"
- "{\n"
- "public:\n"
- " " << m_ClassName.c_str() << "Initialize();\n"
- " ~" << m_ClassName.c_str() << "Initialize();\n"
- "private:\n"
- " static unsigned int Count;\n"
- "};\n"
- "\n"
- "static " << m_ClassName.c_str() << "Initialize " << m_ClassName.c_str() << "Initializer;\n"
- "\n"
- "#endif\n";
- }
- // Generates the file with the implementation of the class. All
- // methods except the actual object creation functions are generated
- // here.
- void
- cmVTKMakeInstantiatorCommand
- ::OldGenerateImplementationFile(std::ostream& os)
- {
- // Write the ClassInitialize method to register all the creation functions.
- os <<
- "#include \"" << m_ClassName.c_str() << ".h\"\n"
- "\n"
- "void " << m_ClassName.c_str() << "::ClassInitialize()\n"
- "{\n";
-
- for(unsigned int i=0;i < m_Classes.size();++i)
- {
- os << " vtkInstantiator::RegisterInstantiator(\""
- << m_Classes[i].c_str() << "\", " << m_ClassName.c_str() << "::Create_"
- << m_Classes[i].c_str() << ");\n";
- }
-
- // Write the ClassFinalize method to unregister all the creation functions.
- os <<
- "}\n"
- "\n"
- "void " << m_ClassName.c_str() << "::ClassFinalize()\n"
- "{\n";
-
- for(unsigned int i=0;i < m_Classes.size();++i)
- {
- os << " vtkInstantiator::UnRegisterInstantiator(\""
- << m_Classes[i].c_str() << "\", " << m_ClassName.c_str() << "::Create_"
- << m_Classes[i].c_str() << ");\n";
- }
-
- // Write the constructor and destructor of the initializer class to
- // call the ClassInitialize and ClassFinalize methods at the right
- // time.
- os <<
- "}\n"
- "\n" <<
- m_ClassName.c_str() << "Initialize::" << m_ClassName.c_str() << "Initialize()\n"
- "{\n"
- " if(++" << m_ClassName.c_str() << "Initialize::Count == 1)\n"
- " { " << m_ClassName.c_str() << "::ClassInitialize(); }\n"
- "}\n"
- "\n" <<
- m_ClassName.c_str() << "Initialize::~" << m_ClassName.c_str() << "Initialize()\n"
- "{\n"
- " if(--" << m_ClassName.c_str() << "Initialize::Count == 0)\n"
- " { " << m_ClassName.c_str() << "::ClassFinalize(); }\n"
- "}\n"
- "\n"
- "// Number of translation units that include this class's header.\n"
- "// Purposely not initialized. Default is static initialization to 0.\n"
- "unsigned int " << m_ClassName.c_str() << "Initialize::Count;\n";
- }
|