from abc import ABC, abstractmethod
from functools import partial
from typing import Dict, List, Union, Type, Optional
from cache.evict.evictor import *
from cache.evict.predictor import *
import numpy as np
import types
import copy
import random

class EvictAlgorithm(ABC):
    """Evict an entry from one cache line
    
    Max size is associativity
    """
    def __init__(self, associativity) -> None:
        self.cache = [None] * associativity
        self.pcs = [None] * associativity
        self.associativity = associativity
    
    def snapshot(self):
        return list(zip(self.cache, self.pcs))
    
    @abstractmethod
    def access(self, pc, address) -> bool:
        pass

    def boost_access(self, pc, address, boost_pred) -> bool:
        return self.access(pc, address)

class PredictAlgorithm(EvictAlgorithm):
    def __init__(self, associativity, evictor_type: Union[Type[Evictor], partial], predictor_type: Union[Predictor, partial]) -> None:
        super().__init__(associativity)
        self.timestamp = 0

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None
        
        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)
        
        self.evictor = evictor_type()
        self.predictor = predictor_type()

        self.cur_boost_pred = None
        self.cur_boost_type = None

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)
    
    def before_pred(self, pc, address):
        if self.cur_boost_type is not None and self.cur_boost_type == 'before':
            self.preds = self.cur_boost_pred
        else:
            preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
            if preds is not None:
                self.preds = preds
    
    def after_pred(self, pc ,address, target_index):
        if self.cur_boost_type is not None and self.cur_boost_type == 'after':
            self.preds[target_index] = self.cur_boost_pred
        else:
            pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
            if pred is not None:
                self.preds[target_index] = pred
        self.timestamp += 1
    
    def boost_access(self, pc, address, boost_pred):
        self.cur_boost_pred = boost_pred
        if self.cur_boost_type is None:
            if isinstance(boost_pred, list):
                self.cur_boost_type = 'before'
            else:
                self.cur_boost_type = 'after'
        return self.access(pc, address)

    def access(self, pc, address):
        target_index = -1
        hit = False

        self.before_pred(pc, address)
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            target_index = self.evictor.evict(list(enumerate(self.preds)))
        self.cache[target_index], self.pcs[target_index] = address, pc
        self.after_pred(pc, address, target_index)
        return hit

######################################################################

class PredictiveMarker(PredictAlgorithm):
    """
    PredictiveMarker algorithm

    Designed by Thodoris Lykouris and Sergei Vassilvitskii. 2018. Competitive Caching with Machine Learned Advice.
    https://dl.acm.org/doi/10.1145/3447579
    """
    def __init__(self, associativity, evictor_type: Union[Type[Evictor], partial], predictor_type: Union[Predictor, partial]) -> None:
        def harmonic_number(k):
            return sum(1 / i for i in range(1, k + 1))
        super().__init__(associativity, evictor_type, predictor_type)
        self.marked = [0] * associativity
        self.tracking_set = []
        self.h_k = harmonic_number(associativity)
        self.chains_len = []
        self.chains_rep = []
    
    def access(self, pc, address):
        hit = False
        self.before_pred(pc, address)
        target_index = -1

        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            if all(mark == 1 for mark in self.marked):
                # new phase
                self.tracking_set = copy.deepcopy(self.cache)
                self.marked = [0] * self.associativity
            if address not in self.tracking_set:
                target_index = self.evictor.evict([(i, self.preds[i]) for i, mark in enumerate(self.marked) if mark == 0])
                self.chains_len.append(1)
                self.chains_rep.append(self.cache[target_index])
            if address in self.tracking_set:
                index = self.chains_rep.index(address)
                if self.chains_len[index] <= self.h_k:
                    target_index = self.evictor.evict([(i, self.preds[i]) for i, mark in enumerate(self.marked) if mark == 0])
                else:
                    target_index = random.choice([i for i, mark in enumerate(self.marked) if mark == 0])
                self.chains_rep[index] = self.cache[target_index]

        self.cache[target_index], self.pcs[target_index] = address, pc
        self.marked[target_index] = 1
        self.after_pred(pc, address, target_index)
        return hit
    

class RPBMarkerAlgorithm(PredictAlgorithm):
    """
    RPBMarker algorithm

    Designed by Thodoris Lykouris and Sergei Vassilvitskii. 2018. Competitive Caching with Machine Learned Advice.
    https://dl.acm.org/doi/10.1145/3447579
    """
    def __init__(self, associativity, evictor_type: Union[Type[Evictor], partial], predictor_type: Union[Predictor, partial], pred_budget: int = 0) -> None:
        def harmonic_number(k):
            return sum(1 / i for i in range(1, k + 1))
        super().__init__(associativity, evictor_type, predictor_type)
        self.marked = [0] * associativity
        self.k = associativity
        self.tracking_set = []
        self.h_k = harmonic_number(associativity)
        self.chains_len = []
        self.chains_rep = []
        self.budget_threshold = int(pred_budget)
    
    def access(self, pc, address):
        hit = False
        self.before_pred(pc, address)
        target_index = -1

        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            if all(mark == 1 for mark in self.marked):
                # new phase
                self.tracking_set = copy.deepcopy(self.cache)
                self.marked = [0] * self.associativity
                self.pred_budget = self.budget_threshold
                self.prev_unmark_num = self.k
    
            if address not in self.tracking_set:
                target_index = self.evictor.evict([(i, self.preds[i]) for i, mark in enumerate(self.marked) if mark == 0])
                self.chains_len.append(1)
                self.chains_rep.append(self.cache[target_index])
            if address in self.tracking_set:
                index = self.chains_rep.index(address)
                unmarked_num = len([i for i, mark in enumerate(self.marked) if mark == 0])
                if unmarked_num <= self.prev_unmark_num / 2.718 - 1:
                    self.pred_budget += 1

                if self.pred_budget > 0:
                    target_index = self.evictor.evict([(i, self.preds[i]) for i, mark in enumerate(self.marked) if mark == 0])
                    self.pred_budget -= 1
                else:
                    target_index = random.choice([i for i, mark in enumerate(self.marked) if mark == 0])
                self.chains_rep[index] = self.cache[target_index]

        self.cache[target_index], self.pcs[target_index] = address, pc
        self.marked[target_index] = 1
        self.after_pred(pc, address, target_index)
        return hit

class LMarker(PredictAlgorithm):
    """
    LMARKER Algorithm

    Designed by Dhruv Rohatgi. 2020. Near-Optimal Bounds for Online Caching with Machine Learned Advice
    https://epubs.siam.org/doi/10.1137/1.9781611975994.112
    """
    def __init__(self, associativity, evictor_type: Union[Type[Evictor], partial], predictor_type: Union[Predictor, partial]) -> None:
        super().__init__(associativity, evictor_type, predictor_type)

        self.stale = []
        self.marked = [0] * associativity
    
    def access(self, pc, address):
        target_index = -1
        hit = False

        self.before_pred(pc, address)
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            if all(mark == 1 for mark in self.marked):
                self.stale = copy.deepcopy(self.cache)
                self.marked = [0] * self.associativity
            
            if address in self.stale:
                target_index = random.choice([i for i, mark in enumerate(self.marked) if mark == 0])
            else:
                target_index = self.evictor.evict([(i, self.preds[i]) for i, mark in enumerate(self.marked) if mark == 0])
        
        self.cache[target_index], self.pcs[target_index] = address, pc
        self.marked[target_index] = 1
        self.after_pred(pc, address, target_index)
        return hit

class LNonMarker(PredictAlgorithm):
    """
    LNONMARKER Algorithm

    Designed by Dhruv Rohatgi. 2020. Near-Optimal Bounds for Online Caching with Machine Learned Advice
    https://epubs.siam.org/doi/10.1137/1.9781611975994.112
    """
    def __init__(self, associativity, evictor_type: Union[Type[Evictor], partial], predictor_type: Union[Predictor, partial]) -> None:
        super().__init__(associativity, evictor_type, predictor_type)

        self.phase = set()
        self.stale = []
        self.marked = [0] * associativity
        self.evicts = {}
    
    def access(self, pc, address):
        target_index = -1
        hit = False
        self.before_pred(pc, address)

        if len(self.phase) == self.associativity:
            self.stale = copy.deepcopy(self.cache)
            self.marked = [0] * self.associativity
            self.evicts = {}
            self.phase = set()

        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            if address in self.stale:
                if self.evicts[address] not in self.stale:
                    target_index = random.choice(range(self.associativity))
                else:
                    target_index = random.choice([i for i, mark in enumerate(self.marked) if mark == 0])
            else:
                target_index = self.evictor.evict([(i, self.preds[i]) for i, mark in enumerate(self.marked) if mark == 0])
        
        self.evicts[self.cache[target_index]] = address
        self.cache[target_index], self.pcs[target_index] = address, pc
        self.marked[target_index] = 1
        self.phase.add(address)
        self.after_pred(pc, address, target_index)
        return hit

class Mark0(PredictAlgorithm):
    """
    MARK0 Eviction Strategy

    Designed by Antonios Antoniadis, Joan Boyar, Marek Eliáš, Lene M. Favrholdt, Ruben Hoeksma, Kim S. Larsen, Adam Polak, and Bertrand Simon. 2023. Paging with Succinct Prediction.
    https://dl.acm.org/doi/10.5555/3618408.3618447
    """
    def __init__(self, associativity, evictor_type: Union[Type[Evictor], partial], predictor_type: Union[Predictor, partial]):
        super().__init__(associativity, evictor_type, predictor_type)
        if not isinstance(self.predictor, BinaryPredictor):
            raise ValueError('Mark0: predictor must be a BinaryPredictor')
        self.marked = [0] * associativity
        self.S_address = [None] * associativity
        self.S_visited = [0] * associativity
    
    def access(self, pc, address):
        target_index = -1
        hit = False

        self.before_pred(pc, address)
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            if address in self.S_address and 0 in self.S_visited:
                target_index = random.choice([i for i, visited in enumerate(self.S_visited) if visited == 0])
            else:
                target_index = self.cache.index(None)
        else:
            if all(visited == 1 for visited in self.S_visited):
                self.marked = [0] * self.associativity
                self.S_address = copy.deepcopy(self.cache)
                self.S_visited = [0] * self.associativity

            if address in self.S_address and 0 in self.S_visited:
                target_index = random.choice([i for i, visited in enumerate(self.S_visited) if visited == 0])
            else:
                target_index = random.choice([i for i, mark in enumerate(self.marked) if mark == 0])
        
        self.S_address[target_index] = None
        self.S_visited[target_index] = 1
        self.marked[target_index] = 1
        self.cache[target_index], self.pcs[target_index] = address, pc
        self.after_pred(pc, address, target_index)
        if self.preds[target_index] == 1:
            self.cache[target_index], self.pcs[target_index] = None, None
        return hit

class MarkAndPredict(PredictAlgorithm):
    """
    MARK&PREDICT Eviction Strategy

    Designed by Antonios Antoniadis, Joan Boyar, Marek Eliáš, Lene M. Favrholdt, Ruben Hoeksma, Kim S. Larsen, Adam Polak, and Bertrand Simon. 2023. Paging with Succinct Prediction.
    https://dl.acm.org/doi/10.5555/3618408.3618447
    """
    def __init__(self, associativity, evictor_type: Union[Type[Evictor], partial], predictor_type: Union[Predictor, partial]):
        super().__init__(associativity, evictor_type, predictor_type)
        if not isinstance(self.predictor, PhasePredictor):
            raise ValueError('MarkAndPredict: predictor must be a PhasePredictor')
        if not isinstance(self.evictor, BinaryEvictor):
            raise ValueError('MarkAndPredict: evictor must be a BinaryEvictor')
        self.marked = [0] * associativity
    
    def access(self, pc, address):
        target_index = -1
        hit = False

        self.before_pred(pc, address)
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            if all(mark == 1 for mark in self.marked):
                self.marked = [0] * self.associativity
            target_index = self.evictor.evict([(i, self.preds[i]) for i, mark in enumerate(self.marked) if mark == 0])
        
        self.cache[target_index], self.pcs[target_index] = address, pc
        self.marked[target_index] = 1
        self.after_pred(pc, address, target_index)
        return hit

