#!/bin/bash

# gpu_scheduler.sh
# Usage: ./gpu_scheduler.sh <gpu_list> <tasks_file>
# Example: ./gpu_scheduler.sh "0,1,2" tasks.txt

GPUS=$1
TASKS_FILE=$2

if [ -z "$GPUS" ] || [ -z "$TASKS_FILE" ]; then
    echo "Usage: $0 <gpu_list> <tasks_file>"
    echo "Example: $0 \"0,1,2\" tasks.txt"
    exit 1
fi

# Normalize delimiters: replace commas with spaces
GPUS_NORMALIZED=${GPUS//,/ }
read -r -a GPU_ARRAY <<< "$GPUS_NORMALIZED"

# Create a named pipe (FIFO)
FIFO_NAME="/tmp/gpu_scheduler_$$.fifo"
mkfifo "$FIFO_NAME"

# Create file descriptor 3 linked to the FIFO
exec 3<>"$FIFO_NAME"
rm "$FIFO_NAME"  # Remove the file entry, FD remains open

# Pre-populate the FIFO with GPU IDs (tokens)
for gpu in "${GPU_ARRAY[@]}"; do
    echo "$gpu" >&3
done

# Function to execute a task on a GPU
run_task() {
    local task_cmd="$1"
    
    # Read a GPU token from the FIFO (blocking)
    read -u 3 gpu_id
    
    echo "[Scheduler] Starting task on GPU $gpu_id: $task_cmd"
    
    # Run the task in a subshell
    (   
        # Ensure token is returned even on failure/interrupt
        trap 'echo "$gpu_id" >&3' EXIT

        # Select GPU
        # export CUDA_VISIBLE_DEVICES="$gpu_id"

        # Export the GPU ID so the task can use it (e.g., via $GPU_ID)
        # We DO NOT set CUDA_VISIBLE_DEVICES here, as requested.
        # The task itself is responsible for using the assigned GPU_ID.
        export GPU="$gpu_id"
        
        eval "$task_cmd"
        
        # Token return is handled by trap
    ) &
}

# Read tasks and dispatch
while IFS= read -r task || [ -n "$task" ]; do
    # Skip empty lines and comments
    [[ -z "$task" || "$task" =~ ^# ]] && continue
    
    run_task "$task"
done < "$TASKS_FILE"

# Wait for all background jobs to finish
wait

echo "[Scheduler] All tasks completed."
exec 3>&- # Close FD
