CREATE_PROJECT: Add support for tests

- Added --tests command line switch
 - Parse test/module.mk to extract the list of test folders
 - Automatically run tests after a successful build
This commit is contained in:
Littleboy 2013-07-07 11:23:19 -04:00
parent 98899c6ce3
commit a949a88220
12 changed files with 262 additions and 78 deletions

1
.gitignore vendored
View File

@ -93,6 +93,7 @@ project.xcworkspace
/dists/msvc*/*.SAV /dists/msvc*/*.SAV
/dists/msvc*/*.dat /dists/msvc*/*.dat
/dists/msvc*/*.dll /dists/msvc*/*.dll
/dists/msvc*/test_runner.cpp
/doc/*.aux /doc/*.aux
/doc/*.dvi /doc/*.dvi

View File

@ -109,7 +109,7 @@ enum ProjectType {
int main(int argc, char *argv[]) { int main(int argc, char *argv[]) {
#ifndef USE_WIN32_API #ifndef USE_WIN32_API
// Initialize random number generator for UUID creation // Initialize random number generator for UUID creation
std::srand((uint)std::time(0)); std::srand((unsigned int)std::time(0));
#endif #endif
if (argc < 2) { if (argc < 2) {
@ -264,7 +264,7 @@ int main(int argc, char *argv[]) {
setup.filePrefix.erase(setup.filePrefix.size() - 1); setup.filePrefix.erase(setup.filePrefix.size() - 1);
} else if (!std::strcmp(argv[i], "--output-dir")) { } else if (!std::strcmp(argv[i], "--output-dir")) {
if (i + 1 >= argc) { if (i + 1 >= argc) {
std::cerr << "ERROR: Missing \"path\" parameter for \"--output-dirx\"!\n"; std::cerr << "ERROR: Missing \"path\" parameter for \"--output-dir\"!\n";
return -1; return -1;
} }
@ -279,12 +279,23 @@ int main(int argc, char *argv[]) {
setup.createInstaller = true; setup.createInstaller = true;
} else if (!std::strcmp(argv[i], "--tools")) { } else if (!std::strcmp(argv[i], "--tools")) {
setup.devTools = true; setup.devTools = true;
} else if (!std::strcmp(argv[i], "--tests")) {
setup.tests = true;
} else { } else {
std::cerr << "ERROR: Unknown parameter \"" << argv[i] << "\"\n"; std::cerr << "ERROR: Unknown parameter \"" << argv[i] << "\"\n";
return -1; return -1;
} }
} }
// When building tests, disable some features
if (setup.tests) {
setFeatureBuildState("mt32emu", setup.features, false);
setFeatureBuildState("eventrecorder", setup.features, false);
for (EngineDescList::iterator j = setup.engines.begin(); j != setup.engines.end(); ++j)
j->enable = false;
}
// Print status // Print status
cout << "Enabled engines:\n\n"; cout << "Enabled engines:\n\n";
for (EngineDescList::const_iterator i = setup.engines.begin(); i != setup.engines.end(); ++i) { for (EngineDescList::const_iterator i = setup.engines.begin(); i != setup.engines.end(); ++i) {
@ -321,6 +332,12 @@ int main(int argc, char *argv[]) {
} }
} }
// Check if tools and tests are enabled simultaneously
if (setup.devTools && setup.tests) {
std::cerr << "ERROR: The tools and tests projects cannot be created simultaneously\n";
return -1;
}
// Setup defines and libraries // Setup defines and libraries
setup.defines = getEngineDefines(setup.engines); setup.defines = getEngineDefines(setup.engines);
setup.libraries = getFeatureLibraries(setup.features); setup.libraries = getFeatureLibraries(setup.features);
@ -358,8 +375,8 @@ int main(int argc, char *argv[]) {
return -1; return -1;
case kProjectCodeBlocks: case kProjectCodeBlocks:
if (setup.devTools) { if (setup.devTools || setup.tests) {
std::cerr << "ERROR: Building tools is not supported for the CodeBlocks project type!\n"; std::cerr << "ERROR: Building tools or tests is not supported for the CodeBlocks project type!\n";
return -1; return -1;
} }
@ -531,8 +548,8 @@ int main(int argc, char *argv[]) {
break; break;
case kProjectXcode: case kProjectXcode:
if (setup.devTools) { if (setup.devTools || setup.tests) {
std::cerr << "ERROR: Building tools is not supported for the XCode project type!\n"; std::cerr << "ERROR: Building tools or tests is not supported for the XCode project type!\n";
return -1; return -1;
} }
@ -573,6 +590,11 @@ int main(int argc, char *argv[]) {
setup.projectDescription += "Tools"; setup.projectDescription += "Tools";
} }
if (setup.tests) {
setup.projectName += "-tests";
setup.projectDescription += "Tests";
}
provider->createProject(setup); provider->createProject(setup);
delete provider; delete provider;
@ -623,6 +645,9 @@ void displayHelp(const char *exe) {
" --tools Create project files for the devtools\n" " --tools Create project files for the devtools\n"
" (ignores --build-events and --installer, as well as engine settings)\n" " (ignores --build-events and --installer, as well as engine settings)\n"
" (default: false)\n" " (default: false)\n"
" --tests Create project files for the tests\n"
" (ignores --build-events and --installer, as well as engine settings)\n"
" (default: false)\n"
"\n" "\n"
"Engines settings:\n" "Engines settings:\n"
" --list-engines list all available engines and their default state\n" " --list-engines list all available engines and their default state\n"
@ -1130,69 +1155,66 @@ ProjectProvider::ProjectProvider(StringList &global_warnings, std::map<std::stri
: _version(version), _globalWarnings(global_warnings), _projectWarnings(project_warnings) { : _version(version), _globalWarnings(global_warnings), _projectWarnings(project_warnings) {
} }
void ProjectProvider::createProject(const BuildSetup &setup) { void ProjectProvider::createProject(BuildSetup &setup) {
std::string targetFolder;
if (setup.devTools) { if (setup.devTools) {
_uuidMap = createToolsUUIDMap(); _uuidMap = createToolsUUIDMap();
targetFolder = "/devtools/";
// We also need to add the UUID of the main project file. } else if (!setup.tests) {
const std::string svmUUID = _uuidMap[setup.projectName] = createUUID();
createWorkspace(setup);
StringList in, ex;
// Create tools project files
for (UUIDMap::const_iterator i = _uuidMap.begin(); i != _uuidMap.end(); ++i) {
if (i->first == setup.projectName)
continue;
in.clear(); ex.clear();
const std::string moduleDir = setup.srcDir + "/devtools/" + i->first;
createModuleList(moduleDir, setup.defines, in, ex);
createProjectFile(i->first, i->second, setup, moduleDir, in, ex);
}
// Create other misc. build files
createOtherBuildFiles(setup);
} else {
_uuidMap = createUUIDMap(setup); _uuidMap = createUUIDMap(setup);
targetFolder = "/engines/";
}
// We also need to add the UUID of the main project file. // We also need to add the UUID of the main project file.
const std::string svmUUID = _uuidMap[setup.projectName] = createUUID(); const std::string svmUUID = _uuidMap[setup.projectName] = createUUID();
// Create Solution/Workspace file
createWorkspace(setup); createWorkspace(setup);
StringList in, ex; StringList in, ex;
// Create engine project files // Create project files
for (UUIDMap::const_iterator i = _uuidMap.begin(); i != _uuidMap.end(); ++i) { for (UUIDMap::const_iterator i = _uuidMap.begin(); i != _uuidMap.end(); ++i) {
if (i->first == setup.projectName) if (i->first == setup.projectName)
continue; continue;
in.clear(); ex.clear(); in.clear(); ex.clear();
const std::string moduleDir = setup.srcDir + "/engines/" + i->first; const std::string moduleDir = setup.srcDir + targetFolder + i->first;
createModuleList(moduleDir, setup.defines, in, ex); createModuleList(moduleDir, setup.defines, setup.testDirs, in, ex);
createProjectFile(i->first, i->second, setup, moduleDir, in, ex); createProjectFile(i->first, i->second, setup, moduleDir, in, ex);
} }
if (setup.tests) {
// Create the main project file.
in.clear(); ex.clear();
createModuleList(setup.srcDir + "/backends", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/backends/platform/sdl", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/base", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/common", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/engines", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/graphics", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/gui", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/audio", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/test", setup.defines, setup.testDirs, in, ex);
createProjectFile(setup.projectName, svmUUID, setup, setup.srcDir, in, ex);
} else if (!setup.devTools) {
// Last but not least create the main project file. // Last but not least create the main project file.
in.clear(); ex.clear(); in.clear(); ex.clear();
// File list for the Project file // File list for the Project file
createModuleList(setup.srcDir + "/backends", setup.defines, in, ex); createModuleList(setup.srcDir + "/backends", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/backends/platform/sdl", setup.defines, in, ex); createModuleList(setup.srcDir + "/backends/platform/sdl", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/base", setup.defines, in, ex); createModuleList(setup.srcDir + "/base", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/common", setup.defines, in, ex); createModuleList(setup.srcDir + "/common", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/engines", setup.defines, in, ex); createModuleList(setup.srcDir + "/engines", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/graphics", setup.defines, in, ex); createModuleList(setup.srcDir + "/graphics", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/gui", setup.defines, in, ex); createModuleList(setup.srcDir + "/gui", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/audio", setup.defines, in, ex); createModuleList(setup.srcDir + "/audio", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/audio/softsynth/mt32", setup.defines, in, ex); createModuleList(setup.srcDir + "/audio/softsynth/mt32", setup.defines, setup.testDirs, in, ex);
createModuleList(setup.srcDir + "/video", setup.defines, in, ex); createModuleList(setup.srcDir + "/video", setup.defines, setup.testDirs, in, ex);
// Resource files // Resource files
in.push_back(setup.srcDir + "/icons/" + setup.projectName + ".ico"); in.push_back(setup.srcDir + "/icons/" + setup.projectName + ".ico");
@ -1211,10 +1233,10 @@ void ProjectProvider::createProject(const BuildSetup &setup) {
// Create the main project file. // Create the main project file.
createProjectFile(setup.projectName, svmUUID, setup, setup.srcDir, in, ex); createProjectFile(setup.projectName, svmUUID, setup, setup.srcDir, in, ex);
}
// Create other misc. build files // Create other misc. build files
createOtherBuildFiles(setup); createOtherBuildFiles(setup);
}
} }
ProjectProvider::UUIDMap ProjectProvider::createUUIDMap(const BuildSetup &setup) const { ProjectProvider::UUIDMap ProjectProvider::createUUIDMap(const BuildSetup &setup) const {
@ -1322,7 +1344,7 @@ void ProjectProvider::addFilesToProject(const std::string &dir, std::ofstream &p
delete files; delete files;
} }
void ProjectProvider::createModuleList(const std::string &moduleDir, const StringList &defines, StringList &includeList, StringList &excludeList) const { void ProjectProvider::createModuleList(const std::string &moduleDir, const StringList &defines, StringList &testDirs, StringList &includeList, StringList &excludeList) const {
const std::string moduleMkFile = moduleDir + "/module.mk"; const std::string moduleMkFile = moduleDir + "/module.mk";
std::ifstream moduleMk(moduleMkFile.c_str()); std::ifstream moduleMk(moduleMkFile.c_str());
if (!moduleMk) if (!moduleMk)
@ -1453,6 +1475,59 @@ void ProjectProvider::createModuleList(const std::string &moduleDir, const Strin
++i; ++i;
} }
} }
} else if (*i == "TESTS") {
if (tokens.size() < 3)
error("Malformed TESTS definition in " + moduleMkFile);
++i;
if (*i != ":=" && *i != "+=" && *i != "=")
error("Malformed TESTS definition in " + moduleMkFile);
++i;
while (i != tokens.end()) {
// Read input
std::string folder = unifyPath(*i);
// Get include folder
const std::string source_dir = "$(srcdir)/";
const std::string selector = getLastPathComponent(folder);
const std::string module = getLastPathComponent(moduleDir);
folder.replace(folder.find(source_dir), source_dir.length(), "");
folder.replace(folder.find(selector), selector.length(), "");
folder.replace(folder.find(module), module.length(), moduleDir);
// Scan all files in the include folder
FileList files = listDirectory(folder);
if (files.empty())
continue;
// Add to list of test folders
testDirs.push_back(folder);
for (FileList::const_iterator f = files.begin(); f != files.end(); ++f) {
if (f->isDirectory)
continue;
std::string filename = folder + f->name;
if (shouldInclude.top()) {
// In case we should include a file, we need to make
// sure it is not in the exclude list already. If it
// is we just drop it from the exclude list.
excludeList.remove(filename);
includeList.push_back(filename);
} else if (std::find(includeList.begin(), includeList.end(), filename) == includeList.end()) {
// We only add the file to the exclude list in case it
// has not yet been added to the include list.
excludeList.push_back(filename);
}
}
++i;
}
} else if (*i == "ifdef") { } else if (*i == "ifdef") {
if (tokens.size() < 2) if (tokens.size() < 2)
error("Malformed ifdef in " + moduleMkFile); error("Malformed ifdef in " + moduleMkFile);

View File

@ -221,13 +221,16 @@ struct BuildSetup {
StringList defines; ///< List of all defines for the build. StringList defines; ///< List of all defines for the build.
StringList libraries; ///< List of all external libraries required for the build. StringList libraries; ///< List of all external libraries required for the build.
StringList testDirs; ///< List of all folders containing tests
bool devTools; ///< Generate project files for the tools bool devTools; ///< Generate project files for the tools
bool tests; ///< Generate project files for the tests
bool runBuildEvents; ///< Run build events as part of the build (generate revision number and copy engine/theme data & needed files to the build folder bool runBuildEvents; ///< Run build events as part of the build (generate revision number and copy engine/theme data & needed files to the build folder
bool createInstaller; ///< Create NSIS installer after the build bool createInstaller; ///< Create NSIS installer after the build
BuildSetup() { BuildSetup() {
devTools = false; devTools = false;
tests = false;
runBuildEvents = false; runBuildEvents = false;
createInstaller = false; createInstaller = false;
} }
@ -339,7 +342,7 @@ public:
* *
* @param setup Description of the desired build setup. * @param setup Description of the desired build setup.
*/ */
void createProject(const BuildSetup &setup); void createProject(BuildSetup &setup);
/** /**
* Returns the last path component. * Returns the last path component.
@ -430,10 +433,11 @@ protected:
* *
* @param moduleDir Path to the module. * @param moduleDir Path to the module.
* @param defines List of set defines. * @param defines List of set defines.
* @param testDirs List of folders containing tests.
* @param includeList Reference to a list, where included files should be added. * @param includeList Reference to a list, where included files should be added.
* @param excludeList Reference to a list, where excluded files should be added. * @param excludeList Reference to a list, where excluded files should be added.
*/ */
void createModuleList(const std::string &moduleDir, const StringList &defines, StringList &includeList, StringList &excludeList) const; void createModuleList(const std::string &moduleDir, const StringList &defines, StringList &testDirs, StringList &includeList, StringList &excludeList) const;
/** /**
* Creates an UUID for every enabled engine of the * Creates an UUID for every enabled engine of the
@ -448,7 +452,7 @@ protected:
* Creates an UUID for every enabled tool of the * Creates an UUID for every enabled tool of the
* passed build description. * passed build description.
* *
* @return A map, which includes UUIDs for all enabled engines. * @return A map, which includes UUIDs for all enabled tools.
*/ */
UUIDMap createToolsUUIDMap() const; UUIDMap createToolsUUIDMap() const;