class FollowerRobust(PredictAlgorithm):
    """
    F&R Algorithm

    Parameters:

    - a

    - lazy_evictor_type

    Designed by Karim Abdel Sadek and Marek Elias. 2024. Algorithms for Caching and MTS with reduced number of predictions.
    https://arxiv.org/abs/2404.06280
    """
    @staticmethod
    def create_windows(S, W, F, k, a):
        def func(i):    
            return (2**(i+1))-1
        for i in range(0, int(np.log2(k)) + 1):
            S.append((int(k - (k // (2 ** i)) + 1)))
        for i in range(1, int(np.log2(k)) + 1):
            n = []
            for h in range(S[i - 1], S[i]):
                n.append(h)
            W.append(n)
        W.append([S[-1]])
        for g in range(0, len(W)-1):
            gap = int(len(W[g])//(func(g+1)-func(g)))
            if (gap >= a):
                for m in W[g][::gap]:
                    F.append(m)
            else:
                for m in range(S[g], S[-1]+1, a):
                    F.append(m)
                break
        if k == 10:
            F = [1,6,9]
        return S, W, F

    @staticmethod
    def differ(a, b):
        aa = list(a).copy()
        bb = list(b).copy()
        for x in bb:
            if x == None:
                continue
            elif x in aa:
                aa.remove(x)
        if aa == []:
            return bb
        return aa

    def __init__(self, associativity, evictor_type: Union[Type[Evictor], partial], predictor_type: Union[Predictor, partial], **kwargs):
        super().__init__(associativity, evictor_type, predictor_type)
        if not isinstance(self.predictor, StatePredictor):
            raise ValueError('FollowerRobust: predictor must be a StatePredictor')

        if 'boost' in kwargs:
            self.boost = kwargs['boost']
        else:
            self.boost = False
        self.boost_beladys = []
        self.online_belady_cache = [None] * associativity
        self.online_belady_dis = [np.inf] * associativity
        self.boost_beladys.append(copy.deepcopy(self.online_belady_cache))
        if self.boost:
            def oracle_access(self, pc, address, next_access_time):
                if address in self.online_belady_cache:
                    target_index = self.online_belady_cache.index(address)
                elif None in self.online_belady_cache:
                    target_index = self.online_belady_cache.index(None)
                else:
                    target_index = self.online_belady_dis.index(max(self.online_belady_dis))
                
                self.online_belady_cache[target_index] = address
                self.online_belady_dis[target_index] = next_access_time
                self.boost_beladys.append(copy.deepcopy(self.online_belady_cache))
                if hasattr(self.predictor, 'oracle_access'):
                    self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        if 'a' in kwargs:
            self.a = kwargs['a']
        else:
            self.a = 1
        if 'lazy_evictor_type' in kwargs:
            if kwargs['lazy_evictor_type'] is None:
                self.lazy_evictor = None
            else:
                self.lazy_evictor = kwargs['lazy_evictor_type']()
        else:
            self.lazy_evictor = LRUEvictor()
        self.key_scores = [np.inf] * self.associativity if self.lazy_evictor is not None else None
        self.sim_cache = [None] * associativity
        self.sim_pcs = [None] * associativity
        self.traces = []
        
        self.S = []
        self.W = []
        self.F = []
        if (self.a == 1):
            self.S, self.W, self.F = FollowerRobust.create_windows(self.S, self.W, self.F, self.associativity, self.a)
        self.skip = 0
        self.pred_gap = 0
        self.follow_cost = 0
        self.belady_cost = 0
        self.marked = []
        self.old = []
        self.unmarked = []
        self.unmarked_for_reload = []
        self.clean = []
        self.prediction = [None] * self.associativity

    def online_belady(self):
        if self.boost:
            return self.boost_beladys[self.timestamp]
        else:
            cache = []
            for i, current in enumerate(self.traces):
                if current in cache:
                    continue
                if len(cache) < self.associativity:
                    cache.append(current)
                else:
                    future_uses = {item: self.traces[i + 1:].index(item) if item in self.traces[i + 1:] else float('inf') for item in cache}
                    to_remove = max(future_uses, key=future_uses.get)
                    cache.remove(to_remove)
                    cache.append(current)
            return cache
    
    def follow_robust(self, pc, address):
        target_index = -1
        # get next state
        if self.cur_boost_type is not None and self.cur_boost_type == 'before':
            preds = self.cur_boost_pred
        else:
            preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        assert(preds is not None)
        f = copy.deepcopy(self.online_belady())
        if address in self.sim_cache:
            target_index = self.sim_cache.index(address)
            self.sim_cache[target_index] = address
        elif None in self.sim_cache:
            index_to_evict = self.sim_cache.index(None)
            self.sim_cache[index_to_evict] = address
            self.prediction = copy.deepcopy(preds)
        if address not in self.sim_cache:
            target_index = None
            if self.skip == 0:
                self.follow_cost += 1
                if address not in f:
                    self.belady_cost +=1
                if address not in self.prediction and (self.follow_cost <= self.belady_cost):
                    if self.pred_gap <= 0:
                        self.prediction = preds
                        self.pred_gap = self.a
                        dd = self.differ(self.sim_cache, self.prediction)
                        target_index = self.sim_cache.index(random.choice(dd))
                        assert(self.sim_cache[target_index] not in self.prediction)
                        self.sim_cache[target_index] = address
                    else:
                        target_index = random.choice(range(self.associativity))
                        self.sim_cache[target_index] = address
                elif address in self.prediction:
                    dd = self.differ(self.sim_cache, self.prediction)
                    target_index = self.sim_cache.index(random.choice(dd))
                    self.sim_cache[target_index] = address
                else:
                    self.follow_cost = 0
                    self.belady_cost = 0
                    self.skip = self.associativity
                    self.old = []
                    for req in self.traces[self.timestamp-1::-1]:
                        if (req not in self.old) and (req != address):
                            self.old.append(req)
                        if len(self.old) >= self.associativity:
                            break
                    assert(len(self.old)==self.associativity)
                    self.unmarked = self.old.copy()
                    self.sim_cache = self.old.copy()
                    assert(address not in self.sim_cache)
                    self.marked = []
                    self.unmarked_for_reload = []
                    self.clean = []
            if self.skip != 0:
                assert(address not in self.sim_cache)
                if address not in self.marked:
                    self.skip -= 1
                    arrival_no = self.associativity-self.skip
                    if address in self.unmarked:
                        self.unmarked.remove(address)
                    if address not in self.marked:
                        self.marked.append(address)
                    assert(len(self.marked) == arrival_no)
                    if address not in self.old:
                        self.clean.append(address)
                    assert(len(self.unmarked) == self.associativity - (arrival_no - len(self.clean)))
                    if ((self.a==1) and (arrival_no in self.F)) or ((self.a > 1) and (self.pred_gap <= 0)):
                        self.pred_gap = self.a
                        self.prediction = copy.deepcopy(preds)
                    if arrival_no in self.S:
                        self.unmarked_for_reload = []
                        for p in self.unmarked:
                            if (p in self.prediction) and (p not in self.sim_cache):
                                self.unmarked_for_reload.append(p)
                    if address in self.unmarked_for_reload:
                        # Lazy sync with predictor
                        assert(address not in self.sim_cache)
                        dd = self.differ(self.sim_cache, self.prediction)
                        target_index = self.sim_cache.index(random.choice(dd))
                        self.sim_cache[target_index] = address
                    if address in self.clean: # Clean arrival
                            assert(address not in self.sim_cache)
                            dd = self.differ(self.sim_cache, self.prediction)
                            target_index = self.sim_cache.index(random.choice(dd))
                            self.sim_cache[target_index] = address
                if address not in self.sim_cache:
                    index_to_evict = None
                    unmarked_slots = []
                    for page in self.sim_cache:
                        if page in self.unmarked:
                            unmarked_slots.append(self.sim_cache.index(page))
                    target_index = random.choice(unmarked_slots)
                    assert(address not in self.sim_cache)
                    self.sim_cache[target_index] = address
                if self.skip == 0:
                    assert(len(self.marked) == self.associativity)
                    assert(len(self.unmarked) == len(self.clean))
        if self.cur_boost_type is not None:
            assert self.cur_boost_type == 'before'
        else:
            pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
            assert pred is None
        self.pred_gap -= 1
        self.traces.append(address)
    
    def access(self, pc, address):
        self.follow_robust(pc, address)

        ## Lazy
        target_index = -1
        hit = False
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            if self.lazy_evictor is None:
                self.cache = copy.deepcopy(self.sim_cache)
                self.pcs = copy.deepcopy(self.sim_pcs)
                target_index = self.cache.index(address)
            else:
                diff_keys = set(self.cache) - set(self.sim_cache)
                target_index = self.lazy_evictor.evict([(self.cache.index(k), self.key_scores[self.cache.index(k)] if self.key_scores is not None else 0) for k in diff_keys])
        
        self.key_scores[target_index] = self.timestamp
        self.cache[target_index], self.pcs[target_index] = address, pc
        self.timestamp += 1
        return hit

class Guard(PredictAlgorithm):
    """
    Guard algorithm

    Parameters:
    
    - follow_if_guarded

    - relax_times

    - relax_prob

    Our work
    """
    def __init__(self, associativity, evictor_type: Union[Type[Evictor], partial], predictor_type: Union[Predictor, partial], **kwargs) -> None:
        super().__init__(associativity, evictor_type, predictor_type)
        self.old_unvisited_set = []
        self.unguarded_set = []
        self.phase_evicted_set = set()
        self.error_times = 0

        if 'follow_if_guarded' in kwargs:
            self.follow_if_guarded = kwargs['follow_if_guarded']
        else:
            self.follow_if_guarded = False
        if 'relax_times' in kwargs:
            self.relax_times = kwargs['relax_times']
        else:
            self.relax_times = 0
        if 'relax_prob' in kwargs:
            self.relax_prob = kwargs['relax_prob']
        else:
            self.relax_prob = 0
    
    def access(self, pc, address):
        to_guard = False
        target_index = -1
        hit = False

        self.before_pred(pc, address)
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            if not self.old_unvisited_set:
                self.old_unvisited_set = list(range(self.associativity))
                self.unguarded_set = list(range(self.associativity))
                self.phase_evicted_set = set()
                self.error_times = 0
            
            if address in self.phase_evicted_set:
                if self.relax_times != 0:
                    self.error_times += 1
                    if self.error_times >= self.relax_times:
                        to_guard = True
                else:
                    if random.random() > self.relax_prob:
                        to_guard = True

            if to_guard and not self.follow_if_guarded:
                target_index = random.choice(self.old_unvisited_set)
            else:
                target_index = self.evictor.evict([(i, self.preds[i]) for i in self.unguarded_set])
            
            self.phase_evicted_set.add(self.cache[target_index])

        if target_index in self.old_unvisited_set:
            self.old_unvisited_set.remove(target_index)

        if to_guard:
            self.unguarded_set.remove(target_index)
        
        self.cache[target_index], self.pcs[target_index] = address, pc
        self.after_pred(pc, address, target_index)
        return hit

#######################################################################

class CombineAlgorithm(EvictAlgorithm):
    def __init__(self, associativity, candidate_algorithms: List[Union[EvictAlgorithm, partial]], lazy_evictor_type: Union[LRUEvictor, RandEvictor, None] = LRUEvictor):
        if lazy_evictor_type is not None and not issubclass(lazy_evictor_type, Evictor):
            raise ValueError('CombineAlgorithm: Invalid Evictor')
        
        super().__init__(associativity)
        self.oracle_algs = []
        self.boost_algs = []
        self.candidate_algs = []
        self.center = 0
        self.timestamp = 0
        self.lazy_evictor = lazy_evictor_type() if lazy_evictor_type is not None else None
        # self.key_scores = {} if lazy_evictor_type == LRUEvictor else None
        self.key_scores = [np.inf] * associativity if lazy_evictor_type == LRUEvictor else None

        for alg_type in candidate_algorithms:
            alg_instance = alg_type(associativity)
            self.candidate_algs.append([alg_instance, 0])
            if hasattr(alg_instance, 'oracle_access'):
                self.oracle_algs.append(alg_instance)
            if hasattr(alg_instance, 'boost_access'):
                self.boost_algs.append(alg_instance)

        if len(self.oracle_algs) != 0:
            def oracle_access(self, pc, address, next_access_time):
                for oracle_alg in self.oracle_algs:
                    oracle_alg.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)
        
        if len(self.candidate_algs) < 2:
            raise ValueError('CombineAlgorithm: Algorithm Count < 2')

    def __push_candidates__(self, pc, address):
        for i, (alg, _) in enumerate(self.candidate_algs):
            if not alg.access(pc, address):
                self.candidate_algs[i][1] += 1
                self.__trigger_miss__(i, address)
    
    def __push_candidates_boost__(self, pc, address, boost_pred):
        for i, (alg, _) in enumerate(self.candidate_algs):
            if alg in self.boost_algs:
                hit = alg.boost_access(pc, address, boost_pred)
            else:
                hit = alg.access(pc, address)
            if not hit:
                self.candidate_algs[i][1] += 1
                self.__trigger_miss__(i, address)
    
    def __trigger_miss__(self, i, address):
        pass

    @abstractmethod
    def __trigger_elect_center__(self):
        pass

    def __process__(self, pc, address):
        target_index = -1
        hit = False
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            self.__trigger_elect_center__()
            center_cache = self.candidate_algs[self.center][0].cache
            if self.lazy_evictor is None:
                self.cache = copy.deepcopy(center_cache)
                self.pcs = copy.deepcopy(self.candidate_algs[self.center][0].pcs)
                target_index = self.cache.index(address)
            else:
                diff_keys = set(self.cache) - set(center_cache)
                target_index = self.lazy_evictor.evict([(self.cache.index(k), self.key_scores[self.cache.index(k)] if self.key_scores is not None else 0) for k in diff_keys])
        if self.key_scores is not None:
            self.key_scores[target_index] = self.timestamp
        self.cache[target_index], self.pcs[target_index] = address, pc
        self.timestamp += 1
        return hit

    def boost_access(self, pc, address, boost_pred):
        self.__push_candidates_boost__(pc, address, boost_pred)
        return self.__process__(pc, address)

    def access(self, pc, address):
        self.__push_candidates__(pc, address)
        return self.__process__(pc, address)

class CombineDeterministicAlgorithm(CombineAlgorithm):
    """
    black-box algorithm

    Designed by Thodoris Lykouris and Sergei Vassilvitskii. 2018. Competitive Caching with Machine Learned Advice.
    https://dl.acm.org/doi/10.1145/3447579
    """
    def __init__(self, associativity, candidate_algorithms: List[Union[EvictAlgorithm, partial]], switch_bound=2, lazy_evictor_type: Union[LRUEvictor, RandEvictor, None] = LRUEvictor):
        super().__init__(associativity, candidate_algorithms, lazy_evictor_type)
        self.switch_bound = switch_bound

    def __trigger_elect_center__(self):
        this_cost = self.candidate_algs[self.center][1]
        min_center, (_, min_cost) = min(enumerate(self.candidate_algs), key=lambda x: x[1][1])
        if this_cost >= self.switch_bound * min_cost:
            self.center = min_center

class CombineRandomAlgorithm(CombineAlgorithm):
    """
    Algorithm THRESH

    Designed by Avrim Blum and Carl Burch. 1997. On-line learning and the metrical task system problem.
    https://dl.acm.org/doi/10.1145/267460.267475
    """
    def __init__(self, associativity, candidate_algorithms: List[Union[EvictAlgorithm, partial]], alpha=0.0, beta=0.99, lazy_evictor_type: Union[LRUEvictor, RandEvictor, None] = LRUEvictor):
        super().__init__(associativity, candidate_algorithms, lazy_evictor_type)
        self.alpha = alpha
        self.beta = beta
        self.n = len(self.candidate_algs)
        self.weights = [1] * self.n
        self.probs = [1/self.n] * self.n
    
    def __trigger_miss__(self, i, key):
        self.weights[i] *= self.beta
    
    def __trigger_elect_center__(self):
        W = sum(self.weights)
        threshold = self.alpha * W / self.n
        new_probs = [w / W for w in self.weights]
        if new_probs[self.center] < self.probs[self.center]:
            threshold = 1 - new_probs[self.center] / self.probs[self.center]
            if random.random() > threshold:
                self.center = self.center
            else:
                index = list(range(self.n))
                index.remove(self.center)
                probs = copy.deepcopy(new_probs)
                probs.pop(self.center)
                self.center = random.choices(index, weights=probs)[0]
        self.probs = new_probs

        # valid_index, valid_weights = zip(*[(i, weight) for i, weight in enumerate(self.weights) if weight > threshold])
        # if valid_weights:
        #     self.center = random.choices(valid_index, weights=valid_weights)[0]

class CombineWeightsAlgorithm(CombineAlgorithm):
    """
    Imitation learing for Parrot
    """
    def __init__(self, associativity, candidate_algorithms: List[Union[EvictAlgorithm, partial]], weights: Union[List[float], None], lazy_evictor_type: Union[LRUEvictor, RandEvictor, None] = LRUEvictor):
        super().__init__(associativity, candidate_algorithms, lazy_evictor_type)
        self.n = len(self.candidate_algs)
        if weights is not None:
            self.weights = weights
        else:
            self.weights = [1] * self.n
    
    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.candidate_algs[self.center][0].preds)

    def reset(self, weights):
        self.weights = weights

    def __trigger_elect_center__(self):
        self.center = random.choices(list(range(self.n)), weights=self.weights)[0]

#######################################################################

class RandAlgorithm(EvictAlgorithm):
    def __init__(self, associativity):
        super().__init__(associativity)
        self.evictor = RandEvictor()
    
    def access(self, pc, address):
        target_index = -1
        hit = False
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            target_index = self.evictor.evict(list(enumerate(self.cache)))
        
        self.cache[target_index] = address
        self.pcs[target_index] = pc
        return hit

class LRUAlgorithm(EvictAlgorithm):
    def __init__(self, associativity):
        super().__init__(associativity)
        self.evictor = LRUEvictor()
        self.scores = [0] * associativity
        self.timestamp = 0
    
    def access(self, pc, address):
        target_index = -1
        hit = False
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            target_index = self.evictor.evict(list(enumerate(self.scores)))
        
        self.cache[target_index] = address
        self.pcs[target_index] = pc
        self.scores[target_index] = self.timestamp
        self.timestamp += 1
        return hit

class MarkerAlgorithm(EvictAlgorithm):
    def __init__(self, associativity):
        super().__init__(associativity)
        self.evictor = MarkerEvictor()
        self.scores = [0] * associativity
    
    def access(self, pc, address):
        if all(x == 1 for x in self.scores):
            self.scores = [0] * self.associativity

        target_index = -1
        hit = False
        if address in self.cache:
            target_index = self.cache.index(address)
            hit = True
        elif None in self.cache:
            target_index = self.cache.index(None)
        else:
            target_index = self.evictor.evict(list(enumerate(self.scores)))
        
        self.cache[target_index] = address
        self.pcs[target_index] = pc
        self.scores[target_index] = 1
        return hit


class OnlineMinAlgorithm(EvictAlgorithm):
    """OnlineMin: A Fast Strongly Competitive Randomized Paging Algorithm.

    This implementation follows the paper's high-level algorithm (Section 3.1):
    - Maintain support layers L1..Lk (Equitable2 update rule with forgiveness).
    - Maintain a random priority (rank) for each page in the support.
    - On a miss, evict the minimum-priority page from a specific prefix of cache
      pages (determined by layers), then update layers.

    Notes for this codebase:
    - The theoretical algorithm starts with k distinct pages already in cache.
      Here we warm up until the set is full, then initialize layers as k
      singleton (revealed) layers in LRU order.
    """

    def __init__(self, associativity: int, max_support_factor: int = 3):
        super().__init__(associativity)
        self.k = associativity
        if max_support_factor < 1:
            raise ValueError('OnlineMin: max_support_factor must be >= 1')
        self.max_support_factor = int(max_support_factor)
        self.max_support = self.max_support_factor * self.k

        # Support layers: index 1..k used; index 0 unused.
        self.layers = [set() for _ in range(self.k + 1)]
        self.support = set()

        # Random priorities (lower = smaller priority). Only for pages in support.
        self.priority = {}
        self._free_priorities = list(range(1, self.max_support + 1))

        # Page -> current layer index (1..k). Only defined for pages in support.
        self.layer_of = {}

        # Warm-up: maintain recency order until cache becomes full.
        self._inited = False
        self._warmup_lru = []  # oldest -> newest

        # Record last access time for pages in support (layers 1..k).
        self.timestamp = 0
        self.support_last_access: Dict[object, int] = {}
        self.support_last_evict: Dict[object, int] = {}

        # Stats: count requests to L0 vs non-L0 (after initialization).
        self.l0_requests = 0
        self.non_l0_requests = 0

    def _assign_priority(self, page):
        if page in self.priority:
            return
        if not self._free_priorities:
            raise ValueError('OnlineMin: priority universe exhausted')
        pr = random.choice(self._free_priorities)
        self._free_priorities.remove(pr)
        self.priority[page] = pr

    def _delete_priority(self, page):
        pr = self.priority.pop(page, None)
        if pr is not None:
            self._free_priorities.append(pr)

    def _rebuild_layer_of(self):
        new_layer_of = {}
        new_support = set()
        for i in range(1, self.k + 1):
            for p in self.layers[i]:
                new_layer_of[p] = i
                new_support.add(p)

        # Reclaim priorities for pages that left the support BEFORE assigning
        # priorities to newly added support pages.
        for p in list(self.priority.keys()):
            if p not in new_support:
                self._delete_priority(p)

        if len(new_support) > self.max_support:
            raise RuntimeError(
                'OnlineMin invariant violated: support size exceeded 3k. '
                f'support_size={len(new_support)} max_support={self.max_support}'
            )

        self.layer_of = new_layer_of
        self.support = new_support

        # Keep last-access records only for pages still in support.
        for p in list(self.support_last_access.keys()):
            if p not in new_support:
                self.support_last_access.pop(p, None)

        # Keep last-evict records only for pages still in support.
        for p in list(self.support_last_evict.keys()):
            if p not in new_support:
                self.support_last_evict.pop(p, None)

        for p in new_support:
            self._assign_priority(p)

    def _init_layers_from_warmup(self):
        # Build k singleton (revealed) layers from warm-up LRU order.
        pages = [p for p in self._warmup_lru if p is not None]
        if len(pages) < self.k:
            return
        pages = pages[-self.k:]
        self.layers = [set() for _ in range(self.k + 1)]
        for i, p in enumerate(pages, start=1):
            self.layers[i] = {p}
            self._assign_priority(p)
        self._rebuild_layer_of()
        self._inited = True

    def _update_layers_after_request(self, page, layer_i: int, forgiveness: bool):
        # Apply Definition 1 (Equitable2 update rule with forgiveness).
        if layer_i == 0:
            if not forgiveness:
                # (L0\{p}, L1, ..., L_{k-2}, L_{k-1} ∪ L_k, {p})
                if self.k >= 2:
                    self.layers[self.k - 1] |= self.layers[self.k]
                self.layers[self.k] = {page}
            else:
                # (L0\{p} ∪ L1, L2, ..., L_k, {p})  => support loses old L1.
                dropped = self.layers[1]
                for q in dropped:
                    self._delete_priority(q)
                for j in range(1, self.k):
                    self.layers[j] = self.layers[j + 1]
                self.layers[self.k] = {page}
        else:
            i = layer_i
            if i < self.k:
                # (.., L_{i-1} ∪ (L_i \ {p}), L_{i+1}, ..., L_k, {p})
                self.layers[i - 1] |= (self.layers[i] - {page})
                for j in range(i, self.k):
                    self.layers[j] = self.layers[j + 1]
                self.layers[self.k] = {page}
            else:
                # i == k: page already in Lk (revealed); keep singleton.
                self.layers[self.k] = {page}

        self._rebuild_layer_of()

    def access(self, pc, address):
        ts = self.timestamp
        # Warm-up phase: behave like a simple LRU fill until full, then init.
        if not self._inited:
            if address in self.cache:
                hit = True
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.support_last_access[address] = ts
                self.timestamp += 1
                return hit

            hit = False
            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.support_last_access[address] = ts
                self.timestamp += 1
                return hit

            # Cache is full but layers not initialized (should be rare). Initialize now.
            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # OnlineMin proper.
        hit = address in self.cache
        if hit:
            idx = self.cache.index(address)
            self.pcs[idx] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)

        # Count requests by layer (L0 is implicit as "not in support").
        if layer_i == 0:
            self.l0_requests += 1
        else:
            self.non_l0_requests += 1

        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        # Fail fast with a clear error instead of crashing with KeyError in
        # `self.layer_of[p]` if invariants were broken earlier.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'OnlineMin invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        if not hit:
            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
            else:
                cache_pages = [p for p in self.cache if p is not None]
                if eviction_layer == 0:
                    victim = min(cache_pages, key=lambda p: self.priority[p])
                else:
                    # Identify the prefix of
                    # Sort cache pages by increasing layer index.
                    cache_pages.sort(key=lambda p: self.layer_of[p])
                    j = None
                    for jj in range(eviction_layer, self.k + 1):
                        if self.layer_of[cache_pages[jj - 1]] == jj:
                            j = jj
                            break
                    if j is None:
                        # Should not happen; fall back to evicting global min priority.
                        victim = min(cache_pages, key=lambda p: self.priority[p])
                    else:
                        # unexpected = 0
                        # for x in range(j + 1, self.k + 1):
                        #     if len(self.layers[x]) > 1:
                        #         print('OnlineMin warning: unexpected layer structure during eviction')
                        #         unexpected = 1
                        #         break
                        # if unexpected:
                        #     for x in range(j + 1, self.k + 1):
                        #         # print(
                        #         #     f'  Critical Layer {x}: {[(p, ts_p) for p in self.layers[x] for ts_p in [self.support_last_access.get(p)] if self.support_last_evict.get(address) is not None and ts_p is not None and ts_p < self.support_last_evict.get(address)]} '
                        #         #     f'addr_last_evict={self.support_last_evict.get(address)}'
                        #         # )
                        #         print(
                        #             f'  Layer {x}: {[(p, self.support_last_access.get(p)) for p in self.layers[x]]} '
                        #             f'addr_last_evict={self.support_last_evict.get(address)}'
                        #         )
                        prefix = cache_pages[:j]
                        victim = min(prefix, key=lambda p: self.priority[p])

                victim_idx = self.cache.index(victim)
                self.support_last_evict[victim] = ts
                self.cache[victim_idx] = address
                self.pcs[victim_idx] = pc

        # Update layers after cache update (as in the paper).
        self._update_layers_after_request(address, layer_i, forgiveness)

        # Record last access for the accessed page (kept only if in support).
        self.support_last_access[address] = ts

        # Support can be larger than the cache (up to 3k), so support\cache is allowed.
        # But cache must never contain pages outside the support.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'OnlineMin invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
                f'extras={extras} '
                f'cache={list(self.cache)} support_size={len(self.support)}'
            )

        self.timestamp += 1
        return hit


class StrictOnlineMinAlgorithm(EvictAlgorithm):
    """StrictOnlineMin: OnlineMin with eviction-chain ES and ES-filtered eviction.

    Same as `OnlineMinAlgorithm` for support/layer/priority maintenance, except:
    - When a true L0 miss causes an eviction of x (cache full), start an eviction
      chain and record:
        ES(x) = t (the current time)
        Y(x)  = #{p in candidate1 : last_access_time[p] < t}
      where candidate1 is the eviction candidate set chosen by OnlineMin.
    - For all other evictions (including forgiveness and non-L0 misses), let the
      requested page be x'. Look up ES(x') and compute OnlineMin candidate1.
      If there exist pages in candidate1 with last_access_time < ES(x'), form
      candidate2 as those pages and evict the minimum-priority page within
      candidate2; otherwise evict the minimum-priority page within candidate1.
    """

    def __init__(self, associativity: int, max_support_factor: int = 3):
        super().__init__(associativity)
        self.k = associativity
        if max_support_factor < 1:
            raise ValueError('StrictOnlineMin: max_support_factor must be >= 1')
        self.max_support_factor = int(max_support_factor)
        self.max_support = self.max_support_factor * self.k

        # Support layers: index 1..k used; index 0 unused.
        self.layers = [set() for _ in range(self.k + 1)]
        self.support = set()

        # Random priorities (lower = smaller priority). Only for pages in support.
        self.priority = {}
        self._free_priorities = list(range(1, self.max_support + 1))

        # Page -> current layer index (1..k). Only defined for pages in support.
        self.layer_of = {}

        # Warm-up: maintain recency order until cache becomes full.
        self._inited = False
        self._warmup_lru = []  # oldest -> newest

        self.timestamp = 0
        self.support_last_access: Dict[object, int] = {}
        self.support_last_evict: Dict[object, int] = {}

        # Eviction-chain attributes for pages evicted on true L0 misses.
        self._es: Dict[object, int] = {}
        self._y: Dict[object, int] = {}

    def _assign_priority(self, page):
        if page in self.priority:
            return
        if not self._free_priorities:
            raise ValueError('StrictOnlineMin: priority universe exhausted')
        pr = random.choice(self._free_priorities)
        self._free_priorities.remove(pr)
        self.priority[page] = pr

    def _delete_priority(self, page):
        pr = self.priority.pop(page, None)
        if pr is not None:
            self._free_priorities.append(pr)

    def _rebuild_layer_of(self):
        new_layer_of = {}
        new_support = set()
        for i in range(1, self.k + 1):
            for p in self.layers[i]:
                new_layer_of[p] = i
                new_support.add(p)

        for p in list(self.priority.keys()):
            if p not in new_support:
                self._delete_priority(p)

        if len(new_support) > self.max_support:
            raise RuntimeError(
                'StrictOnlineMin invariant violated: support size exceeded cap. '
                f'support_size={len(new_support)} max_support={self.max_support}'
            )

        self.layer_of = new_layer_of
        self.support = new_support

        for p in list(self.support_last_access.keys()):
            if p not in new_support:
                self.support_last_access.pop(p, None)

        for p in list(self.support_last_evict.keys()):
            if p not in new_support:
                self.support_last_evict.pop(p, None)

        for p in new_support:
            self._assign_priority(p)

    def _init_layers_from_warmup(self):
        pages = [p for p in self._warmup_lru if p is not None]
        if len(pages) < self.k:
            return
        pages = pages[-self.k :]
        self.layers = [set() for _ in range(self.k + 1)]
        for i, p in enumerate(pages, start=1):
            self.layers[i] = {p}
            self._assign_priority(p)
        self._rebuild_layer_of()
        self._inited = True

    def _update_layers_after_request(self, page, layer_i: int, forgiveness: bool):
        if layer_i == 0:
            if not forgiveness:
                if self.k >= 2:
                    self.layers[self.k - 1] |= self.layers[self.k]
                self.layers[self.k] = {page}
            else:
                dropped = self.layers[1]
                for q in dropped:
                    self._delete_priority(q)
                for j in range(1, self.k):
                    self.layers[j] = self.layers[j + 1]
                self.layers[self.k] = {page}
        else:
            i = layer_i
            if i < self.k:
                self.layers[i - 1] |= (self.layers[i] - {page})
                for j in range(i, self.k):
                    self.layers[j] = self.layers[j + 1]
                self.layers[self.k] = {page}
            else:
                self.layers[self.k] = {page}

        self._rebuild_layer_of()

    def _onlinemin_candidate1(self, eviction_layer: int) -> List[object]:
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return []
        if eviction_layer == 0:
            return cache_pages
        cache_pages.sort(key=lambda p: self.layer_of[p])
        j = None
        for jj in range(eviction_layer, self.k + 1):
            if self.layer_of[cache_pages[jj - 1]] == jj:
                j = jj
                break
        if j is None:
            return cache_pages
        return cache_pages[:j]

    def _min_priority_victim(self, pages: List[object]) -> object:
        if not pages:
            return None
        return min(pages, key=lambda p: self.priority[p])

    def access(self, pc, address):
        ts = self.timestamp

        # Warm-up: identical behavior to OnlineMin warm-up.
        if not self._inited:
            if address in self.cache:
                hit = True
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.support_last_access[address] = ts
                self.timestamp += 1
                return hit

            hit = False
            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.support_last_access[address] = ts
                self.timestamp += 1
                return hit

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        hit = address in self.cache
        if hit:
            idx = self.cache.index(address)
            self.pcs[idx] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        # Invariant: cache ⊆ support.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'StrictOnlineMin invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        if not hit:
            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
            else:
                candidate1 = self._onlinemin_candidate1(eviction_layer)
                if not candidate1:
                    candidate1 = [p for p in self.cache if p is not None]

                if layer_i == 0 and not forgiveness:
                    # True L0 miss eviction: OnlineMin-style eviction + chain record on victim.
                    victim = self._min_priority_victim(candidate1)
                    victim_idx = self.cache.index(victim)
                    self.support_last_evict[victim] = ts

                    self._es[victim] = int(ts)
                    self._y[victim] = int(sum(1 for p in candidate1 if self.support_last_access.get(p, -1) < ts))
                else:
                    # Other evictions: ES-filtered candidate2 rule.
                    es_xprime = self._es.get(address)
                    if es_xprime is not None:
                        candidate2 = [p for p in candidate1 if self.support_last_access.get(p, -1) < es_xprime]
                    else:
                        candidate2 = []
                    victim = self._min_priority_victim(candidate2 if candidate2 else candidate1)
                    victim_idx = self.cache.index(victim)
                    self.support_last_evict[victim] = ts

                self.cache[victim_idx] = address
                self.pcs[victim_idx] = pc

        # Update layers after cache update.
        self._update_layers_after_request(address, layer_i, forgiveness)
        self.support_last_access[address] = ts

        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'StrictOnlineMin invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        self.timestamp += 1
        return hit
    

class OnlineMinRandomAlgorithm(EvictAlgorithm):
    """OnlineMinRandom: A Fast Strongly Competitive Randomized Paging Algorithm.

    This implementation follows the paper's high-level algorithm (Section 3.1):
    - Maintain support layers L1..Lk (Equitable2 update rule with forgiveness).
    - Maintain a random priority (rank) for each page in the support.
    - On a miss, evict the minimum-priority page from a specific prefix of cache
      pages (determined by layers), then update layers.

    Notes for this codebase:
    - The theoretical algorithm starts with k distinct pages already in cache.
      Here we warm up until the set is full, then initialize layers as k
      singleton (revealed) layers in LRU order.
    """

    def __init__(self, associativity: int, max_support_factor: int = 3):
        super().__init__(associativity)
        self.k = associativity
        if max_support_factor < 1:
            raise ValueError('OnlineMinRandom: max_support_factor must be >= 1')
        self.max_support_factor = int(max_support_factor)
        self.max_support = self.max_support_factor * self.k

        # Support layers: index 1..k used; index 0 unused.
        self.layers = [set() for _ in range(self.k + 1)]
        self.support = set()

        # Random priorities (lower = smaller priority). Only for pages in support.
        self.priority = {}
        self._free_priorities = list(range(1, self.max_support + 1))

        # Page -> current layer index (1..k). Only defined for pages in support.
        self.layer_of = {}

        # Warm-up: maintain recency order until cache becomes full.
        self._inited = False
        self._warmup_lru = []  # oldest -> newest

        # Record last access time for pages in support (layers 1..k).
        self.timestamp = 0
        self.support_last_access: Dict[object, int] = {}
        self.support_last_evict: Dict[object, int] = {}

        # Stats: count requests to L0 vs non-L0 (after initialization).
        self.l0_requests = 0
        self.non_l0_requests = 0

    def _assign_priority(self, page):
        if page in self.priority:
            return
        if not self._free_priorities:
            raise ValueError('OnlineMinRandom: priority universe exhausted')
        pr = random.choice(self._free_priorities)
        self._free_priorities.remove(pr)
        self.priority[page] = pr

    def _delete_priority(self, page):
        pr = self.priority.pop(page, None)
        if pr is not None:
            self._free_priorities.append(pr)

    def _rebuild_layer_of(self):
        new_layer_of = {}
        new_support = set()
        for i in range(1, self.k + 1):
            for p in self.layers[i]:
                new_layer_of[p] = i
                new_support.add(p)

        # Reclaim priorities for pages that left the support BEFORE assigning
        # priorities to newly added support pages.
        for p in list(self.priority.keys()):
            if p not in new_support:
                self._delete_priority(p)

        if len(new_support) > self.max_support:
            raise RuntimeError(
                'OnlineMinRandom invariant violated: support size exceeded 3k. '
                f'support_size={len(new_support)} max_support={self.max_support}'
            )

        self.layer_of = new_layer_of
        self.support = new_support

        # Keep last-access records only for pages still in support.
        # for p in list(self.support_last_access.keys()):
        #     if p not in new_support:
        #         self.support_last_access.pop(p, None)

        # Keep last-evict records only for pages still in support.
        # for p in list(self.support_last_evict.keys()):
        #     if p not in new_support:
        #         self.support_last_evict.pop(p, None)

        for p in new_support:
            self._assign_priority(p)

    def _init_layers_from_warmup(self):
        # Build k singleton (revealed) layers from warm-up LRU order.
        pages = [p for p in self._warmup_lru if p is not None]
        if len(pages) < self.k:
            return
        pages = pages[-self.k:]
        self.layers = [set() for _ in range(self.k + 1)]
        for i, p in enumerate(pages, start=1):
            self.layers[i] = {p}
            self._assign_priority(p)
        self._rebuild_layer_of()
        self._inited = True

    def _update_layers_after_request(self, page, layer_i: int, forgiveness: bool):
        # Apply Definition 1 (Equitable2 update rule with forgiveness).
        if layer_i == 0:
            if not forgiveness:
                # (L0\{p}, L1, ..., L_{k-2}, L_{k-1} ∪ L_k, {p})
                if self.k >= 2:
                    self.layers[self.k - 1] |= self.layers[self.k]
                self.layers[self.k] = {page}
            else:
                # (L0\{p} ∪ L1, L2, ..., L_k, {p})  => support loses old L1.
                dropped = self.layers[1]
                for q in dropped:
                    self._delete_priority(q)
                for j in range(1, self.k):
                    self.layers[j] = self.layers[j + 1]
                self.layers[self.k] = {page}
        else:
            i = layer_i
            if i < self.k:
                # (.., L_{i-1} ∪ (L_i \ {p}), L_{i+1}, ..., L_k, {p})
                self.layers[i - 1] |= (self.layers[i] - {page})
                for j in range(i, self.k):
                    self.layers[j] = self.layers[j + 1]
                self.layers[self.k] = {page}
            else:
                # i == k: page already in Lk (revealed); keep singleton.
                self.layers[self.k] = {page}

        self._rebuild_layer_of()

    def access(self, pc, address):
        ts = self.timestamp
        # Warm-up phase: behave like a simple LRU fill until full, then init.
        if not self._inited:
            if address in self.cache:
                hit = True
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.support_last_access[address] = ts
                self.timestamp += 1
                return hit

            hit = False
            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.support_last_access[address] = ts
                self.timestamp += 1
                return hit

            # Cache is full but layers not initialized (should be rare). Initialize now.
            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # OnlineMinRandom proper.
        hit = address in self.cache
        if hit:
            idx = self.cache.index(address)
            self.pcs[idx] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)

        # Count requests by layer (L0 is implicit as "not in support").
        if layer_i == 0:
            self.l0_requests += 1
        else:
            self.non_l0_requests += 1

        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        # Fail fast with a clear error instead of crashing with KeyError in
        # `self.layer_of[p]` if invariants were broken earlier.
        # cache_set = {p for p in self.cache if p is not None}
        # extras = [p for p in cache_set if p not in self.support]
        # if extras:
        #     raise RuntimeError(
        #         'OnlineMinRandom invariant violated: cache contains non-support pages (pre-evict). '
        #         f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
        #     )

        if not hit:
            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
            else:
                cache_pages = [p for p in self.cache if p is not None]
                eviction_candidates = []
                for jj in range(1, self.k + 1):
                    if self.support_last_evict.get(address) == None or self.support_last_access.get(cache_pages[jj - 1]) < self.support_last_evict.get(address):
                        eviction_candidates.append(cache_pages[jj - 1])
                if len(eviction_candidates) == 0:
                    # print("eviction_layer:", eviction_layer)
                    # for jj in range(1, self.k + 1):
                    #     print(f"Last access of cache page {cache_pages[jj - 1]}(layer {self.layer_of.get(cache_pages[jj - 1], 'N/A')}): {self.support_last_access.get(cache_pages[jj - 1])}, last evict of address {address}: {self.support_last_evict.get(address)}")
                    eviction_candidates = cache_pages
                if eviction_layer == 0:
                    #victim = min(cache_pages, key=lambda p: self.priority[p])
                    #victim = random.choice(cache_pages)
                    victim = min(cache_pages, key=lambda p: self.support_last_access.get(p, -1))
                else:
                    
                    #victim = min(eviction_candidates, key=lambda p: self.priority[p])
                    #victim = random.choice(eviction_candidates)
                    victim = min(eviction_candidates, key=lambda p: self.support_last_access.get(p, -1))
                    # Sort cache pages by increasing layer index.
                    # cache_pages.sort(key=lambda p: self.layer_of[p])
                    # j = None
                    # for jj in range(eviction_layer, self.k + 1):
                    #     if self.layer_of[cache_pages[jj - 1]] == jj:
                    #         j = jj
                    #         break
                    # for jj in range(eviction_layer, self.k + 1):
                    #     if self.support_last_access.get(cache_pages[jj - 1]) >= self.support_last_evict.get(address):
                    #         j = jj - 1
                    #         break
                    # for jj in range(self.k + 1, eviction_layer - 1, -1):
                    #     if len(self.layers[jj]) > 1:
                    #         j = jj
                    #         break
                    # if j is None:
                    #     # Should not happen; fall back to evicting global min priority.
                    #     victim = min(cache_pages, key=lambda p: self.priority[p])
                    #     print('OnlineMinRandom warning 1: unexpected layer structure during eviction')
                    # else:
                    #     unexpected = 0
                    #     for x in range(j + 1, self.k + 1):
                    #         if self.support_last_access.get(cache_pages[x - 1]) < self.support_last_evict.get(address):
                    #             print('OnlineMinRandom warning 2: unexpected layer structure during eviction')
                    #             unexpected = 1
                    #             break
                    #     if unexpected:
                    #         for x in range(j + 1, self.k + 1):
                    #             print(
                    #                 f'  Critical Layer {x}: {[(p, ts_p) for p in self.layers[x] for ts_p in [self.support_last_access.get(p)] if self.support_last_evict.get(address) is not None and ts_p is not None and ts_p < self.support_last_evict.get(address)]} '
                    #                 f'addr_last_evict={self.support_last_evict.get(address)}'
                    #             )
                    #             print(
                    #                 f'  Layer {x}: {[(p, self.support_last_access.get(p)) for p in self.layers[x]]} '
                    #                 f'addr_last_evict={self.support_last_evict.get(address)}'
                    #             )
                        
                    #     prefix_idx = j
                    #     for x in range(j + 1, self.k + 1):
                    #         if self.support_last_access.get(cache_pages[x - 1]) < self.support_last_evict.get(address):
                    #             prefix_idx = x
                    #             if prefix_idx > j:
                    #                 print(f'OnlineMinRandom warning 3: cache_pages{prefix_idx} > j{j}, since page {cache_pages[x - 1]} last_access {self.support_last_access.get(cache_pages[x - 1])} < addr last_evict {self.support_last_evict.get(address)}')
                    #         else:
                    #             break
                    #     prefix = cache_pages[:j]
                    #     victim = min(prefix, key=lambda p: self.priority[p])

                victim_idx = self.cache.index(victim)
                self.support_last_evict[victim] = ts
                self.cache[victim_idx] = address
                self.pcs[victim_idx] = pc

        # Update layers after cache update (as in the paper).
        self._update_layers_after_request(address, layer_i, forgiveness)

        # Record last access for the accessed page (kept only if in support).
        self.support_last_access[address] = ts

        # Support can be larger than the cache (up to 3k), so support\cache is allowed.
        # But cache must never contain pages outside the support.
        # cache_set = {p for p in self.cache if p is not None}
        # extras = [p for p in cache_set if p not in self.support]
        # if extras:
        #     raise RuntimeError(
        #         'OnlineMin invariant violated: cache contains non-support pages. '
        #         f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
        #         f'extras={extras} '
        #         f'cache={list(self.cache)} support_size={len(self.support)}'
        #     )

        self.timestamp += 1
        return hit


