1 #include <msp/core/application.h>
2 #include <msp/fs/utils.h>
3 #include <msp/io/print.h>
4 #include <msp/strings/utils.h>
6 #include "csourcefile.h"
7 #include "executable.h"
8 #include "sourcepackage.h"
9 #include "vcxprojectfile.h"
10 #include "vcxprojectgenerator.h"
15 VcxProjectGenerator::VcxProjectGenerator(Builder &b):
19 Target *VcxProjectGenerator::create_target(const list<Target *> &, const string &)
21 throw logic_error("Not implemented");
24 Task *VcxProjectGenerator::run(const Target &target) const
26 const VcxProjectFile &project = dynamic_cast<const VcxProjectFile &>(target);
27 Worker *worker = new Worker(project);
28 return new InternalTask(worker);
32 VcxProjectGenerator::Worker::Worker(const VcxProjectFile &t):
36 void VcxProjectGenerator::Worker::main()
38 const SourcePackage &spkg = *target.get_package();
39 Builder &builder = spkg.get_builder();
41 IO::BufferedFile out(target.get_path().str(), IO::M_WRITE);
42 IO::print(out, "<Project DefaultTargets=\"Build\" ToolsVersion=\"4.0\" xmlns=\"http://schemas.microsoft.com/developer/msbuild/2003\">\n");
44 IO::print(out, "\t<ItemGroup Label=\"ProjectConfigurations\">\n");
45 vector<string> build_types = builder.get_build_types();
46 const char *platforms[] = { "Win32", "x64", 0 };
47 for(const char **i=platforms; *i; ++i)
48 for(vector<string>::const_iterator j=build_types.begin(); j!=build_types.end(); ++j)
50 IO::print(out, "\t\t<ProjectConfiguration Include=\"%s|%s\">\n", *j, *i);
51 IO::print(out, "\t\t\t<Configuration>%s</Configuration>\n", *j);
52 IO::print(out, "\t\t\t<Platform>%s</Platform>\n", *i);
53 IO::print(out, "\t\t</ProjectConfiguration>\n");
55 IO::print(out, "\t</ItemGroup>\n");
57 IO::print(out, "\t<PropertyGroup Label=\"Globals\">\n");
58 IO::print(out, "\t\t<VCProjectVersion>15.0</VCProjectVersion>\n");
59 IO::print(out, "\t\t<Keyword>MakeFileProj</Keyword>\n");
60 IO::print(out, "\t</PropertyGroup>\n");
62 IO::print(out, "\t<Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.Default.props\" />\n");
64 const Target *world = builder.get_build_graph().get_target("world");
65 const Target::Dependencies &world_deps = world->get_dependencies();
66 const Executable *exe = 0;
67 for(Target::Dependencies::const_iterator i=world_deps.begin(); (!exe && i!=world_deps.end()); ++i)
68 if((*i)->get_package()==&spkg)
69 exe = dynamic_cast<const Executable *>(*i);
71 const char *argv0 = Application::get_argv0();
72 const string &toolchain = builder.get_current_arch().get_toolchain();
73 for(const char **i=platforms; *i; ++i)
74 for(vector<string>::const_iterator j=build_types.begin(); j!=build_types.end(); ++j)
76 string base_cmd = format("%s --arch=%s-%s --build-type=%s --prefix=%s", argv0, *i, toolchain, *j, builder.get_prefix());
77 IO::print(out, "\t<PropertyGroup Condition=\"'$(Configuration)|$(Platform)'=='%s|%s'\" Label=\"Configuration\">\n", *j, *i);
78 IO::print(out, "\t\t<ConfigurationType>MakeFile</ConfigurationType>\n");
79 IO::print(out, "\t\t<NMakeBuildCommandLine>%s</NMakeBuildCommandLine>\n", base_cmd);
80 IO::print(out, "\t\t<NMakeCleanCommandLine>%s -c</NMakeCleanCommandLine>\n", base_cmd);
81 IO::print(out, "\t\t<NMakeReBuildCommandLine>%s -B</NMakeReBuildCommandLine>\n", base_cmd);
83 IO::print(out, "\t\t<NMakeOutput>%s</NMakeOutput>\n", exe->get_path());
84 IO::print(out, "\t</PropertyGroup>\n");
87 IO::print(out, "\t<Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.props\" />\n");
89 const BuildGraph::TargetMap &targets = builder.get_build_graph().get_targets();
90 vector<const FileTarget *> sources;
91 vector<const FileTarget *> includes;
92 vector<const FileTarget *> others;
94 for(BuildGraph::TargetMap::const_iterator i=targets.begin(); i!=targets.end(); ++i)
95 if(i->second->get_package()==&spkg)
97 if(i->second->is_buildable())
100 i->second->collect_build_info(tgt_binfo);
101 build_info.update_from(tgt_binfo, BuildInfo::CHAINED);
103 else if(const FileTarget *file = dynamic_cast<const FileTarget *>(i->second))
105 if(dynamic_cast<const CSourceFile *>(file))
107 string ext = tolower(FS::extpart(FS::basename(file->get_path())));
108 if(ext==".h" || ext==".hpp")
109 includes.push_back(file);
111 sources.push_back(file);
114 others.push_back(file);
118 if(!build_info.incpath.empty())
120 IO::print(out, "\t<PropertyGroup>\n");
122 for(BuildInfo::PathList::const_iterator i=build_info.incpath.begin(); i!=build_info.incpath.end(); ++i)
123 append(path_str, ";", i->str());
124 IO::print(out, "\t\t<NMakeIncludeSearchPath>%s</NMakeIncludeSearchPath>\n", path_str);
125 IO::print(out, "\t</PropertyGroup>\n");
128 IO::print(out, "\t<ItemGroup>\n");
129 for(vector<const FileTarget *>::const_iterator i=sources.begin(); i!=sources.end(); ++i)
130 IO::print(out, "\t\t<ClCompile Include=\"%s\" />\n", (*i)->get_path());
131 IO::print(out, "\t</ItemGroup>\n");
133 IO::print(out, "\t<ItemGroup>\n");
134 for(vector<const FileTarget *>::const_iterator i=includes.begin(); i!=includes.end(); ++i)
135 IO::print(out, "\t\t<ClInclude Include=\"%s\" />\n", (*i)->get_path());
136 IO::print(out, "\t</ItemGroup>\n");
138 IO::print(out, "\t<ItemGroup>\n");
139 for(vector<const FileTarget *>::const_iterator i=others.begin(); i!=others.end(); ++i)
140 IO::print(out, "\t\t<None Include=\"%s\" />\n", (*i)->get_path());
141 IO::print(out, "\t</ItemGroup>\n");
143 IO::print(out, "\t<Import Project=\"$(VCTargetsPath)\\Microsoft.Cpp.targets\" />\n");
144 IO::print(out, "</Project>\n");
146 status = Task::SUCCESS;