mirror of
				https://github.com/espressif/ESP8266_RTOS_SDK.git
				synced 2025-10-25 05:25:06 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			177 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			177 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/env python2
 | |
| #
 | |
| # Copyright 2018-2019 Espressif Systems (Shanghai) PTE LTD
 | |
| # 
 | |
| # Licensed under the Apache License, Version 2.0 (the "License");
 | |
| # you may not use this file except in compliance with the License.
 | |
| # You may obtain a copy of the License at
 | |
| # 
 | |
| #     http://www.apache.org/licenses/LICENSE-2.0
 | |
| # 
 | |
| # Unless required by applicable law or agreed to in writing, software
 | |
| # distributed under the License is distributed on an "AS IS" BASIS,
 | |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| # See the License for the specific language governing permissions and
 | |
| # limitations under the License.
 | |
| 
 | |
| import argparse
 | |
| import inspect
 | |
| import sys
 | |
| import binascii
 | |
| import struct
 | |
| 
 | |
| __version__ = "1.0.1"
 | |
| 
 | |
| FLASH_SECTOR_SIZE = 0x1000
 | |
| 
 | |
| PYTHON2 = sys.version_info[0] < 3
 | |
| 
 | |
| def esp8266_crc32(data):
 | |
|     """
 | |
|     CRC32 algorithm used by 8266 SDK bootloader (and gen_appbin.py).
 | |
|     """
 | |
|     crc = binascii.crc32(data, 0) & 0xFFFFFFFF
 | |
|     if crc & 0x80000000:
 | |
|         return crc ^ 0xFFFFFFFF
 | |
|     else:
 | |
|         return crc + 1
 | |
| 
 | |
| def version(args):
 | |
|     print(__version__)
 | |
| 
 | |
| # python pack_fw.py addr1 bin1 addr2 bin2 ...... 
 | |
| # The address must increase.
 | |
| 
 | |
| class proc_addr_file(argparse.Action):
 | |
|     """ Custom parser class for the address/filename pairs passed as arguments """
 | |
|     def __init__(self, option_strings, dest, nargs='+', **kwargs):
 | |
|         super(proc_addr_file, self).__init__(option_strings, dest, nargs, **kwargs)
 | |
| 
 | |
|     def __call__(self, parser, namespace, values, option_string=None):
 | |
|         pairs = []
 | |
|         for i in range(0, len(values) ,2):
 | |
|             try:
 | |
|                 address = int(values[i], 0)
 | |
|             except ValueError:
 | |
|                 raise argparse.ArgumentError(self, 'Address "%s" must be a number' % values[i])
 | |
|             try:
 | |
|                 argfile = open(values[i + 1], 'rb')
 | |
|             except IOError as e:
 | |
|                 raise argparse.ArgumentError(self, e)
 | |
|             pairs.append((address, argfile))
 | |
| 
 | |
|         end = 0
 | |
|         pairs = sorted(pairs)
 | |
|         for address, argfile in pairs:
 | |
|             argfile.seek(0,2)  # seek to end
 | |
|             size = argfile.tell()
 | |
|             argfile.seek(0)
 | |
|             sector_start = address & ~(FLASH_SECTOR_SIZE - 1)
 | |
|             sector_end = ((address + size + FLASH_SECTOR_SIZE - 1) & ~(FLASH_SECTOR_SIZE - 1)) - 1
 | |
|             if sector_start < end:
 | |
|                 message = 'Detected overlap at address: 0x%x for file: %s' % (address, argfile.name)
 | |
|                 raise argparse.ArgumentError(self, message)
 | |
|             end = sector_end
 | |
|         setattr(namespace, self.dest, pairs)
 | |
| 
 | |
| def pack3(args):
 | |
|     fw_data = ''
 | |
| 
 | |
|     try:
 | |
|         output_file = open(args.output, "w+")
 | |
|     except IOError as e:
 | |
|         print(e)
 | |
| 
 | |
|     end_addr = None
 | |
|     for address, argfile in args.addr_filename:
 | |
|         if address == 0:
 | |
|            address = 4096
 | |
| 
 | |
|         if end_addr is not None and address > end_addr:
 | |
|             data = (address - end_addr) * ['ff']
 | |
|             filled = binascii.a2b_hex(''.join(data))
 | |
|             fw_data += filled
 | |
| 
 | |
|         try:
 | |
|             argfile.seek(0, 0)
 | |
|             data = argfile.read()
 | |
|             fw_data += data
 | |
| 
 | |
|             argfile.seek(0, 2)
 | |
|             end_addr = address + argfile.tell() 
 | |
|         except IOError as e:
 | |
|             raise e
 | |
| 
 | |
|     crc32 = esp8266_crc32(fw_data)
 | |
|     fw_data += struct.pack('<I', crc32)
 | |
| 
 | |
|     try:
 | |
|         output_file.write(fw_data)
 | |
|         output_file.close()
 | |
|     except IOError as e:
 | |
|         raise e
 | |
|         
 | |
| 
 | |
| def main():
 | |
|     parser = argparse.ArgumentParser(description='pack_fw v%s - ESP8266 ROM Bootloader Utility' % __version__, prog='pack_fw')
 | |
|  
 | |
|     parser.add_argument(
 | |
|         '--output', '-o',
 | |
|         help='Output file name with full path',
 | |
|         default=None)
 | |
| 
 | |
|     subparsers = parser.add_subparsers(
 | |
|         dest='operation',
 | |
|         help='Run pack_fw {command} -h for additional help')
 | |
| 
 | |
|     parser_pack_fw = subparsers.add_parser(
 | |
|         'pack3',
 | |
|         help='Pack the V3 firmware')
 | |
|     parser_pack_fw.add_argument('addr_filename', metavar='<address> <filename>', help='Address followed by binary filename, separated by space',
 | |
|         action=proc_addr_file)
 | |
| 
 | |
|     print('pack_fw.py v%s' % __version__)
 | |
| 
 | |
|     args = parser.parse_args()
 | |
| 
 | |
|     if args.operation is None:
 | |
|         parser.print_help()
 | |
|         sys.exit(1)
 | |
| 
 | |
|     operation_func = globals()[args.operation]
 | |
| 
 | |
|     if PYTHON2:
 | |
|         # This function is depreciated in Python3
 | |
|         operation_args = inspect.getargspec(operation_func).args
 | |
|     else:
 | |
|         operation_args = inspect.getfullargspec(operation_func).args
 | |
| 
 | |
|     operation_func(args)
 | |
| 
 | |
| class FatalError(RuntimeError):
 | |
|     """
 | |
|     Wrapper class for runtime errors that aren't caused by internal bugs, but by
 | |
|     ESP8266 responses or input content.
 | |
|     """
 | |
|     def __init__(self, message):
 | |
|         RuntimeError.__init__(self, message)
 | |
| 
 | |
|     @staticmethod
 | |
|     def WithResult(message, result):
 | |
|         """
 | |
|         Return a fatal error object that appends the hex values of
 | |
|         'result' as a string formatted argument.
 | |
|         """
 | |
|         message += " (result was %s)" % hexify(result)
 | |
|         return FatalError(message)
 | |
| 
 | |
| def _main():
 | |
|     try:
 | |
|         main()
 | |
|     except FatalError as e:
 | |
|         print('\nA fatal error occurred: %s' % e)
 | |
|         sys.exit(2)
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     _main()
 | 