View File

@ -69,7 +69,7 @@ inline void outputConfiguration(std::ostream &project, const std::string &config
inline void outputConfigurationType(const BuildSetup &setup, std::ostream &project, const std::string &name, const std::string &config, int version) { inline void outputConfigurationType(const BuildSetup &setup, std::ostream &project, const std::string &name, const std::string &config, int version) {
project << "\t<PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='" << config << "'\" Label=\"Configuration\">\n" project << "\t<PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='" << config << "'\" Label=\"Configuration\">\n"
"\t\t<ConfigurationType>" << ((name == setup.projectName || setup.devTools) ? "Application" : "StaticLibrary") << "</ConfigurationType>\n" "\t\t<ConfigurationType>" << ((name == setup.projectName || setup.devTools || setup.tests) ? "Application" : "StaticLibrary") << "</ConfigurationType>\n"
"\t\t<PlatformToolset>v" << version << "0</PlatformToolset>\n" "\t\t<PlatformToolset>v" << version << "0</PlatformToolset>\n"
"\t</PropertyGroup>\n"; "\t</PropertyGroup>\n";
} }
@ -159,10 +159,26 @@ void MSBuildProvider::createProjectFile(const std::string &name, const std::stri
if (name == setup.projectName) if (name == setup.projectName)
writeReferences(setup, project); writeReferences(setup, project);
// Output auto-generated test runner
if (setup.tests) {
project << "\t<ItemGroup>\n";
project << "\t\t<ClCompile Include=\"test_runner.cpp\" />\n";
project << "\t</ItemGroup>\n";
}
project << "\t<Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.targets\" />\n" project << "\t<Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.targets\" />\n"
"\t<ImportGroup Label=\"ExtensionTargets\">\n" "\t<ImportGroup Label=\"ExtensionTargets\">\n"
"\t</ImportGroup>\n" "\t</ImportGroup>\n";
"</Project>\n";
if (setup.tests) {
// We override the normal target to ignore the exit code (this allows us to have a clean output and not message about the command exit code)
project << "\t\t<Target Name=\"PostBuildEvent\">\n"
<< "\t\t\t<Message Text=\"Description: Run tests\" />\n"
<< "\t\t\t<Exec Command=\"$(TargetPath)\" IgnoreExitCode=\"true\" />\n"
<< "\t\t</Target>\n";
}
project << "</Project>\n";
// Output filter file if necessary // Output filter file if necessary
createFiltersFile(setup, name); createFiltersFile(setup, name);
@ -248,7 +264,7 @@ void MSBuildProvider::outputProjectSettings(std::ofstream &project, const std::s
bool disableEditAndContinue = find(_disableEditAndContinue.begin(), _disableEditAndContinue.end(), name) != _disableEditAndContinue.end(); bool disableEditAndContinue = find(_disableEditAndContinue.begin(), _disableEditAndContinue.end(), name) != _disableEditAndContinue.end();
// Nothing to add here, move along! // Nothing to add here, move along!
if (!setup.devTools && name != setup.projectName && !enableLanguageExtensions && !disableEditAndContinue && warningsIterator == _projectWarnings.end()) if ((!setup.devTools || !setup.tests) && name != setup.projectName && !enableLanguageExtensions && !disableEditAndContinue && warningsIterator == _projectWarnings.end())
return; return;
std::string warnings = ""; std::string warnings = "";
@ -260,7 +276,7 @@ void MSBuildProvider::outputProjectSettings(std::ofstream &project, const std::s
"\t\t<ClCompile>\n"; "\t\t<ClCompile>\n";
// Language Extensions // Language Extensions
if (setup.devTools || name == setup.projectName || enableLanguageExtensions) if (setup.devTools || setup.tests || name == setup.projectName || enableLanguageExtensions)
project << "\t\t\t<DisableLanguageExtensions>false</DisableLanguageExtensions>\n"; project << "\t\t\t<DisableLanguageExtensions>false</DisableLanguageExtensions>\n";
// Edit and Continue // Edit and Continue
@ -274,18 +290,18 @@ void MSBuildProvider::outputProjectSettings(std::ofstream &project, const std::s
project << "\t\t</ClCompile>\n"; project << "\t\t</ClCompile>\n";
// Link configuration for main project // Link configuration for main project
if (name == setup.projectName || setup.devTools) { if (name == setup.projectName || setup.devTools || setup.tests) {
std::string libraries; std::string libraries;
for (StringList::const_iterator i = setup.libraries.begin(); i != setup.libraries.end(); ++i) for (StringList::const_iterator i = setup.libraries.begin(); i != setup.libraries.end(); ++i)
libraries += *i + ".lib;"; libraries += *i + ".lib;";
project << "\t\t<Link>\n" project << "\t\t<Link>\n"
"\t\t\t<OutputFile>$(OutDir)" << (setup.devTools ? name : setup.projectName) << ".exe</OutputFile>\n" "\t\t\t<OutputFile>$(OutDir)" << ((setup.devTools || setup.tests) ? name : setup.projectName) << ".exe</OutputFile>\n"
"\t\t\t<AdditionalDependencies>" << libraries << "%(AdditionalDependencies)</AdditionalDependencies>\n" "\t\t\t<AdditionalDependencies>" << libraries << "%(AdditionalDependencies)</AdditionalDependencies>\n"
"\t\t</Link>\n"; "\t\t</Link>\n";
if (!setup.devTools && setup.runBuildEvents) { if (!setup.devTools && !setup.tests && setup.runBuildEvents) {
project << "\t\t<PreBuildEvent>\n" project << "\t\t<PreBuildEvent>\n"
"\t\t\t<Message>Generate revision</Message>\n" "\t\t\t<Message>Generate revision</Message>\n"
"\t\t\t<Command>" << getPreBuildEvent() << "</Command>\n" "\t\t\t<Command>" << getPreBuildEvent() << "</Command>\n"
@ -296,6 +312,11 @@ void MSBuildProvider::outputProjectSettings(std::ofstream &project, const std::s
"\t\t\t<Message>Copy data files to the build folder</Message>\n" "\t\t\t<Message>Copy data files to the build folder</Message>\n"
"\t\t\t<Command>" << getPostBuildEvent(isWin32, setup.createInstaller) << "</Command>\n" "\t\t\t<Command>" << getPostBuildEvent(isWin32, setup.createInstaller) << "</Command>\n"
"\t\t</PostBuildEvent>\n"; "\t\t</PostBuildEvent>\n";
} else if (setup.tests) {
project << "\t\t<PreBuildEvent>\n"
"\t\t\t<Message>Generate runner.cpp</Message>\n"
"\t\t\t<Command>" << getTestPreBuildEvent(setup) << "</Command>\n"
"\t\t</PreBuildEvent>\n";
} }
} }
@ -330,9 +351,9 @@ void MSBuildProvider::outputGlobalPropFile(const BuildSetup &setup, std::ofstrea
"\t\t<ClCompile>\n" "\t\t<ClCompile>\n"
"\t\t\t<DisableLanguageExtensions>true</DisableLanguageExtensions>\n" "\t\t\t<DisableLanguageExtensions>true</DisableLanguageExtensions>\n"
"\t\t\t<DisableSpecificWarnings>" << warnings << ";%(DisableSpecificWarnings)</DisableSpecificWarnings>\n" "\t\t\t<DisableSpecificWarnings>" << warnings << ";%(DisableSpecificWarnings)</DisableSpecificWarnings>\n"
"\t\t\t<AdditionalIncludeDirectories>$(" << LIBS_DEFINE << ")\\include;" << prefix << ";" << prefix << "\\engines;$(TargetDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\n" "\t\t\t<AdditionalIncludeDirectories>$(" << LIBS_DEFINE << ")\\include;" << prefix << ";" << prefix << "\\engines;" << (setup.tests ? prefix + "\\test\\cxxtest;" : "") << "$(TargetDir);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>\n"
"\t\t\t<PreprocessorDefinitions>" << definesList << "%(PreprocessorDefinitions)</PreprocessorDefinitions>\n" "\t\t\t<PreprocessorDefinitions>" << definesList << "%(PreprocessorDefinitions)</PreprocessorDefinitions>\n"
"\t\t\t<ExceptionHandling>" << (setup.devTools ? "Sync" : "") << "</ExceptionHandling>\n"; "\t\t\t<ExceptionHandling>" << ((setup.devTools || setup.tests) ? "Sync" : "") << "</ExceptionHandling>\n";
#if NEEDS_RTTI #if NEEDS_RTTI
properties << "\t\t\t<RuntimeTypeInfo>true</RuntimeTypeInfo>\n"; properties << "\t\t\t<RuntimeTypeInfo>true</RuntimeTypeInfo>\n";
@ -348,7 +369,7 @@ void MSBuildProvider::outputGlobalPropFile(const BuildSetup &setup, std::ofstrea
"\t\t\t<IgnoreSpecificDefaultLibraries>%(IgnoreSpecificDefaultLibraries)</IgnoreSpecificDefaultLibraries>\n" "\t\t\t<IgnoreSpecificDefaultLibraries>%(IgnoreSpecificDefaultLibraries)</IgnoreSpecificDefaultLibraries>\n"
"\t\t\t<SubSystem>Console</SubSystem>\n"; "\t\t\t<SubSystem>Console</SubSystem>\n";
if (!setup.devTools) if (!setup.devTools && !setup.tests)
properties << "\t\t\t<EntryPointSymbol>WinMainCRTStartup</EntryPointSymbol>\n"; properties << "\t\t\t<EntryPointSymbol>WinMainCRTStartup</EntryPointSymbol>\n";
properties << "\t\t</Link>\n" properties << "\t\t</Link>\n"

View File

@ -161,6 +161,16 @@ std::string MSVCProvider::getPreBuildEvent() const {
return cmdLine; return cmdLine;
} }
std::string MSVCProvider::getTestPreBuildEvent(const BuildSetup &setup) const {
// Build list of folders containing tests
std::string target = "";
for (StringList::const_iterator it = setup.testDirs.begin(); it != setup.testDirs.end(); ++it)
target += " $(SolutionDir)" + *it + "*.h";
return "&quot;$(SolutionDir)../../test/cxxtest/cxxtestgen.py&quot; --runner=ParenPrinter --no-std --no-eh -o $(SolutionDir)test_runner.cpp" + target;
}
std::string MSVCProvider::getPostBuildEvent(bool isWin32, bool createInstaller) const { std::string MSVCProvider::getPostBuildEvent(bool isWin32, bool createInstaller) const {
std::string cmdLine = ""; std::string cmdLine = "";

View File

@ -87,6 +87,13 @@ protected:
*/ */
std::string getPreBuildEvent() const; std::string getPreBuildEvent() const;
/**
* Get the command line for the test generator
*
* @param setup Description of the desired build setup.
*/
std::string getTestPreBuildEvent(const BuildSetup &setup) const;
/** /**
* Get the command line for copying data files to the build directory. * Get the command line for copying data files to the build directory.
* *

View File

@ -83,7 +83,7 @@ void VisualStudioProvider::createProjectFile(const std::string &name, const std:
// Check for project-specific warnings: // Check for project-specific warnings:
std::map< std::string, std::list<std::string> >::iterator warningsIterator = _projectWarnings.find(name); std::map< std::string, std::list<std::string> >::iterator warningsIterator = _projectWarnings.find(name);
if (setup.devTools || name == setup.projectName) { if (setup.devTools || setup.tests || name == setup.projectName) {
std::string libraries; std::string libraries;
for (StringList::const_iterator i = setup.libraries.begin(); i != setup.libraries.end(); ++i) for (StringList::const_iterator i = setup.libraries.begin(); i != setup.libraries.end(); ++i)
@ -140,6 +140,11 @@ void VisualStudioProvider::createProjectFile(const std::string &name, const std:
else else
addFilesToProject(moduleDir, project, includeList, excludeList, setup.filePrefix); addFilesToProject(moduleDir, project, includeList, excludeList, setup.filePrefix);
// Output auto-generated test runner
if (setup.tests) {
project << "\t\t<File RelativePath=\"test_runner.cpp\" />\n";
}
project << "\t</Files>\n" project << "\t</Files>\n"
"</VisualStudioProject>\n"; "</VisualStudioProject>\n";
} }
@ -161,7 +166,7 @@ void VisualStudioProvider::outputConfiguration(const BuildSetup &setup, std::ost
} }
void VisualStudioProvider::outputBuildEvents(std::ostream &project, const BuildSetup &setup, const bool isWin32) { void VisualStudioProvider::outputBuildEvents(std::ostream &project, const BuildSetup &setup, const bool isWin32) {
if (!setup.devTools && setup.runBuildEvents) { if (!setup.devTools && !setup.tests && setup.runBuildEvents) {
project << "\t\t\t<Tool\tName=\"VCPreBuildEventTool\"\n" project << "\t\t\t<Tool\tName=\"VCPreBuildEventTool\"\n"
"\t\t\t\tCommandLine=\"" << getPreBuildEvent() << "\"\n" "\t\t\t\tCommandLine=\"" << getPreBuildEvent() << "\"\n"
"\t\t\t/>\n" "\t\t\t/>\n"
@ -169,6 +174,17 @@ void VisualStudioProvider::outputBuildEvents(std::ostream &project, const BuildS
"\t\t\t\tCommandLine=\"" << getPostBuildEvent(isWin32, setup.createInstaller) << "\"\n" "\t\t\t\tCommandLine=\"" << getPostBuildEvent(isWin32, setup.createInstaller) << "\"\n"
"\t\t\t/>\n"; "\t\t\t/>\n";
} }
// Generate runner file before build for tests
if (setup.tests) {
project << "\t\t\t<Tool\tName=\"VCPreBuildEventTool\"\n"
"\t\t\t\tCommandLine=\"" << getTestPreBuildEvent(setup) << "\"\n"
"\t\t\t/>\n";
project << "\t\t\t<Tool\tName=\"VCPostBuildEventTool\"\n"
"\t\t\t\tCommandLine=\"$(TargetPath)\" IgnoreExitCode=\"true\"\n"
"\t\t\t/>\n";
}
} }
void VisualStudioProvider::writeReferences(const BuildSetup &setup, std::ofstream &output) { void VisualStudioProvider::writeReferences(const BuildSetup &setup, std::ofstream &output) {
@ -212,9 +228,9 @@ void VisualStudioProvider::outputGlobalPropFile(const BuildSetup &setup, std::of
"\t\tName=\"VCCLCompilerTool\"\n" "\t\tName=\"VCCLCompilerTool\"\n"
"\t\tDisableLanguageExtensions=\"" << (setup.devTools ? "false" : "true") << "\"\n" "\t\tDisableLanguageExtensions=\"" << (setup.devTools ? "false" : "true") << "\"\n"
"\t\tDisableSpecificWarnings=\"" << warnings << "\"\n" "\t\tDisableSpecificWarnings=\"" << warnings << "\"\n"
"\t\tAdditionalIncludeDirectories=\"" << prefix << ";" << prefix << "\\engines;$(" << LIBS_DEFINE << ")\\include;$(TargetDir)\"\n" "\t\tAdditionalIncludeDirectories=\"" << prefix << ";" << prefix << "\\engines;$(" << LIBS_DEFINE << ")\\include;" << (setup.tests ? prefix + "\\test\\cxxtest;" : "") << "$(TargetDir)\"\n"
"\t\tPreprocessorDefinitions=\"" << definesList << "\"\n" "\t\tPreprocessorDefinitions=\"" << definesList << "\"\n"
"\t\tExceptionHandling=\"" << (setup.devTools ? "1" : "0") << "\"\n"; "\t\tExceptionHandling=\"" << ((setup.devTools || setup.tests) ? "1" : "0") << "\"\n";
#if NEEDS_RTTI #if NEEDS_RTTI
properties << "\t\tRuntimeTypeInfo=\"true\"\n"; properties << "\t\tRuntimeTypeInfo=\"true\"\n";
@ -235,7 +251,7 @@ void VisualStudioProvider::outputGlobalPropFile(const BuildSetup &setup, std::of
"\t\tIgnoreDefaultLibraryNames=\"\"\n" "\t\tIgnoreDefaultLibraryNames=\"\"\n"
"\t\tSubSystem=\"1\"\n"; "\t\tSubSystem=\"1\"\n";
if (!setup.devTools) if (!setup.devTools && !setup.tests)
properties << "\t\tEntryPointSymbol=\"WinMainCRTStartup\"\n"; properties << "\t\tEntryPointSymbol=\"WinMainCRTStartup\"\n";
properties << "\t\tAdditionalLibraryDirectories=\"$(" << LIBS_DEFINE << ")\\lib\\" << ((bits == 32) ? "x86" : "x64") << "\"\n" properties << "\t\tAdditionalLibraryDirectories=\"$(" << LIBS_DEFINE << ")\\lib\\" << ((bits == 32) ? "x86" : "x64") << "\"\n"

View File

@ -10,6 +10,8 @@ if "%~1"=="/all" goto all
if "%~1"=="/ALL" goto all if "%~1"=="/ALL" goto all
if "%~1"=="/tools" goto tools if "%~1"=="/tools" goto tools
if "%~1"=="/TOOLS" goto tools if "%~1"=="/TOOLS" goto tools
if "%~1"=="/tests" goto tests
if "%~1"=="/TESTS" goto tests
if "%~1"=="/clean" goto clean_check if "%~1"=="/clean" goto clean_check
if "%~1"=="/CLEAN" goto clean_check if "%~1"=="/CLEAN" goto clean_check
if "%~1"=="/help" goto command_help if "%~1"=="/help" goto command_help
@ -70,6 +72,13 @@ echo.
create_project ..\.. --tools --msvc --msvc-version 10 create_project ..\.. --tools --msvc --msvc-version 10
goto done goto done
:tests
echo.
echo Creating tests project files
echo.
create_project ..\.. --tests --msvc --msvc-version 10
goto done
:clean_check :clean_check
echo. echo.
set cleananswer=N set cleananswer=N
@ -88,6 +97,7 @@ del /Q *.props > NUL 2>&1
del /Q *.sln* > NUL 2>&1 del /Q *.sln* > NUL 2>&1
del /Q scummvm* > NUL 2>&1 del /Q scummvm* > NUL 2>&1
del /Q devtools* > NUL 2>&1 del /Q devtools* > NUL 2>&1
del /Q test_runner.cpp > NUL 2>&1
goto done goto done
:done :done

View File

@ -10,6 +10,8 @@ if "%~1"=="/all" goto all
if "%~1"=="/ALL" goto all if "%~1"=="/ALL" goto all
if "%~1"=="/tools" goto tools if "%~1"=="/tools" goto tools
if "%~1"=="/TOOLS" goto tools if "%~1"=="/TOOLS" goto tools
if "%~1"=="/tests" goto tests
if "%~1"=="/TESTS" goto tests
if "%~1"=="/clean" goto clean_check if "%~1"=="/clean" goto clean_check
if "%~1"=="/CLEAN" goto clean_check if "%~1"=="/CLEAN" goto clean_check
if "%~1"=="/help" goto command_help if "%~1"=="/help" goto command_help
@ -70,6 +72,13 @@ echo.
create_project ..\.. --tools --msvc --msvc-version 11 create_project ..\.. --tools --msvc --msvc-version 11
goto done goto done
:tests
echo.
echo Creating tests project files
echo.
create_project ..\.. --tests --msvc --msvc-version 11
goto done
:clean_check :clean_check
echo. echo.
set cleananswer=N set cleananswer=N
@ -88,6 +97,7 @@ del /Q *.props > NUL 2>&1
del /Q *.sln* > NUL 2>&1 del /Q *.sln* > NUL 2>&1
del /Q scummvm* > NUL 2>&1 del /Q scummvm* > NUL 2>&1
del /Q devtools* > NUL 2>&1 del /Q devtools* > NUL 2>&1
del /Q test_runner.cpp > NUL 2>&1
goto done goto done
:done :done

View File

@ -10,6 +10,8 @@ if "%~1"=="/all" goto all
if "%~1"=="/ALL" goto all if "%~1"=="/ALL" goto all
if "%~1"=="/tools" goto tools if "%~1"=="/tools" goto tools
if "%~1"=="/TOOLS" goto tools if "%~1"=="/TOOLS" goto tools
if "%~1"=="/tests" goto tests
if "%~1"=="/TESTS" goto tests
if "%~1"=="/clean" goto clean_check if "%~1"=="/clean" goto clean_check
if "%~1"=="/CLEAN" goto clean_check if "%~1"=="/CLEAN" goto clean_check
if "%~1"=="/help" goto command_help if "%~1"=="/help" goto command_help
@ -70,6 +72,13 @@ echo.
create_project ..\.. --tools --msvc --msvc-version 12 create_project ..\.. --tools --msvc --msvc-version 12
goto done goto done
:tests
echo.
echo Creating tests project files
echo.
create_project ..\.. --tests --msvc --msvc-version 12
goto done
:clean_check :clean_check
echo. echo.
set cleananswer=N set cleananswer=N
@ -88,6 +97,7 @@ del /Q *.props > NUL 2>&1
del /Q *.sln* > NUL 2>&1 del /Q *.sln* > NUL 2>&1
del /Q scummvm* > NUL 2>&1 del /Q scummvm* > NUL 2>&1
del /Q devtools* > NUL 2>&1 del /Q devtools* > NUL 2>&1
del /Q test_runner.cpp > NUL 2>&1
goto done goto done
:done :done

View File

@ -10,6 +10,8 @@ if "%~1"=="/all" goto all
if "%~1"=="/ALL" goto all if "%~1"=="/ALL" goto all
if "%~1"=="/tools" goto tools if "%~1"=="/tools" goto tools
if "%~1"=="/TOOLS" goto tools if "%~1"=="/TOOLS" goto tools
if "%~1"=="/tests" goto tests
if "%~1"=="/TESTS" goto tests
if "%~1"=="/clean" goto clean_check if "%~1"=="/clean" goto clean_check
if "%~1"=="/CLEAN" goto clean_check if "%~1"=="/CLEAN" goto clean_check
if "%~1"=="/help" goto command_help if "%~1"=="/help" goto command_help
@ -70,6 +72,13 @@ echo.
create_project ..\.. --tools --msvc --msvc-version 8 create_project ..\.. --tools --msvc --msvc-version 8
goto done goto done
:tests
echo.
echo Creating tests project files
echo.
create_project ..\.. --tests --msvc --msvc-version 8
goto done
:clean_check :clean_check
echo. echo.
set cleananswer=N set cleananswer=N
@ -88,6 +97,7 @@ del /Q *.vsprops > NUL 2>&1
del /Q *.sln* > NUL 2>&1 del /Q *.sln* > NUL 2>&1
del /Q scummvm* > NUL 2>&1 del /Q scummvm* > NUL 2>&1
del /Q devtools* > NUL 2>&1 del /Q devtools* > NUL 2>&1
del /Q test_runner.cpp
goto done goto done
:done :done

View File

@ -10,6 +10,8 @@ if "%~1"=="/all" goto all
if "%~1"=="/ALL" goto all if "%~1"=="/ALL" goto all
if "%~1"=="/tools" goto tools if "%~1"=="/tools" goto tools
if "%~1"=="/TOOLS" goto tools if "%~1"=="/TOOLS" goto tools
if "%~1"=="/tests" goto tests
if "%~1"=="/TESTS" goto tests
if "%~1"=="/clean" goto clean_check if "%~1"=="/clean" goto clean_check
if "%~1"=="/CLEAN" goto clean_check if "%~1"=="/CLEAN" goto clean_check
if "%~1"=="/help" goto command_help if "%~1"=="/help" goto command_help
@ -70,6 +72,13 @@ echo.
create_project ..\.. --tools --msvc --msvc-version 9 create_project ..\.. --tools --msvc --msvc-version 9
goto done goto done
:tests
echo.
echo Creating tests project files
echo.
create_project ..\.. --tests --msvc --msvc-version 9
goto done
:clean_check :clean_check
echo. echo.
set cleananswer=N set cleananswer=N
@ -88,6 +97,7 @@ del /Q *.vsprops > NUL 2>&1
del /Q *.sln* > NUL 2>&1 del /Q *.sln* > NUL 2>&1
del /Q scummvm* > NUL 2>&1 del /Q scummvm* > NUL 2>&1
del /Q devtools* > NUL 2>&1 del /Q devtools* > NUL 2>&1
del /Q test_runner.cpp
goto done goto done
:done :done