class PredictiveOnlineMinAlgorithm(OnlineMinAlgorithm):
    """PredictiveOnlineMin: OnlineMin with predictor-driven eviction on L0 misses.

    Behavior:
    - Same layer/support maintenance as `OnlineMinAlgorithm`.
    - On a cache miss for a page in L0 (i.e. not currently in support) AND the
      cache is full, use the provided `predictor` + `evictor` to choose which
      cache slot to evict.
    - Otherwise, fall back to OnlineMin's eviction rule.

    This class uses the same predictor interface as `PredictAlgorithm` so it can
    be constructed via `PredictAlgorithmFactory.generate_predictive_algorithm`.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Warm-up phase: identical to OnlineMin except that we also update the
        # predictor score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # OnlineMin proper, with predictor-driven eviction on L0 misses.
        # Cache must always be a subset of support (OnlineMin invariant). If this
        # is violated (e.g. due to an unintended interaction with forgiveness),
        # fail fast with a clear error instead of crashing with KeyError later.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveOnlineMin invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                # L0 miss and cache full: consult predictor for victim choice.
                # IMPORTANT: do NOT override OnlineMin's eviction rule during the
                # forgiveness case (support at cap). OnlineMin relies on a
                # specific eviction+layer-update interaction; picking an
                # arbitrary victim can leave cache pages outside the support.
                if layer_i == 0 and (not forgiveness) and self.preds is not None:
                #if layer_i == 0 and self.preds is not None:
                    victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                else:
                    cache_pages = [p for p in self.cache if p is not None]
                    if eviction_layer == 0:
                        victim = min(cache_pages, key=lambda p: self.priority[p])
                    else:
                        cache_pages.sort(key=lambda p: self.layer_of[p])
                        j = None
                        for jj in range(eviction_layer, self.k + 1):
                            if self.layer_of[cache_pages[jj - 1]] == jj:
                                j = jj
                                break
                        if j is None:
                            victim = min(cache_pages, key=lambda p: self.priority[p])
                        else:
                            victim = min(cache_pages[:j], key=lambda p: self.priority[p])
                    victim_idx = self.cache.index(victim)

                target_index = victim_idx
                self.cache[target_index] = address
                self.pcs[target_index] = pc

        # Update layers after cache update (as in OnlineMin).
        self._update_layers_after_request(address, layer_i, forgiveness)

        # Invariant: cache must be subset of support.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveOnlineMin invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        # Update predictor score for the accessed/inserted slot.
        if target_index is None:
            # Should never happen, but keep safe.
            target_index = 0
        self.after_pred(pc, address, target_index)

        return hit


class PredictiveRPBOnlineMinAlgorithm(OnlineMinAlgorithm):
    """RPB-OnlineMin: OnlineMin with conditional predictor eviction.

    Similar to `PredictiveOnlineMinAlgorithm`, but with RPB gating and
    bookkeeping:
    - On a true L0 miss (layer_i==0 and not forgiveness) with cache full, evict
      using the predictor within OnlineMin's current eviction candidate set.
      Let the evicted page be x. Record:
        Y(x) = #{p in candidates : last_access_time[p] < now}
        T(x) = now
    - On forgiveness (support at cap), always use OnlineMin's eviction rule.
    - On other misses that require eviction (non-L0 and not forgiveness), for
      the requested page x': if x' was previously evicted via predictor (i.e.
      we have recorded Y(x'), T(x')) then compute
        Z = #{p in current candidates : last_access_time[p] < T(x')}
      If Z < Y(x')/2, use predictor eviction within candidates; otherwise fall
      back to OnlineMin eviction.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        # Last access time for pages that have appeared in cache.
        self.last_access_time: Dict[object, int] = {}

        # RPB bookkeeping for pages evicted via predictor-driven eviction.
        self._rpb_y: Dict[object, int] = {}
        self._rpb_t: Dict[object, int] = {}

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _onlinemin_eviction_candidate_pages(self, eviction_layer: int) -> List[object]:
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return []
        if eviction_layer == 0:
            return cache_pages

        cache_pages.sort(key=lambda p: self.layer_of[p])
        j = None
        for jj in range(eviction_layer, self.k + 1):
            if self.layer_of[cache_pages[jj - 1]] == jj:
                j = jj
                break
        if j is None:
            # Should not happen; treat as full set.
            return cache_pages
        return cache_pages[:j]

    def _onlinemin_eviction_candidate_indices(self, eviction_layer: int) -> List[int]:
        return [self.cache.index(p) for p in self._onlinemin_eviction_candidate_pages(eviction_layer)]

    def _onlinemin_victim_idx(self, eviction_layer: int) -> int:
        candidate_pages = self._onlinemin_eviction_candidate_pages(eviction_layer)
        if not candidate_pages:
            return 0
        victim = min(candidate_pages, key=lambda p: self.priority[p])
        return self.cache.index(victim)

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        ts = self.timestamp

        # Warm-up phase: identical to PredictiveOnlineMin, plus last-access tracking.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.last_access_time[address] = ts
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.last_access_time[address] = ts
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Invariant: cache must always be a subset of support.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveRPBOnlineMin invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                # Never override OnlineMin during forgiveness.
                if forgiveness:
                    victim_idx = self._onlinemin_victim_idx(eviction_layer)
                    target_index = victim_idx
                    self.cache[target_index] = address
                    self.pcs[target_index] = pc
                else:
                    use_predictor = False
                    if self.preds is not None:
                        if layer_i == 0:
                            # True L0 miss: always predictor-evict.
                            use_predictor = True
                        else:
                            # Non-L0 miss: gate on recorded Y/T for this page.
                            if address in self._rpb_y and address in self._rpb_t:
                                y_prev = self._rpb_y[address]
                                t_prev = self._rpb_t[address]
                                candidate_pages = self._onlinemin_eviction_candidate_pages(eviction_layer)
                                z = sum(1 for p in candidate_pages if self.last_access_time.get(p, -1) < t_prev)
                                use_predictor = (2 * z < y_prev)

                    if use_predictor:
                        candidate_pages = self._onlinemin_eviction_candidate_pages(eviction_layer)
                        candidate_indices = [self.cache.index(p) for p in candidate_pages]
                        if not candidate_indices:
                            candidate_indices = list(range(self.k))

                        y_now = sum(1 for p in candidate_pages if self.last_access_time.get(p, -1) < ts)
                        scored_candidates = [(i, self.preds[i]) for i in candidate_indices]
                        victim_idx = self.evictor.evict(scored_candidates)
                        victim_page = self.cache[victim_idx]
                        if victim_page is not None:
                            self._rpb_y[victim_page] = int(y_now)
                            self._rpb_t[victim_page] = int(ts)
                        target_index = victim_idx
                        self.cache[target_index] = address
                        self.pcs[target_index] = pc
                    else:
                        victim_idx = self._onlinemin_victim_idx(eviction_layer)
                        target_index = victim_idx
                        self.cache[target_index] = address
                        self.pcs[target_index] = pc

        # Update layers after cache update (as in OnlineMin).
        self._update_layers_after_request(address, layer_i, forgiveness)

        # Invariant: cache must be subset of support.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveRPBOnlineMin invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        # Record last access time for the accessed/inserted page.
        self.last_access_time[address] = ts

        # Update predictor score for the accessed/inserted slot.
        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)

        return hit


