idf_size.py 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886
  1. #!/usr/bin/env python
  2. #
  3. # esp-idf alternative to "size" to print ELF file sizes, also analyzes
  4. # the linker map file to dump higher resolution details.
  5. #
  6. # Includes information which is not shown in "xtensa-esp32-elf-size",
  7. # or easy to parse from "xtensa-esp32-elf-objdump" or raw map files.
  8. #
  9. # Copyright 2017-2020 Espressif Systems (Shanghai) PTE LTD
  10. #
  11. # Licensed under the Apache License, Version 2.0 (the "License");
  12. # you may not use this file except in compliance with the License.
  13. # You may obtain a copy of the License at
  14. #
  15. # http://www.apache.org/licenses/LICENSE-2.0
  16. #
  17. # Unless required by applicable law or agreed to in writing, software
  18. # distributed under the License is distributed on an "AS IS" BASIS,
  19. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  20. # See the License for the specific language governing permissions and
  21. # limitations under the License.
  22. #
  23. from __future__ import print_function
  24. from __future__ import unicode_literals
  25. from __future__ import division
  26. from future.utils import iteritems
  27. import argparse
  28. import collections
  29. import json
  30. import os.path
  31. import re
  32. import sys
  33. GLOBAL_JSON_INDENT = 4
  34. GLOBAL_JSON_SEPARATORS = (',', ': ')
  35. class MemRegions(object):
  36. (DRAM_ID, IRAM_ID, DIRAM_ID) = range(3)
  37. @staticmethod
  38. def get_mem_regions(target):
  39. # The target specific memory structure is deduced from soc_memory_types defined in
  40. # $IDF_PATH/components/soc/**/soc_memory_layout.c files.
  41. # The order of variables in the tuple is the same as in the soc_memory_layout.c files
  42. MemRegDef = collections.namedtuple('MemRegDef', ['primary_addr', 'length', 'type', 'secondary_addr'])
  43. if target == 'esp32':
  44. return sorted([
  45. # Consecutive MemRegDefs of the same type are joined into one MemRegDef
  46. MemRegDef(0x3FFAE000, 17 * 0x2000 + 4 * 0x8000 + 4 * 0x4000, MemRegions.DRAM_ID, 0),
  47. # MemRegDef(0x3FFAE000, 0x2000, MemRegions.DRAM_ID, 0),
  48. # MemRegDef(0x3FFB0000, 0x8000, MemRegions.DRAM_ID, 0),
  49. # MemRegDef(0x3FFB8000, 0x8000, MemRegions.DRAM_ID, 0),
  50. # MemRegDef(0x3FFC0000, 0x2000, MemRegions.DRAM_ID, 0),
  51. # MemRegDef(0x3FFC2000, 0x2000, MemRegions.DRAM_ID, 0),
  52. # MemRegDef(0x3FFC4000, 0x2000, MemRegions.DRAM_ID, 0),
  53. # MemRegDef(0x3FFC6000, 0x2000, MemRegions.DRAM_ID, 0),
  54. # MemRegDef(0x3FFC8000, 0x2000, MemRegions.DRAM_ID, 0),
  55. # MemRegDef(0x3FFCA000, 0x2000, MemRegions.DRAM_ID, 0),
  56. # MemRegDef(0x3FFCC000, 0x2000, MemRegions.DRAM_ID, 0),
  57. # MemRegDef(0x3FFCE000, 0x2000, MemRegions.DRAM_ID, 0),
  58. # MemRegDef(0x3FFD0000, 0x2000, MemRegions.DRAM_ID, 0),
  59. # MemRegDef(0x3FFD2000, 0x2000, MemRegions.DRAM_ID, 0),
  60. # MemRegDef(0x3FFD4000, 0x2000, MemRegions.DRAM_ID, 0),
  61. # MemRegDef(0x3FFD6000, 0x2000, MemRegions.DRAM_ID, 0),
  62. # MemRegDef(0x3FFD8000, 0x2000, MemRegions.DRAM_ID, 0),
  63. # MemRegDef(0x3FFDA000, 0x2000, MemRegions.DRAM_ID, 0),
  64. # MemRegDef(0x3FFDC000, 0x2000, MemRegions.DRAM_ID, 0),
  65. # MemRegDef(0x3FFDE000, 0x2000, MemRegions.DRAM_ID, 0),
  66. #
  67. # The bootloader is there and it has to been counted as DRAM
  68. # MemRegDef(0x3FFE0000, 0x4000, MemRegions.DIRAM_ID, 0x400BC000),
  69. # MemRegDef(0x3FFE4000, 0x4000, MemRegions.DIRAM_ID, 0x400B8000),
  70. # MemRegDef(0x3FFE8000, 0x8000, MemRegions.DIRAM_ID, 0x400B0000),
  71. # MemRegDef(0x3FFF0000, 0x8000, MemRegions.DIRAM_ID, 0x400A8000),
  72. # MemRegDef(0x3FFF8000, 0x4000, MemRegions.DIRAM_ID, 0x400A4000),
  73. # MemRegDef(0x3FFFC000, 0x4000, MemRegions.DIRAM_ID, 0x400A0000),
  74. #
  75. MemRegDef(0x40070000, 2 * 0x8000 + 16 * 0x2000, MemRegions.IRAM_ID, 0),
  76. # MemRegDef(0x40070000, 0x8000, MemRegions.IRAM_ID, 0),
  77. # MemRegDef(0x40078000, 0x8000, MemRegions.IRAM_ID, 0),
  78. # MemRegDef(0x40080000, 0x2000, MemRegions.IRAM_ID, 0),
  79. # MemRegDef(0x40082000, 0x2000, MemRegions.IRAM_ID, 0),
  80. # MemRegDef(0x40084000, 0x2000, MemRegions.IRAM_ID, 0),
  81. # MemRegDef(0x40086000, 0x2000, MemRegions.IRAM_ID, 0),
  82. # MemRegDef(0x40088000, 0x2000, MemRegions.IRAM_ID, 0),
  83. # MemRegDef(0x4008A000, 0x2000, MemRegions.IRAM_ID, 0),
  84. # MemRegDef(0x4008C000, 0x2000, MemRegions.IRAM_ID, 0),
  85. # MemRegDef(0x4008E000, 0x2000, MemRegions.IRAM_ID, 0),
  86. # MemRegDef(0x40090000, 0x2000, MemRegions.IRAM_ID, 0),
  87. # MemRegDef(0x40092000, 0x2000, MemRegions.IRAM_ID, 0),
  88. # MemRegDef(0x40094000, 0x2000, MemRegions.IRAM_ID, 0),
  89. # MemRegDef(0x40096000, 0x2000, MemRegions.IRAM_ID, 0),
  90. # MemRegDef(0x40098000, 0x2000, MemRegions.IRAM_ID, 0),
  91. # MemRegDef(0x4009A000, 0x2000, MemRegions.IRAM_ID, 0),
  92. # MemRegDef(0x4009C000, 0x2000, MemRegions.IRAM_ID, 0),
  93. # MemRegDef(0x4009E000, 0x2000, MemRegions.IRAM_ID, 0),
  94. ])
  95. elif target == 'esp32s2':
  96. return sorted([
  97. MemRegDef(0x3FFB2000, 3 * 0x2000 + 18 * 0x4000, MemRegions.DIRAM_ID, 0x40022000),
  98. # MemRegDef(0x3FFB2000, 0x2000, MemRegions.DIRAM_ID, 0x40022000),
  99. # MemRegDef(0x3FFB4000, 0x2000, MemRegions.DIRAM_ID, 0x40024000),
  100. # MemRegDef(0x3FFB6000, 0x2000, MemRegions.DIRAM_ID, 0x40026000),
  101. # MemRegDef(0x3FFB8000, 0x4000, MemRegions.DIRAM_ID, 0x40028000),
  102. # MemRegDef(0x3FFBC000, 0x4000, MemRegions.DIRAM_ID, 0x4002C000),
  103. # MemRegDef(0x3FFC0000, 0x4000, MemRegions.DIRAM_ID, 0x40030000),
  104. # MemRegDef(0x3FFC4000, 0x4000, MemRegions.DIRAM_ID, 0x40034000),
  105. # MemRegDef(0x3FFC8000, 0x4000, MemRegions.DIRAM_ID, 0x40038000),
  106. # MemRegDef(0x3FFCC000, 0x4000, MemRegions.DIRAM_ID, 0x4003C000),
  107. # MemRegDef(0x3FFD0000, 0x4000, MemRegions.DIRAM_ID, 0x40040000),
  108. # MemRegDef(0x3FFD4000, 0x4000, MemRegions.DIRAM_ID, 0x40044000),
  109. # MemRegDef(0x3FFD8000, 0x4000, MemRegions.DIRAM_ID, 0x40048000),
  110. # MemRegDef(0x3FFDC000, 0x4000, MemRegions.DIRAM_ID, 0x4004C000),
  111. # MemRegDef(0x3FFE0000, 0x4000, MemRegions.DIRAM_ID, 0x40050000),
  112. #
  113. # MemRegDef(0x3FFE4000, 0x4000, MemRegions.DIRAM_ID, 0x40054000),
  114. # MemRegDef(0x3FFE8000, 0x4000, MemRegions.DIRAM_ID, 0x40058000),
  115. # MemRegDef(0x3FFEC000, 0x4000, MemRegions.DIRAM_ID, 0x4005C000),
  116. # MemRegDef(0x3FFF0000, 0x4000, MemRegions.DIRAM_ID, 0x40060000),
  117. # MemRegDef(0x3FFF4000, 0x4000, MemRegions.DIRAM_ID, 0x40064000),
  118. # MemRegDef(0x3FFF8000, 0x4000, MemRegions.DIRAM_ID, 0x40068000),
  119. # MemRegDef(0x3FFFC000, 0x4000, MemRegions.DIRAM_ID, 0x4006C000),
  120. ])
  121. else:
  122. return None
  123. def __init__(self, target):
  124. self.chip_mem_regions = self.get_mem_regions(target)
  125. if not self.chip_mem_regions:
  126. raise RuntimeError('Target {} is not implemented in idf_size'.format(target))
  127. def _address_in_range(self, address, length, reg_address, reg_length):
  128. return address >= reg_address and (address - reg_address) <= (reg_length - length)
  129. def get_names(self, dictionary, region_id):
  130. def get_address(d):
  131. try:
  132. return d['address']
  133. except KeyError:
  134. return d['origin']
  135. def get_size(d):
  136. try:
  137. return d['size']
  138. except KeyError:
  139. return d['length']
  140. result = set() # using a set will remove possible duplicates and consequent operations with sets are more
  141. # efficient
  142. for m in self.chip_mem_regions:
  143. if m.type != region_id:
  144. continue
  145. # the following code is intentionally not a one-liner for better readability
  146. for (n, c) in iteritems(dictionary):
  147. if (self._address_in_range(get_address(c), get_size(c), m.primary_addr, m.length) or
  148. (m.type == self.DIRAM_ID and
  149. self._address_in_range(get_address(c), get_size(c), m.secondary_addr, m.length))):
  150. result.add(n)
  151. return result
  152. def scan_to_header(f, header_line):
  153. """ Scan forward in a file until you reach 'header_line', then return """
  154. for line in f:
  155. if line.strip() == header_line:
  156. return
  157. raise RuntimeError("Didn't find line '%s' in file" % header_line)
  158. def format_json(json_object):
  159. return json.dumps(json_object,
  160. allow_nan=False,
  161. indent=GLOBAL_JSON_INDENT,
  162. separators=GLOBAL_JSON_SEPARATORS) + os.linesep
  163. def load_map_data(map_file):
  164. memory_config = load_memory_config(map_file)
  165. sections = load_sections(map_file)
  166. return memory_config, sections
  167. def load_memory_config(map_file):
  168. """ Memory Configuration section is the total size of each output section """
  169. result = {}
  170. scan_to_header(map_file, "Memory Configuration")
  171. RE_MEMORY_SECTION = re.compile(r"(?P<name>[^ ]+) +0x(?P<origin>[\da-f]+) +0x(?P<length>[\da-f]+)")
  172. for line in map_file:
  173. m = RE_MEMORY_SECTION.match(line)
  174. if m is None:
  175. if len(result) == 0:
  176. continue # whitespace or a header, before the content we want
  177. else:
  178. return result # we're at the end of the Memory Configuration
  179. section = {
  180. "name": m.group("name"),
  181. "origin": int(m.group("origin"), 16),
  182. "length": int(m.group("length"), 16),
  183. }
  184. if section["name"] != "*default*":
  185. result[section["name"]] = section
  186. raise RuntimeError("End of file while scanning memory configuration?")
  187. def load_sections(map_file):
  188. """ Load section size information from the MAP file.
  189. Returns a dict of 'sections', where each key is a section name and the value
  190. is a dict with details about this section, including a "sources" key which holds a list of source file line
  191. information for each symbol linked into the section.
  192. """
  193. scan_to_header(map_file, "Linker script and memory map")
  194. # output section header, ie '.iram0.text 0x0000000040080400 0x129a5'
  195. RE_SECTION_HEADER = re.compile(r"(?P<name>[^ ]+) +0x(?P<address>[\da-f]+) +0x(?P<size>[\da-f]+)$")
  196. # source file line, ie
  197. # 0x0000000040080400 0xa4 /home/gus/esp/32/idf/examples/get-started/hello_world/build/esp32/libesp32.a(cpu_start.o)
  198. # cmake build system links some object files directly, not part of any archive, so make that part optional
  199. # .xtensa.info 0x0000000000000000 0x38 CMakeFiles/hello-world.elf.dir/project_elf_src.c.obj
  200. RE_SOURCE_LINE = re.compile(r"\s*(?P<sym_name>\S*) +0x(?P<address>[\da-f]+) +0x(?P<size>[\da-f]+) (?P<archive>.+\.a)?\(?(?P<object_file>.+\.(o|obj))\)?")
  201. # Fast check to see if line is a potential source line before running the slower full regex against it
  202. RE_PRE_FILTER = re.compile(r".*\.(o|obj)\)?")
  203. # Check for lines which only contain the sym name (and rest is on following lines)
  204. RE_SYMBOL_ONLY_LINE = re.compile(r"^ (?P<sym_name>\S*)$")
  205. sections = {}
  206. section = None
  207. sym_backup = None
  208. for line in map_file:
  209. if line.strip() == "Cross Reference Table":
  210. # stop processing lines because we are at the next section in the map file
  211. break
  212. m = RE_SECTION_HEADER.match(line)
  213. if m is not None: # start of a new section
  214. section = {
  215. "name": m.group("name"),
  216. "address": int(m.group("address"), 16),
  217. "size": int(m.group("size"), 16),
  218. "sources": [],
  219. }
  220. sections[section["name"]] = section
  221. continue
  222. if section is not None:
  223. m = RE_SYMBOL_ONLY_LINE.match(line)
  224. if m is not None:
  225. # In some cases the section name appears on the previous line, back it up in here
  226. sym_backup = m.group("sym_name")
  227. continue
  228. if not RE_PRE_FILTER.match(line):
  229. # line does not match our quick check, so skip to next line
  230. continue
  231. m = RE_SOURCE_LINE.match(line)
  232. if m is not None: # input source file details=ma,e
  233. sym_name = m.group("sym_name") if len(m.group("sym_name")) > 0 else sym_backup
  234. archive = m.group("archive")
  235. if archive is None:
  236. # optional named group "archive" was not matched, so assign a value to it
  237. archive = "(exe)"
  238. source = {
  239. "size": int(m.group("size"), 16),
  240. "address": int(m.group("address"), 16),
  241. "archive": os.path.basename(archive),
  242. "object_file": os.path.basename(m.group("object_file")),
  243. "sym_name": sym_name,
  244. }
  245. source["file"] = "%s:%s" % (source["archive"], source["object_file"])
  246. section["sources"] += [source]
  247. return sections
  248. class MemRegNames(object):
  249. @staticmethod
  250. def get(mem_regions, memory_config, sections):
  251. mreg = MemRegNames()
  252. mreg.iram_names = mem_regions.get_names(memory_config, MemRegions.IRAM_ID)
  253. mreg.dram_names = mem_regions.get_names(memory_config, MemRegions.DRAM_ID)
  254. mreg.diram_names = mem_regions.get_names(memory_config, MemRegions.DIRAM_ID)
  255. mreg.used_iram_names = mem_regions.get_names(sections, MemRegions.IRAM_ID)
  256. mreg.used_dram_names = mem_regions.get_names(sections, MemRegions.DRAM_ID)
  257. mreg.used_diram_names = mem_regions.get_names(sections, MemRegions.DIRAM_ID)
  258. return mreg
  259. def main():
  260. parser = argparse.ArgumentParser(description="idf_size - a tool to print size information from an IDF MAP file")
  261. parser.add_argument(
  262. '--json',
  263. help="Output results as JSON",
  264. action="store_true")
  265. parser.add_argument(
  266. 'map_file', help='MAP file produced by linker',
  267. type=argparse.FileType('r'))
  268. parser.add_argument(
  269. '--archives', help='Print per-archive sizes', action='store_true')
  270. parser.add_argument(
  271. '--archive_details', help='Print detailed symbols per archive')
  272. parser.add_argument(
  273. '--files', help='Print per-file sizes', action='store_true')
  274. parser.add_argument(
  275. '--target', help='Set target chip', default='esp32')
  276. parser.add_argument(
  277. '--diff', help='Show the differences in comparison with another MAP file',
  278. metavar='ANOTHER_MAP_FILE',
  279. default=None,
  280. dest='another_map_file')
  281. parser.add_argument(
  282. '-o',
  283. '--output-file',
  284. type=argparse.FileType('w'),
  285. default=sys.stdout,
  286. help="Print output to the specified file instead of stdout")
  287. args = parser.parse_args()
  288. memory_config, sections = load_map_data(args.map_file)
  289. args.map_file.close()
  290. if args.another_map_file:
  291. with open(args.another_map_file, 'r') as f:
  292. memory_config_diff, sections_diff = load_map_data(f)
  293. else:
  294. memory_config_diff, sections_diff = None, None
  295. mem_regions = MemRegions(args.target)
  296. mem_reg = MemRegNames.get(mem_regions, memory_config, sections)
  297. mem_reg_diff = MemRegNames.get(mem_regions, memory_config_diff, sections_diff) if args.another_map_file else None
  298. output = ''
  299. if not args.json or not (args.archives or args.files or args.archive_details):
  300. output += get_summary(args.map_file.name, mem_reg, memory_config, sections,
  301. args.json,
  302. args.another_map_file, mem_reg_diff, memory_config_diff, sections_diff)
  303. if args.archives:
  304. output += get_detailed_sizes(mem_reg, sections, "archive", "Archive File", args.json, sections_diff)
  305. if args.files:
  306. output += get_detailed_sizes(mem_reg, sections, "file", "Object File", args.json, sections_diff)
  307. if args.archive_details:
  308. output += get_archive_symbols(mem_reg, sections, args.archive_details, args.json, sections_diff)
  309. args.output_file.write(output)
  310. args.output_file.close()
  311. class StructureForSummary(object):
  312. (dram_data_names, dram_bss_names, dram_other_names,
  313. diram_data_names, diram_bss_names) = (frozenset(), ) * 5
  314. (total_iram, total_dram, total_dram, total_diram,
  315. used_dram_data, used_dram_bss, used_dram_other,
  316. used_dram, used_dram_ratio,
  317. used_iram, used_iram_ratio,
  318. used_diram_data, used_diram_bss,
  319. used_diram, used_diram_ratio,
  320. flash_code, flash_rodata,
  321. total_size) = (0, ) * 18
  322. @staticmethod
  323. def get(reg, mem_conf, sects):
  324. def _get_size(sects, section):
  325. try:
  326. return sects[section]['size']
  327. except KeyError:
  328. return 0
  329. r = StructureForSummary()
  330. r.dram_data_names = frozenset([n for n in reg.used_dram_names if n.endswith('.data')])
  331. r.dram_bss_names = frozenset([n for n in reg.used_dram_names if n.endswith('.bss')])
  332. r.dram_other_names = reg.used_dram_names - r.dram_data_names - r.dram_bss_names
  333. r.diram_data_names = frozenset([n for n in reg.used_diram_names if n.endswith('.data')])
  334. r.diram_bss_names = frozenset([n for n in reg.used_diram_names if n.endswith('.bss')])
  335. r.total_iram = sum(mem_conf[n]['length'] for n in reg.iram_names)
  336. r.total_dram = sum(mem_conf[n]['length'] for n in reg.dram_names)
  337. r.total_diram = sum(mem_conf[n]['length'] for n in reg.diram_names)
  338. r.used_dram_data = sum(_get_size(sects, n) for n in r.dram_data_names)
  339. r.used_dram_bss = sum(_get_size(sects, n) for n in r.dram_bss_names)
  340. r.used_dram_other = sum(_get_size(sects, n) for n in r.dram_other_names)
  341. r.used_dram = r.used_dram_data + r.used_dram_bss + r.used_dram_other
  342. try:
  343. r.used_dram_ratio = r.used_dram / r.total_dram
  344. except ZeroDivisionError:
  345. r.used_dram_ratio = float('nan')
  346. r.used_iram = sum(_get_size(sects, s) for s in sects if s in reg.used_iram_names)
  347. try:
  348. r.used_iram_ratio = r.used_iram / r.total_iram
  349. except ZeroDivisionError:
  350. r.used_iram_ratio = float('nan')
  351. r.used_diram_data = sum(_get_size(sects, n) for n in r.diram_data_names)
  352. r.used_diram_bss = sum(_get_size(sects, n) for n in r.diram_bss_names)
  353. r.used_diram = sum(_get_size(sects, n) for n in reg.used_diram_names)
  354. try:
  355. r.used_diram_ratio = r.used_diram / r.total_diram
  356. except ZeroDivisionError:
  357. r.used_diram_ratio = float('nan')
  358. r.flash_code = _get_size(sects, '.flash.text')
  359. r.flash_rodata = _get_size(sects, '.flash.rodata')
  360. r.total_size = r.used_dram + r.used_iram + r.used_diram + r.flash_code + r.flash_rodata
  361. return r
  362. def get_json_dic(self):
  363. return collections.OrderedDict([
  364. ('dram_data', self.used_dram_data + self.used_diram_data),
  365. ('dram_bss', self.used_dram_bss + self.used_diram_bss),
  366. ('dram_other', self.used_dram_other),
  367. ('used_dram', self.used_dram),
  368. ('available_dram', self.total_dram - self.used_dram),
  369. ('used_dram_ratio', self.used_dram_ratio if self.total_dram != 0 else 0),
  370. ('used_iram', self.used_iram),
  371. ('available_iram', self.total_iram - self.used_iram),
  372. ('used_iram_ratio', self.used_iram_ratio if self.total_iram != 0 else 0),
  373. ('used_diram', self.used_diram),
  374. ('available_diram', self.total_diram - self.used_diram),
  375. ('used_diram_ratio', self.used_diram_ratio if self.total_diram != 0 else 0),
  376. ('flash_code', self.flash_code),
  377. ('flash_rodata', self.flash_rodata),
  378. ('total_size', self.total_size)
  379. ])
  380. def get_summary(path, mem_reg, memory_config, sections,
  381. as_json=False,
  382. path_diff=None, mem_reg_diff=None, memory_config_diff=None, sections_diff=None):
  383. diff_en = mem_reg_diff and memory_config_diff and sections_diff
  384. current = StructureForSummary.get(mem_reg, memory_config, sections)
  385. reference = StructureForSummary.get(mem_reg_diff,
  386. memory_config_diff,
  387. sections_diff) if diff_en else StructureForSummary()
  388. if as_json:
  389. current_json_dic = current.get_json_dic()
  390. if diff_en:
  391. reference_json_dic = reference.get_json_dic()
  392. diff_json_dic = collections.OrderedDict([(k,
  393. v - reference_json_dic[k]) for k, v in iteritems(current_json_dic)])
  394. output = format_json(collections.OrderedDict([('current', current_json_dic),
  395. ('reference', reference_json_dic),
  396. ('diff', diff_json_dic),
  397. ]))
  398. else:
  399. output = format_json(current_json_dic)
  400. else:
  401. rows = []
  402. if diff_en:
  403. rows += [('<CURRENT> MAP file: {}'.format(path), '', '', '')]
  404. rows += [('<REFERENCE> MAP file: {}'.format(path_diff), '', '', '')]
  405. rows += [('Difference is counted as <CURRENT> - <REFERENCE>, '
  406. 'i.e. a positive number means that <CURRENT> is larger.',
  407. '', '', '')]
  408. rows += [('Total sizes{}:'.format(' of <CURRENT>' if diff_en else ''), '<REFERENCE>', 'Difference', '')]
  409. rows += [(' DRAM .data size: {f_dram_data:>7} bytes', '{f_dram_data_2:>7}', '{f_dram_data_diff:+}', '')]
  410. rows += [(' DRAM .bss size: {f_dram_bss:>7} bytes', '{f_dram_bss_2:>7}', '{f_dram_bss_diff:+}', '')]
  411. if current.used_dram_other > 0 or reference.used_dram_other > 0:
  412. diff_list = ['+{}'.format(x) for x in current.dram_other_names - reference.dram_other_names]
  413. diff_list += ['-{}'.format(x) for x in reference.dram_other_names - current.dram_other_names]
  414. other_diff_str = '' if len(diff_list) == 0 else '({})'.format(', '.join(sorted(diff_list)))
  415. rows += [(' DRAM other size: {f_dram_other:>7} bytes ' + '({})'.format(', '.join(current.dram_other_names)),
  416. '{f_dram_other_2:>7}',
  417. '{f_dram_other_diff:+}',
  418. other_diff_str)]
  419. rows += [('Used static DRAM: {f_used_dram:>7} bytes ({f_dram_avail:>7} available, '
  420. '{f_used_dram_ratio:.1%} used)',
  421. '{f_used_dram_2:>7}',
  422. '{f_used_dram_diff:+}',
  423. '({f_dram_avail_diff:>+7} available, {f_dram_total_diff:>+7} total)')]
  424. rows += [('Used static IRAM: {f_used_iram:>7} bytes ({f_iram_avail:>7} available, '
  425. '{f_used_iram_ratio:.1%} used)',
  426. '{f_used_iram_2:>7}',
  427. '{f_used_iram_diff:+}',
  428. '({f_iram_avail_diff:>+7} available, {f_iram_total_diff:>+7} total)')]
  429. if current.total_diram > 0 or reference.total_diram > 0:
  430. rows += [('Used stat D/IRAM: {f_used_diram:>7} bytes ({f_diram_avail:>7} available, '
  431. '{f_used_diram_ratio:.1%} used)',
  432. '{f_used_diram_2:>7}',
  433. '{f_used_diram_diff:+}',
  434. '({f_diram_avail_diff:>+7} available, {f_diram_total_diff:>+7} total)')]
  435. rows += [(' Flash code: {f_flash_code:>7} bytes',
  436. '{f_flash_code_2:>7}',
  437. '{f_flash_code_diff:+}',
  438. '')]
  439. rows += [(' Flash rodata: {f_flash_rodata:>7} bytes',
  440. '{f_flash_rodata_2:>7}',
  441. '{f_flash_rodata_diff:+}',
  442. '')]
  443. rows += [('Total image size:~{f_total_size:>7} bytes (.bin may be padded larger)',
  444. '{f_total_size_2:>7}',
  445. '{f_total_size_diff:+}',
  446. '')]
  447. f_dic = {'f_dram_data': current.used_dram_data + current.used_diram_data,
  448. 'f_dram_bss': current.used_dram_bss + current.used_diram_bss,
  449. 'f_dram_other': current.used_dram_other,
  450. 'f_used_dram': current.used_dram,
  451. 'f_dram_avail': current.total_dram - current.used_dram,
  452. 'f_used_dram_ratio': current.used_dram_ratio,
  453. 'f_used_iram': current.used_iram,
  454. 'f_iram_avail': current.total_iram - current.used_iram,
  455. 'f_used_iram_ratio': current.used_iram_ratio,
  456. 'f_used_diram': current.used_diram,
  457. 'f_diram_avail': current.total_diram - current.used_diram,
  458. 'f_used_diram_ratio': current.used_diram_ratio,
  459. 'f_flash_code': current.flash_code,
  460. 'f_flash_rodata': current.flash_rodata,
  461. 'f_total_size': current.total_size,
  462. 'f_dram_data_2': reference.used_dram_data + reference.used_diram_data,
  463. 'f_dram_bss_2': reference.used_dram_bss + reference.used_diram_bss,
  464. 'f_dram_other_2': reference.used_dram_other,
  465. 'f_used_dram_2': reference.used_dram,
  466. 'f_used_iram_2': reference.used_iram,
  467. 'f_used_diram_2': reference.used_diram,
  468. 'f_flash_code_2': reference.flash_code,
  469. 'f_flash_rodata_2': reference.flash_rodata,
  470. 'f_total_size_2': reference.total_size,
  471. 'f_dram_total_diff': current.total_dram - reference.total_dram,
  472. 'f_iram_total_diff': current.total_iram - reference.total_iram,
  473. 'f_diram_total_diff': current.total_diram - reference.total_diram,
  474. 'f_dram_data_diff': current.used_dram_data + current.used_diram_data - (reference.used_dram_data +
  475. reference.used_diram_data),
  476. 'f_dram_bss_diff': current.used_dram_bss + current.used_diram_bss - (reference.used_dram_bss +
  477. reference.used_diram_bss),
  478. 'f_dram_other_diff': current.used_dram_other - reference.used_dram_other,
  479. 'f_used_dram_diff': current.used_dram - reference.used_dram,
  480. 'f_dram_avail_diff': current.total_dram - current.used_dram - (reference.total_dram -
  481. reference.used_dram),
  482. 'f_used_iram_diff': current.used_iram - reference.used_iram,
  483. 'f_iram_avail_diff': current.total_iram - current.used_iram - (reference.total_iram -
  484. reference.used_iram),
  485. 'f_used_diram_diff': current.used_diram - reference.used_diram,
  486. 'f_diram_avail_diff': current.total_diram - current.used_diram - (reference.total_diram -
  487. reference.used_diram),
  488. 'f_flash_code_diff': current.flash_code - reference.flash_code,
  489. 'f_flash_rodata_diff': current.flash_rodata - reference.flash_rodata,
  490. 'f_total_size_diff': current.total_size - reference.total_size,
  491. }
  492. lf = '{:70}{:>15}{:>15} {}'
  493. output = os.linesep.join([lf.format(a.format(**f_dic),
  494. b.format(**f_dic) if diff_en else '',
  495. c.format(**f_dic) if (diff_en and
  496. not c.format(**f_dic).startswith('+0')) else '',
  497. d.format(**f_dic) if diff_en else ''
  498. ).rstrip() for a, b, c, d in rows])
  499. output += os.linesep # last line need to be terminated because it won't be printed otherwise
  500. return output
  501. class StructureForDetailedSizes(object):
  502. @staticmethod
  503. def sizes_by_key(sections, key):
  504. """ Takes a dict of sections (from load_sections) and returns
  505. a dict keyed by 'key' with aggregate output size information.
  506. Key can be either "archive" (for per-archive data) or "file" (for per-file data) in the result.
  507. """
  508. result = {}
  509. for _, section in iteritems(sections):
  510. for s in section["sources"]:
  511. if not s[key] in result:
  512. result[s[key]] = {}
  513. archive = result[s[key]]
  514. if not section["name"] in archive:
  515. archive[section["name"]] = 0
  516. archive[section["name"]] += s["size"]
  517. return result
  518. @staticmethod
  519. def get(mem_reg, sections, key):
  520. sizes = StructureForDetailedSizes.sizes_by_key(sections, key)
  521. # these sets are also computed in get_summary() but they are small ones so it should not matter
  522. dram_data_names = frozenset([n for n in mem_reg.used_dram_names if n.endswith('.data')])
  523. dram_bss_names = frozenset([n for n in mem_reg.used_dram_names if n.endswith('.bss')])
  524. dram_other_names = mem_reg.used_dram_names - dram_data_names - dram_bss_names
  525. diram_data_names = frozenset([n for n in mem_reg.used_diram_names if n.endswith('.data')])
  526. diram_bss_names = frozenset([n for n in mem_reg.used_diram_names if n.endswith('.bss')])
  527. s = []
  528. for k, v in iteritems(sizes):
  529. r = [('data', sum(v.get(n, 0) for n in dram_data_names | diram_data_names)),
  530. ('bss', sum(v.get(n, 0) for n in dram_bss_names | diram_bss_names)),
  531. ('other', sum(v.get(n, 0) for n in dram_other_names)),
  532. ('iram', sum(t for (s,t) in iteritems(v) if s in mem_reg.used_iram_names)),
  533. ('diram', sum(t for (s,t) in iteritems(v) if s in mem_reg.used_diram_names)),
  534. ('flash_text', v.get('.flash.text', 0)),
  535. ('flash_rodata', v.get('.flash.rodata', 0))]
  536. r.append(('total', sum([value for _, value in r])))
  537. s.append((k, collections.OrderedDict(r)))
  538. s = sorted(s, key=lambda elem: elem[0])
  539. # do a secondary sort in order to have consistent order (for diff-ing the output)
  540. s = sorted(s, key=lambda elem: elem[1]['total'], reverse=True)
  541. return collections.OrderedDict(s)
  542. def get_detailed_sizes(mem_reg, sections, key, header, as_json=False, sections_diff=None):
  543. diff_en = sections_diff is not None
  544. current = StructureForDetailedSizes.get(mem_reg, sections, key)
  545. reference = StructureForDetailedSizes.get(mem_reg, sections_diff, key) if diff_en else {}
  546. if as_json:
  547. if diff_en:
  548. diff_json_dic = collections.OrderedDict()
  549. for name in sorted(list(frozenset(current.keys()) | frozenset(reference.keys()))):
  550. cur_name_dic = current.get(name, {})
  551. ref_name_dic = reference.get(name, {})
  552. all_keys = sorted(list(frozenset(cur_name_dic.keys()) | frozenset(ref_name_dic.keys())))
  553. diff_json_dic[name] = collections.OrderedDict([(k,
  554. cur_name_dic.get(k, 0) -
  555. ref_name_dic.get(k, 0)) for k in all_keys])
  556. output = format_json(collections.OrderedDict([('current', current),
  557. ('reference', reference),
  558. ('diff', diff_json_dic),
  559. ]))
  560. else:
  561. output = format_json(current)
  562. else:
  563. def _get_output(data, selection):
  564. header_format = '{:>24} {:>10} {:>6} {:>7} {:>6} {:>8} {:>10} {:>8} {:>7}' + os.linesep
  565. output = header_format.format(header,
  566. 'DRAM .data',
  567. '& .bss',
  568. '& other',
  569. 'IRAM',
  570. 'D/IRAM',
  571. 'Flash code',
  572. '& rodata',
  573. 'Total')
  574. for k, v in iteritems(data):
  575. if k not in selection:
  576. continue
  577. try:
  578. _, k = k.split(':', 1)
  579. # print subheadings for key of format archive:file
  580. except ValueError:
  581. # k remains the same
  582. pass
  583. output += header_format.format(k[:24],
  584. v['data'],
  585. v['bss'],
  586. v['other'],
  587. v['iram'],
  588. v['diram'],
  589. v['flash_text'],
  590. v['flash_rodata'],
  591. v['total'],
  592. )
  593. return output
  594. def _get_output_diff(curr, ref):
  595. header_format = '{:>24}' + ' {:>23}' * 8
  596. output = header_format.format(header,
  597. 'DRAM .data',
  598. 'DRAM .bss',
  599. 'DRAM other',
  600. 'IRAM',
  601. 'D/IRAM',
  602. 'Flash code',
  603. 'Flash rodata',
  604. 'Total') + os.linesep
  605. f_print = ('-' * 23, '') * 4
  606. header_line = header_format.format('', *f_print).rstrip() + os.linesep
  607. header_format = '{:>24}' + '|{:>7}|{:>7}|{:>7}' * 8
  608. f_print = ('<C>', '<R>', '<C>-<R>') * 8
  609. output += header_format.format('', *f_print) + os.linesep
  610. output += header_line
  611. for k, v in iteritems(curr):
  612. try:
  613. v2 = ref[k]
  614. except KeyError:
  615. continue
  616. try:
  617. _, k = k.split(':', 1)
  618. # print subheadings for key of format archive:file
  619. except ValueError:
  620. # k remains the same
  621. pass
  622. def _get_items(name):
  623. a = v[name]
  624. b = v2[name]
  625. diff = a - b
  626. # the sign is added here and not in header_format in order to be able to print empty strings
  627. return (a or '', b or '', '' if diff == 0 else '{:+}'.format(diff))
  628. v_data, v2_data, diff_data = _get_items('data')
  629. v_bss, v2_bss, diff_bss = _get_items('bss')
  630. v_other, v2_other, diff_other = _get_items('other')
  631. v_iram, v2_iram, diff_iram = _get_items('iram')
  632. v_diram, v2_diram, diff_diram = _get_items('diram')
  633. v_flash_text, v2_flash_text, diff_flash_text = _get_items('flash_text')
  634. v_flash_rodata, v2_flash_rodata, diff_flash_rodata = _get_items('flash_rodata')
  635. v_total, v2_total, diff_total = _get_items('total')
  636. output += header_format.format(k[:24],
  637. v_data, v2_data, diff_data,
  638. v_bss, v2_bss, diff_bss,
  639. v_other, v2_other, diff_other,
  640. v_iram, v2_iram, diff_iram,
  641. v_diram, v2_diram, diff_diram,
  642. v_flash_text, v2_flash_text, diff_flash_text,
  643. v_flash_rodata, v2_flash_rodata, diff_flash_rodata,
  644. v_total, v2_total, diff_total,
  645. ).rstrip() + os.linesep
  646. return output
  647. output = 'Per-{} contributions to ELF file:{}'.format(key, os.linesep)
  648. if diff_en:
  649. output += _get_output_diff(current, reference)
  650. in_current = frozenset(current.keys())
  651. in_reference = frozenset(reference.keys())
  652. only_in_current = in_current - in_reference
  653. only_in_reference = in_reference - in_current
  654. if len(only_in_current) > 0:
  655. output += 'The following entries are present in <CURRENT> only:{}'.format(os.linesep)
  656. output += _get_output(current, only_in_current)
  657. if len(only_in_reference) > 0:
  658. output += 'The following entries are present in <REFERENCE> only:{}'.format(os.linesep)
  659. output += _get_output(reference, only_in_reference)
  660. else:
  661. output += _get_output(current, current)
  662. return output
  663. class StructureForArchiveSymbols(object):
  664. @staticmethod
  665. def get(mem_reg, archive, sections):
  666. interested_sections = mem_reg.used_dram_names | mem_reg.used_iram_names | mem_reg.used_diram_names
  667. interested_sections |= frozenset(['.flash.text', '.flash.rodata'])
  668. result = dict([(t, {}) for t in interested_sections])
  669. for _, section in iteritems(sections):
  670. section_name = section['name']
  671. if section_name not in interested_sections:
  672. continue
  673. for s in section['sources']:
  674. if archive != s['archive']:
  675. continue
  676. s['sym_name'] = re.sub('(.text.|.literal.|.data.|.bss.|.rodata.)', '', s['sym_name'])
  677. result[section_name][s['sym_name']] = result[section_name].get(s['sym_name'], 0) + s['size']
  678. # build a new ordered dict of each section, where each entry is an ordereddict of symbols to sizes
  679. section_symbols = collections.OrderedDict()
  680. for t in sorted(list(interested_sections)):
  681. s = sorted(list(result[t].items()), key=lambda k_v: k_v[0])
  682. # do a secondary sort in order to have consistent order (for diff-ing the output)
  683. s = sorted(s, key=lambda k_v: k_v[1], reverse=True)
  684. section_symbols[t] = collections.OrderedDict(s)
  685. return section_symbols
  686. def get_archive_symbols(mem_reg, sections, archive, as_json=False, sections_diff=None):
  687. diff_en = sections_diff is not None
  688. current = StructureForArchiveSymbols.get(mem_reg, archive, sections)
  689. reference = StructureForArchiveSymbols.get(mem_reg, archive, sections_diff) if diff_en else {}
  690. if as_json:
  691. if diff_en:
  692. diff_json_dic = collections.OrderedDict()
  693. for name in sorted(list(frozenset(current.keys()) | frozenset(reference.keys()))):
  694. cur_name_dic = current.get(name, {})
  695. ref_name_dic = reference.get(name, {})
  696. all_keys = sorted(list(frozenset(cur_name_dic.keys()) | frozenset(ref_name_dic.keys())))
  697. diff_json_dic[name] = collections.OrderedDict([(key,
  698. cur_name_dic.get(key, 0) -
  699. ref_name_dic.get(key, 0)) for key in all_keys])
  700. output = format_json(collections.OrderedDict([('current', current),
  701. ('reference', reference),
  702. ('diff', diff_json_dic),
  703. ]))
  704. else:
  705. output = format_json(current)
  706. else:
  707. def _get_item_pairs(name, section):
  708. return collections.OrderedDict([(key.replace(name + '.', ''), val) for key, val in iteritems(section)])
  709. def _get_output(section_symbols):
  710. output = ''
  711. for t, s in iteritems(section_symbols):
  712. output += '{}Symbols from section: {}{}'.format(os.linesep, t, os.linesep)
  713. item_pairs = _get_item_pairs(t, s)
  714. output += ' '.join(['{}({})'.format(key, val) for key, val in iteritems(item_pairs)])
  715. section_total = sum([val for _, val in iteritems(item_pairs)])
  716. output += '{}Section total: {}{}'.format(os.linesep if section_total > 0 else '',
  717. section_total,
  718. os.linesep)
  719. return output
  720. output = 'Symbols within the archive: {} (Not all symbols may be reported){}'.format(archive, os.linesep)
  721. if diff_en:
  722. def _generate_line_tuple(curr, ref, name):
  723. cur_val = curr.get(name, 0)
  724. ref_val = ref.get(name, 0)
  725. diff_val = cur_val - ref_val
  726. # string slicing is used just to make sure it will fit into the first column of line_format
  727. return ((' ' * 4 + name)[:40], cur_val, ref_val, '' if diff_val == 0 else '{:+}'.format(diff_val))
  728. line_format = '{:40} {:>12} {:>12} {:>25}'
  729. all_section_names = sorted(list(frozenset(current.keys()) | frozenset(reference.keys())))
  730. for section_name in all_section_names:
  731. current_item_pairs = _get_item_pairs(section_name, current.get(section_name, {}))
  732. reference_item_pairs = _get_item_pairs(section_name, reference.get(section_name, {}))
  733. output += os.linesep + line_format.format(section_name[:40],
  734. '<CURRENT>',
  735. '<REFERENCE>',
  736. '<CURRENT> - <REFERENCE>') + os.linesep
  737. current_section_total = sum([val for _, val in iteritems(current_item_pairs)])
  738. reference_section_total = sum([val for _, val in iteritems(reference_item_pairs)])
  739. diff_section_total = current_section_total - reference_section_total
  740. all_item_names = sorted(list(frozenset(current_item_pairs.keys()) |
  741. frozenset(reference_item_pairs.keys())))
  742. output += os.linesep.join([line_format.format(*_generate_line_tuple(current_item_pairs,
  743. reference_item_pairs,
  744. n)
  745. ).rstrip() for n in all_item_names])
  746. output += os.linesep if current_section_total > 0 or reference_section_total > 0 else ''
  747. output += line_format.format('Section total:',
  748. current_section_total,
  749. reference_section_total,
  750. '' if diff_section_total == 0 else '{:+}'.format(diff_section_total)
  751. ).rstrip() + os.linesep
  752. else:
  753. output += _get_output(current)
  754. return output
  755. if __name__ == '__main__':
  756. main()