| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | #!/usr/bin/env python | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # ESP32 efuse table generation tool | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Converts efuse table to header file efuse_table.h. | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2022-01-18 10:32:56 +08:00
										 |  |  | # SPDX-FileCopyrightText: 2017-2022 Espressif Systems (Shanghai) CO LTD | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | # | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  | # SPDX-License-Identifier: Apache-2.0 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  | from __future__ import division, print_function | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | import argparse | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  | import hashlib | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | import os | 
					
						
							|  |  |  | import re | 
					
						
							|  |  |  | import sys | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  | from datetime import datetime | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | __version__ = '1.0' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | quiet = False | 
					
						
							| 
									
										
										
										
											2018-12-18 22:42:10 +08:00
										 |  |  | max_blk_len = 256 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  | idf_target = 'esp32' | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | def get_copyright(): | 
					
						
							|  |  |  |     copyright_str = '''/*
 | 
					
						
							|  |  |  |  * SPDX-FileCopyrightText: 2017-%d Espressif Systems (Shanghai) CO LTD | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * SPDX-License-Identifier: Apache-2.0 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | '''
 | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |     return copyright_str % datetime.today().year | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | def status(msg): | 
					
						
							|  |  |  |     """ Print status message to stderr """ | 
					
						
							|  |  |  |     if not quiet: | 
					
						
							|  |  |  |         critical(msg) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | def critical(msg): | 
					
						
							|  |  |  |     """ Print critical message to stderr """ | 
					
						
							|  |  |  |     sys.stderr.write(msg) | 
					
						
							|  |  |  |     sys.stderr.write('\n') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | class FuseTable(list): | 
					
						
							|  |  |  |     def __init__(self): | 
					
						
							|  |  |  |         super(FuseTable, self).__init__(self) | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         self.md5_digest_table = '' | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  |     def from_csv(cls, csv_contents): | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         res = FuseTable() | 
					
						
							|  |  |  |         lines = csv_contents.splitlines() | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         def expand_vars(f): | 
					
						
							|  |  |  |             f = os.path.expandvars(f) | 
					
						
							|  |  |  |             m = re.match(r'(?<!\\)\$([A-Za-z_][A-Za-z0-9_]*)', f) | 
					
						
							|  |  |  |             if m: | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  |                 raise InputError("unknown variable '%s'" % (m.group(1))) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |             return f | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for line_no in range(len(lines)): | 
					
						
							|  |  |  |             line = expand_vars(lines[line_no]).strip() | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             if line.startswith('#') or len(line) == 0: | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                 continue | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 res.append(FuseDefinition.from_csv(line)) | 
					
						
							|  |  |  |             except InputError as e: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                 raise InputError('Error at line %d: %s' % (line_no + 1, e)) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |             except Exception: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                 critical('Unexpected error parsing line %d: %s' % (line_no + 1, line)) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                 raise | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # fix up missing bit_start | 
					
						
							|  |  |  |         last_efuse_block = None | 
					
						
							| 
									
										
										
										
											2021-05-26 10:44:20 +02:00
										 |  |  |         for i in res: | 
					
						
							|  |  |  |             if last_efuse_block != i.efuse_block: | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                 last_end = 0 | 
					
						
							| 
									
										
										
										
											2021-05-26 10:44:20 +02:00
										 |  |  |             if i.bit_start is None: | 
					
						
							|  |  |  |                 i.bit_start = last_end | 
					
						
							|  |  |  |             last_end = i.bit_start + i.bit_count | 
					
						
							|  |  |  |             last_efuse_block = i.efuse_block | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         res.verify_duplicate_name() | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         # fix up missing field_name | 
					
						
							|  |  |  |         last_field = None | 
					
						
							| 
									
										
										
										
											2021-05-26 10:44:20 +02:00
										 |  |  |         for i in res: | 
					
						
							|  |  |  |             if i.field_name == '' and last_field is None: | 
					
						
							|  |  |  |                 raise InputError('Error at line %d: %s missing field name' % (line_no + 1, i)) | 
					
						
							|  |  |  |             elif i.field_name == '' and last_field is not None: | 
					
						
							|  |  |  |                 i.field_name = last_field.field_name | 
					
						
							|  |  |  |             last_field = i | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         # fill group | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  |         names = [p.field_name for p in res] | 
					
						
							|  |  |  |         duplicates = set(n for n in names if names.count(n) > 1) | 
					
						
							| 
									
										
										
											
												efuse_table_gen: Fixes wrong joining fields with omitted names