class PredictiveRPBNewOnlineMinAlgorithm(OnlineMinAlgorithm):
    """RPB-OnlineMin: OnlineMin with conditional predictor eviction.

    Similar to `PredictiveOnlineMinAlgorithm`, but with RPB gating and
    bookkeeping:
    - On a true L0 miss (layer_i==0 and not forgiveness) with cache full, evict
      using the predictor within OnlineMin's current eviction candidate set.
      Let the evicted page be x. Record:
        Y(x) = #{p in candidates : last_access_time[p] < now}
        T(x) = now
    - On forgiveness (support at cap), always use OnlineMin's eviction rule.
    - On other misses that require eviction (non-L0 and not forgiveness), for
      the requested page x': if x' was previously evicted via predictor (i.e.
      we have recorded Y(x'), T(x')) then compute
        Z = #{p in current candidates : last_access_time[p] < T(x')}
      If Z < Y(x')/2, use predictor eviction within candidates; otherwise fall
      back to OnlineMin eviction.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        pred_budget: int = 0,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        # Last access time for pages that have appeared in cache.
        self.last_access_time: Dict[object, int] = {}

        # RPB bookkeeping for pages evicted via predictor-driven eviction.
        self._rpb_y: Dict[object, int] = {}
        self._rpb_t: Dict[object, int] = {}

        # Prediction budget for non-L0 predictor use.
        self.pred_budget_init = int(pred_budget)
        self.pred_budget = self.pred_budget_init

        # Previous candidate-set size used by the budget update rule.
        self.prev_can_size = self.k

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _onlinemin_eviction_candidate_pages(self, eviction_layer: int) -> List[object]:
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return []
        if eviction_layer == 0:
            return cache_pages

        cache_pages.sort(key=lambda p: self.layer_of[p])
        j = None
        for jj in range(eviction_layer, self.k + 1):
            if self.layer_of[cache_pages[jj - 1]] == jj:
                j = jj
                break
        if j is None:
            # Should not happen; treat as full set.
            return cache_pages
        return cache_pages[:j]

    def _num_of_revealed_layrs(self) -> int:
        n = 0
        for jj in range(self.k, 0, -1):
            if len(self.layers[jj]) == 1:
                n += 1
            else:
                break
        return n

    def _onlinemin_eviction_candidate_indices(self, eviction_layer: int) -> List[int]:
        return [self.cache.index(p) for p in self._onlinemin_eviction_candidate_pages(eviction_layer)]

    def _onlinemin_victim_idx(self, eviction_layer: int) -> int:
        candidate_pages = self._onlinemin_eviction_candidate_pages(eviction_layer)
        if not candidate_pages:
            return 0
        victim = min(candidate_pages, key=lambda p: self.priority[p])
        return self.cache.index(victim)

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        ts = self.timestamp

        # Warm-up phase: identical to PredictiveOnlineMin, plus last-access tracking.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.last_access_time[address] = ts
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.last_access_time[address] = ts
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Invariant: cache must always be a subset of support.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveRPBOnlineMin invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                # Never override OnlineMin during forgiveness.
                if forgiveness:
                    victim_idx = self._onlinemin_victim_idx(eviction_layer)
                    target_index = victim_idx
                    self.cache[target_index] = address
                    self.pcs[target_index] = pc
                else:
                    use_predictor = False
                    if self.preds is not None:
                        if layer_i == 0:
                            # True L0 miss: always predictor-evict.
                            use_predictor = True
                            self.pred_budget = self.pred_budget_init
                            self._rpb_y.clear()
                            self._rpb_t.clear()
                        else:
                            # Non-L0 miss: gate on recorded Y/T for this page.
                            # if address in self._rpb_y and address in self._rpb_t:
                            #     y_prev = self._rpb_y[address]
                            #     t_prev = self._rpb_t[address]
                            #     candidate_pages = self._onlinemin_eviction_candidate_pages(eviction_layer)
                            #     z = sum(1 for p in candidate_pages if self.last_access_time.get(p, -1) < t_prev)
                            #     use_predictor = (2 * z < y_prev)
                            
                            candidate_pages = self._onlinemin_eviction_candidate_pages(eviction_layer)
                            if len(candidate_pages) <= self.prev_can_size / 2.718 - 1:
                                self.pred_budget += 1
                            
                            if self.pred_budget >= 1:
                                use_predictor = True
                                self.pred_budget -= 1

                    if use_predictor:
                        candidate_pages = self._onlinemin_eviction_candidate_pages(eviction_layer)
                        candidate_indices = [self.cache.index(p) for p in candidate_pages]
                        if not candidate_indices:
                            candidate_indices = list(range(self.k))

                        #y_now = sum(1 for p in candidate_pages if self.last_access_time.get(p, -1) < ts)
                        scored_candidates = [(i, self.preds[i]) for i in candidate_indices]
                        victim_idx = self.evictor.evict(scored_candidates)
                        #victim_page = self.cache[victim_idx]
                        #if victim_page is not None:
                        #    self._rpb_y[victim_page] = int(y_now)
                        #    self._rpb_t[victim_page] = int(ts)
                        target_index = victim_idx
                        self.cache[target_index] = address
                        self.pcs[target_index] = pc
                    else:
                        victim_idx = self._onlinemin_victim_idx(eviction_layer)
                        target_index = victim_idx
                        self.cache[target_index] = address
                        self.pcs[target_index] = pc

        # Update layers after cache update (as in OnlineMin).
        self._update_layers_after_request(address, layer_i, forgiveness)
        self.prev_can_size = self.k - self._num_of_revealed_layrs()

        # Invariant: cache must be subset of support.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveRPBOnlineMin invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        # Record last access time for the accessed/inserted page.
        self.last_access_time[address] = ts

        # Update predictor score for the accessed/inserted slot.
        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)

        return hit

class PredictiveMirrorOnlineMinAlgorithm(EvictAlgorithm):
    """PredictiveMirrorOnlineMin (PMOM).

    - On requests where the page is in L0 (w.r.t. a mirrored OnlineMin) and the
      request misses in this cache (and the cache is full), evict using the
      predictor (same as PredictiveOnlineMin).
    - On requests where the page is not in L0 and the request misses in this
      cache (and the cache is full), first run the mirrored OnlineMin on this
      request, then evict the unique page that is in *this* cache but not in the
      mirrored OnlineMin cache after the request.

    Note: The spec text says to evict a page that is in OnlineMin cache but not
    in this cache; that is not executable (it is not resident here). This
    implementation follows the mirror interpretation above so that the cache can
    resync toward OnlineMin on non-L0 misses.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
    ) -> None:
        super().__init__(associativity)
        self.k = associativity
        self.max_support_factor = int(max_support_factor)

        self._phase_non_l0_pred_used = 0

        self.timestamp = 0

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        # The mirrored OnlineMin instance whose cache content we observe.
        self.mirror_onlinemin = OnlineMinAlgorithm(
            associativity=associativity,
            max_support_factor=self.max_support_factor,
        )

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Determine whether this request is L0 w.r.t the mirrored OnlineMin.
        # During OnlineMin warm-up, treat everything as L0.
        if getattr(self.mirror_onlinemin, '_inited', False):
            layer_i = self.mirror_onlinemin.layer_of.get(address, 0)
        else:
            layer_i = 0

        # New phase starts on every L0 request.
        if layer_i == 0:
            self._phase_non_l0_pred_used = 0

        # Run the mirrored OnlineMin first so we can observe its post-request cache.
        self.mirror_onlinemin.access(pc, address)
        mirror_cache_after = {p for p in self.mirror_onlinemin.cache if p is not None}

        # Normal cache access.
        if address in self.cache:
            idx = self.cache.index(address)
            self.pcs[idx] = pc
            self.after_pred(pc, address, idx)
            return True

        if None in self.cache:
            idx = self.cache.index(None)
            self.cache[idx] = address
            self.pcs[idx] = pc
            self.after_pred(pc, address, idx)
            return False

        # Cache full miss.
        if layer_i == 0:
            # L0 miss: predictor eviction.
            if self.preds is None:
                print('Warning: PredictiveMirrorOnlineMinAlgorithm with no predictor on L0 miss; using random eviction.')
            else:
                victim_idx = self.evictor.evict(list(enumerate(self.preds)))
        else:
            # Mirror OnlineMin by evicting the page that OnlineMin
            # (post-request) does NOT keep.
            candidates = [p for p in self.cache if p is not None and p not in mirror_cache_after]
            if not candidates:
                victim_idx = random.randrange(self.associativity)
            else:
                victim_idx = self.cache.index(candidates[0])

        self.cache[victim_idx] = address
        self.pcs[victim_idx] = pc
        self.after_pred(pc, address, victim_idx)
        return False


