mirror of
				https://github.com/espressif/esp-idf.git
				synced 2025-10-31 04:59:55 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			223 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			223 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #
 | |
| # Copyright 2021 Espressif Systems (Shanghai) CO LTD
 | |
| #
 | |
| # Licensed under the Apache License, Version 2.0 (the "License");
 | |
| # you may not use this file except in compliance with the License.
 | |
| # You may obtain a copy of the License at
 | |
| #
 | |
| #     http://www.apache.org/licenses/LICENSE-2.0
 | |
| #
 | |
| # Unless required by applicable law or agreed to in writing, software
 | |
| # distributed under the License is distributed on an "AS IS" BASIS,
 | |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| # See the License for the specific language governing permissions and
 | |
| # limitations under the License.
 | |
| #
 | |
| 
 | |
| import collections
 | |
| import fnmatch
 | |
| import os
 | |
| from enum import Enum
 | |
| from functools import total_ordering
 | |
| 
 | |
| from pyparsing import (Group, Literal, OneOrMore, ParseException, SkipTo, Suppress, White, Word, ZeroOrMore, alphas,
 | |
|                        nums, restOfLine)
 | |
| 
 | |
| 
 | |
| @total_ordering
 | |
| class Entity():
 | |
|     """
 | |
|     Definition of an entity which can be placed or excluded
 | |
|     from placement.
 | |
|     """
 | |
| 
 | |
|     ALL = '*'
 | |
| 
 | |
|     class Specificity(Enum):
 | |
|         NONE = 0
 | |
|         ARCHIVE = 1
 | |
|         OBJ = 2
 | |
|         SYMBOL = 3
 | |
| 
 | |
|     def __init__(self, archive=None, obj=None, symbol=None):
 | |
|         archive_spec = archive and archive != Entity.ALL
 | |
|         obj_spec = obj and obj != Entity.ALL
 | |
|         symbol_spec = symbol and symbol != Entity.ALL
 | |
| 
 | |
|         if not archive_spec and not obj_spec and not symbol_spec:
 | |
|             self.specificity = Entity.Specificity.NONE
 | |
|         elif archive_spec and not obj_spec and not symbol_spec:
 | |
|             self.specificity = Entity.Specificity.ARCHIVE
 | |
|         elif archive_spec and obj_spec and not symbol_spec:
 | |
|             self.specificity = Entity.Specificity.OBJ
 | |
|         elif archive_spec and obj_spec and symbol_spec:
 | |
|             self.specificity = Entity.Specificity.SYMBOL
 | |
|         else:
 | |
|             raise ValueError("Invalid arguments '(%s, %s, %s)'" % (archive, obj, symbol))
 | |
| 
 | |
|         self.archive  = archive
 | |
|         self.obj = obj
 | |
|         self.symbol = symbol
 | |
| 
 | |
|     def __eq__(self, other):
 | |
|         return (self.specificity.value == other.specificity.value and
 | |
|                 self.archive == other.archive and
 | |
|                 self.obj == other.obj and
 | |
|                 self.symbol == other.symbol)
 | |
| 
 | |
|     def __lt__(self, other):
 | |
|         res = False
 | |
|         if self.specificity.value < other.specificity.value:
 | |
|             res = True
 | |
|         elif self.specificity == other.specificity:
 | |
|             for s in Entity.Specificity:
 | |
|                 a = self[s] if self[s] else ''
 | |
|                 b = other[s] if other[s] else ''
 | |
| 
 | |
|                 if a != b:
 | |
|                     res = a < b
 | |
|                     break
 | |
|         else:
 | |
|             res = False
 | |
|         return res
 | |
| 
 | |
|     def __hash__(self):
 | |
|         return hash(self.__repr__())
 | |
| 
 | |
|     def __str__(self):
 | |
|         return '%s:%s %s' % self.__repr__()
 | |
| 
 | |
|     def __repr__(self):
 | |
|         return (self.archive, self.obj, self.symbol)
 | |
| 
 | |
|     def __getitem__(self, spec):
 | |
|         res = None
 | |
|         if spec == Entity.Specificity.ARCHIVE:
 | |
|             res = self.archive
 | |
|         elif spec == Entity.Specificity.OBJ:
 | |
|             res = self.obj
 | |
|         elif spec == Entity.Specificity.SYMBOL:
 | |
|             res = self.symbol
 | |
|         else:
 | |
|             res = None
 | |
|         return res
 | |
| 
 | |
| 
 | |
| class EntityDB():
 | |
|     """
 | |
|     Encapsulates an output of objdump. Contains information about the static library sections
 | |
|     and names
 | |
|     """
 | |
| 
 | |
|     __info = collections.namedtuple('__info', 'filename content')
 | |
| 
 | |
|     def __init__(self):
 | |
|         self.sections = dict()
 | |
| 
 | |
|     def add_sections_info(self, sections_info_dump):
 | |
|         first_line = sections_info_dump.readline()
 | |
| 
 | |
