# multicore/masking_mapping.py
"""
Iterative masked-greedy mapper used for mapping tasks to K cores at a single time step.

Algorithm (iterative masked-greedy):
  1. Start with all cores free and all available tasks candidate.
  2. Compute a score for every (task, core) pair. Here we use per-task Q values as proxy
     and assume identical cores; extension: per-core affinity may be used.
  3. Pick the (task, core) pair with maximum score, assign that task to that core.
  4. Remove the task from candidate pool and mark core as busy.
  5. Repeat until no free cores or no candidate tasks remain.

This module exposes MaskedGreedyMapper with optional thresholding (skip low scores -> idle).
"""
from typing import List, Optional, Tuple
import numpy as np
import torch

IDLE_TASK = -1

class MaskedGreedyMapperMultiCore:
    def __init__(self, n_tasks: int, n_cores: int, idle_threshold: float = -1e9):
        """
        n_tasks: number of tasks
        n_cores: number of CPU cores to map in this timestep
        idle_threshold: minimum score to accept an assignment (scores below -> keep core idle)
        """
        self.n_tasks = n_tasks
        self.n_cores = n_cores
        self.idle_threshold = idle_threshold

    def map(self, per_task_scores: torch.Tensor, availability_mask: Optional[List[bool]] = None) -> List[int]:
        """
        per_task_scores: torch.Tensor [n_tasks] (higher -> more desirable)
        availability_mask: optional bool list len n_tasks (True if candidate)
        Returns: list length n_cores with task_id assigned or IDLE_TASK
        """
        if isinstance(per_task_scores, torch.Tensor):
            scores = per_task_scores.detach().cpu().numpy().astype(float).copy()
        else:
            scores = np.asarray(per_task_scores, dtype=float).copy()

        if availability_mask is None:
            available = np.ones(self.n_tasks, dtype=bool)
        else:
            available = np.asarray(availability_mask, dtype=bool)

        assigned = [IDLE_TASK] * self.n_cores
        assigned_tasks = set()
        for core_idx in range(self.n_cores):
            # candidate indices exclude already-assigned
            cand_idx = [i for i in range(self.n_tasks) if available[i] and (i not in assigned_tasks)]
            if len(cand_idx) == 0:
                assigned[core_idx] = IDLE_TASK
                continue
            cand_scores = scores[cand_idx]
            best_pos = int(np.argmax(cand_scores))
            best_task = cand_idx[best_pos]
            if cand_scores[best_pos] < self.idle_threshold:
                # too low; leave core idle
                assigned[core_idx] = IDLE_TASK
                continue
            assigned[core_idx] = int(best_task)
            assigned_tasks.add(best_task)
        return assigned
