https://bugzilla.tianocore.org/show_bug.cgi?id=1373 Replace BSD 2-Clause License with BSD+Patent License. This change is based on the following emails: https://lists.01.org/pipermail/edk2-devel/2019-February/036260.html https://lists.01.org/pipermail/edk2-devel/2018-October/030385.html RFCs with detailed process for the license change: V3: https://lists.01.org/pipermail/edk2-devel/2019-March/038116.html V2: https://lists.01.org/pipermail/edk2-devel/2019-March/037669.html V1: https://lists.01.org/pipermail/edk2-devel/2019-March/037500.html Contributed-under: TianoCore Contribution Agreement 1.1 Signed-off-by: Michael D Kinney <michael.d.kinney@intel.com> Reviewed-by: Bob Feng <bob.c.feng@intel.com>
		
			
				
	
	
		
			255 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			255 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| ## @file
 | |
| # Target Tool Parser
 | |
| #
 | |
| #  Copyright (c) 2007 - 2018, Intel Corporation. All rights reserved.<BR>
 | |
| #
 | |
| #  SPDX-License-Identifier: BSD-2-Clause-Patent
 | |
| #
 | |
| 
 | |
| from __future__ import print_function
 | |
| import Common.LongFilePathOs as os
 | |
| import sys
 | |
| import traceback
 | |
| from optparse import OptionParser
 | |
| 
 | |
| import Common.EdkLogger as EdkLogger
 | |
| import Common.BuildToolError as BuildToolError
 | |
| from Common.DataType import *
 | |
| from Common.BuildVersion import gBUILD_VERSION
 | |
| from Common.LongFilePathSupport import OpenLongFilePath as open
 | |
| 
 | |
| # To Do 1.set clean, 2. add item, if the line is disabled.
 | |
| 
 | |
| class TargetTool():
 | |
|     def __init__(self, opt, args):
 | |
|         self.WorkSpace = os.path.normpath(os.getenv('WORKSPACE'))
 | |
|         self.Opt       = opt
 | |
|         self.Arg       = args[0]
 | |
|         self.FileName  = os.path.normpath(os.path.join(self.WorkSpace, 'Conf', 'target.txt'))
 | |
|         if os.path.isfile(self.FileName) == False:
 | |
|             print("%s does not exist." % self.FileName)
 | |
|             sys.exit(1)
 | |
|         self.TargetTxtDictionary = {
 | |
|             TAB_TAT_DEFINES_ACTIVE_PLATFORM                            : None,
 | |
|             TAB_TAT_DEFINES_TOOL_CHAIN_CONF                            : None,
 | |
|             TAB_TAT_DEFINES_MAX_CONCURRENT_THREAD_NUMBER               : None,
 | |
|             TAB_TAT_DEFINES_TARGET                                     : None,
 | |
|             TAB_TAT_DEFINES_TOOL_CHAIN_TAG                             : None,
 | |
|             TAB_TAT_DEFINES_TARGET_ARCH                                : None,
 | |
|             TAB_TAT_DEFINES_BUILD_RULE_CONF                            : None,
 | |
|         }
 | |
|         self.LoadTargetTxtFile(self.FileName)
 | |
| 
 | |
|     def LoadTargetTxtFile(self, filename):
 | |
|         if os.path.exists(filename) and os.path.isfile(filename):
 | |
|             return self.ConvertTextFileToDict(filename, '#', '=')
 | |
|         else:
 | |
|             raise ParseError('LoadTargetTxtFile() : No Target.txt file exists.')
 | |
|             return 1
 | |
| 
 | |
| #
 | |
| # Convert a text file to a dictionary
 | |
| #
 | |
|     def ConvertTextFileToDict(self, FileName, CommentCharacter, KeySplitCharacter):
 | |
|         """Convert a text file to a dictionary of (name:value) pairs."""
 | |
|         try:
 | |
|             f = open(FileName, 'r')
 | |
|             for Line in f:
 | |
|                 if Line.startswith(CommentCharacter) or Line.strip() == '':
 | |
|                     continue
 | |
|                 LineList = Line.split(KeySplitCharacter, 1)
 | |
|                 if len(LineList) >= 2:
 | |
|                     Key = LineList[0].strip()
 | |
|                     if Key.startswith(CommentCharacter) == False and Key in self.TargetTxtDictionary:
 | |
|                         if Key == TAB_TAT_DEFINES_ACTIVE_PLATFORM or Key == TAB_TAT_DEFINES_TOOL_CHAIN_CONF \
 | |
|                           or Key == TAB_TAT_DEFINES_MAX_CONCURRENT_THREAD_NUMBER \
 | |