The issue is related to the non-sequential way of description when
such fields going together sequential.
Related to esp32h2 chip for eFuses: MAC_FACTORY and MAC_EXT.
The issue is in wrong indexes of MAC_EXT.
MAC_EXT got indexes like it is joined to MAC_FACTORY.
const esp_efuse_desc_t* ESP_EFUSE_MAC_FACTORY[] = {
    &MAC_FACTORY[0],
    &MAC_FACTORY[1],
    &MAC_FACTORY[2],
    &MAC_FACTORY[3],
    &MAC_FACTORY[4],
    &MAC_FACTORY[5],
    NULL
};
const esp_efuse_desc_t* ESP_EFUSE_MAC_EXT[] = {
    &MAC_EXT[6],
    &MAC_EXT[7],
    NULL
};
This commit fixed it to:
const esp_efuse_desc_t* ESP_EFUSE_MAC_EXT[] = {
    &MAC_EXT[0],
    &MAC_EXT[1],
    NULL
};
											
										 
											2021-10-28 20:15:24 +08:00
										 |  |  |         for dname in duplicates: | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |             i_count = 0 | 
					
						
							|  |  |  |             for p in res: | 
					
						
							| 
									
										
										
											
												efuse_table_gen: Fixes wrong joining fields with omitted names
The issue is related to the non-sequential way of description when
such fields going together sequential.
Related to esp32h2 chip for eFuses: MAC_FACTORY and MAC_EXT.
The issue is in wrong indexes of MAC_EXT.
MAC_EXT got indexes like it is joined to MAC_FACTORY.
const esp_efuse_desc_t* ESP_EFUSE_MAC_FACTORY[] = {
    &MAC_FACTORY[0],
    &MAC_FACTORY[1],
    &MAC_FACTORY[2],
    &MAC_FACTORY[3],
    &MAC_FACTORY[4],
    &MAC_FACTORY[5],
    NULL
};
const esp_efuse_desc_t* ESP_EFUSE_MAC_EXT[] = {
    &MAC_EXT[6],
    &MAC_EXT[7],
    NULL
};
This commit fixed it to:
const esp_efuse_desc_t* ESP_EFUSE_MAC_EXT[] = {
    &MAC_EXT[0],
    &MAC_EXT[1],
    NULL
};
											
										 
											2021-10-28 20:15:24 +08:00
										 |  |  |                 if p.field_name != dname: | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                 if len(duplicates.intersection([p.field_name])) != 0: | 
					
						
							|  |  |  |                     p.group = str(i_count) | 
					
						
							|  |  |  |                     i_count += 1 | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     i_count = 0 | 
					
						
							|  |  |  |         res.verify_duplicate_name() | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |         # clac md5 for table | 
					
						
							|  |  |  |         res.calc_md5() | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         return res | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def verify_duplicate_name(self): | 
					
						
							|  |  |  |         # check on duplicate name | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  |         names = [p.field_name for p in self] | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |         names += [name.replace('.', '_') for name in names if '.' in name] | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  |         duplicates = set(n for n in names if names.count(n) > 1) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         # print sorted duplicate partitions by name | 
					
						
							|  |  |  |         if len(duplicates) != 0: | 
					
						
							|  |  |  |             fl_error = False | 
					
						
							|  |  |  |             for p in self: | 
					
						
							|  |  |  |                 field_name = p.field_name + p.group | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                 if field_name != '' and len(duplicates.intersection([field_name])) != 0: | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                     fl_error = True | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                     print('Field at %s, %s, %s, %s have dublicate field_name' % | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  |                           (p.field_name, p.efuse_block, p.bit_start, p.bit_count)) | 
					
						
							|  |  |  |             if fl_error is True: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                 raise InputError('Field names must be unique') | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |     def check_struct_field_name(self): | 
					
						
							|  |  |  |         # check that stuctured fields have a root field | 
					
						
							|  |  |  |         for p in self: | 
					
						
							|  |  |  |             if '.' in p.field_name: | 
					
						
							|  |  |  |                 name = '' | 
					
						
							|  |  |  |                 for sub in p.field_name.split('.')[:-1]: | 
					
						
							|  |  |  |                     name = sub if name == '' else name + '.' + sub | 
					
						
							|  |  |  |                     missed_name = True | 
					
						
							|  |  |  |                     for d in self: | 
					
						
							|  |  |  |                         if p is not d and p.efuse_block == d.efuse_block and name == d.field_name: | 
					
						
							|  |  |  |                             missed_name = False | 
					
						
							|  |  |  |                     if missed_name: | 
					
						
							|  |  |  |                         raise InputError('%s is not found' % name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  |     def verify(self, type_table=None): | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |         def check(p, n): | 
					
						
							|  |  |  |             left = n.bit_start | 
					
						
							|  |  |  |             right = n.bit_start + n.bit_count - 1 | 
					
						
							|  |  |  |             start = p.bit_start | 
					
						
							|  |  |  |             end = p.bit_start + p.bit_count - 1 | 
					
						
							|  |  |  |             if left <= start <= right: | 
					
						
							|  |  |  |                 if left <= end <= right: | 
					
						
							|  |  |  |                     return 'included in'  # [n  [p...p]  n] | 
					
						
							|  |  |  |                 return 'intersected with'  # [n  [p..n]..p] | 
					
						
							|  |  |  |             if left <= end <= right: | 
					
						
							|  |  |  |                 return 'intersected with'  # [p..[n..p] n] | 
					
						
							|  |  |  |             if start <= left and right <= end: | 
					
						
							|  |  |  |                 return 'wraps'  # [p  [n...n]  p] | 
					
						
							|  |  |  |             return 'ok'  # [p] [n]  or  [n] [p] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def print_error(p, n, state): | 
					
						
							|  |  |  |             raise InputError('Field at %s, %s, %s, %s  %s  %s, %s, %s, %s' % | 
					
						
							|  |  |  |                              (p.field_name, p.efuse_block, p.bit_start, p.bit_count, state, | 
					
						
							|  |  |  |                               n.field_name, n.efuse_block, n.bit_start, n.bit_count)) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         for p in self: | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |             p.verify(type_table) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.verify_duplicate_name() | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |         if type_table != 'custom_table': | 
					
						
							|  |  |  |             # check will be done for common and custom tables together | 
					
						
							|  |  |  |             self.check_struct_field_name() | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         # check for overlaps | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |         for p in self: | 
					
						
							|  |  |  |             for n in self: | 
					
						
							|  |  |  |                 if p is not n and p.efuse_block == n.efuse_block: | 
					
						
							|  |  |  |                     state = check(p, n) | 
					
						
							|  |  |  |                     if state != 'ok': | 
					
						
							|  |  |  |                         if '.' in p.field_name: | 
					
						
							|  |  |  |                             name = '' | 
					
						
							|  |  |  |                             for sub in p.field_name.split('.'): | 
					
						
							|  |  |  |                                 name = sub if name == '' else name + '.' + sub | 
					
						
							|  |  |  |                                 for d in self: | 
					
						
							|  |  |  |                                     if p is not d and p.efuse_block == d.efuse_block and name == d.field_name: | 
					
						
							|  |  |  |                                         state = check(p, d) | 
					
						
							|  |  |  |                                         if state == 'included in': | 
					
						
							|  |  |  |                                             break | 
					
						
							|  |  |  |                                         elif state != 'intersected with': | 
					
						
							|  |  |  |                                             state = 'out of range' | 
					
						
							|  |  |  |                                         print_error(p, d, state) | 
					
						
							|  |  |  |                             continue | 
					
						
							|  |  |  |                         elif '.' in n.field_name: | 
					
						
							|  |  |  |                             continue | 
					
						
							|  |  |  |                         print_error(p, n, state) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |     def calc_md5(self): | 
					
						
							|  |  |  |         txt_table = '' | 
					
						
							|  |  |  |         for p in self: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             txt_table += '%s %s %d %s %s' % (p.field_name, p.efuse_block, p.bit_start, str(p.get_bit_count()), p.comment) + '\n' | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  |         self.md5_digest_table = hashlib.md5(txt_table.encode('utf-8')).hexdigest() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def show_range_used_bits(self): | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         # print used and free bits | 
					
						
							|  |  |  |         rows = '' | 
					
						
							|  |  |  |         rows += 'Sorted efuse table:\n' | 
					
						
							|  |  |  |         num = 1 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         rows += '{0} \t{1:<30} \t{2} \t{3} \t{4}'.format('#', 'field_name', 'efuse_block', 'bit_start', 'bit_count') + '\n' | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         for p in sorted(self, key=lambda x:(x.efuse_block, x.bit_start)): | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             rows += '{0} \t{1:<30} \t{2} \t{3:^8} \t{4:^8}'.format(num, p.field_name, p.efuse_block, p.bit_start, p.bit_count) + '\n' | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |             num += 1 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         rows += '\nUsed bits in efuse table:\n' | 
					
						
							|  |  |  |         last = None | 
					
						
							|  |  |  |         for p in sorted(self, key=lambda x:(x.efuse_block, x.bit_start)): | 
					
						
							|  |  |  |             if last is None: | 
					
						
							|  |  |  |                 rows += '%s \n[%d ' % (p.efuse_block, p.bit_start) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  |             if last is not None: | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                 if last.efuse_block != p.efuse_block: | 
					
						
							|  |  |  |                     rows += '%d] \n\n%s \n[%d ' % (last.bit_start + last.bit_count - 1, p.efuse_block, p.bit_start) | 
					
						
							|  |  |  |                 elif last.bit_start + last.bit_count != p.bit_start: | 
					
						
							|  |  |  |                     rows += '%d] [%d ' % (last.bit_start + last.bit_count - 1, p.bit_start) | 
					
						
							|  |  |  |             last = p | 
					
						
							|  |  |  |         rows += '%d] \n' % (last.bit_start + last.bit_count - 1) | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  |         rows += '\nNote: Not printed ranges are free for using. (bits in EFUSE_BLK0 are reserved for Espressif)\n' | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         return rows | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  |     def get_str_position_last_free_bit_in_blk(self, blk): | 
					
						
							|  |  |  |         last_used_bit = 0 | 
					
						
							|  |  |  |         for p in self: | 
					
						
							|  |  |  |             if p.efuse_block == blk: | 
					
						
							|  |  |  |                 if p.define is not None: | 
					
						
							|  |  |  |                     return p.get_bit_count() | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     if last_used_bit < p.bit_start + p.bit_count: | 
					
						
							|  |  |  |                         last_used_bit = p.bit_start + p.bit_count | 
					
						
							|  |  |  |         if last_used_bit == 0: | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  |         return str(last_used_bit) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     def to_header(self, file_name): | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |         rows = [get_copyright()] | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         rows += ['#ifdef __cplusplus', | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                  'extern "C" {', | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                  '#endif', | 
					
						
							|  |  |  |                  '', | 
					
						
							| 
									
										
										
										
											2022-08-01 21:19:38 +08:00
										 |  |  |                  '#include "esp_efuse.h"', | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                  '', | 
					
						
							|  |  |  |                  '// md5_digest_table ' + self.md5_digest_table, | 
					
						
							|  |  |  |                  '// This file was generated from the file ' + file_name + '.csv. DO NOT CHANGE THIS FILE MANUALLY.', | 
					
						
							|  |  |  |                  '// If you want to change some fields, you need to change ' + file_name + '.csv file', | 
					
						
							|  |  |  |                  '// then run `efuse_common_table` or `efuse_custom_table` command it will generate this file.', | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  |                  "// To show efuse_table run the command 'show_efuse_table'.", | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                  '', | 
					
						
							|  |  |  |                  ''] | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         last_field_name = '' | 
					
						
							|  |  |  |         for p in self: | 
					
						
							|  |  |  |             if (p.field_name != last_field_name): | 
					
						
							| 
									
										
										
										
											2023-02-15 16:51:51 +08:00
										 |  |  |                 name = 'ESP_EFUSE_' + p.field_name.replace('.', '_') | 
					
						
							|  |  |  |                 rows += ['extern const esp_efuse_desc_t* ' + name + '[];'] | 
					
						
							|  |  |  |                 for alt_name in p.get_alt_names(): | 
					
						
							|  |  |  |                     alt_name = 'ESP_EFUSE_' + alt_name.replace('.', '_') | 
					
						
							|  |  |  |                     rows += ['#define ' + alt_name + ' ' + name] | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                 last_field_name = p.field_name | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         rows += ['', | 
					
						
							|  |  |  |                  '#ifdef __cplusplus', | 
					
						
							|  |  |  |                  '}', | 
					
						
							|  |  |  |                  '#endif', | 
					
						
							|  |  |  |                  ''] | 
					
						
							| 
									
										
										
										
											2020-06-24 10:52:51 +08:00
										 |  |  |         return '\n'.join(rows) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def to_c_file(self, file_name, debug): | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |         rows = [get_copyright()] | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  |         rows += ['#include "sdkconfig.h"', | 
					
						
							|  |  |  |                  '#include "esp_efuse.h"', | 
					
						
							|  |  |  |                  '#include <assert.h>', | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                  '#include "' + file_name + '.h"', | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                  '', | 
					
						
							|  |  |  |                  '// md5_digest_table ' + self.md5_digest_table, | 
					
						
							|  |  |  |                  '// This file was generated from the file ' + file_name + '.csv. DO NOT CHANGE THIS FILE MANUALLY.', | 
					
						
							|  |  |  |                  '// If you want to change some fields, you need to change ' + file_name + '.csv file', | 
					
						
							|  |  |  |                  '// then run `efuse_common_table` or `efuse_custom_table` command it will generate this file.', | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  |                  "// To show efuse_table run the command 'show_efuse_table'."] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         rows += [''] | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         if idf_target == 'esp32': | 
					
						
							|  |  |  |             rows += ['#define MAX_BLK_LEN CONFIG_EFUSE_MAX_BLK_LEN'] | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             rows += [''] | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             last_free_bit_blk1 = self.get_str_position_last_free_bit_in_blk('EFUSE_BLK1') | 
					
						
							|  |  |  |             last_free_bit_blk2 = self.get_str_position_last_free_bit_in_blk('EFUSE_BLK2') | 
					
						
							|  |  |  |             last_free_bit_blk3 = self.get_str_position_last_free_bit_in_blk('EFUSE_BLK3') | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             rows += ['// The last free bit in the block is counted over the entire file.'] | 
					
						
							| 
									
										
										
										
											2019-06-18 19:34:05 +08:00
										 |  |  |             if last_free_bit_blk1 is not None: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                 rows += ['#define LAST_FREE_BIT_BLK1 ' + last_free_bit_blk1] | 
					
						
							| 
									
										
										
										
											2019-06-18 19:34:05 +08:00
										 |  |  |             if last_free_bit_blk2 is not None: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                 rows += ['#define LAST_FREE_BIT_BLK2 ' + last_free_bit_blk2] | 
					
						
							| 
									
										
										
										
											2019-06-18 19:34:05 +08:00
										 |  |  |             if last_free_bit_blk3 is not None: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                 rows += ['#define LAST_FREE_BIT_BLK3 ' + last_free_bit_blk3] | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             rows += [''] | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-18 19:34:05 +08:00
										 |  |  |             if last_free_bit_blk1 is not None: | 
					
						
							|  |  |  |                 rows += ['_Static_assert(LAST_FREE_BIT_BLK1 <= MAX_BLK_LEN, "The eFuse table does not match the coding scheme. ' | 
					
						
							|  |  |  |                          'Edit the table and restart the efuse_common_table or efuse_custom_table command to regenerate the new files.");'] | 
					
						
							|  |  |  |             if last_free_bit_blk2 is not None: | 
					
						
							|  |  |  |                 rows += ['_Static_assert(LAST_FREE_BIT_BLK2 <= MAX_BLK_LEN, "The eFuse table does not match the coding scheme. ' | 
					
						
							|  |  |  |                          'Edit the table and restart the efuse_common_table or efuse_custom_table command to regenerate the new files.");'] | 
					
						
							|  |  |  |             if last_free_bit_blk3 is not None: | 
					
						
							|  |  |  |                 rows += ['_Static_assert(LAST_FREE_BIT_BLK3 <= MAX_BLK_LEN, "The eFuse table does not match the coding scheme. ' | 
					
						
							|  |  |  |                          'Edit the table and restart the efuse_common_table or efuse_custom_table command to regenerate the new files.");'] | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             rows += [''] | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         last_name = '' | 
					
						
							|  |  |  |         for p in self: | 
					
						
							|  |  |  |             if (p.field_name != last_name): | 
					
						
							|  |  |  |                 if last_name != '': | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                     rows += ['};\n'] | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |                 rows += ['static const esp_efuse_desc_t ' + p.field_name.replace('.', '_') + '[] = {'] | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                 last_name = p.field_name | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             rows += [p.to_struct(debug) + ','] | 
					
						
							|  |  |  |         rows += ['};\n'] | 
					
						
							|  |  |  |         rows += ['\n\n\n'] | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         last_name = '' | 
					
						
							|  |  |  |         for p in self: | 
					
						
							|  |  |  |             if (p.field_name != last_name): | 
					
						
							|  |  |  |                 if last_name != '': | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                     rows += ['    NULL', | 
					
						
							|  |  |  |                              '};\n'] | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |                 rows += ['const esp_efuse_desc_t* ' + 'ESP_EFUSE_' + p.field_name.replace('.', '_') + '[] = {'] | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |             last_name = p.field_name | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             index = str(0) if str(p.group) == '' else str(p.group) | 
					
						
							| 
									
										
										
											
												efuse: Adds support structure of efuses in efuse_table
Supported a new format of efuse description using '.' in the name.
It means that RD_DIS.KEYx belongs to the range of the RD_DIS name.
RD_DIS,                           EFUSE_BLK0,   32,    7,     Read protection
    RD_DIS.KEY0,                  EFUSE_BLK0,   32,    1,     Read protection for EFUSE_BLK4.
    RD_DIS.KEY1,                  EFUSE_BLK0,   33,    1,     Read protection for EFUSE_BLK5.
											
										 
											2021-05-21 16:21:03 +08:00
										 |  |  |             rows += ['    &' + p.field_name.replace('.', '_') + '[' + index + '],    \t\t// ' + p.comment] | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         rows += ['    NULL', | 
					
						
							|  |  |  |                  '};\n'] | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-24 10:52:51 +08:00
										 |  |  |         return '\n'.join(rows) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class FuseDefinition(object): | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     def __init__(self): | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         self.field_name = '' | 
					
						
							|  |  |  |         self.group = '' | 
					
						
							|  |  |  |         self.efuse_block = '' | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         self.bit_start = None | 
					
						
							|  |  |  |         self.bit_count = None | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  |         self.define = None | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         self.comment = '' | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def from_csv(cls, line): | 
					
						
							|  |  |  |         """ Parse a line from the CSV """ | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         line_w_defaults = line + ',,,,'  # lazy way to support default fields | 
					
						
							|  |  |  |         fields = [f.strip() for f in line_w_defaults.split(',')] | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         res = FuseDefinition() | 
					
						
							|  |  |  |         res.field_name = fields[0] | 
					
						
							|  |  |  |         res.efuse_block = res.parse_block(fields[1]) | 
					
						
							|  |  |  |         res.bit_start = res.parse_num(fields[2]) | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  |         res.bit_count = res.parse_bit_count(fields[3]) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         if res.bit_count is None or res.bit_count == 0: | 
					
						
							|  |  |  |             raise InputError("Field bit_count can't be empty") | 
					
						
							|  |  |  |         res.comment = fields[4] | 
					
						
							|  |  |  |         return res | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def parse_num(self, strval): | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         if strval == '': | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |             return None  # Field will fill in default | 
					
						
							|  |  |  |         return self.parse_int(strval) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  |     def parse_bit_count(self, strval): | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         if strval == 'MAX_BLK_LEN': | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  |             self.define = strval | 
					
						
							|  |  |  |             return self.get_max_bits_of_block() | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return self.parse_num(strval) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     def parse_int(self, v): | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return int(v, 0) | 
					
						
							|  |  |  |         except ValueError: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             raise InputError('Invalid field value %s' % v) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     def parse_block(self, strval): | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         if strval == '': | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |             raise InputError("Field 'efuse_block' can't be left empty.") | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         if idf_target == 'esp32': | 
					
						
							|  |  |  |             if strval not in ['EFUSE_BLK0', 'EFUSE_BLK1', 'EFUSE_BLK2', 'EFUSE_BLK3']: | 
					
						
							| 
									
										
										
										
											2019-06-18 19:34:05 +08:00
										 |  |  |                 raise InputError("Field 'efuse_block' should be one of EFUSE_BLK0..EFUSE_BLK3") | 
					
						
							| 
									
										
										
										
											2020-09-17 17:04:01 +08:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             if strval not in ['EFUSE_BLK0', 'EFUSE_BLK1', 'EFUSE_BLK2', 'EFUSE_BLK3', 'EFUSE_BLK4', | 
					
						
							|  |  |  |                               'EFUSE_BLK5', 'EFUSE_BLK6', 'EFUSE_BLK7', 'EFUSE_BLK8', 'EFUSE_BLK9', | 
					
						
							|  |  |  |                               'EFUSE_BLK10']: | 
					
						
							| 
									
										
										
										
											2019-06-18 19:34:05 +08:00
										 |  |  |                 raise InputError("Field 'efuse_block' should be one of EFUSE_BLK0..EFUSE_BLK10") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         return strval | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |     def get_max_bits_of_block(self): | 
					
						
							|  |  |  |         '''common_table: EFUSE_BLK0, EFUSE_BLK1, EFUSE_BLK2, EFUSE_BLK3
 | 
					
						
							|  |  |  |            custom_table: ----------, ----------, ----------, EFUSE_BLK3(some reserved in common_table) | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         if self.efuse_block == 'EFUSE_BLK0': | 
					
						
							| 
									
										
										
										
											2018-12-18 22:42:10 +08:00
										 |  |  |             return 256 | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2018-12-18 22:42:10 +08:00
										 |  |  |             return max_blk_len | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |     def verify(self, type_table): | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         if self.efuse_block is None: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             raise ValidationError(self, 'efuse_block field is not set') | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         if self.bit_count is None: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             raise ValidationError(self, 'bit_count field is not set') | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |         max_bits = self.get_max_bits_of_block() | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         if self.bit_start + self.bit_count > max_bits: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             raise ValidationError(self, 'The field is outside the boundaries(max_bits = %d) of the %s block' % (max_bits, self.efuse_block)) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-12 15:50:31 +08:00
										 |  |  |     def get_bit_count(self, check_define=True): | 
					
						
							|  |  |  |         if check_define is True and self.define is not None: | 
					
						
							|  |  |  |             return self.define | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return self.bit_count | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     def to_struct(self, debug): | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         start = '    {' | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  |         if debug is True: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |             start = '    {' + '"' + self.field_name + '" ,' | 
					
						
							|  |  |  |         return ', '.join([start + self.efuse_block, | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |                          str(self.bit_start), | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |                          str(self.get_bit_count()) + '}, \t // ' + self.comment]) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-15 16:51:51 +08:00
										 |  |  |     def get_alt_names(self): | 
					
						
							| 
									
										
										
										
											2023-03-14 00:03:27 +08:00
										 |  |  |         result = re.search(r'^\[(.*?)\]', self.comment) | 
					
						
							| 
									
										
										
										
											2023-02-15 16:51:51 +08:00
										 |  |  |         if result: | 
					
						
							|  |  |  |             return result.group(1).split() | 
					
						
							|  |  |  |         return [] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  | def process_input_file(file, type_table): | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |     status('Parsing efuse CSV input file ' + file.name + ' ...') | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     input = file.read() | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  |     table = FuseTable.from_csv(input) | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |     status('Verifying efuse table...') | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |     table.verify(type_table) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     return table | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  | def ckeck_md5_in_file(md5, filename): | 
					
						
							|  |  |  |     if os.path.exists(filename): | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  |         with open(filename, 'r') as f: | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |             for line in f: | 
					
						
							|  |  |  |                 if md5 in line: | 
					
						
							|  |  |  |                     return True | 
					
						
							|  |  |  |     return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | def create_output_files(name, output_table, debug): | 
					
						
							|  |  |  |     file_name = os.path.splitext(os.path.basename(name))[0] | 
					
						
							|  |  |  |     gen_dir = os.path.dirname(name) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |     dir_for_file_h = gen_dir + '/include' | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     try: | 
					
						
							|  |  |  |         os.stat(dir_for_file_h) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  |     except Exception: | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |         os.mkdir(dir_for_file_h) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |     file_h_path = os.path.join(dir_for_file_h, file_name + '.h') | 
					
						
							|  |  |  |     file_c_path = os.path.join(gen_dir, file_name + '.c') | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |     # src files are the same | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  |     if ckeck_md5_in_file(output_table.md5_digest_table, file_c_path) is False: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         status('Creating efuse *.h file ' + file_h_path + ' ...') | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |         output = output_table.to_header(file_name) | 
					
						
							|  |  |  |         with open(file_h_path, 'w') as f: | 
					
						
							|  |  |  |             f.write(output) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         status('Creating efuse *.c file ' + file_c_path + ' ...') | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |         output = output_table.to_c_file(file_name, debug) | 
					
						
							|  |  |  |         with open(file_c_path, 'w') as f: | 
					
						
							|  |  |  |             f.write(output) | 
					
						
							|  |  |  |     else: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         print('Source files do not require updating correspond to csv file.') | 
					
						
							| 
									
										
										
										
											2018-12-06 17:49:02 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | def main(): | 
					
						
							|  |  |  |     global quiet | 
					
						
							| 
									
										
										
										
											2018-12-18 22:42:10 +08:00
										 |  |  |     global max_blk_len | 
					
						
							| 
									
										
										
										
											2019-06-18 19:34:05 +08:00
										 |  |  |     global idf_target | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     parser = argparse.ArgumentParser(description='ESP32 eFuse Manager') | 
					
						
							| 
									
										
										
										
											2021-11-06 17:23:21 +08:00
										 |  |  |     parser.add_argument('--idf_target', '-t', help='Target chip type', choices=['esp32', 'esp32s2', 'esp32s3', 'esp32c3', | 
					
						
							| 
									
										
										
										
											2023-07-27 15:10:50 +08:00
										 |  |  |                         'esp32c2', 'esp32c6', 'esp32h2', 'esp32p4'], default='esp32') | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     parser.add_argument('--quiet', '-q', help="Don't print non-critical status messages to stderr", action='store_true') | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |     parser.add_argument('--debug', help='Create header file with debug info', default=False, action='store_false') | 
					
						
							|  |  |  |     parser.add_argument('--info', help='Print info about range of used bits', default=False, action='store_true') | 
					
						
							| 
									
										
										
										
											2019-06-18 19:34:05 +08:00
										 |  |  |     parser.add_argument('--max_blk_len', help='Max number of bits in BLOCKs', type=int, default=256) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  |     parser.add_argument('common_input', help='Path to common CSV file to parse.', type=argparse.FileType('r')) | 
					
						
							|  |  |  |     parser.add_argument('custom_input', help='Path to custom CSV file to parse.', type=argparse.FileType('r'), nargs='?', default=None) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     args = parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-18 19:34:05 +08:00
										 |  |  |     idf_target = args.idf_target | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-18 22:42:10 +08:00
										 |  |  |     max_blk_len = args.max_blk_len | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |     print('Max number of bits in BLK %d' % (max_blk_len)) | 
					
						
							| 
									
										
										
										
											2018-12-18 22:42:10 +08:00
										 |  |  |     if max_blk_len not in [256, 192, 128]: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         raise InputError('Unsupported block length = %d' % (max_blk_len)) | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     quiet = args.quiet | 
					
						
							|  |  |  |     debug = args.debug | 
					
						
							|  |  |  |     info = args.info | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |     common_table = process_input_file(args.common_input, 'common_table') | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |     two_table = common_table | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     if args.custom_input is not None: | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         custom_table = process_input_file(args.custom_input, 'custom_table') | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |         two_table += custom_table | 
					
						
							|  |  |  |         two_table.verify() | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     # save files. | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  |     if info is False: | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |         if args.custom_input is None: | 
					
						
							|  |  |  |             create_output_files(args.common_input.name, common_table, debug) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             create_output_files(args.custom_input.name, custom_table, debug) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2018-11-21 16:09:36 +08:00
										 |  |  |         print(two_table.show_range_used_bits()) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  |     return 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | class InputError(RuntimeError): | 
					
						
							|  |  |  |     def __init__(self, e): | 
					
						
							|  |  |  |         super(InputError, self).__init__(e) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | class ValidationError(InputError): | 
					
						
							|  |  |  |     def __init__(self, p, message): | 
					
						
							| 
									
										
										
										
											2021-01-26 10:49:01 +08:00
										 |  |  |         super(ValidationError, self).__init__('Entry %s invalid: %s' % (p.field_name, message)) | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-07 20:32:58 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-19 11:46:21 +08:00
										 |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         main() | 
					
						
							|  |  |  |     except InputError as e: | 
					
						
							|  |  |  |         print(e, file=sys.stderr) | 
					
						
							|  |  |  |         sys.exit(2) |