Deficit Round Robin Queue Scheduling
The router uses a Deficit Round Robin (DRR) variant that is work-conserving across dispatchable physical policy-class heads. DRR determines which class can dispatch next; the configured FCFS or WSPT policy determines request order within that class.
This separation provides:
- Weighted service across policy classes with different request sizes.
- Independent within-class ordering and strict-priority tiers.
- Progress for requests whose token cost is much larger than their class quantum.
- Bounded arbitration work that does not loop once per token or DRR round.
Request Cost and Quantum
Each request receives an immutable queue snapshot when it is enqueued:
The router uses exact uncached tokens for cache-bucket classification and uses
the clamped scheduling_cost for DRR and WSPT. The snapshot is not recomputed
while the request waits.
Each physical policy class defines a positive quantum, measured in uncached
tokens. A class with quantum 4096 earns four times as much DRR credit per
round as a class with quantum 1024. This weighting controls token service,
not request count: variable-size requests consume correspondingly different
amounts of credit.
DRR State
The scheduler maintains the following state for each physical class:
- A pending heap ordered by strict priority and the class’s FCFS or WSPT policy.
- A deficit containing earned but unspent credit.
- A quantum controlling how quickly the deficit grows.
The scheduler also maintains a ring cursor identifying the first class to visit on the next arbitration call. Starting each scan at the cursor prevents the configured class order from becoming a permanent preference.
Selecting the Next Request
For each class in cursor order, the scheduler examines only the class head:
- If the class is empty, reset its deficit to zero.
- If its head cannot currently dispatch, retain its deficit but add no credit.
- If existing deficit covers the head cost, dispatch it without adding another quantum.
- Otherwise, add one quantum and dispatch if the head is now affordable.
- If the head remains unaffordable, continue to the next class.
Quantum is granted per ring round, not per request. A class that retained enough credit can dispatch multiple requests from the same weighted allocation:
Adding another quantum for every request would let small requests accumulate credit faster than they consume it and would violate the configured weighting.
Bulk Credit for Oversized Requests
A request may cost many times its class quantum. Repeatedly scanning the ring once per virtual round would make arbitration time proportional to request size. Instead, after one complete ring makes no progress, the scheduler calculates how many additional complete rounds are required for each dispatchable class:
It selects the minimum rounds_needed, adds that number of virtual rounds to
every dispatchable class, and scans the ring once more. Each class receives:
Applying the same virtual-round count preserves weighting because every class still scales credit by its own quantum.
For example:
The scheduler fast-forwards four rounds. standard gains 4000 credit and
reaches 5000, while latency gains 8000 and reaches 10000.
latency can then dispatch and retains 1000 credit after paying its cost.
If every class head is blocked, there are no dispatchable classes and the scheduler adds no bulk credit.
Charging and Cursor Movement
After selecting a request, the scheduler subtracts its immutable scheduling cost from the class deficit.
- If the class becomes empty, its deficit resets and the cursor advances.
- If the next head is already affordable, the cursor stays on the class so it can spend the remainder of its weighted burst.
- Otherwise, the class retains its remaining deficit and the cursor advances.
Blocked classes retain previously earned credit but do not accumulate more credit while blocked. This prevents unavailable classes from building an unbounded burst while preserving work they had already earned.
Dispatchability and Head-of-Line Behavior
A class head is dispatchable when its eligible workers are not all above the class busy threshold. If no eligible endpoint remains, the head proceeds to worker selection so the router can return the appropriate error instead of parking it indefinitely. Eligibility continues to enforce exact pins, worker allow-lists, DP-rank bounds, taints, and overload filtering.
Only the class head participates in arbitration. A constrained or pinned head can therefore block later requests in the same class even if those later requests could use another worker. This is intentional current behavior; FCFS/WSPT ordering is not bypassed to search deeper in a class heap.
New arrivals also join an existing backlog in their resolved class instead of bypassing queued work. Queue limits, ordering, and DRR charging apply equally to allow-listed and unconstrained requests.
Complexity and Progress
One arbitration call performs:
- At most one ring scan across all classes.
- One linear calculation for bulk virtual rounds when required.
- At most one final ring scan.
For C configured classes, arbitration is therefore O(C) regardless of the
request cost or quantum. The queue actor calls arbitration repeatedly while
work remains dispatchable, but each individual selection is bounded and
continuation draining remains local to the actor.
With only the synthetic no-YAML default class, the ring contains one class
and DRR reduces to ordinary single-queue dispatch.
Configuration
Set each physical class’s quantum in the router policy YAML:
Use larger quantum ratios only when the corresponding classes should receive larger shares of uncached-token service. For the complete policy-family and cache-bucket schema, thresholds, and per-worker queue limits, see Configuration and Tuning. See the tested sample policy for a complete profile.