2010-09-06 04:01:57 +02:00
## @file
# This file include GenVpd class for fix the Vpd type PCD offset, and PcdEntry for describe
# and process each entry of vpd type PCD.
#
2018-01-29 08:49:56 +01:00
# Copyright (c) 2010 - 2018, Intel Corporation. All rights reserved.<BR>
2010-09-06 04:01:57 +02:00
#
# 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.
#
2018-07-13 12:18:39 +02:00
from __future__ import absolute_import
2014-08-15 05:06:48 +02:00
import Common . LongFilePathOs as os
2018-06-25 12:31:36 +02:00
from io import BytesIO
2018-07-13 12:18:39 +02:00
from . import StringTable as st
2010-09-06 04:01:57 +02:00
import array
2010-11-15 03:51:34 +01:00
import re
2014-08-15 05:06:48 +02:00
from Common . LongFilePathSupport import OpenLongFilePath as open
2010-09-06 04:01:57 +02:00
from struct import *
import Common . EdkLogger as EdkLogger
import Common . BuildToolError as BuildToolError
_FORMAT_CHAR = { 1 : ' B ' ,
2 : ' H ' ,
4 : ' I ' ,
8 : ' Q '
}
2010-10-11 08:26:52 +02:00
## The VPD PCD data structure for store and process each VPD PCD entry.
#
2018-07-05 11:40:04 +02:00
# This class contain method to format and pack pcd's value.
2010-10-11 08:26:52 +02:00
#
2010-09-06 04:01:57 +02:00
class PcdEntry :
2018-07-05 11:40:04 +02:00
def __init__ ( self , PcdCName , SkuId , PcdOffset , PcdSize , PcdValue , Lineno = None , FileName = None , PcdUnpackValue = None ,
2018-01-29 08:49:56 +01:00
PcdBinOffset = None , PcdBinSize = None , Alignment = None ) :
2010-09-06 04:01:57 +02:00
self . PcdCName = PcdCName . strip ( )
2013-11-18 08:41:21 +01:00
self . SkuId = SkuId . strip ( )
2010-09-06 04:01:57 +02:00
self . PcdOffset = PcdOffset . strip ( )
self . PcdSize = PcdSize . strip ( )
self . PcdValue = PcdValue . strip ( )
self . Lineno = Lineno . strip ( )
self . FileName = FileName . strip ( )
self . PcdUnpackValue = PcdUnpackValue
self . PcdBinOffset = PcdBinOffset
self . PcdBinSize = PcdBinSize
2018-01-29 08:49:56 +01:00
self . Alignment = Alignment
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
if self . PcdValue == ' ' :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid PCD format(Name: %s File: %s line: %s ) , no Value specified! " % ( self . PcdCName , self . FileName , self . Lineno ) )
2010-09-06 04:01:57 +02:00
if self . PcdOffset == ' ' :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid PCD format(Name: %s File: %s Line: %s ) , no Offset specified! " % ( self . PcdCName , self . FileName , self . Lineno ) )
2010-09-06 04:01:57 +02:00
if self . PcdSize == ' ' :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid PCD format(Name: %s File: %s Line: %s ), no PcdSize specified! " % ( self . PcdCName , self . FileName , self . Lineno ) )
2010-09-06 04:01:57 +02:00
self . _GenOffsetValue ( )
2015-12-01 05:22:16 +01:00
2018-01-30 16:01:31 +01:00
## Analyze the string value to judge the PCD's datum type equal to Boolean or not.
2018-07-05 11:40:04 +02:00
#
2010-10-11 08:26:52 +02:00
# @param ValueString PCD's value
# @param Size PCD's size
2018-07-05 11:40:04 +02:00
#
2010-10-11 08:26:52 +02:00
# @retval True PCD's datum type is Boolean
2018-07-05 11:40:04 +02:00
# @retval False PCD's datum type is not Boolean.
#
2010-10-11 08:26:52 +02:00
def _IsBoolean ( self , ValueString , Size ) :
if ( Size == " 1 " ) :
if ValueString . upper ( ) in [ " TRUE " , " FALSE " ] :
return True
elif ValueString in [ " 0 " , " 1 " , " 0x0 " , " 0x1 " , " 0x00 " , " 0x01 " ] :
2015-12-01 05:22:16 +01:00
return True
2010-09-06 04:01:57 +02:00
return False
2015-12-01 05:22:16 +01:00
2010-10-11 08:26:52 +02:00
## Convert the PCD's value from string to integer.
2018-07-05 11:40:04 +02:00
#
2010-10-11 08:26:52 +02:00
# This function will try to convert the Offset value form string to integer
# for both hexadecimal and decimal.
2018-07-05 11:40:04 +02:00
#
2010-09-06 04:01:57 +02:00
def _GenOffsetValue ( self ) :
if self . PcdOffset != " * " :
try :
self . PcdBinOffset = int ( self . PcdOffset )
except :
try :
self . PcdBinOffset = int ( self . PcdOffset , 16 )
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid offset value %s for PCD %s (File: %s Line: %s ) " % ( self . PcdOffset , self . PcdCName , self . FileName , self . Lineno ) )
2010-10-11 08:26:52 +02:00
## Pack Boolean type VPD PCD's value form string to binary type.
2018-07-05 11:40:04 +02:00
#
2010-10-11 08:26:52 +02:00
# @param ValueString The boolean type string for pack.
2018-07-05 11:40:04 +02:00
#
#
2010-09-06 04:01:57 +02:00
def _PackBooleanValue ( self , ValueString ) :
2010-10-11 08:26:52 +02:00
if ValueString . upper ( ) == " TRUE " or ValueString in [ " 1 " , " 0x1 " , " 0x01 " ] :
2015-12-01 05:22:16 +01:00
try :
self . PcdValue = pack ( _FORMAT_CHAR [ 1 ] , 1 )
2010-09-06 04:01:57 +02:00
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid size or value for PCD %s to pack(File: %s Line: %s ). " % ( self . PcdCName , self . FileName , self . Lineno ) )
2010-09-06 04:01:57 +02:00
else :
try :
2015-12-01 05:22:16 +01:00
self . PcdValue = pack ( _FORMAT_CHAR [ 1 ] , 0 )
2010-09-06 04:01:57 +02:00
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid size or value for PCD %s to pack(File: %s Line: %s ). " % ( self . PcdCName , self . FileName , self . Lineno ) )
2010-10-11 08:26:52 +02:00
## Pack Integer type VPD PCD's value form string to binary type.
2018-07-05 11:40:04 +02:00
#
2010-10-11 08:26:52 +02:00
# @param ValueString The Integer type string for pack.
2018-07-05 11:40:04 +02:00
#
#
2010-09-06 04:01:57 +02:00
def _PackIntValue ( self , IntValue , Size ) :
2018-04-17 16:40:15 +02:00
if Size not in _FORMAT_CHAR :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid size %d for PCD %s in integer datum size(File: %s Line: %s ). " % ( Size , self . PcdCName , self . FileName , self . Lineno ) )
if Size == 1 :
2010-10-11 08:26:52 +02:00
if IntValue < 0 :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
2010-10-11 08:26:52 +02:00
" PCD can ' t be set to negative value %d for PCD %s in UINT8 datum type(File: %s Line: %s ). " % ( IntValue , self . PcdCName , self . FileName , self . Lineno ) )
elif IntValue > = 0x100 :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Too large PCD value %d for datum type UINT8 for PCD %s (File: %s Line: %s ). " % ( IntValue , self . PcdCName , self . FileName , self . Lineno ) )
2010-10-11 08:26:52 +02:00
elif Size == 2 :
if IntValue < 0 :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
2010-10-11 08:26:52 +02:00
" PCD can ' t be set to negative value %d for PCD %s in UINT16 datum type(File: %s Line: %s ). " % ( IntValue , self . PcdCName , self . FileName , self . Lineno ) )
elif IntValue > = 0x10000 :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Too large PCD value %d for datum type UINT16 for PCD %s (File: %s Line: %s ). " % ( IntValue , self . PcdCName , self . FileName , self . Lineno ) )
2010-10-11 08:26:52 +02:00
elif Size == 4 :
if IntValue < 0 :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
2010-10-11 08:26:52 +02:00
" PCD can ' t be set to negative value %d for PCD %s in UINT32 datum type(File: %s Line: %s ). " % ( IntValue , self . PcdCName , self . FileName , self . Lineno ) )
elif IntValue > = 0x100000000 :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
2010-10-11 08:26:52 +02:00
" Too large PCD value %d for datum type UINT32 for PCD %s (File: %s Line: %s ). " % ( IntValue , self . PcdCName , self . FileName , self . Lineno ) )
elif Size == 8 :
if IntValue < 0 :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
2010-10-11 08:26:52 +02:00
" PCD can ' t be set to negative value %d for PCD %s in UINT32 datum type(File: %s Line: %s ). " % ( IntValue , self . PcdCName , self . FileName , self . Lineno ) )
elif IntValue > = 0x10000000000000000 :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
2010-10-11 08:26:52 +02:00
" Too large PCD value %d for datum type UINT32 for PCD %s (File: %s Line: %s ). " % ( IntValue , self . PcdCName , self . FileName , self . Lineno ) )
else :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid size %d for PCD %s in integer datum size(File: %s Line: %s ). " % ( Size , self . PcdCName , self . FileName , self . Lineno ) )
2010-09-06 04:01:57 +02:00
try :
2015-12-01 05:22:16 +01:00
self . PcdValue = pack ( _FORMAT_CHAR [ Size ] , IntValue )
2010-09-06 04:01:57 +02:00
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid size or value for PCD %s to pack(File: %s Line: %s ). " % ( self . PcdCName , self . FileName , self . Lineno ) )
2010-10-11 08:26:52 +02:00
## Pack VOID* type VPD PCD's value form string to binary type.
#
# The VOID* type of string divided into 3 sub-type:
2018-01-30 16:01:31 +01:00
# 1: L"String"/L'String', Unicode type string.
# 2: "String"/'String', Ascii type string.
2010-10-11 08:26:52 +02:00
# 3: {bytearray}, only support byte-array.
#
# @param ValueString The Integer type string for pack.
2018-07-05 11:40:04 +02:00
#
2010-09-06 04:01:57 +02:00
def _PackPtrValue ( self , ValueString , Size ) :
2018-01-30 16:01:31 +01:00
if ValueString . startswith ( ' L " ' ) or ValueString . startswith ( " L ' " ) :
2010-09-06 04:01:57 +02:00
self . _PackUnicode ( ValueString , Size )
elif ValueString . startswith ( ' { ' ) and ValueString . endswith ( ' } ' ) :
self . _PackByteArray ( ValueString , Size )
2018-01-30 16:01:31 +01:00
elif ( ValueString . startswith ( ' " ' ) and ValueString . endswith ( ' " ' ) ) or ( ValueString . startswith ( " ' " ) and ValueString . endswith ( " ' " ) ) :
2010-09-06 04:01:57 +02:00
self . _PackString ( ValueString , Size )
else :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid VOID* type PCD %s value %s (File: %s Line: %s ) " % ( self . PcdCName , ValueString , self . FileName , self . Lineno ) )
2010-10-11 08:26:52 +02:00
## Pack an Ascii PCD value.
2018-07-05 11:40:04 +02:00
#
2018-01-30 16:01:31 +01:00
# An Ascii string for a PCD should be in format as ""/''.
2018-07-05 11:40:04 +02:00
#
2010-09-06 04:01:57 +02:00
def _PackString ( self , ValueString , Size ) :
if ( Size < 0 ) :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
2010-09-06 04:01:57 +02:00
" Invalid parameter Size %s of PCD %s !(File: %s Line: %s ) " % ( self . PcdBinSize , self . PcdCName , self . FileName , self . Lineno ) )
if ( ValueString == " " ) :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID , " Invalid parameter ValueString %s of PCD %s !(File: %s Line: %s ) " % ( self . PcdUnpackValue , self . PcdCName , self . FileName , self . Lineno ) )
2018-01-30 16:01:31 +01:00
QuotedFlag = True
if ValueString . startswith ( " ' " ) :
QuotedFlag = False
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
ValueString = ValueString [ 1 : - 1 ]
2018-07-05 11:40:04 +02:00
# No null-terminator in 'string'
2018-01-30 16:01:31 +01:00
if ( QuotedFlag and len ( ValueString ) + 1 > Size ) or ( not QuotedFlag and len ( ValueString ) > Size ) :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . RESOURCE_OVERFLOW ,
2010-09-06 04:01:57 +02:00
" PCD value string %s is exceed to size %d (File: %s Line: %s ) " % ( ValueString , Size , self . FileName , self . Lineno ) )
try :
2015-12-01 05:22:16 +01:00
self . PcdValue = pack ( ' %d s ' % Size , ValueString )
2010-09-06 04:01:57 +02:00
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" Invalid size or value for PCD %s to pack(File: %s Line: %s ). " % ( self . PcdCName , self . FileName , self . Lineno ) )
2010-10-11 08:26:52 +02:00
## Pack a byte-array PCD value.
2018-07-05 11:40:04 +02:00
#
2010-10-11 08:26:52 +02:00
# A byte-array for a PCD should be in format as {0x01, 0x02, ...}.
2018-07-05 11:40:04 +02:00
#
2010-09-06 04:01:57 +02:00
def _PackByteArray ( self , ValueString , Size ) :
2015-12-01 05:22:16 +01:00
if ( Size < 0 ) :
2010-09-06 04:01:57 +02:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID , " Invalid parameter Size %s of PCD %s !(File: %s Line: %s ) " % ( self . PcdBinSize , self . PcdCName , self . FileName , self . Lineno ) )
if ( ValueString == " " ) :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID , " Invalid parameter ValueString %s of PCD %s !(File: %s Line: %s ) " % ( self . PcdUnpackValue , self . PcdCName , self . FileName , self . Lineno ) )
2010-09-06 04:01:57 +02:00
ValueString = ValueString . strip ( )
ValueString = ValueString . lstrip ( ' { ' ) . strip ( ' } ' )
ValueList = ValueString . split ( ' , ' )
ValueList = [ item . strip ( ) for item in ValueList ]
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
if len ( ValueList ) > Size :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . RESOURCE_OVERFLOW ,
2010-09-06 04:01:57 +02:00
" The byte array %s is too large for size %d (File: %s Line: %s ) " % ( ValueString , Size , self . FileName , self . Lineno ) )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
ReturnArray = array . array ( ' B ' )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
for Index in xrange ( len ( ValueList ) ) :
Value = None
2012-05-23 10:27:14 +02:00
if ValueList [ Index ] . lower ( ) . startswith ( ' 0x ' ) :
2010-09-06 04:01:57 +02:00
# translate hex value
try :
Value = int ( ValueList [ Index ] , 16 )
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
2010-09-06 04:01:57 +02:00
" The value item %s in byte array %s is an invalid HEX value.(File: %s Line: %s ) " % \
( ValueList [ Index ] , ValueString , self . FileName , self . Lineno ) )
else :
# translate decimal value
try :
Value = int ( ValueList [ Index ] , 10 )
except :
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" The value item %s in byte array %s is an invalid DECIMAL value.(File: %s Line: %s ) " % \
( ValueList [ Index ] , ValueString , self . FileName , self . Lineno ) )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
if Value > 255 :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
" The value item %s in byte array %s do not in range 0 ~ 0xFF(File: %s Line: %s ) " % \
2010-09-06 04:01:57 +02:00
( ValueList [ Index ] , ValueString , self . FileName , self . Lineno ) )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
ReturnArray . append ( Value )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
for Index in xrange ( len ( ValueList ) , Size ) :
ReturnArray . append ( 0 )
2015-12-01 05:22:16 +01:00
self . PcdValue = ReturnArray . tolist ( )
2010-09-06 04:01:57 +02:00
## Pack a unicode PCD value into byte array.
2018-07-05 11:40:04 +02:00
#
2018-01-30 16:01:31 +01:00
# A unicode string for a PCD should be in format as L""/L''.
2010-09-06 04:01:57 +02:00
#
def _PackUnicode ( self , UnicodeString , Size ) :
2015-12-01 05:22:16 +01:00
if ( Size < 0 ) :
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID , " Invalid parameter Size %s of PCD %s !(File: %s Line: %s ) " % \
2010-09-06 04:01:57 +02:00
( self . PcdBinSize , self . PcdCName , self . FileName , self . Lineno ) )
2015-12-01 05:22:16 +01:00
2018-01-30 16:01:31 +01:00
QuotedFlag = True
if UnicodeString . startswith ( " L ' " ) :
2018-07-05 11:40:04 +02:00
QuotedFlag = False
2010-09-06 04:01:57 +02:00
UnicodeString = UnicodeString [ 2 : - 1 ]
2015-12-01 05:22:16 +01:00
2018-01-30 16:01:31 +01:00
# No null-terminator in L'string'
if ( QuotedFlag and ( len ( UnicodeString ) + 1 ) * 2 > Size ) or ( not QuotedFlag and len ( UnicodeString ) * 2 > Size ) :
2010-09-06 04:01:57 +02:00
EdkLogger . error ( " BPDG " , BuildToolError . RESOURCE_OVERFLOW ,
" The size of unicode string %s is too larger for size %s (File: %s Line: %s ) " % \
( UnicodeString , Size , self . FileName , self . Lineno ) )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
ReturnArray = array . array ( ' B ' )
for Value in UnicodeString :
try :
ReturnArray . append ( ord ( Value ) )
ReturnArray . append ( 0 )
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID ,
2010-09-06 04:01:57 +02:00
" Invalid unicode character %s in unicode string %s (File: %s Line: %s ) " % \
( Value , UnicodeString , self . FileName , self . Lineno ) )
2015-12-01 05:22:16 +01:00
2013-11-18 08:41:21 +01:00
for Index in xrange ( len ( UnicodeString ) * 2 , Size ) :
2010-09-06 04:01:57 +02:00
ReturnArray . append ( 0 )
2015-12-01 05:22:16 +01:00
self . PcdValue = ReturnArray . tolist ( )
2010-10-11 08:26:52 +02:00
## The class implementing the BPDG VPD PCD offset fix process
#
# The VPD PCD offset fix process includes:
# 1. Parse the input guided.txt file and store it in the data structure;
# 2. Format the input file data to remove unused lines;
# 3. Fixed offset if needed;
# 4. Generate output file, including guided.map and guided.bin file;
2018-07-05 11:40:04 +02:00
#
2015-12-01 05:22:16 +01:00
class GenVPD :
2010-09-06 04:01:57 +02:00
## Constructor of DscBuildData
#
# Initialize object of GenVPD
# @Param InputFileName The filename include the vpd type pcd information
# @param MapFileName The filename of map file that stores vpd type pcd information.
# This file will be generated by the BPDG tool after fix the offset
# and adjust the offset to make the pcd data aligned.
# @param VpdFileName The filename of Vpd file that hold vpd pcd information.
#
def __init__ ( self , InputFileName , MapFileName , VpdFileName ) :
self . InputFileName = InputFileName
self . MapFileName = MapFileName
self . VpdFileName = VpdFileName
self . FileLinesList = [ ]
self . PcdFixedOffsetSizeList = [ ]
self . PcdUnknownOffsetList = [ ]
try :
fInputfile = open ( InputFileName , " r " , 0 )
try :
self . FileLinesList = fInputfile . readlines ( )
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FILE_READ_FAILURE , " File read failed for %s " % InputFileName , None )
2010-09-06 04:01:57 +02:00
finally :
fInputfile . close ( )
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FILE_OPEN_FAILURE , " File open failed for %s " % InputFileName , None )
2010-09-06 04:01:57 +02:00
##
2018-07-05 11:40:04 +02:00
# Parser the input file which is generated by the build tool. Convert the value of each pcd's
2010-09-06 04:01:57 +02:00
# from string to it's real format. Also remove the useless line in the input file.
2018-07-05 11:40:04 +02:00
#
2010-09-06 04:01:57 +02:00
def ParserInputFile ( self ) :
2015-12-01 05:22:16 +01:00
count = 0
2010-09-06 04:01:57 +02:00
for line in self . FileLinesList :
# Strip "\r\n" generated by readlines ().
line = line . strip ( )
line = line . rstrip ( os . linesep )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
# Skip the comment line
2010-11-15 03:51:34 +01:00
if ( not line . startswith ( " # " ) ) and len ( line ) > 1 :
#
# Enhanced for support "|" character in the string.
#
2018-06-25 12:31:33 +02:00
ValueList = [ ' ' , ' ' , ' ' , ' ' , ' ' ]
2015-12-01 05:22:16 +01:00
ValueRe = re . compile ( r ' \ s*L? \ " .* \ |.* \ " \ s*$ ' )
2010-11-15 03:51:34 +01:00
PtrValue = ValueRe . findall ( line )
2015-12-01 05:22:16 +01:00
2010-11-15 03:51:34 +01:00
ValueUpdateFlag = False
2015-12-01 05:22:16 +01:00
2010-11-15 03:51:34 +01:00
if len ( PtrValue ) > = 1 :
line = re . sub ( ValueRe , ' ' , line )
2015-12-01 05:22:16 +01:00
ValueUpdateFlag = True
2010-11-15 03:51:34 +01:00
TokenList = line . split ( ' | ' )
ValueList [ 0 : len ( TokenList ) ] = TokenList
2015-12-01 05:22:16 +01:00
2010-11-15 03:51:34 +01:00
if ValueUpdateFlag :
2016-01-18 02:47:50 +01:00
ValueList [ 4 ] = PtrValue [ 0 ]
2010-11-15 03:51:34 +01:00
self . FileLinesList [ count ] = ValueList
2010-09-06 04:01:57 +02:00
# Store the line number
2015-12-01 05:22:16 +01:00
self . FileLinesList [ count ] . append ( str ( count + 1 ) )
2010-09-06 04:01:57 +02:00
elif len ( line ) < = 1 :
# Set the blank line to "None"
self . FileLinesList [ count ] = None
else :
# Set the comment line to "None"
self . FileLinesList [ count ] = None
count + = 1
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
# The line count contain usage information
2015-12-01 05:22:16 +01:00
count = 0
2010-09-06 04:01:57 +02:00
# Delete useless lines
while ( True ) :
try :
2018-03-26 22:25:43 +02:00
if ( self . FileLinesList [ count ] is None ) :
2010-09-06 04:01:57 +02:00
del ( self . FileLinesList [ count ] )
else :
count + = 1
except :
2015-12-01 05:22:16 +01:00
break
2010-09-06 04:01:57 +02:00
#
# After remove the useless line, if there are no data remain in the file line list,
# Report warning messages to user's.
2018-07-05 11:40:04 +02:00
#
2010-09-06 04:01:57 +02:00
if len ( self . FileLinesList ) == 0 :
2015-12-01 05:22:16 +01:00
EdkLogger . warn ( ' BPDG ' , BuildToolError . RESOURCE_NOT_AVAILABLE ,
2010-09-06 04:01:57 +02:00
" There are no VPD type pcds defined in DSC file, Please check it. " )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
# Process the pcds one by one base on the pcd's value and size
count = 0
2015-12-01 05:22:16 +01:00
for line in self . FileLinesList :
2018-03-26 22:25:43 +02:00
if line is not None :
2018-06-25 12:31:33 +02:00
PCD = PcdEntry ( line [ 0 ] , line [ 1 ] , line [ 2 ] , line [ 3 ] , line [ 4 ] , line [ 5 ] , self . InputFileName )
2010-09-06 04:01:57 +02:00
# Strip the space char
PCD . PcdCName = PCD . PcdCName . strip ( ' ' )
2013-11-18 08:41:21 +01:00
PCD . SkuId = PCD . SkuId . strip ( ' ' )
2010-09-06 04:01:57 +02:00
PCD . PcdOffset = PCD . PcdOffset . strip ( ' ' )
PCD . PcdSize = PCD . PcdSize . strip ( ' ' )
2016-01-18 02:47:50 +01:00
PCD . PcdValue = PCD . PcdValue . strip ( ' ' )
2010-09-06 04:01:57 +02:00
PCD . Lineno = PCD . Lineno . strip ( ' ' )
2016-01-18 02:47:50 +01:00
2010-09-06 04:01:57 +02:00
#
# Store the original pcd value.
# This information will be useful while generate the output map file.
#
2016-01-18 02:47:50 +01:00
PCD . PcdUnpackValue = str ( PCD . PcdValue )
2016-03-13 11:13:04 +01:00
#
# Translate PCD size string to an integer value.
PackSize = None
try :
PackSize = int ( PCD . PcdSize , 10 )
PCD . PcdBinSize = PackSize
except :
try :
PackSize = int ( PCD . PcdSize , 16 )
PCD . PcdBinSize = PackSize
except :
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID , " Invalid PCD size value %s at file: %s line: %s " % ( PCD . PcdSize , self . InputFileName , PCD . Lineno ) )
2016-01-18 02:47:50 +01:00
#
# If value is Unicode string (e.g. L""), then use 2-byte alignment
# If value is byte array (e.g. {}), then use 8-byte alignment
#
2016-03-13 11:13:04 +01:00
PCD . PcdOccupySize = PCD . PcdBinSize
2016-01-18 02:47:50 +01:00
if PCD . PcdUnpackValue . startswith ( " { " ) :
Alignment = 8
elif PCD . PcdUnpackValue . startswith ( " L " ) :
Alignment = 2
else :
Alignment = 1
2016-03-09 07:03:39 +01:00
2018-01-29 08:49:56 +01:00
PCD . Alignment = Alignment
2016-03-09 07:03:39 +01:00
if PCD . PcdOffset != ' * ' :
if PCD . PcdOccupySize % Alignment != 0 :
if PCD . PcdUnpackValue . startswith ( " { " ) :
EdkLogger . warn ( " BPDG " , " The offset value of PCD %s is not 8-byte aligned! " % ( PCD . PcdCName ) , File = self . InputFileName )
else :
EdkLogger . error ( " BPDG " , BuildToolError . FORMAT_INVALID , ' The offset value of PCD %s should be %s -byte aligned. ' % ( PCD . PcdCName , Alignment ) )
else :
if PCD . PcdOccupySize % Alignment != 0 :
PCD . PcdOccupySize = ( PCD . PcdOccupySize / Alignment + 1 ) * Alignment
2010-09-06 04:01:57 +02:00
2018-01-29 08:49:56 +01:00
PackSize = PCD . PcdOccupySize
2010-10-11 08:26:52 +02:00
if PCD . _IsBoolean ( PCD . PcdValue , PCD . PcdSize ) :
2010-09-06 04:01:57 +02:00
PCD . _PackBooleanValue ( PCD . PcdValue )
self . FileLinesList [ count ] = PCD
count + = 1
continue
#
# Try to translate value to an integer firstly.
#
IsInteger = True
2015-12-01 05:22:16 +01:00
PackValue = None
2010-09-06 04:01:57 +02:00
try :
PackValue = int ( PCD . PcdValue )
except :
try :
PackValue = int ( PCD . PcdValue , 16 )
except :
IsInteger = False
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
if IsInteger :
PCD . _PackIntValue ( PackValue , PackSize )
else :
PCD . _PackPtrValue ( PCD . PcdValue , PackSize )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
self . FileLinesList [ count ] = PCD
count + = 1
else :
continue
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
##
2018-07-05 11:40:04 +02:00
# This function used to create a clean list only contain useful information and reorganized to make it
2010-09-06 04:01:57 +02:00
# easy to be sorted
#
def FormatFileLine ( self ) :
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
for eachPcd in self . FileLinesList :
if eachPcd . PcdOffset != ' * ' :
2018-07-05 11:40:04 +02:00
# Use pcd's Offset value as key, and pcd's Value as value
2010-09-06 04:01:57 +02:00
self . PcdFixedOffsetSizeList . append ( eachPcd )
else :
# Use pcd's CName as key, and pcd's Size as value
self . PcdUnknownOffsetList . append ( eachPcd )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
##
# This function is use to fix the offset value which the not specified in the map file.
# Usually it use the star (meaning any offset) character in the offset field
2018-07-05 11:40:04 +02:00
#
2015-12-01 05:22:16 +01:00
def FixVpdOffset ( self ) :
2010-09-06 04:01:57 +02:00
# At first, the offset should start at 0
# Sort fixed offset list in order to find out where has free spaces for the pcd's offset
2018-07-05 11:40:04 +02:00
# value is "*" to insert into.
2015-12-01 05:22:16 +01:00
self . PcdFixedOffsetSizeList . sort ( lambda x , y : cmp ( x . PcdBinOffset , y . PcdBinOffset ) )
2010-09-06 04:01:57 +02:00
#
# Sort the un-fixed pcd's offset by it's size.
#
2015-12-01 05:22:16 +01:00
self . PcdUnknownOffsetList . sort ( lambda x , y : cmp ( x . PcdBinSize , y . PcdBinSize ) )
2017-12-22 13:23:21 +01:00
index = 0
for pcd in self . PcdUnknownOffsetList :
index + = 1
2018-06-25 12:31:33 +02:00
if pcd . PcdCName == " . " . join ( ( " gEfiMdeModulePkgTokenSpaceGuid " , " PcdNvStoreDefaultValueBuffer " ) ) :
2017-12-22 13:23:21 +01:00
if index != len ( self . PcdUnknownOffsetList ) :
for i in range ( len ( self . PcdUnknownOffsetList ) - index ) :
2018-06-25 12:31:33 +02:00
self . PcdUnknownOffsetList [ index + i - 1 ] , self . PcdUnknownOffsetList [ index + i ] = self . PcdUnknownOffsetList [ index + i ] , self . PcdUnknownOffsetList [ index + i - 1 ]
2017-12-22 13:23:21 +01:00
2010-09-06 04:01:57 +02:00
#
# Process all Offset value are "*"
#
if ( len ( self . PcdFixedOffsetSizeList ) == 0 ) and ( len ( self . PcdUnknownOffsetList ) != 0 ) :
# The offset start from 0
NowOffset = 0
2015-12-01 05:22:16 +01:00
for Pcd in self . PcdUnknownOffsetList :
2018-01-29 08:49:56 +01:00
if NowOffset % Pcd . Alignment != 0 :
NowOffset = ( NowOffset / Pcd . Alignment + 1 ) * Pcd . Alignment
2010-09-06 04:01:57 +02:00
Pcd . PcdBinOffset = NowOffset
Pcd . PcdOffset = str ( hex ( Pcd . PcdBinOffset ) )
2016-01-18 02:47:50 +01:00
NowOffset + = Pcd . PcdOccupySize
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
self . PcdFixedOffsetSizeList = self . PcdUnknownOffsetList
return
2015-12-01 05:22:16 +01:00
2018-07-05 11:40:04 +02:00
# Check the offset of VPD type pcd's offset start from 0.
2015-12-01 05:22:16 +01:00
if self . PcdFixedOffsetSizeList [ 0 ] . PcdBinOffset != 0 :
2010-09-06 04:01:57 +02:00
EdkLogger . warn ( " BPDG " , " The offset of VPD type pcd should start with 0, please check it. " ,
2015-12-01 05:22:16 +01:00
None )
2010-09-06 04:01:57 +02:00
# Judge whether the offset in fixed pcd offset list is overlapped or not.
lenOfList = len ( self . PcdFixedOffsetSizeList )
2018-07-05 11:40:04 +02:00
count = 0
2010-09-06 04:01:57 +02:00
while ( count < lenOfList - 1 ) :
PcdNow = self . PcdFixedOffsetSizeList [ count ]
PcdNext = self . PcdFixedOffsetSizeList [ count + 1 ]
2018-07-05 11:40:04 +02:00
# Two pcd's offset is same
2010-09-06 04:01:57 +02:00
if PcdNow . PcdBinOffset == PcdNext . PcdBinOffset :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . ATTRIBUTE_GET_FAILURE ,
" The offset of %s at line: %s is same with %s at line: %s in file %s " % \
2010-09-06 04:01:57 +02:00
( PcdNow . PcdCName , PcdNow . Lineno , PcdNext . PcdCName , PcdNext . Lineno , PcdNext . FileName ) ,
None )
2015-12-01 05:22:16 +01:00
2018-07-05 11:40:04 +02:00
# Overlapped
2016-01-18 02:47:50 +01:00
if PcdNow . PcdBinOffset + PcdNow . PcdOccupySize > PcdNext . PcdBinOffset :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . ATTRIBUTE_GET_FAILURE ,
" The offset of %s at line: %s is overlapped with %s at line: %s in file %s " % \
2010-09-06 04:01:57 +02:00
( PcdNow . PcdCName , PcdNow . Lineno , PcdNext . PcdCName , PcdNext . Lineno , PcdNext . FileName ) ,
None )
2015-12-01 05:22:16 +01:00
2018-07-05 11:40:04 +02:00
# Has free space, raise a warning message
2016-01-18 02:47:50 +01:00
if PcdNow . PcdBinOffset + PcdNow . PcdOccupySize < PcdNext . PcdBinOffset :
2015-12-01 05:22:16 +01:00
EdkLogger . warn ( " BPDG " , BuildToolError . ATTRIBUTE_GET_FAILURE ,
" The offsets have free space of between %s at line: %s and %s at line: %s in file %s " % \
2010-09-06 04:01:57 +02:00
( PcdNow . PcdCName , PcdNow . Lineno , PcdNext . PcdCName , PcdNext . Lineno , PcdNext . FileName ) ,
None )
count + = 1
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
LastOffset = self . PcdFixedOffsetSizeList [ 0 ] . PcdBinOffset
FixOffsetSizeListCount = 0
lenOfList = len ( self . PcdFixedOffsetSizeList )
lenOfUnfixedList = len ( self . PcdUnknownOffsetList )
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
##
2018-07-05 11:40:04 +02:00
# Insert the un-fixed offset pcd's list into fixed offset pcd's list if has free space between those pcds.
#
2010-09-06 04:01:57 +02:00
while ( FixOffsetSizeListCount < lenOfList ) :
2018-07-05 11:40:04 +02:00
eachFixedPcd = self . PcdFixedOffsetSizeList [ FixOffsetSizeListCount ]
2010-09-06 04:01:57 +02:00
NowOffset = eachFixedPcd . PcdBinOffset
2018-07-05 11:40:04 +02:00
# Has free space
2010-09-06 04:01:57 +02:00
if LastOffset < NowOffset :
if lenOfUnfixedList != 0 :
countOfUnfixedList = 0
2015-12-01 05:22:16 +01:00
while ( countOfUnfixedList < lenOfUnfixedList ) :
2010-09-06 04:01:57 +02:00
eachUnfixedPcd = self . PcdUnknownOffsetList [ countOfUnfixedList ]
2016-01-18 02:47:50 +01:00
needFixPcdSize = eachUnfixedPcd . PcdOccupySize
2010-09-06 04:01:57 +02:00
# Not been fixed
if eachUnfixedPcd . PcdOffset == ' * ' :
2018-01-29 08:49:56 +01:00
if LastOffset % eachUnfixedPcd . Alignment != 0 :
LastOffset = ( LastOffset / eachUnfixedPcd . Alignment + 1 ) * eachUnfixedPcd . Alignment
2010-09-06 04:01:57 +02:00
# The offset un-fixed pcd can write into this free space
if needFixPcdSize < = ( NowOffset - LastOffset ) :
# Change the offset value of un-fixed pcd
eachUnfixedPcd . PcdOffset = str ( hex ( LastOffset ) )
eachUnfixedPcd . PcdBinOffset = LastOffset
# Insert this pcd into fixed offset pcd list.
2018-06-25 12:31:33 +02:00
self . PcdFixedOffsetSizeList . insert ( FixOffsetSizeListCount , eachUnfixedPcd )
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
# Delete the item's offset that has been fixed and added into fixed offset list
self . PcdUnknownOffsetList . pop ( countOfUnfixedList )
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
# After item added, should enlarge the length of fixed pcd offset list
2018-07-05 11:40:04 +02:00
lenOfList + = 1
2010-09-06 04:01:57 +02:00
FixOffsetSizeListCount + = 1
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
# Decrease the un-fixed pcd offset list's length
lenOfUnfixedList - = 1
2018-07-05 11:40:04 +02:00
# Modify the last offset value
LastOffset + = needFixPcdSize
2010-09-06 04:01:57 +02:00
else :
2010-11-15 03:51:34 +01:00
# It can not insert into those two pcds, need to check still has other space can store it.
2016-01-18 02:47:50 +01:00
LastOffset = NowOffset + self . PcdFixedOffsetSizeList [ FixOffsetSizeListCount ] . PcdOccupySize
2010-09-06 04:01:57 +02:00
FixOffsetSizeListCount + = 1
2010-11-15 03:51:34 +01:00
break
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
# Set the FixOffsetSizeListCount = lenOfList for quit the loop
else :
2018-07-05 11:40:04 +02:00
FixOffsetSizeListCount = lenOfList
# No free space, smoothly connect with previous pcd.
2010-09-06 04:01:57 +02:00
elif LastOffset == NowOffset :
2016-01-18 02:47:50 +01:00
LastOffset = NowOffset + eachFixedPcd . PcdOccupySize
2010-09-06 04:01:57 +02:00
FixOffsetSizeListCount + = 1
2018-07-05 11:40:04 +02:00
# Usually it will not enter into this thunk, if so, means it overlapped.
2010-09-06 04:01:57 +02:00
else :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . ATTRIBUTE_NOT_AVAILABLE ,
" The offset value definition has overlapped at pcd: %s , it ' s offset is: %s , in file: %s line: %s " % \
2010-09-06 04:01:57 +02:00
( eachFixedPcd . PcdCName , eachFixedPcd . PcdOffset , eachFixedPcd . InputFileName , eachFixedPcd . Lineno ) ,
None )
FixOffsetSizeListCount + = 1
2018-07-05 11:40:04 +02:00
# Continue to process the un-fixed offset pcd's list, add this time, just append them behind the fixed pcd's offset list.
2010-09-06 04:01:57 +02:00
lenOfUnfixedList = len ( self . PcdUnknownOffsetList )
lenOfList = len ( self . PcdFixedOffsetSizeList )
while ( lenOfUnfixedList > 0 ) :
# Still has items need to process
# The last pcd instance
LastPcd = self . PcdFixedOffsetSizeList [ lenOfList - 1 ]
NeedFixPcd = self . PcdUnknownOffsetList [ 0 ]
2018-07-05 11:40:04 +02:00
2016-01-18 02:47:50 +01:00
NeedFixPcd . PcdBinOffset = LastPcd . PcdBinOffset + LastPcd . PcdOccupySize
2018-01-29 08:49:56 +01:00
if NeedFixPcd . PcdBinOffset % NeedFixPcd . Alignment != 0 :
NeedFixPcd . PcdBinOffset = ( NeedFixPcd . PcdBinOffset / NeedFixPcd . Alignment + 1 ) * NeedFixPcd . Alignment
2010-09-06 04:01:57 +02:00
NeedFixPcd . PcdOffset = str ( hex ( NeedFixPcd . PcdBinOffset ) )
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
# Insert this pcd into fixed offset pcd list's tail.
self . PcdFixedOffsetSizeList . insert ( lenOfList , NeedFixPcd )
# Delete the item's offset that has been fixed and added into fixed offset list
self . PcdUnknownOffsetList . pop ( 0 )
2018-07-05 11:40:04 +02:00
2010-09-06 04:01:57 +02:00
lenOfList + = 1
2018-07-05 11:40:04 +02:00
lenOfUnfixedList - = 1
2010-09-06 04:01:57 +02:00
##
# Write the final data into output files.
2018-07-05 11:40:04 +02:00
#
2010-09-06 04:01:57 +02:00
def GenerateVpdFile ( self , MapFileName , BinFileName ) :
#Open an VPD file to process
try :
2015-12-01 05:22:16 +01:00
fVpdFile = open ( BinFileName , " wb " , 0 )
2010-09-06 04:01:57 +02:00
except :
# Open failed
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FILE_OPEN_FAILURE , " File open failed for %s " % self . VpdFileName , None )
2010-09-06 04:01:57 +02:00
try :
2015-12-01 05:22:16 +01:00
fMapFile = open ( MapFileName , " w " , 0 )
2010-09-06 04:01:57 +02:00
except :
# Open failed
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FILE_OPEN_FAILURE , " File open failed for %s " % self . MapFileName , None )
2018-06-25 12:31:36 +02:00
# Use a instance of BytesIO to cache data
fStringIO = BytesIO ( ' ' )
2015-12-01 05:22:16 +01:00
2010-09-06 04:01:57 +02:00
# Write the header of map file.
try :
fMapFile . write ( st . MAP_FILE_COMMENT_TEMPLATE + " \n " )
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FILE_WRITE_FAILURE , " Write data to file %s failed, please check whether the file been locked or using by other applications. " % self . MapFileName , None )
2010-09-06 04:01:57 +02:00
for eachPcd in self . PcdFixedOffsetSizeList :
# write map file
try :
2018-06-25 12:31:33 +02:00
fMapFile . write ( " %s | %s | %s | %s | %s \n " % ( eachPcd . PcdCName , eachPcd . SkuId , eachPcd . PcdOffset , eachPcd . PcdSize , eachPcd . PcdUnpackValue ) )
2010-09-06 04:01:57 +02:00
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FILE_WRITE_FAILURE , " Write data to file %s failed, please check whether the file been locked or using by other applications. " % self . MapFileName , None )
2010-09-06 04:01:57 +02:00
# Write Vpd binary file
2015-12-01 05:22:16 +01:00
fStringIO . seek ( eachPcd . PcdBinOffset )
2010-09-06 04:01:57 +02:00
if isinstance ( eachPcd . PcdValue , list ) :
ValueList = [ chr ( Item ) for Item in eachPcd . PcdValue ]
2015-12-01 05:22:16 +01:00
fStringIO . write ( ' ' . join ( ValueList ) )
else :
2010-09-06 04:01:57 +02:00
fStringIO . write ( eachPcd . PcdValue )
2015-12-01 05:22:16 +01:00
try :
2010-09-06 04:01:57 +02:00
fVpdFile . write ( fStringIO . getvalue ( ) )
except :
2015-12-01 05:22:16 +01:00
EdkLogger . error ( " BPDG " , BuildToolError . FILE_WRITE_FAILURE , " Write data to file %s failed, please check whether the file been locked or using by other applications. " % self . VpdFileName , None )
2010-09-06 04:01:57 +02:00
fStringIO . close ( )
fVpdFile . close ( )
fMapFile . close ( )
2018-07-05 11:40:04 +02:00