class PredictiveRelaxedMirrorOnlineMinAlgorithm(EvictAlgorithm):
    """PredictiveRelaxedMirrorOnlineMin.

        Similar to `PredictiveMirrorOnlineMinAlgorithm` but maintains a per-phase
        `budget` (initialized to 1) where a phase starts on every L0 request
        (w.r.t. the mirrored OnlineMin).

        Rules:
        - If request is L0: on cache-full miss, evict using predictions.
        - If request is non-L0:
            (1) this-miss & OnlineMin-hit  => budget -= 1 and sync toward OnlineMin
            (2) this-hit  & OnlineMin-miss => budget += 1
            (3) this-miss & OnlineMin-miss => if budget > 0, predictor-evict within
                    the OnlineMin-based range; else sync toward OnlineMin

    Sync means: evict any resident page that OnlineMin (post-request) does NOT
    keep, then insert the requested page.

    Predictor-evict with OnlineMin candidates means: compute the same eviction
    candidate prefix as OnlineMin (sort by OnlineMin layer index, choose the
    smallest prefix length j >= eviction_layer satisfying OnlineMin's condition),
    then evict within that prefix using predictor scores.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        budget: int = 1,
    ) -> None:
        super().__init__(associativity)
        self.k = associativity
        self.max_support_factor = int(max_support_factor)

        self.timestamp = 0

        # Phase/budget state.
        self.budget_init = int(budget)
        self.budget = self.budget_init

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        self.mirror_onlinemin = OnlineMinAlgorithm(
            associativity=associativity,
            max_support_factor=self.max_support_factor,
        )

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _sync_victim_idx(self, mirror_cache_after: set) -> int:
        candidates = [p for p in self.cache if p is not None and p not in mirror_cache_after]
        if not candidates:
            return random.randrange(self.associativity)
        return self.cache.index(candidates[0])

    def _onlinemin_candidate_indices(self, eviction_layer: int, mirror_layer_of: Dict[object, int]) -> List[int]:
        """Compute OnlineMin-style eviction candidate indices for *this* cache.

        Mirrors OnlineMin's candidate prefix computation:
        - Sort cache pages by their OnlineMin layer index.
        - Find the smallest j >= eviction_layer such that layer(sorted[j-1]) == j.
        - Candidates are the prefix of length j (or all pages if j doesn't exist).

        Pages not in OnlineMin's support are treated as being in layer k+1.
        """
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return []
        if eviction_layer == 0:
            return [self.cache.index(p) for p in cache_pages]

        def layer_idx(p):
            return mirror_layer_of.get(p, self.k + 1)

        cache_pages.sort(key=layer_idx)

        j = None
        # OnlineMin searches jj in [eviction_layer..k]. Clamp to valid range.
        start = max(1, int(eviction_layer))
        for jj in range(start, self.k + 1):
            if jj - 1 < len(cache_pages) and layer_idx(cache_pages[jj - 1]) == jj:
                j = jj
                break

        prefix_pages = cache_pages if j is None else cache_pages[:j]
        return [self.cache.index(p) for p in prefix_pages]

    def _pred_victim_idx_within_onlinemin_candidates(self, eviction_candidates: List[int]) -> int:
        """Pick victim using predictions, restricted to given candidate indices."""
        if not eviction_candidates:
            return random.randrange(self.associativity)
        if self.preds is None:
            return random.choice(eviction_candidates)
        return self.evictor.evict([(i, self.preds[i]) for i in eviction_candidates])

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Determine whether this request is L0 w.r.t the mirrored OnlineMin.
        # During OnlineMin warm-up, treat everything as L0.
        mirror_inited = getattr(self.mirror_onlinemin, '_inited', False)
        if mirror_inited:
            layer_i = self.mirror_onlinemin.layer_of.get(address, 0)
            support_size = len(self.mirror_onlinemin.support)
            mirror_layer_of_pre = dict(self.mirror_onlinemin.layer_of)
        else:
            layer_i = 0
            support_size = 0
            mirror_layer_of_pre = {}

        forgiveness = (layer_i == 0 and support_size == getattr(self.mirror_onlinemin, 'max_support', float('inf')))
        eviction_layer = 1 if forgiveness else layer_i

        # Phase boundary: every L0 request starts a new phase.
        if layer_i == 0:
            self.budget = self.budget_init

        # Observe OnlineMin behavior by running it first.
        mirror_hit = self.mirror_onlinemin.access(pc, address)
        mirror_cache_after = {p for p in self.mirror_onlinemin.cache if p is not None}

        # Normal cache access.
        if address in self.cache:
            idx = self.cache.index(address)
            self.pcs[idx] = pc
            if layer_i > 0 and (not mirror_hit):
                self.budget += 1
            self.after_pred(pc, address, idx)
            return True

        if None in self.cache:
            idx = self.cache.index(None)
            self.cache[idx] = address
            self.pcs[idx] = pc
            self.after_pred(pc, address, idx)
            return False

        # Cache full miss.
        if layer_i == 0:
            # L0 miss: predictor eviction.
            if self.preds is None:
                print('Warning: PredictiveRelaxedMirrorOnlineMinAlgorithm with no predictor on L0 miss; using random eviction.')
                victim_idx = random.randrange(self.associativity)
            else:
                victim_idx = self.evictor.evict(list(enumerate(self.preds)))
        else:
            # non-L0
            if mirror_hit:
                # (1) this-miss & OnlineMin-hit => budget--, then sync.
                self.budget -= 1
                victim_idx = self._sync_victim_idx(mirror_cache_after)
            else:
                # OnlineMin miss too.
                if self.budget > 0:
                    # (2) budget>0 => predictor-evict with OnlineMin candidate rule.
                    eviction_candidates = self._onlinemin_candidate_indices(eviction_layer, mirror_layer_of_pre)
                    victim_idx = self._pred_victim_idx_within_onlinemin_candidates(eviction_candidates)
                else:
                    # (2) budget<=0 => sync.
                    victim_idx = self._sync_victim_idx(mirror_cache_after)

        self.cache[victim_idx] = address
        self.pcs[victim_idx] = pc
        self.after_pred(pc, address, victim_idx)
        return False


class PredictiveExtendedRelaxedMirrorOnlineMinAlgorithm(EvictAlgorithm):
    """PredictiveExtendedRelaxedMirrorOnlineMin.

    Identical to `PredictiveRelaxedMirrorOnlineMinAlgorithm` except in case (3)
    (non-L0, both miss, budget>0) where the eviction candidate set is
    chosen using an *extended* rule similar to `PredictiveValidExtendedAlgorithm`
    (for eviction_layer != 0), rather than OnlineMin's "valid prefix" rule.

    Concretely, we:
    - Sort cached pages by the mirrored OnlineMin layer assignment (pre-request).
    - Compute the revealed suffix start index from mirrored OnlineMin layers
      (pre-request), same criterion as in ValidExtended.
    - Prefer candidates in the first k positions that were accessed before the
      last time the requested page was evicted, skipping revealed suffix pages.
    - If none, fall back to the same set without the access-time filter.
    - Finally, pick the victim within candidates using predictor scores.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        budget: int = 1,
    ) -> None:
        super().__init__(associativity)
        self.k = associativity
        self.max_support_factor = int(max_support_factor)

        self.timestamp = 0

        # Phase/budget state.
        self.budget_init = int(budget)
        self.budget = self.budget_init

        # Extended candidate-selection state (as in PredictiveValidExtendedAlgorithm).
        self.last_access_time: Dict[object, int] = {}
        self.last_evict_time: Dict[object, int] = {}

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        self.mirror_onlinemin = OnlineMinAlgorithm(
            associativity=associativity,
            max_support_factor=self.max_support_factor,
        )

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _sync_victim_idx(self, mirror_cache_after: set) -> int:
        candidates = [p for p in self.cache if p is not None and p not in mirror_cache_after]
        if not candidates:
            return random.randrange(self.associativity)
        return self.cache.index(candidates[0])

    def _extended_onlinemin_candidate_indices(
        self,
        eviction_layer: int,
        mirror_layer_of: Dict[object, int],
        mirror_layers: List[set],
        address,
    ) -> List[int]:
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return []
        if eviction_layer == 0:
            return [self.cache.index(p) for p in cache_pages]

        cache_pages.sort(key=lambda p: mirror_layer_of.get(p, self.k + 1))

        revealed_start_idx = self.k + 1
        for x in range(self.k, 0, -1):
            if x < len(mirror_layers) and len(mirror_layers[x]) == 1:
                revealed_start_idx = x
            else:
                break

        eviction_candidates = []
        last_evict_of_addr = self.last_evict_time.get(address, -1)

        for jj in range(1, self.k + 1):
            if jj - 1 >= len(cache_pages):
                break
            p = cache_pages[jj - 1]
            if self.last_access_time.get(p, -1) < last_evict_of_addr:
                if p in mirror_layer_of and mirror_layer_of.get(p) >= revealed_start_idx:
                    continue
                eviction_candidates.append(p)

        if len(eviction_candidates) == 0:
            for jj in range(1, self.k + 1):
                if jj - 1 >= len(cache_pages):
                    break
                p = cache_pages[jj - 1]
                if p in mirror_layer_of and mirror_layer_of.get(p) >= revealed_start_idx:
                    continue
                eviction_candidates.append(p)

        return [self.cache.index(p) for p in eviction_candidates]

    def _pred_victim_idx_within_onlinemin_candidates(self, eviction_candidates: List[int]) -> int:
        if not eviction_candidates:
            return random.randrange(self.associativity)
        if self.preds is None:
            return random.choice(eviction_candidates)
        return self.evictor.evict([(i, self.preds[i]) for i in eviction_candidates])

    def access(self, pc, address) -> bool:
        ts = self.timestamp

        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Determine whether this request is L0 w.r.t the mirrored OnlineMin.
        # During OnlineMin warm-up, treat everything as L0.
        mirror_inited = getattr(self.mirror_onlinemin, '_inited', False)
        if mirror_inited:
            layer_i = self.mirror_onlinemin.layer_of.get(address, 0)
            support_size = len(self.mirror_onlinemin.support)
            mirror_layer_of_pre = dict(self.mirror_onlinemin.layer_of)
            mirror_layers_pre = [set() for _ in range(self.k + 1)]
            for i in range(1, self.k + 1):
                mirror_layers_pre[i] = set(self.mirror_onlinemin.layers[i])
        else:
            layer_i = 0
            support_size = 0
            mirror_layer_of_pre = {}
            mirror_layers_pre = [set() for _ in range(self.k + 1)]

        forgiveness = (layer_i == 0 and support_size == getattr(self.mirror_onlinemin, 'max_support', float('inf')))
        eviction_layer = 1 if forgiveness else layer_i

        # Phase boundary: every L0 request starts a new phase.
        if layer_i == 0:
            self.budget = self.budget_init

        # Observe OnlineMin behavior by running it first.
        mirror_hit = self.mirror_onlinemin.access(pc, address)
        mirror_cache_after = {p for p in self.mirror_onlinemin.cache if p is not None}

        # Normal cache access.
        if address in self.cache:
            idx = self.cache.index(address)
            self.pcs[idx] = pc
            self.last_access_time[address] = ts
            if layer_i > 0 and (not mirror_hit):
                self.budget += 1
            self.after_pred(pc, address, idx)
            return True

        if None in self.cache:
            idx = self.cache.index(None)
            self.cache[idx] = address
            self.pcs[idx] = pc
            self.last_access_time[address] = ts
            self.after_pred(pc, address, idx)
            return False

        # Cache full miss.
        self.last_access_time[address] = ts

        if layer_i == 0:
            # L0 miss: predictor eviction.
            if self.preds is None:
                print('Warning: PredictiveExtendedRelaxedMirrorOnlineMinAlgorithm with no predictor on L0 miss; using random eviction.')
                victim_idx = random.randrange(self.associativity)
            else:
                victim_idx = self.evictor.evict(list(enumerate(self.preds)))
        else:
            # non-L0
            if mirror_hit:
                # (1) this-miss & OnlineMin-hit => budget--, then sync.
                self.budget -= 1
                victim_idx = self._sync_victim_idx(mirror_cache_after)
            else:
                # OnlineMin miss too.
                if self.budget > 0:
                    eviction_candidates = self._extended_onlinemin_candidate_indices(
                        eviction_layer,
                        mirror_layer_of_pre,
                        mirror_layers_pre,
                        address,
                    )
                    victim_idx = self._pred_victim_idx_within_onlinemin_candidates(eviction_candidates)
                else:
                    victim_idx = self._sync_victim_idx(mirror_cache_after)

        victim = self.cache[victim_idx]
        self.last_evict_time[victim] = ts

        self.cache[victim_idx] = address
        self.pcs[victim_idx] = pc
        self.after_pred(pc, address, victim_idx)
        return False


class RDMAlgorithm(EvictAlgorithm):
    """RDM (Recency Duration Mix) plugged into the OnOPT priority framework.

    From Moruz & Negoescu:
    - Maintain the offset-function layer partition L0..Lk and apply the
      standard update rule after each request.
    - Eviction policy (OnOPT):
        * If requested page p ∈ L0 on a miss: evict the cached page with the
          smallest priority.
        * If p ∈ Li (i>0) on a miss: find the smallest j ≥ i such that the cache
          contains exactly j pages from L1 ∪ ... ∪ Lj, then evict (among those j
          cached pages) the one with smallest priority.
    - Priority assignment uses a global counter t which increments only on
      requests to non-revealed pages. For each page p in the support we store
      t0(p) (time when p entered support, i.e. when requested from L0).
      Upon each request to p, set:
        priority(p) = 0.8*t + 0.1*(t - t0(p))

    Practical note: the exact OnOPT framework can have support size growing with
    the number of distinct pages. For this simulator we provide an optional
    forgiveness-style cap similar to OnlineMin via max_support_factor.
    """

    def __init__(self, associativity: int, max_support_factor: int = 3):
        super().__init__(associativity)
        self.k = associativity
        if max_support_factor < 1:
            raise ValueError('RDM: max_support_factor must be >= 1')
        self.max_support_factor = int(max_support_factor)
        self.max_support = self.max_support_factor * self.k

        # Layers for the offset-function representation: index 1..k used;
        # index 0 unused (L0 tracked implicitly as "not in support").
        self.layers = [set() for _ in range(self.k + 1)]
        self.support = set()
        self.layer_of = {}

        # RDM priority state.
        self.t = 0
        self.t0 = {}        # page -> entry time into support
        self.priority = {}  # page -> float

        # Warm-up: maintain recency order until cache becomes full.
        self._inited = False
        self._warmup_lru = []  # oldest -> newest

    def _priority_key(self, page):
        # Stable tie-break to keep behavior deterministic.
        return (self.priority.get(page, float('-inf')), page)

    def _rebuild_layer_of(self):
        new_layer_of = {}
        new_support = set()
        for i in range(1, self.k + 1):
            for p in self.layers[i]:
                new_layer_of[p] = i
                new_support.add(p)

        # Drop state for pages that leave the support (only possible with
        # forgiveness/capping).
        for p in list(self.priority.keys()):
            if p not in new_support:
                self.priority.pop(p, None)
                self.t0.pop(p, None)

        self.layer_of = new_layer_of
        self.support = new_support

    def _init_layers_from_warmup(self):
        pages = [p for p in self._warmup_lru if p is not None]
        if len(pages) < self.k:
            return
        pages = pages[-self.k:]
        self.layers = [set() for _ in range(self.k + 1)]
        for i, p in enumerate(pages, start=1):
            self.layers[i] = {p}
            self.t0[p] = 0
            self.priority[p] = 0.0
        self._rebuild_layer_of()
        self._inited = True

    def _revealed_pages(self):
        # r = smallest index such that L_r..L_k are all singletons.
        r = self.k + 1
        suffix_singletons = True
        for i in range(self.k, 0, -1):
            if suffix_singletons and len(self.layers[i]) == 1:
                r = i
            else:
                suffix_singletons = False
        if r > self.k:
            return set()
        revealed = set()
        for i in range(r, self.k + 1):
            revealed |= self.layers[i]
        return revealed

    def _assign_priority_on_request(self, page, layer_i: int, revealed: bool):
        # Increment t only on requests to non-revealed pages.
        if not revealed:
            self.t += 1

        # On requests from L0, define entry time into support.
        if layer_i == 0:
            self.t0[page] = self.t

        if page not in self.t0:
            # Shouldn't happen unless warm-up/forgiveness removed state.
            self.t0[page] = self.t

        # priority(p) = 0.8*t + 0.1*(t - t0(p))
        self.priority[page] = 0.8 * self.t + 0.1 * (self.t - self.t0[page])

    def _update_layers_after_request(self, page, layer_i: int, forgiveness: bool):
        # Offset-function layer update (Section 2.1), plus an optional
        # forgiveness-style cap to keep support bounded.
        if layer_i == 0:
            if not forgiveness:
                if self.k >= 2:
                    self.layers[self.k - 1] |= self.layers[self.k]
                self.layers[self.k] = {page}
            else:
                # Bounded-support variant: drop the oldest unrevealed layer.
                dropped = self.layers[1]
                for q in dropped:
                    self.priority.pop(q, None)
                    self.t0.pop(q, None)
                for j in range(1, self.k):
                    self.layers[j] = self.layers[j + 1]
                self.layers[self.k] = {page}
        else:
            i = layer_i
            if i < self.k:
                self.layers[i - 1] |= (self.layers[i] - {page})
                for j in range(i, self.k):
                    self.layers[j] = self.layers[j + 1]
                self.layers[self.k] = {page}
            else:
                self.layers[self.k] = {page}

        self._rebuild_layer_of()

        if len(self.support) > self.max_support:
            raise RuntimeError(
                'RDM invariant violated: support size exceeded cap. '
                f'support_size={len(self.support)} max_support={self.max_support}'
            )

    def access(self, pc, address) -> bool:
        # Warm-up phase: behave like a simple LRU fill until full, then init.
        if not self._inited:
            if address in self.cache:
                hit = True
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                return hit

            hit = False
            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                return hit

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # RDM proper.
        revealed = self._revealed_pages()
        is_revealed = address in revealed

        hit = address in self.cache
        if hit:
            idx = self.cache.index(address)
            self.pcs[idx] = pc

        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and len(self.support) == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
            else:
                cache_pages = [p for p in self.cache if p is not None]
                if eviction_layer == 0:
                    victim = min(cache_pages, key=self._priority_key)
                else:
                    cache_pages.sort(key=lambda p: self.layer_of[p])
                    j = None
                    for jj in range(eviction_layer, self.k + 1):
                        if self.layer_of[cache_pages[jj - 1]] == jj:
                            j = jj
                            break
                    if j is None:
                        victim = min(cache_pages, key=self._priority_key)
                    else:
                        victim = min(cache_pages[:j], key=self._priority_key)

                victim_idx = self.cache.index(victim)
                self.cache[victim_idx] = address
                self.pcs[victim_idx] = pc

        # Priority assignment happens on each request.
        self._assign_priority_on_request(address, layer_i, is_revealed)

        # Update layers after cache update.
        self._update_layers_after_request(address, layer_i, forgiveness)

        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'RDM invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} revealed={is_revealed} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        return hit

####################################################################

class PredictAlgorithmFactory:
    predictor_evict_dict = {
        "PLECO": (MaxEvictor, PLECOPredictor),
        "PLECO-State": (DummyEvictor, PLECOStatePredictor),
        "PLECO-Bin": (BinaryEvictor, PLECOBinPredictor),
        "GBM": (BinaryEvictor, GBMBinPredictor),
        "POPU": (MaxEvictor, POPUPredictor),
        "POPU-State": (DummyEvictor, POPUStatePredictor),
        "Parrot": (MaxEvictor, ParrotPredictor),
        "Parrot-State": (DummyEvictor, ParrotStatePredictor),
        "OracleDis": (ReuseDistanceEvictor, OracleReuseDistancePredictor),
        "OracleBin": (BinaryEvictor, OracleBinaryPredictor),
        "OraclePhase": (BinaryEvictor, OraclePhasePredictor),
        "OracleState": (DummyEvictor, OracleStatePredictor),
    }

    @staticmethod
    def generate_predictive_algorithm(alg_type: Union[Type[PredictAlgorithm], partial], pred_type_str: str, **kwargs) -> partial:
        evictor_type, predictor_type = PredictAlgorithmFactory.predictor_evict_dict[pred_type_str]
        
        evictor_partial = evictor_type
        predictor_partial = predictor_type
        if pred_type_str == 'Parrot' or pred_type_str == 'Parrot-State' or pred_type_str == 'GBM':
            # shared_model
            if 'shared_model' not in kwargs:
                raise ValueError('PredictAlgorithmFactory: Parrot need [shared_model]')
            
            if pred_type_str == 'Parrot-State':
                if 'associativity' not in kwargs:
                    raise ValueError(f'PredictAlgorithmFactory: {pred_type_str} need [associativity]')
                associativity = kwargs['associativity']
                predictor_partial = partial(predictor_type, shared_model=kwargs['shared_model'], associativity=associativity)
            else:
                predictor_partial = partial(predictor_type, shared_model=kwargs['shared_model']) 
        elif pred_type_str.startswith('Oracle'):
            reuse_dis_noise_sigma = 0
            lognormal = True
            if 'reuse_dis_noise_sigma' in kwargs:
                reuse_dis_noise_sigma = kwargs['reuse_dis_noise_sigma']
            if 'lognormal' in kwargs:
                lognormal = kwargs['lognormal']

            if pred_type_str == 'OracleDis':
                predictor_partial = partial(predictor_type, reuse_dis_noise_sigma=reuse_dis_noise_sigma, lognormal=lognormal)
            else:
                if 'associativity' not in kwargs:
                    raise ValueError(f'PredictAlgorithmFactory: {pred_type_str} need [associativity]')
                associativity = kwargs['associativity']
                if pred_type_str == 'OracleState':
                    predictor_partial = partial(predictor_type, associativity=associativity, reuse_dis_noise_sigma=reuse_dis_noise_sigma, lognormal=lognormal)
                else:
                    bin_noise_prob = 0
                    if 'bin_noise_prob' in kwargs:
                        bin_noise_prob = kwargs['bin_noise_prob']
                    predictor_partial = partial(predictor_type, associativity=associativity, bin_noise_prob=bin_noise_prob, reuse_dis_noise_sigma=reuse_dis_noise_sigma, lognormal=lognormal)
        elif pred_type_str.endswith('State'):
            if 'associativity' not in kwargs:
                raise ValueError(f'PredictAlgorithmFactory: {pred_type_str} need [associativity]')
            associativity = kwargs['associativity']
            predictor_partial = partial(predictor_type, associativity=associativity)
        elif pred_type_str == 'PLECO-Bin':
            if 'threshold' not in kwargs:
                raise ValueError(f'PredictAlgorithmFactory: {pred_type_str} need [threshold]')
            threshold = kwargs['threshold']
            predictor_partial = partial(predictor_type, threshold=threshold)

        if isinstance(alg_type, partial):
            this_partial = copy.deepcopy(alg_type)
            this_partial.keywords['evictor_type'] = evictor_partial
            this_partial.keywords['predictor_type'] = predictor_partial
            return this_partial
        else:
            return partial(alg_type, evictor_type=evictor_partial, predictor_type=predictor_partial)

