fix creation of shlwapi vcproject file
[reactos.git] / reactos / tools / rbuild / backend / msvc / vcprojmaker.cpp
1 /*
2 * Copyright (C) 2002 Patrik Stridvall
3 * Copyright (C) 2005 Royce Mitchell III
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18 */
19
20 #ifdef _MSC_VER
21 #pragma warning ( disable : 4786 )
22 #endif//_MSC_VER
23
24 #include <string>
25 #include <vector>
26
27 #include <stdio.h>
28
29 #include "msvc.h"
30
31 using std::string;
32 using std::vector;
33
34 void
35 MSVCBackend::_generate_vcproj ( const Module& module )
36 {
37 size_t i;
38 // TODO FIXME wine hack?
39 //const bool wine = false;
40
41 string vcproj_file = VcprojFileName(module);
42 printf ( "Creating MSVC.NET project: '%s'\n", vcproj_file.c_str() );
43 FILE* OUT = fopen ( vcproj_file.c_str(), "wb" );
44
45 vector<string> imports;
46 for ( i = 0; i < module.non_if_data.libraries.size(); i++ )
47 {
48 imports.push_back ( module.non_if_data.libraries[i]->name );
49 }
50
51 string module_type = GetExtension(module.GetTargetName());
52 bool lib = (module_type == ".lib") || (module_type == ".a");
53 bool dll = (module_type == ".dll") || (module_type == ".cpl");
54 bool exe = (module_type == ".exe");
55 // TODO FIXME - need more checks here for 'sys' and possibly 'drv'?
56
57 bool console = exe && (module.type == Win32CUI);
58
59 // TODO FIXME - not sure if the count here is right...
60 int parts = 0;
61 const char* p = strpbrk ( vcproj_file.c_str(), "/\\" );
62 while ( p )
63 {
64 ++parts;
65 p = strpbrk ( p+1, "/\\" );
66 }
67 string msvc_wine_dir = "..";
68 while ( --parts )
69 msvc_wine_dir += "\\..";
70
71 string wine_include_dir = msvc_wine_dir + "\\include";
72
73 //$progress_current++;
74 //$output->progress("$dsp_file (file $progress_current of $progress_max)");
75
76 // TODO FIXME - what's diff. betw. 'c_srcs' and 'source_files'?
77 string vcproj_path = module.GetBasePath();
78 vector<string> c_srcs, source_files, resource_files, includes, libraries, defines;
79 vector<const IfableData*> ifs_list;
80 ifs_list.push_back ( &module.project.non_if_data );
81 ifs_list.push_back ( &module.non_if_data );
82
83 // this is a define in MinGW w32api, but not Microsoft's headers
84 defines.push_back ( "STDCALL=__stdcall" );
85
86 while ( ifs_list.size() )
87 {
88 const IfableData& data = *ifs_list.back();
89 ifs_list.pop_back();
90 // TODO FIXME - refactor needed - we're discarding if conditions
91 for ( i = 0; i < data.ifs.size(); i++ )
92 ifs_list.push_back ( &data.ifs[i]->data );
93 const vector<File*>& files = data.files;
94 for ( i = 0; i < files.size(); i++ )
95 {
96 // TODO FIXME - do we want the full path of the file here?
97 string file = string(".") + &files[i]->name[vcproj_path.size()];
98
99 source_files.push_back ( file );
100 if ( !stricmp ( Right(file,2).c_str(), ".c" ) )
101 c_srcs.push_back ( file );
102 if ( !stricmp ( Right(file,3).c_str(), ".rc" ) )
103 resource_files.push_back ( file );
104 }
105 const vector<Include*>& incs = data.includes;
106 for ( i = 0; i < incs.size(); i++ )
107 {
108 // explicitly omit win32api directories
109 if ( !strncmp(incs[i]->directory.c_str(), "w32api", 6 ) )
110 continue;
111
112 // explicitly omit include/wine directories
113 if ( !strncmp(incs[i]->directory.c_str(), "include\\wine", 12 ) )
114 continue;
115
116 string path = Path::RelativeFromDirectory (
117 incs[i]->directory,
118 module.GetBasePath() );
119 includes.push_back ( path );
120 }
121 const vector<Library*>& libs = data.libraries;
122 for ( i = 0; i < libs.size(); i++ )
123 {
124 libraries.push_back ( libs[i]->name + ".lib" );
125 }
126 const vector<Define*>& defs = data.defines;
127 for ( i = 0; i < defs.size(); i++ )
128 {
129 if ( defs[i]->value[0] )
130 defines.push_back ( defs[i]->name + "=" + defs[i]->value );
131 else
132 defines.push_back ( defs[i]->name );
133 }
134 }
135
136 vector<string> header_files;
137
138 bool no_cpp = true;
139 bool no_msvc_headers = true;
140
141 std::vector<std::string> cfgs;
142
143 cfgs.push_back ( module.name + " - Win32" );
144
145 if (!no_cpp)
146 {
147 std::vector<std::string> _cfgs;
148 for ( i = 0; i < cfgs.size(); i++ )
149 {
150 _cfgs.push_back ( cfgs[i] + " C" );
151 _cfgs.push_back ( cfgs[i] + " C++" );
152 }
153 cfgs.resize(0);
154 cfgs = _cfgs;
155 }
156
157 if (!no_msvc_headers)
158 {
159 std::vector<std::string> _cfgs;
160 for ( i = 0; i < cfgs.size(); i++ )
161 {
162 _cfgs.push_back ( cfgs[i] + " MSVC Headers" );
163 _cfgs.push_back ( cfgs[i] + " Wine Headers" );
164 }
165 cfgs.resize(0);
166 cfgs = _cfgs;
167 }
168
169 string default_cfg = cfgs.back();
170
171 fprintf ( OUT, "<?xml version=\"1.0\" encoding = \"Windows-1252\"?>\r\n" );
172 fprintf ( OUT, "<VisualStudioProject\r\n" );
173 fprintf ( OUT, "\tProjectType=\"Visual C++\"\r\n" );
174
175 if (configuration.VSProjectVersion.empty())
176 configuration.VSProjectVersion = MS_VS_DEF_VERSION;
177
178 fprintf ( OUT, "\tVersion=\"%s\"\r\n", configuration.VSProjectVersion.c_str() );
179 fprintf ( OUT, "\tName=\"%s\"\r\n", module.name.c_str() );
180 fprintf ( OUT, "\tKeyword=\"Win32Proj\">\r\n" );
181
182 fprintf ( OUT, "\t<Platforms>\r\n" );
183 fprintf ( OUT, "\t\t<Platform\r\n" );
184 fprintf ( OUT, "\t\t\tName=\"Win32\"/>\r\n" );
185 fprintf ( OUT, "\t</Platforms>\r\n" );
186
187 int n = 0;
188
189 std::string output_dir;
190
191 fprintf ( OUT, "\t<Configurations>\r\n" );
192 for ( size_t icfg = 0; icfg < cfgs.size(); icfg++ )
193 {
194 std::string& cfg = cfgs[icfg];
195
196 bool debug = !strstr ( cfg.c_str(), "Release" );
197 //bool msvc_headers = ( 0 != strstr ( cfg.c_str(), "MSVC Headers" ) );
198
199 fprintf ( OUT, "\t\t<Configuration\r\n" );
200 fprintf ( OUT, "\t\t\tName=\"%s|Win32\"\r\n", cfg.c_str() );
201 fprintf ( OUT, "\t\t\tOutputDirectory=\"%s\"\r\n", cfg.c_str() );
202 fprintf ( OUT, "\t\t\tIntermediateDirectory=\"%s\"\r\n", cfg.c_str() );
203 fprintf ( OUT, "\t\t\tConfigurationType=\"%d\"\r\n", exe ? 1 : dll ? 2 : lib ? 4 : -1 );
204 fprintf ( OUT, "\t\t\tCharacterSet=\"2\">\r\n" );
205
206 fprintf ( OUT, "\t\t\t<Tool\r\n" );
207 fprintf ( OUT, "\t\t\t\tName=\"VCCLCompilerTool\"\r\n" );
208 fprintf ( OUT, "\t\t\t\tOptimization=\"%d\"\r\n", debug ? 0 : 2 );
209
210 fprintf ( OUT, "\t\t\t\tAdditionalIncludeDirectories=\"" );
211 bool multiple_includes = false;
212 fprintf ( OUT, "./;" );
213 for ( i = 0; i < includes.size(); i++ )
214 {
215 const string& include = includes[i];
216 if ( strcmp ( include.c_str(), "." ) )
217 {
218 if ( multiple_includes )
219 fprintf ( OUT, ";" );
220 fprintf ( OUT, "%s", include.c_str() );
221 multiple_includes = true;
222 }
223 }
224 fprintf ( OUT, "\"\r\n " );
225
226 if ( debug )
227 {
228 defines.push_back ( "_DEBUG" );
229 if ( lib || exe )
230 {
231 defines.push_back ( "_LIB" );
232 }
233 else
234 {
235 defines.push_back ( "_WINDOWS" );
236 defines.push_back ( "_USRDLL" );
237 }
238 }
239 else
240 {
241 defines.push_back ( "NDEBUG" );
242 if ( lib || exe )
243 {
244 defines.push_back ( "_LIB" );
245 }
246 else
247 {
248 defines.push_back ( "_WINDOWS" );
249 defines.push_back ( "_USRDLL" );
250 }
251 }
252
253 fprintf ( OUT, "\t\t\t\tPreprocessorDefinitions=\"" );
254 for ( i = 0; i < defines.size(); i++ )
255 {
256 if ( i > 0 )
257 fprintf ( OUT, ";" );
258
259 defines[i] = _replace_str(defines[i], "\"","&quot;");
260 fprintf ( OUT, "%s", defines[i].c_str() );
261 }
262 fprintf ( OUT, "\"\r\n" );
263
264 fprintf ( OUT, "\t\t\t\tMinimalRebuild=\"TRUE\"\r\n" );
265 fprintf ( OUT, "\t\t\t\tBasicRuntimeChecks=\"3\"\r\n" );
266 fprintf ( OUT, "\t\t\t\tRuntimeLibrary=\"5\"\r\n" );
267 fprintf ( OUT, "\t\t\t\tBufferSecurityCheck=\"%s\"\r\n", debug ? "TRUE" : "FALSE" );
268 fprintf ( OUT, "\t\t\t\tEnableFunctionLevelLinking=\"%s\"\r\n", debug ? "TRUE" : "FALSE" );
269 fprintf ( OUT, "\t\t\t\tUsePrecompiledHeader=\"0\"\r\n" );
270 fprintf ( OUT, "\t\t\t\tWarningLevel=\"1\"\r\n" );
271 fprintf ( OUT, "\t\t\t\tDetect64BitPortabilityProblems=\"TRUE\"\r\n" );
272 fprintf ( OUT, "\t\t\t\tDebugInformationFormat=\"4\"/>\r\n" );
273
274 fprintf ( OUT, "\t\t\t<Tool\r\n" );
275 fprintf ( OUT, "\t\t\t\tName=\"VCCustomBuildTool\"/>\r\n" );
276
277 if ( lib )
278 {
279 fprintf ( OUT, "\t\t\t<Tool\r\n" );
280 fprintf ( OUT, "\t\t\t\tName=\"VCLibrarianTool\"\r\n" );
281 fprintf ( OUT, "\t\t\t\tOutputFile=\"$(OutDir)/%s.%s\"/>\r\n", module.name.c_str(), module_type.c_str() );
282 }
283 else
284 {
285 fprintf ( OUT, "\t\t\t<Tool\r\n" );
286 fprintf ( OUT, "\t\t\t\tName=\"VCLinkerTool\"\r\n" );
287
288 fprintf ( OUT, "\t\t\t\tAdditionalDependencies=\"" );
289 for ( i = 0; i < libraries.size(); i++ )
290 {
291 if ( i > 0 )
292 fprintf ( OUT, " " );
293 fprintf ( OUT, "%s", libraries[i].c_str() );
294 }
295 fprintf ( OUT, "\"\r\n" );
296
297 fprintf ( OUT, "\t\t\t\tOutputFile=\"$(OutDir)/%s.%s\"\r\n", module.name.c_str(), module_type.c_str() );
298 fprintf ( OUT, "\t\t\t\tLinkIncremental=\"%d\"\r\n", debug ? 2 : 1 );
299 fprintf ( OUT, "\t\t\t\tGenerateDebugInformation=\"TRUE\"\r\n" );
300
301 if ( debug )
302 fprintf ( OUT, "\t\t\t\tProgramDatabaseFile=\"$(OutDir)/%s.pdb\"\r\n", module.name.c_str() );
303
304 fprintf ( OUT, "\t\t\t\tSubSystem=\"%d\"\r\n", console ? 1 : 2 );
305 fprintf ( OUT, "\t\t\t\tTargetMachine=\"%d\"/>\r\n", 1 );
306 }
307
308 fprintf ( OUT, "\t\t\t<Tool\r\n" );
309 fprintf ( OUT, "\t\t\t\tName=\"VCResourceCompilerTool\"\r\n" );
310 fprintf ( OUT, "\t\t\t\tAdditionalIncludeDirectories=\"" );
311 multiple_includes = false;
312 fprintf ( OUT, "./;" );
313 for ( i = 0; i < includes.size(); i++ )
314 {
315 const string& include = includes[i];
316 if ( strcmp ( include.c_str(), "." ) )
317 {
318 if ( multiple_includes )
319 fprintf ( OUT, ";" );
320 fprintf ( OUT, "%s", include.c_str() );
321 multiple_includes = true;
322 }
323 }
324 fprintf ( OUT, "\"/>\r\n " );
325
326 fprintf ( OUT, "\t\t\t<Tool\r\n" );
327 fprintf ( OUT, "\t\t\t\tName=\"VCMIDLTool\"/>\r\n" );
328 fprintf ( OUT, "\t\t\t<Tool\r\n" );
329 fprintf ( OUT, "\t\t\t\tName=\"VCPostBuildEventTool\"/>\r\n" );
330 fprintf ( OUT, "\t\t\t<Tool\r\n" );
331 fprintf ( OUT, "\t\t\t\tName=\"VCPreBuildEventTool\"/>\r\n" );
332 fprintf ( OUT, "\t\t\t<Tool\r\n" );
333 fprintf ( OUT, "\t\t\t\tName=\"VCPreLinkEventTool\"/>\r\n" );
334 fprintf ( OUT, "\t\t\t<Tool\r\n" );
335 fprintf ( OUT, "\t\t\t\tName=\"VCWebServiceProxyGeneratorTool\"/>\r\n" );
336 fprintf ( OUT, "\t\t\t<Tool\r\n" );
337 fprintf ( OUT, "\t\t\t\tName=\"VCWebDeploymentTool\"/>\r\n" );
338 fprintf ( OUT, "\t\t</Configuration>\r\n" );
339
340 n++;
341 }
342 fprintf ( OUT, "\t</Configurations>\r\n" );
343
344 fprintf ( OUT, "\t<Files>\r\n" );
345
346 // Source files
347 fprintf ( OUT, "\t\t<Filter\r\n" );
348 fprintf ( OUT, "\t\t\tName=\"Source Files\"\r\n" );
349 fprintf ( OUT, "\t\t\tFilter=\"cpp;c;cxx;rc;def;r;odl;idl;hpj;bat\">\r\n" );
350 for ( size_t isrcfile = 0; isrcfile < source_files.size(); isrcfile++ )
351 {
352 const string& source_file = DosSeparator(source_files[isrcfile]);
353 fprintf ( OUT, "\t\t\t<File\r\n" );
354 fprintf ( OUT, "\t\t\t\tRelativePath=\"%s\">\r\n", source_file.c_str() );
355 fprintf ( OUT, "\t\t\t</File>\r\n" );
356 }
357 fprintf ( OUT, "\t\t</Filter>\r\n" );
358
359 // Header files
360 fprintf ( OUT, "\t\t<Filter\r\n" );
361 fprintf ( OUT, "\t\t\tName=\"Header Files\"\r\n" );
362 fprintf ( OUT, "\t\t\tFilter=\"h;hpp;hxx;hm;inl\">\r\n" );
363 for ( i = 0; i < header_files.size(); i++ )
364 {
365 const string& header_file = header_files[i];
366 fprintf ( OUT, "\t\t\t<File\r\n" );
367 fprintf ( OUT, "\t\t\t\tRelativePath=\"%s\">\r\n", header_file.c_str() );
368 fprintf ( OUT, "\t\t\t</File>\r\n" );
369 }
370 fprintf ( OUT, "\t\t</Filter>\r\n" );
371
372 // Resource files
373 fprintf ( OUT, "\t\t<Filter\r\n" );
374 fprintf ( OUT, "\t\t\tName=\"Resource Files\"\r\n" );
375 fprintf ( OUT, "\t\t\tFilter=\"ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe\">\r\n" );
376 for ( i = 0; i < header_files.size(); i++ )
377 {
378 const string& resource_file = resource_files[i];
379 fprintf ( OUT, "\t\t\t<File\r\n" );
380 fprintf ( OUT, "\t\t\t\tRelativePath=\"%s\">\r\n", resource_file.c_str() );
381 fprintf ( OUT, "\t\t\t</File>\r\n" );
382 }
383 fprintf ( OUT, "\t\t</Filter>\r\n" );
384
385 fprintf ( OUT, "\t</Files>\r\n" );
386 fprintf ( OUT, "\t<Globals>\r\n" );
387 fprintf ( OUT, "\t</Globals>\r\n" );
388 fprintf ( OUT, "</VisualStudioProject>\r\n" );
389 fclose(OUT);
390 }
391
392 std::string
393 MSVCBackend::_replace_str(std::string string1, const std::string &find_str, const std::string &replace_str)
394 {
395 std::string::size_type pos = string1.find(find_str, 0);
396 int intLen = find_str.length();
397
398 while(std::string::npos != pos)
399 {
400 string1.replace(pos, intLen, replace_str);
401 pos = string1.find(find_str, intLen + pos);
402 }
403
404 return string1;
405 }
406
407 void
408 MSVCBackend::_generate_sln_header ( FILE* OUT )
409 {
410 if (configuration.VSProjectVersion.empty())
411 configuration.VSProjectVersion = MS_VS_DEF_VERSION;
412
413 string version;
414
415 if (configuration.VSProjectVersion == "7.00")
416 version = "7.00";
417
418 if (configuration.VSProjectVersion == "7.10")
419 version = "8.00";
420
421 if (configuration.VSProjectVersion == "8.00")
422 version = "9.00";
423
424 fprintf ( OUT, "Microsoft Visual Studio Solution File, Format Version %s\r\n", version.c_str() );
425 fprintf ( OUT, "# Visual Studio 2005\r\n" );
426 fprintf ( OUT, "\r\n" );
427 }
428
429
430 void
431 MSVCBackend::_generate_sln_project (
432 FILE* OUT,
433 const Module& module,
434 std::string vcproj_file,
435 std::string sln_guid,
436 std::string vcproj_guid,
437 const std::vector<Dependency*>& dependencies )
438 {
439
440 vcproj_file = DosSeparator ( std::string(".\\") + vcproj_file );
441
442 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() );
443 fprintf ( OUT, " ProjectSection(ProjectDependencies) = postProject\r\n" );
444 fprintf ( OUT, " EndProjectSection\r\n" );
445 fprintf ( OUT, "EndProject\r\n" );
446 }
447
448
449 void
450 MSVCBackend::_generate_sln_footer ( FILE* OUT )
451 {
452 fprintf ( OUT, "Global\r\n" );
453
454 fprintf ( OUT, " GlobalSection(SolutionConfiguration) = preSolution\r\n" );
455 fprintf ( OUT, " Debug = Debug\r\n" );
456 fprintf ( OUT, " Release = Release\r\n" );
457 fprintf ( OUT, " EndGlobalSection\r\n" );
458
459 fprintf ( OUT, " GlobalSection(ExtensibilityGlobals) = postSolution\r\n" );
460 fprintf ( OUT, " EndGlobalSection\r\n" );
461 fprintf ( OUT, " GlobalSection(ExtensibilityAddIns) = postSolution\r\n" );
462 fprintf ( OUT, " EndGlobalSection\r\n" );
463 fprintf ( OUT, "EndGlobal\r\n" );
464
465
466 fprintf ( OUT, "\r\n" );
467 }
468
469
470 void
471 MSVCBackend::_generate_sln ( FILE* OUT )
472 {
473 string sln_guid = "{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}";
474
475 _generate_sln_header(OUT);
476 // TODO FIXME - is it necessary to sort them?
477 for ( size_t i = 0; i < ProjectNode.modules.size(); i++ )
478 {
479 Module& module = *ProjectNode.modules[i];
480
481 std::string vcproj_file = VcprojFileName ( module );
482 std::string vcproj_guid = _gen_guid();
483 _generate_sln_project ( OUT, module, vcproj_file, sln_guid, vcproj_guid, module.dependencies );
484 }
485 _generate_sln_footer ( OUT );
486 }
487