|                           or Key == TAB_TAT_DEFINES_ACTIVE_MODULE:
 | |
|                             self.TargetTxtDictionary[Key] = LineList[1].replace('\\', '/').strip()
 | |
|                         elif Key == TAB_TAT_DEFINES_TARGET or Key == TAB_TAT_DEFINES_TARGET_ARCH \
 | |
|                           or Key == TAB_TAT_DEFINES_TOOL_CHAIN_TAG or Key == TAB_TAT_DEFINES_BUILD_RULE_CONF:
 | |
|                             self.TargetTxtDictionary[Key] = LineList[1].split()
 | |
|             f.close()
 | |
|             return 0
 | |
|         except:
 | |
|             last_type, last_value, last_tb = sys.exc_info()
 | |
|             traceback.print_exception(last_type, last_value, last_tb)
 | |
| 
 | |
|     def Print(self):
 | |
|         errMsg  = ''
 | |
|         for Key in self.TargetTxtDictionary:
 | |
|             if isinstance(self.TargetTxtDictionary[Key], type([])):
 | |
|                 print("%-30s = %s" % (Key, ''.join(elem + ' ' for elem in self.TargetTxtDictionary[Key])))
 | |
|             elif self.TargetTxtDictionary[Key] is None:
 | |
|                 errMsg += "  Missing %s configuration information, please use TargetTool to set value!" % Key + os.linesep
 | |
|             else:
 | |
|                 print("%-30s = %s" % (Key, self.TargetTxtDictionary[Key]))
 | |
| 
 | |
|         if errMsg != '':
 | |
|             print(os.linesep + 'Warning:' + os.linesep + errMsg)
 | |
| 
 | |
|     def RWFile(self, CommentCharacter, KeySplitCharacter, Num):
 | |
|         try:
 | |
|             fr = open(self.FileName, 'r')
 | |
|             fw = open(os.path.normpath(os.path.join(self.WorkSpace, 'Conf\\targetnew.txt')), 'w')
 | |
| 
 | |
|             existKeys = []
 | |
|             for Line in fr:
 | |
|                 if Line.startswith(CommentCharacter) or Line.strip() == '':
 | |
|                     fw.write(Line)
 | |
|                 else:
 | |
|                     LineList = Line.split(KeySplitCharacter, 1)
 | |
|                     if len(LineList) >= 2:
 | |
|                         Key = LineList[0].strip()
 | |
|                         if Key.startswith(CommentCharacter) == False and Key in self.TargetTxtDictionary:
 | |
|                             if Key not in existKeys:
 | |
|                                 existKeys.append(Key)
 | |
|                             else:
 | |
|                                 print("Warning: Found duplicate key item in original configuration files!")
 | |
| 
 | |
|                             if Num == 0:
 | |
|                                 Line = "%-30s = \n" % Key
 | |
|                             else:
 | |
|                                 ret = GetConfigureKeyValue(self, Key)
 | |
|                                 if ret is not None:
 | |
|                                     Line = ret
 | |
|                             fw.write(Line)
 | |
|             for key in self.TargetTxtDictionary:
 | |
|                 if key not in existKeys:
 | |
|                     print("Warning: %s does not exist in original configuration file" % key)
 | |
|                     Line = GetConfigureKeyValue(self, key)
 | |
|                     if Line is None:
 | |
|                         Line = "%-30s = " % key
 | |
|                     fw.write(Line)
 | |
| 
 | |
|             fr.close()
 | |
|             fw.close()
 | |
|             os.remove(self.FileName)
 | |
|             os.rename(os.path.normpath(os.path.join(self.WorkSpace, 'Conf\\targetnew.txt')), self.FileName)
 | |
| 
 | |
|         except:
 | |
|             last_type, last_value, last_tb = sys.exc_info()
 | |
|             traceback.print_exception(last_type, last_value, last_tb)
 | |
| 
 | |
| def GetConfigureKeyValue(self, Key):
 | |
|     Line = None
 | |
|     if Key == TAB_TAT_DEFINES_ACTIVE_PLATFORM and self.Opt.DSCFILE is not None:
 | |
|         dscFullPath = os.path.join(self.WorkSpace, self.Opt.DSCFILE)
 | |
|         if os.path.exists(dscFullPath):
 | |
|             Line = "%-30s = %s\n" % (Key, self.Opt.DSCFILE)
 | |
|         else:
 | |
|             EdkLogger.error("TargetTool", BuildToolError.FILE_NOT_FOUND,
 | |
|                             "DSC file %s does not exist!" % self.Opt.DSCFILE, RaiseError=False)
 | |