def format_guard(relax_times, relax_prob):
    if relax_times == 0 and relax_prob == 0:
        return "-no-relax"
    elif relax_times == 0 and relax_prob != 0:
        return f"-relax-prob-{relax_prob}"
    elif relax_times != 0 and relax_prob == 0:
        return f"-relax-times-{relax_times}"
    else:
        raise ValueError('relax_times and relax_prob invaild')

def format_oracle(reuse_dis_noise_sigma, bin_noise_prob):
    if reuse_dis_noise_sigma == 0 and bin_noise_prob == 0:
        return "-oracle"
    elif reuse_dis_noise_sigma == 0 and bin_noise_prob != 0:
        return f"-bin-{bin_noise_prob}"
    elif reuse_dis_noise_sigma != 0 and bin_noise_prob == 0:
        return f"-dis-{reuse_dis_noise_sigma}"
    else:
        return f"-dis-{reuse_dis_noise_sigma}-bin-{bin_noise_prob}"

def pretty_print(callable: Union[EvictAlgorithm, partial], verbose=False) -> str:
    this_cls = callable
    if hasattr(callable, 'func'):
        this_cls = callable.func
    this_cls_name = this_cls.__name__.replace("Algorithm", '').replace("CombineDeterministic", 'CombDet').replace('CombineRandomAlgorithm', 'CombRand').replace("MarkAndPredict", "Mark&Predict").replace('PredictiveMarker', 'PredMark').replace('PredictiveRPBOnlineMin', 'RPB-OnlineMin').replace('PredictiveRPBNewOnline', 'RPB-new-Online').replace('PredictiveRPBNewRDM', 'RPB-new-RDM')
    metadata = this_cls_name
    if hasattr(callable, 'keywords'):
        kw = callable.keywords
        if issubclass(this_cls, CombineAlgorithm):
            algs = kw['candidate_algorithms']
            alg_names = []
            for alg in algs:
                alg_names.append(pretty_print(alg, verbose))
            metadata += ("[" + (", ".join(alg_names)) + "]")
        
        if 'predictor_type' in kw:
            predictor_type = kw['predictor_type']
            pred_kw = {}
            if hasattr(predictor_type, 'func'):
                pred_kw = predictor_type.keywords
                predictor_type = predictor_type.func
            predictor = predictor_type.__name__.replace("Predictor", '').replace('OracleReuseDistance', 'Belady').replace('OracleBinary', 'FBP')
            metadata += f'[{predictor}]'

            if issubclass(predictor_type, OraclePredictor) and verbose:
                reuse_dis_noise_sigma = bin_noise_prob = 0
                if 'reuse_dis_noise_sigma' in pred_kw:
                    reuse_dis_noise_sigma = pred_kw['reuse_dis_noise_sigma']
                if 'bin_noise_prob' in pred_kw:
                    bin_noise_prob = pred_kw['bin_noise_prob']
                metadata += format_oracle(reuse_dis_noise_sigma, bin_noise_prob) 

        if issubclass(this_cls, Guard):
            follow_if_guarded = False
            relax_times = relax_prob = 0
            if 'follow_if_guarded' in kw:
                follow_if_guarded = kw['follow_if_guarded']
            if follow_if_guarded:
                metadata += '-unv'
            else:
                metadata += '-f-pred'
            if 'relax_times' in kw:
                relax_times = kw['relax_times']
            if 'relax_prob' in kw:
                relax_prob = kw['relax_prob']
            metadata += format_guard(relax_times, relax_prob)

        if issubclass(this_cls, OnlineMinAlgorithm):
            if 'max_support_factor' in kw:
                metadata += f"-msf-{kw['max_support_factor']}"

        if issubclass(this_cls, (PredictiveRPBNewOnlineMinAlgorithm, PredictiveRPBNewRDMAlgorithm, RPBMarkerAlgorithm)):
            metadata += f"-pb-{kw.get('pred_budget', 0)}"

        if issubclass(
            this_cls,
            (
                PredictiveMirrorOnlineMinAlgorithm,
                PredictiveRelaxedMirrorOnlineMinAlgorithm,
                PredictiveExtendedRelaxedMirrorOnlineMinAlgorithm,
            ),
        ):
            if 'max_support_factor' in kw:
                metadata += f"-msf-{kw['max_support_factor']}"

        if issubclass(
            this_cls,
            (
                PredictiveMirrorRDMAlgorithm,
                PredictiveRelaxedMirrorRDMAlgorithm,
                PredictiveExtendedRelaxedMirrorRDMAlgorithm,
            ),
        ):
            if 'max_support_factor' in kw:
                metadata += f"-msf-{kw['max_support_factor']}"

        if issubclass(this_cls, (PredictiveRelaxedMirrorOnlineMinAlgorithm, PredictiveRelaxedMirrorRDMAlgorithm)):
            if 'budget' in kw:
                metadata += f"-bud-{kw['budget']}"
            elif 'credit_threshold' in kw:
                # Backward-compatible naming if a stale partial still uses credit_threshold.
                metadata += f"-bud-{kw['credit_threshold']}"

        if issubclass(
            this_cls,
            (
                PredictiveExtendedRelaxedMirrorOnlineMinAlgorithm,
                PredictiveExtendedRelaxedMirrorRDMAlgorithm,
            ),
        ):
            if 'budget' in kw:
                metadata += f"-bud-{kw['budget']}"
            elif 'credit_threshold' in kw:
                # Backward-compatible naming if a stale partial still uses credit_threshold.
                metadata += f"-bud-{kw['credit_threshold']}"

        if issubclass(this_cls, RDMAlgorithm):
            if 'max_support_factor' in kw:
                metadata += f"-msf-{kw['max_support_factor']}"

        if issubclass(
            this_cls,
            (
                PredictiveBoundingRDMAlgorithm,
                PredictiveBoundingOnlineMinAlgorithm,
                PredictiveBoundingMirrorOnlineMinAlgorithm,
                PredictiveValidExtendedBoundingOnlineMinAlgorithm,
                PredictiveValidExtendedBoundingRDMAlgorithm,
            ),
        ):
            if 'sync_threshold' in kw:
                metadata += f"-st-{kw['sync_threshold']}"
            
    return metadata


class PredictiveRDMAlgorithm(RDMAlgorithm):
    """PredictiveRDM: RDM with predictor-driven eviction on L0 misses.

    Behavior:
    - Same layer/support/priority maintenance as `RDMAlgorithm`.
    - On a cache miss for a page in L0 (i.e. not currently in support) AND the
      cache is full, use the provided `predictor` + `evictor` to choose which
      cache slot to evict.
    - Otherwise, fall back to RDM's eviction rule.

    IMPORTANT: During the bounded-support forgiveness case (support at cap), we
    must *not* override RDM's eviction rule, because the subsequent layer shift
    can drop L1 pages from support; picking an arbitrary victim may leave a
    dropped page in cache (violating cache ⊆ support).
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        phase_pred_misses: int = 0,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Warm-up: identical to RDM except that we also update the predictor
        # score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Fail fast if invariants are already broken.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveRDM invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        revealed = self._revealed_pages()
        is_revealed = address in revealed

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and len(self.support) == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                cache_pages = [p for p in self.cache if p is not None]

                if eviction_layer == 0:
                    # True L0 miss (no forgiveness) and cache full: predictor eviction.
                    victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                else:
                    # eviction_layer > 0 covers:
                    # - non-L0 misses
                    # - forgiveness case for an L0 request (eviction_layer forced to 1)
                    # In both cases, compute RDM's valid prefix; only allow predictor
                    # within that prefix for non-L0 misses (never during forgiveness).
                    cache_pages.sort(key=lambda p: self.layer_of[p])
                    j = None
                    for jj in range(eviction_layer, self.k + 1):
                        if self.layer_of[cache_pages[jj - 1]] == jj:
                            j = jj
                            break

                    victim = min(cache_pages[:j], key=self._priority_key)
                    victim_idx = self.cache.index(victim)

                target_index = victim_idx
                self.cache[target_index] = address
                self.pcs[target_index] = pc

        # Priority assignment happens on each request.
        self._assign_priority_on_request(address, layer_i, is_revealed)

        # Update layers after cache update.
        self._update_layers_after_request(address, layer_i, forgiveness)

        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveRDM invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} revealed={is_revealed} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)

        return hit


class PredictiveBoundingRDMAlgorithm(RDMAlgorithm):
    """PredictiveBoundingRDM: PredictiveRDM with bounded divergence syncing.

    This algorithm behaves like `PredictiveRDMAlgorithm` except on **non-L0**
    cache-full misses.

    On such an access:
    - Let X be the cache contents produced by running baseline RDM (equivalently
      PredictiveRDM, since predictor is only used on L0 misses) on the *same
      pre-request state* for this request.
    - If the divergence between current cache C and X is greater than
      `sync_threshold` (default 3), then "sync" toward X by evicting a resident
      page that X would *not* keep (i.e. a page in C \ X), and insert the
      requested page.
    - Otherwise, fall back to the normal PredictiveRDM eviction logic.

    Divergence is measured as the symmetric difference size |C △ X|.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        sync_threshold: int = 3,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0
        self.sync_threshold = int(sync_threshold)
        if self.sync_threshold < 0:
            raise ValueError('PredictiveBoundingRDM: sync_threshold must be >= 0')

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _baseline_rdm_cache_after(self, pc, address):
        """Return baseline RDM cache set after applying (pc,address) to a clone."""
        clone = RDMAlgorithm(associativity=self.associativity, max_support_factor=self.max_support_factor)

        clone.k = self.k
        clone.max_support_factor = self.max_support_factor
        clone.max_support = self.max_support

        clone.layers = [set() for _ in range(self.k + 1)]
        for i in range(1, self.k + 1):
            clone.layers[i] = set(self.layers[i])

        clone.t = self.t
        clone.t0 = dict(self.t0)
        clone.priority = dict(self.priority)

        clone._inited = self._inited
        clone._warmup_lru = list(self._warmup_lru)

        clone.cache = list(self.cache)
        clone.pcs = list(self.pcs)

        # Rebuild derived structures.
        clone._rebuild_layer_of()

        clone.access(pc, address)
        return {p for p in clone.cache if p is not None}

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Warm-up: identical to PredictiveRDM except that we also update the
        # predictor score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Fail fast if invariants are already broken.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveBoundingRDM invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        revealed = self._revealed_pages()
        is_revealed = address in revealed

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and len(self.support) == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                # Cache full miss.
                # Non-L0 miss: optionally sync toward baseline RDM if diverged.
                if layer_i > 0:
                    baseline_after = self._baseline_rdm_cache_after(pc, address)
                    current_set = {p for p in self.cache if p is not None}
                    divergence = len(current_set.symmetric_difference(baseline_after))

                    if divergence > self.sync_threshold:
                        # Sync: evict a resident page baseline would not keep.
                        candidates = [p for p in self.cache if p is not None and p not in baseline_after]
                        if candidates:
                            victim = candidates[0]
                            victim_idx = self.cache.index(victim)
                        else:
                            # Already aligned; fall back to baseline RDM eviction.
                            cache_pages = [p for p in self.cache if p is not None]
                            if eviction_layer == 0:
                                victim = min(cache_pages, key=self._priority_key)
                            else:
                                cache_pages.sort(key=lambda p: self.layer_of[p])
                                j = None
                                for jj in range(eviction_layer, self.k + 1):
                                    if self.layer_of[cache_pages[jj - 1]] == jj:
                                        j = jj
                                        break
                                if j is None:
                                    victim = min(cache_pages, key=self._priority_key)
                                else:
                                    victim = min(cache_pages[:j], key=self._priority_key)
                            victim_idx = self.cache.index(victim)

                        target_index = victim_idx
                        self.cache[target_index] = address
                        self.pcs[target_index] = pc
                    else:
                        # Within bound: use standard PredictiveRDM logic.
                        cache_pages = [p for p in self.cache if p is not None]
                        if eviction_layer == 0:
                            victim = min(cache_pages, key=self._priority_key)
                        else:
                            cache_pages.sort(key=lambda p: self.layer_of[p])
                            j = None
                            for jj in range(eviction_layer, self.k + 1):
                                if self.layer_of[cache_pages[jj - 1]] == jj:
                                    j = jj
                                    break
                            if j is None:
                                victim = min(cache_pages, key=self._priority_key)
                            else:
                                victim = min(cache_pages[:j], key=self._priority_key)
                        victim_idx = self.cache.index(victim)
                        target_index = victim_idx
                        self.cache[target_index] = address
                        self.pcs[target_index] = pc
                else:
                    # L0 miss and cache full: consult predictor for victim choice.
                    # Do NOT override the eviction rule during forgiveness.
                    if layer_i == 0 and (not forgiveness) and self.preds is not None:
                        victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                    else:
                        cache_pages = [p for p in self.cache if p is not None]
                        if eviction_layer == 0:
                            victim = min(cache_pages, key=self._priority_key)
                        else:
                            cache_pages.sort(key=lambda p: self.layer_of[p])
                            j = None
                            for jj in range(eviction_layer, self.k + 1):
                                if self.layer_of[cache_pages[jj - 1]] == jj:
                                    j = jj
                                    break
                            if j is None:
                                victim = min(cache_pages, key=self._priority_key)
                            else:
                                victim = min(cache_pages[:j], key=self._priority_key)
                        victim_idx = self.cache.index(victim)

                    target_index = victim_idx
                    self.cache[target_index] = address
                    self.pcs[target_index] = pc

        # Priority assignment happens on each request.
        self._assign_priority_on_request(address, layer_i, is_revealed)

        # Update layers after cache update.
        self._update_layers_after_request(address, layer_i, forgiveness)

        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveBoundingRDM invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} revealed={is_revealed} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)

        return hit


class PredictiveBoundingOnlineMinAlgorithm(OnlineMinAlgorithm):
    """PredictiveBoundingOnlineMin: PredictiveOnlineMin with bounded divergence syncing.

    This algorithm mirrors `PredictiveBoundingRDMAlgorithm` but runs in the
    OnlineMin framework.

    The only behavioral difference vs `PredictiveBoundingRDMAlgorithm` is that
    when eviction is NOT based on predictor scores, it uses OnlineMin's
    `priority` (random ranks) to choose the victim.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        sync_threshold: int = 3,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0
        self.sync_threshold = int(sync_threshold)
        if self.sync_threshold < 0:
            raise ValueError('PredictiveBoundingOnlineMin: sync_threshold must be >= 0')

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _priority_key(self, page):
        return (self.priority.get(page, float('inf')), page)

    def _evict_by_onlinemin_priority(self, eviction_layer: int) -> int:
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return 0

        if eviction_layer == 0:
            victim = min(cache_pages, key=self._priority_key)
            return self.cache.index(victim)

        cache_pages.sort(key=lambda p: self.layer_of.get(p, self.k + 1))
        j = None
        for jj in range(eviction_layer, self.k + 1):
            if self.layer_of.get(cache_pages[jj - 1], self.k + 1) == jj:
                j = jj
                break

        if j is None:
            victim = min(cache_pages, key=self._priority_key)
        else:
            victim = min(cache_pages[:j], key=self._priority_key)

        return self.cache.index(victim)

    def _baseline_onlinemin_cache_after(self, pc, address):
        """Return baseline OnlineMin cache set after applying (pc,address) to a clone."""
        clone = OnlineMinAlgorithm(associativity=self.associativity, max_support_factor=self.max_support_factor)

        clone.k = self.k
        clone.max_support_factor = self.max_support_factor
        clone.max_support = self.max_support

        clone.layers = [set() for _ in range(self.k + 1)]
        for i in range(1, self.k + 1):
            clone.layers[i] = set(self.layers[i])

        clone.priority = dict(self.priority)
        clone._free_priorities = list(self._free_priorities)

        clone.layer_of = dict(self.layer_of)
        clone.support = set(self.support)

        clone._inited = self._inited
        clone._warmup_lru = list(self._warmup_lru)

        clone.cache = list(self.cache)
        clone.pcs = list(self.pcs)

        clone.timestamp = self.timestamp
        clone.support_last_access = dict(getattr(self, 'support_last_access', {}))
        clone.support_last_evict = dict(getattr(self, 'support_last_evict', {}))

        clone._rebuild_layer_of()

        rng_state = random.getstate()
        try:
            clone.access(pc, address)
        finally:
            random.setstate(rng_state)

        return {p for p in clone.cache if p is not None}

    def access(self, pc, address) -> bool:
        ts = self.timestamp

        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Warm-up: identical to OnlineMin except that we also update predictor
        # score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.support_last_access[address] = ts
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.support_last_access[address] = ts
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Fail fast if invariants are already broken.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveBoundingOnlineMin invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                # Cache full miss.
                if layer_i > 0:
                    # Non-L0 miss: optionally sync toward baseline OnlineMin if diverged.
                    baseline_after = self._baseline_onlinemin_cache_after(pc, address)
                    current_set = {p for p in self.cache if p is not None}
                    divergence = len(current_set.symmetric_difference(baseline_after))

                    if divergence > self.sync_threshold:
                        candidates = [p for p in self.cache if p is not None and p not in baseline_after]
                        if candidates:
                            victim_idx = self.cache.index(candidates[0])
                        else:
                            victim_idx = self._evict_by_onlinemin_priority(eviction_layer)
                    else:
                        victim_idx = self._evict_by_onlinemin_priority(eviction_layer)

                    target_index = victim_idx
                    victim = self.cache[target_index]
                    if victim is not None:
                        self.support_last_evict[victim] = ts
                    self.cache[target_index] = address
                    self.pcs[target_index] = pc
                else:
                    # L0 miss and cache full: consult predictor for victim choice.
                    # Do NOT override the eviction rule during forgiveness.
                    if layer_i == 0 and (not forgiveness) and self.preds is not None:
                        victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                    else:
                        victim_idx = self._evict_by_onlinemin_priority(eviction_layer)

                    target_index = victim_idx
                    victim = self.cache[target_index]
                    if victim is not None:
                        self.support_last_evict[victim] = ts
                    self.cache[target_index] = address
                    self.pcs[target_index] = pc

        # Update layers after cache update.
        self._update_layers_after_request(address, layer_i, forgiveness)
        self.support_last_access[address] = ts

        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveBoundingOnlineMin invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)
        return hit


class PredictiveBoundingMirrorOnlineMinAlgorithm(OnlineMinAlgorithm):
    """PredictiveBoundingMirrorOnlineMin: bounding + mirror-OnlineMin syncing.

    This algorithm is similar to `PredictiveBoundingOnlineMinAlgorithm` but it
    additionally maintains a mirrored `OnlineMinAlgorithm` instance.

    On cache-full misses where eviction is NOT based on predictor scores, it may
    synchronize toward the mirrored OnlineMin by evicting an arbitrary resident
    page that the mirrored OnlineMin (post-request) does not keep.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        sync_threshold: int = 3,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0
        self.sync_threshold = int(sync_threshold)
        if self.sync_threshold < 0:
            raise ValueError('PredictiveBoundingMirrorOnlineMin: sync_threshold must be >= 0')

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        # Mirror OnlineMin whose post-request cache we sync toward.
        self.mirror_onlinemin = OnlineMinAlgorithm(
            associativity=associativity,
            max_support_factor=int(max_support_factor),
        )

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _priority_key(self, page):
        return (self.priority.get(page, float('inf')), page)

    def _evict_by_onlinemin_priority(self, eviction_layer: int) -> int:
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return 0

        if eviction_layer == 0:
            victim = min(cache_pages, key=self._priority_key)
            return self.cache.index(victim)

        cache_pages.sort(key=lambda p: self.layer_of.get(p, self.k + 1))
        j = None
        for jj in range(eviction_layer, self.k + 1):
            if self.layer_of.get(cache_pages[jj - 1], self.k + 1) == jj:
                j = jj
                break

        if j is None:
            victim = min(cache_pages, key=self._priority_key)
        else:
            victim = min(cache_pages[:j], key=self._priority_key)

        return self.cache.index(victim)

    def access(self, pc, address) -> bool:
        ts = self.timestamp

        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Run mirrored OnlineMin first so we can observe its post-request cache.
        self.mirror_onlinemin.access(pc, address)
        mirror_cache_after = {p for p in self.mirror_onlinemin.cache if p is not None}

        # Warm-up: identical to OnlineMin except that we also update predictor
        # score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.support_last_access[address] = ts
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.support_last_access[address] = ts
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Fail fast if invariants are already broken.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveBoundingMirrorOnlineMin invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                # Cache full miss.
                use_pred = (layer_i == 0 and (not forgiveness) and self.preds is not None)

                if use_pred:
                    victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                else:
                    # Non-pred eviction: optionally sync toward the mirrored OnlineMin.
                    current_set = {p for p in self.cache if p is not None}
                    divergence = len(current_set.symmetric_difference(mirror_cache_after))

                    if not forgiveness and divergence > self.sync_threshold:
                        candidates = [p for p in self.cache if p is not None and p not in mirror_cache_after]
                        if candidates:
                            victim_idx = self.cache.index(candidates[0])
                        else:
                            victim_idx = self._evict_by_onlinemin_priority(eviction_layer)
                    else:
                        victim_idx = self._evict_by_onlinemin_priority(eviction_layer)

                target_index = victim_idx
                victim = self.cache[target_index]
                if victim is not None:
                    self.support_last_evict[victim] = ts
                self.cache[target_index] = address
                self.pcs[target_index] = pc

        # Update layers after cache update.
        self._update_layers_after_request(address, layer_i, forgiveness)
        self.support_last_access[address] = ts

        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveBoundingMirrorOnlineMin invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)
        return hit


