commit - release 1
This commit is contained in:
@@ -0,0 +1,175 @@
|
||||
"""Performance report formatter for bench results."""
|
||||
from __future__ import annotations
|
||||
|
||||
|
||||
_DTYPE_BITS: dict[str, int] = {
|
||||
"f16": 16, "fp16": 16, "float16": 16, "bf16": 16,
|
||||
"f32": 32, "fp32": 32, "float32": 32,
|
||||
"i8": 8, "int8": 8, "i16": 16, "int16": 16, "i32": 32, "int32": 32,
|
||||
}
|
||||
|
||||
|
||||
def format_report(
|
||||
traces: list[dict],
|
||||
title: str = "Benchmark",
|
||||
spec: dict | None = None,
|
||||
) -> str:
|
||||
"""Format collected traces into a human-readable performance report.
|
||||
|
||||
spec: topology spec dict for peak TFLOPS / BW extraction.
|
||||
"""
|
||||
peak_tflops_f16, peak_hbm_bw_gbs = _extract_peaks(spec)
|
||||
num_pes = _count_pes(spec)
|
||||
|
||||
lines: list[str] = []
|
||||
title_line = f"-- {title} Performance Report "
|
||||
|
||||
deploy_entries = [t for t in traces if t.get("phase") not in ("kernel",)]
|
||||
kernel_entries = [t for t in traces if t.get("phase") == "kernel"]
|
||||
|
||||
# ── Title ──
|
||||
# Compute max header width for consistent separator lengths
|
||||
_cmd_hdr = (f"{'Cmd':<10} {'Name':<12} {'SIP':>4} {'Cube':>5} {'PE':>4} {'Bytes':>10} "
|
||||
f"{'Lat(ns)':>10} {'Xfer(ns)':>10} {'Proc(ns)':>10} "
|
||||
f"{'BW(GB/s)':>10} {'MinBW':>10} {'Util%':>7}")
|
||||
report_width = len(_cmd_hdr)
|
||||
lines.append(title_line + "-" * max(0, report_width - len(title_line)))
|
||||
|
||||
# ── Command summary ──
|
||||
if deploy_entries:
|
||||
lines.append("")
|
||||
hdr = (f"{'Cmd':<10} {'Name':<12} {'SIP':>4} {'Cube':>5} {'PE':>4} {'Bytes':>10} "
|
||||
f"{'Lat(ns)':>10} {'Xfer(ns)':>10} {'Proc(ns)':>10} "
|
||||
f"{'BW(GB/s)':>10} {'MinBW':>10} {'Util%':>7}")
|
||||
lines.append(hdr)
|
||||
lines.append("-" * len(hdr))
|
||||
for e in deploy_entries:
|
||||
lat = e.get("total_ns", 0.0)
|
||||
nb = e.get("nbytes", 0)
|
||||
sip = e.get("sip", "-")
|
||||
pe = e.get("pe", "-")
|
||||
cube = e.get("cube", "-")
|
||||
cmd = e.get("phase", "deploy")
|
||||
xfer_ns = e.get("xfer_ns", 0.0)
|
||||
proc_ns = lat - xfer_ns if xfer_ns > 0 else 0.0
|
||||
bw = nb / lat if lat > 0 else 0.0
|
||||
min_bw = nb / xfer_ns if xfer_ns > 0 else 0.0
|
||||
util = (xfer_ns / lat * 100) if lat > 0 and xfer_ns > 0 else 0.0
|
||||
lines.append(
|
||||
f"{cmd:<10} {e.get('name', '?'):<12} {str(sip):>4} {str(cube):>5} {str(pe):>4} {nb:>10} "
|
||||
f"{lat:>10.1f} {xfer_ns:>10.1f} {proc_ns:>10.1f} "
|
||||
f"{bw:>10.1f} {min_bw:>10.1f} {util:>6.1f}%"
|
||||
)
|
||||
|
||||
# ── Kernel summary ──
|
||||
if kernel_entries:
|
||||
lines.append("")
|
||||
k_hdr = (f"{'Phase':<10} {'Name':<12} {'PE':>4} {'E2E(ns)':>10} "
|
||||
f"{'PE(ns)':>10} {'DMA(ns)':>10} {'Comp(ns)':>10} "
|
||||
f"{'Bound':<8} {'TFLOPS':>8} {'Peak':>8} {'Util%':>7}")
|
||||
lines.append(k_hdr)
|
||||
lines.append("-" * len(k_hdr))
|
||||
for e in kernel_entries:
|
||||
e2e_ns = e.get("total_ns", 0.0)
|
||||
pe_ns = e.get("pe_exec_ns", e2e_ns)
|
||||
dma_ns = e.get("dma_ns", 0.0)
|
||||
compute_ns = e.get("compute_ns", 0.0)
|
||||
target_pe = e.get("target_pe", "-")
|
||||
scalars = e.get("scalars", [])
|
||||
pe_str = "all" if target_pe == "all" else str(target_pe)
|
||||
n_active = num_pes if target_pe == "all" else 1
|
||||
|
||||
# Bound indicator based on measured DMA vs compute time
|
||||
if dma_ns > 0 or compute_ns > 0:
|
||||
bound = "memory" if dma_ns >= compute_ns else "compute"
|
||||
else:
|
||||
bound = "-"
|
||||
|
||||
achieved = _calc_tflops(scalars, pe_ns)
|
||||
peak_total = peak_tflops_f16 * n_active
|
||||
util = (achieved / peak_total * 100) if peak_total > 0 else 0.0
|
||||
lines.append(
|
||||
f"{'kernel':<10} {e.get('name', '?'):<12} {pe_str:>4} {e2e_ns:>10.1f} "
|
||||
f"{pe_ns:>10.1f} {dma_ns:>10.1f} {compute_ns:>10.1f} "
|
||||
f"{bound:<8} {achieved:>8.3f} {peak_total:>8.1f} {util:>6.1f}%"
|
||||
)
|
||||
|
||||
# ── Per-PE summary ──
|
||||
pe_deploy = _per_pe_deploy(deploy_entries)
|
||||
if len(pe_deploy) > 1:
|
||||
lines.append("")
|
||||
pe_title = (f"-- Per-PE Summary (peak: {peak_tflops_f16:.1f} TFLOPS/PE, "
|
||||
f"{peak_hbm_bw_gbs:.0f} GB/s HBM BW) ")
|
||||
pe_hdr = (f"{'PE':>4} {'Deploy(ns)':>10} {'BW(GB/s)':>10} {'BW Util':>8} "
|
||||
f"{'Kernel(ns)':>10} {'TFLOPS':>8} {'Util':>7}")
|
||||
pe_width = max(len(pe_title), len(pe_hdr))
|
||||
lines.append(pe_title + "-" * max(0, pe_width - len(pe_title)))
|
||||
lines.append(pe_hdr)
|
||||
lines.append("-" * pe_width)
|
||||
|
||||
k_ns = sum(e.get("pe_exec_ns", e.get("total_ns", 0.0)) for e in kernel_entries)
|
||||
k_scalars = kernel_entries[0].get("scalars", []) if kernel_entries else []
|
||||
n_active = len(pe_deploy)
|
||||
total_achieved = _calc_tflops(k_scalars, k_ns)
|
||||
per_pe_tflops = total_achieved / n_active if n_active > 0 else 0.0
|
||||
pe_util = (per_pe_tflops / peak_tflops_f16 * 100) if peak_tflops_f16 > 0 else 0.0
|
||||
|
||||
for pe_id in sorted(pe_deploy):
|
||||
d_ns, d_bytes = pe_deploy[pe_id]
|
||||
d_bw = d_bytes / d_ns if d_ns > 0 else 0.0
|
||||
d_util = (d_bw / peak_hbm_bw_gbs * 100) if peak_hbm_bw_gbs > 0 else 0.0
|
||||
lines.append(
|
||||
f"{pe_id:>4} {d_ns:>10.1f} {d_bw:>10.1f} {d_util:>7.1f}% "
|
||||
f"{k_ns:>10.1f} {per_pe_tflops:>8.3f} {pe_util:>6.1f}%"
|
||||
)
|
||||
lines.append("")
|
||||
|
||||
return "\n".join(lines)
|
||||
|
||||
|
||||
def _extract_peaks(spec: dict | None) -> tuple[float, float]:
|
||||
"""Extract peak TFLOPS (f16) and HBM BW (GB/s) from spec."""
|
||||
if spec is None:
|
||||
return 0.0, 0.0
|
||||
cube = spec.get("cube", {})
|
||||
pe_template = cube.get("pe_template", {})
|
||||
comps = pe_template.get("components", {})
|
||||
gemm_attrs = comps.get("pe_gemm", {}).get("attrs", {})
|
||||
peak_tflops = float(gemm_attrs.get("peak_tflops_f16", 0.0))
|
||||
cube_links = cube.get("links", {})
|
||||
hbm_bw = float(cube_links.get("xbar_to_hbm_bw_gbs", 0.0))
|
||||
return peak_tflops, hbm_bw
|
||||
|
||||
|
||||
def _count_pes(spec: dict | None) -> int:
|
||||
if spec is None:
|
||||
return 8
|
||||
cube = spec.get("cube", {})
|
||||
layout = cube.get("pe_layout", {})
|
||||
per_corner = layout.get("pe_per_corner", 2)
|
||||
corners = len(layout.get("corners", ["NW", "NE", "SW", "SE"]))
|
||||
return per_corner * corners
|
||||
|
||||
|
||||
def _calc_tflops(scalars: list, latency_ns: float) -> float:
|
||||
"""Calculate achieved TFLOPS from scalar args [M, K, N] and latency."""
|
||||
if len(scalars) < 3 or latency_ns <= 0:
|
||||
return 0.0
|
||||
m, k, n = scalars[0], scalars[1], scalars[2]
|
||||
flops = 2.0 * m * k * n
|
||||
return flops / (latency_ns * 1e-9) / 1e12
|
||||
|
||||
|
||||
def _per_pe_deploy(deploy_entries: list[dict]) -> dict[int, tuple[float, int]]:
|
||||
"""Aggregate deploy latency and bytes per PE."""
|
||||
result: dict[int, tuple[float, int]] = {}
|
||||
for e in deploy_entries:
|
||||
pe = e.get("pe", 0)
|
||||
lat = e.get("total_ns", 0.0)
|
||||
nb = e.get("nbytes", 0)
|
||||
if pe in result:
|
||||
old_ns, old_bytes = result[pe]
|
||||
result[pe] = (old_ns + lat, old_bytes + nb)
|
||||
else:
|
||||
result[pe] = (lat, nb)
|
||||
return result
|
||||
Reference in New Issue
Block a user