mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-25 13:13:48 +01:00 
			
		
		
		
	merge
This commit is contained in:
		| @@ -174,7 +174,7 @@ class MemoryAnalyzer: | ||||
|                         self.sections[mapped_section].total_size += size | ||||
|  | ||||
|         except subprocess.CalledProcessError as e: | ||||
|             _LOGGER.error(f"Failed to parse sections: {e}") | ||||
|             _LOGGER.error("Failed to parse sections: %s", e) | ||||
|             raise | ||||
|  | ||||
|     def _parse_symbols(self) -> None: | ||||
| @@ -252,7 +252,7 @@ class MemoryAnalyzer: | ||||
|                         seen_addresses.add(address) | ||||
|  | ||||
|         except subprocess.CalledProcessError as e: | ||||
|             _LOGGER.error(f"Failed to parse symbols: {e}") | ||||
|             _LOGGER.error("Failed to parse symbols: %s", e) | ||||
|             raise | ||||
|  | ||||
|     def _categorize_symbols(self) -> None: | ||||
| @@ -399,8 +399,9 @@ class MemoryAnalyzer: | ||||
|                 # If batch fails, cache originals | ||||
|                 for symbol in symbols: | ||||
|                     self._demangle_cache[symbol] = symbol | ||||
|         except Exception: | ||||
|         except (subprocess.SubprocessError, OSError, UnicodeDecodeError) as e: | ||||
|             # On error, cache originals | ||||
|             _LOGGER.debug("Failed to batch demangle symbols: %s", e) | ||||
|             for symbol in symbols: | ||||
|                 self._demangle_cache[symbol] = symbol | ||||
|  | ||||
| @@ -424,267 +425,6 @@ class MemoryAnalyzer: | ||||
|  | ||||
|         return "Other Core" | ||||
|  | ||||
|     def generate_report(self, detailed: bool = False) -> str: | ||||
|         """Generate a formatted memory report.""" | ||||
|         components = sorted( | ||||
|             self.components.items(), key=lambda x: x[1].flash_total, reverse=True | ||||
|         ) | ||||
|  | ||||
|         # Calculate totals | ||||
|         total_flash = sum(c.flash_total for _, c in components) | ||||
|         total_ram = sum(c.ram_total for _, c in components) | ||||
|  | ||||
|         # Build report | ||||
|         lines = [] | ||||
|  | ||||
|         # Column width constants | ||||
|         COL_COMPONENT = 29 | ||||
|         COL_FLASH_TEXT = 14 | ||||
|         COL_FLASH_DATA = 14 | ||||
|         COL_RAM_DATA = 12 | ||||
|         COL_RAM_BSS = 12 | ||||
|         COL_TOTAL_FLASH = 15 | ||||
|         COL_TOTAL_RAM = 12 | ||||
|         COL_SEPARATOR = 3  # " | " | ||||
|  | ||||
|         # Core analysis column widths | ||||
|         COL_CORE_SUBCATEGORY = 30 | ||||
|         COL_CORE_SIZE = 12 | ||||
|         COL_CORE_COUNT = 6 | ||||
|         COL_CORE_PERCENT = 10 | ||||
|  | ||||
|         # Calculate the exact table width | ||||
|         table_width = ( | ||||
|             COL_COMPONENT | ||||
|             + COL_SEPARATOR | ||||
|             + COL_FLASH_TEXT | ||||
|             + COL_SEPARATOR | ||||
|             + COL_FLASH_DATA | ||||
|             + COL_SEPARATOR | ||||
|             + COL_RAM_DATA | ||||
|             + COL_SEPARATOR | ||||
|             + COL_RAM_BSS | ||||
|             + COL_SEPARATOR | ||||
|             + COL_TOTAL_FLASH | ||||
|             + COL_SEPARATOR | ||||
|             + COL_TOTAL_RAM | ||||
|         ) | ||||
|  | ||||
|         lines.append("=" * table_width) | ||||
|         lines.append("Component Memory Analysis".center(table_width)) | ||||
|         lines.append("=" * table_width) | ||||
|         lines.append("") | ||||
|  | ||||
|         # Main table - fixed column widths | ||||
|         lines.append( | ||||
|             f"{'Component':<{COL_COMPONENT}} | {'Flash (text)':>{COL_FLASH_TEXT}} | {'Flash (data)':>{COL_FLASH_DATA}} | {'RAM (data)':>{COL_RAM_DATA}} | {'RAM (bss)':>{COL_RAM_BSS}} | {'Total Flash':>{COL_TOTAL_FLASH}} | {'Total RAM':>{COL_TOTAL_RAM}}" | ||||
|         ) | ||||
|         lines.append( | ||||
|             "-" * COL_COMPONENT | ||||
|             + "-+-" | ||||
|             + "-" * COL_FLASH_TEXT | ||||
|             + "-+-" | ||||
|             + "-" * COL_FLASH_DATA | ||||
|             + "-+-" | ||||
|             + "-" * COL_RAM_DATA | ||||
|             + "-+-" | ||||
|             + "-" * COL_RAM_BSS | ||||
|             + "-+-" | ||||
|             + "-" * COL_TOTAL_FLASH | ||||
|             + "-+-" | ||||
|             + "-" * COL_TOTAL_RAM | ||||
|         ) | ||||
|  | ||||
|         for name, mem in components: | ||||
|             if mem.flash_total > 0 or mem.ram_total > 0: | ||||
|                 flash_rodata = mem.rodata_size + mem.data_size | ||||
|                 lines.append( | ||||
|                     f"{name:<{COL_COMPONENT}} | {mem.text_size:>{COL_FLASH_TEXT - 2},} B | {flash_rodata:>{COL_FLASH_DATA - 2},} B | " | ||||
|                     f"{mem.data_size:>{COL_RAM_DATA - 2},} B | {mem.bss_size:>{COL_RAM_BSS - 2},} B | " | ||||
|                     f"{mem.flash_total:>{COL_TOTAL_FLASH - 2},} B | {mem.ram_total:>{COL_TOTAL_RAM - 2},} B" | ||||
|                 ) | ||||
|  | ||||
|         lines.append( | ||||
|             "-" * COL_COMPONENT | ||||
|             + "-+-" | ||||
|             + "-" * COL_FLASH_TEXT | ||||
|             + "-+-" | ||||
|             + "-" * COL_FLASH_DATA | ||||
|             + "-+-" | ||||
|             + "-" * COL_RAM_DATA | ||||
|             + "-+-" | ||||
|             + "-" * COL_RAM_BSS | ||||
|             + "-+-" | ||||
|             + "-" * COL_TOTAL_FLASH | ||||
|             + "-+-" | ||||
|             + "-" * COL_TOTAL_RAM | ||||
|         ) | ||||
|         lines.append( | ||||
|             f"{'TOTAL':<{COL_COMPONENT}} | {' ':>{COL_FLASH_TEXT}} | {' ':>{COL_FLASH_DATA}} | " | ||||
|             f"{' ':>{COL_RAM_DATA}} | {' ':>{COL_RAM_BSS}} | " | ||||
|             f"{total_flash:>{COL_TOTAL_FLASH - 2},} B | {total_ram:>{COL_TOTAL_RAM - 2},} B" | ||||
|         ) | ||||
|  | ||||
|         # Top consumers | ||||
|         lines.append("") | ||||
|         lines.append("Top Flash Consumers:") | ||||
|         for i, (name, mem) in enumerate(components[:25]): | ||||
|             if mem.flash_total > 0: | ||||
|                 percentage = ( | ||||
|                     (mem.flash_total / total_flash * 100) if total_flash > 0 else 0 | ||||
|                 ) | ||||
|                 lines.append( | ||||
|                     f"{i + 1}. {name} ({mem.flash_total:,} B) - {percentage:.1f}% of analyzed flash" | ||||
|                 ) | ||||
|  | ||||
|         lines.append("") | ||||
|         lines.append("Top RAM Consumers:") | ||||
|         ram_components = sorted(components, key=lambda x: x[1].ram_total, reverse=True) | ||||
|         for i, (name, mem) in enumerate(ram_components[:25]): | ||||
|             if mem.ram_total > 0: | ||||
|                 percentage = (mem.ram_total / total_ram * 100) if total_ram > 0 else 0 | ||||
|                 lines.append( | ||||
|                     f"{i + 1}. {name} ({mem.ram_total:,} B) - {percentage:.1f}% of analyzed RAM" | ||||
|                 ) | ||||
|  | ||||
|         lines.append("") | ||||
|         lines.append( | ||||
|             "Note: This analysis covers symbols in the ELF file. Some runtime allocations may not be included." | ||||
|         ) | ||||
|         lines.append("=" * table_width) | ||||
|  | ||||
|         # Add ESPHome core detailed analysis if there are core symbols | ||||
|         if self._esphome_core_symbols: | ||||
|             lines.append("") | ||||
|             lines.append("=" * table_width) | ||||
|             lines.append("[esphome]core Detailed Analysis".center(table_width)) | ||||
|             lines.append("=" * table_width) | ||||
|             lines.append("") | ||||
|  | ||||
|             # Group core symbols by subcategory | ||||
|             core_subcategories: dict[str, list[tuple[str, str, int]]] = defaultdict( | ||||
|                 list | ||||
|             ) | ||||
|  | ||||
|             for symbol, demangled, size in self._esphome_core_symbols: | ||||
|                 # Categorize based on demangled name patterns | ||||
|                 subcategory = self._categorize_esphome_core_symbol(demangled) | ||||
|                 core_subcategories[subcategory].append((symbol, demangled, size)) | ||||
|  | ||||
|             # Sort subcategories by total size | ||||
|             sorted_subcategories = sorted( | ||||
|                 [ | ||||
|                     (name, symbols, sum(s[2] for s in symbols)) | ||||
|                     for name, symbols in core_subcategories.items() | ||||
|                 ], | ||||
|                 key=lambda x: x[2], | ||||
|                 reverse=True, | ||||
|             ) | ||||
|  | ||||
|             lines.append( | ||||
|                 f"{'Subcategory':<{COL_CORE_SUBCATEGORY}} | {'Size':>{COL_CORE_SIZE}} | " | ||||
|                 f"{'Count':>{COL_CORE_COUNT}} | {'% of Core':>{COL_CORE_PERCENT}}" | ||||
|             ) | ||||
|             lines.append( | ||||
|                 "-" * COL_CORE_SUBCATEGORY | ||||
|                 + "-+-" | ||||
|                 + "-" * COL_CORE_SIZE | ||||
|                 + "-+-" | ||||
|                 + "-" * COL_CORE_COUNT | ||||
|                 + "-+-" | ||||
|                 + "-" * COL_CORE_PERCENT | ||||
|             ) | ||||
|  | ||||
|             core_total = sum(size for _, _, size in self._esphome_core_symbols) | ||||
|  | ||||
|             for subcategory, symbols, total_size in sorted_subcategories: | ||||
|                 percentage = (total_size / core_total * 100) if core_total > 0 else 0 | ||||
|                 lines.append( | ||||
|                     f"{subcategory:<{COL_CORE_SUBCATEGORY}} | {total_size:>{COL_CORE_SIZE - 2},} B | " | ||||
|                     f"{len(symbols):>{COL_CORE_COUNT}} | {percentage:>{COL_CORE_PERCENT - 1}.1f}%" | ||||
|                 ) | ||||
|  | ||||
|             # Top 10 largest core symbols | ||||
|             lines.append("") | ||||
|             lines.append("Top 10 Largest [esphome]core Symbols:") | ||||
|             sorted_core_symbols = sorted( | ||||
|                 self._esphome_core_symbols, key=lambda x: x[2], reverse=True | ||||
|             ) | ||||
|  | ||||
|             for i, (symbol, demangled, size) in enumerate(sorted_core_symbols[:15]): | ||||
|                 lines.append(f"{i + 1}. {demangled} ({size:,} B)") | ||||
|  | ||||
|             lines.append("=" * table_width) | ||||
|  | ||||
|         # Add detailed analysis for top ESPHome and external components | ||||
|         esphome_components = [ | ||||
|             (name, mem) | ||||
|             for name, mem in components | ||||
|             if name.startswith("[esphome]") and name != "[esphome]core" | ||||
|         ] | ||||
|         external_components = [ | ||||
|             (name, mem) for name, mem in components if name.startswith("[external]") | ||||
|         ] | ||||
|  | ||||
|         top_esphome_components = sorted( | ||||
|             esphome_components, key=lambda x: x[1].flash_total, reverse=True | ||||
|         )[:30] | ||||
|  | ||||
|         # Include all external components (they're usually important) | ||||
|         top_external_components = sorted( | ||||
|             external_components, key=lambda x: x[1].flash_total, reverse=True | ||||
|         ) | ||||
|  | ||||
|         # Check if API component exists and ensure it's included | ||||
|         api_component = None | ||||
|         for name, mem in components: | ||||
|             if name == "[esphome]api": | ||||
|                 api_component = (name, mem) | ||||
|                 break | ||||
|  | ||||
|         # Combine all components to analyze: top ESPHome + all external + API if not already included | ||||
|         components_to_analyze = list(top_esphome_components) + list( | ||||
|             top_external_components | ||||
|         ) | ||||
|         if api_component and api_component not in components_to_analyze: | ||||
|             components_to_analyze.append(api_component) | ||||
|  | ||||
|         if components_to_analyze: | ||||
|             for comp_name, comp_mem in components_to_analyze: | ||||
|                 comp_symbols = self._component_symbols.get(comp_name, []) | ||||
|                 if comp_symbols: | ||||
|                     lines.append("") | ||||
|                     lines.append("=" * table_width) | ||||
|                     lines.append(f"{comp_name} Detailed Analysis".center(table_width)) | ||||
|                     lines.append("=" * table_width) | ||||
|                     lines.append("") | ||||
|  | ||||
|                     # Sort symbols by size | ||||
|                     sorted_symbols = sorted( | ||||
|                         comp_symbols, key=lambda x: x[2], reverse=True | ||||
|                     ) | ||||
|  | ||||
|                     lines.append(f"Total symbols: {len(sorted_symbols)}") | ||||
|                     lines.append(f"Total size: {comp_mem.flash_total:,} B") | ||||
|                     lines.append("") | ||||
|  | ||||
|                     # Show all symbols > 100 bytes for better visibility | ||||
|                     large_symbols = [ | ||||
|                         (sym, dem, size) | ||||
|                         for sym, dem, size in sorted_symbols | ||||
|                         if size > 100 | ||||
|                     ] | ||||
|  | ||||
|                     lines.append( | ||||
|                         f"{comp_name} Symbols > 100 B ({len(large_symbols)} symbols):" | ||||
|                     ) | ||||
|                     for i, (symbol, demangled, size) in enumerate(large_symbols): | ||||
|                         lines.append(f"{i + 1}. {demangled} ({size:,} B)") | ||||
|  | ||||
|                     lines.append("=" * table_width) | ||||
|  | ||||
|         return "\n".join(lines) | ||||
|  | ||||
|     def to_json(self) -> str: | ||||
|         """Export analysis results as JSON.""" | ||||
|         data = { | ||||
| @@ -707,63 +447,8 @@ class MemoryAnalyzer: | ||||
|         } | ||||
|         return json.dumps(data, indent=2) | ||||
|  | ||||
|     def dump_uncategorized_symbols(self, output_file: str | None = None) -> None: | ||||
|         """Dump uncategorized symbols for analysis.""" | ||||
|         # Sort by size descending | ||||
|         sorted_symbols = sorted( | ||||
|             self._uncategorized_symbols, key=lambda x: x[2], reverse=True | ||||
|         ) | ||||
|  | ||||
|         lines = ["Uncategorized Symbols Analysis", "=" * 80] | ||||
|         lines.append(f"Total uncategorized symbols: {len(sorted_symbols)}") | ||||
|         lines.append( | ||||
|             f"Total uncategorized size: {sum(s[2] for s in sorted_symbols):,} bytes" | ||||
|         ) | ||||
|         lines.append("") | ||||
|         lines.append(f"{'Size':>10} | {'Symbol':<60} | Demangled") | ||||
|         lines.append("-" * 10 + "-+-" + "-" * 60 + "-+-" + "-" * 40) | ||||
|  | ||||
|         for symbol, demangled, size in sorted_symbols[:100]:  # Top 100 | ||||
|             if symbol != demangled: | ||||
|                 lines.append(f"{size:>10,} | {symbol[:60]:<60} | {demangled[:100]}") | ||||
|             else: | ||||
|                 lines.append(f"{size:>10,} | {symbol[:60]:<60} | [not demangled]") | ||||
|  | ||||
|         if len(sorted_symbols) > 100: | ||||
|             lines.append(f"\n... and {len(sorted_symbols) - 100} more symbols") | ||||
|  | ||||
|         content = "\n".join(lines) | ||||
|  | ||||
|         if output_file: | ||||
|             with open(output_file, "w") as f: | ||||
|                 f.write(content) | ||||
|         else: | ||||
|             print(content) | ||||
|  | ||||
|  | ||||
| def analyze_elf( | ||||
|     elf_path: str, | ||||
|     objdump_path: str | None = None, | ||||
|     readelf_path: str | None = None, | ||||
|     detailed: bool = False, | ||||
|     external_components: set[str] | None = None, | ||||
| ) -> str: | ||||
|     """Analyze an ELF file and return a memory report.""" | ||||
|     analyzer = MemoryAnalyzer(elf_path, objdump_path, readelf_path, external_components) | ||||
|     analyzer.analyze() | ||||
|     return analyzer.generate_report(detailed) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     import sys | ||||
|     from .cli import main | ||||
|  | ||||
|     if len(sys.argv) < 2: | ||||
|         print("Usage: analyze_memory.py <elf_file>") | ||||
|         sys.exit(1) | ||||
|  | ||||
|     try: | ||||
|         report = analyze_elf(sys.argv[1]) | ||||
|         print(report) | ||||
|     except Exception as e: | ||||
|         print(f"Error: {e}") | ||||
|         sys.exit(1) | ||||
|     main() | ||||
|   | ||||
							
								
								
									
										6
									
								
								esphome/analyze_memory/__main__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								esphome/analyze_memory/__main__.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| """Main entry point for running the memory analyzer as a module.""" | ||||