class PredictiveValidAlgorithm(OnlineMinAlgorithm):
    """PredictiveValid: OnlineMin framework with predictor-driven eviction.

    - Maintains the same support/layer structure as `OnlineMinAlgorithm`.
    - On a cache-full miss, computes OnlineMin's eviction candidate set:
      * if eviction_layer==0: all cache pages
      * else: the OnlineMin "valid prefix" `cache_pages[:j]`
    - Always evicts *within the candidate set* using predictor scores.

    This algorithm intentionally does NOT use any RDM-style priority.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        phase_policy: bool = False,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0
        self.phase_policy = bool(phase_policy)

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Warm-up: identical to OnlineMin except that we also update predictor
        # score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Fail fast if invariants are already broken.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveValid invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                cache_pages = [p for p in self.cache if p is not None]

                if eviction_layer == 0:
                    # Candidate set is the full cache.
                    victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                else:
                    # Candidate set is OnlineMin's valid prefix.
                    cache_pages.sort(key=lambda p: self.layer_of[p])
                    j = None
                    for jj in range(eviction_layer, self.k + 1):
                        if self.layer_of[cache_pages[jj - 1]] == jj:
                            j = jj
                            break

                    prefix_pages = cache_pages if j is None else cache_pages[:j]
                    prefix_indices = [self.cache.index(p) for p in prefix_pages]
                    victim_idx = self.evictor.evict([(i, self.preds[i]) for i in prefix_indices])

                target_index = victim_idx
                self.cache[target_index] = address
                self.pcs[target_index] = pc

        # Update layers after cache update (as in OnlineMin).
        self._update_layers_after_request(address, layer_i, forgiveness)

        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveValid invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)
        return hit


class PredictiveValidRandomAlgorithm(OnlineMinAlgorithm):
    """PredictiveValidRandom: OnlineMin framework with predictor-driven eviction.

    - Maintains the same support/layer structure as `OnlineMinAlgorithm`.
    - On a cache-full miss, computes OnlineMin's eviction candidate set:
      * if eviction_layer==0: all cache pages
      * else: the OnlineMin "valid prefix" `cache_pages[:j]`
    - Always evicts *within the candidate set* using predictor scores.

    This algorithm intentionally does NOT use any RDM-style priority.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        phase_policy: bool = False,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0
        self.phase_policy = bool(phase_policy)

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Warm-up: identical to OnlineMin except that we also update predictor
        # score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Fail fast if invariants are already broken.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveValidRandom invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                cache_pages = [p for p in self.cache if p is not None]

                if eviction_layer == 0:
                    # Candidate set is the full cache.
                    victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                else:
                    # Candidate set is OnlineMin's valid prefix.
                    cache_pages.sort(key=lambda p: self.layer_of[p])
                    j = None
                    for jj in range(eviction_layer, self.k + 1):
                        if self.layer_of[cache_pages[jj - 1]] == jj:
                            j = jj
                            break

                    prefix_pages = cache_pages if j is None else cache_pages[:j]
                    prefix_indices = [self.cache.index(p) for p in prefix_pages]
                    victim_idx = random.choice(prefix_indices)

                target_index = victim_idx
                self.cache[target_index] = address
                self.pcs[target_index] = pc

        # Update layers after cache update (as in OnlineMin).
        self._update_layers_after_request(address, layer_i, forgiveness)

        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveValidRandom invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)
        return hit

class PredictiveValidExtendedAlgorithm(OnlineMinAlgorithm):
    """PredictiveValidExtended: OnlineMin framework with predictor-driven eviction.

    - Maintains the same support/layer structure as `OnlineMinAlgorithm`.
    - On a cache-full miss, computes OnlineMin's eviction candidate set:
      * if eviction_layer==0: all cache pages
      * else: the OnlineMin "valid prefix" `cache_pages[:j]`
    - Always evicts *within the candidate set* using predictor scores.

    This algorithm intentionally does NOT use any RDM-style priority.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0

        self.last_access_time: Dict[object, int] = {}
        self.last_evict_time: Dict[object, int] = {}

        # Phase definition: each phase starts with one L0 request (w.r.t. the
        # current support) followed by zero or more non-L0 requests.
        # Track whether the *previous* phase contained any non-L0 miss; if so,
        # then on the next phase's L0 cache-full miss we evict randomly.

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _select_eviction_candidates_non_layer0(self, address, cache_pages):
        cache_pages.sort(key=lambda p: self.layer_of[p] if p in self.layer_of else self.k + 1)

        revealed_start_idx = self.k + 1
        for x in range(self.k, 0, -1):
            if len(self.layers[x]) == 1:
                revealed_start_idx = x
            else:
                break

        eviction_candidates = []
        for jj in range(1, self.k + 1):
            if self.last_access_time.get(cache_pages[jj - 1], -1) < self.last_evict_time.get(address, -1):
                if cache_pages[jj - 1] in self.layer_of and self.layer_of.get(cache_pages[jj - 1]) >= revealed_start_idx:
                    continue
                eviction_candidates.append(cache_pages[jj - 1])

        if len(eviction_candidates) == 0:
            for jj in range(1, self.k + 1):
                if cache_pages[jj - 1] in self.layer_of and self.layer_of.get(cache_pages[jj - 1]) >= revealed_start_idx:
                    continue
                eviction_candidates.append(cache_pages[jj - 1])

        return eviction_candidates

    def access(self, pc, address) -> bool:
        ts = self.timestamp

        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Warm-up: identical to OnlineMin except that we also update predictor
        # score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                self.last_access_time[address] = ts
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                self.last_access_time[address] = ts
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Fail fast if invariants are already broken.
        # cache_set = {p for p in self.cache if p is not None}
        # extras = [p for p in cache_set if p not in self.support]
        # if extras:
        #     raise RuntimeError(
        #         'PredictiveValidExtended invariant violated: cache contains non-support pages (pre-evict). '
        #         f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
        #     )

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        self.last_access_time[address] = ts

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                cache_pages = [p for p in self.cache if p is not None]

                victim = None

                if eviction_layer == 0:
                    # Candidate set is the full cache.
                    victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                else:
                    # Candidate set is OnlineMin's valid prefix.
                    eviction_candidates = self._select_eviction_candidates_non_layer0(address, cache_pages)
   
                    eviction_candidate_indices = [self.cache.index(p) for p in eviction_candidates]
                    victim_idx = self.evictor.evict([(i, self.preds[i]) for i in eviction_candidate_indices])

                victim = self.cache[victim_idx]
                self.last_evict_time[victim] = ts

                target_index = victim_idx
                self.cache[target_index] = address
                self.pcs[target_index] = pc

        # Update layers after cache update (as in OnlineMin).
        self._update_layers_after_request(address, layer_i, forgiveness)

        # cache_set = {p for p in self.cache if p is not None}
        # extras = [p for p in cache_set if p not in self.support]
        # if extras:
        #     raise RuntimeError(
        #         'PredictiveValidExtended invariant violated: cache contains non-support pages. '
        #         f'addr={address} hit={hit} layer_i={layer_i} forgiveness={forgiveness} '
        #         f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
        #     )

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)
        return hit


class PredictiveValidExtendedBoundingOnlineMinAlgorithm(OnlineMinAlgorithm):
    """PredictiveValidExtendedBoundingOnlineMin.

    Same as `PredictiveValidExtendedAlgorithm`, except on non-L0 cache-full misses
    it compares against a mirrored `OnlineMinAlgorithm`'s post-request cache.

    Define swap distance as `|mirror_cache_after \\ current_cache|`. If
    `swap_distance > sync_threshold`, we synchronize toward mirrored OnlineMin by
    evicting an arbitrary resident page that mirrored OnlineMin does not keep.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        sync_threshold: int = 3,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0
        self.sync_threshold = int(sync_threshold)
        if self.sync_threshold < 0:
            raise ValueError('PredictiveValidExtendedBoundingOnlineMin: sync_threshold must be >= 0')

        self.last_access_time: Dict[object, int] = {}
        self.last_evict_time: Dict[object, int] = {}

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        self.mirror_onlinemin = OnlineMinAlgorithm(
            associativity=associativity,
            max_support_factor=int(max_support_factor),
        )

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def access(self, pc, address) -> bool:
        ts = self.timestamp

        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Run mirrored OnlineMin first so we can observe its post-request cache.
        self.mirror_onlinemin.access(pc, address)
        mirror_cache_after = {p for p in self.mirror_onlinemin.cache if p is not None}

        # Warm-up: identical to OnlineMin except that we also update predictor
        # score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                self.last_access_time[address] = ts
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                self.last_access_time[address] = ts
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        self.last_access_time[address] = ts

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                cache_pages = [p for p in self.cache if p is not None]

                # Cache full miss.
                if layer_i > 0:
                    current_set = {p for p in self.cache if p is not None}
                    swap_distance = len(mirror_cache_after.difference(current_set))

                    if swap_distance > self.sync_threshold:
                        candidates = [p for p in self.cache if p is not None and p not in mirror_cache_after]
                        if candidates:
                            victim_idx = self.cache.index(candidates[0])
                        else:
                            victim_idx = None
                    else:
                        victim_idx = None
                else:
                    victim_idx = None

                if victim_idx is None:
                    if eviction_layer == 0:
                        # Candidate set is the full cache.
                        victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                    else:
                        # Candidate set is OnlineMin's valid prefix (extended rule).
                        cache_pages.sort(
                            key=lambda p: self.layer_of[p] if p in self.layer_of else self.k + 1
                        )

                        revealed_start_idx = self.k + 1
                        for x in range(self.k, 0, -1):
                            if len(self.layers[x]) == 1:
                                revealed_start_idx = x
                            else:
                                break

                        eviction_candidates = []
                        for jj in range(1, self.k + 1):
                            if self.last_access_time.get(cache_pages[jj - 1], -1) < self.last_evict_time.get(address, -1):
                                if cache_pages[jj - 1] in self.layer_of and self.layer_of.get(cache_pages[jj - 1]) >= revealed_start_idx:
                                    continue
                                eviction_candidates.append(cache_pages[jj - 1])

                        if len(eviction_candidates) == 0:
                            for jj in range(1, self.k + 1):
                                if cache_pages[jj - 1] in self.layer_of and self.layer_of.get(cache_pages[jj - 1]) >= revealed_start_idx:
                                    continue
                                eviction_candidates.append(cache_pages[jj - 1])

                        eviction_candidate_indices = [self.cache.index(p) for p in eviction_candidates]
                        victim_idx = self.evictor.evict([(i, self.preds[i]) for i in eviction_candidate_indices])

                victim = self.cache[victim_idx]
                self.last_evict_time[victim] = ts

                target_index = victim_idx
                self.cache[target_index] = address
                self.pcs[target_index] = pc

        # Update layers after cache update (as in OnlineMin).
        self._update_layers_after_request(address, layer_i, forgiveness)

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)
        return hit


class PredictiveValidExtendedBoundingRDMAlgorithm(OnlineMinAlgorithm):
    """PredictiveValidExtendedBoundingRDM.

    Identical to `PredictiveValidExtendedBoundingOnlineMinAlgorithm` except the
    reference baseline used for swap-distance and resync is `RDMAlgorithm`
    (mirrored and advanced per-request), not `OnlineMinAlgorithm`.

    Swap distance is `|mirror_cache_after \\ current_cache|`. If
    `swap_distance > sync_threshold`, we synchronize toward mirrored RDM by
    evicting an arbitrary resident page that mirrored RDM does not keep.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        sync_threshold: int = 3,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0
        self.sync_threshold = int(sync_threshold)
        if self.sync_threshold < 0:
            raise ValueError('PredictiveValidExtendedBoundingRDM: sync_threshold must be >= 0')

        self.last_access_time: Dict[object, int] = {}
        self.last_evict_time: Dict[object, int] = {}

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        self.mirror_rdm = RDMAlgorithm(
            associativity=associativity,
            max_support_factor=int(max_support_factor),
        )

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def access(self, pc, address) -> bool:
        ts = self.timestamp

        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Run mirrored RDM first so we can observe its post-request cache.
        self.mirror_rdm.access(pc, address)
        mirror_cache_after = {p for p in self.mirror_rdm.cache if p is not None}

        # Warm-up: identical to OnlineMin except that we also update predictor
        # score for the affected slot.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                self.last_access_time[address] = ts
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                self._assign_priority(address)
                self.last_access_time[address] = ts
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        self.last_access_time[address] = ts

        support_size = len(self.support)
        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and support_size == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                cache_pages = [p for p in self.cache if p is not None]

                # Cache full miss.
                if layer_i > 0:
                    current_set = {p for p in self.cache if p is not None}
                    swap_distance = len(mirror_cache_after.difference(current_set))

                    if swap_distance > self.sync_threshold:
                        candidates = [p for p in self.cache if p is not None and p not in mirror_cache_after]
                        if candidates:
                            victim_idx = self.cache.index(candidates[0])
                        else:
                            victim_idx = None
                    else:
                        victim_idx = None
                else:
                    victim_idx = None

                if victim_idx is None:
                    if eviction_layer == 0:
                        # Candidate set is the full cache.
                        victim_idx = self.evictor.evict(list(enumerate(self.preds)))
                    else:
                        # Candidate set is OnlineMin's valid prefix (extended rule).
                        cache_pages.sort(
                            key=lambda p: self.layer_of[p] if p in self.layer_of else self.k + 1
                        )

                        revealed_start_idx = self.k + 1
                        for x in range(self.k, 0, -1):
                            if len(self.layers[x]) == 1:
                                revealed_start_idx = x
                            else:
                                break

                        eviction_candidates = []
                        for jj in range(1, self.k + 1):
                            if self.last_access_time.get(cache_pages[jj - 1], -1) < self.last_evict_time.get(address, -1):
                                if cache_pages[jj - 1] in self.layer_of and self.layer_of.get(cache_pages[jj - 1]) >= revealed_start_idx:
                                    continue
                                eviction_candidates.append(cache_pages[jj - 1])

                        if len(eviction_candidates) == 0:
                            for jj in range(1, self.k + 1):
                                if cache_pages[jj - 1] in self.layer_of and self.layer_of.get(cache_pages[jj - 1]) >= revealed_start_idx:
                                    continue
                                eviction_candidates.append(cache_pages[jj - 1])

                        eviction_candidate_indices = [self.cache.index(p) for p in eviction_candidates]
                        victim_idx = self.evictor.evict([(i, self.preds[i]) for i in eviction_candidate_indices])

                victim = self.cache[victim_idx]
                self.last_evict_time[victim] = ts

                target_index = victim_idx
                self.cache[target_index] = address
                self.pcs[target_index] = pc

        # Update layers after cache update (as in OnlineMin).
        self._update_layers_after_request(address, layer_i, forgiveness)

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)
        return hit

class PredictiveMirrorRDMAlgorithm(EvictAlgorithm):
    """PredictiveMirrorRDM (PMR).

    - Maintain a mirrored `RDMAlgorithm` instance to track the RDM cache state.
    - On L0 requests (w.r.t. the mirrored RDM) that miss in this cache and the
      cache is full, evict using the predictor.
    - On non-L0 requests that miss in this cache and the cache is full, run the
      mirrored RDM on this request, then evict a page from *this* cache that the
      mirrored RDM does NOT keep after the request.

    Note: The spec text mentions evicting a page that is in the mirrored RDM
    cache but not in this cache; that is not executable (it is not resident
    here). This implementation uses the executable mirror-difference eviction
    above to resync toward RDM.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
    ) -> None:
        super().__init__(associativity)
        self.k = associativity
        self.max_support_factor = int(max_support_factor)

        self.timestamp = 0

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        self.mirror_rdm = RDMAlgorithm(
            associativity=associativity,
            max_support_factor=self.max_support_factor,
        )

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def access(self, pc, address) -> bool:
        self.before_pred(pc, address)

        # Determine L0 vs non-L0 w.r.t the mirrored RDM (pre-request state).
        if getattr(self.mirror_rdm, '_inited', False):
            layer_i = self.mirror_rdm.layer_of.get(address, 0)
        else:
            layer_i = 0

        # Run mirrored RDM first to observe its post-request cache.
        self.mirror_rdm.access(pc, address)
        mirror_cache_after = {p for p in self.mirror_rdm.cache if p is not None}

        # Normal cache access.
        if address in self.cache:
            idx = self.cache.index(address)
            self.pcs[idx] = pc
            self.after_pred(pc, address, idx)
            return True

        if None in self.cache:
            idx = self.cache.index(None)
            self.cache[idx] = address
            self.pcs[idx] = pc
            self.after_pred(pc, address, idx)
            return False

        # Cache full miss.
        if layer_i == 0:
            # L0 miss: predictor eviction.
            victim_idx = self.evictor.evict(list(enumerate(self.preds)))
        else:
            # Non-L0 miss: evict the page this cache keeps but mirrored RDM doesn't.
            candidates = [p for p in self.cache if p is not None and p not in mirror_cache_after]
            if not candidates:
                victim_idx = random.randrange(self.associativity)
                print('Warning: PredictiveMirrorRDMAlgorithm could not find a victim from mirror difference; using random eviction.')
            else:
                victim_idx = self.cache.index(candidates[0])

        self.cache[victim_idx] = address
        self.pcs[victim_idx] = pc
        self.after_pred(pc, address, victim_idx)
        return False


