log_parser.py 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. #!/usr/bin/env python3
  2. #
  3. # Copyright (c) 2021 Intel Corporation
  4. #
  5. # SPDX-License-Identifier: Apache-2.0
  6. """
  7. Log Parser for Dictionary-based Logging
  8. This uses the JSON database file to decode the input binary
  9. log data and print the log messages.
  10. """
  11. import argparse
  12. import binascii
  13. import logging
  14. import sys
  15. import dictionary_parser
  16. from dictionary_parser.log_database import LogDatabase
  17. LOGGER_FORMAT = "%(message)s"
  18. logger = logging.getLogger("parser")
  19. LOG_HEX_SEP = "##ZLOGV1##"
  20. def parse_args():
  21. """Parse command line arguments"""
  22. argparser = argparse.ArgumentParser()
  23. argparser.add_argument("dbfile", help="Dictionary Logging Database file")
  24. argparser.add_argument("logfile", help="Log Data file")
  25. argparser.add_argument("--hex", action="store_true",
  26. help="Log Data file is in hexadecimal strings")
  27. argparser.add_argument("--rawhex", action="store_true",
  28. help="Log file only contains hexadecimal log data")
  29. argparser.add_argument("--debug", action="store_true",
  30. help="Print extra debugging information")
  31. return argparser.parse_args()
  32. def main():
  33. """Main function of log parser"""
  34. args = parse_args()
  35. # Setup logging for parser
  36. logging.basicConfig(format=LOGGER_FORMAT)
  37. if args.debug:
  38. logger.setLevel(logging.DEBUG)
  39. else:
  40. logger.setLevel(logging.INFO)
  41. # Read from database file
  42. database = LogDatabase.read_json_database(args.dbfile)
  43. if database is None:
  44. logger.error("ERROR: Cannot open database file: %s, exiting...", args.dbfile)
  45. sys.exit(1)
  46. # Open log data file for reading
  47. if args.hex:
  48. if args.rawhex:
  49. # Simply log file with only hexadecimal data
  50. logdata = dictionary_parser.utils.convert_hex_file_to_bin(args.logfile)
  51. else:
  52. hexdata = ''
  53. with open(args.logfile, "r") as hexfile:
  54. for line in hexfile.readlines():
  55. hexdata += line.strip()
  56. if LOG_HEX_SEP not in hexdata:
  57. logger.error("ERROR: Cannot find start of log data, exiting...")
  58. sys.exit(1)
  59. idx = hexdata.index(LOG_HEX_SEP) + len(LOG_HEX_SEP)
  60. hexdata = hexdata[idx:]
  61. if len(hexdata) % 2 != 0:
  62. # Make sure there are even number of characters
  63. idx = int(len(hexdata) / 2) * 2
  64. hexdata = hexdata[:idx]
  65. idx = 0
  66. while idx < len(hexdata):
  67. # When running QEMU via west or ninja, there may be additional
  68. # strings printed by QEMU, west or ninja (for example, QEMU
  69. # is terminated, or user interrupted, etc). So we need to
  70. # figure out where the end of log data stream by
  71. # trying to convert from hex to bin.
  72. idx += 2
  73. try:
  74. binascii.unhexlify(hexdata[:idx])
  75. except binascii.Error:
  76. idx -= 2
  77. break
  78. logdata = binascii.unhexlify(hexdata[:idx])
  79. else:
  80. logfile = open(args.logfile, "rb")
  81. if not logfile:
  82. logger.error("ERROR: Cannot open binary log data file: %s, exiting...", args.logfile)
  83. sys.exit(1)
  84. logdata = logfile.read()
  85. logfile.close()
  86. log_parser = dictionary_parser.get_parser(database)
  87. if log_parser is not None:
  88. logger.debug("# Build ID: %s", database.get_build_id())
  89. logger.debug("# Target: %s, %d-bit", database.get_arch(), database.get_tgt_bits())
  90. if database.is_tgt_little_endian():
  91. logger.debug("# Endianness: Little")
  92. else:
  93. logger.debug("# Endianness: Big")
  94. ret = log_parser.parse_log_data(logdata, debug=args.debug)
  95. if not ret:
  96. logger.error("ERROR: there were error(s) parsing log data")
  97. sys.exit(1)
  98. else:
  99. logger.error("ERROR: Cannot find a suitable parser matching database version!")
  100. sys.exit(1)
  101. if __name__ == "__main__":
  102. main()