/*========================================================================= 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 "cmCreateTestSourceList.h" #include "cmSourceFile.h" // cmCreateTestSourceList bool cmCreateTestSourceList::InitialPass(std::vector const& argsIn) { if (argsIn.size() < 3) { this->SetError("called with wrong number of arguments."); return false; } std::vector args; cmSystemTools::ExpandListArguments(argsIn, args); std::vector::iterator i = args.begin(); std::string extraInclude; std::string function; std::vector tests; // extract extra include and function ot for(; i != args.end(); i++) { if(*i == "EXTRA_INCLUDE") { ++i; if(i == args.end()) { this->SetError("incorrect arguments to EXTRA_INCLUDE"); return false; } extraInclude = *i; } else if(*i == "FUNCTION") { ++i; if(i == args.end()) { this->SetError("incorrect arguments to FUNCTION"); return false; } function = *i; } else { tests.push_back(*i); } } i = tests.begin(); // Name of the source list const char* sourceList = i->c_str(); ++i; // Name of the test driver // make sure they specified an extension if (cmSystemTools::GetFilenameExtension(*i).size() < 2) { this->SetError("You must specify a file extenion for the test driver file."); return false; } std::string driver = m_Makefile->GetCurrentOutputDirectory(); driver += "/"; driver += *i; ++i; std::ofstream fout(driver.c_str()); if (!fout) { std::string err = "Could not create file "; err += driver; err += " for cmCreateTestSourceList command."; this->SetError(err.c_str()); return false; } // Create the test driver file fout << "#include \n" "#include \n" "#include \n" "#include \n"; fout << "#if defined(_MSC_VER) && defined(_DEBUG)\n" "/* MSVC debug hook to prevent dialogs when running from DART. */\n" "# include \n" "static int TestDriverDebugReport(int type, char* message, int* retVal)\n" "{\n" " (void)type; (void)retVal;\n" " if(getenv(\"DART_TEST_FROM_DART\"))\n" " {\n" " fprintf(stderr, message);\n" " exit(1);\n" " }\n" " return 0;\n" "}\n" "#endif\n"; if(extraInclude.size()) { fout << "#include \"" << extraInclude << "\"\n"; } fout << "\n" "/* Forward declare test functions. */\n" "\n"; std::vector::iterator testsBegin = i; std::vector tests_func_name; // The rest of the arguments consist of a list of test source files. // Sadly, they can be in directories. Let's find a unique function // name for the corresponding test, and push it to the tests_func_name // list. // For the moment: // - replace spaces ' ', ':' and '/' with underscores '_' for(i = testsBegin; i != tests.end(); ++i) { if(*i == "EXTRA_INCLUDE") { break; } std::string func_name; if (cmSystemTools::GetFilenamePath(*i).size() > 0) { func_name = cmSystemTools::GetFilenamePath(*i) + "/" + cmSystemTools::GetFilenameWithoutLastExtension(*i); } else { func_name = cmSystemTools::GetFilenameWithoutLastExtension(*i); } cmSystemTools::ConvertToUnixSlashes(func_name); cmSystemTools::ReplaceString(func_name, " ", "_"); cmSystemTools::ReplaceString(func_name, "/", "_"); cmSystemTools::ReplaceString(func_name, ":", "_"); tests_func_name.push_back(func_name); fout << "int " << func_name << "(int, char*[]);\n"; } fout << "\n" "/* Create map. */\n" "\n" "typedef int (*MainFuncPointer)(int , char*[]);\n" "typedef struct\n" "{\n" " const char* name;\n" " MainFuncPointer func;\n" "} functionMapEntry;\n" "\n" "functionMapEntry cmakeGeneratedFunctionMapEntries[] = {\n"; int numTests = 0; std::vector::iterator j; for(i = testsBegin, j = tests_func_name.begin(); i != tests.end(); ++i, ++j) { std::string func_name; if (cmSystemTools::GetFilenamePath(*i).size() > 0) { func_name = cmSystemTools::GetFilenamePath(*i) + "/" + cmSystemTools::GetFilenameWithoutLastExtension(*i); } else { func_name = cmSystemTools::GetFilenameWithoutLastExtension(*i); } fout << " {\n" " \"" << func_name << "\",\n" " " << *j << "\n" " },\n"; numTests++; } fout << "};\n" "\n" "/* Allocate and create a lowercased copy of string\n" " (note that it has to be free'd manually) */\n" "\n" "char* lowercase(const char *string)\n" "{\n" " char *new_string, *p;\n" " new_string = (char *)malloc(sizeof(char) * (size_t)(strlen(string) + 1));\n" " if (!new_string)\n" " {\n" " return NULL;\n" " }\n" " strcpy(new_string, string);\n" " p = new_string;\n" " while (*p != 0)\n" " {\n" " *p = (char)tolower(*p);\n" " ++p;\n" " }\n" " return new_string;\n" "}\n" "\n" "int main(int ac, char *av[])\n" "{\n" " int i, NumTests, testNum, partial_match;\n" " char *arg, *test_name;\n" " \n" "#if defined(_MSC_VER) && defined(_DEBUG)\n" " /* Put in hook for debug library. */\n" " _CrtSetReportHook(TestDriverDebugReport);\n" "#endif\n" " \n" " NumTests = " << numTests << ";\n" " \n" " /* If no test name was given */\n"; if(function.size()) { fout << " /* process command line with user function. */\n" << " " << function << "(&ac, &av);\n"; } fout << " if (ac < 2)\n" " {\n" " /* Ask for a test. */\n" " printf(\"Available tests:\\n\");\n" " for (i =0; i < NumTests; ++i)\n" " {\n" " printf(\"%3d. %s\\n\", i, cmakeGeneratedFunctionMapEntries[i].name);\n" " }\n" " printf(\"To run a test, enter the test number: \");\n" " testNum = 0;\n" " scanf(\"%d\", &testNum);\n" " if (testNum >= NumTests)\n" " {\n" " printf(\"%3d is an invalid test number.\\n\", testNum);\n" " return -1;\n" " }\n" " return (*cmakeGeneratedFunctionMapEntries[testNum].func)(ac-1, av+1);\n" " }\n" " \n" " /* If partial match is requested. */\n" " partial_match = (strcmp(av[1], \"-R\") == 0) ? 1 : 0;\n" " if (partial_match && ac < 3)\n" " {\n" " printf(\"-R needs an additional parameter.\\n\");\n" " return -1;\n" " }\n" " \n" " arg = lowercase(av[1 + partial_match]);\n" " for (i =0; i < NumTests; ++i)\n" " {\n" " test_name = lowercase(cmakeGeneratedFunctionMapEntries[i].name);\n" " if (partial_match && strstr(test_name, arg) != NULL)\n" " {\n" " free(arg);\n" " free(test_name);\n" " return (*cmakeGeneratedFunctionMapEntries[i].func)(ac - 2, av + 2);\n" " }\n" " else if (!partial_match && strcmp(test_name, arg) == 0)\n" " {\n" " free(arg);\n" " free(test_name);\n" " return (*cmakeGeneratedFunctionMapEntries[i].func)(ac - 1, av + 1);\n" " }\n" " free(test_name);\n" " }\n" " free(arg);\n" " \n" " /* Nothing was run, display the test names. */\n" " printf(\"Available tests:\\n\");\n" " for (i =0; i < NumTests; ++i)\n" " {\n" " printf(\"%3d. %s\\n\", i, cmakeGeneratedFunctionMapEntries[i].name);\n" " }\n" " printf(\"Failed: %s is an invalid test name.\\n\", av[1]);\n" " \n" " return -1;\n" "}\n"; fout.close(); // Create the source list cmSourceFile cfile; std::string sourceListValue; cfile.SetProperty("ABSTRACT","0"); cfile.SetName(cmSystemTools::GetFilenameWithoutExtension(args[1]).c_str(), m_Makefile->GetCurrentOutputDirectory(), cmSystemTools::GetFilenameExtension(args[1]).c_str()+1, false); m_Makefile->AddSource(cfile); sourceListValue = args[1]; for(i = testsBegin; i != tests.end(); ++i) { cmSourceFile icfile; icfile.SetProperty("ABSTRACT","0"); icfile.SetName(i->c_str(), m_Makefile->GetCurrentDirectory(), m_Makefile->GetSourceExtensions(), m_Makefile->GetHeaderExtensions()); m_Makefile->AddSource(icfile); sourceListValue += ";"; sourceListValue += *i; } m_Makefile->AddDefinition(sourceList, sourceListValue.c_str()); return true; }