class PredictiveRelaxedMirrorRDMAlgorithm(EvictAlgorithm):
    """PredictiveRelaxedMirrorRDM.

    Same as `PredictiveRelaxedMirrorOnlineMinAlgorithm`, but the mirrored
    baseline is `RDMAlgorithm` instead of OnlineMin.

        Phase/budget rules:
        - A new phase starts on every L0 request (w.r.t. mirrored RDM); budget=1.
        - If request is L0: on cache-full miss, evict using predictions.
        - If request is non-L0:
            (1) this-miss & RDM-hit  => budget -= 1 and sync toward RDM
            (2) this-hit  & RDM-miss => budget += 1
            (3) this-miss & RDM-miss => if budget > 0, predictor-evict with RDM
                    candidate rule; else sync toward RDM

    Sync: evict any resident page that RDM (post-request) does NOT keep.

    RDM candidate rule (for case (3), credit>=0): compute the same eviction
    candidate prefix as `RDMAlgorithm` (using mirrored RDM's pre-request layer
    assignment), then choose the victim within that prefix using predictor
    scores.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        budget: int = 1,
    ) -> None:
        super().__init__(associativity)
        self.k = associativity
        self.max_support_factor = int(max_support_factor)

        self.timestamp = 0

        # Phase/budget state.
        self.budget_init = int(budget)
        self.budget = self.budget_init

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        self.mirror_rdm = RDMAlgorithm(
            associativity=associativity,
            max_support_factor=self.max_support_factor,
        )

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _sync_victim_idx(self, mirror_cache_after: set) -> int:
        candidates = [p for p in self.cache if p is not None and p not in mirror_cache_after]
        if not candidates:
            return random.randrange(self.associativity)
        return self.cache.index(candidates[0])

    def _rdm_candidate_indices(self, eviction_layer: int, mirror_layer_of: Dict[object, int]) -> List[int]:
        """Compute RDM-style eviction candidate indices for *this* cache.

        Mirrors `RDMAlgorithm` candidate prefix computation.
        Pages not in mirrored RDM's support are treated as being in layer k+1.
        """
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return []
        if eviction_layer == 0:
            return [self.cache.index(p) for p in cache_pages]

        def layer_idx(p):
            return mirror_layer_of.get(p, self.k + 1)

        cache_pages.sort(key=layer_idx)

        j = None
        start = max(1, int(eviction_layer))
        for jj in range(start, self.k + 1):
            if jj - 1 < len(cache_pages) and layer_idx(cache_pages[jj - 1]) == jj:
                j = jj
                break

        prefix_pages = cache_pages if j is None else cache_pages[:j]
        return [self.cache.index(p) for p in prefix_pages]

    def _pred_victim_idx_within_candidates(self, candidate_indices: List[int]) -> int:
        if not candidate_indices:
            return random.randrange(self.associativity)
        if self.preds is None:
            return random.choice(candidate_indices)
        return self.evictor.evict([(i, self.preds[i]) for i in candidate_indices])

    def access(self, pc, address) -> bool:
        self.before_pred(pc, address)

        # Determine L0 vs non-L0 w.r.t mirrored RDM (pre-request state).
        mirror_inited = getattr(self.mirror_rdm, '_inited', False)
        if mirror_inited:
            layer_i = self.mirror_rdm.layer_of.get(address, 0)
            support_size = len(self.mirror_rdm.support)
            mirror_layer_of_pre = dict(self.mirror_rdm.layer_of)
        else:
            layer_i = 0
            support_size = 0
            mirror_layer_of_pre = {}

        forgiveness = (layer_i == 0 and support_size == getattr(self.mirror_rdm, 'max_support', float('inf')))
        eviction_layer = 1 if forgiveness else layer_i

        # Phase boundary: every L0 request starts a new phase.
        if layer_i == 0:
            self.budget = self.budget_init

        # Observe RDM behavior by running it first.
        mirror_hit = self.mirror_rdm.access(pc, address)
        mirror_cache_after = {p for p in self.mirror_rdm.cache if p is not None}

        # Normal cache access.
        if address in self.cache:
            idx = self.cache.index(address)
            self.pcs[idx] = pc
            if layer_i > 0 and (not mirror_hit):
                self.budget += 1
            self.after_pred(pc, address, idx)
            return True

        if None in self.cache:
            idx = self.cache.index(None)
            self.cache[idx] = address
            self.pcs[idx] = pc
            self.after_pred(pc, address, idx)
            return False

        # Cache full miss.
        if layer_i == 0:
            # L0 miss: predictor eviction.
            if self.preds is None:
                print('Warning: PredictiveRelaxedMirrorRDMAlgorithm with no predictor on L0 miss; using random eviction.')
                victim_idx = random.randrange(self.associativity)
            else:
                victim_idx = self.evictor.evict(list(enumerate(self.preds)))
        else:
            # non-L0
            if mirror_hit:
                # (1) this-miss & RDM-hit => budget--, then sync.
                self.budget -= 1
                victim_idx = self._sync_victim_idx(mirror_cache_after)
            else:
                # RDM miss too.
                if self.budget > 0:
                    # (2) budget>0 => predictor-evict with RDM candidate rule.
                    candidate_indices = self._rdm_candidate_indices(eviction_layer, mirror_layer_of_pre)
                    victim_idx = self._pred_victim_idx_within_candidates(candidate_indices)
                else:
                    # (2) budget<=0 => sync.
                    victim_idx = self._sync_victim_idx(mirror_cache_after)

        self.cache[victim_idx] = address
        self.pcs[victim_idx] = pc
        self.after_pred(pc, address, victim_idx)
        return False


class PredictiveExtendedRelaxedMirrorRDMAlgorithm(EvictAlgorithm):
    """PredictiveExtendedRelaxedMirrorRDM.

    Identical to `PredictiveRelaxedMirrorRDMAlgorithm` except in case (3)
    (non-L0, both miss, budget>0) where the eviction candidate set is
    chosen using an *extended* rule similar to `PredictiveValidExtendedAlgorithm`
    (for eviction_layer != 0), rather than RDM's "valid prefix" rule.

    Concretely, we:
    - Sort cached pages by the mirrored RDM layer assignment (pre-request).
    - Compute the revealed suffix start index from mirrored RDM layers
      (pre-request), same criterion as in ValidExtended.
    - Prefer candidates in the first k positions that were accessed before the
      last time the requested page was evicted, skipping revealed suffix pages.
    - If none, fall back to the same set without the access-time filter.
    - Finally, pick the victim within candidates using predictor scores.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        budget: int = 1,
    ) -> None:
        super().__init__(associativity)
        self.k = associativity
        self.max_support_factor = int(max_support_factor)

        self.timestamp = 0

        # Phase/budget state.
        self.budget_init = int(budget)
        self.budget = self.budget_init

        # Extended candidate-selection state (as in PredictiveValidExtendedAlgorithm).
        self.last_access_time: Dict[object, int] = {}
        self.last_evict_time: Dict[object, int] = {}

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        self.mirror_rdm = RDMAlgorithm(
            associativity=associativity,
            max_support_factor=self.max_support_factor,
        )

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _sync_victim_idx(self, mirror_cache_after: set) -> int:
        candidates = [p for p in self.cache if p is not None and p not in mirror_cache_after]
        if not candidates:
            return random.randrange(self.associativity)
        return self.cache.index(candidates[0])

    def _extended_rdm_candidate_indices(
        self,
        eviction_layer: int,
        mirror_layer_of: Dict[object, int],
        mirror_layers: List[set],
        address,
    ) -> List[int]:
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return []
        if eviction_layer == 0:
            return [self.cache.index(p) for p in cache_pages]

        cache_pages.sort(key=lambda p: mirror_layer_of.get(p, self.k + 1))

        revealed_start_idx = self.k + 1
        for x in range(self.k, 0, -1):
            if x < len(mirror_layers) and len(mirror_layers[x]) == 1:
                revealed_start_idx = x
            else:
                break

        eviction_candidates = []
        last_evict_of_addr = self.last_evict_time.get(address, -1)

        for jj in range(1, self.k + 1):
            if jj - 1 >= len(cache_pages):
                break
            p = cache_pages[jj - 1]
            if self.last_access_time.get(p, -1) < last_evict_of_addr:
                if p in mirror_layer_of and mirror_layer_of.get(p) >= revealed_start_idx:
                    continue
                eviction_candidates.append(p)

        if len(eviction_candidates) == 0:
            for jj in range(1, self.k + 1):
                if jj - 1 >= len(cache_pages):
                    break
                p = cache_pages[jj - 1]
                if p in mirror_layer_of and mirror_layer_of.get(p) >= revealed_start_idx:
                    continue
                eviction_candidates.append(p)

        return [self.cache.index(p) for p in eviction_candidates]

    def _pred_victim_idx_within_candidates(self, candidate_indices: List[int]) -> int:
        if not candidate_indices:
            return random.randrange(self.associativity)
        if self.preds is None:
            return random.choice(candidate_indices)
        return self.evictor.evict([(i, self.preds[i]) for i in candidate_indices])

    def access(self, pc, address) -> bool:
        ts = self.timestamp

        self.before_pred(pc, address)

        # Determine L0 vs non-L0 w.r.t mirrored RDM (pre-request state).
        mirror_inited = getattr(self.mirror_rdm, '_inited', False)
        if mirror_inited:
            layer_i = self.mirror_rdm.layer_of.get(address, 0)
            support_size = len(self.mirror_rdm.support)
            mirror_layer_of_pre = dict(self.mirror_rdm.layer_of)
            mirror_layers_pre = [set() for _ in range(self.k + 1)]
            for i in range(1, self.k + 1):
                mirror_layers_pre[i] = set(self.mirror_rdm.layers[i])
        else:
            layer_i = 0
            support_size = 0
            mirror_layer_of_pre = {}
            mirror_layers_pre = [set() for _ in range(self.k + 1)]

        forgiveness = (layer_i == 0 and support_size == getattr(self.mirror_rdm, 'max_support', float('inf')))
        eviction_layer = 1 if forgiveness else layer_i

        # Phase boundary: every L0 request starts a new phase.
        if layer_i == 0:
            self.budget = self.budget_init

        # Observe RDM behavior by running it first.
        mirror_hit = self.mirror_rdm.access(pc, address)
        mirror_cache_after = {p for p in self.mirror_rdm.cache if p is not None}

        # Normal cache access.
        if address in self.cache:
            idx = self.cache.index(address)
            self.pcs[idx] = pc
            self.last_access_time[address] = ts
            if layer_i > 0 and (not mirror_hit):
                self.budget += 1
            self.after_pred(pc, address, idx)
            return True

        if None in self.cache:
            idx = self.cache.index(None)
            self.cache[idx] = address
            self.pcs[idx] = pc
            self.last_access_time[address] = ts
            self.after_pred(pc, address, idx)
            return False

        # Cache full miss.
        self.last_access_time[address] = ts

        if layer_i == 0:
            # L0 miss: predictor eviction.
            if self.preds is None:
                print('Warning: PredictiveExtendedRelaxedMirrorRDMAlgorithm with no predictor on L0 miss; using random eviction.')
                victim_idx = random.randrange(self.associativity)
            else:
                victim_idx = self.evictor.evict(list(enumerate(self.preds)))
        else:
            # non-L0
            if mirror_hit:
                # (1) this-miss & RDM-hit => budget--, then sync.
                self.budget -= 1
                victim_idx = self._sync_victim_idx(mirror_cache_after)
            else:
                # RDM miss too.
                if self.budget > 0:
                    candidate_indices = self._extended_rdm_candidate_indices(
                        eviction_layer,
                        mirror_layer_of_pre,
                        mirror_layers_pre,
                        address,
                    )
                    victim_idx = self._pred_victim_idx_within_candidates(candidate_indices)
                else:
                    victim_idx = self._sync_victim_idx(mirror_cache_after)

        victim = self.cache[victim_idx]
        self.last_evict_time[victim] = ts

        self.cache[victim_idx] = address
        self.pcs[victim_idx] = pc
        self.after_pred(pc, address, victim_idx)
        return False
    


class PredictiveRPBNewRDMAlgorithm(RDMAlgorithm):
    """RPB-new-RDM: RDM with conditional predictor eviction.

    Mirrors the current logic of `PredictiveRPBNewOnlineMinAlgorithm` (including
    the `pred_budget` mechanism). The ONLY behavioral difference is that when
    we do NOT use predictions to evict, we evict using `RDMAlgorithm`'s
    priority rule (OnOPT framework) instead of OnlineMin's priority.
    """

    def __init__(
        self,
        associativity: int,
        evictor_type: Union[Type[Evictor], partial],
        predictor_type: Union[Predictor, partial],
        max_support_factor: int = 3,
        pred_budget: int = 0,
    ) -> None:
        super().__init__(associativity=associativity, max_support_factor=max_support_factor)

        self.timestamp = 0

        cls_type = predictor_type.func if hasattr(predictor_type, 'func') else predictor_type
        if issubclass(cls_type, ReuseDistancePredictor):
            self.preds = [np.inf] * associativity
        elif issubclass(cls_type, BinaryPredictor):
            self.preds = [0] * associativity
        elif issubclass(cls_type, PhasePredictor):
            self.preds = [1] * associativity
        elif issubclass(cls_type, StatePredictor):
            self.preds = [None] * associativity
        else:
            self.preds = None

        if issubclass(cls_type, OraclePredictor):
            def oracle_access(self, pc, address, next_access_time):
                self.predictor.oracle_access(pc, address, next_access_time)
            self.oracle_access = types.MethodType(oracle_access, self)

        self.evictor = evictor_type()
        self.predictor = predictor_type()

        self.pred_budget_init = int(pred_budget)
        self.pred_budget = self.pred_budget_init

        self.prev_can_size = self.k

    def snapshot(self):
        return (list(zip(self.cache, self.pcs)), self.preds)

    def before_pred(self, pc, address):
        preds = self.predictor.refresh_scores(self.timestamp, pc, address, self.snapshot()[0])
        if preds is not None:
            self.preds = preds

    def after_pred(self, pc, address, target_index):
        pred = self.predictor.predict_score(self.timestamp, pc, address, self.snapshot()[0])
        if pred is not None and self.preds is not None:
            self.preds[target_index] = pred
        self.timestamp += 1

    def _rdm_eviction_candidate_pages(self, eviction_layer: int) -> List[object]:
        cache_pages = [p for p in self.cache if p is not None]
        if not cache_pages:
            return []
        if eviction_layer == 0:
            return cache_pages

        cache_pages.sort(key=lambda p: self.layer_of[p])
        j = None
        for jj in range(eviction_layer, self.k + 1):
            if self.layer_of[cache_pages[jj - 1]] == jj:
                j = jj
                break
        return cache_pages if j is None else cache_pages[:j]

    def _rdm_victim_idx(self, eviction_layer: int) -> int:
        candidate_pages = self._rdm_eviction_candidate_pages(eviction_layer)
        if not candidate_pages:
            return 0
        victim = min(candidate_pages, key=self._priority_key)
        return self.cache.index(victim)

    def _num_of_revealed_layrs(self) -> int:
        n = 0
        for jj in range(self.k, 0, -1):
            if len(self.layers[jj]) == 1:
                n += 1
            else:
                break
        return n

    def access(self, pc, address) -> bool:
        # Keep predictor state up to date.
        self.before_pred(pc, address)

        # Warm-up: identical to PredictiveRDM, plus predictor score update.
        if not self._inited:
            if address in self.cache:
                idx = self.cache.index(address)
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                self.after_pred(pc, address, idx)
                return True

            if None in self.cache:
                idx = self.cache.index(None)
                self.cache[idx] = address
                self.pcs[idx] = pc
                if address in self._warmup_lru:
                    self._warmup_lru.remove(address)
                self._warmup_lru.append(address)
                if None not in self.cache:
                    self._init_layers_from_warmup()
                self.after_pred(pc, address, idx)
                return False

            if not self._warmup_lru:
                self._warmup_lru = [p for p in self.cache if p is not None]
            self._init_layers_from_warmup()

        # Fail fast if invariants are already broken.
        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveRPBNewRDM invariant violated: cache contains non-support pages (pre-evict). '
                f'addr={address} extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        revealed = self._revealed_pages()
        is_revealed = address in revealed

        hit = address in self.cache
        target_index = None
        if hit:
            target_index = self.cache.index(address)
            self.pcs[target_index] = pc

        layer_i = self.layer_of.get(address, 0)
        forgiveness = (layer_i == 0 and len(self.support) == self.max_support)
        eviction_layer = 1 if forgiveness else layer_i

        if not hit:
            if None in self.cache:
                target_index = self.cache.index(None)
                self.cache[target_index] = address
                self.pcs[target_index] = pc
            else:
                use_predictor = False
                if self.preds is not None:
                    if layer_i == 0:
                        use_predictor = True
                        self.pred_budget = self.pred_budget_init
                    else:
                        candidate_pages = self._rdm_eviction_candidate_pages(eviction_layer)
                        if len(candidate_pages) <= int(self.prev_can_size / 2.718) - 1:
                            self.pred_budget += 1
                        if self.pred_budget >= 1:
                            use_predictor = True
                            self.pred_budget -= 1

                if use_predictor:
                    candidate_pages = self._rdm_eviction_candidate_pages(eviction_layer)
                    candidate_indices = [self.cache.index(p) for p in candidate_pages]
                    if not candidate_indices:
                        candidate_indices = list(range(self.k))
                    scored_candidates = [(i, self.preds[i]) for i in candidate_indices]
                    victim_idx = self.evictor.evict(scored_candidates)
                    target_index = victim_idx
                else:
                    target_index = self._rdm_victim_idx(eviction_layer)

                self.cache[target_index] = address
                self.pcs[target_index] = pc

        # Priority assignment happens on each request.
        self._assign_priority_on_request(address, layer_i, is_revealed)

        # Update layers after cache update.
        self._update_layers_after_request(address, layer_i, forgiveness)
        self.prev_can_size = self.k - self._num_of_revealed_layrs()

        cache_set = {p for p in self.cache if p is not None}
        extras = [p for p in cache_set if p not in self.support]
        if extras:
            raise RuntimeError(
                'PredictiveRPBNewRDM invariant violated: cache contains non-support pages. '
                f'addr={address} hit={hit} layer_i={layer_i} revealed={is_revealed} '
                f'extras={extras} cache={list(self.cache)} support_size={len(self.support)}'
            )

        if target_index is None:
            target_index = 0
        self.after_pred(pc, address, target_index)

        return hit