log_database.py 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. #!/usr/bin/env python3
  2. #
  3. # Copyright (c) 2021 Intel Corporation
  4. #
  5. # SPDX-License-Identifier: Apache-2.0
  6. """
  7. Class for Dictionary-based Logging Database
  8. """
  9. import base64
  10. import copy
  11. import json
  12. from .utils import extract_string_from_section
  13. ARCHS = {
  14. "arc" : {
  15. "kconfig": "CONFIG_ARC",
  16. },
  17. "arm" : {
  18. "kconfig": "CONFIG_ARM",
  19. },
  20. "arm64" : {
  21. "kconfig": "CONFIG_ARM64",
  22. },
  23. "sparc" : {
  24. "kconfig": "CONFIG_SPARC",
  25. },
  26. "x86" : {
  27. "kconfig": "CONFIG_X86",
  28. },
  29. "nios2" : {
  30. "kconfig": "CONFIG_NIOS2",
  31. # Small static strings are put into section "datas"
  32. # so we need to include them also.
  33. #
  34. # See include/arch/nios2/linker.ld on .sdata.*
  35. # for explanation.
  36. "extra_string_section": ['datas'],
  37. },
  38. "riscv" : {
  39. "kconfig": "CONFIG_RISCV",
  40. },
  41. "xtensa" : {
  42. "kconfig": "CONFIG_XTENSA",
  43. },
  44. }
  45. class LogDatabase():
  46. """Class of log database"""
  47. # Update this if binary format of dictionary based logging
  48. # has changed
  49. ZEPHYR_DICT_LOG_VER = 1
  50. LITTLE_ENDIAN = True
  51. BIG_ENDIAN = False
  52. def __init__(self):
  53. new_db = dict()
  54. new_db['version'] = self.ZEPHYR_DICT_LOG_VER
  55. new_db['target'] = dict()
  56. new_db['sections'] = dict()
  57. new_db['log_subsys'] = dict()
  58. new_db['log_subsys']['log_instances'] = dict()
  59. new_db['build_id'] = None
  60. new_db['arch'] = None
  61. new_db['kconfigs'] = dict()
  62. self.database = new_db
  63. def get_version(self):
  64. """Get Database Version"""
  65. return self.database['version']
  66. def get_build_id(self):
  67. """Get Build ID"""
  68. return self.database['build_id']
  69. def set_build_id(self, build_id):
  70. """Set Build ID in Database"""
  71. self.database['build_id'] = build_id
  72. def get_arch(self):
  73. """Get the Target Architecture"""
  74. return self.database['arch']
  75. def set_arch(self, arch):
  76. """Set the Target Architecture"""
  77. self.database['arch'] = arch
  78. def get_tgt_bits(self):
  79. """Get Target Bitness: 32 or 64"""
  80. if 'bits' in self.database['target']:
  81. return self.database['target']['bits']
  82. return None
  83. def set_tgt_bits(self, bits):
  84. """Set Target Bitness: 32 or 64"""
  85. self.database['target']['bits'] = bits
  86. def is_tgt_64bit(self):
  87. """Return True if target is 64-bit, False if 32-bit.
  88. None if error."""
  89. if 'bits' not in self.database['target']:
  90. return None
  91. if self.database['target']['bits'] == 32:
  92. return False
  93. if self.database['target']['bits'] == 64:
  94. return True
  95. return None
  96. def get_tgt_endianness(self):
  97. """
  98. Get Target Endianness.
  99. Return True if little endian, False if big.
  100. """
  101. if 'little_endianness' in self.database['target']:
  102. return self.database['target']['little_endianness']
  103. return None
  104. def set_tgt_endianness(self, endianness):
  105. """
  106. Set Target Endianness
  107. True if little endian, False if big.
  108. """
  109. self.database['target']['little_endianness'] = endianness
  110. def is_tgt_little_endian(self):
  111. """Return True if target is little endian"""
  112. if 'little_endianness' not in self.database['target']:
  113. return None
  114. return self.database['target']['little_endianness'] == self.LITTLE_ENDIAN
  115. def add_string_section(self, name, sect_dict):
  116. """Add a static string section to the collection"""
  117. self.database['sections'][name] = sect_dict
  118. def has_string_sections(self):
  119. """Return True if there are any static string sections"""
  120. return len(self.database['sections']) != 0
  121. def find_string(self, string_ptr):
  122. """Find string pointed by string_ptr from any static string section.
  123. Return None if not found."""
  124. for _, sect in self.database['sections'].items():
  125. one_str = extract_string_from_section(sect, string_ptr)
  126. if one_str is not None:
  127. return one_str
  128. return None
  129. def add_log_instance(self, source_id, name, level, address):
  130. """Add one log instance into database"""
  131. self.database['log_subsys']['log_instances'][source_id] = {
  132. 'source_id' : source_id,
  133. 'name' : name,
  134. 'level' : level,
  135. 'addr' : address,
  136. }
  137. def get_log_source_string(self, domain_id, source_id):
  138. """Get the source string based on source ID"""
  139. # JSON stores key as string, so we need to convert
  140. src_id = str(source_id)
  141. if src_id in self.database['log_subsys']['log_instances']:
  142. return self.database['log_subsys']['log_instances'][src_id]['name']
  143. return f"unknown<{domain_id}:{source_id}>"
  144. def add_kconfig(self, name, val):
  145. """Add a kconfig name-value pair into database"""
  146. self.database['kconfigs'][name] = val
  147. def get_kconfigs(self):
  148. """Return kconfig name-value pairs"""
  149. return self.database['kconfigs']
  150. @staticmethod
  151. def read_json_database(db_file_name):
  152. """Read database from file and return a LogDatabase object"""
  153. try:
  154. with open(db_file_name, "r") as db_fd:
  155. json_db = json.load(db_fd)
  156. except (OSError, json.JSONDecodeError):
  157. return None
  158. # Decode data in JSON back into binary data
  159. for _, sect in json_db['sections'].items():
  160. sect['data'] = base64.b64decode(sect['data_b64'])
  161. database = LogDatabase()
  162. database.database = json_db
  163. return database
  164. @staticmethod
  165. def write_json_database(db_file_name, database):
  166. """Write the database into file"""
  167. json_db = copy.deepcopy(database.database)
  168. # Make database object into something JSON can dump
  169. for _, sect in json_db['sections'].items():
  170. encoded = base64.b64encode(sect['data'])
  171. sect['data_b64'] = encoded.decode('ascii')
  172. del sect['data']
  173. try:
  174. with open(db_file_name, "w") as db_fd:
  175. db_fd.write(json.dumps(json_db))
  176. except OSError:
  177. return False
  178. return True