|  | #!/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() |