|         archive_path = (Literal('In archive').suppress() +
 | |
|                         White().suppress() +
 | |
|                         # trim the colon and line ending characters from archive_path
 | |
|                         restOfLine.setResultsName('archive_path').setParseAction(lambda s, loc, toks: s.rstrip(':\n\r ')))
 | |
|         parser = archive_path
 | |
| 
 | |
|         results = None
 | |
| 
 | |
|         try:
 | |
|             results = parser.parseString(first_line, parseAll=True)
 | |
|         except ParseException as p:
 | |
|             raise ParseException('Parsing sections info for library ' + sections_info_dump.name + ' failed. ' + p.msg)
 | |
| 
 | |
|         archive = os.path.basename(results.archive_path)
 | |
|         self.sections[archive] = EntityDB.__info(sections_info_dump.name, sections_info_dump.read())
 | |
| 
 | |
|     def _get_infos_from_file(self, info):
 | |
|         # {object}:  file format elf32-xtensa-le
 | |
|         object_line = SkipTo(':').setResultsName('object') + Suppress(restOfLine)
 | |
| 
 | |
|         # Sections:
 | |
|         # Idx Name ...
 | |
|         section_start = Suppress(Literal('Sections:'))
 | |
|         section_header = Suppress(OneOrMore(Word(alphas)))
 | |
| 
 | |
|         # 00 {section} 0000000 ...
 | |
|         #              CONTENTS, ALLOC, ....
 | |
|         section_entry = Suppress(Word(nums)) + SkipTo(' ') + Suppress(restOfLine) + \
 | |
|             Suppress(ZeroOrMore(Word(alphas) + Literal(',')) + Word(alphas))
 | |
| 
 | |
|         content = Group(object_line + section_start + section_header + Group(OneOrMore(section_entry)).setResultsName('sections'))
 | |
|         parser = Group(ZeroOrMore(content)).setResultsName('contents')
 | |
| 
 | |
|         results = None
 | |
| 
 | |
|         try:
 | |
|             results = parser.parseString(info.content, parseAll=True)
 | |
|         except ParseException as p:
 | |
|             raise ParseException('Unable to parse section info file ' + info.filename + '. ' + p.msg)
 | |
| 
 | |
|         return results
 | |
| 
 | |
|     def _process_archive(self, archive):
 | |
|         stored = self.sections[archive]
 | |
| 
 | |
|         # Parse the contents of the sections file on-demand,
 | |
|         # save the result for later
 | |
|         if not isinstance(stored, dict):
 | |
|             parsed = self._get_infos_from_file(stored)
 | |
|             stored = dict()
 | |
|             for content in parsed.contents:
 | |
|                 sections = list(map(lambda s: s, content.sections))
 | |
|                 stored[content.object] = sections
 | |
|             self.sections[archive] = stored
 | |
| 
 | |
|     def get_archives(self):
 | |
|         return self.sections.keys()
 | |
| 
 | |
|     def get_objects(self, archive):
 | |
|         try:
 | |
|             self._process_archive(archive)
 | |
|         except KeyError:
 | |
|             return []
 | |
| 
 | |
|         return self.sections[archive].keys()
 | |
| 
 | |
|     def _match_obj(self, archive, obj):
 | |
|         objs = self.get_objects(archive)
 | |
|         match_objs = fnmatch.filter(objs, obj + '.o') + fnmatch.filter(objs, obj + '.*.obj') + fnmatch.filter(objs, obj + '.obj')
 | |
| 
 | |
|         if len(match_objs) > 1:
 | |
|             raise ValueError("Multiple matches for object: '%s: %s': %s" % (archive, obj, str(match_objs)))
 | |
| 
 | |
|         try:
 | |
|             return match_objs[0]
 | |
|         except IndexError:
 | |
|             return None
 | |
| 
 | |
|     def get_sections(self, archive, obj):
 | |
|         obj = self._match_obj(archive, obj)
 | |
|         res = []
 | |
|         if obj:
 | |
|             res = self.sections[archive][obj]
 | |
|         return res
 | |
| 
 | |
|     def _match_symbol(self, archive, obj, symbol):
 | |
|         sections = self.get_sections(archive, obj)
 | |
|         return [s for s in sections if s.endswith(symbol)]
 | |
| 
 | |
|     def check_exists(self, entity):
 | |
|         res = True
 | |
| 
 | |
|         if entity.specificity != Entity.Specificity.NONE:
 | |
|             if entity.specificity == Entity.Specificity.ARCHIVE:
 | |
|                 res = entity.archive in self.get_archives()
 | |
|             elif entity.specificity == Entity.Specificity.OBJ:
 | |
|                 res = self._match_obj(entity.archive, entity.obj) is not None
 | |
|             elif entity.specificity == Entity.Specificity.SYMBOL:
 | |
|                 res = len(self._match_symbol(entity.archive, entity.obj, entity.symbol)) > 0
 | |
|             else:
 | |
|                 res = False
 | |
| 
 | |
|         return res
 | 
