Two Qwen3 models on one DGX Spark: the residency math Alibaba's Qwen3-80B and Qwen3-4B models were successfully co-located on a single NVIDIA DGX Spark using vLLM containers behind a LiteLLM proxy, but the 80B model's inability to emit tool calls in auto mode rendered it incompatible with agent SDKs like Hermes. The setup required careful GPU memory allocation, with gpu_memory_utilization fractions summing below 0.95 to avoid out-of-memory errors. Two Qwen3 Models on One DGX Spark: The Residency Math for Local LLM Setup The residency math, the gpu memory utilization trap, and what to verify first. Notes from my experiments with local LLMs. My agent stack with Hermes runs on a workstation. The models run on a DGX Spark on the same LAN. The split is deliberate: the workstation stays responsive, the Spark does the GPU work, and they talk over an HTTP proxy. Since I started managing the agent fleet through Clawrium https://github.com/ric03uec/clawrium , the Hermes count has climbed. More agents on more hosts, more concurrent traffic, all hitting the same Spark. What was a one-laptop, one-model setup is now a small fleet against a single backend — and the shape of the load is exactly what a single-model server can’t serve. The Spark served models through ollama for months. It worked. One model up, single config, easy to bring down. But ollama owns the card. There’s no per-process memory budget, no gpu memory utilization knob, no straightforward way to coresident a heavy model for reasoning and a fast model for quick turns. KV cache management is whatever the underlying llama.cpp backend gives you. PagedAttention isn’t there. vLLM fixes all of that. PagedAttention reclaims KV blocks instead of contiguous-pinning them. gpu memory utilization gives you a per-container budget. One Spark GB10, 119.67 GiB unified memory can run multiple vLLM containers behind a LiteLLM proxy on :4000 , and Hermes hits one URL to route to either model. The promise: serve Qwen3-Next-80B-Instruct-FP8 for the heavy work and Qwen3-4B-Instruct-2507 for fast turns, coresident, both reachable from a single endpoint. That’s the why. What follows is what it took to make the promise hold. Spark hardware will happily hold two Qwen3 models if the numbers line up. They didn’t, for several days. That’s where my last weekend went. Attempt one: trust the target First 80B config: gpu memory utilization: 0.75 , max model len: 65536 , max num seqs: 4 . vLLM’s KV cache init crashed with “No available memory for the cache blocks.” Qwen3-Next is mostly Mamba; the per-block page alignment pushes KV pool demand higher than the ~14 GiB residue after weights. Bumped to 0.85. Now the free-memory check crashed: “Free memory on device 98.51/119.67 GiB is less than desired GPU memory utilization 0.85, 101.72 GiB .” The 4B was already resident at ~16 GiB. The 80B’s 0.85 target was reading the whole card, not what was free. That’s the first lesson. gpu memory utilization i s a fraction of total GPU memory, not free memory . Two co-resident vLLM processes need their fractions to sum below ~0.95 to leave room for CUDA framework overhead. If your math assumes free, you’ll oscillate between OOMs and silent KV starvation. Settled at 0.80 / 32k / 2 for the 80B. Loaded clean. KV pool ~20.8 GiB after weights. Attempt two: point Hermes at it Then Hermes came online and tool calls came back as plain text.