|     elif Key == TAB_TAT_DEFINES_TOOL_CHAIN_CONF and self.Opt.TOOL_DEFINITION_FILE is not None:
 | |
|         tooldefFullPath = os.path.join(self.WorkSpace, self.Opt.TOOL_DEFINITION_FILE)
 | |
|         if os.path.exists(tooldefFullPath):
 | |
|             Line = "%-30s = %s\n" % (Key, self.Opt.TOOL_DEFINITION_FILE)
 | |
|         else:
 | |
|             EdkLogger.error("TargetTool", BuildToolError.FILE_NOT_FOUND,
 | |
|                             "Tooldef file %s does not exist!" % self.Opt.TOOL_DEFINITION_FILE, RaiseError=False)
 | |
| 
 | |
|     elif self.Opt.NUM >= 2:
 | |
|         Line = "%-30s = %s\n" % (Key, 'Enable')
 | |
|     elif self.Opt.NUM <= 1:
 | |
|         Line = "%-30s = %s\n" % (Key, 'Disable')
 | |
|     elif Key == TAB_TAT_DEFINES_MAX_CONCURRENT_THREAD_NUMBER and self.Opt.NUM is not None:
 | |
|         Line = "%-30s = %s\n" % (Key, str(self.Opt.NUM))
 | |
|     elif Key == TAB_TAT_DEFINES_TARGET and self.Opt.TARGET is not None:
 | |
|         Line = "%-30s = %s\n" % (Key, ''.join(elem + ' ' for elem in self.Opt.TARGET))
 | |
|     elif Key == TAB_TAT_DEFINES_TARGET_ARCH and self.Opt.TARGET_ARCH is not None:
 | |
|         Line = "%-30s = %s\n" % (Key, ''.join(elem + ' ' for elem in self.Opt.TARGET_ARCH))
 | |
|     elif Key == TAB_TAT_DEFINES_TOOL_CHAIN_TAG and self.Opt.TOOL_CHAIN_TAG is not None:
 | |
|         Line = "%-30s = %s\n" % (Key, self.Opt.TOOL_CHAIN_TAG)
 | |
|     elif Key == TAB_TAT_DEFINES_BUILD_RULE_CONF and self.Opt.BUILD_RULE_FILE is not None:
 | |
|         buildruleFullPath = os.path.join(self.WorkSpace, self.Opt.BUILD_RULE_FILE)
 | |
|         if os.path.exists(buildruleFullPath):
 | |
|             Line = "%-30s = %s\n" % (Key, self.Opt.BUILD_RULE_FILE)
 | |
|         else:
 | |
|             EdkLogger.error("TagetTool", BuildToolError.FILE_NOT_FOUND,
 | |
|                             "Build rule file %s does not exist!" % self.Opt.BUILD_RULE_FILE, RaiseError=False)
 | |
|     return Line
 | |
| 
 | |
| VersionNumber = ("0.01" + " " + gBUILD_VERSION)
 | |
| __version__ = "%prog Version " + VersionNumber
 | |
| __copyright__ = "Copyright (c) 2007 - 2018, Intel Corporation  All rights reserved."
 | |
| __usage__ = "%prog [options] {args} \
 | |
| \nArgs:                                                  \
 | |
| \n Clean  clean the all default configuration of target.txt. \
 | |
| \n Print  print the all default configuration of target.txt. \
 | |
| \n Set    replace the default configuration with expected value specified by option."
 | |
| 
 | |
| gParamCheck = []
 | |
| def SingleCheckCallback(option, opt_str, value, parser):
 | |
|     if option not in gParamCheck:
 | |
|         setattr(parser.values, option.dest, value)
 | |
|         gParamCheck.append(option)
 | |
|     else:
 | |
|         parser.error("Option %s only allows one instance in command line!" % option)
 | |
| 
 | |
| def RangeCheckCallback(option, opt_str, value, parser):
 | |
|     if option not in gParamCheck:
 | |
|         gParamCheck.append(option)
 | |
|         if value < 1 or value > 8:
 | |
|             parser.error("The count of multi-thread is not in valid range of 1 ~ 8.")
 | |
|         else:
 | |
|             setattr(parser.values, option.dest, value)
 | |
|     else:
 | |
|         parser.error("Option %s only allows one instance in command line!" % option)
 | |
| 
 | |
| def MyOptionParser():
 | |
|     parser = OptionParser(version=__version__, prog="TargetTool.exe", usage=__usage__, description=__copyright__)
 | |
