from __future__ import annotations

from dataclasses import dataclass
from pathlib import Path

# Base directory for the repo (parent of veoplace/)
BASE_DIR = Path(__file__).resolve().parents[2]

# LEF file collections for each technology (in Enablements directory)
_NANGATE45_LEFS = tuple(
        Path(f"thirdparty/MacroPlacement/Enablements/NanGate45/lef/{f}")
        for f in [
                "NangateOpenCellLibrary.tech.lef",
                "NangateOpenCellLibrary.macro.mod.lef",
                "fakeram45_128x116.lef",
                "fakeram45_128x256.lef",
                "fakeram45_128x32.lef",
                "fakeram45_256x16.lef",
                "fakeram45_256x32.lef",
                "fakeram45_256x48.lef",
                "fakeram45_256x64.lef",
                "fakeram45_32x32.lef",
                "fakeram45_512x64.lef",
                "fakeram45_64x124.lef",
                "fakeram45_64x256.lef",
                "fakeram45_64x62.lef",
                "fakeram45_64x64.lef",
        ]
)

_ASAP7_LEFS = tuple(
        Path(f"thirdparty/MacroPlacement/Enablements/ASAP7/lef/{f}")
        for f in [
                "asap7_tech_1x_201209.lef",
                "asap7_tech_4x_201209.lef",
                "asap7sc7p5t_27_R_1x_201211.lef",
                "asap7sc7p5t_27_R_4x_201211.lef",
                "sram_asap7_16x256_1rw.lef",
                "sram_asap7_32x256_1rw.lef",
                "sram_asap7_64x256_1rw.lef",
                "sram_asap7_64x64_1rw.lef",
        ]
)

_SKY130HD_LEFS = tuple(
        Path(f"thirdparty/MacroPlacement/Enablements/SKY130HD/lef/{f}")
        for f in [
                "sky130_fd_sc_hd.tlef",
                "sky130_fd_sc_hd_merged.lef",
                "fakeram130_256x16.lef",
                "fakeram130_256x32.lef",
                "fakeram130_256x512.lef",
                "fakeram130_256x64.lef",
                "fakeram130_64x64.lef",
        ]
)


@dataclass(frozen=True)
class BenchmarkSpec:
    name: str
    format: str  # "bookshelf" | "lefdef"

    # LEF/DEF format (relative paths)
    def_file: Path | None = None
    lef_files: tuple[Path, ...] = ()
    verilog_file: Path | None = None  # Optional Verilog netlist for LEF/DEF

    # Bookshelf format (relative path)
    aux_file: Path | None = None

    # DREAMPlace config (relative path)
    dreamplace_config: Path | None = None

    # Color config (relative path)
    color_config: Path | None = None

    def abs_def_file(self) -> Path | None:
        return BASE_DIR / self.def_file if self.def_file else None

    def abs_lef_files(self) -> tuple[Path, ...]:
        return tuple(BASE_DIR / lef for lef in self.lef_files)

    def abs_verilog_file(self) -> Path | None:
        return BASE_DIR / self.verilog_file if self.verilog_file else None

    def abs_aux_file(self) -> Path | None:
        return BASE_DIR / self.aux_file if self.aux_file else None

    def abs_dreamplace_config(self) -> Path | None:
        return BASE_DIR / self.dreamplace_config if self.dreamplace_config else None

    def abs_color_config(self) -> Path | None:
        return BASE_DIR / self.color_config if self.color_config else None

    def abs_benchmark_dir(self) -> Path:
        """Get the directory containing the netlist files."""
        if self.format == "bookshelf":
            return (BASE_DIR / self.aux_file).parent
        elif self.format == "lefdef":
            return (BASE_DIR / self.def_file).parent
        else:
            raise ValueError(f"Unknown format: {self.format}")

    def validate(self) -> None:
        """Validate required fields for this format. Raises ValueError if invalid."""
        if self.format == "bookshelf":
            missing = []
            if not self.aux_file:
                missing.append("aux_file")
            if not self.dreamplace_config:
                missing.append("dreamplace_config")
            if missing:
                raise ValueError(
                        f"Bookshelf format '{self.name}' missing: {', '.join(missing)}"
                )
        elif self.format == "lefdef":
            missing = []
            if not self.def_file:
                missing.append("def_file")
            if not self.lef_files:
                missing.append("lef_files")
            if not self.dreamplace_config:
                missing.append("dreamplace_config")
            if missing:
                raise ValueError(
                        f"LEF/DEF format '{self.name}' missing: {', '.join(missing)}"
                )
        else:
            raise ValueError(
                    f"Unknown format '{self.format}' for '{self.name}'")


