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