|     parser.add_option("-a", "--arch", action="append", type="choice", choices=['IA32', 'X64', 'EBC', 'ARM', 'AARCH64', '0'], dest="TARGET_ARCH",
 | |
|         help="ARCHS is one of list: IA32, X64, ARM, AARCH64 or EBC, which replaces target.txt's TARGET_ARCH definition. To specify more archs, please repeat this option. 0 will clear this setting in target.txt and can't combine with other value.")
 | |
|     parser.add_option("-p", "--platform", action="callback", type="string", dest="DSCFILE", callback=SingleCheckCallback,
 | |
|         help="Specify a DSC file, which replace target.txt's ACTIVE_PLATFORM definition. 0 will clear this setting in target.txt and can't combine with other value.")
 | |
|     parser.add_option("-c", "--tooldef", action="callback", type="string", dest="TOOL_DEFINITION_FILE", callback=SingleCheckCallback,
 | |
|         help="Specify the WORKSPACE relative path of tool_def.txt file, which replace target.txt's TOOL_CHAIN_CONF definition. 0 will clear this setting in target.txt and can't combine with other value.")
 | |
|     parser.add_option("-t", "--target", action="append", type="choice", choices=['DEBUG', 'RELEASE', '0'], dest="TARGET",
 | |
|         help="TARGET is one of list: DEBUG, RELEASE, which replaces target.txt's TARGET definition. To specify more TARGET, please repeat this option. 0 will clear this setting in target.txt and can't combine with other value.")
 | |
|     parser.add_option("-n", "--tagname", action="callback", type="string", dest="TOOL_CHAIN_TAG", callback=SingleCheckCallback,
 | |
|         help="Specify the Tool Chain Tagname, which replaces target.txt's TOOL_CHAIN_TAG definition. 0 will clear this setting in target.txt and can't combine with other value.")
 | |
|     parser.add_option("-r", "--buildrule", action="callback", type="string", dest="BUILD_RULE_FILE", callback=SingleCheckCallback,
 | |
|         help="Specify the build rule configure file, which replaces target.txt's BUILD_RULE_CONF definition. If not specified, the default value Conf/build_rule.txt will be set.")
 | |
|     parser.add_option("-m", "--multithreadnum", action="callback", type="int", dest="NUM", callback=RangeCheckCallback,
 | |
|         help="Specify the multi-thread number which replace target.txt's MAX_CONCURRENT_THREAD_NUMBER. If the value is less than 2, MULTIPLE_THREAD will be disabled. If the value is larger than 1, MULTIPLE_THREAD will be enabled.")
 | |
|     (opt, args)=parser.parse_args()
 | |
|     return (opt, args)
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     EdkLogger.Initialize()
 | |
|     EdkLogger.SetLevel(EdkLogger.QUIET)
 | |
|     if os.getenv('WORKSPACE') is None:
 | |
|         print("ERROR: WORKSPACE should be specified or edksetup script should be executed before run TargetTool")
 | |
|         sys.exit(1)
 | |
| 
 | |
|     (opt, args) = MyOptionParser()
 | |
|     if len(args) != 1 or (args[0].lower() != 'print' and args[0].lower() != 'clean' and args[0].lower() != 'set'):
 | |
|         print("The number of args isn't 1 or the value of args is invalid.")
 | |
|         sys.exit(1)
 | |
|     if opt.NUM is not None and opt.NUM < 1:
 | |
|         print("The MAX_CONCURRENT_THREAD_NUMBER must be larger than 0.")
 | |
|         sys.exit(1)
 | |
|     if opt.TARGET is not None and len(opt.TARGET) > 1:
 | |
|         for elem in opt.TARGET:
 | |
|             if elem == '0':
 | |
|                 print("0 will clear the TARGET setting in target.txt and can't combine with other value.")
 | |
|                 sys.exit(1)
 | |
|     if opt.TARGET_ARCH is not None and len(opt.TARGET_ARCH) > 1:
 | |
|         for elem in opt.TARGET_ARCH:
 | |
|             if elem == '0':
 | |
|                 print("0 will clear the TARGET_ARCH setting in target.txt and can't combine with other value.")
 | |
|                 sys.exit(1)
 | |
| 
 | |
|     try:
 | |
|         FileHandle = TargetTool(opt, args)
 | |
|         if FileHandle.Arg.lower() == 'print':
 | |
|             FileHandle.Print()
 | |
|             sys.exit(0)
 | |
|         elif FileHandle.Arg.lower() == 'clean':
 | |
|             FileHandle.RWFile('#', '=', 0)
 | |
|         else:
 | |
|             FileHandle.RWFile('#', '=', 1)
 | |
|     except Exception as e:
 | |
|         last_type, last_value, last_tb = sys.exc_info()
 | |
|         traceback.print_exception(last_type, last_value, last_tb)
 | |
| 
 |