mirror of
https://github.com/Dasharo/edk2.git
synced 2026-03-06 14:47:27 -08:00
71cac3f791
1) use decorators 2) also change some private functions to public when all callers are external 3) change external callers to use functions instead of directly accessing private data. Cc: Liming Gao <liming.gao@intel.com> Cc: Yonghong Zhu <yonghong.zhu@intel.com> Contributed-under: TianoCore Contribution Agreement 1.1 Signed-off-by: Jaben Carsey <jaben.carsey@intel.com> Reviewed-by: Yonghong Zhu <yonghong.zhu@intel.com>
1164 lines
55 KiB
Python
1164 lines
55 KiB
Python
## @file
|
|
# This file is used to create a database used by build tool
|
|
#
|
|
# Copyright (c) 2008 - 2018, Intel Corporation. All rights reserved.<BR>
|
|
# (C) Copyright 2016 Hewlett Packard Enterprise Development LP<BR>
|
|
# This program and the accompanying materials
|
|
# are licensed and made available under the terms and conditions of the BSD License
|
|
# which accompanies this distribution. The full text of the license may be found at
|
|
# http://opensource.org/licenses/bsd-license.php
|
|
#
|
|
# THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
|
|
#
|
|
|
|
from __future__ import absolute_import
|
|
from Common.StringUtils import *
|
|
from Common.DataType import *
|
|
from Common.Misc import *
|
|
from types import *
|
|
from .MetaFileParser import *
|
|
from collections import OrderedDict
|
|
|
|
from Workspace.BuildClassObject import ModuleBuildClassObject, LibraryClassObject, PcdClassObject
|
|
## Module build information from INF file
|
|
#
|
|
# This class is used to retrieve information stored in database and convert them
|
|
# into ModuleBuildClassObject form for easier use for AutoGen.
|
|
#
|
|
class InfBuildData(ModuleBuildClassObject):
|
|
# dict used to convert PCD type in database to string used by build tool
|
|
_PCD_TYPE_STRING_ = {
|
|
MODEL_PCD_FIXED_AT_BUILD : TAB_PCDS_FIXED_AT_BUILD,
|
|
MODEL_PCD_PATCHABLE_IN_MODULE : TAB_PCDS_PATCHABLE_IN_MODULE,
|
|
MODEL_PCD_FEATURE_FLAG : TAB_PCDS_FEATURE_FLAG,
|
|
MODEL_PCD_DYNAMIC : TAB_PCDS_DYNAMIC,
|
|
MODEL_PCD_DYNAMIC_DEFAULT : TAB_PCDS_DYNAMIC,
|
|
MODEL_PCD_DYNAMIC_HII : TAB_PCDS_DYNAMIC_HII,
|
|
MODEL_PCD_DYNAMIC_VPD : TAB_PCDS_DYNAMIC_VPD,
|
|
MODEL_PCD_DYNAMIC_EX : TAB_PCDS_DYNAMIC_EX,
|
|
MODEL_PCD_DYNAMIC_EX_DEFAULT : TAB_PCDS_DYNAMIC_EX,
|
|
MODEL_PCD_DYNAMIC_EX_HII : TAB_PCDS_DYNAMIC_EX_HII,
|
|
MODEL_PCD_DYNAMIC_EX_VPD : TAB_PCDS_DYNAMIC_EX_VPD,
|
|
}
|
|
|
|
# dict used to convert part of [Defines] to members of InfBuildData directly
|
|
_PROPERTY_ = {
|
|
#
|
|
# Required Fields
|
|
#
|
|
TAB_INF_DEFINES_BASE_NAME : "_BaseName",
|
|
TAB_INF_DEFINES_FILE_GUID : "_Guid",
|
|
TAB_INF_DEFINES_MODULE_TYPE : "_ModuleType",
|
|
#
|
|
# Optional Fields
|
|
#
|
|
# TAB_INF_DEFINES_INF_VERSION : "_AutoGenVersion",
|
|
TAB_INF_DEFINES_COMPONENT_TYPE : "_ComponentType",
|
|
TAB_INF_DEFINES_MAKEFILE_NAME : "_MakefileName",
|
|
# TAB_INF_DEFINES_CUSTOM_MAKEFILE : "_CustomMakefile",
|
|
TAB_INF_DEFINES_DPX_SOURCE :"_DxsFile",
|
|
TAB_INF_DEFINES_VERSION_NUMBER : "_Version",
|
|
TAB_INF_DEFINES_VERSION_STRING : "_Version",
|
|
TAB_INF_DEFINES_VERSION : "_Version",
|
|
TAB_INF_DEFINES_PCD_IS_DRIVER : "_PcdIsDriver",
|
|
TAB_INF_DEFINES_SHADOW : "_Shadow",
|
|
|
|
TAB_COMPONENTS_SOURCE_OVERRIDE_PATH : "_SourceOverridePath",
|
|
}
|
|
|
|
# regular expression for converting XXX_FLAGS in [nmake] section to new type
|
|
_NMAKE_FLAG_PATTERN_ = re.compile("(?:EBC_)?([A-Z]+)_(?:STD_|PROJ_|ARCH_)?FLAGS(?:_DLL|_ASL|_EXE)?", re.UNICODE)
|
|
# dict used to convert old tool name used in [nmake] section to new ones
|
|
_TOOL_CODE_ = {
|
|
"C" : "CC",
|
|
BINARY_FILE_TYPE_LIB : "SLINK",
|
|
"LINK" : "DLINK",
|
|
}
|
|
|
|
|
|
## Constructor of DscBuildData
|
|
#
|
|
# Initialize object of DscBuildData
|
|
#
|
|
# @param FilePath The path of platform description file
|
|
# @param RawData The raw data of DSC file
|
|
# @param BuildDataBase Database used to retrieve module/package information
|
|
# @param Arch The target architecture
|
|
# @param Platform The name of platform employing this module
|
|
# @param Macros Macros used for replacement in DSC file
|
|
#
|
|
def __init__(self, FilePath, RawData, BuildDatabase, Arch=TAB_ARCH_COMMON, Target=None, Toolchain=None):
|
|
self.MetaFile = FilePath
|
|
self._ModuleDir = FilePath.Dir
|
|
self._RawData = RawData
|
|
self._Bdb = BuildDatabase
|
|
self._Arch = Arch
|
|
self._Target = Target
|
|
self._Toolchain = Toolchain
|
|
self._Platform = TAB_COMMON
|
|
self._SourceOverridePath = None
|
|
if FilePath.Key in GlobalData.gOverrideDir:
|
|
self._SourceOverridePath = GlobalData.gOverrideDir[FilePath.Key]
|
|
self._Clear()
|
|
|
|
## XXX[key] = value
|
|
def __setitem__(self, key, value):
|
|
self.__dict__[self._PROPERTY_[key]] = value
|
|
|
|
## value = XXX[key]
|
|
def __getitem__(self, key):
|
|
return self.__dict__[self._PROPERTY_[key]]
|
|
|
|
## "in" test support
|
|
def __contains__(self, key):
|
|
return key in self._PROPERTY_
|
|
|
|
## Set all internal used members of InfBuildData to None
|
|
def _Clear(self):
|
|
self._HeaderComments = None
|
|
self._TailComments = None
|
|
self._Header_ = None
|
|
self._AutoGenVersion = None
|
|
self._BaseName = None
|
|
self._DxsFile = None
|
|
self._ModuleType = None
|
|
self._ComponentType = None
|
|
self._BuildType = None
|
|
self._Guid = None
|
|
self._Version = None
|
|
self._PcdIsDriver = None
|
|
self._BinaryModule = None
|
|
self._Shadow = None
|
|
self._MakefileName = None
|
|
self._CustomMakefile = None
|
|
self._Specification = None
|
|
self._LibraryClass = None
|
|
self._ModuleEntryPointList = None
|
|
self._ModuleUnloadImageList = None
|
|
self._ConstructorList = None
|
|
self._DestructorList = None
|
|
self._Defs = OrderedDict()
|
|
self._Binaries = None
|
|
self._Sources = None
|
|
self._LibraryClasses = None
|
|
self._Libraries = None
|
|
self._Protocols = None
|
|
self._ProtocolComments = None
|
|
self._Ppis = None
|
|
self._PpiComments = None
|
|
self._Guids = None
|
|
self._GuidsUsedByPcd = OrderedDict()
|
|
self._GuidComments = None
|
|
self._Includes = None
|
|
self._Packages = None
|
|
self._Pcds = None
|
|
self._PcdComments = None
|
|
self._BuildOptions = None
|
|
self._Depex = None
|
|
self._DepexExpression = None
|
|
self._MacroDict = None
|
|
|
|
## Get current effective macros
|
|
@property
|
|
def _Macros(self):
|
|
if self._MacroDict is None:
|
|
self._MacroDict = {}
|
|
# EDK_GLOBAL defined macros can be applied to EDK module
|
|
if self.AutoGenVersion < 0x00010005:
|
|
self._MacroDict.update(GlobalData.gEdkGlobal)
|
|
self._MacroDict.update(GlobalData.gGlobalDefines)
|
|
return self._MacroDict
|
|
|
|
## Get architecture
|
|
@property
|
|
def Arch(self):
|
|
return self._Arch
|
|
|
|
## Return the name of platform employing this module
|
|
@property
|
|
def Platform(self):
|
|
return self._Platform
|
|
|
|
|
|
@property
|
|
def HeaderComments(self):
|
|
if not self._HeaderComments:
|
|
self._HeaderComments = [a[0] for a in self._RawData[MODEL_META_DATA_HEADER_COMMENT]]
|
|
return self._HeaderComments
|
|
|
|
@property
|
|
def TailComments(self):
|
|
if not self._TailComments:
|
|
self._TailComments = [a[0] for a in self._RawData[MODEL_META_DATA_TAIL_COMMENT]]
|
|
return self._TailComments
|
|
|
|
## Retrieve all information in [Defines] section
|
|
#
|
|
# (Retriving all [Defines] information in one-shot is just to save time.)
|
|
#
|
|
def _GetHeaderInfo(self):
|
|
RecordList = self._RawData[MODEL_META_DATA_HEADER, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
Name, Value = Record[1], ReplaceMacro(Record[2], self._Macros, False)
|
|
# items defined _PROPERTY_ don't need additional processing
|
|
if Name in self:
|
|
self[Name] = Value
|
|
self._Defs[Name] = Value
|
|
self._Macros[Name] = Value
|
|
# some special items in [Defines] section need special treatment
|
|
elif Name in ('EFI_SPECIFICATION_VERSION', 'UEFI_SPECIFICATION_VERSION', 'EDK_RELEASE_VERSION', 'PI_SPECIFICATION_VERSION'):
|
|
if Name in ('EFI_SPECIFICATION_VERSION', 'UEFI_SPECIFICATION_VERSION'):
|
|
Name = 'UEFI_SPECIFICATION_VERSION'
|
|
if self._Specification is None:
|
|
self._Specification = OrderedDict()
|
|
self._Specification[Name] = GetHexVerValue(Value)
|
|
if self._Specification[Name] is None:
|
|
EdkLogger.error("build", FORMAT_NOT_SUPPORTED,
|
|
"'%s' format is not supported for %s" % (Value, Name),
|
|
File=self.MetaFile, Line=Record[-1])
|
|
elif Name == 'LIBRARY_CLASS':
|
|
if self._LibraryClass is None:
|
|
self._LibraryClass = []
|
|
ValueList = GetSplitValueList(Value)
|
|
LibraryClass = ValueList[0]
|
|
if len(ValueList) > 1:
|
|
SupModuleList = GetSplitValueList(ValueList[1], ' ')
|
|
else:
|
|
SupModuleList = SUP_MODULE_LIST
|
|
self._LibraryClass.append(LibraryClassObject(LibraryClass, SupModuleList))
|
|
elif Name == 'ENTRY_POINT':
|
|
if self._ModuleEntryPointList is None:
|
|
self._ModuleEntryPointList = []
|
|
self._ModuleEntryPointList.append(Value)
|
|
elif Name == 'UNLOAD_IMAGE':
|
|
if self._ModuleUnloadImageList is None:
|
|
self._ModuleUnloadImageList = []
|
|
if not Value:
|
|
continue
|
|
self._ModuleUnloadImageList.append(Value)
|
|
elif Name == 'CONSTRUCTOR':
|
|
if self._ConstructorList is None:
|
|
self._ConstructorList = []
|
|
if not Value:
|
|
continue
|
|
self._ConstructorList.append(Value)
|
|
elif Name == 'DESTRUCTOR':
|
|
if self._DestructorList is None:
|
|
self._DestructorList = []
|
|
if not Value:
|
|
continue
|
|
self._DestructorList.append(Value)
|
|
elif Name == TAB_INF_DEFINES_CUSTOM_MAKEFILE:
|
|
TokenList = GetSplitValueList(Value)
|
|
if self._CustomMakefile is None:
|
|
self._CustomMakefile = {}
|
|
if len(TokenList) < 2:
|
|
self._CustomMakefile[TAB_COMPILER_MSFT] = TokenList[0]
|
|
self._CustomMakefile['GCC'] = TokenList[0]
|
|
else:
|
|
if TokenList[0] not in [TAB_COMPILER_MSFT, 'GCC']:
|
|
EdkLogger.error("build", FORMAT_NOT_SUPPORTED,
|
|
"No supported family [%s]" % TokenList[0],
|
|
File=self.MetaFile, Line=Record[-1])
|
|
self._CustomMakefile[TokenList[0]] = TokenList[1]
|
|
else:
|
|
self._Defs[Name] = Value
|
|
self._Macros[Name] = Value
|
|
|
|
#
|
|
# Retrieve information in sections specific to Edk.x modules
|
|
#
|
|
if self.AutoGenVersion >= 0x00010005:
|
|
if not self._ModuleType:
|
|
EdkLogger.error("build", ATTRIBUTE_NOT_AVAILABLE,
|
|
"MODULE_TYPE is not given", File=self.MetaFile)
|
|
if self._ModuleType not in SUP_MODULE_LIST:
|
|
RecordList = self._RawData[MODEL_META_DATA_HEADER, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
Name = Record[1]
|
|
if Name == "MODULE_TYPE":
|
|
LineNo = Record[6]
|
|
break
|
|
EdkLogger.error("build", FORMAT_NOT_SUPPORTED,
|
|
"MODULE_TYPE %s is not supported for EDK II, valid values are:\n %s" % (self._ModuleType, ' '.join(l for l in SUP_MODULE_LIST)),
|
|
File=self.MetaFile, Line=LineNo)
|
|
if (self._Specification is None) or (not 'PI_SPECIFICATION_VERSION' in self._Specification) or (int(self._Specification['PI_SPECIFICATION_VERSION'], 16) < 0x0001000A):
|
|
if self._ModuleType == SUP_MODULE_SMM_CORE:
|
|
EdkLogger.error("build", FORMAT_NOT_SUPPORTED, "SMM_CORE module type can't be used in the module with PI_SPECIFICATION_VERSION less than 0x0001000A", File=self.MetaFile)
|
|
if (self._Specification is None) or (not 'PI_SPECIFICATION_VERSION' in self._Specification) or (int(self._Specification['PI_SPECIFICATION_VERSION'], 16) < 0x00010032):
|
|
if self._ModuleType == SUP_MODULE_MM_CORE_STANDALONE:
|
|
EdkLogger.error("build", FORMAT_NOT_SUPPORTED, "MM_CORE_STANDALONE module type can't be used in the module with PI_SPECIFICATION_VERSION less than 0x00010032", File=self.MetaFile)
|
|
if self._ModuleType == SUP_MODULE_MM_STANDALONE:
|
|
EdkLogger.error("build", FORMAT_NOT_SUPPORTED, "MM_STANDALONE module type can't be used in the module with PI_SPECIFICATION_VERSION less than 0x00010032", File=self.MetaFile)
|
|
if 'PCI_DEVICE_ID' in self._Defs and 'PCI_VENDOR_ID' in self._Defs \
|
|
and 'PCI_CLASS_CODE' in self._Defs and 'PCI_REVISION' in self._Defs:
|
|
self._BuildType = 'UEFI_OPTIONROM'
|
|
if 'PCI_COMPRESS' in self._Defs:
|
|
if self._Defs['PCI_COMPRESS'] not in ('TRUE', 'FALSE'):
|
|
EdkLogger.error("build", FORMAT_INVALID, "Expected TRUE/FALSE for PCI_COMPRESS: %s" % self.MetaFile)
|
|
|
|
elif 'UEFI_HII_RESOURCE_SECTION' in self._Defs \
|
|
and self._Defs['UEFI_HII_RESOURCE_SECTION'] == 'TRUE':
|
|
self._BuildType = 'UEFI_HII'
|
|
else:
|
|
self._BuildType = self._ModuleType.upper()
|
|
|
|
if self._DxsFile:
|
|
File = PathClass(NormPath(self._DxsFile), self._ModuleDir, Arch=self._Arch)
|
|
# check the file validation
|
|
ErrorCode, ErrorInfo = File.Validate(".dxs", CaseSensitive=False)
|
|
if ErrorCode != 0:
|
|
EdkLogger.error('build', ErrorCode, ExtraData=ErrorInfo,
|
|
File=self.MetaFile, Line=LineNo)
|
|
if self.Sources is None:
|
|
self._Sources = []
|
|
self._Sources.append(File)
|
|
else:
|
|
if not self._ComponentType:
|
|
EdkLogger.error("build", ATTRIBUTE_NOT_AVAILABLE,
|
|
"COMPONENT_TYPE is not given", File=self.MetaFile)
|
|
self._BuildType = self._ComponentType.upper()
|
|
if self._ComponentType in COMPONENT_TO_MODULE_MAP_DICT:
|
|
self._ModuleType = COMPONENT_TO_MODULE_MAP_DICT[self._ComponentType]
|
|
if self._ComponentType == EDK_COMPONENT_TYPE_LIBRARY:
|
|
self._LibraryClass = [LibraryClassObject(self._BaseName, SUP_MODULE_LIST)]
|
|
# make use some [nmake] section macros
|
|
Macros = self._Macros
|
|
Macros["EDK_SOURCE"] = GlobalData.gEcpSource
|
|
Macros['PROCESSOR'] = self._Arch
|
|
RecordList = self._RawData[MODEL_META_DATA_NMAKE, self._Arch, self._Platform]
|
|
for Name, Value, Dummy, Arch, Platform, ID, LineNo in RecordList:
|
|
Value = ReplaceMacro(Value, Macros, True)
|
|
if Name == "IMAGE_ENTRY_POINT":
|
|
if self._ModuleEntryPointList is None:
|
|
self._ModuleEntryPointList = []
|
|
self._ModuleEntryPointList.append(Value)
|
|
elif Name == "DPX_SOURCE":
|
|
File = PathClass(NormPath(Value), self._ModuleDir, Arch=self._Arch)
|
|
# check the file validation
|
|
ErrorCode, ErrorInfo = File.Validate(".dxs", CaseSensitive=False)
|
|
if ErrorCode != 0:
|
|
EdkLogger.error('build', ErrorCode, ExtraData=ErrorInfo,
|
|
File=self.MetaFile, Line=LineNo)
|
|
if self.Sources is None:
|
|
self._Sources = []
|
|
self._Sources.append(File)
|
|
else:
|
|
ToolList = self._NMAKE_FLAG_PATTERN_.findall(Name)
|
|
if len(ToolList) == 1:
|
|
if self._BuildOptions is None:
|
|
self._BuildOptions = OrderedDict()
|
|
|
|
if ToolList[0] in self._TOOL_CODE_:
|
|
Tool = self._TOOL_CODE_[ToolList[0]]
|
|
else:
|
|
Tool = ToolList[0]
|
|
ToolChain = "*_*_*_%s_FLAGS" % Tool
|
|
# Edk.x only support MSFT tool chain
|
|
# ignore not replaced macros in value
|
|
ValueList = GetSplitList(' ' + Value, '/D')
|
|
Dummy = ValueList[0]
|
|
for Index in range(1, len(ValueList)):
|
|
if ValueList[Index][-1] == '=' or ValueList[Index] == '':
|
|
continue
|
|
Dummy = Dummy + ' /D ' + ValueList[Index]
|
|
Value = Dummy.strip()
|
|
if (TAB_COMPILER_MSFT, ToolChain) not in self._BuildOptions:
|
|
self._BuildOptions[TAB_COMPILER_MSFT, ToolChain] = Value
|
|
else:
|
|
OptionString = self._BuildOptions[TAB_COMPILER_MSFT, ToolChain]
|
|
self._BuildOptions[TAB_COMPILER_MSFT, ToolChain] = OptionString + " " + Value
|
|
# set _Header to non-None in order to avoid database re-querying
|
|
self._Header_ = 'DUMMY'
|
|
|
|
## Retrieve file version
|
|
@property
|
|
def AutoGenVersion(self):
|
|
if self._AutoGenVersion is None:
|
|
RecordList = self._RawData[MODEL_META_DATA_HEADER, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
if Record[1] == TAB_INF_DEFINES_INF_VERSION:
|
|
if '.' in Record[2]:
|
|
ValueList = Record[2].split('.')
|
|
Major = '%04o' % int(ValueList[0], 0)
|
|
Minor = '%04o' % int(ValueList[1], 0)
|
|
self._AutoGenVersion = int('0x' + Major + Minor, 0)
|
|
else:
|
|
self._AutoGenVersion = int(Record[2], 0)
|
|
break
|
|
if self._AutoGenVersion is None:
|
|
self._AutoGenVersion = 0x00010000
|
|
return self._AutoGenVersion
|
|
|
|
## Retrieve BASE_NAME
|
|
@property
|
|
def BaseName(self):
|
|
if self._BaseName is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._BaseName is None:
|
|
EdkLogger.error('build', ATTRIBUTE_NOT_AVAILABLE, "No BASE_NAME name", File=self.MetaFile)
|
|
return self._BaseName
|
|
|
|
## Retrieve DxsFile
|
|
@property
|
|
def DxsFile(self):
|
|
if self._DxsFile is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._DxsFile is None:
|
|
self._DxsFile = ''
|
|
return self._DxsFile
|
|
|
|
## Retrieve MODULE_TYPE
|
|
@property
|
|
def ModuleType(self):
|
|
if self._ModuleType is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._ModuleType is None:
|
|
self._ModuleType = SUP_MODULE_BASE
|
|
if self._ModuleType not in SUP_MODULE_LIST:
|
|
self._ModuleType = SUP_MODULE_USER_DEFINED
|
|
return self._ModuleType
|
|
|
|
## Retrieve COMPONENT_TYPE
|
|
@property
|
|
def ComponentType(self):
|
|
if self._ComponentType is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._ComponentType is None:
|
|
self._ComponentType = SUP_MODULE_USER_DEFINED
|
|
return self._ComponentType
|
|
|
|
## Retrieve "BUILD_TYPE"
|
|
@property
|
|
def BuildType(self):
|
|
if self._BuildType is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if not self._BuildType:
|
|
self._BuildType = SUP_MODULE_BASE
|
|
return self._BuildType
|
|
|
|
## Retrieve file guid
|
|
@property
|
|
def Guid(self):
|
|
if self._Guid is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._Guid is None:
|
|
self._Guid = '00000000-0000-0000-0000-000000000000'
|
|
return self._Guid
|
|
|
|
## Retrieve module version
|
|
@property
|
|
def Version(self):
|
|
if self._Version is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._Version is None:
|
|
self._Version = '0.0'
|
|
return self._Version
|
|
|
|
## Retrieve PCD_IS_DRIVER
|
|
@property
|
|
def PcdIsDriver(self):
|
|
if self._PcdIsDriver is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._PcdIsDriver is None:
|
|
self._PcdIsDriver = ''
|
|
return self._PcdIsDriver
|
|
|
|
## Retrieve SHADOW
|
|
@property
|
|
def Shadow(self):
|
|
if self._Shadow is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._Shadow is not None and self._Shadow.upper() == 'TRUE':
|
|
self._Shadow = True
|
|
else:
|
|
self._Shadow = False
|
|
return self._Shadow
|
|
|
|
## Retrieve CUSTOM_MAKEFILE
|
|
@property
|
|
def CustomMakefile(self):
|
|
if self._CustomMakefile is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._CustomMakefile is None:
|
|
self._CustomMakefile = {}
|
|
return self._CustomMakefile
|
|
|
|
## Retrieve EFI_SPECIFICATION_VERSION
|
|
@property
|
|
def Specification(self):
|
|
if self._Specification is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._Specification is None:
|
|
self._Specification = {}
|
|
return self._Specification
|
|
|
|
## Retrieve LIBRARY_CLASS
|
|
@property
|
|
def LibraryClass(self):
|
|
if self._LibraryClass is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._LibraryClass is None:
|
|
self._LibraryClass = []
|
|
return self._LibraryClass
|
|
|
|
## Retrieve ENTRY_POINT
|
|
@property
|
|
def ModuleEntryPointList(self):
|
|
if self._ModuleEntryPointList is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._ModuleEntryPointList is None:
|
|
self._ModuleEntryPointList = []
|
|
return self._ModuleEntryPointList
|
|
|
|
## Retrieve UNLOAD_IMAGE
|
|
@property
|
|
def ModuleUnloadImageList(self):
|
|
if self._ModuleUnloadImageList is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._ModuleUnloadImageList is None:
|
|
self._ModuleUnloadImageList = []
|
|
return self._ModuleUnloadImageList
|
|
|
|
## Retrieve CONSTRUCTOR
|
|
@property
|
|
def ConstructorList(self):
|
|
if self._ConstructorList is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._ConstructorList is None:
|
|
self._ConstructorList = []
|
|
return self._ConstructorList
|
|
|
|
## Retrieve DESTRUCTOR
|
|
@property
|
|
def DestructorList(self):
|
|
if self._DestructorList is None:
|
|
if self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
if self._DestructorList is None:
|
|
self._DestructorList = []
|
|
return self._DestructorList
|
|
|
|
## Retrieve definies other than above ones
|
|
@property
|
|
def Defines(self):
|
|
if len(self._Defs) == 0 and self._Header_ is None:
|
|
self._GetHeaderInfo()
|
|
return self._Defs
|
|
|
|
## Retrieve binary files
|
|
def _GetBinaries(self):
|
|
if self._Binaries is None:
|
|
self._Binaries = []
|
|
RecordList = self._RawData[MODEL_EFI_BINARY_FILE, self._Arch, self._Platform]
|
|
Macros = self._Macros
|
|
Macros["EDK_SOURCE"] = GlobalData.gEcpSource
|
|
Macros['PROCESSOR'] = self._Arch
|
|
for Record in RecordList:
|
|
FileType = Record[0]
|
|
LineNo = Record[-1]
|
|
Target = TAB_COMMON
|
|
FeatureFlag = []
|
|
if Record[2]:
|
|
TokenList = GetSplitValueList(Record[2], TAB_VALUE_SPLIT)
|
|
if TokenList:
|
|
Target = TokenList[0]
|
|
if len(TokenList) > 1:
|
|
FeatureFlag = Record[1:]
|
|
|
|
File = PathClass(NormPath(Record[1], Macros), self._ModuleDir, '', FileType, True, self._Arch, '', Target)
|
|
# check the file validation
|
|
ErrorCode, ErrorInfo = File.Validate()
|
|
if ErrorCode != 0:
|
|
EdkLogger.error('build', ErrorCode, ExtraData=ErrorInfo, File=self.MetaFile, Line=LineNo)
|
|
self._Binaries.append(File)
|
|
return self._Binaries
|
|
|
|
## Retrieve binary files with error check.
|
|
@property
|
|
def Binaries(self):
|
|
Binaries = self._GetBinaries()
|
|
if GlobalData.gIgnoreSource and Binaries == []:
|
|
ErrorInfo = "The INF file does not contain any Binaries to use in creating the image\n"
|
|
EdkLogger.error('build', RESOURCE_NOT_AVAILABLE, ExtraData=ErrorInfo, File=self.MetaFile)
|
|
|
|
return Binaries
|
|
|
|
## Retrieve source files
|
|
@property
|
|
def Sources(self):
|
|
# Ignore all source files in a binary build mode
|
|
if GlobalData.gIgnoreSource:
|
|
self._Sources = []
|
|
return self._Sources
|
|
|
|
if self._Sources is None:
|
|
self._Sources = []
|
|
RecordList = self._RawData[MODEL_EFI_SOURCE_FILE, self._Arch, self._Platform]
|
|
Macros = self._Macros
|
|
for Record in RecordList:
|
|
LineNo = Record[-1]
|
|
ToolChainFamily = Record[1]
|
|
TagName = Record[2]
|
|
ToolCode = Record[3]
|
|
FeatureFlag = Record[4]
|
|
if self.AutoGenVersion < 0x00010005:
|
|
Macros["EDK_SOURCE"] = GlobalData.gEcpSource
|
|
Macros['PROCESSOR'] = self._Arch
|
|
SourceFile = NormPath(Record[0], Macros)
|
|
if SourceFile[0] == os.path.sep:
|
|
SourceFile = mws.join(GlobalData.gWorkspace, SourceFile[1:])
|
|
# old module source files (Edk)
|
|
File = PathClass(SourceFile, self._ModuleDir, self._SourceOverridePath,
|
|
'', False, self._Arch, ToolChainFamily, '', TagName, ToolCode)
|
|
# check the file validation
|
|
ErrorCode, ErrorInfo = File.Validate(CaseSensitive=False)
|
|
if ErrorCode != 0:
|
|
if File.Ext.lower() == '.h':
|
|
EdkLogger.warn('build', 'Include file not found', ExtraData=ErrorInfo,
|
|
File=self.MetaFile, Line=LineNo)
|
|
continue
|
|
else:
|
|
EdkLogger.error('build', ErrorCode, ExtraData=File, File=self.MetaFile, Line=LineNo)
|
|
else:
|
|
File = PathClass(NormPath(Record[0], Macros), self._ModuleDir, '',
|
|
'', False, self._Arch, ToolChainFamily, '', TagName, ToolCode)
|
|
# check the file validation
|
|
ErrorCode, ErrorInfo = File.Validate()
|
|
if ErrorCode != 0:
|
|
EdkLogger.error('build', ErrorCode, ExtraData=ErrorInfo, File=self.MetaFile, Line=LineNo)
|
|
|
|
self._Sources.append(File)
|
|
return self._Sources
|
|
|
|
## Retrieve library classes employed by this module
|
|
@property
|
|
def LibraryClasses(self):
|
|
if self._LibraryClasses is None:
|
|
self._LibraryClasses = OrderedDict()
|
|
RecordList = self._RawData[MODEL_EFI_LIBRARY_CLASS, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
Lib = Record[0]
|
|
Instance = Record[1]
|
|
if Instance:
|
|
Instance = NormPath(Instance, self._Macros)
|
|
self._LibraryClasses[Lib] = Instance
|
|
return self._LibraryClasses
|
|
|
|
## Retrieve library names (for Edk.x style of modules)
|
|
@property
|
|
def Libraries(self):
|
|
if self._Libraries is None:
|
|
self._Libraries = []
|
|
RecordList = self._RawData[MODEL_EFI_LIBRARY_INSTANCE, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
LibraryName = ReplaceMacro(Record[0], self._Macros, False)
|
|
# in case of name with '.lib' extension, which is unusual in Edk.x inf
|
|
LibraryName = os.path.splitext(LibraryName)[0]
|
|
if LibraryName not in self._Libraries:
|
|
self._Libraries.append(LibraryName)
|
|
return self._Libraries
|
|
|
|
@property
|
|
def ProtocolComments(self):
|
|
self.Protocols
|
|
return self._ProtocolComments
|
|
|
|
## Retrieve protocols consumed/produced by this module
|
|
@property
|
|
def Protocols(self):
|
|
if self._Protocols is None:
|
|
self._Protocols = OrderedDict()
|
|
self._ProtocolComments = OrderedDict()
|
|
RecordList = self._RawData[MODEL_EFI_PROTOCOL, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
CName = Record[0]
|
|
Value = ProtocolValue(CName, self.Packages, self.MetaFile.Path)
|
|
if Value is None:
|
|
PackageList = "\n\t".join(str(P) for P in self.Packages)
|
|
EdkLogger.error('build', RESOURCE_NOT_AVAILABLE,
|
|
"Value of Protocol [%s] is not found under [Protocols] section in" % CName,
|
|
ExtraData=PackageList, File=self.MetaFile, Line=Record[-1])
|
|
self._Protocols[CName] = Value
|
|
CommentRecords = self._RawData[MODEL_META_DATA_COMMENT, self._Arch, self._Platform, Record[5]]
|
|
Comments = []
|
|
for CmtRec in CommentRecords:
|
|
Comments.append(CmtRec[0])
|
|
self._ProtocolComments[CName] = Comments
|
|
return self._Protocols
|
|
|
|
@property
|
|
def PpiComments(self):
|
|
self.Ppis
|
|
return self._PpiComments
|
|
|
|
## Retrieve PPIs consumed/produced by this module
|
|
@property
|
|
def Ppis(self):
|
|
if self._Ppis is None:
|
|
self._Ppis = OrderedDict()
|
|
self._PpiComments = OrderedDict()
|
|
RecordList = self._RawData[MODEL_EFI_PPI, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
CName = Record[0]
|
|
Value = PpiValue(CName, self.Packages, self.MetaFile.Path)
|
|
if Value is None:
|
|
PackageList = "\n\t".join(str(P) for P in self.Packages)
|
|
EdkLogger.error('build', RESOURCE_NOT_AVAILABLE,
|
|
"Value of PPI [%s] is not found under [Ppis] section in " % CName,
|
|
ExtraData=PackageList, File=self.MetaFile, Line=Record[-1])
|
|
self._Ppis[CName] = Value
|
|
CommentRecords = self._RawData[MODEL_META_DATA_COMMENT, self._Arch, self._Platform, Record[5]]
|
|
Comments = []
|
|
for CmtRec in CommentRecords:
|
|
Comments.append(CmtRec[0])
|
|
self._PpiComments[CName] = Comments
|
|
return self._Ppis
|
|
|
|
@property
|
|
def GuidComments(self):
|
|
self.Guids
|
|
return self._GuidComments
|
|
|
|
## Retrieve GUIDs consumed/produced by this module
|
|
@property
|
|
def Guids(self):
|
|
if self._Guids is None:
|
|
self._Guids = OrderedDict()
|
|
self._GuidComments = OrderedDict()
|
|
RecordList = self._RawData[MODEL_EFI_GUID, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
CName = Record[0]
|
|
Value = GuidValue(CName, self.Packages, self.MetaFile.Path)
|
|
if Value is None:
|
|
PackageList = "\n\t".join(str(P) for P in self.Packages)
|
|
EdkLogger.error('build', RESOURCE_NOT_AVAILABLE,
|
|
"Value of Guid [%s] is not found under [Guids] section in" % CName,
|
|
ExtraData=PackageList, File=self.MetaFile, Line=Record[-1])
|
|
self._Guids[CName] = Value
|
|
CommentRecords = self._RawData[MODEL_META_DATA_COMMENT, self._Arch, self._Platform, Record[5]]
|
|
Comments = []
|
|
for CmtRec in CommentRecords:
|
|
Comments.append(CmtRec[0])
|
|
self._GuidComments[CName] = Comments
|
|
return self._Guids
|
|
|
|
## Retrieve include paths necessary for this module (for Edk.x style of modules)
|
|
@property
|
|
def Includes(self):
|
|
if self._Includes is None:
|
|
self._Includes = []
|
|
if self._SourceOverridePath:
|
|
self._Includes.append(self._SourceOverridePath)
|
|
|
|
Macros = self._Macros
|
|
Macros['PROCESSOR'] = GlobalData.gEdkGlobal.get('PROCESSOR', self._Arch)
|
|
RecordList = self._RawData[MODEL_EFI_INCLUDE, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
if Record[0].find('EDK_SOURCE') > -1:
|
|
Macros['EDK_SOURCE'] = GlobalData.gEcpSource
|
|
File = NormPath(Record[0], self._Macros)
|
|
if File[0] == '.':
|
|
File = os.path.join(self._ModuleDir, File)
|
|
else:
|
|
File = os.path.join(GlobalData.gWorkspace, File)
|
|
File = RealPath(os.path.normpath(File))
|
|
if File:
|
|
self._Includes.append(File)
|
|
|
|
# TRICK: let compiler to choose correct header file
|
|
Macros['EDK_SOURCE'] = GlobalData.gEdkSource
|
|
File = NormPath(Record[0], self._Macros)
|
|
if File[0] == '.':
|
|
File = os.path.join(self._ModuleDir, File)
|
|
else:
|
|
File = os.path.join(GlobalData.gWorkspace, File)
|
|
File = RealPath(os.path.normpath(File))
|
|
if File:
|
|
self._Includes.append(File)
|
|
else:
|
|
File = NormPath(Record[0], Macros)
|
|
if File[0] == '.':
|
|
File = os.path.join(self._ModuleDir, File)
|
|
else:
|
|
File = mws.join(GlobalData.gWorkspace, File)
|
|
File = RealPath(os.path.normpath(File))
|
|
if File:
|
|
self._Includes.append(File)
|
|
if not File and Record[0].find('EFI_SOURCE') > -1:
|
|
# tricky to regard WorkSpace as EFI_SOURCE
|
|
Macros['EFI_SOURCE'] = GlobalData.gWorkspace
|
|
File = NormPath(Record[0], Macros)
|
|
if File[0] == '.':
|
|
File = os.path.join(self._ModuleDir, File)
|
|
else:
|
|
File = os.path.join(GlobalData.gWorkspace, File)
|
|
File = RealPath(os.path.normpath(File))
|
|
if File:
|
|
self._Includes.append(File)
|
|
return self._Includes
|
|
|
|
## Retrieve packages this module depends on
|
|
@property
|
|
def Packages(self):
|
|
if self._Packages is None:
|
|
self._Packages = []
|
|
RecordList = self._RawData[MODEL_META_DATA_PACKAGE, self._Arch, self._Platform]
|
|
Macros = self._Macros
|
|
Macros['EDK_SOURCE'] = GlobalData.gEcpSource
|
|
for Record in RecordList:
|
|
File = PathClass(NormPath(Record[0], Macros), GlobalData.gWorkspace, Arch=self._Arch)
|
|
LineNo = Record[-1]
|
|
# check the file validation
|
|
ErrorCode, ErrorInfo = File.Validate('.dec')
|
|
if ErrorCode != 0:
|
|
EdkLogger.error('build', ErrorCode, ExtraData=ErrorInfo, File=self.MetaFile, Line=LineNo)
|
|
# parse this package now. we need it to get protocol/ppi/guid value
|
|
Package = self._Bdb[File, self._Arch, self._Target, self._Toolchain]
|
|
self._Packages.append(Package)
|
|
return self._Packages
|
|
|
|
## Retrieve PCD comments
|
|
@property
|
|
def PcdComments(self):
|
|
self.Pcds
|
|
return self._PcdComments
|
|
|
|
## Retrieve PCDs used in this module
|
|
@property
|
|
def Pcds(self):
|
|
if self._Pcds is None:
|
|
self._Pcds = OrderedDict()
|
|
self._PcdComments = OrderedDict()
|
|
self._Pcds.update(self._GetPcd(MODEL_PCD_FIXED_AT_BUILD))
|
|
self._Pcds.update(self._GetPcd(MODEL_PCD_PATCHABLE_IN_MODULE))
|
|
self._Pcds.update(self._GetPcd(MODEL_PCD_FEATURE_FLAG))
|
|
self._Pcds.update(self._GetPcd(MODEL_PCD_DYNAMIC))
|
|
self._Pcds.update(self._GetPcd(MODEL_PCD_DYNAMIC_EX))
|
|
return self._Pcds
|
|
|
|
## Retrieve build options specific to this module
|
|
@property
|
|
def BuildOptions(self):
|
|
if self._BuildOptions is None:
|
|
self._BuildOptions = OrderedDict()
|
|
RecordList = self._RawData[MODEL_META_DATA_BUILD_OPTION, self._Arch, self._Platform]
|
|
for Record in RecordList:
|
|
ToolChainFamily = Record[0]
|
|
ToolChain = Record[1]
|
|
Option = Record[2]
|
|
if (ToolChainFamily, ToolChain) not in self._BuildOptions or Option.startswith('='):
|
|
self._BuildOptions[ToolChainFamily, ToolChain] = Option
|
|
else:
|
|
# concatenate the option string if they're for the same tool
|
|
OptionString = self._BuildOptions[ToolChainFamily, ToolChain]
|
|
self._BuildOptions[ToolChainFamily, ToolChain] = OptionString + " " + Option
|
|
return self._BuildOptions
|
|
|
|
## Retrieve dependency expression
|
|
@property
|
|
def Depex(self):
|
|
if self._Depex is None:
|
|
self._Depex = tdict(False, 2)
|
|
RecordList = self._RawData[MODEL_EFI_DEPEX, self._Arch]
|
|
|
|
# If the module has only Binaries and no Sources, then ignore [Depex]
|
|
if self.Sources is None or self.Sources == []:
|
|
if self.Binaries is not None and self.Binaries != []:
|
|
return self._Depex
|
|
|
|
# PEIM and DXE drivers must have a valid [Depex] section
|
|
if len(self.LibraryClass) == 0 and len(RecordList) == 0:
|
|
if self.ModuleType == SUP_MODULE_DXE_DRIVER or self.ModuleType == SUP_MODULE_PEIM or self.ModuleType == SUP_MODULE_DXE_SMM_DRIVER or \
|
|
self.ModuleType == SUP_MODULE_DXE_SAL_DRIVER or self.ModuleType == SUP_MODULE_DXE_RUNTIME_DRIVER:
|
|
EdkLogger.error('build', RESOURCE_NOT_AVAILABLE, "No [Depex] section or no valid expression in [Depex] section for [%s] module" \
|
|
% self.ModuleType, File=self.MetaFile)
|
|
|
|
if len(RecordList) != 0 and self.ModuleType == SUP_MODULE_USER_DEFINED:
|
|
for Record in RecordList:
|
|
if Record[4] not in [SUP_MODULE_PEIM, SUP_MODULE_DXE_DRIVER, SUP_MODULE_DXE_SMM_DRIVER]:
|
|
EdkLogger.error('build', FORMAT_INVALID,
|
|
"'%s' module must specify the type of [Depex] section" % self.ModuleType,
|
|
File=self.MetaFile)
|
|
|
|
Depex = OrderedDict()
|
|
for Record in RecordList:
|
|
DepexStr = ReplaceMacro(Record[0], self._Macros, False)
|
|
Arch = Record[3]
|
|
ModuleType = Record[4]
|
|
TokenList = DepexStr.split()
|
|
if (Arch, ModuleType) not in Depex:
|
|
Depex[Arch, ModuleType] = []
|
|
DepexList = Depex[Arch, ModuleType]
|
|
for Token in TokenList:
|
|
if Token in DEPEX_SUPPORTED_OPCODE_SET:
|
|
DepexList.append(Token)
|
|
elif Token.endswith(".inf"): # module file name
|
|
ModuleFile = os.path.normpath(Token)
|
|
Module = self.BuildDatabase[ModuleFile]
|
|
if Module is None:
|
|
EdkLogger.error('build', RESOURCE_NOT_AVAILABLE, "Module is not found in active platform",
|
|
ExtraData=Token, File=self.MetaFile, Line=Record[-1])
|
|
DepexList.append(Module.Guid)
|
|
else:
|
|
# it use the Fixed PCD format
|
|
if '.' in Token:
|
|
if tuple(Token.split('.')[::-1]) not in self.Pcds:
|
|
EdkLogger.error('build', RESOURCE_NOT_AVAILABLE, "PCD [{}] used in [Depex] section should be listed in module PCD section".format(Token), File=self.MetaFile, Line=Record[-1])
|
|
else:
|
|
if self.Pcds[tuple(Token.split('.')[::-1])].DatumType != TAB_VOID:
|
|
EdkLogger.error('build', FORMAT_INVALID, "PCD [{}] used in [Depex] section should be VOID* datum type".format(Token), File=self.MetaFile, Line=Record[-1])
|
|
Value = Token
|
|
else:
|
|
# get the GUID value now
|
|
Value = ProtocolValue(Token, self.Packages, self.MetaFile.Path)
|
|
if Value is None:
|
|
Value = PpiValue(Token, self.Packages, self.MetaFile.Path)
|
|
if Value is None:
|
|
Value = GuidValue(Token, self.Packages, self.MetaFile.Path)
|
|
|
|
if Value is None:
|
|
PackageList = "\n\t".join(str(P) for P in self.Packages)
|
|
EdkLogger.error('build', RESOURCE_NOT_AVAILABLE,
|
|
"Value of [%s] is not found in" % Token,
|
|
ExtraData=PackageList, File=self.MetaFile, Line=Record[-1])
|
|
DepexList.append(Value)
|
|
for Arch, ModuleType in Depex:
|
|
self._Depex[Arch, ModuleType] = Depex[Arch, ModuleType]
|
|
return self._Depex
|
|
|
|
## Retrieve depedency expression
|
|
@property
|
|
def DepexExpression(self):
|
|
if self._DepexExpression is None:
|
|
self._DepexExpression = tdict(False, 2)
|
|
RecordList = self._RawData[MODEL_EFI_DEPEX, self._Arch]
|
|
DepexExpression = OrderedDict()
|
|
for Record in RecordList:
|
|
DepexStr = ReplaceMacro(Record[0], self._Macros, False)
|
|
Arch = Record[3]
|
|
ModuleType = Record[4]
|
|
TokenList = DepexStr.split()
|
|
if (Arch, ModuleType) not in DepexExpression:
|
|
DepexExpression[Arch, ModuleType] = ''
|
|
for Token in TokenList:
|
|
DepexExpression[Arch, ModuleType] = DepexExpression[Arch, ModuleType] + Token.strip() + ' '
|
|
for Arch, ModuleType in DepexExpression:
|
|
self._DepexExpression[Arch, ModuleType] = DepexExpression[Arch, ModuleType]
|
|
return self._DepexExpression
|
|
|
|
def GetGuidsUsedByPcd(self):
|
|
return self._GuidsUsedByPcd
|
|
|
|
## Retrieve PCD for given type
|
|
def _GetPcd(self, Type):
|
|
Pcds = OrderedDict()
|
|
PcdDict = tdict(True, 4)
|
|
PcdList = []
|
|
RecordList = self._RawData[Type, self._Arch, self._Platform]
|
|
for TokenSpaceGuid, PcdCName, Setting, Arch, Platform, Id, LineNo in RecordList:
|
|
PcdDict[Arch, Platform, PcdCName, TokenSpaceGuid] = (Setting, LineNo)
|
|
PcdList.append((PcdCName, TokenSpaceGuid))
|
|
# get the guid value
|
|
if TokenSpaceGuid not in self.Guids:
|
|
Value = GuidValue(TokenSpaceGuid, self.Packages, self.MetaFile.Path)
|
|
if Value is None:
|
|
PackageList = "\n\t".join(str(P) for P in self.Packages)
|
|
EdkLogger.error('build', RESOURCE_NOT_AVAILABLE,
|
|
"Value of Guid [%s] is not found under [Guids] section in" % TokenSpaceGuid,
|
|
ExtraData=PackageList, File=self.MetaFile, Line=LineNo)
|
|
self.Guids[TokenSpaceGuid] = Value
|
|
self._GuidsUsedByPcd[TokenSpaceGuid] = Value
|
|
CommentRecords = self._RawData[MODEL_META_DATA_COMMENT, self._Arch, self._Platform, Id]
|
|
Comments = []
|
|
for CmtRec in CommentRecords:
|
|
Comments.append(CmtRec[0])
|
|
self._PcdComments[TokenSpaceGuid, PcdCName] = Comments
|
|
|
|
# resolve PCD type, value, datum info, etc. by getting its definition from package
|
|
_GuidDict = self.Guids.copy()
|
|
for PcdCName, TokenSpaceGuid in PcdList:
|
|
PcdRealName = PcdCName
|
|
Setting, LineNo = PcdDict[self._Arch, self.Platform, PcdCName, TokenSpaceGuid]
|
|
if Setting is None:
|
|
continue
|
|
ValueList = AnalyzePcdData(Setting)
|
|
DefaultValue = ValueList[0]
|
|
Pcd = PcdClassObject(
|
|
PcdCName,
|
|
TokenSpaceGuid,
|
|
'',
|
|
'',
|
|
DefaultValue,
|
|
'',
|
|
'',
|
|
{},
|
|
False,
|
|
self.Guids[TokenSpaceGuid]
|
|
)
|
|
if Type == MODEL_PCD_PATCHABLE_IN_MODULE and ValueList[1]:
|
|
# Patch PCD: TokenSpace.PcdCName|Value|Offset
|
|
Pcd.Offset = ValueList[1]
|
|
|
|
if (PcdRealName, TokenSpaceGuid) in GlobalData.MixedPcd:
|
|
for Package in self.Packages:
|
|
for key in Package.Pcds:
|
|
if (Package.Pcds[key].TokenCName, Package.Pcds[key].TokenSpaceGuidCName) == (PcdRealName, TokenSpaceGuid):
|
|
for item in GlobalData.MixedPcd[(PcdRealName, TokenSpaceGuid)]:
|
|
Pcd_Type = item[0].split('_')[-1]
|
|
if Pcd_Type == Package.Pcds[key].Type:
|
|
Value = Package.Pcds[key]
|
|
Value.TokenCName = Package.Pcds[key].TokenCName + '_' + Pcd_Type
|
|
if len(key) == 2:
|
|
newkey = (Value.TokenCName, key[1])
|
|
elif len(key) == 3:
|
|
newkey = (Value.TokenCName, key[1], key[2])
|
|
del Package.Pcds[key]
|
|
Package.Pcds[newkey] = Value
|
|
break
|
|
else:
|
|
pass
|
|
else:
|
|
pass
|
|
|
|
# get necessary info from package declaring this PCD
|
|
for Package in self.Packages:
|
|
#
|
|
# 'dynamic' in INF means its type is determined by platform;
|
|
# if platform doesn't give its type, use 'lowest' one in the
|
|
# following order, if any
|
|
#
|
|
# TAB_PCDS_FIXED_AT_BUILD, TAB_PCDS_PATCHABLE_IN_MODULE, TAB_PCDS_FEATURE_FLAG, TAB_PCDS_DYNAMIC, TAB_PCDS_DYNAMIC_EX
|
|
#
|
|
_GuidDict.update(Package.Guids)
|
|
PcdType = self._PCD_TYPE_STRING_[Type]
|
|
if Type == MODEL_PCD_DYNAMIC:
|
|
Pcd.Pending = True
|
|
for T in PCD_TYPE_LIST:
|
|
if (PcdRealName, TokenSpaceGuid) in GlobalData.MixedPcd:
|
|
for item in GlobalData.MixedPcd[(PcdRealName, TokenSpaceGuid)]:
|
|
if str(item[0]).endswith(T) and (item[0], item[1], T) in Package.Pcds:
|
|
PcdType = T
|
|
PcdCName = item[0]
|
|
break
|
|
else:
|
|
pass
|
|
break
|
|
else:
|
|
if (PcdRealName, TokenSpaceGuid, T) in Package.Pcds:
|
|
PcdType = T
|
|
break
|
|
|
|
else:
|
|
Pcd.Pending = False
|
|
if (PcdRealName, TokenSpaceGuid) in GlobalData.MixedPcd:
|
|
for item in GlobalData.MixedPcd[(PcdRealName, TokenSpaceGuid)]:
|
|
Pcd_Type = item[0].split('_')[-1]
|
|
if Pcd_Type == PcdType:
|
|
PcdCName = item[0]
|
|
break
|
|
else:
|
|
pass
|
|
else:
|
|
pass
|
|
|
|
if (PcdCName, TokenSpaceGuid, PcdType) in Package.Pcds:
|
|
PcdInPackage = Package.Pcds[PcdCName, TokenSpaceGuid, PcdType]
|
|
Pcd.Type = PcdType
|
|
Pcd.TokenValue = PcdInPackage.TokenValue
|
|
|
|
#
|
|
# Check whether the token value exist or not.
|
|
#
|
|
if Pcd.TokenValue is None or Pcd.TokenValue == "":
|
|
EdkLogger.error(
|
|
'build',
|
|
FORMAT_INVALID,
|
|
"No TokenValue for PCD [%s.%s] in [%s]!" % (TokenSpaceGuid, PcdRealName, str(Package)),
|
|
File=self.MetaFile, Line=LineNo,
|
|
ExtraData=None
|
|
)
|
|
#
|
|
# Check hexadecimal token value length and format.
|
|
#
|
|
ReIsValidPcdTokenValue = re.compile(r"^[0][x|X][0]*[0-9a-fA-F]{1,8}$", re.DOTALL)
|
|
if Pcd.TokenValue.startswith("0x") or Pcd.TokenValue.startswith("0X"):
|
|
if ReIsValidPcdTokenValue.match(Pcd.TokenValue) is None:
|
|
EdkLogger.error(
|
|
'build',
|
|
FORMAT_INVALID,
|
|
"The format of TokenValue [%s] of PCD [%s.%s] in [%s] is invalid:" % (Pcd.TokenValue, TokenSpaceGuid, PcdRealName, str(Package)),
|
|
File=self.MetaFile, Line=LineNo,
|
|
ExtraData=None
|
|
)
|
|
|
|
#
|
|
# Check decimal token value length and format.
|
|
#
|
|
else:
|
|
try:
|
|
TokenValueInt = int (Pcd.TokenValue, 10)
|
|
if (TokenValueInt < 0 or TokenValueInt > 4294967295):
|
|
EdkLogger.error(
|
|
'build',
|
|
FORMAT_INVALID,
|
|
"The format of TokenValue [%s] of PCD [%s.%s] in [%s] is invalid, as a decimal it should between: 0 - 4294967295!" % (Pcd.TokenValue, TokenSpaceGuid, PcdRealName, str(Package)),
|
|
File=self.MetaFile, Line=LineNo,
|
|
ExtraData=None
|
|
)
|
|
except:
|
|
EdkLogger.error(
|
|
'build',
|
|
FORMAT_INVALID,
|
|
"The format of TokenValue [%s] of PCD [%s.%s] in [%s] is invalid, it should be hexadecimal or decimal!" % (Pcd.TokenValue, TokenSpaceGuid, PcdRealName, str(Package)),
|
|
File=self.MetaFile, Line=LineNo,
|
|
ExtraData=None
|
|
)
|
|
|
|
Pcd.DatumType = PcdInPackage.DatumType
|
|
Pcd.MaxDatumSize = PcdInPackage.MaxDatumSize
|
|
Pcd.InfDefaultValue = Pcd.DefaultValue
|
|
if not Pcd.DefaultValue:
|
|
Pcd.DefaultValue = PcdInPackage.DefaultValue
|
|
else:
|
|
try:
|
|
Pcd.DefaultValue = ValueExpressionEx(Pcd.DefaultValue, Pcd.DatumType, _GuidDict)(True)
|
|
except BadExpression as Value:
|
|
EdkLogger.error('Parser', FORMAT_INVALID, 'PCD [%s.%s] Value "%s", %s' %(TokenSpaceGuid, PcdRealName, Pcd.DefaultValue, Value),
|
|
File=self.MetaFile, Line=LineNo)
|
|
break
|
|
else:
|
|
EdkLogger.error(
|
|
'build',
|
|
FORMAT_INVALID,
|
|
"PCD [%s.%s] in [%s] is not found in dependent packages:" % (TokenSpaceGuid, PcdRealName, self.MetaFile),
|
|
File=self.MetaFile, Line=LineNo,
|
|
ExtraData="\t%s" % '\n\t'.join(str(P) for P in self.Packages)
|
|
)
|
|
Pcds[PcdCName, TokenSpaceGuid] = Pcd
|
|
|
|
return Pcds
|
|
|
|
## check whether current module is binary module
|
|
@property
|
|
def IsBinaryModule(self):
|
|
if (self.Binaries and not self.Sources) or GlobalData.gIgnoreSource:
|
|
return True
|
|
return False
|