2 This file is ANT task FpdParserTask.
4 FpdParserTask is used to parse FPD (Framework Platform Description) and generate
5 build.out.xml. It is for Package or Platform build use.
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
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.
16 package org
.tianocore
.build
.fpd
;
18 import java
.io
.BufferedWriter
;
20 import java
.io
.FileWriter
;
21 import java
.util
.HashMap
;
22 import java
.util
.Iterator
;
23 import java
.util
.LinkedHashMap
;
24 import java
.util
.LinkedHashSet
;
27 import java
.util
.Vector
;
28 import java
.util
.TreeMap
;
30 import org
.apache
.tools
.ant
.BuildException
;
31 import org
.apache
.tools
.ant
.Task
;
32 import org
.apache
.tools
.ant
.taskdefs
.Ant
;
33 import org
.apache
.tools
.ant
.taskdefs
.Property
;
34 import org
.apache
.xmlbeans
.XmlObject
;
36 import org
.tianocore
.build
.global
.GlobalData
;
37 import org
.tianocore
.build
.global
.OutputManager
;
38 import org
.tianocore
.build
.global
.SurfaceAreaQuery
;
39 import org
.tianocore
.build
.id
.FpdModuleIdentification
;
40 import org
.tianocore
.build
.id
.ModuleIdentification
;
41 import org
.tianocore
.build
.id
.PlatformIdentification
;
42 import org
.tianocore
.build
.pcd
.action
.ActionMessage
;
43 import org
.tianocore
.build
.pcd
.action
.CollectPCDAction
;
44 import org
.tianocore
.build
.pcd
.exception
.EntityException
;
45 import org
.tianocore
.build
.toolchain
.ToolChainAttribute
;
46 import org
.tianocore
.build
.toolchain
.ToolChainElement
;
47 import org
.tianocore
.build
.toolchain
.ToolChainMap
;
48 import org
.tianocore
.exception
.EdkException
;
51 <code>FpdParserTask</code> is an ANT task. The main function is parsing Framework
52 Platform Descritpion (FPD) XML file and generating its ANT build script for
53 corresponding platform.
55 <p>The task sets global properties PLATFORM, PLATFORM_DIR, PLATFORM_RELATIVE_DIR
58 <p>The task generates ${PLATFORM}_build.xml file which will be called by top level
59 build.xml. The task also generate Fv.inf files (File is for Tool GenFvImage)
60 and flash definition file (File is for Tool FlashMap) if necessary. </p>
62 <p>FpdParserTask task stores all FPD information to GlobalData. And parse
63 tools definition file to set up compiler options for different Target and
64 different ToolChainTag. </p>
66 <p>The method parseFpdFile is also prepared for single module build. </p>
68 <p>The usage is (take NT32 Platform for example):</p>
71 <FPDParser platformName="Nt32" />
74 <p>The task will initialize all information through parsing Framework Database,
75 SPD, Tool chain configuration files. </p>
79 public class FpdParserTask
extends Task
{
81 private String platformName
;
83 private File fpdFile
= null;
85 private PlatformIdentification platformId
;
93 /// Mapping from modules identification to out put file name
95 private Map
<FpdModuleIdentification
, String
> outfiles
= new LinkedHashMap
<FpdModuleIdentification
, String
>();
98 /// Mapping from FV name to its modules
100 private Map
<String
, Set
<FpdModuleIdentification
>> fvs
= new HashMap
<String
, Set
<FpdModuleIdentification
>>();
103 /// Mapping from sequence number to FV names
105 private Map
<String
, Set
<String
>> sequences
= new TreeMap
<String
, Set
<String
>>();
108 /// FpdParserTask can specify some ANT properties.
110 private Vector
<Property
> properties
= new Vector
<Property
>();
112 private boolean isUnified
= true;
116 Public construct method. It is necessary for ANT task.
118 public FpdParserTask() {
122 ANT task's entry method. The main steps is described as following:
125 <li>Initialize global information (Framework DB, SPD files and all MSA files
126 listed in SPD). This step will execute only once in whole build process;</li>
127 <li>Parse specified FPD file; </li>
128 <li>Generate FV.inf files; </li>
129 <li>Generate PlatformName_build.xml file for Flatform build; </li>
130 <li>Collect PCD information. </li>
133 @throws BuildException
134 Surface area is not valid.
136 public void execute() throws BuildException
{
138 if ( fpdFile
== null) {
139 if (platformName
== null) {
140 throw new BuildException("FpdParserTask parameter error. Please specify platform name or FPD file. ");
142 platformId
= GlobalData
.getPlatform(platformName
);
143 fpdFile
= platformId
.getFpdFile();
152 // Pcd Collection. Call CollectPCDAction to collect pcd info.
155 System
.out
.println("Begin PCD collecttion!");
156 CollectPCDAction ca
= new CollectPCDAction();
157 ca
.perform(GlobalData
.getWorkspacePath(),platformId
.getFpdFile().getPath(),ActionMessage
.NULL_MESSAGE_LEVEL
);
158 System
.out
.println("End PCD collection!");
159 } catch (Exception e
){
160 throw new BuildException(e
.getMessage());
166 isUnified
= OutputManager
.getInstance().prepareBuildDir(getProject());
169 // Generate FDF (Flash Definition File) file
173 // For every Target and ToolChain
175 String
[] targetList
= GlobalData
.getToolChainInfo().getTargets();
176 for (int i
= 0; i
< targetList
.length
; i
++){
177 String
[] toolchainList
= GlobalData
.getToolChainInfo().getTagnames();
178 for(int j
= 0; j
< toolchainList
.length
; j
++){
182 String ffsCommonDir
= getProject().getProperty("BUILD_DIR") + File
.separatorChar
183 + targetList
[i
] + File
.separatorChar
185 File fvDir
= new File(ffsCommonDir
+ File
.separatorChar
+ "FV");
187 getProject().setProperty("FV_DIR", fvDir
.getPath().replaceAll("(\\\\)", "/"));
192 genFvInfFiles(ffsCommonDir
);
199 PlatformBuildFileGenerator fileGenerator
= new PlatformBuildFileGenerator(getProject(), outfiles
, fvs
, sequences
, isUnified
);
200 fileGenerator
.genBuildFile();
203 // Ant call ${PLATFORM}_build.xml
207 ant
.setProject(getProject());
208 ant
.setAntfile(platformId
.getFpdFile().getParent() + File
.separatorChar
+ platformId
.getName() + "_build.xml");
210 ant
.setInheritAll(true);
214 // GlobalData.log.info("Fpd build end. ");
218 Generate Fv.inf files. The Fv.inf file is composed with four
219 parts: Options, Attributes, Components and Files. The Fv.inf files
220 will be under FV_DIR.
222 @throws BuildException
223 File write FV.inf files error.
225 private void genFvInfFiles(String ffsCommonDir
) throws BuildException
{
226 String
[] validFv
= SurfaceAreaQuery
.getFpdValidImageNames();
227 for (int i
= 0; i
< validFv
.length
; i
++) {
229 // Get all global variables from FPD and set them to properties
231 String
[][] globalVariables
= SurfaceAreaQuery
.getFpdGlobalVariable();
232 for (int j
= 0; j
< globalVariables
.length
; j
++) {
233 getProject().setProperty(globalVariables
[j
][0], globalVariables
[j
][1]);
236 getProject().setProperty("FV_FILENAME", validFv
[i
].toUpperCase());
238 File fvFile
= new File(getProject().replaceProperties( getProject().getProperty("FV_DIR") + File
.separatorChar
+ validFv
[i
].toUpperCase() + ".inf"));
239 fvFile
.getParentFile().mkdirs();
242 FileWriter fw
= new FileWriter(fvFile
);
243 BufferedWriter bw
= new BufferedWriter(fw
);
248 String
[][] options
= SurfaceAreaQuery
.getFpdOptions(validFv
[i
]);
249 if (options
.length
> 0) {
250 bw
.write("[options]");
252 for (int j
= 0; j
< options
.length
; j
++) {
253 StringBuffer str
= new StringBuffer(100);
254 str
.append(options
[j
][0]);
255 while (str
.length() < 40) {
259 str
.append(options
[j
][1]);
260 bw
.write(getProject().replaceProperties(str
.toString()));
269 String
[][] attributes
= SurfaceAreaQuery
.getFpdAttributes(validFv
[i
]);
270 if (attributes
.length
> 0) {
271 bw
.write("[attributes]");
273 for (int j
= 0; j
< attributes
.length
; j
++) {
274 StringBuffer str
= new StringBuffer(100);
275 str
.append(attributes
[j
][0]);
276 while (str
.length() < 40) {
280 str
.append(attributes
[j
][1]);
281 bw
.write(getProject().replaceProperties(str
.toString()));
290 String
[][] components
= SurfaceAreaQuery
.getFpdComponents(validFv
[i
]);
291 if (components
.length
> 0) {
292 bw
.write("[components]");
294 for (int j
= 0; j
< components
.length
; j
++) {
295 StringBuffer str
= new StringBuffer(100);
296 str
.append(components
[j
][0]);
297 while (str
.length() < 40) {
301 str
.append(components
[j
][1]);
302 bw
.write(getProject().replaceProperties(str
.toString()));
311 Set
<FpdModuleIdentification
> filesSet
= fvs
.get(validFv
[i
].toUpperCase());
312 if (filesSet
!= null) {
313 FpdModuleIdentification
[] files
= filesSet
.toArray(new FpdModuleIdentification
[filesSet
.size()]);
316 for (int j
= 0; j
< files
.length
; j
++) {
317 String str
= ffsCommonDir
+ File
.separatorChar
+ outfiles
.get(files
[j
]);
318 bw
.write(getProject().replaceProperties("EFI_FILE_NAME = " + str
));
325 } catch (Exception e
) {
327 throw new BuildException("Generate FV file [" + fvFile
.getPath() + "] failed. \n" + e
.getMessage());
332 This method is used for Single Module Build.
335 @throws BuildException
336 FPD file is not valid.
338 public void parseFpdFile(File fpdFile
) throws BuildException
{
339 this.fpdFile
= fpdFile
;
346 @throws BuildException
347 FPD file is not valid.
349 private void parseFpdFile() throws BuildException
{
351 XmlObject doc
= XmlObject
.Factory
.parse(fpdFile
);
353 if (!doc
.validate()) {
354 throw new BuildException("Platform Surface Area file [" + fpdFile
.getPath() + "] is invalid.");
357 Map
<String
, XmlObject
> map
= new HashMap
<String
, XmlObject
>();
358 map
.put("PlatformSurfaceArea", doc
);
359 SurfaceAreaQuery
.setDoc(map
);
364 platformId
= SurfaceAreaQuery
.getFpdHeader();
365 platformId
.setFpdFile(fpdFile
);
366 getProject().setProperty("PLATFORM", platformId
.getName());
367 getProject().setProperty("PLATFORM_DIR", platformId
.getFpdFile().getParent().replaceAll("(\\\\)", "/"));
368 getProject().setProperty("PLATFORM_RELATIVE_DIR", platformId
.getPlatformRelativeDir().replaceAll("(\\\\)", "/"));
371 // Build mode. User-defined output dir.
373 String buildMode
= SurfaceAreaQuery
.getFpdIntermediateDirectories();
374 String userDefinedOutputDir
= SurfaceAreaQuery
.getFpdOutputDirectory();
376 OutputManager
.getInstance().setup(userDefinedOutputDir
, buildMode
);
379 // TBD. Deal PCD and BuildOption related Info
381 GlobalData
.setFpdBuildOptions(SurfaceAreaQuery
.getFpdBuildOptions());
383 GlobalData
.setToolChainPlatformInfo(SurfaceAreaQuery
.getFpdToolChainInfo());
386 // Parse all list modules SA
388 parseModuleSAFiles();
391 // TBD. Deal PCD and BuildOption related Info
393 parseToolChainFamilyOptions();
394 parseToolChainOptions();
396 SurfaceAreaQuery
.setDoc(map
);
397 } catch (Exception e
) {
399 throw new BuildException("Load FPD file [" + fpdFile
.getPath() + "] error. \n" + e
.getMessage());
406 Parse all modules listed in FPD file.
408 private void parseModuleSAFiles() throws EdkException
{
409 Map
<FpdModuleIdentification
, Map
<String
, XmlObject
>> moduleSAs
= SurfaceAreaQuery
.getFpdModules();
412 // For every Module lists in FPD file.
414 Set
<FpdModuleIdentification
> keys
= moduleSAs
.keySet();
415 Iterator iter
= keys
.iterator();
416 while (iter
.hasNext()) {
417 FpdModuleIdentification fpdModuleId
= (FpdModuleIdentification
) iter
.next();
420 // Judge if Module is existed?
423 GlobalData
.registerFpdModuleSA(fpdModuleId
, moduleSAs
.get(fpdModuleId
));
426 // Put fpdModuleId to the corresponding FV
428 SurfaceAreaQuery
.push(GlobalData
.getDoc(fpdModuleId
));
429 String fvBinding
= SurfaceAreaQuery
.getModuleFvBindingKeyword();
430 SurfaceAreaQuery
.pop();
432 fpdModuleId
.setFvBinding(fvBinding
);
433 String fvSequence
= fpdModuleId
.getSequence();
434 updateFvs(fvSequence
, fvBinding
, fpdModuleId
);
437 // Prepare for out put file name
439 ModuleIdentification moduleId
= fpdModuleId
.getModule();
440 SurfaceAreaQuery
.push(GlobalData
.getDoc(fpdModuleId
));
441 String baseName
= SurfaceAreaQuery
.getModuleOutputFileBasename();
442 SurfaceAreaQuery
.pop();
443 if (baseName
== null) {
444 baseName
= moduleId
.getName();
446 outfiles
.put(fpdModuleId
, fpdModuleId
.getArch() + File
.separatorChar
447 + moduleId
.getGuid() + "-" + baseName
448 + getSuffix(moduleId
.getModuleType()));
451 // parse module build options, if any
453 SurfaceAreaQuery
.push(GlobalData
.getDoc(fpdModuleId
));
454 GlobalData
.addModuleToolChainOption(fpdModuleId
, parseModuleBuildOptions(false));
455 GlobalData
.addModuleToolChainFamilyOption(fpdModuleId
, parseModuleBuildOptions(true));
456 SurfaceAreaQuery
.pop();
460 private ToolChainMap
parseModuleBuildOptions(boolean toolChainFamilyFlag
) throws EdkException
{
461 String
[][] options
= SurfaceAreaQuery
.getModuleBuildOptions(toolChainFamilyFlag
);
462 if (options
== null || options
.length
== 0) {
465 return parseOptions(options
);
468 private ToolChainMap
parsePlatformBuildOptions(boolean toolChainFamilyFlag
) throws EdkException
{
469 String
[][] options
= SurfaceAreaQuery
.getPlatformBuildOptions(toolChainFamilyFlag
);
470 if (options
== null || options
.length
== 0) {
473 return parseOptions(options
);
476 private ToolChainMap
parseOptions(String
[][] options
) throws EdkException
{
477 ToolChainMap map
= new ToolChainMap();
478 int flagIndex
= ToolChainElement
.ATTRIBUTE
.value
;
480 for (int i
= 0; i
< options
.length
; ++i
) {
481 String flagString
= options
[i
][flagIndex
];
482 if (flagString
== null) {
485 options
[i
][flagIndex
] = ToolChainAttribute
.FLAGS
+ "";
486 map
.put(options
[i
], flagString
.trim());
492 private void parseToolChainFamilyOptions() throws EdkException
{
493 GlobalData
.setPlatformToolChainFamilyOption(parsePlatformBuildOptions(true));
496 private void parseToolChainOptions() throws EdkException
{
497 GlobalData
.setPlatformToolChainOption(parsePlatformBuildOptions(false));
501 Add the current module to corresponding FV.
503 @param fvName current FV name
504 @param moduleName current module identification
506 private void updateFvs(String fvSequence
, String fvName
, FpdModuleIdentification fpdModuleId
) {
507 String upcaseFvName
= fvName
.toUpperCase();
508 String
[] fvNameArray
= upcaseFvName
.split("[, \t]+");
509 for (int i
= 0; i
< fvNameArray
.length
; i
++) {
511 // Put module to corresponding fvName
513 if (fvs
.containsKey(fvNameArray
[i
])) {
514 Set
<FpdModuleIdentification
> set
= fvs
.get(fvNameArray
[i
]);
515 set
.add(fpdModuleId
);
518 Set
<FpdModuleIdentification
> set
= new LinkedHashSet
<FpdModuleIdentification
>();
519 set
.add(fpdModuleId
);
520 fvs
.put(fvNameArray
[i
], set
);
524 // Put fvName to corresponding fvSequence
526 if (sequences
.containsKey(fvSequence
)) {
527 Set
<String
> set
= sequences
.get(fvSequence
);
528 set
.add(fvNameArray
[i
]);
531 Set
<String
> set
= new LinkedHashSet
<String
>();
532 set
.add(fvNameArray
[i
]);
533 sequences
.put(fvSequence
, set
);
539 Get the suffix based on module type. Current relationship are listed:
542 <b>ModuleType</b> <b>Suffix</b>
549 DXE_RUNTIME_DRIVER .DXE
554 UEFI_APPLICATION .APP
558 @param moduleType module type
560 @throws BuildException
561 If module type is null
563 public static String
getSuffix(String moduleType
) throws BuildException
{
564 if (moduleType
== null) {
565 throw new BuildException("Module type is not specified.");
568 String
[][] suffix
= { { "BASE", ".FFS"},
569 { "SEC", ".SEC" }, { "PEI_CORE", ".PEI" },
570 { "PEIM", ".PEI" }, { "DXE_CORE", ".DXE" },
571 { "DXE_DRIVER", ".DXE" }, { "DXE_RUNTIME_DRIVER", ".DXE" },
572 { "DXE_SAL_DRIVER", ".DXE" }, { "DXE_SMM_DRIVER", ".DXE" },
573 { "TOOL", ".FFS" }, { "UEFI_DRIVER", ".DXE" },
574 { "UEFI_APPLICATION", ".APP" }, { "USER_DEFINED", ".FFS" } };
576 for (int i
= 0; i
< suffix
.length
; i
++) {
577 if (suffix
[i
][0].equalsIgnoreCase(moduleType
)) {
591 public void addProperty(Property p
) {
592 properties
.addElement(p
);
595 public void setPlatformName(String platformName
) {
596 this.platformName
= platformName
;
599 public void setFpdFile(File fpdFile
) {
600 this.fpdFile
= fpdFile
;
603 public void setType(String type
) {