Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[tuner] add the output file #815

Merged
merged 7 commits into from
Jan 14, 2025
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
26 changes: 23 additions & 3 deletions tuner/examples/simple/simple_tuner.py
Original file line number Diff line number Diff line change
Expand Up @@ -80,7 +80,7 @@ def main():
stop_after_phase: str = args.stop_after

print("Setup logging")
libtuner.setup_logging(args, path_config)
root_logger = libtuner.setup_logging(args, path_config)
print(path_config.run_log, end="\n\n")

if not args.dry_run:
Expand All @@ -93,8 +93,16 @@ def main():
args.simple_model_benchmark_flags_file
)

summary_log_file = path_config.base_dir / "summary.log"
path_config.set_summary_log(summary_log_file)
summary_handler = logging.FileHandler(summary_log_file)
summary_handler.setLevel(logging.INFO)
summary_handler.setFormatter(
logging.Formatter("%(asctime)s - %(levelname)s - %(message)s")
)
print("Generating candidate tuning specs...")
with TunerContext() as tuner_context:
with TunerContext(logger=root_logger) as tuner_context:
tuner_context.add_logging_handler(summary_handler)
simple_tuner = SimpleTuner(tuner_context)
candidates = libtuner.generate_candidate_specs(
args, path_config, candidate_trackers, simple_tuner
Expand All @@ -114,15 +122,20 @@ def main():
return

print("Benchmarking compiled dispatch candidates...")
logging.info(f"Summarization about top dispatch candidates:")
bangtianliu marked this conversation as resolved.
Show resolved Hide resolved
simple_tuner.benchmark_flags = ["--input=1", "--benchmark_repetitions=3"]
top_candidates = libtuner.benchmark(
tuner_context,
args,
path_config,
compiled_candidates,
candidate_trackers,
simple_tuner,
args.simple_num_dispatch_candidates,
)
logging.info(f"Top dispatch candidates: {top_candidates}")
for id in top_candidates:
logging.info(f"{candidate_trackers[id].spec_path.resolve()}")
if stop_after_phase == libtuner.ExecutionPhases.benchmark_dispatches:
return

Expand All @@ -141,18 +154,25 @@ def main():
return

print("Benchmarking compiled model candidates...")
logging.info(f"Summarization about top model candidates:")
bangtianliu marked this conversation as resolved.
Show resolved Hide resolved
simple_tuner.benchmark_flags = model_benchmark_flags
simple_tuner.benchmark_timeout = 60
top_model_candidates = libtuner.benchmark(
tuner_context,
args,
path_config,
compiled_model_candidates,
candidate_trackers,
simple_tuner,
args.simple_num_model_candidates,
)

logging.info(f"Top model candidates: {top_model_candidates}")
for id in top_model_candidates:
logging.info(f"{candidate_trackers[id].spec_path.resolve()}")
print(f"Top model candidates: {top_model_candidates}")

print("Check the detailed execution logs in:")
print(path_config.run_log.resolve())

print("Check the tuning results in:")
bangtianliu marked this conversation as resolved.
Show resolved Hide resolved
print(path_config.summary_log.resolve())
4 changes: 2 additions & 2 deletions tuner/tuner/candidate_gen.py
Original file line number Diff line number Diff line change
Expand Up @@ -195,12 +195,12 @@ def generate_configs_and_td_specs(
):
if i >= limit:
break
tune_logger.info(f"Solution #{i+1}: {config}")
tune_logger.debug(f"Solution #{i+1}: {config}")
td_spec_module = dispatch_tuner.get_td_spec(input_module, config)
assert td_spec_module, "Failed to generate transform dialect spec"
config_specs.append(td_spec_module)

tune_logger.info(f"Generated {len(config_specs)} tuning specs")
tune_logger.debug(f"Generated {len(config_specs)} tuning specs")
return config_specs


Expand Down
3 changes: 3 additions & 0 deletions tuner/tuner/common.py
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,9 @@ def __enter__(self) -> "TunerContext":
self.mlir_ctx.__enter__()
return self

def add_logging_handler(self, handler: logging.Handler) -> None:
self.logger.addHandler(handler)
bangtianliu marked this conversation as resolved.
Show resolved Hide resolved

def __exit__(
self,
exc_type: type[BaseException] | None,
Expand Down
2 changes: 1 addition & 1 deletion tuner/tuner/dispatch_constraints.py
Original file line number Diff line number Diff line change
Expand Up @@ -376,7 +376,7 @@ def generate_solutions(
codegen_pipeline,
)
M, N, K = problem_size.MNK
tuner_ctx.logger.info(f"{M},{N},{K}")
tuner_ctx.logger.debug(f"{M},{N},{K}")
m_vars = [z3.Int(f"m{i}") for i in range(len(M))]
n_vars = [z3.Int(f"n{i}") for i in range(len(N))]
k_vars = [z3.Int(f"k{i}") for i in range(len(K))]
Expand Down
2 changes: 1 addition & 1 deletion tuner/tuner/dispatch_parser.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ def parse_mlir(mlir_text: str, ctx: TunerContext) -> ir.Module:
mlir_module = None
try:
mlir_module = ir.Module.parse(mlir_text, ctx.mlir_ctx)
ctx.logger.info("MLIR parsing successful!")
ctx.logger.debug("MLIR parsing successful!")
except ir.MLIRError as e:
ctx.logger.error(f"Error parsing MLIR: {e}")
raise RuntimeError(f"Error parsing MLIR: {e}")
Expand Down
36 changes: 26 additions & 10 deletions tuner/tuner/libtuner.py
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,7 @@ class PathConfig:

# To be set outside of class
run_log: Optional[Path] = field(init=False, default=None)
summary_log: Optional[Path] = field(init=False, default=None)

def __post_init__(self):
object.__setattr__(self, "base_dir", self._name_base_dir())
Expand All @@ -92,9 +93,12 @@ def _name_base_dir(self) -> Path:
base_dir = Path(f"./tuning_{timestamp}")
return base_dir

def _set_run_log(self, run_log: Path):
def set_run_log(self, run_log: Path):
object.__setattr__(self, "run_log", run_log)

def set_summary_log(self, summary_log: Path):
object.__setattr__(self, "summary_log", summary_log)
bangtianliu marked this conversation as resolved.
Show resolved Hide resolved

def get_candidate_spec_filename(self, candidate_id: int) -> str:
return f"{candidate_id}_spec.mlir"

Expand Down Expand Up @@ -334,10 +338,10 @@ def parse_arguments(
return parser.parse_args()


def setup_logging(args: argparse.Namespace, path_config: PathConfig):
def setup_logging(args: argparse.Namespace, path_config: PathConfig) -> logging.Logger:
log_file_name = f"autotune_{args.input_file.stem}.log"
run_log_path = path_config.base_dir / log_file_name
path_config._set_run_log(run_log_path)
path_config.set_run_log(run_log_path)

# Create file handler for logging to a file
if path_config.run_log is None:
Expand Down Expand Up @@ -384,7 +388,9 @@ def format(self, record):
# Log all arguments
logging.debug(f"Input Arguments:")
for arg, value in vars(args).items():
tune_logger.info(f"{arg}: {value}")
logging.debug(f"{arg}: {value}")

return logging.getLogger()


def handle_error(
Expand Down Expand Up @@ -717,10 +723,18 @@ def generate_candidate_specs(
tune_logger.exception("Error in candidate_gen.py:")
raise

logging.info(f"Generated [{len(candidates) - 1}] candidates")
logging.debug(f"Generated [{len(candidates) - 1}] candidates")
return candidates


def get_compilation_success_rate(compiled_candiates: list[Optional[int]]) -> float:
if not compiled_candiates:
return 0.0
successful_candidates = [c for c in compiled_candiates if c is not None]
success_rate = float(len(successful_candidates)) / float(len(compiled_candiates))
bangtianliu marked this conversation as resolved.
Show resolved Hide resolved
return success_rate


def collision_handler(index_hash_list: list[tuple[int, str]]) -> tuple[bool, list[int]]:
"""If a collision is found, generate a list of new indexes. If no collision, `unique_indexes = []`"""
# Check if candidate produces tbe same .vmfb
Expand Down Expand Up @@ -800,11 +814,11 @@ def compile(
compiled_candidates = multiprocess_progress_wrapper(
num_worker=num_worker, task_list=task_list, function=run_iree_compile_command
)
compiled_candidates = [c for c in compiled_candidates if c is not None]
success_rate = float(len(compiled_candidates)) / float(len(candidates))
logging.info(
success_rate = get_compilation_success_rate(compiled_candidates)
logging.debug(
f"Successfully compiled [{len(compiled_candidates)}] candidates. Success rate: {success_rate:.2f}"
)
compiled_candidates = [c for c in compiled_candidates if c is not None]

# Remove duplicate vmfbs from the candidate list.
compiled_candidate_hashes = []
Expand All @@ -818,7 +832,7 @@ def compile(
if collision_detected:
compiled_candidates = unique_compiled_candidates

logging.info(f"Produced [{len(compiled_candidates)}] unique vmfbs")
logging.debug(f"Produced [{len(compiled_candidates)}] unique vmfbs")
return compiled_candidates


Expand Down Expand Up @@ -875,11 +889,13 @@ def get_speedup(result: BenchmarkResult) -> float:
speedup = f"{round(get_speedup(r) * 100, 2)}% of baseline"
else:
speedup = "baseline unavailable"
logging.info(f"Candidate {r.candidate_id} time: {r.time:.2f} ({speedup})")
result = f"Candidate {r.candidate_id} time: {r.time:.2f} ms ({speedup})"
logging.info(result)
return best_results


def benchmark(
tuner_context: TunerContext,
bangtianliu marked this conversation as resolved.
Show resolved Hide resolved
args: argparse.Namespace,
path_config: PathConfig,
compiled_candidates: list[int],
Expand Down
16 changes: 16 additions & 0 deletions tuner/tuner/libtuner_test.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,9 @@
import json
from subprocess import CompletedProcess
from unittest.mock import call, patch, MagicMock
from typing import Generator
from . import libtuner
from . import common

"""
Usage: python -m pytest libtuner_test.py
Expand Down Expand Up @@ -176,6 +178,20 @@ def test_validate_devices_with_invalid_device() -> None:
assert expected_call in mock_handle_error.call_args_list


def test_get_compilation_success_rate():
compiled_candidates = [0, None, 2, None, 4]
assert libtuner.get_compilation_success_rate(compiled_candidates) == 3.0 / 5.0

compiled_candidates = [0, 1, 2, 3, 4]
assert libtuner.get_compilation_success_rate(compiled_candidates) == 1.0

compiled_candidates = [None, None, None]
assert libtuner.get_compilation_success_rate(compiled_candidates) == 0.0
bangtianliu marked this conversation as resolved.
Show resolved Hide resolved

compiled_candidates = []
assert libtuner.get_compilation_success_rate(compiled_candidates) == 0.0


def test_select_best_benchmark_results() -> None:
candidate_results = [
libtuner.BenchmarkResult(1, 0.5, "hip://0"),
Expand Down
Loading