{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "90f49387",
   "metadata": {},
   "outputs": [],
   "source": [
    "#!/usr/bin/env python3\n",
    "\n",
    "\n",
    "from collections import deque, defaultdict, Counter\n",
    "import math\n",
    "import random\n",
    "from statistics import mean, pstdev\n",
    "from typing import List, Set, Dict, Tuple, Optional\n",
    "\n",
    "# ----------------- Golomb rulers / placements -----------------\n",
    "\n",
    "def get_golomb_offsets_builtin(r: int) -> Optional[List[int]]:\n",
    "    \"\"\"Optional library of known offsets (marks) for some r.\"\"\"\n",
    "    rulers = {\n",
    "        3:  [0, 1, 3],\n",
    "        4:  [0, 1, 4, 6],\n",
    "        5:  [0, 1, 4, 9, 11],\n",
    "        6:  [0, 1, 4, 10, 12, 17],\n",
    "        7:  [0, 1, 4, 10, 18, 23, 25],\n",
    "        8:  [0, 1, 4, 9, 15, 22, 32, 34],\n",
    "        9:  [0, 1, 5, 12, 25, 27, 35, 41, 44],\n",
    "        10: [0, 1, 6, 10, 23, 26, 34, 41, 53, 55],\n",
    "        11: [0, 1, 4, 13, 28, 33, 47, 54, 64, 70, 72],\n",
    "        12: [0, 2, 6, 24, 29, 40, 43, 55, 68, 75, 76, 85],\n",
    "        13: [0, 2, 5, 25, 37, 43, 59, 70, 85, 89, 98, 99, 106],\n",
    "        14: [0, 4, 6, 20, 35, 52, 59, 77, 78, 86, 89, 99, 122, 127],\n",
    "        15: [0, 4, 20, 30, 57, 59, 62, 76, 100, 111, 123, 136, 144, 145, 151],\n",
    "        16: [0, 1, 4, 11, 26, 32, 56, 68, 76, 115, 117, 134, 150, 163, 168, 177],\n",
    "        17: [0, 5, 7, 17, 52, 56, 67, 80, 81, 100, 122, 138, 159, 165, 168, 191, 199],\n",
    "        18: [0, 2, 10, 22, 53, 56, 82, 83, 89, 98, 130, 148, 153, 167, 188, 192, 205, 216],\n",
    "        19: [0, 1, 6, 25, 32, 72, 100, 108, 120, 130, 153, 169, 187, 190, 204, 231, 233, 242, 246],\n",
    "        20: [0, 1, 8, 11, 68, 77, 94, 116, 121, 156, 158, 179, 194, 208, 212, 228, 240, 253, 259, 283],\n",
    "        21: [0, 2, 24, 56, 77, 82, 83, 95, 129, 144, 179, 186, 195, 255, 265, 285, 293, 296, 310, 329, 333],\n",
    "        22: [0, 1, 9, 14, 43, 70, 106, 122, 124, 128, 159, 179, 204, 223, 253, 263, 270, 291, 330, 341, 353, 356],\n",
    "        23: [0, 3, 7, 17, 61, 66, 91, 99, 114, 159, 171, 199, 200, 226, 235, 246, 277, 316, 329, 348, 350, 366, 372],\n",
    "        24: [0, 9, 33, 37, 38, 97, 122, 129, 140, 142, 152, 191, 205, 208, 252, 278, 286, 326, 332, 353, 368, 384, 403, 425],\n",
    "        25: [0, 12, 29, 39, 72, 91, 146, 157, 160, 161, 166, 191, 207, 214, 258, 290, 316, 354, 372, 394, 396, 431, 459, 467, 480],\n",
    "        26: [0, 1, 33, 83, 104, 110, 124, 163, 185, 200, 203, 249, 251, 258, 314, 318, 343, 356, 386, 430, 440, 456, 464, 475, 487, 492],\n",
    "    }\n",
    "    return rulers.get(r, None)\n",
    "\n",
    "def construct_unique_diffs_offsets(N: int, r: int, rng: random.Random, max_tries: int = 20000) -> List[int]:\n",
    "    \"\"\"\n",
    "    Construct offsets G = {g_0=0, g_1, ..., g_{r-1}} in Z_N such that\n",
    "    all absolute cyclic differences |g_i - g_j|_N are distinct (i<j).\n",
    "    This ensures t_max <= 1 for the cyclic placement H_i = {i + g : g in G}.\n",
    "    Works reliably when N >> r^2.\n",
    "    \"\"\"\n",
    "    G = [0]\n",
    "    used = set()  # used absolute diffs in {1,..., floor((N-1)/2]}\n",
    "    attempts = 0\n",
    "    while len(G) < r and attempts < max_tries:\n",
    "        attempts += 1\n",
    "        cand = rng.randrange(1, N)  # 0 is already in G\n",
    "        if cand in G:\n",
    "            continue\n",
    "        ok = True\n",
    "        new_diffs = []\n",
    "        for g in G:\n",
    "            d = (cand - g) % N\n",
    "            d_abs = d if d <= N - d else N - d\n",
    "            if d_abs == 0 or d_abs in used:\n",
    "                ok = False\n",
    "                break\n",
    "            new_diffs.append(d_abs)\n",
    "        if ok:\n",
    "            G.append(cand)\n",
    "            for d_abs in new_diffs:\n",
    "                used.add(d_abs)\n",
    "    if len(G) != r:\n",
    "        raise RuntimeError(f\"Failed to construct unique-difference offsets for r={r} with N={N}. \"\n",
    "                           f\"Try larger N or pass explicit `offsets`.\")\n",
    "    G.sort()\n",
    "    return G\n",
    "\n",
    "def get_offsets(N: int, r: int, rng: random.Random, offsets: Optional[List[int]]) -> List[int]:\n",
    "    \"\"\"Resolve offsets: explicit > builtin > construct unique-differences.\"\"\"\n",
    "    if offsets is not None:\n",
    "        if len(offsets) != r:\n",
    "            raise ValueError(\"`offsets` length must equal r.\")\n",
    "        if 0 not in offsets:\n",
    "            offsets = [0] + [x for x in offsets if x != 0]\n",
    "        return offsets\n",
    "    builtin = get_golomb_offsets_builtin(r)\n",
    "    if builtin is not None:\n",
    "        return builtin\n",
    "    return construct_unique_diffs_offsets(N, r, rng)\n",
    "\n",
    "# ----------------- Mappings -----------------\n",
    "\n",
    "def build_hosts_for_types(N: int, offsets: List[int]) -> List[Tuple[int, ...]]:\n",
    "    \"\"\"hosts_for_types[t] -> tuple of workers that host type t.\"\"\"\n",
    "    hosts = []\n",
    "    for t in range(N):\n",
    "        hosts.append(tuple((t - g) % N for g in offsets))\n",
    "    return hosts\n",
    "\n",
    "def build_types_for_workers(N: int, offsets: List[int]) -> List[Tuple[int, ...]]:\n",
    "    \"\"\"types_for_workers[w] -> tuple of r types stored by worker w in initial order.\"\"\"\n",
    "    types = []\n",
    "    for w in range(N):\n",
    "        types.append(tuple((w + g) % N for g in offsets))\n",
    "    return types\n",
    "\n",
    "# ----------------- Hopcroft–Karp -----------------\n",
    "\n",
    "class HopcroftKarp:\n",
    "    \"\"\"\n",
    "    Bipartite maximum matching (unit capacity on both sides):\n",
    "      - Left:   N data types (0..N-1)\n",
    "      - Right:  surviving workers' clones (capacity T per worker -> T clones per worker)\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self, adj: List[List[int]], n_left: int, n_right: int):\n",
    "        self.adj = adj\n",
    "        self.n_left = n_left\n",
    "        self.n_right = n_right\n",
    "        self.pairU = [-1] * n_left\n",
    "        self.pairV = [-1] * n_right\n",
    "        self.dist = [0] * n_left\n",
    "\n",
    "    def bfs(self) -> bool:\n",
    "        q = deque()\n",
    "        for u in range(self.n_left):\n",
    "            if self.pairU[u] == -1:\n",
    "                self.dist[u] = 0\n",
    "                q.append(u)\n",
    "            else:\n",
    "                self.dist[u] = float('inf')\n",
    "        reachable_free = False\n",
    "        while q:\n",
    "            u = q.popleft()\n",
    "            for v in self.adj[u]:\n",
    "                pu = self.pairV[v]\n",
    "                if pu == -1:\n",
    "                    reachable_free = True\n",
    "                else:\n",
    "                    if self.dist[pu] == float('inf'):\n",
    "                        self.dist[pu] = self.dist[u] + 1\n",
    "                        q.append(pu)\n",
    "        return reachable_free\n",
    "\n",
    "    def dfs(self, u: int) -> bool:\n",
    "        for v in self.adj[u]:\n",
    "            pu = self.pairV[v]\n",
    "            if pu == -1 or (self.dist[pu] == self.dist[u] + 1 and self.dfs(pu)):\n",
    "                self.pairU[u] = v\n",
    "                self.pairV[v] = u\n",
    "                return True\n",
    "        self.dist[u] = float('inf')\n",
    "        return False\n",
    "\n",
    "    def max_matching(self) -> int:\n",
    "        matching = 0\n",
    "        while self.bfs():\n",
    "            for u in range(self.n_left):\n",
    "                if self.pairU[u] == -1 and self.dfs(u):\n",
    "                    matching += 1\n",
    "        return matching\n",
    "\n",
    "# ----------------- Persistent resequencing helpers -----------------\n",
    "\n",
    "def init_sequences(N: int, offsets: List[int]) -> Tuple[List[List[int]], List[Dict[int,int]]]:\n",
    "    \"\"\"\n",
    "    Initialize per-worker persistent sequences seq[w] and pos_map[w]: type -> position.\n",
    "    seq[w] is an ordered list of the r types this worker holds (initial Golomb order).\n",
    "    \"\"\"\n",
    "    types_for_workers = build_types_for_workers(N, offsets)\n",
    "    seq = [list(tup) for tup in types_for_workers]\n",
    "    pos_map: List[Dict[int,int]] = []\n",
    "    for w in range(N):\n",
    "        pm = {t: i for i, t in enumerate(seq[w])}\n",
    "        pos_map.append(pm)\n",
    "    return seq, pos_map\n",
    "\n",
    "def commit_prefix(w: int, T: int, new_prefix: List[int],\n",
    "                  seq: List[List[int]], pos_map: List[Dict[int,int]]) -> None:\n",
    "    \"\"\"\n",
    "    Commit a new first-T order for worker w. Preserve the relative order\n",
    "    of the remaining r-T types after the prefix.\n",
    "    \"\"\"\n",
    "    old = seq[w]\n",
    "    used = set(new_prefix)\n",
    "    tail = [t for t in old if t not in used]\n",
    "    seq[w] = list(new_prefix) + tail\n",
    "    pos_map[w] = {t: i for i, t in enumerate(seq[w])}\n",
    "\n",
    "# ----------------- Matching builders -----------------\n",
    "\n",
    "def fixed_adjacency(\n",
    "    N: int,\n",
    "    T: int,\n",
    "    survivors: List[int],\n",
    "    hosts_for_types: List[Tuple[int, ...]],\n",
    "    seq: List[List[int]],\n",
    "    pos_map: List[Dict[int,int]],\n",
    ") -> Tuple[List[List[int]], Dict[int, Tuple[int,int]]]:\n",
    "    \"\"\"\n",
    "    Build adjacency where each clone (w,k) can ONLY compute the type at seq[w][k].\n",
    "    Returns (adj, clone_to_wk) with clone_to_wk[v] = (worker, k).\n",
    "    \"\"\"\n",
    "    L = len(survivors)\n",
    "    adj: List[List[int]] = [[] for _ in range(N)]\n",
    "    clone_to_wk: Dict[int, Tuple[int,int]] = {}\n",
    "    base = {w: i*T for i, w in enumerate(survivors)}\n",
    "    for i, w in enumerate(survivors):\n",
    "        for k in range(T):\n",
    "            v = i*T + k\n",
    "            clone_to_wk[v] = (w, k)\n",
    "    for t in range(N):\n",
    "        for w in hosts_for_types[t]:\n",
    "            if w in base:\n",
    "                k = pos_map[w].get(t, None)\n",
    "                if k is not None and k < T:\n",
    "                    adj[t].append(base[w] + k)\n",
    "    return adj, clone_to_wk\n",
    "\n",
    "def free_adjacency(\n",
    "    N: int,\n",
    "    T: int,\n",
    "    survivors: List[int],\n",
    "    hosts_for_types: List[Tuple[int, ...]],\n",
    ") -> Tuple[List[List[int]], Dict[int, Tuple[int,int]]]:\n",
    "    \"\"\"\n",
    "    Build adjacency where each clone (w,k) can compute ANY of its worker's hosted types.\n",
    "    This models a resequencing of the first T slots (to be committed afterward).\n",
    "    \"\"\"\n",
    "    L = len(survivors)\n",
    "    adj: List[List[int]] = [[] for _ in range(N)]\n",
    "    clone_to_wk: Dict[int, Tuple[int,int]] = {}\n",
    "    base = {w: i*T for i, w in enumerate(survivors)}\n",
    "    for i, w in enumerate(survivors):\n",
    "        for k in range(T):\n",
    "            v = i*T + k\n",
    "            clone_to_wk[v] = (w, k)\n",
    "    for t in range(N):\n",
    "        for w in hosts_for_types[t]:\n",
    "            if w in base:\n",
    "                b = base[w]\n",
    "                for k in range(T):\n",
    "                    adj[t].append(b + k)\n",
    "    return adj, clone_to_wk\n",
    "\n",
    "# ----------------- W1/W2 helpers -----------------\n",
    "\n",
    "def compute_pinned_loads(\n",
    "    N: int,\n",
    "    failed: Set[int],\n",
    "    hosts_for_types: List[Tuple[int, ...]]\n",
    ") -> Tuple[Dict[int, int], int, Optional[int]]:\n",
    "    \"\"\"\n",
    "    For current failure set, compute per-survivor pinned loads:\n",
    "    O_w = # types whose only surviving host is w.\n",
    "    Returns (O, max_O, argmax_w) where O is dict for all survivors.\n",
    "    \"\"\"\n",
    "    survivors = set(range(N)) - failed\n",
    "    O: Dict[int, int] = {w: 0 for w in survivors}\n",
    "    for t in range(N):\n",
    "        live_hosts = [h for h in hosts_for_types[t] if h in survivors]\n",
    "        if len(live_hosts) == 1:\n",
    "            O[live_hosts[0]] += 1\n",
    "    max_w = None\n",
    "    max_O = 0\n",
    "    for w, val in O.items():\n",
    "        if val > max_O:\n",
    "            max_O = val\n",
    "            max_w = w\n",
    "    return O, max_O, max_w\n",
    "\n",
    "# ----------------- Core per-step solver (with logging) -----------------\n",
    "\n",
    "def find_T_with_persistence(\n",
    "    N: int,\n",
    "    r: int,\n",
    "    failed: Set[int],\n",
    "    hosts_for_types: List[Tuple[int, ...]],\n",
    "    seq: List[List[int]],\n",
    "    pos_map: List[Dict[int,int]],\n",
    "    wlog: Dict[str, List],  # witness logger (accumulated across steps)\n",
    ") -> Tuple[int, bool]:\n",
    "    \"\"\"\n",
    "    Return (T, resequenced_flag) for current failure set, enforcing minimal makespan:\n",
    "      For T = s_cap, ..., r:\n",
    "        1) FIXED at T\n",
    "        2) if fails, FREE at the SAME T (then COMMIT prefixes if succeeds)\n",
    "    Also:\n",
    "      - If minimal T > s_cap, log W1 or W2 with details in `wlog`.\n",
    "    \"\"\"\n",
    "    survivors = [w for w in range(N) if w not in failed]\n",
    "    L = len(survivors)\n",
    "\n",
    "    # Wipe-out check: if any type has all its r hosts failed, no feasible T ≤ r.\n",
    "    for t in range(N):\n",
    "        if all(h in failed for h in hosts_for_types[t]):\n",
    "            return r + 1, False\n",
    "\n",
    "    # Capacity floor\n",
    "    s_cap = (N + L - 1) // L  # ceil(N / L)\n",
    "    s_cap = max(1, s_cap)\n",
    "\n",
    "    minimal_T = None\n",
    "    resequenced = False\n",
    "\n",
    "    for T in range(s_cap, r + 1):\n",
    "        # Phase 1: FIXED at this T\n",
    "        adj_fixed, _ = fixed_adjacency(N, T, survivors, hosts_for_types, seq, pos_map)\n",
    "        hk = HopcroftKarp(adj_fixed, N, len(survivors) * T)\n",
    "        if hk.max_matching() == N:\n",
    "            minimal_T = T\n",
    "            resequenced = False\n",
    "            break\n",
    "\n",
    "        # Phase 2: FREE at the same T\n",
    "        adj_free, clone_to_wk = free_adjacency(N, T, survivors, hosts_for_types)\n",
    "        hk2 = HopcroftKarp(adj_free, N, len(survivors) * T)\n",
    "        if hk2.max_matching() == N:\n",
    "            # Commit the new first-T order per worker (sticky stacks)\n",
    "            used_prefix: Dict[int, List[Optional[int]]] = {w: [None]*T for w in survivors}\n",
    "            for t in range(N):\n",
    "                v = hk2.pairU[t]\n",
    "                if v == -1:\n",
    "                    continue\n",
    "                w, k = clone_to_wk[v]\n",
    "                used_prefix[w][k] = t\n",
    "            for w in survivors:\n",
    "                prefix = used_prefix[w]\n",
    "                if None in prefix:\n",
    "                    have = {x for x in prefix if x is not None}\n",
    "                    filler = [t for t in seq[w] if t not in have]\n",
    "                    fi = 0\n",
    "                    for i in range(T):\n",
    "                        if prefix[i] is None:\n",
    "                            prefix[i] = filler[fi]\n",
    "                            fi += 1\n",
    "                commit_prefix(w, T, prefix, seq, pos_map)\n",
    "            minimal_T = T\n",
    "            resequenced = True\n",
    "            break\n",
    "\n",
    "    if minimal_T is None:\n",
    "        # Should only happen if a wipe-out exists (caught above), but keep sentinel.\n",
    "        return r + 1, False\n",
    "\n",
    "    # If minimal_T > s_cap, classify W1 vs W2 and log.\n",
    "    if minimal_T > s_cap:\n",
    "        O, max_O, argmax_w = compute_pinned_loads(N, failed, hosts_for_types)\n",
    "        if max_O > s_cap:\n",
    "            # W1: single-worker pinned overload\n",
    "            wlog[\"kind\"].append(\"W1\")\n",
    "        else:\n",
    "            # W2: multi-worker Hall witness\n",
    "            wlog[\"kind\"].append(\"W2\")\n",
    "        wlog[\"k\"].append(len(failed))\n",
    "        wlog[\"s_cap\"].append(s_cap)\n",
    "        wlog[\"T\"].append(minimal_T)\n",
    "        wlog[\"w\"].append(argmax_w)\n",
    "        wlog[\"max_O\"].append(max_O)\n",
    "\n",
    "    return minimal_T, resequenced\n",
    "\n",
    "# ----------------- Wipe-out detector -----------------\n",
    "\n",
    "def first_wipeout_reached(\n",
    "    new_fail: int,\n",
    "    failed: Set[int],\n",
    "    hosts_for_types: List[Tuple[int, ...]],\n",
    "    types_for_workers: List[Tuple[int, ...]],\n",
    ") -> bool:\n",
    "    \"\"\"Only types stored on new_fail can newly wipe out.\"\"\"\n",
    "    for t in types_for_workers[new_fail]:\n",
    "        if all(h in failed for h in hosts_for_types[t]):\n",
    "            return True\n",
    "    return False\n",
    "\n",
    "# ----------------- Trial simulation -----------------\n",
    "\n",
    "def simulate_one_trial(\n",
    "    N: int,\n",
    "    r: int,\n",
    "    rng: random.Random,\n",
    "    offsets: List[int],\n",
    ") -> Tuple[float, int, List[int], Dict[str, List], Counter, Counter]:\n",
    "    \"\"\"\n",
    "    One random failure path with PERSISTENT stacks:\n",
    "      - Start with Golomb order as seq[w].\n",
    "      - Each failure: find minimal T (fixed-then-free at SAME T) and commit if resequenced.\n",
    "      - Stop at first wipe-out.\n",
    "\n",
    "    Returns:\n",
    "      mean_overhead_before_wipeout, F, T_path,\n",
    "      witness_log (dict of lists),\n",
    "      per-T histogram (Counter: T -> step count before wipe-out),\n",
    "      transition histogram (Counter: (T_prev, T_cur) -> count)\n",
    "    \"\"\"\n",
    "    hosts_for_types = build_hosts_for_types(N, offsets)\n",
    "    types_for_workers = build_types_for_workers(N, offsets)\n",
    "    seq, pos_map = init_sequences(N, offsets)\n",
    "\n",
    "    failed: Set[int] = set()\n",
    "    order = list(range(N))\n",
    "    rng.shuffle(order)\n",
    "\n",
    "    # Witness logger\n",
    "    wlog = {\"kind\": [], \"k\": [], \"s_cap\": [], \"T\": [], \"w\": [], \"max_O\": []}\n",
    "\n",
    "    T_path: List[int] = []\n",
    "    T_hist = Counter()\n",
    "    trans_hist = Counter()\n",
    "\n",
    "    # k = 0\n",
    "    T0, _ = find_T_with_persistence(N, r, failed, hosts_for_types, seq, pos_map, wlog)\n",
    "    T_path.append(T0)\n",
    "    T_hist[T0] += 1\n",
    "\n",
    "    for step, w in enumerate(order, start=1):\n",
    "        failed.add(w)\n",
    "        if first_wipeout_reached(w, failed, hosts_for_types, types_for_workers):\n",
    "            F = step\n",
    "            break\n",
    "\n",
    "        T_cur, _resequenced = find_T_with_persistence(N, r, failed, hosts_for_types, seq, pos_map, wlog)\n",
    "        if T_cur == r + 1:\n",
    "            F = step\n",
    "            break\n",
    "        trans_hist[(T_path[-1], T_cur)] += 1\n",
    "        T_path.append(T_cur)\n",
    "        T_hist[T_cur] += 1\n",
    "    else:\n",
    "        F = N  # shouldn't happen\n",
    "\n",
    "    mean_overhead = sum(T_path[:F]) / F\n",
    "    return mean_overhead, F, T_path, wlog, T_hist, trans_hist\n",
    "\n",
    "# ----------------- Public API -----------------\n",
    "\n",
    "def run_simulation(\n",
    "    N: int = 600,\n",
    "    r: Optional[int] = 20,\n",
    "    trials: int = 200,\n",
    "    seed: int = 0,\n",
    "    offsets: Optional[List[int]] = None,\n",
    ") -> None:\n",
    "    \"\"\"\n",
    "    Monte Carlo with persistent resequencing + ALWAYS-ON W1/W2 logger + histograms.\n",
    "    Either pass `r` and let the code build/construct offsets, or pass a custom `offsets` list.\n",
    "\n",
    "    Prints:\n",
    "      - E[F], E[mean T] with stddevs.\n",
    "      - Average #steps at each T per trial.\n",
    "      - Counts of T transitions (e.g., 2->3, 3->4).\n",
    "      - Exceed-capacity events per trial and distribution by (kind, s_cap -> T).\n",
    "    \"\"\"\n",
    "    rng = random.Random(seed)\n",
    "    if offsets is None:\n",
    "        if r is None:\n",
    "            raise ValueError(\"Provide either r or offsets.\")\n",
    "        offsets = get_offsets(N, r, rng, offsets=None)\n",
    "    else:\n",
    "        r = len(offsets)\n",
    "\n",
    "    overheads: List[float] = []\n",
    "    Fs: List[int] = []\n",
    "\n",
    "    # Aggregates\n",
    "    total_T_hist = Counter()\n",
    "    total_trans_hist = Counter()\n",
    "    total_W1 = 0\n",
    "    total_W2 = 0\n",
    "    pair_counts = Counter()  # (kind, s_cap, T)\n",
    "    trials_with_W1 = 0\n",
    "    trials_with_W2 = 0\n",
    "\n",
    "    for _ in range(trials):\n",
    "        m, F, _T_path, wlog, T_hist, trans_hist = simulate_one_trial(N, r, rng, offsets)\n",
    "        overheads.append(m)\n",
    "        Fs.append(F)\n",
    "        total_T_hist.update(T_hist)\n",
    "        total_trans_hist.update(trans_hist)\n",
    "\n",
    "        w1 = sum(1 for k in wlog[\"kind\"] if k == \"W1\")\n",
    "        w2 = sum(1 for k in wlog[\"kind\"] if k == \"W2\")\n",
    "        total_W1 += w1\n",
    "        total_W2 += w2\n",
    "        if w1 > 0: trials_with_W1 += 1\n",
    "        if w2 > 0: trials_with_W2 += 1\n",
    "        for kind, s_cap, T in zip(wlog[\"kind\"], wlog[\"s_cap\"], wlog[\"T\"]):\n",
    "            pair_counts[(kind, s_cap, T)] += 1\n",
    "\n",
    "    # ----- Summary printing -----\n",
    "    print(f\"(N={N}, r={r}, offsets={offsets}) over {trials} trials\")\n",
    "    print(f\"Average failures until first wipe-out   E[F]     ≈ {mean(Fs):.3f}  (std={pstdev(Fs):.3f})\")\n",
    "    print(f\"Average per-step computational overhead E[mean T] ≈ x{mean(overheads):.4f}  (std={pstdev(overheads):.4f})\")\n",
    "\n",
    "    # Overhead histogram averaged per trial\n",
    "    max_T_seen = max(total_T_hist.keys()) if total_T_hist else 0\n",
    "    print(\"\\nAverage #failure steps at each stack count (per trial):\")\n",
    "    for T in range(1, max_T_seen + 1):\n",
    "        avg_steps_T = total_T_hist[T] / trials\n",
    "        print(f\"  T={T}: {avg_steps_T:.3f} steps on average\")\n",
    "\n",
    "    # Transition histogram\n",
    "    if total_trans_hist:\n",
    "        print(\"\\nObserved transitions T_prev -> T_cur (counts, total over all trials):\")\n",
    "        for (t0, t1), cnt in sorted(total_trans_hist.items()):\n",
    "            print(f\"  {t0} -> {t1}: {cnt}\")\n",
    "\n",
    "    # Witness statistics\n",
    "    print(\"\\nExceed-capacity events (T > s_cap):\")\n",
    "    print(f\"  W1 (single-worker pinned overload): total {total_W1}  \"\n",
    "          f\"(occurred in {trials_with_W1}/{trials} trials; avg {total_W1/trials:.4f} per trial)\")\n",
    "    print(f\"  W2 (multi-worker Hall witness)    : total {total_W2}  \"\n",
    "          f\"(occurred in {trials_with_W2}/{trials} trials; avg {total_W2/trials:.4f} per trial)\")\n",
    "\n",
    "    if pair_counts:\n",
    "        print(\"\\nBreakdown by (kind, s_cap -> T) for exceed-capacity events:\")\n",
    "        for (kind, s_cap, T), cnt in sorted(pair_counts.items()):\n",
    "            print(f\"  {kind}: {s_cap} -> {T}  count={cnt}\")\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "846fe4a8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=2, offsets=[0, 586]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 22.314  (std=10.819)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9370  (std=0.0542)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 21.314 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 20314\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=2, trials=1000, seed=10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "91c19d5b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=3, offsets=[0, 1, 3]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 64.651  (std=22.245)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9818  (std=0.0107)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 63.640 steps on average\n",
      "  T=3: 0.011 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 62640\n",
      "  2 -> 3: 2\n",
      "  3 -> 3: 9\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 11  (occurred in 2/1000 trials; avg 0.0110 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W2: 2 -> 3  count=11\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=3, trials=1000, seed=11)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "743772a0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=4, offsets=[0, 1, 4, 6]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 111.495  (std=29.830)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9906  (std=0.0082)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 110.414 steps on average\n",
      "  T=3: 0.081 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 109414\n",
      "  2 -> 3: 5\n",
      "  3 -> 3: 76\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 12  (occurred in 1/1000 trials; avg 0.0120 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 69  (occurred in 4/1000 trials; avg 0.0690 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=12\n",
      "  W2: 2 -> 3  count=69\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=4, trials=1000, seed=12)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "21f802cf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=5, offsets=[0, 1, 4, 9, 11]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 154.107  (std=34.398)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9931  (std=0.0023)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 153.107 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 152107\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=5, trials=1000, seed=13)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "8332cad5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=6, offsets=[0, 1, 4, 10, 12, 17]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 191.539  (std=36.782)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9949  (std=0.0064)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 190.443 steps on average\n",
      "  T=3: 0.096 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 189443\n",
      "  2 -> 3: 6\n",
      "  3 -> 3: 90\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 41  (occurred in 3/1000 trials; avg 0.0410 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 55  (occurred in 3/1000 trials; avg 0.0550 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=41\n",
      "  W2: 2 -> 3  count=55\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=6, trials=1000, seed=14)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "edd8939b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=7, offsets=[0, 1, 4, 10, 18, 23, 25]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 226.686  (std=36.106)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9960  (std=0.0058)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 225.531 steps on average\n",
      "  T=3: 0.155 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 224531\n",
      "  2 -> 3: 12\n",
      "  3 -> 3: 143\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 6  (occurred in 1/1000 trials; avg 0.0060 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 79  (occurred in 9/1000 trials; avg 0.0790 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=6\n",
      "  W2: 2 -> 3  count=79\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=7, trials=1000, seed=15)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d29d9f67",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=8, offsets=[0, 1, 4, 9, 15, 22, 32, 34]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 256.067  (std=36.000)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0004  (std=0.0164)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 253.675 steps on average\n",
      "  T=3: 1.392 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 252675\n",
      "  2 -> 3: 99\n",
      "  3 -> 3: 1293\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 2  (occurred in 1/1000 trials; avg 0.0020 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 216  (occurred in 33/1000 trials; avg 0.2160 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=2\n",
      "  W2: 2 -> 3  count=216\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=8, trials=1000, seed=16)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "03b66dbc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=9, offsets=[0, 1, 5, 12, 25, 27, 35, 41, 44]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 279.986  (std=36.797)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0135  (std=0.0343)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 273.346 steps on average\n",
      "  T=3: 5.640 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 272346\n",
      "  2 -> 3: 307\n",
      "  3 -> 3: 5333\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 39  (occurred in 10/1000 trials; avg 0.0390 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W2: 2 -> 3  count=39\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=9, trials=1000, seed=17)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "0da6f4cc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=10, offsets=[0, 1, 6, 10, 23, 26, 34, 41, 53, 55]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 303.535  (std=36.142)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0425  (std=0.0532)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 287.054 steps on average\n",
      "  T=3: 15.481 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 286054\n",
      "  2 -> 3: 579\n",
      "  3 -> 3: 14902\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 7  (occurred in 1/1000 trials; avg 0.0070 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W2: 2 -> 3  count=7\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=10, trials=1000, seed=18)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "99e95729",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=11, offsets=[0, 1, 4, 13, 28, 33, 47, 54, 64, 70, 72]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 322.333  (std=35.009)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0763  (std=0.0670)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 293.604 steps on average\n",
      "  T=3: 27.729 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 292604\n",
      "  2 -> 3: 744\n",
      "  3 -> 3: 26985\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=11, trials=1000, seed=19)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "eb5a25bf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=12, offsets=[0, 2, 6, 24, 29, 40, 43, 55, 68, 75, 76, 85]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 339.038  (std=32.739)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.1101  (std=0.0733)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 297.469 steps on average\n",
      "  T=3: 40.477 steps on average\n",
      "  T=4: 0.092 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 296469\n",
      "  2 -> 3: 869\n",
      "  3 -> 3: 39608\n",
      "  3 -> 4: 13\n",
      "  4 -> 4: 79\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=12, trials=1000, seed=20)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "cc670cd4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=13, offsets=[0, 2, 5, 25, 37, 43, 59, 70, 85, 89, 98, 99, 106]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 355.413  (std=34.415)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.1491  (std=0.0776)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 298.323 steps on average\n",
      "  T=3: 55.575 steps on average\n",
      "  T=4: 0.515 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 297323\n",
      "  2 -> 3: 925\n",
      "  3 -> 3: 54650\n",
      "  3 -> 4: 68\n",
      "  4 -> 4: 447\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=13, trials=1000, seed=21)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "a5c5c325",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=14, offsets=[0, 4, 6, 20, 35, 52, 59, 77, 78, 86, 89, 99, 122, 127]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 369.758  (std=31.286)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.1830  (std=0.0755)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 299.432 steps on average\n",
      "  T=3: 67.631 steps on average\n",
      "  T=4: 1.695 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 298432\n",
      "  2 -> 3: 968\n",
      "  3 -> 3: 66663\n",
      "  3 -> 4: 151\n",
      "  4 -> 4: 1544\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=14, trials=1000, seed=22)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "7a088ca9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=15, offsets=[0, 4, 20, 30, 57, 59, 62, 76, 100, 111, 123, 136, 144, 145, 151]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 381.258  (std=30.417)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.2126  (std=0.0772)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 299.660 steps on average\n",
      "  T=3: 76.833 steps on average\n",
      "  T=4: 3.765 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 298660\n",
      "  2 -> 3: 986\n",
      "  3 -> 3: 75847\n",
      "  3 -> 4: 276\n",
      "  4 -> 4: 3489\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=15, trials=1000, seed=23)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "421bfe32",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=16, offsets=[0, 1, 4, 11, 26, 32, 56, 68, 76, 115, 117, 134, 150, 163, 168, 177]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 392.799  (std=29.165)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.2442  (std=0.0822)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 299.955 steps on average\n",
      "  T=3: 84.399 steps on average\n",
      "  T=4: 7.433 steps on average\n",
      "  T=5: 0.012 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 298955\n",
      "  2 -> 3: 994\n",
      "  3 -> 3: 83405\n",
      "  3 -> 4: 413\n",
      "  4 -> 4: 7020\n",
      "  4 -> 5: 3\n",
      "  5 -> 5: 9\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=16, trials=1000, seed=24)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "00c5043b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=17, offsets=[0, 5, 7, 17, 52, 56, 67, 80, 81, 100, 122, 138, 159, 165, 168, 191, 199]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 401.789  (std=27.092)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.2707  (std=0.0809)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 299.995 steps on average\n",
      "  T=3: 89.700 steps on average\n",
      "  T=4: 11.063 steps on average\n",
      "  T=5: 0.031 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 298995\n",
      "  2 -> 3: 998\n",
      "  3 -> 3: 88702\n",
      "  3 -> 4: 552\n",
      "  4 -> 4: 10511\n",
      "  4 -> 5: 6\n",
      "  5 -> 5: 25\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=17, trials=1000, seed=25)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "cf27558f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=18, offsets=[0, 2, 10, 22, 53, 56, 82, 83, 89, 98, 130, 148, 153, 167, 188, 192, 205, 216]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 411.308  (std=27.278)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.3020  (std=0.0859)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 299.963 steps on average\n",
      "  T=3: 93.293 steps on average\n",
      "  T=4: 16.904 steps on average\n",
      "  T=5: 0.148 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 298963\n",
      "  2 -> 3: 997\n",
      "  3 -> 3: 92296\n",
      "  3 -> 4: 709\n",
      "  4 -> 4: 16195\n",
      "  4 -> 5: 28\n",
      "  5 -> 5: 120\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=18, trials=1000, seed=26)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "db362ad5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=19, offsets=[0, 1, 6, 25, 32, 72, 100, 108, 120, 130, 153, 169, 187, 190, 204, 231, 233, 242, 246]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 419.778  (std=27.038)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.3318  (std=0.0895)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 299.959 steps on average\n",
      "  T=3: 95.518 steps on average\n",
      "  T=4: 22.748 steps on average\n",
      "  T=5: 0.553 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 298959\n",
      "  2 -> 3: 997\n",
      "  3 -> 3: 94521\n",
      "  3 -> 4: 806\n",
      "  4 -> 4: 21942\n",
      "  4 -> 5: 74\n",
      "  5 -> 5: 479\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=19, trials=1000, seed=27)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "898bcd39",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=600, r=20, offsets=[0, 1, 8, 11, 68, 77, 94, 116, 121, 156, 158, 179, 194, 208, 212, 228, 240, 253, 259, 283]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 426.411  (std=25.832)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.3560  (std=0.0908)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 299.986 steps on average\n",
      "  T=3: 96.905 steps on average\n",
      "  T=4: 27.361 steps on average\n",
      "  T=5: 1.159 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 298986\n",
      "  2 -> 3: 999\n",
      "  3 -> 3: 95906\n",
      "  3 -> 4: 850\n",
      "  4 -> 4: 26511\n",
      "  4 -> 5: 144\n",
      "  5 -> 5: 1015\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=600, r=20, trials=1000, seed=28)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "c8fac328",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=2, offsets=[0, 141]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 13.237  (std=6.231)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.8953  (std=0.0834)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 12.237 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 11237\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=2, trials=1000, seed=29)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "053bf760",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=3, offsets=[0, 1, 3]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 31.495  (std=10.607)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9631  (std=0.0210)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 30.484 steps on average\n",
      "  T=3: 0.011 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 29484\n",
      "  2 -> 3: 2\n",
      "  3 -> 3: 9\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 11  (occurred in 2/1000 trials; avg 0.0110 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W2: 2 -> 3  count=11\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=3, trials=1000, seed=30)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "b97ac1dc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=4, offsets=[0, 1, 4, 6]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 49.852  (std=12.985)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9794  (std=0.0199)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 48.746 steps on average\n",
      "  T=3: 0.106 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 47746\n",
      "  2 -> 3: 13\n",
      "  3 -> 3: 93\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 34  (occurred in 4/1000 trials; avg 0.0340 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 72  (occurred in 10/1000 trials; avg 0.0720 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=34\n",
      "  W2: 2 -> 3  count=72\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=4, trials=1000, seed=31)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "fdd81022",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=5, offsets=[0, 1, 4, 9, 11]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 64.403  (std=14.096)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9842  (std=0.0098)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 63.345 steps on average\n",
      "  T=3: 0.058 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 62345\n",
      "  2 -> 3: 13\n",
      "  3 -> 3: 45\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 43  (occurred in 9/1000 trials; avg 0.0430 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 15  (occurred in 5/1000 trials; avg 0.0150 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=43\n",
      "  W2: 2 -> 3  count=15\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=5, trials=1000, seed=32)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "e785c589",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=6, offsets=[0, 1, 4, 10, 12, 17]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 78.143  (std=13.790)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9891  (std=0.0137)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 76.893 steps on average\n",
      "  T=3: 0.250 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 75893\n",
      "  2 -> 3: 52\n",
      "  3 -> 3: 198\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 8  (occurred in 3/1000 trials; avg 0.0080 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 106  (occurred in 29/1000 trials; avg 0.1060 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=8\n",
      "  W2: 2 -> 3  count=106\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=6, trials=1000, seed=33)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "cab0ec3e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=7, offsets=[0, 1, 4, 10, 18, 23, 25]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 89.171  (std=14.092)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0003  (std=0.0311)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 86.856 steps on average\n",
      "  T=3: 1.315 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 85856\n",
      "  2 -> 3: 190\n",
      "  3 -> 3: 1125\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 6  (occurred in 1/1000 trials; avg 0.0060 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 10  (occurred in 5/1000 trials; avg 0.0100 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=6\n",
      "  W2: 2 -> 3  count=10\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=7, trials=1000, seed=34)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "5063fd0b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=8, offsets=[0, 1, 4, 9, 15, 22, 32, 34]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 98.642  (std=14.395)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0280  (std=0.0551)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 93.249 steps on average\n",
      "  T=3: 4.393 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 92249\n",
      "  2 -> 3: 475\n",
      "  3 -> 3: 3918\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=8, trials=1000, seed=35)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "72112906",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=9, offsets=[0, 1, 5, 12, 25, 27, 35, 41, 44]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 107.202  (std=13.678)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0688  (std=0.0739)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 96.922 steps on average\n",
      "  T=3: 9.273 steps on average\n",
      "  T=4: 0.007 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 95922\n",
      "  2 -> 3: 693\n",
      "  3 -> 3: 8580\n",
      "  3 -> 4: 1\n",
      "  4 -> 4: 6\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=9, trials=1000, seed=36)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "0d9a2ada",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=10, offsets=[0, 1, 6, 10, 23, 26, 34, 41, 53, 55]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 114.081  (std=13.260)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.1107  (std=0.0828)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 98.540 steps on average\n",
      "  T=3: 14.408 steps on average\n",
      "  T=4: 0.133 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 97540\n",
      "  2 -> 3: 832\n",
      "  3 -> 3: 13576\n",
      "  3 -> 4: 34\n",
      "  4 -> 4: 99\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=10, trials=1000, seed=37)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "17f74307",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=11, offsets=[0, 1, 4, 13, 28, 33, 47, 54, 64, 70, 72]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 119.820  (std=12.577)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.1484  (std=0.0866)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 99.374 steps on average\n",
      "  T=3: 19.046 steps on average\n",
      "  T=4: 0.400 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 98374\n",
      "  2 -> 3: 911\n",
      "  3 -> 3: 18135\n",
      "  3 -> 4: 103\n",
      "  4 -> 4: 297\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=11, trials=1000, seed=38)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "daa47226",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=200, r=12, offsets=[0, 2, 6, 24, 29, 40, 43, 55, 68, 75, 76, 85]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 125.218  (std=12.005)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.1894  (std=0.0892)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 99.670 steps on average\n",
      "  T=3: 23.332 steps on average\n",
      "  T=4: 1.214 steps on average\n",
      "  T=5: 0.002 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 98670\n",
      "  2 -> 3: 954\n",
      "  3 -> 3: 22378\n",
      "  3 -> 4: 227\n",
      "  4 -> 4: 987\n",
      "  4 -> 5: 1\n",
      "  5 -> 5: 1\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=200, r=12, trials=1000, seed=39)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "437de12e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=2, offsets=[0, 470]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 28.464  (std=15.009)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9472  (std=0.0522)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 27.464 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 26464\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=2, trials=1000, seed=40)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "33223969",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=3, offsets=[0, 1, 3]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 89.728  (std=32.324)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9867  (std=0.0078)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 88.721 steps on average\n",
      "  T=3: 0.007 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 87721\n",
      "  2 -> 3: 1\n",
      "  3 -> 3: 6\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 7  (occurred in 1/1000 trials; avg 0.0070 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W2: 2 -> 3  count=7\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=3, trials=1000, seed=41)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "26b7d996",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=4, offsets=[0, 1, 4, 6]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 163.154  (std=45.199)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9933  (std=0.0038)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 162.135 steps on average\n",
      "  T=3: 0.019 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 161135\n",
      "  2 -> 3: 2\n",
      "  3 -> 3: 17\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 19  (occurred in 2/1000 trials; avg 0.0190 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=19\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=4, trials=1000, seed=42)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "d20c2aa4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=5, offsets=[0, 1, 4, 9, 11]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 230.367  (std=50.872)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9955  (std=0.0029)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 229.339 steps on average\n",
      "  T=3: 0.028 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 228339\n",
      "  2 -> 3: 1\n",
      "  3 -> 3: 27\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 28  (occurred in 1/1000 trials; avg 0.0280 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W2: 2 -> 3  count=28\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=5, trials=1000, seed=43)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "cc03fa39",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=6, offsets=[0, 1, 4, 10, 12, 17]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 296.333  (std=55.012)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9965  (std=0.0014)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 295.316 steps on average\n",
      "  T=3: 0.017 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 294316\n",
      "  2 -> 3: 2\n",
      "  3 -> 3: 15\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 12  (occurred in 1/1000 trials; avg 0.0120 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 5  (occurred in 1/1000 trials; avg 0.0050 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=12\n",
      "  W2: 2 -> 3  count=5\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=6, trials=1000, seed=44)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "6acff390",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=7, offsets=[0, 1, 4, 10, 18, 23, 25]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 349.821  (std=56.599)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9971  (std=0.0016)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 348.786 steps on average\n",
      "  T=3: 0.035 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 347786\n",
      "  2 -> 3: 4\n",
      "  3 -> 3: 31\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 9  (occurred in 1/1000 trials; avg 0.0090 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 26  (occurred in 4/1000 trials; avg 0.0260 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=9\n",
      "  W2: 2 -> 3  count=26\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=7, trials=1000, seed=45)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "11a2d330",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=8, offsets=[0, 1, 4, 9, 15, 22, 32, 34]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 399.261  (std=59.220)\n",
      "Average per-step computational overhead E[mean T] ≈ x1.9985  (std=0.0074)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 397.721 steps on average\n",
      "  T=3: 0.540 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 396721\n",
      "  2 -> 3: 35\n",
      "  3 -> 3: 505\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 16  (occurred in 2/1000 trials; avg 0.0160 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 248  (occurred in 25/1000 trials; avg 0.2480 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W1: 2 -> 3  count=16\n",
      "  W2: 2 -> 3  count=248\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=8, trials=1000, seed=46)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "fe0cb3a3",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=9, offsets=[0, 1, 5, 12, 25, 27, 35, 41, 44]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 443.616  (std=58.070)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0045  (std=0.0199)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 439.020 steps on average\n",
      "  T=3: 3.596 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 438020\n",
      "  2 -> 3: 165\n",
      "  3 -> 3: 3431\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 430  (occurred in 50/1000 trials; avg 0.4300 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W2: 2 -> 3  count=430\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=9, trials=1000, seed=47)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "1652434c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=10, offsets=[0, 1, 6, 10, 23, 26, 34, 41, 53, 55]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 477.233  (std=55.702)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0192  (std=0.0359)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 464.603 steps on average\n",
      "  T=3: 11.630 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 463603\n",
      "  2 -> 3: 376\n",
      "  3 -> 3: 11254\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 157  (occurred in 28/1000 trials; avg 0.1570 per trial)\n",
      "\n",
      "Breakdown by (kind, s_cap -> T) for exceed-capacity events:\n",
      "  W2: 2 -> 3  count=157\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=10, trials=1000, seed=48)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "09522628",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=11, offsets=[0, 1, 4, 13, 28, 33, 47, 54, 64, 70, 72]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 510.154  (std=55.828)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0464  (std=0.0540)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 481.885 steps on average\n",
      "  T=3: 27.258 steps on average\n",
      "  T=4: 0.011 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 480885\n",
      "  2 -> 3: 606\n",
      "  3 -> 3: 26652\n",
      "  3 -> 4: 1\n",
      "  4 -> 4: 10\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=11, trials=1000, seed=49)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "e3d73877",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=12, offsets=[0, 2, 6, 24, 29, 40, 43, 55, 68, 75, 76, 85]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 542.951  (std=53.189)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.0839  (std=0.0648)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 492.138 steps on average\n",
      "  T=3: 49.813 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 491138\n",
      "  2 -> 3: 800\n",
      "  3 -> 3: 49013\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=12, trials=1000, seed=50)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "92eb5afb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=13, offsets=[0, 2, 5, 25, 37, 43, 59, 70, 85, 89, 98, 99, 106]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 568.070  (std=50.332)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.1163  (std=0.0684)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 496.679 steps on average\n",
      "  T=3: 70.356 steps on average\n",
      "  T=4: 0.035 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 495679\n",
      "  2 -> 3: 886\n",
      "  3 -> 3: 69470\n",
      "  3 -> 4: 3\n",
      "  4 -> 4: 32\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=13, trials=1000, seed=51)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "17260029",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=14, offsets=[0, 4, 6, 20, 35, 52, 59, 77, 78, 86, 89, 99, 122, 127]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 592.349  (std=49.618)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.1497  (std=0.0696)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 498.638 steps on average\n",
      "  T=3: 92.328 steps on average\n",
      "  T=4: 0.383 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 497638\n",
      "  2 -> 3: 954\n",
      "  3 -> 3: 91374\n",
      "  3 -> 4: 33\n",
      "  4 -> 4: 350\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=14, trials=1000, seed=52)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "b32c09c5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=15, offsets=[0, 4, 20, 30, 57, 59, 62, 76, 100, 111, 123, 136, 144, 145, 151]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 608.114  (std=51.077)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.1729  (std=0.0716)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 498.729 steps on average\n",
      "  T=3: 107.000 steps on average\n",
      "  T=4: 1.385 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 497729\n",
      "  2 -> 3: 963\n",
      "  3 -> 3: 106037\n",
      "  3 -> 4: 95\n",
      "  4 -> 4: 1290\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=15, trials=1000, seed=53)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "30822ccd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=16, offsets=[0, 1, 4, 11, 26, 32, 56, 68, 76, 115, 117, 134, 150, 163, 168, 177]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 633.081  (std=47.348)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.2099  (std=0.0710)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 499.547 steps on average\n",
      "  T=3: 127.851 steps on average\n",
      "  T=4: 4.683 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 498547\n",
      "  2 -> 3: 987\n",
      "  3 -> 3: 126864\n",
      "  3 -> 4: 233\n",
      "  4 -> 4: 4450\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=16, trials=1000, seed=54)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "460a3c83",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=17, offsets=[0, 5, 7, 17, 52, 56, 67, 80, 81, 100, 122, 138, 159, 165, 168, 191, 199]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 647.262  (std=47.026)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.2335  (std=0.0759)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 499.837 steps on average\n",
      "  T=3: 137.211 steps on average\n",
      "  T=4: 9.214 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 498837\n",
      "  2 -> 3: 990\n",
      "  3 -> 3: 136221\n",
      "  3 -> 4: 374\n",
      "  4 -> 4: 8840\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=17, trials=1000, seed=55)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "68b8f329",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=18, offsets=[0, 2, 10, 22, 53, 56, 82, 83, 89, 98, 130, 148, 153, 167, 188, 192, 205, 216]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 663.715  (std=45.739)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.2621  (std=0.0788)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 499.875 steps on average\n",
      "  T=3: 147.241 steps on average\n",
      "  T=4: 15.558 steps on average\n",
      "  T=5: 0.041 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 498875\n",
      "  2 -> 3: 995\n",
      "  3 -> 3: 146246\n",
      "  3 -> 4: 543\n",
      "  4 -> 4: 15015\n",
      "  4 -> 5: 5\n",
      "  5 -> 5: 36\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=18, trials=1000, seed=56)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "af1aca30",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=19, offsets=[0, 1, 6, 25, 32, 72, 100, 108, 120, 130, 153, 169, 187, 190, 204, 231, 233, 242, 246]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 682.356  (std=43.492)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.2981  (std=0.0824)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 499.964 steps on average\n",
      "  T=3: 154.987 steps on average\n",
      "  T=4: 26.275 steps on average\n",
      "  T=5: 0.130 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 498964\n",
      "  2 -> 3: 999\n",
      "  3 -> 3: 153988\n",
      "  3 -> 4: 689\n",
      "  4 -> 4: 25586\n",
      "  4 -> 5: 19\n",
      "  5 -> 5: 111\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=19, trials=1000, seed=57)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "f7323967",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=20, offsets=[0, 1, 8, 11, 68, 77, 94, 116, 121, 156, 158, 179, 194, 208, 212, 228, 240, 253, 259, 283]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 691.561  (std=40.657)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.3166  (std=0.0803)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 500.000 steps on average\n",
      "  T=3: 158.152 steps on average\n",
      "  T=4: 32.181 steps on average\n",
      "  T=5: 0.228 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 499000\n",
      "  2 -> 3: 1000\n",
      "  3 -> 3: 157152\n",
      "  3 -> 4: 763\n",
      "  4 -> 4: 31418\n",
      "  4 -> 5: 32\n",
      "  5 -> 5: 196\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=20, trials=1000, seed=58)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "6f2ebd90",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=21, offsets=[0, 2, 24, 56, 77, 82, 83, 95, 129, 144, 179, 186, 195, 255, 265, 285, 293, 296, 310, 329, 333]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 704.886  (std=41.194)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.3461  (std=0.0866)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 500.000 steps on average\n",
      "  T=3: 160.621 steps on average\n",
      "  T=4: 41.958 steps on average\n",
      "  T=5: 1.307 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 499000\n",
      "  2 -> 3: 1000\n",
      "  3 -> 3: 159621\n",
      "  3 -> 4: 828\n",
      "  4 -> 4: 41130\n",
      "  4 -> 5: 102\n",
      "  5 -> 5: 1205\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=21, trials=1000, seed=59)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "1a2343e4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=22, offsets=[0, 1, 9, 14, 43, 70, 106, 122, 124, 128, 159, 179, 204, 223, 253, 263, 270, 291, 330, 341, 353, 356]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 714.356  (std=40.206)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.3667  (std=0.0879)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 500.000 steps on average\n",
      "  T=3: 162.488 steps on average\n",
      "  T=4: 48.673 steps on average\n",
      "  T=5: 2.195 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 499000\n",
      "  2 -> 3: 1000\n",
      "  3 -> 3: 161488\n",
      "  3 -> 4: 882\n",
      "  4 -> 4: 47791\n",
      "  4 -> 5: 166\n",
      "  5 -> 5: 2029\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=22, trials=1000, seed=60)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "75ad3fe5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=23, offsets=[0, 3, 7, 17, 61, 66, 91, 99, 114, 159, 171, 199, 200, 226, 235, 246, 277, 316, 329, 348, 350, 366, 372]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 724.567  (std=39.380)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.3908  (std=0.0898)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 500.000 steps on average\n",
      "  T=3: 163.564 steps on average\n",
      "  T=4: 55.923 steps on average\n",
      "  T=5: 4.069 steps on average\n",
      "  T=6: 0.011 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 499000\n",
      "  2 -> 3: 1000\n",
      "  3 -> 3: 162564\n",
      "  3 -> 4: 908\n",
      "  4 -> 4: 55015\n",
      "  4 -> 5: 257\n",
      "  5 -> 5: 3812\n",
      "  5 -> 6: 2\n",
      "  6 -> 6: 9\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=23, trials=1000, seed=61)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "de69ec6b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=24, offsets=[0, 9, 33, 37, 38, 97, 122, 129, 140, 142, 152, 191, 205, 208, 252, 278, 286, 326, 332, 353, 368, 384, 403, 425]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 736.185  (std=37.930)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.4198  (std=0.0926)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 500.000 steps on average\n",
      "  T=3: 164.679 steps on average\n",
      "  T=4: 62.733 steps on average\n",
      "  T=5: 7.713 steps on average\n",
      "  T=6: 0.060 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 499000\n",
      "  2 -> 3: 1000\n",
      "  3 -> 3: 163679\n",
      "  3 -> 4: 941\n",
      "  4 -> 4: 61792\n",
      "  4 -> 5: 381\n",
      "  5 -> 5: 7332\n",
      "  5 -> 6: 9\n",
      "  6 -> 6: 51\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=24, trials=1000, seed=62)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "61e318ea",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=25, offsets=[0, 12, 29, 39, 72, 91, 146, 157, 160, 161, 166, 191, 207, 214, 258, 290, 316, 354, 372, 394, 396, 431, 459, 467, 480]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 745.801  (std=36.260)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.4446  (std=0.0924)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 500.000 steps on average\n",
      "  T=3: 165.239 steps on average\n",
      "  T=4: 68.217 steps on average\n",
      "  T=5: 11.164 steps on average\n",
      "  T=6: 0.181 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 499000\n",
      "  2 -> 3: 1000\n",
      "  3 -> 3: 164239\n",
      "  3 -> 4: 969\n",
      "  4 -> 4: 67248\n",
      "  4 -> 5: 495\n",
      "  5 -> 5: 10669\n",
      "  5 -> 6: 25\n",
      "  6 -> 6: 156\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=25, trials=1000, seed=63)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "e7d5d5d8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(N=1000, r=26, offsets=[0, 1, 33, 83, 104, 110, 124, 163, 185, 200, 203, 249, 251, 258, 314, 318, 343, 356, 386, 430, 440, 456, 464, 475, 487, 492]) over 1000 trials\n",
      "Average failures until first wipe-out   E[F]     ≈ 751.857  (std=34.904)\n",
      "Average per-step computational overhead E[mean T] ≈ x2.4610  (std=0.0923)\n",
      "\n",
      "Average #failure steps at each stack count (per trial):\n",
      "  T=1: 1.000 steps on average\n",
      "  T=2: 500.000 steps on average\n",
      "  T=3: 165.527 steps on average\n",
      "  T=4: 71.051 steps on average\n",
      "  T=5: 13.915 steps on average\n",
      "  T=6: 0.364 steps on average\n",
      "\n",
      "Observed transitions T_prev -> T_cur (counts, total over all trials):\n",
      "  1 -> 2: 1000\n",
      "  2 -> 2: 499000\n",
      "  2 -> 3: 1000\n",
      "  3 -> 3: 164527\n",
      "  3 -> 4: 981\n",
      "  4 -> 4: 70070\n",
      "  4 -> 5: 550\n",
      "  5 -> 5: 13365\n",
      "  5 -> 6: 44\n",
      "  6 -> 6: 320\n",
      "\n",
      "Exceed-capacity events (T > s_cap):\n",
      "  W1 (single-worker pinned overload): total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n",
      "  W2 (multi-worker Hall witness)    : total 0  (occurred in 0/1000 trials; avg 0.0000 per trial)\n"
     ]
    }
   ],
   "source": [
    "run_simulation(N=1000, r=26, trials=1000, seed=64)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
