/* * Copyright (C) 2002 Patrik Stridvall * Copyright (C) 2005 Royce Mitchell III * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #ifdef _MSC_VER #pragma warning ( disable : 4786 ) #endif//_MSC_VER #include #include #include #include "msvc.h" using std::string; using std::vector; #ifdef OUT #undef OUT #endif//OUT void MSVCBackend::_generate_vcproj ( const Module& module ) { size_t i; // TODO FIXME wine hack? //const bool wine = false; string vcproj_file = VcprojFileName(module); printf ( "Creating MSVC.NET project: '%s'\n", vcproj_file.c_str() ); FILE* OUT = fopen ( vcproj_file.c_str(), "wb" ); vector imports; string module_type = GetExtension(module.GetTargetName()); bool lib = (module.type == ObjectLibrary) || (module_type == ".lib") || (module_type == ".a"); bool dll = (module_type == ".dll") || (module_type == ".cpl"); bool exe = (module_type == ".exe"); bool sys = (module_type == ".sys"); string path_basedir = module.GetPathToBaseDir (); string intenv = Environment::GetIntermediatePath (); string outenv = Environment::GetOutputPath (); string outdir; string intdir; if ( intenv == "obj-i386" ) intdir = path_basedir + "obj-i386"; /* append relative dir from project dir */ else intdir = intenv; if ( outenv == "output-i386" ) outdir = path_basedir + "output-i386"; else outdir = outenv; // TODO FIXME - need more checks here for 'sys' and possibly 'drv'? bool console = exe && (module.type == Win32CUI); // TODO FIXME - not sure if the count here is right... int parts = 0; const char* p = strpbrk ( vcproj_file.c_str(), "/\\" ); while ( p ) { ++parts; p = strpbrk ( p+1, "/\\" ); } string msvc_wine_dir = ".."; while ( --parts ) msvc_wine_dir += "\\.."; string wine_include_dir = msvc_wine_dir + "\\include"; //$progress_current++; //$output->progress("$dsp_file (file $progress_current of $progress_max)"); string vcproj_path = module.GetBasePath(); vector source_files, resource_files, includes, libraries, defines; vector ifs_list; ifs_list.push_back ( &module.project.non_if_data ); ifs_list.push_back ( &module.non_if_data ); // this is a define in MinGW w32api, but not Microsoft's headers defines.push_back ( "_CRT_SECURE_NO_DEPRECATE" ); defines.push_back ( "_CRT_NON_CONFORMING_SWPRINTFS" ); defines.push_back ( "STDCALL=__stdcall" ); string baseaddr; while ( ifs_list.size() ) { const IfableData& data = *ifs_list.back(); ifs_list.pop_back(); // TODO FIXME - refactor needed - we're discarding if conditions for ( i = 0; i < data.ifs.size(); i++ ) ifs_list.push_back ( &data.ifs[i]->data ); const vector& files = data.files; for ( i = 0; i < files.size(); i++ ) { // TODO FIXME - do we want the full path of the file here? string file = string(".") + &files[i]->name[vcproj_path.size()]; if ( !stricmp ( Right(file,3).c_str(), ".rc" ) ) resource_files.push_back ( file ); else source_files.push_back ( file ); } const vector& incs = data.includes; for ( i = 0; i < incs.size(); i++ ) { // explicitly omit win32api directories if ( !strncmp(incs[i]->directory.c_str(), "w32api", 6 ) ) continue; // explicitly omit include/wine directories if ( !strncmp(incs[i]->directory.c_str(), "include\\wine", 12 ) ) continue; string path = Path::RelativeFromDirectory ( incs[i]->directory, module.GetBasePath() ); includes.push_back ( path ); } const vector& libs = data.libraries; for ( i = 0; i < libs.size(); i++ ) { #if 0 // this code is deactivated untill the tree builds fine with msvc // --- is appended to each library path which is later // replaced by the configuration // i.e. ../output-i386/lib/rtl/---/rtl.lib becomes // ../output-i386/lib/rtl/Debug/rtl.lib // etc libs[i]->importedModule-> string libpath = outdir + "\\" + libs[i]->importedModule->GetBasePath() + "\\---\\" + libs[i]->name + ".lib"; libraries.push_back ( libpath ); #else libraries.push_back ( libs[i]->name + ".lib" ); #endif } const vector& defs = data.defines; for ( i = 0; i < defs.size(); i++ ) { if ( defs[i]->value[0] ) defines.push_back ( defs[i]->name + "=" + defs[i]->value ); else defines.push_back ( defs[i]->name ); } for ( i = 0; i < data.properties.size(); i++ ) { Property& prop = *data.properties[i]; if ( strstr ( module.baseaddress.c_str(), prop.name.c_str() ) ) baseaddr = prop.value; } } vector header_files; bool no_cpp = true; bool no_msvc_headers = true; std::vector cfgs; cfgs.push_back ( "Debug" ); cfgs.push_back ( "Release" ); cfgs.push_back ( "Speed" ); if (!no_cpp) { std::vector _cfgs; for ( i = 0; i < cfgs.size(); i++ ) { _cfgs.push_back ( cfgs[i] + " C" ); _cfgs.push_back ( cfgs[i] + " C++" ); } cfgs.resize(0); cfgs = _cfgs; } if (!no_msvc_headers) { std::vector _cfgs; for ( i = 0; i < cfgs.size(); i++ ) { _cfgs.push_back ( cfgs[i] + " MSVC Headers" ); _cfgs.push_back ( cfgs[i] + " Wine Headers" ); } cfgs.resize(0); cfgs = _cfgs; } string default_cfg = cfgs.back(); string include_string; fprintf ( OUT, "\r\n" ); fprintf ( OUT, "\r\n" ); fprintf ( OUT, "\t\r\n" ); fprintf ( OUT, "\t\t\r\n" ); fprintf ( OUT, "\t\r\n" ); fprintf ( OUT, "\t\r\n" ); fprintf ( OUT, "\t\t\r\n", path.c_str() ); fprintf ( OUT, "\t\r\n" ); int n = 0; std::string output_dir; fprintf ( OUT, "\t\r\n" ); for ( size_t icfg = 0; icfg < cfgs.size(); icfg++ ) { std::string& cfg = cfgs[icfg]; bool debug = strstr ( cfg.c_str(), "Debug" ) != NULL; bool speed = strstr ( cfg.c_str(), "Speed" ) != NULL; bool release = (!debug && !speed ); //bool msvc_headers = ( 0 != strstr ( cfg.c_str(), "MSVC Headers" ) ); fprintf ( OUT, "\t\t\r\n" ); fprintf ( OUT, "\t\t\t 0 ) fprintf ( OUT, ";" ); defines[i] = _replace_str(defines[i], "\"","""); fprintf ( OUT, "%s", defines[i].c_str() ); } fprintf ( OUT, "\"\r\n" ); fprintf ( OUT, "\t\t\t\tMinimalRebuild=\"%s\"\r\n", speed ? "FALSE" : "TRUE" ); fprintf ( OUT, "\t\t\t\tBasicRuntimeChecks=\"%s\"\r\n", debug ? "3" : "0" ); fprintf ( OUT, "\t\t\t\tRuntimeLibrary=\"5\"\r\n" ); fprintf ( OUT, "\t\t\t\tBufferSecurityCheck=\"%s\"\r\n", debug ? "TRUE" : "FALSE" ); fprintf ( OUT, "\t\t\t\tEnableFunctionLevelLinking=\"%s\"\r\n", debug ? "TRUE" : "FALSE" ); if ( module.pch != NULL ) { fprintf ( OUT, "\t\t\t\tUsePrecompiledHeader=\"2\"\r\n" ); string pch_path = Path::RelativeFromDirectory ( module.pch->file.name, module.GetBasePath() ); fprintf ( OUT, "\t\t\t\tPrecompiledHeaderThrough=\"%s\"\r\n", pch_path.c_str() ); } else { fprintf ( OUT, "\t\t\t\tUsePrecompiledHeader=\"0\"\r\n" ); } fprintf ( OUT, "\t\t\t\tWholeProgramOptimization=\"%s\"\r\n", release ? "TRUE" : "FALSE"); if ( release ) { fprintf ( OUT, "\t\t\t\tFavorSizeOrSpeed=\"1\"\r\n" ); fprintf ( OUT, "\t\t\t\tStringPooling=\"true\"\r\n" ); } fprintf ( OUT, "\t\t\t\tEnablePREfast=\"%s\"\r\n", debug ? "TRUE" : "FALSE"); fprintf ( OUT, "\t\t\t\tDisableSpecificWarnings=\"4201;4127;4214\"\r\n" ); fprintf ( OUT, "\t\t\t\tWarningLevel=\"%s\"\r\n", release ? "0" : "4" ); fprintf ( OUT, "\t\t\t\tDetect64BitPortabilityProblems=\"%s\"\r\n", release ? "FALSE" : "TRUE"); if ( !module.cplusplus ) fprintf ( OUT, "\t\t\t\tCompileAs=\"1\"\r\n" ); fprintf ( OUT, "\t\t\t\tDebugInformationFormat=\"%s\"/>\r\n", speed ? "0" : "4"); fprintf ( OUT, "\t\t\t\r\n" ); if ( lib ) { fprintf ( OUT, "\t\t\t\r\n", module.name.c_str() ); } else { fprintf ( OUT, "\t\t\t 0 ) fprintf ( OUT, " " ); #if 0 // this code is deactivated untill // msvc can build the whole tree string libpath = libraries[i].c_str(); libpath.replace (libpath.find("---"), //See HACK 3, cfg); fprintf ( OUT, "%s", libpath.c_str() ); #else fprintf ( OUT, "%s", libraries[i].c_str() ); #endif } fprintf ( OUT, "\"\r\n" ); fprintf ( OUT, "\t\t\t\tOutputFile=\"$(OutDir)/%s%s\"\r\n", module.name.c_str(), module_type.c_str() ); fprintf ( OUT, "\t\t\t\tLinkIncremental=\"%d\"\r\n", debug ? 2 : 1 ); fprintf ( OUT, "\t\t\t\tGenerateDebugInformation=\"%s\"\r\n", speed ? "FALSE" : "TRUE" ); if ( debug ) fprintf ( OUT, "\t\t\t\tProgramDatabaseFile=\"$(OutDir)/%s.pdb\"\r\n", module.name.c_str() ); if ( sys ) { fprintf ( OUT, "\t\t\t\tAdditionalOptions=\" /DRIVER /ALIGN:0x20 /SUBSYSTEM:NATIVE /SECTION:INIT,D /NODEFAULTLIB /IGNORE:4001,4037,4039,4065,4070,4078,4087,4089,4096\"\r\n" ); fprintf ( OUT, "\t\t\t\tIgnoreAllDefaultLibraries=\"TRUE\"\r\n" ); fprintf ( OUT, "\t\t\t\tEntryPointSymbol=\"%s\"\r\n", module.entrypoint == "" ? "DriverEntry" : module.entrypoint.c_str ()); fprintf ( OUT, "\t\t\t\tBaseAddress=\"%s\"\r\n", baseaddr == "" ? "0x10000" : baseaddr.c_str ()); } else if ( exe ) { if ( module.type == Kernel ) { fprintf ( OUT, "\t\t\t\tAdditionalOptions=\" /SUBSYSTEM:NATIVE /NODEFAULTLIB /SECTION:INIT,D /ALIGN:0x80\"\r\n" ); fprintf ( OUT, "\t\t\t\tIgnoreAllDefaultLibraries=\"TRUE\"\r\n" ); fprintf ( OUT, "\t\t\t\tEntryPointSymbol=\"KiSystemStartup\"\r\n" ); fprintf ( OUT, "\t\t\t\tBaseAddress=\"%s\"\r\n", baseaddr.c_str ()); } else if ( module.type == NativeCUI ) { fprintf ( OUT, "\t\t\t\tAdditionalOptions=\" /SUBSYSTEM:NATIVE /NODEFAULTLIB /ALIGN:0x20\"\r\n" ); fprintf ( OUT, "\t\t\t\tIgnoreAllDefaultLibraries=\"TRUE\"\r\n" ); fprintf ( OUT, "\t\t\t\tEntryPointSymbol=\"NtProcessStartup\"\r\n" ); fprintf ( OUT, "\t\t\t\tBaseAddress=\"%s\"\r\n", baseaddr.c_str ()); } else if ( module.type == Win32CUI || module.type == Win32GUI ) { fprintf ( OUT, "\t\t\t\tSubSystem=\"%d\"\r\n", console ? 1 : 2 ); } } else if ( dll ) { fprintf ( OUT, "\t\t\t\tEntryPointSymbol=\"%s\"\r\n", module.entrypoint == "" ? "DllMain" : module.entrypoint.c_str ()); fprintf ( OUT, "\t\t\t\tBaseAddress=\"%s\"\r\n", baseaddr == "" ? "0x40000" : baseaddr.c_str ()); } fprintf ( OUT, "\t\t\t\tTargetMachine=\"%d\"/>\r\n", 1 ); } fprintf ( OUT, "\t\t\t\r\n " ); fprintf ( OUT, "\t\t\t\r\n" ); fprintf ( OUT, "\t\t\t\r\n" ); fprintf ( OUT, "\t\t\t\r\n" ); fprintf ( OUT, "\t\t\t\r\n" ); fprintf ( OUT, "\t\t\t\r\n" ); fprintf ( OUT, "\t\t\t\r\n" ); fprintf ( OUT, "\t\t\r\n" ); n++; } fprintf ( OUT, "\t\r\n" ); fprintf ( OUT, "\t\r\n" ); // Source files fprintf ( OUT, "\t\t\r\n" ); for ( size_t isrcfile = 0; isrcfile < source_files.size(); isrcfile++ ) { string source_file = DosSeparator(source_files[isrcfile]); fprintf ( OUT, "\t\t\t\r\n", source_file.c_str() ); for ( size_t iconfig = 0; iconfig < cfgs.size(); iconfig++ ) { std::string& config = cfgs[iconfig]; if (( isrcfile == 0 ) && ( module.pch != NULL )) { /* little hack to speed up PCH */ fprintf ( OUT, "\t\t\t\t\r\n" ); fprintf ( OUT, "\t\t\t\t\t\r\n" ); fprintf ( OUT, "\t\t\t\t\r\n" ); } if (configuration.VSProjectVersion < "8.00") { if ((source_file.find(".idl") != string::npos) || ((source_file.find(".asm") != string::npos || tolower(source_file.at(source_file.size() - 1)) == 's'))) { fprintf ( OUT, "\t\t\t\t\r\n" ); fprintf ( OUT, "\t\t\t\t\t\r\n" ); } else if ((source_file.find(".asm") != string::npos || tolower(source_file.at(source_file.size() - 1)) == 's')) { fprintf ( OUT, "\t\t\t\t\t\tName=\"VCCustomBuildTool\"\r\n" ); fprintf ( OUT, "\t\t\t\t\t\tCommandLine=\"cl /E "$(InputPath)" %s /D__ASM__ | as -o "$(OutDir)\\(InputName).obj"\"\r\n",include_string.c_str() ); fprintf ( OUT, "\t\t\t\t\t\tOutputs=\"$(OutDir)\\(InputName).obj\"/>\r\n" ); } fprintf ( OUT, "\t\t\t\t\r\n" ); } } } fprintf ( OUT, "\t\t\t\r\n" ); } fprintf ( OUT, "\t\t\r\n" ); // Header files fprintf ( OUT, "\t\t\r\n" ); for ( i = 0; i < header_files.size(); i++ ) { const string& header_file = header_files[i]; fprintf ( OUT, "\t\t\t\r\n", header_file.c_str() ); fprintf ( OUT, "\t\t\t\r\n" ); } fprintf ( OUT, "\t\t\r\n" ); // Resource files fprintf ( OUT, "\t\t\r\n" ); for ( i = 0; i < resource_files.size(); i++ ) { const string& resource_file = resource_files[i]; fprintf ( OUT, "\t\t\t\r\n", resource_file.c_str() ); fprintf ( OUT, "\t\t\t\r\n" ); } fprintf ( OUT, "\t\t\r\n" ); fprintf ( OUT, "\t\r\n" ); fprintf ( OUT, "\t\r\n" ); fprintf ( OUT, "\t\r\n" ); fprintf ( OUT, "\r\n" ); fclose(OUT); } std::string MSVCBackend::_replace_str(std::string string1, const std::string &find_str, const std::string &replace_str) { std::string::size_type pos = string1.find(find_str, 0); int intLen = find_str.length(); while(std::string::npos != pos) { string1.replace(pos, intLen, replace_str); pos = string1.find(find_str, intLen + pos); } return string1; } std::string MSVCBackend::_get_solution_verion ( void ) { string version; if (configuration.VSProjectVersion.empty()) configuration.VSProjectVersion = MS_VS_DEF_VERSION; if (configuration.VSProjectVersion == "7.00") version = "7.00"; if (configuration.VSProjectVersion == "7.10") version = "8.00"; if (configuration.VSProjectVersion == "8.00") version = "9.00"; return version; } void MSVCBackend::_generate_rules_file ( FILE* OUT ) { fprintf ( OUT, "\r\n" ); fprintf ( OUT, "\r\n" ); fprintf ( OUT, "\t\r\n" ); fprintf ( OUT, "\t\t\r\n" ); fprintf ( OUT, "\t\t\t\r\n" ); fprintf ( OUT, "\t\t\t\r\n" ); fprintf ( OUT, "\t\t\r\n" ); fprintf ( OUT, "\t\r\n" ); fprintf ( OUT, "\r\n" ); } void MSVCBackend::_generate_sln_header ( FILE* OUT ) { fprintf ( OUT, "Microsoft Visual Studio Solution File, Format Version %s\r\n", _get_solution_verion().c_str() ); fprintf ( OUT, "# Visual Studio 2005\r\n" ); fprintf ( OUT, "\r\n" ); } void MSVCBackend::_generate_sln_project ( FILE* OUT, const Module& module, std::string vcproj_file, std::string sln_guid, std::string vcproj_guid, const std::vector& dependencies ) { vcproj_file = DosSeparator ( std::string(".\\") + vcproj_file ); fprintf ( OUT, "Project(\"%s\") = \"%s\", \"%s\", \"%s\"\r\n", sln_guid.c_str() , module.name.c_str(), vcproj_file.c_str(), vcproj_guid.c_str() ); //FIXME: only omit ProjectDependencies in VS 2005 when there are no dependencies //NOTE: VS 2002 do not use ProjectSection; it uses GlobalSection instead if ((configuration.VSProjectVersion == "7.10") || (dependencies.size() > 0)) { fprintf ( OUT, "\tProjectSection(ProjectDependencies) = postProject\r\n" ); for ( size_t i = 0; i < dependencies.size(); i++ ) { Dependency& dependency = *dependencies[i]; fprintf ( OUT, "\t\t%s = %s\r\n", dependency.module.guid.c_str(), dependency.module.guid.c_str() ); } fprintf ( OUT, "\tEndProjectSection\r\n" ); } fprintf ( OUT, "EndProject\r\n" ); } void MSVCBackend::_generate_sln_footer ( FILE* OUT ) { fprintf ( OUT, "Global\r\n" ); fprintf ( OUT, "\tGlobalSection(SolutionConfiguration) = preSolution\r\n" ); fprintf ( OUT, "\t\tDebug = Debug\r\n" ); fprintf ( OUT, "\t\tRelease = Release\r\n" ); fprintf ( OUT, "\tEndGlobalSection\r\n" ); fprintf ( OUT, "\tGlobalSection(ProjectConfiguration) = postSolution\r\n" ); for ( size_t i = 0; i < ProjectNode.modules.size(); i++ ) { Module& module = *ProjectNode.modules[i]; std::string guid = module.guid; _generate_sln_configurations ( OUT, guid.c_str() ); } fprintf ( OUT, "\tEndGlobalSection\r\n" ); fprintf ( OUT, "\tGlobalSection(ExtensibilityGlobals) = postSolution\r\n" ); fprintf ( OUT, "\tEndGlobalSection\r\n" ); fprintf ( OUT, "\tGlobalSection(ExtensibilityAddIns) = postSolution\r\n" ); fprintf ( OUT, "\tEndGlobalSection\r\n" ); if (configuration.VSProjectVersion == "7.00") { fprintf ( OUT, "\tGlobalSection(ProjectDependencies) = postSolution\r\n" ); //FIXME: Add dependencies for VS 2002 fprintf ( OUT, "\tEndGlobalSection\r\n" ); } if (configuration.VSProjectVersion == "8.00") { fprintf ( OUT, "\tGlobalSection(SolutionProperties) = preSolution\r\n" ); fprintf ( OUT, "\t\tHideSolutionNode = FALSE\r\n" ); fprintf ( OUT, "\tEndGlobalSection\r\n" ); } fprintf ( OUT, "EndGlobal\r\n" ); fprintf ( OUT, "\r\n" ); } void MSVCBackend::_generate_sln_configurations ( FILE* OUT, std::string vcproj_guid ) { fprintf ( OUT, "\t\t%s.Debug.ActiveCfg = Debug|Win32\r\n", vcproj_guid.c_str() ); fprintf ( OUT, "\t\t%s.Debug.Build.0 = Debug|Win32\r\n", vcproj_guid.c_str() ); fprintf ( OUT, "\t\t%s.Debug.Release.ActiveCfg = Release|Win32\r\n", vcproj_guid.c_str() ); fprintf ( OUT, "\t\t%s.Debug.Release.Build.0 = Release|Win32\r\n", vcproj_guid.c_str() ); } void MSVCBackend::_generate_sln ( FILE* OUT ) { string sln_guid = "{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}"; vector guids; _generate_sln_header(OUT); // TODO FIXME - is it necessary to sort them? for ( size_t i = 0; i < ProjectNode.modules.size(); i++ ) { Module& module = *ProjectNode.modules[i]; std::string vcproj_file = VcprojFileName ( module ); _generate_sln_project ( OUT, module, vcproj_file, sln_guid, module.guid, module.dependencies ); } _generate_sln_footer ( OUT ); }