|  | ||||
| from .cli import main | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
							
								
								
									
										338
									
								
								esphome/analyze_memory/cli.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										338
									
								
								esphome/analyze_memory/cli.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,338 @@ | ||||
| """CLI interface for memory analysis with report generation.""" | ||||
|  | ||||
| from collections import defaultdict | ||||
| import subprocess | ||||
| import sys | ||||
|  | ||||
| from . import MemoryAnalyzer | ||||
|  | ||||
|  | ||||
| class MemoryAnalyzerCLI(MemoryAnalyzer): | ||||
|     """Memory analyzer with CLI-specific report generation.""" | ||||
|  | ||||
|     def generate_report(self, detailed: bool = False) -> str: | ||||
|         """Generate a formatted memory report.""" | ||||
|         components = sorted( | ||||
|             self.components.items(), key=lambda x: x[1].flash_total, reverse=True | ||||
|         ) | ||||
|  | ||||
|         # Calculate totals | ||||
|         total_flash = sum(c.flash_total for _, c in components) | ||||
|         total_ram = sum(c.ram_total for _, c in components) | ||||
|  | ||||
|         # Build report | ||||
|         lines = [] | ||||
|  | ||||
|         # Column width constants | ||||
|         COL_COMPONENT = 29 | ||||
|         COL_FLASH_TEXT = 14 | ||||
|         COL_FLASH_DATA = 14 | ||||
|         COL_RAM_DATA = 12 | ||||
|         COL_RAM_BSS = 12 | ||||
|         COL_TOTAL_FLASH = 15 | ||||
|         COL_TOTAL_RAM = 12 | ||||
|         COL_SEPARATOR = 3  # " | " | ||||
|  | ||||
|         # Core analysis column widths | ||||
|         COL_CORE_SUBCATEGORY = 30 | ||||
|         COL_CORE_SIZE = 12 | ||||
|         COL_CORE_COUNT = 6 | ||||
|         COL_CORE_PERCENT = 10 | ||||
|  | ||||
|         # Calculate the exact table width | ||||
|         table_width = ( | ||||
|             COL_COMPONENT | ||||
|             + COL_SEPARATOR | ||||
|             + COL_FLASH_TEXT | ||||
|             + COL_SEPARATOR | ||||
|             + COL_FLASH_DATA | ||||
|             + COL_SEPARATOR | ||||
|             + COL_RAM_DATA | ||||
|             + COL_SEPARATOR | ||||
|             + COL_RAM_BSS | ||||
|             + COL_SEPARATOR | ||||
|             + COL_TOTAL_FLASH | ||||
|             + COL_SEPARATOR | ||||
|             + COL_TOTAL_RAM | ||||
|         ) | ||||
|  | ||||
|         lines.append("=" * table_width) | ||||
|         lines.append("Component Memory Analysis".center(table_width)) | ||||
|         lines.append("=" * table_width) | ||||
|         lines.append("") | ||||
|  | ||||
|         # Main table - fixed column widths | ||||
|         lines.append( | ||||
|             f"{'Component':<{COL_COMPONENT}} | {'Flash (text)':>{COL_FLASH_TEXT}} | {'Flash (data)':>{COL_FLASH_DATA}} | {'RAM (data)':>{COL_RAM_DATA}} | {'RAM (bss)':>{COL_RAM_BSS}} | {'Total Flash':>{COL_TOTAL_FLASH}} | {'Total RAM':>{COL_TOTAL_RAM}}" | ||||
|         ) | ||||
|         lines.append( | ||||
|             "-" * COL_COMPONENT | ||||
|             + "-+-" | ||||
|             + "-" * COL_FLASH_TEXT | ||||
|             + "-+-" | ||||
|             + "-" * COL_FLASH_DATA | ||||
|             + "-+-" | ||||
|             + "-" * COL_RAM_DATA | ||||
|             + "-+-" | ||||
|             + "-" * COL_RAM_BSS | ||||
|             + "-+-" | ||||
|             + "-" * COL_TOTAL_FLASH | ||||
|             + "-+-" | ||||
|             + "-" * COL_TOTAL_RAM | ||||
|         ) | ||||
|  | ||||
|         for name, mem in components: | ||||
|             if mem.flash_total > 0 or mem.ram_total > 0: | ||||
|                 flash_rodata = mem.rodata_size + mem.data_size | ||||
|                 lines.append( | ||||
|                     f"{name:<{COL_COMPONENT}} | {mem.text_size:>{COL_FLASH_TEXT - 2},} B | {flash_rodata:>{COL_FLASH_DATA - 2},} B | " | ||||
|                     f"{mem.data_size:>{COL_RAM_DATA - 2},} B | {mem.bss_size:>{COL_RAM_BSS - 2},} B | " | ||||
|                     f"{mem.flash_total:>{COL_TOTAL_FLASH - 2},} B | {mem.ram_total:>{COL_TOTAL_RAM - 2},} B" | ||||
|                 ) | ||||
|  | ||||
|         lines.append( | ||||
|             "-" * COL_COMPONENT | ||||
|             + "-+-" | ||||
|             + "-" * COL_FLASH_TEXT | ||||
|             + "-+-" | ||||
|             + "-" * COL_FLASH_DATA | ||||
|             + "-+-" | ||||
|             + "-" * COL_RAM_DATA | ||||
|             + "-+-" | ||||
|             + "-" * COL_RAM_BSS | ||||
|             + "-+-" | ||||
|             + "-" * COL_TOTAL_FLASH | ||||
|             + "-+-" | ||||
|             + "-" * COL_TOTAL_RAM | ||||
|         ) | ||||
|         lines.append( | ||||
|             f"{'TOTAL':<{COL_COMPONENT}} | {' ':>{COL_FLASH_TEXT}} | {' ':>{COL_FLASH_DATA}} | " | ||||
|             f"{' ':>{COL_RAM_DATA}} | {' ':>{COL_RAM_BSS}} | " | ||||
|             f"{total_flash:>{COL_TOTAL_FLASH - 2},} B | {total_ram:>{COL_TOTAL_RAM - 2},} B" | ||||
|         ) | ||||
|  | ||||
|         # Top consumers | ||||
|         lines.append("") | ||||
|         lines.append("Top Flash Consumers:") | ||||
|         for i, (name, mem) in enumerate(components[:25]): | ||||
|             if mem.flash_total > 0: | ||||
|                 percentage = ( | ||||
|                     (mem.flash_total / total_flash * 100) if total_flash > 0 else 0 | ||||
|                 ) | ||||
|                 lines.append( | ||||
|                     f"{i + 1}. {name} ({mem.flash_total:,} B) - {percentage:.1f}% of analyzed flash" | ||||
|                 ) | ||||
|  | ||||
|         lines.append("") | ||||
|         lines.append("Top RAM Consumers:") | ||||
|         ram_components = sorted(components, key=lambda x: x[1].ram_total, reverse=True) | ||||
|         for i, (name, mem) in enumerate(ram_components[:25]): | ||||
|             if mem.ram_total > 0: | ||||
|                 percentage = (mem.ram_total / total_ram * 100) if total_ram > 0 else 0 | ||||
|                 lines.append( | ||||
|                     f"{i + 1}. {name} ({mem.ram_total:,} B) - {percentage:.1f}% of analyzed RAM" | ||||
|                 ) | ||||
|  | ||||
|         lines.append("") | ||||
|         lines.append( | ||||
|             "Note: This analysis covers symbols in the ELF file. Some runtime allocations may not be included." | ||||
|         ) | ||||
|         lines.append("=" * table_width) | ||||
|  | ||||
|         # Add ESPHome core detailed analysis if there are core symbols | ||||
|         if self._esphome_core_symbols: | ||||
|             lines.append("") | ||||
|             lines.append("=" * table_width) | ||||
|             lines.append("[esphome]core Detailed Analysis".center(table_width)) | ||||
|             lines.append("=" * table_width) | ||||
|             lines.append("") | ||||
|  | ||||
|             # Group core symbols by subcategory | ||||
|             core_subcategories: dict[str, list[tuple[str, str, int]]] = defaultdict( | ||||
|                 list | ||||
|             ) | ||||
|  | ||||
|             for symbol, demangled, size in self._esphome_core_symbols: | ||||
|                 # Categorize based on demangled name patterns | ||||
|                 subcategory = self._categorize_esphome_core_symbol(demangled) | ||||
|                 core_subcategories[subcategory].append((symbol, demangled, size)) | ||||
|  | ||||
|             # Sort subcategories by total size | ||||
|             sorted_subcategories = sorted( | ||||
|                 [ | ||||
|                     (name, symbols, sum(s[2] for s in symbols)) | ||||
|                     for name, symbols in core_subcategories.items() | ||||
|                 ], | ||||
|                 key=lambda x: x[2], | ||||
|                 reverse=True, | ||||
|             ) | ||||
|  | ||||
|             lines.append( | ||||
|                 f"{'Subcategory':<{COL_CORE_SUBCATEGORY}} | {'Size':>{COL_CORE_SIZE}} | " | ||||
|                 f"{'Count':>{COL_CORE_COUNT}} | {'% of Core':>{COL_CORE_PERCENT}}" | ||||
|             ) | ||||
|             lines.append( | ||||
|                 "-" * COL_CORE_SUBCATEGORY | ||||
|                 + "-+-" | ||||
|                 + "-" * COL_CORE_SIZE | ||||
|                 + "-+-" | ||||
|                 + "-" * COL_CORE_COUNT | ||||
|                 + "-+-" | ||||
|                 + "-" * COL_CORE_PERCENT | ||||
|             ) | ||||
|  | ||||
|             core_total = sum(size for _, _, size in self._esphome_core_symbols) | ||||
|  | ||||
|             for subcategory, symbols, total_size in sorted_subcategories: | ||||
|                 percentage = (total_size / core_total * 100) if core_total > 0 else 0 | ||||
|                 lines.append( | ||||
|                     f"{subcategory:<{COL_CORE_SUBCATEGORY}} | {total_size:>{COL_CORE_SIZE - 2},} B | " | ||||
|                     f"{len(symbols):>{COL_CORE_COUNT}} | {percentage:>{COL_CORE_PERCENT - 1}.1f}%" | ||||
|                 ) | ||||
|  | ||||
|             # Top 10 largest core symbols | ||||
|             lines.append("") | ||||
|             lines.append("Top 10 Largest [esphome]core Symbols:") | ||||
|             sorted_core_symbols = sorted( | ||||
|                 self._esphome_core_symbols, key=lambda x: x[2], reverse=True | ||||
|             ) | ||||
|  | ||||
|             for i, (symbol, demangled, size) in enumerate(sorted_core_symbols[:15]): | ||||
|                 lines.append(f"{i + 1}. {demangled} ({size:,} B)") | ||||
|  | ||||
|             lines.append("=" * table_width) | ||||
|  | ||||
|         # Add detailed analysis for top ESPHome and external components | ||||
|         esphome_components = [ | ||||
|             (name, mem) | ||||
|             for name, mem in components | ||||
|             if name.startswith("[esphome]") and name != "[esphome]core" | ||||
|         ] | ||||
|         external_components = [ | ||||
|             (name, mem) for name, mem in components if name.startswith("[external]") | ||||
|         ] | ||||
|  | ||||
|         top_esphome_components = sorted( | ||||
|             esphome_components, key=lambda x: x[1].flash_total, reverse=True | ||||
|         )[:30] | ||||
|  | ||||
|         # Include all external components (they're usually important) | ||||
|         top_external_components = sorted( | ||||
|             external_components, key=lambda x: x[1].flash_total, reverse=True | ||||
|         ) | ||||
|  | ||||
|         # Check if API component exists and ensure it's included | ||||
|         api_component = None | ||||
|         for name, mem in components: | ||||
|             if name == "[esphome]api": | ||||
|                 api_component = (name, mem) | ||||
|                 break | ||||
|  | ||||
|         # Combine all components to analyze: top ESPHome + all external + API if not already included | ||||
|         components_to_analyze = list(top_esphome_components) + list( | ||||
|             top_external_components | ||||
|         ) | ||||
|         if api_component and api_component not in components_to_analyze: | ||||
|             components_to_analyze.append(api_component) | ||||
|  | ||||
|         if components_to_analyze: | ||||
|             for comp_name, comp_mem in components_to_analyze: | ||||
|                 comp_symbols = self._component_symbols.get(comp_name, []) | ||||
|                 if comp_symbols: | ||||
|                     lines.append("") | ||||
|                     lines.append("=" * table_width) | ||||
|                     lines.append(f"{comp_name} Detailed Analysis".center(table_width)) | ||||
|                     lines.append("=" * table_width) | ||||
|                     lines.append("") | ||||
|  | ||||
|                     # Sort symbols by size | ||||
|                     sorted_symbols = sorted( | ||||
|                         comp_symbols, key=lambda x: x[2], reverse=True | ||||
|                     ) | ||||
|  | ||||
|                     lines.append(f"Total symbols: {len(sorted_symbols)}") | ||||
|                     lines.append(f"Total size: {comp_mem.flash_total:,} B") | ||||
|                     lines.append("") | ||||
|  | ||||
|                     # Show all symbols > 100 bytes for better visibility | ||||
|                     large_symbols = [ | ||||
|                         (sym, dem, size) | ||||
|                         for sym, dem, size in sorted_symbols | ||||
|                         if size > 100 | ||||
|                     ] | ||||
|  | ||||
|                     lines.append( | ||||
|                         f"{comp_name} Symbols > 100 B ({len(large_symbols)} symbols):" | ||||
|                     ) | ||||
|                     for i, (symbol, demangled, size) in enumerate(large_symbols): | ||||
|                         lines.append(f"{i + 1}. {demangled} ({size:,} B)") | ||||
|  | ||||
|                     lines.append("=" * table_width) | ||||
|  | ||||
|         return "\n".join(lines) | ||||
|  | ||||
|     def dump_uncategorized_symbols(self, output_file: str | None = None) -> None: | ||||
|         """Dump uncategorized symbols for analysis.""" | ||||
|         # Sort by size descending | ||||
|         sorted_symbols = sorted( | ||||
|             self._uncategorized_symbols, key=lambda x: x[2], reverse=True | ||||
|         ) | ||||
|  | ||||
|         lines = ["Uncategorized Symbols Analysis", "=" * 80] | ||||
|         lines.append(f"Total uncategorized symbols: {len(sorted_symbols)}") | ||||
|         lines.append( | ||||
|             f"Total uncategorized size: {sum(s[2] for s in sorted_symbols):,} bytes" | ||||
|         ) | ||||
|         lines.append("") | ||||
|         lines.append(f"{'Size':>10} | {'Symbol':<60} | Demangled") | ||||
|         lines.append("-" * 10 + "-+-" + "-" * 60 + "-+-" + "-" * 40) | ||||
|  | ||||
|         for symbol, demangled, size in sorted_symbols[:100]:  # Top 100 | ||||
|             if symbol != demangled: | ||||
|                 lines.append(f"{size:>10,} | {symbol[:60]:<60} | {demangled[:100]}") | ||||
|             else: | ||||
|                 lines.append(f"{size:>10,} | {symbol[:60]:<60} | [not demangled]") | ||||
|  | ||||
|         if len(sorted_symbols) > 100: | ||||
|             lines.append(f"\n... and {len(sorted_symbols) - 100} more symbols") | ||||
|  | ||||
|         content = "\n".join(lines) | ||||
|  | ||||
|         if output_file: | ||||
|             with open(output_file, "w", encoding="utf-8") as f: | ||||
|                 f.write(content) | ||||
|         else: | ||||
|             print(content) | ||||
|  | ||||
|  | ||||
| def analyze_elf( | ||||
|     elf_path: str, | ||||
|     objdump_path: str | None = None, | ||||
|     readelf_path: str | None = None, | ||||
|     detailed: bool = False, | ||||
|     external_components: set[str] | None = None, | ||||
| ) -> str: | ||||
|     """Analyze an ELF file and return a memory report.""" | ||||
|     analyzer = MemoryAnalyzerCLI( | ||||
|         elf_path, objdump_path, readelf_path, external_components | ||||
|     ) | ||||
|     analyzer.analyze() | ||||
|     return analyzer.generate_report(detailed) | ||||
|  | ||||
|  | ||||
| def main(): | ||||
|     """CLI entrypoint for memory analysis.""" | ||||
|     if len(sys.argv) < 2: | ||||
|         print("Usage: analyze_memory.py <elf_file>") | ||||
|         sys.exit(1) | ||||
|  | ||||
|     try: | ||||
|         report = analyze_elf(sys.argv[1]) | ||||
|         print(report) | ||||
|     except (subprocess.CalledProcessError, FileNotFoundError, OSError) as e: | ||||
|         print(f"Error: {e}") | ||||
|         sys.exit(1) | ||||
|  | ||||
|  | ||||
| if __name__ == "__main__": | ||||
|     main() | ||||
		Reference in New Issue
	
	Block a user