| #!/usr/bin/env python3 | 
 |  | 
 | #  Print the percentage of instructions spent in each phase of QEMU | 
 | #  execution. | 
 | # | 
 | #  Syntax: | 
 | #  dissect.py [-h] -- <qemu executable> [<qemu executable options>] \ | 
 | #                   <target executable> [<target executable options>] | 
 | # | 
 | #  [-h] - Print the script arguments help message. | 
 | # | 
 | #  Example of usage: | 
 | #  dissect.py -- qemu-arm coulomb_double-arm | 
 | # | 
 | #  This file is a part of the project "TCG Continuous Benchmarking". | 
 | # | 
 | #  Copyright (C) 2020  Ahmed Karaman <ahmedkhaledkaraman@gmail.com> | 
 | #  Copyright (C) 2020  Aleksandar Markovic <aleksandar.qemu.devel@gmail.com> | 
 | # | 
 | #  This program is free software: you can redistribute it and/or modify | 
 | #  it under the terms of the GNU General Public License as published by | 
 | #  the Free Software Foundation, either version 2 of the License, or | 
 | #  (at your option) any later version. | 
 | # | 
 | #  This program is distributed in the hope that it will be useful, | 
 | #  but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 | #  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
 | #  GNU General Public License for more details. | 
 | # | 
 | #  You should have received a copy of the GNU General Public License | 
 | #  along with this program. If not, see <https://www.gnu.org/licenses/>. | 
 |  | 
 | import argparse | 
 | import os | 
 | import subprocess | 
 | import sys | 
 | import tempfile | 
 |  | 
 |  | 
 | def get_JIT_line(callgrind_data): | 
 |     """ | 
 |     Search for the first instance of the JIT call in | 
 |     the callgrind_annotate output when ran using --tree=caller | 
 |     This is equivalent to the self number of instructions of JIT. | 
 |  | 
 |     Parameters: | 
 |     callgrind_data (list): callgrind_annotate output | 
 |  | 
 |     Returns: | 
 |     (int): Line number | 
 |     """ | 
 |     line = -1 | 
 |     for i in range(len(callgrind_data)): | 
 |         if callgrind_data[i].strip('\n') and \ | 
 |                 callgrind_data[i].split()[-1] == "[???]": | 
 |             line = i | 
 |             break | 
 |     if line == -1: | 
 |         sys.exit("Couldn't locate the JIT call ... Exiting.") | 
 |     return line | 
 |  | 
 |  | 
 | def main(): | 
 |     # Parse the command line arguments | 
 |     parser = argparse.ArgumentParser( | 
 |         usage='dissect.py [-h] -- ' | 
 |         '<qemu executable> [<qemu executable options>] ' | 
 |         '<target executable> [<target executable options>]') | 
 |  | 
 |     parser.add_argument('command', type=str, nargs='+', help=argparse.SUPPRESS) | 
 |  | 
 |     args = parser.parse_args() | 
 |  | 
 |     # Extract the needed variables from the args | 
 |     command = args.command | 
 |  | 
 |     # Insure that valgrind is installed | 
 |     check_valgrind = subprocess.run( | 
 |         ["which", "valgrind"], stdout=subprocess.DEVNULL) | 
 |     if check_valgrind.returncode: | 
 |         sys.exit("Please install valgrind before running the script.") | 
 |  | 
 |     # Save all intermediate files in a temporary directory | 
 |     with tempfile.TemporaryDirectory() as tmpdirname: | 
 |         # callgrind output file path | 
 |         data_path = os.path.join(tmpdirname, "callgrind.data") | 
 |         # callgrind_annotate output file path | 
 |         annotate_out_path = os.path.join(tmpdirname, "callgrind_annotate.out") | 
 |  | 
 |         # Run callgrind | 
 |         callgrind = subprocess.run((["valgrind", | 
 |                                      "--tool=callgrind", | 
 |                                      "--callgrind-out-file=" + data_path] | 
 |                                     + command), | 
 |                                    stdout=subprocess.DEVNULL, | 
 |                                    stderr=subprocess.PIPE) | 
 |         if callgrind.returncode: | 
 |             sys.exit(callgrind.stderr.decode("utf-8")) | 
 |  | 
 |         # Save callgrind_annotate output | 
 |         with open(annotate_out_path, "w") as output: | 
 |             callgrind_annotate = subprocess.run( | 
 |                 ["callgrind_annotate", data_path, "--tree=caller"], | 
 |                 stdout=output, | 
 |                 stderr=subprocess.PIPE) | 
 |             if callgrind_annotate.returncode: | 
 |                 sys.exit(callgrind_annotate.stderr.decode("utf-8")) | 
 |  | 
 |         # Read the callgrind_annotate output to callgrind_data[] | 
 |         callgrind_data = [] | 
 |         with open(annotate_out_path, 'r') as data: | 
 |             callgrind_data = data.readlines() | 
 |  | 
 |         # Line number with the total number of instructions | 
 |         total_instructions_line_number = 20 | 
 |         # Get the total number of instructions | 
 |         total_instructions_line_data = \ | 
 |             callgrind_data[total_instructions_line_number] | 
 |         total_instructions = total_instructions_line_data.split()[0] | 
 |         total_instructions = int(total_instructions.replace(',', '')) | 
 |  | 
 |         # Line number with the JIT self number of instructions | 
 |         JIT_self_instructions_line_number = get_JIT_line(callgrind_data) | 
 |         # Get the JIT self number of instructions | 
 |         JIT_self_instructions_line_data = \ | 
 |             callgrind_data[JIT_self_instructions_line_number] | 
 |         JIT_self_instructions = JIT_self_instructions_line_data.split()[0] | 
 |         JIT_self_instructions = int(JIT_self_instructions.replace(',', '')) | 
 |  | 
 |         # Line number with the JIT self + inclusive number of instructions | 
 |         # It's the line above the first JIT call when running with --tree=caller | 
 |         JIT_total_instructions_line_number = JIT_self_instructions_line_number-1 | 
 |         # Get the JIT self + inclusive number of instructions | 
 |         JIT_total_instructions_line_data = \ | 
 |             callgrind_data[JIT_total_instructions_line_number] | 
 |         JIT_total_instructions = JIT_total_instructions_line_data.split()[0] | 
 |         JIT_total_instructions = int(JIT_total_instructions.replace(',', '')) | 
 |  | 
 |         # Calculate number of instructions in helpers and code generation | 
 |         helpers_instructions = JIT_total_instructions-JIT_self_instructions | 
 |         code_generation_instructions = total_instructions-JIT_total_instructions | 
 |  | 
 |         # Print results (Insert commas in large numbers) | 
 |         # Print total number of instructions | 
 |         print('{:<20}{:>20}\n'. | 
 |               format("Total Instructions:", | 
 |                      format(total_instructions, ','))) | 
 |         # Print code generation instructions and percentage | 
 |         print('{:<20}{:>20}\t{:>6.3f}%'. | 
 |               format("Code Generation:", | 
 |                      format(code_generation_instructions, ","), | 
 |                      (code_generation_instructions / total_instructions) * 100)) | 
 |         # Print JIT instructions and percentage | 
 |         print('{:<20}{:>20}\t{:>6.3f}%'. | 
 |               format("JIT Execution:", | 
 |                      format(JIT_self_instructions, ","), | 
 |                      (JIT_self_instructions / total_instructions) * 100)) | 
 |         # Print helpers instructions and percentage | 
 |         print('{:<20}{:>20}\t{:>6.3f}%'. | 
 |               format("Helpers:", | 
 |                      format(helpers_instructions, ","), | 
 |                      (helpers_instructions/total_instructions)*100)) | 
 |  | 
 |  | 
 | if __name__ == "__main__": | 
 |     main() |