| ## @file | |
| # process rule section generation | |
| # | |
| # Copyright (c) 2007 - 2018, Intel Corporation. All rights reserved.<BR> | |
| # | |
| # SPDX-License-Identifier: BSD-2-Clause-Patent | |
| # | |
| ## | |
| # Import Modules | |
| # | |
| from __future__ import absolute_import | |
| from struct import * | |
| from . import Section | |
| from .GenFdsGlobalVariable import GenFdsGlobalVariable | |
| import subprocess | |
| from .Ffs import SectionSuffix | |
| import Common.LongFilePathOs as os | |
| from CommonDataClass.FdfClass import EfiSectionClassObject | |
| from Common import EdkLogger | |
| from Common.BuildToolError import * | |
| from Common.Misc import PeImageClass | |
| from Common.LongFilePathSupport import OpenLongFilePath as open | |
| from Common.LongFilePathSupport import CopyLongFilePath | |
| from Common.DataType import * | |
| ## generate rule section | |
| # | |
| # | |
| class EfiSection (EfiSectionClassObject): | |
| ## The constructor | |
| # | |
| # @param self The object pointer | |
| # | |
| def __init__(self): | |
| EfiSectionClassObject.__init__(self) | |
| ## GenSection() method | |
| # | |
| # Generate rule section | |
| # | |
| # @param self The object pointer | |
| # @param OutputPath Where to place output file | |
| # @param ModuleName Which module this section belongs to | |
| # @param SecNum Index of section | |
| # @param KeyStringList Filter for inputs of section generation | |
| # @param FfsInf FfsInfStatement object that contains this section data | |
| # @param Dict dictionary contains macro and its value | |
| # @retval tuple (Generated file name list, section alignment) | |
| # | |
| def GenSection(self, OutputPath, ModuleName, SecNum, KeyStringList, FfsInf = None, Dict = None, IsMakefile = False) : | |
| if self.FileName is not None and self.FileName.startswith('PCD('): | |
| self.FileName = GenFdsGlobalVariable.GetPcdValue(self.FileName) | |
| """Prepare the parameter of GenSection""" | |
| if FfsInf is not None : | |
| InfFileName = FfsInf.InfFileName | |
| SectionType = FfsInf.__ExtendMacro__(self.SectionType) | |
| Filename = FfsInf.__ExtendMacro__(self.FileName) | |
| BuildNum = FfsInf.__ExtendMacro__(self.BuildNum) | |
| StringData = FfsInf.__ExtendMacro__(self.StringData) | |
| ModuleNameStr = FfsInf.__ExtendMacro__('$(MODULE_NAME)') | |
| NoStrip = True | |
| if FfsInf.ModuleType in (SUP_MODULE_SEC, SUP_MODULE_PEI_CORE, SUP_MODULE_PEIM, SUP_MODULE_MM_CORE_STANDALONE) and SectionType in (BINARY_FILE_TYPE_TE, BINARY_FILE_TYPE_PE32): | |
| if FfsInf.KeepReloc is not None: | |
| NoStrip = FfsInf.KeepReloc | |
| elif FfsInf.KeepRelocFromRule is not None: | |
| NoStrip = FfsInf.KeepRelocFromRule | |
| elif self.KeepReloc is not None: | |
| NoStrip = self.KeepReloc | |
| elif FfsInf.ShadowFromInfFile is not None: | |
| NoStrip = FfsInf.ShadowFromInfFile | |
| else: | |
| EdkLogger.error("GenFds", GENFDS_ERROR, "Module %s apply rule for None!" %ModuleName) | |
| """If the file name was pointed out, add it in FileList""" | |
| FileList = [] | |
| if Dict is None: | |
| Dict = {} | |
| if Filename is not None: | |
| Filename = GenFdsGlobalVariable.MacroExtend(Filename, Dict) | |
| # check if the path is absolute or relative | |
| if os.path.isabs(Filename): | |
| Filename = os.path.normpath(Filename) | |
| else: | |
| Filename = os.path.normpath(os.path.join(FfsInf.EfiOutputPath, Filename)) | |
| if not self.Optional: | |
| FileList.append(Filename) | |
| elif os.path.exists(Filename): | |
| FileList.append(Filename) | |
| elif IsMakefile: | |
| SuffixMap = FfsInf.GetFinalTargetSuffixMap() | |
| if '.depex' in SuffixMap: | |
| FileList.append(Filename) | |
| else: | |
| FileList, IsSect = Section.Section.GetFileList(FfsInf, self.FileType, self.FileExtension, Dict, IsMakefile=IsMakefile, SectionType=SectionType) | |
| if IsSect : | |
| return FileList, self.Alignment | |
| Index = 0 | |
| Align = self.Alignment | |
| """ If Section type is 'VERSION'""" | |
| OutputFileList = [] | |
| if SectionType == 'VERSION': | |
| InfOverrideVerString = False | |
| if FfsInf.Version is not None: | |
| #StringData = FfsInf.Version | |
| BuildNum = FfsInf.Version | |
| InfOverrideVerString = True | |
| if InfOverrideVerString: | |
| #VerTuple = ('-n', '"' + StringData + '"') | |
| if BuildNum is not None and BuildNum != '': | |
| BuildNumTuple = ('-j', BuildNum) | |
| else: | |
| BuildNumTuple = tuple() | |
| Num = SecNum | |
| OutputFile = os.path.join( OutputPath, ModuleName + SUP_MODULE_SEC + str(Num) + SectionSuffix.get(SectionType)) | |
| GenFdsGlobalVariable.GenerateSection(OutputFile, [], 'EFI_SECTION_VERSION', | |
| #Ui=StringData, | |
| Ver=BuildNum, | |
| IsMakefile=IsMakefile) | |
| OutputFileList.append(OutputFile) | |
| elif FileList != []: | |
| for File in FileList: | |
| Index = Index + 1 | |
| Num = '%s.%d' %(SecNum, Index) | |
| OutputFile = os.path.join(OutputPath, ModuleName + SUP_MODULE_SEC + Num + SectionSuffix.get(SectionType)) | |
| f = open(File, 'r') | |
| VerString = f.read() | |
| f.close() | |
| BuildNum = VerString | |
| if BuildNum is not None and BuildNum != '': | |
| BuildNumTuple = ('-j', BuildNum) | |
| GenFdsGlobalVariable.GenerateSection(OutputFile, [], 'EFI_SECTION_VERSION', | |
| #Ui=VerString, | |
| Ver=BuildNum, | |
| IsMakefile=IsMakefile) | |
| OutputFileList.append(OutputFile) | |
| else: | |
| BuildNum = StringData | |
| if BuildNum is not None and BuildNum != '': | |
| BuildNumTuple = ('-j', BuildNum) | |
| else: | |
| BuildNumTuple = tuple() | |
| BuildNumString = ' ' + ' '.join(BuildNumTuple) | |
| #if VerString == '' and | |
| if BuildNumString == '': | |
| if self.Optional == True : | |
| GenFdsGlobalVariable.VerboseLogger( "Optional Section doesn't exist!") | |
| return [], None | |
| else: | |
| EdkLogger.error("GenFds", GENFDS_ERROR, "File: %s miss Version Section value" %InfFileName) | |
| Num = SecNum | |
| OutputFile = os.path.join( OutputPath, ModuleName + SUP_MODULE_SEC + str(Num) + SectionSuffix.get(SectionType)) | |
| GenFdsGlobalVariable.GenerateSection(OutputFile, [], 'EFI_SECTION_VERSION', | |
| #Ui=VerString, | |
| Ver=BuildNum, | |
| IsMakefile=IsMakefile) | |
| OutputFileList.append(OutputFile) | |
| # | |
| # If Section Type is BINARY_FILE_TYPE_UI | |
| # | |
| elif SectionType == BINARY_FILE_TYPE_UI: | |
| InfOverrideUiString = False | |
| if FfsInf.Ui is not None: | |
| StringData = FfsInf.Ui | |
| InfOverrideUiString = True | |
| if InfOverrideUiString: | |
| Num = SecNum | |
| if IsMakefile and StringData == ModuleNameStr: | |
| StringData = "$(MODULE_NAME)" | |
| OutputFile = os.path.join( OutputPath, ModuleName + SUP_MODULE_SEC + str(Num) + SectionSuffix.get(SectionType)) | |
| GenFdsGlobalVariable.GenerateSection(OutputFile, [], 'EFI_SECTION_USER_INTERFACE', | |
| Ui=StringData, IsMakefile=IsMakefile) | |
| OutputFileList.append(OutputFile) | |
| elif FileList != []: | |
| for File in FileList: | |
| Index = Index + 1 | |
| Num = '%s.%d' %(SecNum, Index) | |
| OutputFile = os.path.join(OutputPath, ModuleName + SUP_MODULE_SEC + Num + SectionSuffix.get(SectionType)) | |
| f = open(File, 'r') | |
| UiString = f.read() | |
| f.close() | |
| if IsMakefile and UiString == ModuleNameStr: | |
| UiString = "$(MODULE_NAME)" | |
| GenFdsGlobalVariable.GenerateSection(OutputFile, [], 'EFI_SECTION_USER_INTERFACE', | |
| Ui=UiString, IsMakefile=IsMakefile) | |
| OutputFileList.append(OutputFile) | |
| else: | |
| if StringData is not None and len(StringData) > 0: | |
| UiTuple = ('-n', '"' + StringData + '"') | |
| else: | |
| UiTuple = tuple() | |
| if self.Optional == True : | |
| GenFdsGlobalVariable.VerboseLogger( "Optional Section don't exist!") | |
| return '', None | |
| else: | |
| EdkLogger.error("GenFds", GENFDS_ERROR, "File: %s miss UI Section value" %InfFileName) | |
| Num = SecNum | |
| if IsMakefile and StringData == ModuleNameStr: | |
| StringData = "$(MODULE_NAME)" | |
| OutputFile = os.path.join( OutputPath, ModuleName + SUP_MODULE_SEC + str(Num) + SectionSuffix.get(SectionType)) | |
| GenFdsGlobalVariable.GenerateSection(OutputFile, [], 'EFI_SECTION_USER_INTERFACE', | |
| Ui=StringData, IsMakefile=IsMakefile) | |
| OutputFileList.append(OutputFile) | |
| # | |
| # If Section Type is BINARY_FILE_TYPE_RAW | |
| # | |
| elif SectionType == BINARY_FILE_TYPE_RAW: | |
| """If File List is empty""" | |
| if FileList == []: | |
| if self.Optional == True: | |
| GenFdsGlobalVariable.VerboseLogger("Optional Section don't exist!") | |
| return [], None | |
| else: | |
| EdkLogger.error("GenFds", GENFDS_ERROR, "Output file for %s section could not be found for %s" % (SectionType, InfFileName)) | |
| elif len(FileList) > 1: | |
| EdkLogger.error("GenFds", GENFDS_ERROR, | |
| "Files suffixed with %s are not allowed to have more than one file in %s[Binaries] section" % ( | |
| self.FileExtension, InfFileName)) | |
| else: | |
| for File in FileList: | |
| File = GenFdsGlobalVariable.MacroExtend(File, Dict) | |
| OutputFileList.append(File) | |
| else: | |
| """If File List is empty""" | |
| if FileList == [] : | |
| if self.Optional == True: | |
| GenFdsGlobalVariable.VerboseLogger("Optional Section don't exist!") | |
| return [], None | |
| else: | |
| EdkLogger.error("GenFds", GENFDS_ERROR, "Output file for %s section could not be found for %s" % (SectionType, InfFileName)) | |
| else: | |
| """Convert the File to Section file one by one """ | |
| for File in FileList: | |
| """ Copy Map file to FFS output path """ | |
| Index = Index + 1 | |
| Num = '%s.%d' %(SecNum, Index) | |
| OutputFile = os.path.join( OutputPath, ModuleName + SUP_MODULE_SEC + Num + SectionSuffix.get(SectionType)) | |
| File = GenFdsGlobalVariable.MacroExtend(File, Dict) | |
| #Get PE Section alignment when align is set to AUTO | |
| if self.Alignment == 'Auto' and (SectionType == BINARY_FILE_TYPE_PE32 or SectionType == BINARY_FILE_TYPE_TE): | |
| Align = "0" | |
| if File[(len(File)-4):] == '.efi' and FfsInf.InfModule.BaseName == os.path.basename(File)[:-4]: | |
| MapFile = File.replace('.efi', '.map') | |
| CopyMapFile = os.path.join(OutputPath, ModuleName + '.map') | |
| if IsMakefile: | |
| if GenFdsGlobalVariable.CopyList == []: | |
| GenFdsGlobalVariable.CopyList = [(MapFile, CopyMapFile)] | |
| else: | |
| GenFdsGlobalVariable.CopyList.append((MapFile, CopyMapFile)) | |
| else: | |
| if os.path.exists(MapFile): | |
| if not os.path.exists(CopyMapFile) or \ | |
| (os.path.getmtime(MapFile) > os.path.getmtime(CopyMapFile)): | |
| CopyLongFilePath(MapFile, CopyMapFile) | |
| if not NoStrip: | |
| FileBeforeStrip = os.path.join(OutputPath, ModuleName + '.efi') | |
| if IsMakefile: | |
| if GenFdsGlobalVariable.CopyList == []: | |
| GenFdsGlobalVariable.CopyList = [(File, FileBeforeStrip)] | |
| else: | |
| GenFdsGlobalVariable.CopyList.append((File, FileBeforeStrip)) | |
| else: | |
| if not os.path.exists(FileBeforeStrip) or \ | |
| (os.path.getmtime(File) > os.path.getmtime(FileBeforeStrip)): | |
| CopyLongFilePath(File, FileBeforeStrip) | |
| StrippedFile = os.path.join(OutputPath, ModuleName + '.stripped') | |
| GenFdsGlobalVariable.GenerateFirmwareImage( | |
| StrippedFile, | |
| [File], | |
| Strip=True, | |
| IsMakefile = IsMakefile | |
| ) | |
| File = StrippedFile | |
| """For TE Section call GenFw to generate TE image""" | |
| if SectionType == BINARY_FILE_TYPE_TE: | |
| TeFile = os.path.join( OutputPath, ModuleName + 'Te.raw') | |
| GenFdsGlobalVariable.GenerateFirmwareImage( | |
| TeFile, | |
| [File], | |
| Type='te', | |
| IsMakefile = IsMakefile | |
| ) | |
| File = TeFile | |
| """Call GenSection""" | |
| GenFdsGlobalVariable.GenerateSection(OutputFile, | |
| [File], | |
| Section.Section.SectionType.get (SectionType), | |
| IsMakefile=IsMakefile | |
| ) | |
| OutputFileList.append(OutputFile) | |
| return OutputFileList, Align |