# Per-benchmark DREAMPlace parameter overrides
# These override the defaults when calling get_*_dreamplace_params()
DREAMPLACE_OVERRIDES = {
        # superblue* has K-Reorder boundary issues with detailed placement
        "superblue1_movable": {
                "detailed_place_flag": False,
        },
        "superblue3_movable": {
                "detailed_place_flag": False,
        },
        "superblue4_movable": {
                "detailed_place_flag": False,
        },
        "superblue5_movable": {
                "detailed_place_flag": False,
        },
        "superblue7_movable": {
                "detailed_place_flag": False,
        },
        "superblue10_movable": {
                "detailed_place_flag": False,
        },
        "superblue16_movable": {
                "detailed_place_flag": False,
        },
        "superblue18_movable": {
                "detailed_place_flag": False,
        },
        # ibm* has row height mismatch causing abacus_legalize assertion failure
        "ibm01": {
                "detailed_place_flag": False,
                "legalize_flag": False,
        },
        "ibm02": {
                "detailed_place_flag": False,
                "legalize_flag": False,
        },
        "ibm03": {
                "detailed_place_flag": False,
                "legalize_flag": False,
        },
        "ibm04": {
                "detailed_place_flag": False,
                "legalize_flag": False,
        },
}


def get_dreamplace_overrides(benchmark_id: str) -> dict:
    """Get DREAMPlace parameter overrides for a benchmark.

    Returns empty dict if no overrides are defined.
    """
    return DREAMPLACE_OVERRIDES.get(benchmark_id, {})


