2016-08-17 23:08:22 +08:00
#!/usr/bin/env python
#
# ESP32 partition table generation tool
#
# Converts partition tables to/from CSV and binary formats.
#
2018-06-18 17:07:16 +10:00
# See https://docs.espressif.com/projects/esp-idf/en/latest/api-guides/partition-tables.html
2017-05-12 11:58:28 +10:00
# for explanation of partition table structure and uses.
#
# Copyright 2015-2016 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.
2017-05-12 12:25:41 +10:00
from __future__ import print_function , division
2016-08-17 23:08:22 +08:00
import argparse
2017-01-26 01:47:53 +00:00
import os
import re
import struct
2016-08-17 23:08:22 +08:00
import sys
2018-01-31 14:45:12 +01:00
import hashlib
import binascii
2016-08-17 23:08:22 +08:00
2016-11-07 15:32:21 +11:00
MAX_PARTITION_LENGTH = 0xC00 # 3K for partition data (96 entries) leaves 1K in a 4K sector for signature
2018-01-31 14:45:12 +01:00
MD5_PARTITION_BEGIN = b " \xEB \xEB " + b " \xFF " * 14 # The first 2 bytes are like magic numbers for MD5 sum
2018-04-19 09:42:26 +05:00
PARTITION_TABLE_SIZE = 0x1000 # Size of partition table
2016-11-07 15:32:21 +11:00
2018-04-19 09:42:26 +05:00
__version__ = ' 1.1 '
2016-08-17 23:08:22 +08:00
quiet = False
2018-02-16 11:12:16 +01:00
md5sum = True
2018-07-13 11:52:57 +10:00
secure = False
2018-04-19 09:42:26 +05:00
offset_part_table = 0
2016-08-17 23:08:22 +08:00
def status ( msg ) :
""" Print status message to stderr """
if not quiet :
critical ( msg )
def critical ( msg ) :
""" Print critical message to stderr """
if not quiet :
sys . stderr . write ( msg )
sys . stderr . write ( ' \n ' )
class PartitionTable ( list ) :
def __init__ ( self ) :
super ( PartitionTable , self ) . __init__ ( self )
@classmethod
def from_csv ( cls , csv_contents ) :
res = PartitionTable ( )
2017-07-25 18:12:31 +02:00
lines = csv_contents . splitlines ( )
def expand_vars ( f ) :
f = os . path . expandvars ( f )
m = re . match ( r ' (?<! \\ ) \ $([A-Za-z_][A-Za-z0-9_]*) ' , f )
if m :
raise InputError ( " unknown variable ' %s ' " % m . group ( 1 ) )
return f
2016-08-17 23:08:22 +08:00
for line_no in range ( len ( lines ) ) :
2017-07-25 18:12:31 +02:00
line = expand_vars ( lines [ line_no ] ) . strip ( )
2016-08-17 23:08:22 +08:00
if line . startswith ( " # " ) or len ( line ) == 0 :
continue
try :
res . append ( PartitionDefinition . from_csv ( line ) )
except InputError as e :
raise InputError ( " Error at line %d : %s " % ( line_no + 1 , e ) )
except Exception :
critical ( " Unexpected error parsing line %d : %s " % ( line_no + 1 , line ) )
raise
# fix up missing offsets & negative sizes
2018-04-19 09:42:26 +05:00
last_end = offset_part_table + PARTITION_TABLE_SIZE # first offset after partition table
2016-08-17 23:08:22 +08:00
for e in res :
2018-04-19 09:42:26 +05:00
if offset_part_table != 0 and e . offset is not None and e . offset < last_end :
critical ( " WARNING: 0x %x address in the partition table is below 0x %x " % ( e . offset , last_end ) )
e . offset = None
2016-08-17 23:08:22 +08:00
if e . offset is None :
pad_to = 0x10000 if e . type == PartitionDefinition . APP_TYPE else 4
if last_end % pad_to != 0 :
last_end + = pad_to - ( last_end % pad_to )
e . offset = last_end
if e . size < 0 :
e . size = - e . size - e . offset
last_end = e . offset + e . size
return res
def __getitem__ ( self , item ) :
""" Allow partition table access via name as well as by
numeric index . """
if isinstance ( item , str ) :
for x in self :
if x . name == item :
return x
raise ValueError ( " No partition entry named ' %s ' " % item )
else :
return super ( PartitionTable , self ) . __getitem__ ( item )
2018-06-22 11:14:22 +10:00
def find_by_type ( self , ptype , subtype ) :
""" Return a partition by type & subtype, returns
None if not found """
TYPES = PartitionDefinition . TYPES
SUBTYPES = PartitionDefinition . SUBTYPES
# convert ptype & subtypes names (if supplied this way) to integer values
try :
ptype = TYPES [ ptype ]
except KeyError :
try :
ptypes = int ( ptype , 0 )
except TypeError :
pass
try :
subtype = SUBTYPES [ int ( ptype ) ] [ subtype ]
except KeyError :
try :
ptypes = int ( ptype , 0 )
except TypeError :
pass
for p in self :
if p . type == ptype and p . subtype == subtype :
return p
return None
def find_by_name ( self , name ) :
for p in self :
if p . name == name :
return p
return None
2016-08-17 23:08:22 +08:00
def verify ( self ) :
# verify each partition individually
for p in self :
p . verify ( )
# check for overlaps
last = None
2017-05-06 12:47:26 -06:00
for p in sorted ( self , key = lambda x : x . offset ) :
2018-04-19 09:42:26 +05:00
if p . offset < offset_part_table + PARTITION_TABLE_SIZE :
raise InputError ( " Partition offset 0x %x is below 0x %x " % ( p . offset , offset_part_table + PARTITION_TABLE_SIZE ) )
2016-08-17 23:08:22 +08:00
if last is not None and p . offset < last . offset + last . size :
raise InputError ( " Partition at 0x %x overlaps 0x %x -0x %x " % ( p . offset , last . offset , last . offset + last . size - 1 ) )
last = p
2018-04-20 16:57:15 +10:00
def flash_size ( self ) :
""" Return the size that partitions will occupy in flash
( ie the offset the last partition ends at )
"""
try :
last = sorted ( self , reverse = True ) [ 0 ]
except IndexError :
return 0 # empty table!
return last . offset + last . size
2016-08-17 23:08:22 +08:00
@classmethod
def from_binary ( cls , b ) :
2018-01-31 14:45:12 +01:00
md5 = hashlib . md5 ( ) ;
2016-08-17 23:08:22 +08:00
result = cls ( )
for o in range ( 0 , len ( b ) , 32 ) :
2016-11-07 15:45:57 +11:00
data = b [ o : o + 32 ]
if len ( data ) != 32 :
2016-11-11 17:00:34 +11:00
raise InputError ( " Partition table length must be a multiple of 32 bytes " )
2017-05-12 12:25:41 +10:00
if data == b ' \xFF ' * 32 :
2016-11-11 17:00:34 +11:00
return result # got end marker
2018-02-16 11:12:16 +01:00
if md5sum and data [ : 2 ] == MD5_PARTITION_BEGIN [ : 2 ] : #check only the magic number part
2018-01-31 14:45:12 +01:00
if data [ 16 : ] == md5 . digest ( ) :
continue # the next iteration will check for the end marker
else :
raise InputError ( " MD5 checksums don ' t match! (computed: 0x %s , parsed: 0x %s ) " % ( md5 . hexdigest ( ) , binascii . hexlify ( data [ 16 : ] ) ) )
else :
md5 . update ( data )
2016-11-07 15:45:57 +11:00
result . append ( PartitionDefinition . from_binary ( data ) )
2016-11-11 17:00:34 +11:00
raise InputError ( " Partition table is missing an end-of-table marker " )
2016-08-17 23:08:22 +08:00
def to_binary ( self ) :
2017-05-06 12:47:26 -06:00
result = b " " . join ( e . to_binary ( ) for e in self )
2018-02-16 11:12:16 +01:00
if md5sum :
result + = MD5_PARTITION_BEGIN + hashlib . md5 ( result ) . digest ( )
2016-11-07 15:32:21 +11:00
if len ( result ) > = MAX_PARTITION_LENGTH :
raise InputError ( " Binary partition table length ( %d ) longer than max " % len ( result ) )
2017-05-06 12:47:26 -06:00
result + = b " \xFF " * ( MAX_PARTITION_LENGTH - len ( result ) ) # pad the sector, for signing
2016-11-07 15:32:21 +11:00
return result
2016-08-17 23:08:22 +08:00
def to_csv ( self , simple_formatting = False ) :
rows = [ " # Espressif ESP32 Partition Table " ,
2016-11-11 17:00:34 +11:00
" # Name, Type, SubType, Offset, Size, Flags " ]
2016-08-17 23:08:22 +08:00
rows + = [ x . to_csv ( simple_formatting ) for x in self ]
return " \n " . join ( rows ) + " \n "
class PartitionDefinition ( object ) :
APP_TYPE = 0x00
DATA_TYPE = 0x01
TYPES = {
" app " : APP_TYPE ,
" data " : DATA_TYPE ,
}
2017-07-25 18:12:31 +02:00
2016-11-15 18:35:10 +08:00
# Keep this map in sync with esp_partition_subtype_t enum in esp_partition.h
2016-08-17 23:08:22 +08:00
SUBTYPES = {
APP_TYPE : {
" factory " : 0x00 ,
" test " : 0x20 ,
} ,
DATA_TYPE : {
" ota " : 0x00 ,
2016-11-15 18:35:10 +08:00
" phy " : 0x01 ,
" nvs " : 0x02 ,
2017-01-10 14:48:47 +03:00
" coredump " : 0x03 ,
2016-11-15 18:35:10 +08:00
" esphttpd " : 0x80 ,
" fat " : 0x81 ,
" spiffs " : 0x82 ,
2016-08-17 23:08:22 +08:00
} ,
}
2017-05-06 12:47:26 -06:00
MAGIC_BYTES = b " \xAA \x50 "
2016-08-17 23:08:22 +08:00
ALIGNMENT = {
2017-05-12 12:07:59 +10:00
APP_TYPE : 0x10000 ,
2016-08-17 23:08:22 +08:00
DATA_TYPE : 0x04 ,
}
2016-11-11 17:00:34 +11:00
# dictionary maps flag name (as used in CSV flags list, property name)
# to bit set in flags words in binary format
FLAGS = {
2017-01-24 14:42:02 +00:00
" encrypted " : 0
2016-11-11 17:00:34 +11:00
}
2018-06-22 11:14:22 +10:00
# add subtypes for the 16 OTA slot values ("ota_XX, etc.")
2016-08-17 23:08:22 +08:00
for ota_slot in range ( 16 ) :
SUBTYPES [ TYPES [ " app " ] ] [ " ota_ %d " % ota_slot ] = 0x10 + ota_slot
def __init__ ( self ) :
self . name = " "
self . type = None
self . subtype = None
self . offset = None
self . size = None
2016-11-11 17:00:34 +11:00
self . encrypted = False
2016-08-17 23:08:22 +08:00
@classmethod
def from_csv ( cls , line ) :
""" Parse a line from the CSV """
2016-11-11 17:00:34 +11:00
line_w_defaults = line + " ,,,, " # lazy way to support default fields
2017-07-25 18:12:31 +02:00
fields = [ f . strip ( ) for f in line_w_defaults . split ( " , " ) ]
2016-08-17 23:08:22 +08:00
res = PartitionDefinition ( )
res . name = fields [ 0 ]
res . type = res . parse_type ( fields [ 1 ] )
res . subtype = res . parse_subtype ( fields [ 2 ] )
res . offset = res . parse_address ( fields [ 3 ] )
res . size = res . parse_address ( fields [ 4 ] )
if res . size is None :
raise InputError ( " Size field can ' t be empty " )
2016-11-11 17:00:34 +11:00
flags = fields [ 5 ] . split ( " : " )
for flag in flags :
if flag in cls . FLAGS :
setattr ( res , flag , True )
elif len ( flag ) > 0 :
raise InputError ( " CSV flag column contains unknown flag ' %s ' " % ( flag ) )
2016-08-17 23:08:22 +08:00
return res
def __eq__ ( self , other ) :
return self . name == other . name and self . type == other . type \
and self . subtype == other . subtype and self . offset == other . offset \
and self . size == other . size
def __repr__ ( self ) :
def maybe_hex ( x ) :
return " 0x %x " % x if x is not None else " None "
return " PartitionDefinition( ' %s ' , 0x %x , 0x %x , %s , %s ) " % ( self . name , self . type , self . subtype or 0 ,
maybe_hex ( self . offset ) , maybe_hex ( self . size ) )
def __str__ ( self ) :
return " Part ' %s ' %d / %d @ 0x %x size 0x %x " % ( self . name , self . type , self . subtype , self . offset or - 1 , self . size or - 1 )
def __cmp__ ( self , other ) :
return self . offset - other . offset
2018-05-31 16:26:38 +02:00
def __lt__ ( self , other ) :
return self . offset < other . offset
def __gt__ ( self , other ) :
return self . offset > other . offset
def __le__ ( self , other ) :
return self . offset < = other . offset
def __ge__ ( self , other ) :
return self . offset > = other . offset
2016-08-17 23:08:22 +08:00
def parse_type ( self , strval ) :
if strval == " " :
raise InputError ( " Field ' type ' can ' t be left empty. " )
return parse_int ( strval , self . TYPES )
def parse_subtype ( self , strval ) :
if strval == " " :
return 0 # default
return parse_int ( strval , self . SUBTYPES . get ( self . type , { } ) )
def parse_address ( self , strval ) :
if strval == " " :
return None # PartitionTable will fill in default
return parse_int ( strval )
def verify ( self ) :
if self . type is None :
2017-05-12 12:07:59 +10:00
raise ValidationError ( self , " Type field is not set " )
2016-08-17 23:08:22 +08:00
if self . subtype is None :
2017-05-12 12:07:59 +10:00
raise ValidationError ( self , " Subtype field is not set " )
2016-08-17 23:08:22 +08:00
if self . offset is None :
2017-05-12 12:07:59 +10:00
raise ValidationError ( self , " Offset field is not set " )
2016-08-17 23:08:22 +08:00
align = self . ALIGNMENT . get ( self . type , 4 )
if self . offset % align :
2017-05-12 12:07:59 +10:00
raise ValidationError ( self , " Offset 0x %x is not aligned to 0x %x " % ( self . offset , align ) )
2018-07-13 11:52:57 +10:00
if self . size % align and secure :
raise ValidationError ( self , " Size 0x %x is not aligned to 0x %x " % ( self . size , align ) )
2016-08-17 23:08:22 +08:00
if self . size is None :
2017-05-12 12:07:59 +10:00
raise ValidationError ( self , " Size field is not set " )
2016-08-17 23:08:22 +08:00
STRUCT_FORMAT = " <2sBBLL16sL "
@classmethod
def from_binary ( cls , b ) :
if len ( b ) != 32 :
raise InputError ( " Partition definition length must be exactly 32 bytes. Got %d bytes. " % len ( b ) )
res = cls ( )
( magic , res . type , res . subtype , res . offset ,
2016-11-11 17:00:34 +11:00
res . size , res . name , flags ) = struct . unpack ( cls . STRUCT_FORMAT , b )
2017-05-12 12:25:41 +10:00
if b " \x00 " in res . name : # strip null byte padding from name string
res . name = res . name [ : res . name . index ( b " \x00 " ) ]
res . name = res . name . decode ( )
2016-08-17 23:08:22 +08:00
if magic != cls . MAGIC_BYTES :
raise InputError ( " Invalid magic bytes ( %r ) for partition definition " % magic )
2016-11-11 17:00:34 +11:00
for flag , bit in cls . FLAGS . items ( ) :
if flags & ( 1 << bit ) :
setattr ( res , flag , True )
flags & = ~ ( 1 << bit )
if flags != 0 :
critical ( " WARNING: Partition definition had unknown flag(s) 0x %08x . Newer binary format? " % flags )
2016-08-17 23:08:22 +08:00
return res
2016-11-11 17:00:34 +11:00
def get_flags_list ( self ) :
return [ flag for flag in self . FLAGS . keys ( ) if getattr ( self , flag ) ]
2016-08-17 23:08:22 +08:00
def to_binary ( self ) :
2016-11-11 17:00:34 +11:00
flags = sum ( ( 1 << self . FLAGS [ flag ] ) for flag in self . get_flags_list ( ) )
2016-08-17 23:08:22 +08:00
return struct . pack ( self . STRUCT_FORMAT ,
self . MAGIC_BYTES ,
self . type , self . subtype ,
self . offset , self . size ,
2017-05-06 12:47:26 -06:00
self . name . encode ( ) ,
2016-11-11 17:00:34 +11:00
flags )
2016-08-17 23:08:22 +08:00
def to_csv ( self , simple_formatting = False ) :
def addr_format ( a , include_sizes ) :
if not simple_formatting and include_sizes :
for ( val , suffix ) in [ ( 0x100000 , " M " ) , ( 0x400 , " K " ) ] :
if a % val == 0 :
2017-05-12 12:25:41 +10:00
return " %d %s " % ( a / / val , suffix )
2016-08-17 23:08:22 +08:00
return " 0x %x " % a
def lookup_keyword ( t , keywords ) :
for k , v in keywords . items ( ) :
if simple_formatting == False and t == v :
return k
return " %d " % t
2016-11-11 17:00:34 +11:00
def generate_text_flags ( ) :
""" colon-delimited list of flags """
return " : " . join ( self . get_flags_list ( ) )
2016-08-17 23:08:22 +08:00
return " , " . join ( [ self . name ,
lookup_keyword ( self . type , self . TYPES ) ,
lookup_keyword ( self . subtype , self . SUBTYPES . get ( self . type , { } ) ) ,
addr_format ( self . offset , False ) ,
2016-11-11 17:00:34 +11:00
addr_format ( self . size , True ) ,
generate_text_flags ( ) ] )
2016-08-17 23:08:22 +08:00
def parse_int ( v , keywords = { } ) :
""" Generic parser for integer fields - int(x,0) with provision for
k / m / K / M suffixes and ' keyword ' value lookup .
"""
try :
for letter , multiplier in [ ( " k " , 1024 ) , ( " m " , 1024 * 1024 ) ] :
if v . lower ( ) . endswith ( letter ) :
return parse_int ( v [ : - 1 ] , keywords ) * multiplier
return int ( v , 0 )
except ValueError :
if len ( keywords ) == 0 :
raise InputError ( " Invalid field value %s " % v )
try :
return keywords [ v . lower ( ) ]
except KeyError :
raise InputError ( " Value ' %s ' is not valid. Known keywords: %s " % ( v , " , " . join ( keywords ) ) )
def main ( ) :
global quiet
2018-02-16 11:12:16 +01:00
global md5sum
2018-04-19 09:42:26 +05:00
global offset_part_table
2018-07-13 11:52:57 +10:00
global secure
2016-08-17 23:08:22 +08:00
parser = argparse . ArgumentParser ( description = ' ESP32 partition table utility ' )
2018-04-20 16:57:15 +10:00
parser . add_argument ( ' --flash-size ' , help = ' Optional flash size limit, checks partition table fits in flash ' ,
nargs = ' ? ' , choices = [ ' 1MB ' , ' 2MB ' , ' 4MB ' , ' 8MB ' , ' 16MB ' ] )
2018-02-16 11:12:16 +01:00
parser . add_argument ( ' --disable-md5sum ' , help = ' Disable md5 checksum for the partition table ' , default = False , action = ' store_true ' )
2016-08-17 23:08:22 +08:00
parser . add_argument ( ' --verify ' , ' -v ' , help = ' Verify partition table fields ' , default = True , action = ' store_false ' )
parser . add_argument ( ' --quiet ' , ' -q ' , help = " Don ' t print status messages to stderr " , action = ' store_true ' )
2018-04-19 09:42:26 +05:00
parser . add_argument ( ' --offset ' , ' -o ' , help = ' Set offset partition table ' , default = ' 0x8000 ' )
2018-07-13 11:52:57 +10:00
parser . add_argument ( ' --secure ' , help = " Require app partitions to be suitable for secure boot " , action = ' store_true ' )
2017-05-12 12:25:41 +10:00
parser . add_argument ( ' input ' , help = ' Path to CSV or binary file to parse. Will use stdin if omitted. ' , type = argparse . FileType ( ' rb ' ) , default = sys . stdin )
2016-08-17 23:08:22 +08:00
parser . add_argument ( ' output ' , help = ' Path to output converted binary or CSV file. Will use stdout if omitted, unless the --display argument is also passed (in which case only the summary is printed.) ' ,
nargs = ' ? ' ,
default = ' - ' )
args = parser . parse_args ( )
quiet = args . quiet
2018-02-16 11:12:16 +01:00
md5sum = not args . disable_md5sum
2018-07-13 11:52:57 +10:00
secure = args . secure
2018-04-19 09:42:26 +05:00
offset_part_table = int ( args . offset , 0 )
2016-08-17 23:08:22 +08:00
input = args . input . read ( )
input_is_binary = input [ 0 : 2 ] == PartitionDefinition . MAGIC_BYTES
if input_is_binary :
status ( " Parsing binary partition input... " )
table = PartitionTable . from_binary ( input )
else :
2017-05-12 12:25:41 +10:00
input = input . decode ( )
2016-08-17 23:08:22 +08:00
status ( " Parsing CSV input... " )
table = PartitionTable . from_csv ( input )
if args . verify :
status ( " Verifying table... " )
table . verify ( )
2018-04-20 16:57:15 +10:00
if args . flash_size :
size_mb = int ( args . flash_size . replace ( " MB " , " " ) )
size = size_mb * 1024 * 1024 # flash memory uses honest megabytes!
table_size = table . flash_size ( )
if size < table_size :
raise InputError ( " Partitions defined in ' %s ' occupy %.1f MB of flash ( %d bytes) which does not fit in configured flash size %d MB. Change the flash size in menuconfig under the ' Serial Flasher Config ' menu. " %
( args . input . name , table_size / 1024.0 / 1024.0 , table_size , size_mb ) )
2016-08-17 23:08:22 +08:00
if input_is_binary :
output = table . to_csv ( )
2017-05-06 12:47:26 -06:00
with sys . stdout if args . output == ' - ' else open ( args . output , ' w ' ) as f :
f . write ( output )
2016-08-17 23:08:22 +08:00
else :
output = table . to_binary ( )
2017-05-06 12:47:26 -06:00
with sys . stdout . buffer if args . output == ' - ' else open ( args . output , ' wb ' ) as f :
f . write ( output )
2016-08-17 23:08:22 +08:00
2017-05-12 12:07:59 +10:00
class InputError ( RuntimeError ) :
def __init__ ( self , e ) :
super ( InputError , self ) . __init__ ( e )
class ValidationError ( InputError ) :
def __init__ ( self , partition , message ) :
super ( ValidationError , self ) . __init__ (
" Partition %s invalid: %s " % ( partition . name , message ) )
2016-08-17 23:08:22 +08:00
if __name__ == ' __main__ ' :
try :
main ( )
except InputError as e :
2017-05-12 12:25:41 +10:00
print ( e , file = sys . stderr )
2016-08-17 23:08:22 +08:00
sys . exit ( 2 )