]> git.proxmox.com Git - mirror_edk2.git/blob - Tools/Source/GenBuild/org/tianocore/build/GenBuildTask.java
Fully support active platform policy.
[mirror_edk2.git] / Tools / Source / GenBuild / org / tianocore / build / GenBuildTask.java
1 /** @file
2 This file is ANT task GenBuild.
3
4 The file is used to parse a specified Module, and generate its build time
5 ANT script build.xml, then call the the ANT script to build the module.
6
7 Copyright (c) 2006, Intel Corporation
8 All rights reserved. This program and the accompanying materials
9 are licensed and made available under the terms and conditions of the BSD License
10 which accompanies this distribution. The full text of the license may be found at
11 http://opensource.org/licenses/bsd-license.php
12
13 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
14 WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
15 **/
16 package org.tianocore.build;
17
18 import java.io.File;
19 import java.util.Hashtable;
20 import java.util.Iterator;
21 import java.util.LinkedHashSet;
22 import java.util.List;
23 import java.util.Map;
24 import java.util.Set;
25 import java.util.Stack;
26 import java.util.Vector;
27 import java.util.regex.Matcher;
28 import java.util.regex.Pattern;
29
30 import org.apache.tools.ant.BuildException;
31 import org.apache.tools.ant.taskdefs.Ant;
32 import org.apache.tools.ant.taskdefs.Property;
33 import org.apache.xmlbeans.XmlObject;
34
35 import org.tianocore.build.autogen.AutoGen;
36 import org.tianocore.build.fpd.FpdParserTask;
37 import org.tianocore.build.global.GenBuildLogger;
38 import org.tianocore.build.global.GlobalData;
39 import org.tianocore.build.global.OutputManager;
40 import org.tianocore.build.global.SurfaceAreaQuery;
41 import org.tianocore.build.id.FpdModuleIdentification;
42 import org.tianocore.build.id.ModuleIdentification;
43 import org.tianocore.build.id.PackageIdentification;
44 import org.tianocore.build.id.PlatformIdentification;
45 import org.tianocore.build.tools.ModuleItem;
46 import org.tianocore.exception.EdkException;
47 import org.tianocore.logger.EdkLog;
48
49 /**
50 <p>
51 <code>GenBuildTask</code> is an ANT task that can be used in ANT build
52 system. The main function of this task is to parse module's surface area,
53 then generate the corresponding <em>BaseName_build.xml</em> (the real ANT
54 build script) and call this to build the module. The whole process including:
55 1. generate AutoGen.c and AutoGen.h; 2. build all dependent library instances;
56 3. build all source files inlcude AutoGen.c; 4. generate sections;
57 5. generate FFS file if it is driver module while LIB file if it is Library module.
58 </p>
59
60 <p>
61 The usage is (take module <em>HelloWorld</em> for example):
62 </p>
63
64 <pre>
65 &lt;GenBuild
66 msaFilename=&quot;HelloWorld.msa&quot;/&gt;
67 processTo=&quot;ALL&quot;/&gt;
68 </pre>
69
70 <p><code>processTo</code> provides a way to customize the whole build process.
71 processTo can be one value of ALL, AUTOGEN, FILES, LIBRARYINSTANCES, SECTIONS, NONE.
72 Default is ALL, means whole
73 </p>
74
75 <p>
76 This task calls <code>AutoGen</code> to generate <em>AutoGen.c</em> and
77 <em>AutoGen.h</em>. The task also parses the development environment
78 configuration files, such as collecting package information, setting compiler
79 flags and so on.
80 </p>
81
82
83 @since GenBuild 1.0
84 **/
85 public class GenBuildTask extends Ant {
86
87 ///
88 /// Module surface area file.
89 ///
90 File msaFile;
91
92 ///
93 ///
94 ///
95 private String type = "all"; // = "build";
96
97 ///
98 /// Module's Identification.
99 ///
100 private ModuleIdentification moduleId;
101
102 private Vector<Property> properties = new Vector<Property>();
103
104 private static Stack<Hashtable> backupPropertiesStack = new Stack<Hashtable>();
105
106 private boolean isSingleModuleBuild = false;
107
108 /**
109 Public construct method. It is necessary for ANT task.
110 **/
111 public GenBuildTask() {
112 }
113
114 /**
115
116 @throws BuildException
117 From module build, exception from module surface area invalid.
118 **/
119 public void execute() throws BuildException {
120 //
121 // set Logger
122 //
123 GenBuildLogger logger = new GenBuildLogger(getProject());
124 EdkLog.setLogLevel(getProject().getProperty("env.LOGLEVEL"));
125 EdkLog.setLogger(logger);
126 // remove !!
127 try {
128 pushProperties();
129 //
130 // Enable all specified properties
131 //
132 Iterator<Property> iter = properties.iterator();
133 while (iter.hasNext()) {
134 Property item = iter.next();
135 getProject().setProperty(item.getName(), item.getValue());
136 }
137
138 //
139 // GenBuild should specify either msaFile or moduleGuid & packageGuid
140 //
141 if (msaFile == null ) {
142 String moduleGuid = getProject().getProperty("MODULE_GUID");
143 String moduleVersion = getProject().getProperty("MODULE_VERSION");
144 String packageGuid = getProject().getProperty("PACKAGE_GUID");
145 String packageVersion = getProject().getProperty("PACKAGE_VERSION");
146 if (moduleGuid == null || packageGuid == null) {
147 throw new BuildException("GenBuild parameters error. ");
148 }
149 PackageIdentification packageId = new PackageIdentification(packageGuid, packageVersion);
150 moduleId = new ModuleIdentification(moduleGuid, moduleVersion);
151 moduleId.setPackage(packageId);
152 Map<String, XmlObject> doc = GlobalData.getNativeMsa(moduleId);
153 SurfaceAreaQuery.setDoc(doc);
154 moduleId = SurfaceAreaQuery.getMsaHeader();
155 }
156 else {
157 Map<String, XmlObject> doc = GlobalData.getNativeMsa(msaFile);
158 SurfaceAreaQuery.setDoc(doc);
159 moduleId = SurfaceAreaQuery.getMsaHeader();
160 }
161 String[] producedLibraryClasses = SurfaceAreaQuery.getLibraryClasses("ALWAYS_PRODUCED");
162 if (producedLibraryClasses.length == 0) {
163 moduleId.setLibrary(false);
164 }
165 else {
166 moduleId.setLibrary(true);
167 }
168
169 //
170 // Judge whether it is single module build or not
171 //
172 if (isSingleModuleBuild) {
173 //
174 // Single Module build
175 //
176 prepareSingleModuleBuild();
177 }
178 else {
179 //
180 // Platform build. Restore the platform related info
181 //
182 String filename = getProject().getProperty("PLATFORM_FILE");
183 PlatformIdentification platformId = GlobalData.getPlatform(filename);
184 getProject().setProperty("PLATFORM_DIR", platformId.getFpdFile().getParent().replaceAll("(\\\\)", "/"));
185 getProject().setProperty("PLATFORM_RELATIVE_DIR", platformId.getPlatformRelativeDir().replaceAll("(\\\\)", "/"));
186
187 String packageGuid = getProject().getProperty("PACKAGE_GUID");
188 String packageVersion = getProject().getProperty("PACKAGE_VERSION");
189 PackageIdentification packageId = new PackageIdentification(packageGuid, packageVersion);
190 moduleId.setPackage(packageId);
191 }
192
193 //
194 // If single module : intersection MSA supported ARCHs and tools def!!
195 // else, get arch from pass down
196 //
197 String[] archList = new String[0];
198 if ( getProject().getProperty("ARCH") != null ) {
199 archList = getProject().getProperty("ARCH").split(" ");
200 }
201 else {
202 archList = GlobalData.getToolChainInfo().getArchs();
203 }
204
205
206 //
207 // Judge if arch is all supported by current module. If not, throw Exception.
208 //
209 List moduleSupportedArchs = SurfaceAreaQuery.getModuleSupportedArchs();
210 if (moduleSupportedArchs != null) {
211 for (int k = 0; k < archList.length; k++) {
212 if ( ! moduleSupportedArchs.contains(archList[k])) {
213 throw new BuildException("ARCH [" + archList[k] + "] is not supported by " + moduleId + ". " + moduleId + " only supports [" + moduleSupportedArchs + "].");
214 }
215 }
216 }
217
218 for (int k = 0; k < archList.length; k++) {
219 getProject().setProperty("ARCH", archList[k]);
220
221 FpdModuleIdentification fpdModuleId = new FpdModuleIdentification(moduleId, archList[k]);
222
223 //
224 // Whether the module is built before
225 //
226 if (GlobalData.isModuleBuilt(fpdModuleId)) {
227 return ;
228 }
229 else {
230 GlobalData.registerBuiltModule(fpdModuleId);
231 }
232
233 //
234 // For Every TOOLCHAIN, TARGET
235 //
236 String[] targetList = GlobalData.getToolChainInfo().getTargets();
237 for (int i = 0; i < targetList.length; i ++){
238 //
239 // Prepare for target related common properties
240 // TARGET
241 //
242 getProject().setProperty("TARGET", targetList[i]);
243 String[] toolchainList = GlobalData.getToolChainInfo().getTagnames();
244 for(int j = 0; j < toolchainList.length; j ++){
245 //
246 // Prepare for toolchain related common properties
247 // TOOLCHAIN
248 //
249 getProject().setProperty("TOOLCHAIN", toolchainList[j]);
250
251 System.out.println("Build " + moduleId + " start >>>");
252 System.out.println("Target: " + targetList[i] + " Tagname: " + toolchainList[j] + " Arch: " + archList[k]);
253 SurfaceAreaQuery.setDoc(GlobalData.getDoc(fpdModuleId));
254
255 //
256 // Prepare for all other common properties
257 // PACKAGE, PACKAGE_GUID, PACKAGE_VERSION, PACKAGE_DIR, PACKAGE_RELATIVE_DIR
258 // MODULE or BASE_NAME, GUID or FILE_GUID, VERSION, MODULE_TYPE
259 // MODULE_DIR, MODULE_RELATIVE_DIR
260 // SUBSYSTEM, ENTRYPOINT, EBC_TOOL_LIB_PATH
261 // LIBS, OBJECTS, SDB_FILES
262 //
263 setModuleCommonProperties(archList[k]);
264
265 //
266 // OutputManage prepare for
267 // BIN_DIR, DEST_DIR_DEBUG, DEST_DIR_OUTPUT, BUILD_DIR, FV_DIR
268 //
269 OutputManager.getInstance().update(getProject());
270
271 if (type.equalsIgnoreCase("all") || type.equalsIgnoreCase("build")) {
272 applyBuild(targetList[i], toolchainList[j], fpdModuleId);
273 }
274 else if (type.equalsIgnoreCase("clean")) {
275 applyClean(fpdModuleId);
276 }
277 else if (type.equalsIgnoreCase("cleanall")) {
278 applyCleanall(fpdModuleId);
279 }
280 }
281 }
282 }
283 popProperties();
284 }catch (Exception e){
285 e.printStackTrace();
286 throw new BuildException(e.getMessage());
287 }
288 }
289
290 /**
291 This method is used to prepare Platform-related information.
292
293 <p>In Single Module Build mode, platform-related information is not ready.
294 The method read the system environment variable <code>ACTIVE_PLATFORM</code>
295 and search in the Framework Database. Note that platform name in the Framework
296 Database must be unique. </p>
297
298 **/
299 private void prepareSingleModuleBuild(){
300 //
301 // Find out the package which the module belongs to
302 // TBD: Enhance it!!!!
303 //
304 PackageIdentification packageId = GlobalData.getPackageForModule(moduleId);
305
306 moduleId.setPackage(packageId);
307
308 //
309 // Read ACTIVE_PLATFORM's FPD file (Call FpdParserTask's method)
310 //
311 String filename = getProject().getProperty("PLATFORM_FILE");
312
313 if (filename == null){
314 throw new BuildException("Plese set ACTIVE_PLATFORM if you want to build a single module. ");
315 }
316
317 PlatformIdentification platformId = GlobalData.getPlatform(filename);
318
319 //
320 // Read FPD file
321 //
322 FpdParserTask fpdParser = new FpdParserTask();
323 fpdParser.setProject(getProject());
324 fpdParser.parseFpdFile(platformId.getFpdFile());
325
326 //
327 // Prepare for Platform related common properties
328 // PLATFORM, PLATFORM_DIR, PLATFORM_RELATIVE_DIR
329 //
330 getProject().setProperty("PLATFORM", platformId.getName());
331 getProject().setProperty("PLATFORM_DIR", platformId.getFpdFile().getParent().replaceAll("(\\\\)", "/"));
332 getProject().setProperty("PLATFORM_RELATIVE_DIR", platformId.getPlatformRelativeDir().replaceAll("(\\\\)", "/"));
333 }
334
335
336 /**
337 Set Module-Related information to properties.
338 **/
339 private void setModuleCommonProperties(String arch) {
340 //
341 // Prepare for all other common properties
342 // PACKAGE, PACKAGE_GUID, PACKAGE_VERSION, PACKAGE_DIR, PACKAGE_RELATIVE_DIR
343 //
344 PackageIdentification packageId = moduleId.getPackage();
345 getProject().setProperty("PACKAGE", packageId.getName());
346 getProject().setProperty("PACKAGE_GUID", packageId.getGuid());
347 getProject().setProperty("PACKAGE_VERSION", packageId.getVersion());
348 getProject().setProperty("PACKAGE_DIR", packageId.getPackageDir().replaceAll("(\\\\)", "/"));
349 getProject().setProperty("PACKAGE_RELATIVE_DIR", packageId.getPackageRelativeDir().replaceAll("(\\\\)", "/"));
350
351 //
352 // MODULE or BASE_NAME, GUID or FILE_GUID, VERSION, MODULE_TYPE
353 // MODULE_DIR, MODULE_RELATIVE_DIR
354 //
355 getProject().setProperty("MODULE", moduleId.getName());
356 String baseName = SurfaceAreaQuery.getModuleOutputFileBasename();
357 if (baseName == null) {
358 getProject().setProperty("BASE_NAME", moduleId.getName());
359 }
360 else {
361 getProject().setProperty("BASE_NAME", baseName);
362 }
363 getProject().setProperty("GUID", moduleId.getGuid());
364 getProject().setProperty("FILE_GUID", moduleId.getGuid());
365 getProject().setProperty("VERSION", moduleId.getVersion());
366 getProject().setProperty("MODULE_TYPE", moduleId.getModuleType());
367 getProject().setProperty("MODULE_DIR", moduleId.getMsaFile().getParent().replaceAll("(\\\\)", "/"));
368 getProject().setProperty("MODULE_RELATIVE_DIR", moduleId.getModuleRelativePath().replaceAll("(\\\\)", "/"));
369
370 //
371 // SUBSYSTEM
372 //
373 String[][] subsystemMap = { { "BASE", "EFI_BOOT_SERVICE_DRIVER"},
374 { "SEC", "EFI_BOOT_SERVICE_DRIVER" },
375 { "PEI_CORE", "EFI_BOOT_SERVICE_DRIVER" },
376 { "PEIM", "EFI_BOOT_SERVICE_DRIVER" },
377 { "DXE_CORE", "EFI_BOOT_SERVICE_DRIVER" },
378 { "DXE_DRIVER", "EFI_BOOT_SERVICE_DRIVER" },
379 { "DXE_RUNTIME_DRIVER", "EFI_RUNTIME_DRIVER" },
380 { "DXE_SAL_DRIVER", "EFI_BOOT_SERVICE_DRIVER" },
381 { "DXE_SMM_DRIVER", "EFI_BOOT_SERVICE_DRIVER" },
382 { "TOOL", "EFI_BOOT_SERVICE_DRIVER" },
383 { "UEFI_DRIVER", "EFI_BOOT_SERVICE_DRIVER" },
384 { "UEFI_APPLICATION", "EFI_APPLICATION" },
385 { "USER_DEFINED", "EFI_BOOT_SERVICE_DRIVER"} };
386
387 String subsystem = "EFI_BOOT_SERVICE_DRIVER";
388 for (int i = 0; i < subsystemMap.length; i++) {
389 if (moduleId.getModuleType().equalsIgnoreCase(subsystemMap[i][0])) {
390 subsystem = subsystemMap[i][1];
391 break ;
392 }
393 }
394 getProject().setProperty("SUBSYSTEM", subsystem);
395
396 //
397 // ENTRYPOINT
398 //
399 if (arch.equalsIgnoreCase("EBC")) {
400 getProject().setProperty("ENTRYPOINT", "EfiStart");
401 }
402 else {
403 getProject().setProperty("ENTRYPOINT", "_ModuleEntryPoint");
404 }
405
406 //
407 // LIBS, OBJECTS, SDB_FILES
408 //
409 getProject().setProperty("OBJECTS", "");
410 getProject().setProperty("SDB_FILES", "");
411 getProject().setProperty("LIBS", "");
412 }
413
414 private void getCompilerFlags(String target, String toolchain, FpdModuleIdentification fpdModuleId) throws EdkException {
415 String[] cmd = GlobalData.getToolChainInfo().getCommands();
416 for ( int m = 0; m < cmd.length; m++) {
417 //
418 // Set cmd, like CC, DLINK
419 //
420 String[] key = new String[]{target, toolchain, fpdModuleId.getArch(), cmd[m], null};
421 key[4] = "PATH";
422 String cmdPath = GlobalData.getCommandSetting(key, fpdModuleId);
423 key[4] = "NAME";
424 String cmdName = GlobalData.getCommandSetting(key, fpdModuleId);
425 File cmdFile = new File(cmdPath + File.separatorChar + cmdName);
426 // GlobalData.log.info("PATH: " + cmdFile.getPath());
427 getProject().setProperty(cmd[m], cmdFile.getPath().replaceAll("(\\\\)", "/"));
428
429 //
430 // set CC_FLAGS
431 //
432 key[4] = "FLAGS";
433 String cmdFlags = GlobalData.getCommandSetting(key, fpdModuleId);
434 // GlobalData.log.info("Flags: " + cmdFlags);
435 Set<String> addset = new LinkedHashSet<String>();
436 Set<String> subset = new LinkedHashSet<String>();
437 putFlagsToSet(addset, cmdFlags);
438 getProject().setProperty(cmd[m] + "_FLAGS", getProject().replaceProperties(getFlags(addset, subset)));
439
440 //
441 // Set CC_EXT
442 //
443 key[4] = "EXT";
444 String extName = GlobalData.getCommandSetting(key, fpdModuleId);
445 // GlobalData.log.info("Ext: " + extName);
446 if ( extName != null && ! extName.equalsIgnoreCase("")) {
447 getProject().setProperty(cmd[m] + "_EXT", extName);
448 }
449 else {
450 getProject().setProperty(cmd[m] + "_EXT", "");
451 }
452
453 //
454 // set CC_FAMILY
455 //
456 key[4] = "FAMILY";
457 String toolChainFamily = GlobalData.getCommandSetting(key, fpdModuleId);
458 // GlobalData.log.info("FAMILY: " + toolChainFamily);
459 if (toolChainFamily != null) {
460 getProject().setProperty(cmd[m] + "_FAMILY", toolChainFamily);
461 }
462
463 //
464 // set CC_SPATH
465 //
466 key[4] = "SPATH";
467 String spath = GlobalData.getCommandSetting(key, fpdModuleId);
468 // GlobalData.log.info("SPATH: " + spath);
469 if (spath != null) {
470 getProject().setProperty(cmd[m] + "_SPATH", spath.replaceAll("(\\\\)", "/"));
471 }
472 else {
473 getProject().setProperty(cmd[m] + "_SPATH", "");
474 }
475
476 //
477 // set CC_DPATH
478 //
479 key[4] = "DPATH";
480 String dpath = GlobalData.getCommandSetting(key, fpdModuleId);
481 // GlobalData.log.info("DPATH: " + dpath);
482 if (dpath != null) {
483 getProject().setProperty(cmd[m] + "_DPATH", dpath.replaceAll("(\\\\)", "/"));
484 }
485 else {
486 getProject().setProperty(cmd[m] + "_DPATH", "");
487 }
488 }
489 }
490
491 public void setMsaFile(File msaFile) {
492 this.msaFile = msaFile;
493 }
494
495 /**
496 Method is for ANT to initialize MSA file.
497
498 @param msaFilename MSA file name
499 **/
500 public void setMsaFile(String msaFilename) {
501 String moduleDir = getProject().getProperty("MODULE_DIR");
502
503 //
504 // If is Single Module Build, then use the Base Dir defined in build.xml
505 //
506 if (moduleDir == null) {
507 moduleDir = getProject().getBaseDir().getPath();
508 }
509 msaFile = new File(moduleDir + File.separatorChar + msaFilename);
510 }
511
512 public void addConfiguredModuleItem(ModuleItem moduleItem) {
513 PackageIdentification packageId = new PackageIdentification(moduleItem.getPackageGuid(), moduleItem.getPackageVersion());
514 ModuleIdentification moduleId = new ModuleIdentification(moduleItem.getModuleGuid(), moduleItem.getModuleVersion());
515 moduleId.setPackage(packageId);
516 this.moduleId = moduleId;
517 }
518
519 /**
520 Add a property.
521
522 @param p property
523 **/
524 public void addProperty(Property p) {
525 properties.addElement(p);
526 }
527
528 public void setType(String type) {
529 this.type = type;
530 }
531
532 private void applyBuild(String buildTarget, String buildTagname, FpdModuleIdentification fpdModuleId) throws EdkException{
533 //
534 // AutoGen
535 //
536
537 AutoGen autogen = new AutoGen(getProject().getProperty("DEST_DIR_DEBUG"), fpdModuleId.getModule(),fpdModuleId.getArch());
538 autogen.genAutogen();
539
540
541 //
542 // Get compiler flags
543 //
544 getCompilerFlags(buildTarget, buildTagname, fpdModuleId);
545
546 //
547 // Prepare LIBS
548 //
549 ModuleIdentification[] libinstances = SurfaceAreaQuery.getLibraryInstance(fpdModuleId.getArch());
550 String propertyLibs = "";
551 for (int i = 0; i < libinstances.length; i++) {
552 propertyLibs += " " + getProject().getProperty("BIN_DIR") + File.separatorChar + libinstances[i].getName() + ".lib";
553 }
554 getProject().setProperty("LIBS", propertyLibs.replaceAll("(\\\\)", "/"));
555
556 //
557 // if it is CUSTOM_BUILD
558 // then call the exist BaseName_build.xml directly.
559 //
560 if (moduleId.getModuleType().equalsIgnoreCase("USER_DEFINED")) {
561 GlobalData.log.info("Call user-defined " + moduleId.getName() + "_build.xml");
562 Ant ant = new Ant();
563 ant.setProject(getProject());
564 ant.setAntfile(getProject().getProperty("MODULE_DIR") + File.separatorChar + moduleId.getName() + "_build.xml");
565 ant.setInheritAll(true);
566 ant.init();
567 ant.execute();
568 return ;
569 }
570
571 //
572 // Generate ${BASE_NAME}_build.xml
573 // TBD
574 //
575 String ffsKeyword = SurfaceAreaQuery.getModuleFfsKeyword();
576 ModuleBuildFileGenerator fileGenerator = new ModuleBuildFileGenerator(getProject(), ffsKeyword, fpdModuleId);
577 String buildFilename = getProject().getProperty("DEST_DIR_OUTPUT") + File.separatorChar + moduleId.getName() + "_build.xml";
578 fileGenerator.genBuildFile(buildFilename);
579
580 //
581 // Ant call ${BASE_NAME}_build.xml
582 //
583 Ant ant = new Ant();
584 ant.setProject(getProject());
585 ant.setAntfile(getProject().getProperty("DEST_DIR_OUTPUT") + File.separatorChar + moduleId.getName() + "_build.xml");
586 ant.setInheritAll(true);
587 ant.init();
588 ant.execute();
589 }
590
591 private void applyClean(FpdModuleIdentification fpdModuleId){
592 //
593 // if it is CUSTOM_BUILD
594 // then call the exist BaseName_build.xml directly.
595 //
596 if (moduleId.getModuleType().equalsIgnoreCase("USER_DEFINED")) {
597 GlobalData.log.info("Call user-defined " + moduleId.getName() + "_build.xml");
598 Ant ant = new Ant();
599 ant.setProject(getProject());
600 ant.setAntfile(getProject().getProperty("MODULE_DIR") + File.separatorChar + moduleId.getName() + "_build.xml");
601 ant.setTarget("clean");
602 ant.setInheritAll(true);
603 ant.init();
604 ant.execute();
605 return ;
606 }
607
608 Ant ant = new Ant();
609 ant.setProject(getProject());
610 ant.setAntfile(getProject().getProperty("DEST_DIR_OUTPUT") + File.separatorChar + moduleId.getName() + "_build.xml");
611 ant.setTarget("clean");
612 ant.setInheritAll(true);
613 ant.init();
614 ant.execute();
615
616 //
617 // Delete current module's DEST_DIR_OUTPUT
618 // TBD
619 }
620
621 private void applyCleanall(FpdModuleIdentification fpdModuleId){
622 //
623 // if it is CUSTOM_BUILD
624 // then call the exist BaseName_build.xml directly.
625 //
626 if (moduleId.getModuleType().equalsIgnoreCase("USER_DEFINED")) {
627 GlobalData.log.info("Call user-defined " + moduleId.getName() + "_build.xml");
628 Ant ant = new Ant();
629 ant.setProject(getProject());
630 ant.setAntfile(getProject().getProperty("MODULE_DIR") + File.separatorChar + moduleId.getName() + "_build.xml");
631 ant.setTarget("cleanall");
632 ant.setInheritAll(true);
633 ant.init();
634 ant.execute();
635 return ;
636 }
637
638 Ant ant = new Ant();
639 ant.setProject(getProject());
640 ant.setAntfile(getProject().getProperty("DEST_DIR_OUTPUT") + File.separatorChar + moduleId.getName() + "_build.xml");
641 ant.setTarget("cleanall");
642 ant.setInheritAll(true);
643 ant.init();
644 ant.execute();
645
646 //
647 // Delete current module's DEST_DIR_OUTPUT
648 // TBD
649 }
650
651
652
653
654 /**
655 Separate the string and instore in set.
656
657 <p> String is separated by Java Regulation Expression
658 "[^\\\\]?(\".*?[^\\\\]\")[ \t,]+". </p>
659
660 <p>For example: </p>
661
662 <pre>
663 "/nologo", "/W3", "/WX"
664 "/C", "/DSTRING_DEFINES_FILE=\"BdsStrDefs.h\""
665 </pre>
666
667 @param set store the separated string
668 @param str string to separate
669 **/
670 private void putFlagsToSet(Set<String> set, String str) {
671 if (str == null || str.length() == 0) {
672 return;
673 }
674
675 Pattern myPattern = Pattern.compile("[^\\\\]?(\".*?[^\\\\]\")[ \t,]+");
676 Matcher matcher = myPattern.matcher(str + " ");
677 while (matcher.find()) {
678 String item = str.substring(matcher.start(1), matcher.end(1));
679 set.add(item);
680 }
681 }
682
683 /**
684 Generate the final flags string will be used by compile command.
685
686 @param add the add flags set
687 @param sub the sub flags set
688 @return final flags after add set substract sub set
689 **/
690 private String getFlags(Set<String> add, Set<String> sub) {
691 String result = "";
692 add.removeAll(sub);
693 Iterator iter = add.iterator();
694 while (iter.hasNext()) {
695 String str = (String) iter.next();
696 result += str.substring(1, str.length() - 1) + " ";
697 }
698 return result;
699 }
700
701 /**
702 Generate the flags string with original format. The format is defined by
703 Java Regulation Expression "[^\\\\]?(\".*?[^\\\\]\")[ \t,]+". </p>
704
705 <p>For example: </p>
706
707 <pre>
708 "/nologo", "/W3", "/WX"
709 "/C", "/DSTRING_DEFINES_FILE=\"BdsStrDefs.h\""
710 </pre>
711
712 @param add the add flags set
713 @param sub the sub flags set
714 @return flags with original format
715 **/
716 private String getRawFlags(Set<String> add, Set<String> sub) {
717 String result = null;
718 add.removeAll(sub);
719 Iterator iter = add.iterator();
720 while (iter.hasNext()) {
721 String str = (String) iter.next();
722 result += "\"" + str.substring(1, str.length() - 1) + "\", ";
723 }
724 return result;
725 }
726
727 private String parseOptionString(String optionString, Set<String> addSet, Set<String> subSet) {
728 boolean overrideOption = false;
729 Pattern pattern = Pattern.compile("ADD\\.\\[(.+)\\]");
730 Matcher matcher = pattern.matcher(optionString);
731
732 while (matcher.find()) {
733 overrideOption = true;
734 String addOption = optionString.substring(matcher.start(1), matcher.end(1)).trim();
735 putFlagsToSet(addSet, addOption);
736
737 }
738
739 pattern = Pattern.compile("SUB\\.\\[(.+)\\]");
740 matcher = pattern.matcher(optionString);
741
742 while (matcher.find()) {
743 overrideOption = true;
744 String subOption = optionString.substring(matcher.start(1), matcher.end(1)).trim();
745 putFlagsToSet(subSet, subOption);
746 }
747
748 if (overrideOption == true) {
749 return null;
750 }
751
752 return optionString;
753 }
754
755 private void pushProperties() {
756 backupPropertiesStack.push(getProject().getProperties());
757 }
758
759 private void popProperties() {
760 Hashtable backupProperties = backupPropertiesStack.pop();
761 Set keys = backupProperties.keySet();
762 Iterator iter = keys.iterator();
763 while (iter.hasNext()) {
764 String item = (String)iter.next();
765 getProject().setProperty(item, (String)backupProperties.get(item));
766 }
767 }
768
769 public void setSingleModuleBuild(boolean isSingleModuleBuild) {
770 this.isSingleModuleBuild = isSingleModuleBuild;
771 }
772 }