#!/usr/bin/env python3
# SPDX-License-Identifier: GPL-2.0
#
# Copyright (C) xFusion Digital Technologies Co., Ltd., 2023
#
# Author: Wang Jinchao <[email protected]>
#
"""
A tool for comparing tcrypt speed test logs.
Please note that for such a comparison, stability depends
on whether we allow frequency to float or pin the frequency.
Both support tests for operations within one second and
cycles of operation.
For example, use it in the bash script below.
```bash
#!/bin/bash
# log file prefix
seq_num=0
# When sec=0, it will perform cycle tests;
# otherwise, it indicates the duration of a single test
sec=0
num_mb=8
mode=211
# base speed test
lsmod | grep pcrypt && modprobe -r pcrypt
dmesg -C
modprobe tcrypt alg="pcrypt(rfc4106(gcm(aes)))" type=3
modprobe tcrypt mode=${mode} sec=${sec} num_mb=${num_mb}
dmesg > ${seq_num}_base_dmesg.log
# new speed test
lsmod | grep pcrypt && modprobe -r pcrypt
dmesg -C
modprobe tcrypt alg="pcrypt(rfc4106(gcm(aes)))" type=3
modprobe tcrypt mode=${mode} sec=${sec} num_mb=${num_mb}
dmesg > ${seq_num}_new_dmesg.log
lsmod | grep pcrypt && modprobe -r pcrypt
tools/crypto/tcrypt/tcrypt_speed_compare.py \
${seq_num}_base_dmesg.log \
${seq_num}_new_dmesg.log \
>${seq_num}_compare.log
grep 'average' -A2 -B0 --group-separator="" ${seq_num}_compare.log
```
"""
import sys
import re
def parse_title(line):
pattern = r'tcrypt: testing speed of (.*?) (encryption|decryption)'
match = re.search(pattern, line)
if match:
alg = match.group(1)
op = match.group(2)
return alg, op
else:
return "", ""
def parse_item(line):
pattern_operations = r'\((\d+) bit key, (\d+) byte blocks\): (\d+) operations'
pattern_cycles = r'\((\d+) bit key, (\d+) byte blocks\): 1 operation in (\d+) cycles'
match = re.search(pattern_operations, line)
if match:
res = {
"bit_key": int(match.group(1)),
"byte_blocks": int(match.group(2)),
"operations": int(match.group(3)),
}
return res
match = re.search(pattern_cycles, line)
if match:
res = {
"bit_key": int(match.group(1)),
"byte_blocks": int(match.group(2)),
"cycles": int(match.group(3)),
}
return res
return None
def parse(filepath):
result = {}
alg, op = "", ""
with open(filepath, 'r') as file:
for line in file:
if not line:
continue
_alg, _op = parse_title(line)
if _alg:
alg, op = _alg, _op
if alg not in result:
result[alg] = {}
if op not in result[alg]:
result[alg][op] = []
continue
parsed_result = parse_item(line)
if parsed_result:
result[alg][op].append(parsed_result)
return result
def merge(base, new):
merged = {}
for alg in base.keys():
merged[alg] = {}
for op in base[alg].keys():
if op not in merged[alg]:
merged[alg][op] = []
for index in range(len(base[alg][op])):
merged_item = {
"bit_key": base[alg][op][index]["bit_key"],
"byte_blocks": base[alg][op][index]["byte_blocks"],
}
if "operations" in base[alg][op][index].keys():
merged_item["base_ops"] = base[alg][op][index]["operations"]
merged_item["new_ops"] = new[alg][op][index]["operations"]
else:
merged_item["base_cycles"] = base[alg][op][index]["cycles"]
merged_item["new_cycles"] = new[alg][op][index]["cycles"]
merged[alg][op].append(merged_item)
return merged
def format(merged):
for alg in merged.keys():
for op in merged[alg].keys():
base_sum = 0
new_sum = 0
differ_sum = 0
differ_cnt = 0
print()
hlen = 80
print("="*hlen)
print(f"{alg}")
print(f"{' '*(len(alg)//3) + op}")
print("-"*hlen)
key = ""
if "base_ops" in merged[alg][op][0]:
key = "ops"
print(f"bit key | byte blocks | base ops | new ops | differ(%)")
else:
key = "cycles"
print(f"bit key | byte blocks | base cycles | new cycles | differ(%)")
for index in range(len(merged[alg][op])):
item = merged[alg][op][index]
base_cnt = item[f"base_{key}"]
new_cnt = item[f"new_{key}"]
base_sum += base_cnt
new_sum += new_cnt
differ = round((new_cnt - base_cnt)*100/base_cnt, 2)
differ_sum += differ
differ_cnt += 1
bit_key = item["bit_key"]
byte_blocks = item["byte_blocks"]
print(
f"{bit_key:<7} | {byte_blocks:<11} | {base_cnt:<11} | {new_cnt:<11} | {differ:<8}")
average_speed_up = "{:.2f}".format(differ_sum/differ_cnt)
ops_total_speed_up = "{:.2f}".format(
(base_sum - new_sum) * 100 / base_sum)
print('-'*hlen)
print(f"average differ(%s) | total_differ(%)")
print('-'*hlen)
print(f"{average_speed_up:<21} | {ops_total_speed_up:<10}")
print('='*hlen)
def main(base_log, new_log):
base = parse(base_log)
new = parse(new_log)
merged = merge(base, new)
format(merged)
if __name__ == "__main__":
if len(sys.argv) != 3:
print(f"usage: {sys.argv[0]} base_log new_log")
exit(-1)
main(sys.argv[1], sys.argv[2])