# All paths relative to repo root
BENCHMARKS = {
        # =========================================================================
        # ISPD2005free benchmarks (Bookshelf format)
        # =========================================================================
        "adaptec1": BenchmarkSpec(
                name="adaptec1",
                format="bookshelf",
                aux_file=Path(
                        "data/netlists/ispd2005free/adaptec1/adaptec1.aux"),
                dreamplace_config=Path("configs/adaptec1/dreamplace.json"),
                color_config=Path("configs/adaptec1/color_config.json"),
        ),
        "adaptec2": BenchmarkSpec(
                name="adaptec2",
                format="bookshelf",
                aux_file=Path(
                        "data/netlists/ispd2005free/adaptec2/adaptec2.aux"),
                dreamplace_config=Path("configs/adaptec2/dreamplace.json"),
                color_config=Path("configs/adaptec2/color_config.json"),
        ),
        "adaptec3": BenchmarkSpec(
                name="adaptec3",
                format="bookshelf",
                aux_file=Path(
                        "data/netlists/ispd2005free/adaptec3/adaptec3.aux"),
                dreamplace_config=Path("configs/adaptec3/dreamplace.json"),
                color_config=Path("configs/adaptec3/color_config.json"),
        ),
        "adaptec4": BenchmarkSpec(
                name="adaptec4",
                format="bookshelf",
                aux_file=Path(
                        "data/netlists/ispd2005free/adaptec4/adaptec4.aux"),
                dreamplace_config=Path("configs/adaptec4/dreamplace.json"),
                color_config=Path("configs/adaptec4/color_config.json"),
        ),
        "bigblue1": BenchmarkSpec(
                name="bigblue1",
                format="bookshelf",
                aux_file=Path(
                        "data/netlists/ispd2005free/bigblue1/bigblue1.aux"),
                dreamplace_config=Path("configs/bigblue1/dreamplace.json"),
                color_config=Path("configs/bigblue1/color_config.json"),
        ),
        "bigblue2": BenchmarkSpec(
                name="bigblue2",
                format="bookshelf",
                aux_file=Path(
                        "data/netlists/ispd2005free/bigblue2/bigblue2.aux"),
                dreamplace_config=Path("configs/bigblue2/dreamplace.json"),
                color_config=Path("configs/bigblue2/color_config.json"),
        ),
        "bigblue3": BenchmarkSpec(
                name="bigblue3",
                format="bookshelf",
                aux_file=Path(
                        "data/netlists/ispd2005free/bigblue3/bigblue3.aux"),
                dreamplace_config=Path("configs/bigblue3/dreamplace.json"),
                color_config=Path("configs/bigblue3/color_config.json"),
        ),
        "bigblue4": BenchmarkSpec(
                name="bigblue4",
                format="bookshelf",
                aux_file=Path(
                        "data/netlists/ispd2005free/bigblue4/bigblue4.aux"),
                dreamplace_config=Path("configs/bigblue4/dreamplace.json"),
                color_config=Path("configs/bigblue4/color_config.json"),
        ),
        "ibm01": BenchmarkSpec(
                name="ibm01",
                format="bookshelf",
                aux_file=Path("data/netlists/ispd2005free/ibm01/ibm01.aux"),
                dreamplace_config=Path("configs/ibm01/dreamplace.json"),
                color_config=Path("configs/ibm01/color_config.json"),
        ),
        "ibm02": BenchmarkSpec(
                name="ibm02",
                format="bookshelf",
                aux_file=Path("data/netlists/ispd2005free/ibm02/ibm02.aux"),
                dreamplace_config=Path("configs/ibm02/dreamplace.json"),
                color_config=Path("configs/ibm02/color_config.json"),
        ),
        "ibm03": BenchmarkSpec(
                name="ibm03",
                format="bookshelf",
                aux_file=Path("data/netlists/ispd2005free/ibm03/ibm03.aux"),
                dreamplace_config=Path("configs/ibm03/dreamplace.json"),
                color_config=Path("configs/ibm03/color_config.json"),
        ),
        "ibm04": BenchmarkSpec(
                name="ibm04",
                format="bookshelf",
                aux_file=Path("data/netlists/ispd2005free/ibm04/ibm04.aux"),
                dreamplace_config=Path("configs/ibm04/dreamplace.json"),
                color_config=Path("configs/ibm04/color_config.json"),
        ),

        # =========================================================================
        # NanGate45 benchmarks (LEF/DEF format from MacroPlacement)
        # =========================================================================
        "ariane133_nangate45": BenchmarkSpec(
                name="ariane133",
                format="lefdef",
                # Full ungrouped DEF with 87K components + 113K nets
                def_file=Path(
                        "thirdparty/MacroPlacement/CodeElements/FormatTranslators/test/LefDef2ProtocolBufferFormat/design/ariane.def"),
                lef_files=_NANGATE45_LEFS,
                dreamplace_config=Path(
                        "configs/ariane133_nangate45/dreamplace.json"),
                color_config=Path(
                        "configs/ariane133_nangate45/color_config.json"),
        ),
        # Note: nvdla, mempool_tile, mempool_group, bp_quad removed for all tech nodes
        # - No full DEF with nets available (only floorplan DEFs)

        # =========================================================================
        # ICCAD 2015 benchmarks (LEF/DEF format, timing-driven placement)
        # _movable versions have macros PLACED, IO pins FIXED
        # =========================================================================
        "superblue1_movable": BenchmarkSpec(
                name="superblue1_movable",
                format="lefdef",
                def_file=Path(
                        "data/netlists/iccad2015/superblue1/superblue1_movable.def"),
                lef_files=(Path(
                        "data/netlists/iccad2015/superblue1/superblue1.lef"),),
                verilog_file=Path(
                        "data/netlists/iccad2015/superblue1/superblue1.v"),
                dreamplace_config=Path(
                        "configs/superblue1_movable/dreamplace.json"),
        ),
        "superblue3_movable": BenchmarkSpec(
                name="superblue3_movable",
                format="lefdef",
                def_file=Path(
                        "data/netlists/iccad2015/superblue3/superblue3_movable.def"),
                lef_files=(Path(
                        "data/netlists/iccad2015/superblue3/superblue3.lef"),),
                verilog_file=Path(
                        "data/netlists/iccad2015/superblue3/superblue3.v"),
                dreamplace_config=Path(
                        "configs/superblue3_movable/dreamplace.json"),
        ),
        "superblue4_movable": BenchmarkSpec(
                name="superblue4_movable",
                format="lefdef",
                def_file=Path(
                        "data/netlists/iccad2015/superblue4/superblue4_movable.def"),
                lef_files=(Path(
                        "data/netlists/iccad2015/superblue4/superblue4.lef"),),
                verilog_file=Path(
                        "data/netlists/iccad2015/superblue4/superblue4.v"),
                dreamplace_config=Path(
                        "configs/superblue4_movable/dreamplace.json"),
        ),
        "superblue5_movable": BenchmarkSpec(
                name="superblue5_movable",
                format="lefdef",
                def_file=Path(
                        "data/netlists/iccad2015/superblue5/superblue5_movable.def"),
                lef_files=(Path(
                        "data/netlists/iccad2015/superblue5/superblue5.lef"),),
                verilog_file=Path(
                        "data/netlists/iccad2015/superblue5/superblue5.v"),
                dreamplace_config=Path(
                        "configs/superblue5_movable/dreamplace.json"),
        ),
        "superblue7_movable": BenchmarkSpec(
                name="superblue7_movable",
                format="lefdef",
                def_file=Path(
                        "data/netlists/iccad2015/superblue7/superblue7_movable.def"),
                lef_files=(Path(
                        "data/netlists/iccad2015/superblue7/superblue7.lef"),),
                verilog_file=Path(
                        "data/netlists/iccad2015/superblue7/superblue7.v"),
                dreamplace_config=Path(
                        "configs/superblue7_movable/dreamplace.json"),
        ),
        "superblue10_movable": BenchmarkSpec(
                name="superblue10_movable",
                format="lefdef",
                def_file=Path(
                        "data/netlists/iccad2015/superblue10/superblue10_movable.def"),
                lef_files=(Path(
                        "data/netlists/iccad2015/superblue10/superblue10.lef"),),
                verilog_file=Path(
                        "data/netlists/iccad2015/superblue10/superblue10.v"),
                dreamplace_config=Path(
                        "configs/superblue10_movable/dreamplace.json"),
        ),
        "superblue16_movable": BenchmarkSpec(
                name="superblue16_movable",
                format="lefdef",
                def_file=Path(
                        "data/netlists/iccad2015/superblue16/superblue16_movable.def"),
                lef_files=(Path(
                        "data/netlists/iccad2015/superblue16/superblue16.lef"),),
                verilog_file=Path(
                        "data/netlists/iccad2015/superblue16/superblue16.v"),
                dreamplace_config=Path(
                        "configs/superblue16_movable/dreamplace.json"),
        ),
        "superblue18_movable": BenchmarkSpec(
                name="superblue18_movable",
                format="lefdef",
                def_file=Path(
                        "data/netlists/iccad2015/superblue18/superblue18_movable.def"),
                lef_files=(Path(
                        "data/netlists/iccad2015/superblue18/superblue18.lef"),),
                verilog_file=Path(
                        "data/netlists/iccad2015/superblue18/superblue18.v"),
                dreamplace_config=Path(
                        "configs/superblue18_movable/dreamplace.json"),
        ),
}


def get(benchmark_id: str) -> BenchmarkSpec:
    """Get benchmark spec by ID."""
    if benchmark_id not in BENCHMARKS:
        available = ", ".join(sorted(BENCHMARKS.keys()))
        raise KeyError(
                f"Unknown benchmark '{benchmark_id}'. Available: {available}")
    return BENCHMARKS[benchmark_id]


def get_all() -> dict[str, BenchmarkSpec]:
    """Get all benchmarks."""
    return BENCHMARKS


def config_dir(benchmark_id: str) -> Path:
    """Get the config directory for a benchmark. All derived files go here."""
    return BASE_DIR / "configs" / benchmark_id
