<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom"><channel><title>Kv-Cache on lo0 — Blog Técnico</title><link>https://blog.lo0.es/tags/kv-cache/</link><description>Recent content in Kv-Cache on lo0 — Blog Técnico</description><generator>Hugo -- gohugo.io</generator><language>es</language><lastBuildDate>Mon, 18 May 2026 15:00:00 +0200</lastBuildDate><atom:link href="https://blog.lo0.es/tags/kv-cache/index.xml" rel="self" type="application/rss+xml"/><item><title>PagedAttention por dentro: bloques, tabla de páginas, evicción y el estado del arte del KV cache en 2026</title><link>https://blog.lo0.es/posts/pagedattention-deep-dive/</link><pubDate>Mon, 18 May 2026 15:00:00 +0200</pubDate><guid>https://blog.lo0.es/posts/pagedattention-deep-dive/</guid><description>&lt;h2 id="tldr">TL;DR&lt;/h2>
&lt;p>PagedAttention (Kwon et al., SOSP 2023) fue la idea que convirtió la gestión del KV cache de un problema de &lt;strong>malloc clásico&lt;/strong> —reservar contiguo, malgastar el 60-80%— en un problema resuelto &lt;strong>como lo resuelven los sistemas operativos desde hace medio siglo&lt;/strong>: bloques pequeños de tamaño fijo, una tabla de páginas por proceso, asignación bajo demanda. El paper midió un desperdicio menor al 4% y 2-4× más throughput agregado en el mismo hardware. Tres años después, PagedAttention sigue siendo el modelo mental dominante, pero su implementación literal ya no es la de ningún sistema de inferencia serio: la propia documentación de vLLM califica al paper original de &amp;ldquo;documento histórico&amp;rdquo;. Han llegado &lt;strong>vAttention&lt;/strong> (paginar usando la MMU de CUDA, no la indirección software), &lt;strong>EvicPress&lt;/strong> (combinar compresión y evicción), &lt;strong>KVTC&lt;/strong> (transform coding del cache), &lt;strong>LaProx&lt;/strong> (evicción como aproximación matricial), &lt;strong>disaggregated serving&lt;/strong> (prefill y decode en GPUs distintas, en producción en NVIDIA Dynamo, llm-d, Mooncake y media docena más), &lt;strong>RadixAttention&lt;/strong> de SGLang (trie de prefijos compartidos, con hit rates del 85% en cargas de agentes) y la nueva generación de &lt;strong>speculative decoding&lt;/strong> (EAGLE-3, DeepSeek MTP, Mirror Speculative). Este artículo desmonta PagedAttention al nivel del bloque, explica qué hace vLLM hoy en su lugar, y traza el mapa del estado del arte para que no te pierdas eligiendo entre quince siglas en la primera reunión.&lt;/p>
&lt;blockquote>
&lt;p>Este artículo cierra una mini-serie. El primero —&lt;a href="https://blog.lo0.es/posts/kv-cache-fundamentos/">KV cache: la memoria de trabajo que sostiene la inferencia LLM&lt;/a>— explicó por qué cada token consume VRAM. El segundo —&lt;a href="https://blog.lo0.es/posts/vllm-kubernetes/">vLLM en Kubernetes: la pieza de inferencia LLM que sí escala&lt;/a>— mostró cómo se sirve eso en producción. Éste baja al fondo: cómo se gestiona el cache &lt;strong>dentro&lt;/strong> del motor, y qué hay después de PagedAttention.&lt;/p>
&lt;/blockquote>
&lt;h2 id="la-analogía-pasar-de-malloc-al-kernel-multiproceso">La analogía: pasar de &lt;code>malloc()&lt;/code> al kernel multiproceso&lt;/h2>
&lt;p>Un programa C ingenuo pide memoria con &lt;code>malloc(N)&lt;/code> y recibe un bloque contiguo de N bytes. Si pide muchos bloques de tamaños distintos y los libera en cualquier orden, el heap se llena de huecos: hay tres megabytes libres en total, pero ningún hueco contiguo de un megabyte, y el siguiente &lt;code>malloc(1MB)&lt;/code> falla. Fragmentación externa. Si reserva siempre el peor caso &amp;ldquo;para estar seguro&amp;rdquo; —&lt;code>malloc(MAX_POSSIBLE_SIZE)&lt;/code>— el heap se queda lleno con bloques medio vacíos. Fragmentación interna.&lt;/p>
&lt;p>Los sistemas operativos modernos no permiten que eso pase con la memoria virtual de un proceso. La memoria virtual se divide en &lt;strong>páginas&lt;/strong> (4 KB típicamente), cada una asignada a un &lt;strong>marco físico&lt;/strong> en RAM mediante una &lt;strong>tabla de páginas&lt;/strong> específica del proceso. El proceso ve un espacio contiguo enorme; el SO lo respalda con marcos físicos dispersos, asignados bajo demanda y liberados cuando dejan de usarse. El concepto tiene 50 años y funciona.&lt;/p>
&lt;p>Antes de PagedAttention, &lt;strong>los motores de inferencia LLM eran programas C ingenuos&lt;/strong>. Cada sesión reservaba un bloque contiguo de KV cache dimensionado al peor caso &lt;code>max_context_len × bytes_per_token × n_layers × 2&lt;/code>. Una conversación que usa 273 tokens reservaba sitio para 32 768. Cuando el motor servía 50 sesiones simultáneas, el 60-80% de la VRAM dedicada a KV cache estaba reservada y vacía. El paper de PagedAttention midió este desperdicio en cargas reales y propuso lo evidente: tratar el KV cache como &lt;strong>memoria virtual&lt;/strong>. Bloques físicos pequeños (16 tokens), tabla de páginas por sesión, asignación bajo demanda. El resultado: &amp;lt; 4% de desperdicio, 2-4× más throughput agregado en el mismo hardware.&lt;/p>
&lt;p>La idea no era nueva fuera del mundo LLM, era nueva &lt;strong>dentro&lt;/strong>. Y eso vale como contribución: a veces traer una técnica madura de otro campo es más impactante que inventar algo desde cero.&lt;/p>
&lt;h2 id="el-paper-original-en-cristiano">El paper original, en cristiano&lt;/h2>
&lt;p>Kwon et al. publicaron &lt;em>Efficient Memory Management for Large Language Model Serving with PagedAttention&lt;/em> en SOSP 2023 e implementaron simultáneamente vLLM, que en seis meses pasó de proyecto académico a &amp;ldquo;el motor de inferencia que todo el mundo usa&amp;rdquo;. Las tres aportaciones del paper, en orden de importancia:&lt;/p>
&lt;ol>
&lt;li>&lt;strong>Cuantificación del problema&lt;/strong>: medir el desperdicio en sistemas existentes y mostrar que el 60-80% de la VRAM se estaba quemando en &lt;em>peor-caso reservations&lt;/em> que no se usaban.&lt;/li>
&lt;li>&lt;strong>El algoritmo de paging&lt;/strong>: cómo dividir el KV cache, qué tamaño de bloque elegir, cómo gestionar la tabla de páginas en GPU.&lt;/li>
&lt;li>&lt;strong>El kernel CUDA&lt;/strong>: cómo implementar la operación de atención cuando los tokens de una secuencia están dispersos por la VRAM, sin destruir el rendimiento.&lt;/li>
&lt;/ol>
&lt;h3 id="el-modelo-de-bloques">El modelo de bloques&lt;/h3>
&lt;p>El KV cache se divide en bloques de tamaño fijo. La elección por defecto en vLLM es &lt;strong>16 tokens por bloque&lt;/strong>, decisión que el paper justifica con un barrido empírico: bloques más pequeños reducen la fragmentación interna pero aumentan el overhead de metadata y de indirección; bloques más grandes mejoran throughput pero pierden eficiencia. 16 es el punto razonable para los modelos y cargas medidas.&lt;/p>
&lt;p>Cada bloque almacena los &lt;strong>K y V de N tokens consecutivos&lt;/strong> de &lt;strong>una sola sesión&lt;/strong> en &lt;strong>una sola capa&lt;/strong> del modelo. Para un Llama 3 8B con 32 capas, una sesión de 128 tokens necesita aproximadamente &lt;code>128 / 16 × 32 = 256 bloques&lt;/code> (uno por capa por grupo de 16 tokens). Los bloques son lógicamente independientes entre sí: pueden vivir en cualquier dirección física de VRAM.&lt;/p>
&lt;h3 id="la-tabla-de-páginas-block-table">La tabla de páginas (block table)&lt;/h3>
&lt;p>Cada sesión tiene asociada una &lt;strong>block table&lt;/strong>: una lista ordenada de identificadores de bloques físicos. Cuando vLLM calcula la atención para el token 200 de la sesión X, mira la block table de X, encuentra que el bloque que contiene el token 200 está en la posición &lt;code>200 / 16 = 12&lt;/code> de la lista, lee qué bloque físico corresponde y va a buscarlo.&lt;/p>
&lt;p>La block table vive en VRAM, no en RAM como la tabla de páginas del SO. Si viviese en CPU, cada paso de decode tendría que hacer una indirección PCIe, lo que mataría el throughput. Está en VRAM, junto al cache, y el kernel CUDA la lee como una estructura más durante el cómputo.&lt;/p>
&lt;div class="diagram" style="max-width:720px;margin:1.5rem auto;">
&lt;svg viewBox="0 0 720 280" xmlns="http://www.w3.org/2000/svg" role="img" aria-label="Block table apuntando a bloques físicos dispersos">
&lt;style>.title{font:600 13px sans-serif;fill:#222}.lbl{font:11px sans-serif;fill:#444}.bt{fill:#ffe9d6;stroke:#666}.blk{fill:#d6eaff;stroke:#666}.free{fill:#eee;stroke:#bbb;stroke-dasharray:3 2}.arr{stroke:#888;stroke-width:1.2;fill:none;marker-end:url(#ah)}&lt;/style>
&lt;defs>&lt;marker id="ah" viewBox="0 0 10 10" refX="9" refY="5" markerWidth="6" markerHeight="6" orient="auto">&lt;path d="M0,0 L10,5 L0,10 z" fill="#888"/>&lt;/marker>&lt;/defs>
&lt;text x="120" y="20" text-anchor="middle" class="title">Block table (sesión X)&lt;/text>
&lt;text x="500" y="20" text-anchor="middle" class="title">VRAM (pool de bloques físicos)&lt;/text>
&lt;rect x="40" y="40" width="160" height="22" class="bt"/>&lt;text x="120" y="56" text-anchor="middle" class="lbl">posición 0 → bloque #7&lt;/text>
&lt;rect x="40" y="65" width="160" height="22" class="bt"/>&lt;text x="120" y="81" text-anchor="middle" class="lbl">posición 1 → bloque #2&lt;/text>
&lt;rect x="40" y="90" width="160" height="22" class="bt"/>&lt;text x="120" y="106" text-anchor="middle" class="lbl">posición 2 → bloque #11&lt;/text>
&lt;rect x="40" y="115" width="160" height="22" class="bt"/>&lt;text x="120" y="131" text-anchor="middle" class="lbl">posición 3 → bloque #5&lt;/text>
&lt;rect x="40" y="140" width="160" height="22" class="bt"/>&lt;text x="120" y="156" text-anchor="middle" class="lbl">posición 4 → bloque #9&lt;/text>
&lt;rect x="300" y="40" width="60" height="22" class="free"/>&lt;text x="330" y="56" text-anchor="middle" class="lbl">#0 libre&lt;/text>
&lt;rect x="365" y="40" width="60" height="22" class="free"/>&lt;text x="395" y="56" text-anchor="middle" class="lbl">#1 libre&lt;/text>
&lt;rect x="430" y="40" width="60" height="22" class="blk"/>&lt;text x="460" y="56" text-anchor="middle" class="lbl">#2 sesión X&lt;/text>
&lt;rect x="495" y="40" width="60" height="22" class="blk"/>&lt;text x="525" y="56" text-anchor="middle" class="lbl">#3 sesión Y&lt;/text>
&lt;rect x="560" y="40" width="60" height="22" class="blk"/>&lt;text x="590" y="56" text-anchor="middle" class="lbl">#4 sesión Z&lt;/text>
&lt;rect x="625" y="40" width="60" height="22" class="blk"/>&lt;text x="655" y="56" text-anchor="middle" class="lbl">#5 sesión X&lt;/text>
&lt;rect x="300" y="70" width="60" height="22" class="blk"/>&lt;text x="330" y="86" text-anchor="middle" class="lbl">#6 sesión Y&lt;/text>
&lt;rect x="365" y="70" width="60" height="22" class="blk"/>&lt;text x="395" y="86" text-anchor="middle" class="lbl">#7 sesión X&lt;/text>
&lt;rect x="430" y="70" width="60" height="22" class="free"/>&lt;text x="460" y="86" text-anchor="middle" class="lbl">#8 libre&lt;/text>
&lt;rect x="495" y="70" width="60" height="22" class="blk"/>&lt;text x="525" y="86" text-anchor="middle" class="lbl">#9 sesión X&lt;/text>
&lt;rect x="560" y="70" width="60" height="22" class="blk"/>&lt;text x="590" y="86" text-anchor="middle" class="lbl">#10 sesión Z&lt;/text>
&lt;rect x="625" y="70" width="60" height="22" class="blk"/>&lt;text x="655" y="86" text-anchor="middle" class="lbl">#11 sesión X&lt;/text>
&lt;path class="arr" d="M200,51 L365,51"/>
&lt;path class="arr" d="M200,76 L430,51"/>
&lt;path class="arr" d="M200,101 L625,76"/>
&lt;path class="arr" d="M200,126 L625,51"/>
&lt;path class="arr" d="M200,151 L495,81"/>
&lt;text x="360" y="200" text-anchor="middle" class="lbl">los bloques de una misma sesión están dispersos; la block table reconstruye su orden lógico&lt;/text>
&lt;text x="360" y="225" text-anchor="middle" class="lbl">cuando un bloque queda libre (sesión termina), vuelve al pool y otra sesión lo ocupa en el siguiente paso&lt;/text>
&lt;/svg>
&lt;/div>
&lt;p>Cuando una sesión genera su token N-ésimo, vLLM mira si el último bloque de la block table aún tiene huecos (&lt;code>N mod 16 != 0&lt;/code>). Si los tiene, escribe ahí. Si no, pide un bloque nuevo del &lt;strong>pool global&lt;/strong>, lo añade al final de la block table y escribe en su primera posición. Crecer la sesión cuesta &lt;strong>una asignación O(1) en el pool global más una append O(1) a la block table&lt;/strong>. Liberar una sesión devuelve sus bloques al pool: también O(N_bloques) y rapidísimo.&lt;/p>
&lt;h3 id="el-pool-de-bloques">El pool de bloques&lt;/h3>
&lt;p>El pool global se dimensiona al arrancar el motor. Lo típico:&lt;/p>
&lt;pre tabindex="0">&lt;code>bloques_disponibles = (VRAM_total - modelo - activations - overhead) / block_size_bytes
&lt;/code>&lt;/pre>&lt;p>Para una RTX 4090 (24 GB) sirviendo Llama 3 8B BF16 con cache también en BF16:&lt;/p>
&lt;pre tabindex="0">&lt;code>modelo: ~16 GB
activations: ~1.5 GB
overhead vLLM: ~1 GB
disponible para KV cache: ~5.5 GB
block_size = 16 tokens × 32 capas × 2 (K,V) × 8 KV heads × 128 head_dim × 2 bytes = 2 MB
bloques disponibles ≈ 5.5 GB / 2 MB ≈ 2800 bloques
tokens cacheables totales (todas sesiones) ≈ 2800 × 16 = 44800 ≈ 44 K tokens
&lt;/code>&lt;/pre>&lt;p>Si una sola sesión pide 32 K tokens, ocupa 2 000 bloques (de 2 800). Si las sesiones son más cortas, caben más simultáneas. El pool es &lt;strong>un recurso compartido global&lt;/strong>, no per-sesión, y ahí está la clave del aprovechamiento.&lt;/p>
&lt;h3 id="copy-on-write-para-sampling-paralelo">Copy-on-write para sampling paralelo&lt;/h3>
&lt;p>Una sutileza elegante del paper: cuando una petición usa sampling paralelo o beam search, las N secuencias &lt;strong>comparten el prefijo&lt;/strong> (el prompt + lo que se haya generado hasta el punto de divergencia). En lugar de duplicar el KV cache de ese prefijo, vLLM hace que las N secuencias &lt;strong>compartan los bloques físicos&lt;/strong> vía la block table. Solo cuando una secuencia diverge —genera un token distinto que las otras— vLLM &lt;strong>copia el último bloque&lt;/strong> afectado (no toda la secuencia) y la rama esa pasa a tener su propia versión.&lt;/p>
&lt;p>Esto es exactamente lo que hace el kernel de Linux con &lt;code>fork()&lt;/code>: copy-on-write de las páginas. La memoria solo se duplica cuando se modifica. En beam search con N=4 y prefijos largos, el ahorro es enorme.&lt;/p>
&lt;h3 id="el-kernel-cuda">El kernel CUDA&lt;/h3>
&lt;p>El reto técnico no obvio: el cómputo de atención &lt;strong>debe seguir la indirección de la block table&lt;/strong> para cada token. En la versión naïve (cache contiguo), el kernel asume que los tokens 0..N-1 de la sesión X están en direcciones contiguas y los lee de un tirón. Con paging, los tokens 0..15 están en el bloque #7, los 16..31 en el #2, los 32..47 en el #11, etc.&lt;/p>
&lt;p>El kernel &lt;code>paged_attention&lt;/code> de vLLM resuelve esto con &lt;strong>block-aware tiling&lt;/strong>: divide el cómputo de atención en chunks alineados con el tamaño de bloque (16 tokens), y para cada chunk localiza el bloque físico vía la block table y lo procesa. Es más complejo que el kernel contiguo, pero el coste medido es solo &lt;strong>5-10% de latencia adicional&lt;/strong> frente a la operación contigua equivalente, contra una ganancia de 2-4× en throughput agregado por la mejor utilización de VRAM. Compromiso aplastante.&lt;/p>
&lt;h2 id="evicción-y-preemption-qué-hace-cuando-el-pool-se-agota">Evicción y preemption: qué hace cuando el pool se agota&lt;/h2>
&lt;p>El KV cache crece. Cada token nuevo en cualquier sesión consume bloques. En un servidor con tráfico alto, el pool global se vacía. ¿Qué hacer cuando llega una nueva petición y no hay bloques libres?&lt;/p>
&lt;p>Tres opciones: &lt;strong>rechazar&lt;/strong> la petición (mala UX), &lt;strong>bloquear&lt;/strong> hasta que algo se libere (mala latencia), o &lt;strong>expulsar&lt;/strong> alguna sesión existente para hacer sitio (preemption). vLLM elige la tercera, con dos estrategias seleccionables:&lt;/p>
&lt;h3 id="estrategia-1-recompute">Estrategia 1: recompute&lt;/h3>
&lt;p>Cuando vLLM expulsa una sesión, &lt;strong>libera todos sus bloques&lt;/strong> y la pone en cola de espera. Cuando vuelve a haber sitio (otras sesiones terminan), vLLM rehace el prefill entero de la sesión expulsada desde el prompt original. El KV cache se reconstruye desde cero.&lt;/p>
&lt;p>Ventaja: liberación instantánea, no consume bandwidth de PCIe.
Coste: la sesión rehace &lt;strong>todo el cómputo del prefill&lt;/strong>, segundos o decenas de segundos para prompts largos.&lt;/p>
&lt;h3 id="estrategia-2-swap">Estrategia 2: swap&lt;/h3>
&lt;p>vLLM mueve los bloques de la sesión expulsada &lt;strong>a RAM de CPU&lt;/strong> (vía PCIe), liberando la VRAM. Cuando la sesión vuelva a tocar, vLLM la trae de vuelta a VRAM.&lt;/p>
&lt;p>Ventaja: conserva el cache, no rehace cómputo.
Coste: tiempo de transferencia PCIe (~32 GB/s en PCIe gen4 x16). Mover 4 GB de KV cache cuesta ~125 ms ida y vuelta.&lt;/p>
&lt;p>vLLM elige entre las dos en función del tamaño del cache de la sesión y de la latencia esperada. Para sesiones cortas, recompute suele ganar; para sesiones largas con prompts grandes, swap. Es configurable con &lt;code>--swap-space&lt;/code>.&lt;/p>
&lt;h3 id="el-problema-de-la-preemption-agresiva">El problema de la preemption agresiva&lt;/h3>
&lt;p>Hay un fallo de modo: si el sistema está saturado y vLLM no para de expulsar y reincorporar las mismas sesiones, todas hacen poco progreso y el throughput se hunde. Este es &lt;strong>thrashing&lt;/strong>, exactamente el mismo problema que tiene un SO cuando la presión de paginación es muy alta.&lt;/p>
&lt;p>La solución operativa es la misma que en SO: &lt;strong>admission control&lt;/strong>. Configurar &lt;code>--max-num-seqs&lt;/code> para limitar cuántas sesiones puede atender vLLM simultáneamente. Si llegan más, esperan en la cola HTTP. Mejor tener 10 sesiones avanzando rápido que 100 thrasheando.&lt;/p>
&lt;h2 id="lo-que-vllm-hace-hoy-más-allá-del-paper-original">Lo que vLLM hace hoy: más allá del paper original&lt;/h2>
&lt;p>La documentación oficial de vLLM señala que el &lt;a href="https://docs.vllm.ai/en/latest/design/paged_attention/">paper de PagedAttention es ya un documento histórico&lt;/a> que &lt;strong>ya no describe la implementación actual&lt;/strong>. ¿Qué ha cambiado?&lt;/p>
&lt;h3 id="chunked-prefill-integrado-con-paged-kv">Chunked prefill integrado con paged KV&lt;/h3>
&lt;p>El kernel original asumía que el prefill ocupaba el batch entero un paso, y el decode ocupaba batches separados. El motor actual mezcla prefill (troceado en chunks) con decode en el mismo paso, usando el mismo paged KV cache para ambos. Esto mejora la utilización de tensor cores cuando hay pocas peticiones en prefill y muchas en decode.&lt;/p>
&lt;h3 id="prefix-caching-cross-session">Prefix caching cross-session&lt;/h3>
&lt;p>El paper original ya tenía copy-on-write para sampling paralelo en una sola petición. La extensión natural fue compartir bloques de prefijo entre &lt;strong>peticiones distintas&lt;/strong> que llegan con el mismo system prompt. En vLLM se activa con &lt;code>--enable-prefix-caching&lt;/code>. Es una versión más simple que la de SGLang (no usa radix tree explícito, hace hash de bloques) pero efectiva: 30-70% mejora de TTFT en cargas con prompts compartidos.&lt;/p>
&lt;h3 id="sliding-window-attention">Sliding window attention&lt;/h3>
&lt;p>Modelos como Mistral 7B usan atención con ventana deslizante: solo atienden a los últimos K tokens (4 096 en Mistral). El motor mantiene únicamente los bloques de la ventana activa, liberando los más viejos. Esto cambia la economía: para esos modelos, el cache no crece sin límite.&lt;/p>
&lt;h3 id="flashattention-3-paged">FlashAttention-3 paged&lt;/h3>
&lt;p>Las versiones recientes de FlashAttention (especialmente FA-3) tienen kernels paged-aware optimizados para Hopper (H100). vLLM los usa por defecto en H100 cuando están disponibles, con ganancias adicionales del 15-30% sobre el kernel paged original.&lt;/p>
&lt;h2 id="vattention-paging-sin-reescribir-el-kernel">vAttention: paging sin reescribir el kernel&lt;/h2>
&lt;p>El paper de &lt;a href="https://arxiv.org/abs/2405.04437">vAttention (Prabhu et al., arxiv 2405.04437)&lt;/a> hace una observación incómoda: el coste de PagedAttention no es solo el 5-10% del kernel. Hay dos costes ocultos:&lt;/p>
&lt;ol>
&lt;li>&lt;strong>Inadaptable a kernels nuevos&lt;/strong>: cada vez que sale una optimización de atención (FlashAttention-2, FlashAttention-3, kernel custom), hay que &lt;strong>reescribir su versión paged&lt;/strong>. Eso ha hecho que vLLM frecuentemente esté 1-2 versiones por detrás del frente de FlashAttention.&lt;/li>
&lt;li>&lt;strong>Block tables en VRAM&lt;/strong>: pequeño pero constante. Para muchas sesiones, las block tables ocupan VRAM y cuestan accesos.&lt;/li>
&lt;/ol>
&lt;p>La propuesta de vAttention: usar &lt;strong>CUDA Virtual Memory Management (VMM)&lt;/strong>, las primitivas de virtual memory que NVIDIA expone desde CUDA 11.2. Con VMM puedes &lt;strong>reservar un rango virtual contiguo enorme&lt;/strong> y &lt;strong>asignar memoria física bajo demanda&lt;/strong> en porciones, mapeándolas en posiciones del rango virtual. El kernel de atención ve un rango contiguo (no necesita ser paged-aware); el runtime mete el paging dentro de la API de CUDA.&lt;/p>
&lt;p>Resultado medido en el paper: hasta &lt;strong>1.99× decode throughput&lt;/strong> sobre vLLM con FlashAttention-2 original. Y el kernel de atención es el de FlashAttention estándar, sin modificar.&lt;/p>
&lt;p>La idea es disruptiva porque sugiere que &lt;strong>la abstracción del paper de PagedAttention era inadecuada&lt;/strong>: el problema nunca fue que el cache tenía que ser físicamente paginado, sino que la asignación tenía que ser dinámica. La forma de resolverlo es delegar el paging al hardware (MMU + VMM de CUDA), no implementarlo en software.&lt;/p>
&lt;p>vAttention no ha desplazado a PagedAttention en vLLM por inercia y por consideraciones de portabilidad (VMM no está disponible en GPUs AMD ni Intel; PagedAttention sí). Pero los runtimes nuevos —y algunos forks de vLLM— ya lo están adoptando. Es plausible que en 2027 sea el default.&lt;/p>
&lt;h2 id="compresión-y-evicción-inteligente-lo-que-ha-llegado-en-2025-2026">Compresión y evicción inteligente: lo que ha llegado en 2025-2026&lt;/h2>
&lt;p>PagedAttention y vAttention atacan &lt;strong>dónde&lt;/strong> vive el cache. Otra línea de trabajo ataca &lt;strong>qué&lt;/strong> vive en el cache: si no necesitas todo el KV de un contexto largo, ¿por qué guardarlo todo?&lt;/p>
&lt;h3 id="streamingllm-xiao-et-al-2024-los-attention-sinks">StreamingLLM (Xiao et al., 2024): los attention sinks&lt;/h3>
&lt;p>El precursor conceptual. Observación: los primeros 4 tokens de cualquier contexto reciben atención desproporcionada de los tokens posteriores, incluso cuando semánticamente no son relevantes (son &amp;ldquo;sinks&amp;rdquo; para que el softmax se normalice). Si descartas todo el cache excepto los primeros 4 tokens más una ventana deslizante de los últimos K, el modelo sigue generando con calidad razonable indefinidamente.&lt;/p>
&lt;p>Impacto: permite &lt;strong>contexto efectivamente infinito&lt;/strong> con cache acotado. Coste: olvido real del contenido medio.&lt;/p>
&lt;h3 id="h2o-snapkv-2024-eviction-por-attention-score">H2O, SnapKV (2024): eviction por attention score&lt;/h3>
&lt;p>Variantes que mantienen un score acumulado de atención por token y, cuando el cache se llena, descartan los tokens con menor score. Son métodos por sesión, no por sistema: cada sesión decide qué partes de su propio cache descartar.&lt;/p>
&lt;h3 id="evicpress-microsoft-research-2026">EvicPress (Microsoft Research, 2026)&lt;/h3>
&lt;p>El paper &lt;a href="https://arxiv.org/abs/2512.14946">EvicPress: Joint KV-Cache Compression and Eviction for Efficient LLM Serving&lt;/a> hace una observación elegante: hasta ahora, evicción y compresión se han tratado como técnicas separadas. &lt;strong>Si vas a expulsar un bloque, ¿por qué no comprimirlo y guardarlo en RAM o NVMe en lugar de tirarlo?&lt;/strong> Y si lo tienes comprimido en un tier más lento, ¿cuándo merece la pena descomprimirlo y volver a HBM?&lt;/p>
&lt;p>EvicPress modela el problema como &lt;strong>optimización conjunta&lt;/strong> sobre múltiples tiers de almacenamiento (HBM, RAM, NVMe), aplica compresión lossy a los bloques candidatos a evicción y mantiene metadata para decidir cuándo trasladar de un tier a otro. Resultados: &lt;strong>2.19× faster TTFT&lt;/strong> a igual calidad de generación.&lt;/p>
&lt;p>La idea importa porque cambia el framing: el KV cache deja de ser &amp;ldquo;está o no está&amp;rdquo; para pasar a ser &amp;ldquo;está, en qué tier, con qué fidelidad&amp;rdquo;. Es directamente análogo a la jerarquía de caches L1/L2/L3 en CPUs.&lt;/p>
&lt;h3 id="kv-cache-transform-coding-kvtc-2026">KV Cache Transform Coding (KVTC, 2026)&lt;/h3>
&lt;p>&lt;a href="https://arxiv.org/abs/2511.01815">KV Cache Transform Coding for Compact Storage in LLM Inference (arxiv 2511.01815)&lt;/a> aplica al KV cache una técnica clásica de compresión de imágenes y vídeo: &lt;strong>transform coding&lt;/strong>, similar a DCT en JPEG/MPEG. Descompone los bloques de KV en una base de transformadas, descarta los coeficientes de menor energía y guarda el resto. Testeado con Llama 3, Mistral NeMo y R1-Qwen 2.5, &lt;strong>supera a quantization (INT4) y a SVD&lt;/strong> como métodos de compresión del cache. Importante: el resultado es &lt;strong>un cache comprimido reutilizable&lt;/strong>, no comprimido on-the-fly cada vez.&lt;/p>
&lt;h3 id="laprox-2026">LaProx (2026)&lt;/h3>
&lt;p>&lt;a href="https://arxiv.org/abs/2605.07234">LaProx: Reformulating KV Cache Eviction Problem for Long-Context LLM Inference (arxiv 2605.07234)&lt;/a> reformula la evicción de KV cache. Hasta ahora la mayoría de métodos son &lt;strong>head-wise y por promedios&lt;/strong> —miran scores por cabeza de atención y los promedian para decidir qué descartar—. LaProx la convierte en un problema &lt;strong>output-aware&lt;/strong> y &lt;strong>layer-wise&lt;/strong>: aproximar la multiplicación entre los attention maps y los projected value states como una matriz que se puede comprimir minimizando el error en la salida real del modelo, no en métricas auxiliares.&lt;/p>
&lt;p>La consecuencia práctica: las decisiones de evicción mejoran porque están alineadas con lo que realmente afecta a la generación, no con un proxy.&lt;/p>
&lt;h2 id="disaggregated-serving-separar-prefill-de-decode">Disaggregated serving: separar prefill de decode&lt;/h2>
&lt;p>PagedAttention y derivados optimizan &lt;strong>un motor&lt;/strong> sirviendo peticiones mezcladas. La siguiente revolución conceptual fue darse cuenta de que &lt;strong>prefill y decode no deberían correr en la misma GPU&lt;/strong>.&lt;/p>
&lt;h3 id="el-problema-de-mezclarlos">El problema de mezclarlos&lt;/h3>
&lt;p>Prefill es &lt;em>compute-bound&lt;/em>: usa los tensor cores intensamente. Decode es &lt;em>memory-bound&lt;/em>: mueve el KV cache a través del HBM. Si los mezclas en el mismo batch, una de las dos fases siempre va a ralentizar a la otra. Si entra una petición con prompt de 32 K tokens mientras hay 50 sesiones en decode, el prefill pausa a todas durante un segundo o más. Si llega una avalancha de prefills, los decodes en curso ven su latencia de token siguiente subir.&lt;/p>
&lt;h3 id="distserve-zhong-et-al-2024">DistServe (Zhong et al., 2024)&lt;/h3>
&lt;p>&lt;a href="https://arxiv.org/abs/2401.09670">DistServe (arxiv 2401.09670)&lt;/a> propuso lo evidente: &lt;strong>dedicar GPUs distintas a prefill y a decode&lt;/strong>. Las peticiones llegan a una GPU de prefill, que procesa el prompt y produce el KV cache inicial; ese KV cache se &lt;strong>transfiere&lt;/strong> a una GPU de decode, que se encarga de generar los tokens uno a uno. Resultado: &lt;strong>7.4× más goodput&lt;/strong>, o el mismo throughput con SLO 12.6× más estrictos.&lt;/p>
&lt;p>El truco no obvio es la transferencia del KV cache entre nodos. En GPUs con NVLink/NVSwitch del mismo nodo es trivial (~300 GB/s). Entre nodos con InfiniBand, el coste es manejable pero no despreciable. DistServe asume topologías que lo soporten.&lt;/p>
&lt;h3 id="splitwise-microsoft-2024">Splitwise (Microsoft, 2024)&lt;/h3>
&lt;p>&lt;a href="https://www.microsoft.com/en-us/research/publication/splitwise-efficient-generative-llm-inference-using-phase-splitting/">Splitwise&lt;/a> llevó la idea un paso más allá: &lt;strong>GPUs heterogéneas&lt;/strong>. Los prefills, compute-bound, corren en H100 o A100 (compute-optimizadas). Los decodes, memory-bound, corren en GPUs con más memoria por dólar pero menor compute (algunas variantes datacenter). Ganancia: &lt;strong>1.4× más throughput por dólar&lt;/strong>.&lt;/p>
&lt;h3 id="2026-producción">2026: producción&lt;/h3>
&lt;p>Disaggregated serving es ya &lt;strong>producción mainstream&lt;/strong>:&lt;/p>
&lt;ul>
&lt;li>&lt;strong>NVIDIA Dynamo&lt;/strong> (sucesor de Triton): primitivo nativo.&lt;/li>
&lt;li>&lt;strong>vLLM&lt;/strong>: soporta disaggregation con flags &lt;code>--disaggregation-prefill-instances&lt;/code> / &lt;code>--disaggregation-decode-instances&lt;/code>.&lt;/li>
&lt;li>&lt;strong>SGLang&lt;/strong>, &lt;strong>Ray Serve LLM&lt;/strong>, &lt;strong>llm-d&lt;/strong>, &lt;strong>LMCache&lt;/strong>, &lt;strong>Mooncake&lt;/strong>: idem.&lt;/li>
&lt;li>Operadores con stacks propios: Fireworks, Perplexity, Meta, Amazon, Modular, DeepInfra, Weka.&lt;/li>
&lt;/ul>
&lt;p>&lt;a href="https://haoailab.com/blogs/distserve-retro/">&lt;em>Disaggregated Inference: 18 Months Later&lt;/em> (Hao AI Lab, 2026)&lt;/a> hace una retrospectiva: lo que en 2024 era investigación es, en 2026, &amp;ldquo;como tener separados webservers de bases de datos&amp;rdquo;. Asumido.&lt;/p>
&lt;h3 id="ppd-no-todos-los-prefills-son-iguales-2026">PPD: no todos los prefills son iguales (2026)&lt;/h3>
&lt;p>El refinamiento más reciente: &lt;a href="https://arxiv.org/pdf/2603.13358">Not All Prefills Are Equal: PPD Disaggregation for Multi-turn LLM Serving (arxiv 2603.13358)&lt;/a>. Observación: en cargas multi-turn (asistentes conversacionales, agentes), los &amp;ldquo;prefills&amp;rdquo; sucesivos tienen estructura distinta: el primer turno es prompt nuevo, los siguientes son extensiones del cache anterior. PPD discrimina entre tipos de prefill y los enruta a clusters distintos, mejorando aún el aprovechamiento.&lt;/p>
&lt;h2 id="radixattention-el-camino-alternativo-sglang">RadixAttention: el camino alternativo (SGLang)&lt;/h2>
&lt;p>Mientras vLLM iteraba sobre PagedAttention con prefix caching basado en hashing, &lt;a href="https://github.com/sgl-project/sglang">SGLang&lt;/a> tomó otra ruta: &lt;strong>mantener un trie (radix tree) explícito de todos los prefijos que existen actualmente en el cache&lt;/strong>.&lt;/p>
&lt;h3 id="la-idea">La idea&lt;/h3>
&lt;p>Cuando llega una petición nueva con tokens &lt;code>[t1, t2, t3, ..., tN]&lt;/code>, SGLang baja por el trie tokens-a-tokens. Si los primeros K tokens del prompt coinciden con un camino del trie, esos K tokens &lt;strong>ya tienen su KV cache calculado&lt;/strong> y se reutilizan. Solo se procesa el prefill de los tokens N-K restantes.&lt;/p>
&lt;p>Esto es prefix caching, pero con una estructura de datos que captura &lt;strong>todas las relaciones de prefijo entre todas las sesiones activas simultáneamente&lt;/strong>, no solo los matches exactos de hash. Si dos peticiones comparten 137 tokens iniciales, RadixAttention lo encuentra; si una tercera comparte 89, también.&lt;/p>
&lt;h3 id="eviction-inteligente-del-trie">Eviction inteligente del trie&lt;/h3>
&lt;p>Los nodos del trie tienen un score basado en cuántas veces se han usado recientemente y cuántos descendientes tienen. Cuando hay presión de memoria, SGLang descarta los nodos menos valiosos primero, manteniendo los caminos más &amp;ldquo;calientes&amp;rdquo;. Esto es LRU + un peso por reutilización potencial.&lt;/p>
&lt;h3 id="resultados">Resultados&lt;/h3>
&lt;p>&lt;a href="https://arxiv.org/pdf/2312.07104">El paper de SGLang&lt;/a> y benchmarks posteriores reportan &lt;strong>hasta 6.4× throughput vs sin prefix caching&lt;/strong>, y un gap consistente del &lt;strong>29%&lt;/strong> sobre el prefix caching basado en hash de vLLM en cargas mixtas. En cargas con prefijos muy compartidos (agentes ReAct, multi-tenant SaaS, repo Q&amp;amp;A con system prompt común), los hit rates llegan al &lt;strong>60-85%&lt;/strong> y el ahorro de coste por petición es de &lt;strong>5-12×&lt;/strong>.&lt;/p>
&lt;h3 id="producción">Producción&lt;/h3>
&lt;p>SGLang está en producción en xAI (sirviendo Grok 3) y Microsoft Azure (DeepSeek R1 en GPUs AMD), entre otros. No es un experimento; es un sistema de inferencia maduro.&lt;/p>
&lt;h3 id="cuándo-elegirlo-sobre-vllm">Cuándo elegirlo sobre vLLM&lt;/h3>
&lt;p>Para cargas con prefijos compartidos masivos y predecibles, &lt;strong>SGLang gana claramente&lt;/strong>. Para cargas genéricas mezcladas, &lt;strong>vLLM rinde mejor por simplicidad operativa&lt;/strong>. El criterio operativo: si tu hit rate de prefix caching estimado en vLLM pasaría del 50%, plantéate SGLang.&lt;/p>
&lt;h2 id="speculative-decoding-la-dimensión-ortogonal">Speculative decoding: la dimensión ortogonal&lt;/h2>
&lt;p>PagedAttention y sus sucesores optimizan &lt;strong>dónde y cómo&lt;/strong> vive el cache. Speculative decoding ataca &lt;strong>cómo se generan los tokens&lt;/strong>, ortogonalmente al cache. La idea genérica: usar un modelo pequeño y rápido para &lt;em>adivinar&lt;/em> varios tokens por adelantado, validarlos en paralelo con el modelo grande y aceptar los que coinciden.&lt;/p>
&lt;h3 id="eagle-3-2025">EAGLE-3 (2025)&lt;/h3>
&lt;p>&lt;a href="https://huggingface.co/papers/2401.15077">EAGLE-3 (huggingface.co/papers/2401.15077, versión 3 de 2025)&lt;/a> entrena una cabeza auto-regresiva pequeña que se condiciona en &lt;strong>tres puntos del hidden state del modelo target&lt;/strong> (early, middle, late layers) en lugar de solo en el último. Esta fusión tri-layer es la razón por la que EAGLE-3 supera a EAGLE-2 en un &lt;strong>20-40%&lt;/strong>. Latencia medida: &lt;strong>2-6× speedup&lt;/strong> según tamaño de modelo y batch.&lt;/p>
&lt;h3 id="medusa-y-deepseek-mtp">Medusa y DeepSeek MTP&lt;/h3>
&lt;p>Medusa fija N cabezas de decodificación adicionales al modelo, cada una prediciendo posición +1, +2, +3. DeepSeek-V3 ships con MTP (Multi-Token Prediction) nativo, n=4, &lt;strong>entrenado conjuntamente&lt;/strong> con el modelo principal (no es un drafter externo). En inferencia, basta un flag en SGLang o vLLM (&lt;code>--speculative-model deepseek-v3-mtp&lt;/code>) y obtienes &lt;strong>1.8× speedup out of the box&lt;/strong>, sin entrenar nada adicional, sin pesos extras que hospedar.&lt;/p>
&lt;h3 id="mirror-speculative-decoding-2025">Mirror Speculative Decoding (2025)&lt;/h3>
&lt;p>&lt;a href="https://arxiv.org/pdf/2510.13161">Mirror Speculative Decoding (arxiv 2510.13161)&lt;/a> ataca un límite que se daba por dado: la verificación de los tokens especulados sigue siendo serial dentro del modelo target. Mirror Decoding reorganiza el cómputo para &lt;strong>paralelizar también la verificación&lt;/strong>, rompiendo la barrera serial del paradigma original. Las ganancias añadidas dependen del modelo y del batch, pero el paper lo posiciona como el próximo paso de la trayectoria EAGLE → EAGLE-2 → EAGLE-3.&lt;/p>
&lt;h3 id="estado-en-2026">Estado en 2026&lt;/h3>
&lt;p>Speculative decoding &lt;strong>dejó de ser optimización experimental en 2026&lt;/strong> para convertirse en &lt;strong>capa por defecto de cualquier stack serio&lt;/strong>. Combinado con KV cache optimizado, los números reportados son &lt;strong>2.8× menos latencia&lt;/strong> y &lt;strong>47% menos coste por token&lt;/strong>.&lt;/p>
&lt;p>Caveat operativo: speculative decoding es contraproducente en cargas de baja concurrencia. Si el modelo target tiene poco batch para llenar la GPU, las cabezas especulativas no compensan su overhead. Por debajo de ~4 sesiones simultáneas, suele bajar el throughput. Por encima, lo sube. Mídelo en tu carga antes de activarlo.&lt;/p>
&lt;h2 id="implicaciones-operativas-el-config-2026-para-vllm">Implicaciones operativas: el config 2026 para vLLM&lt;/h2>
&lt;p>Si en 2026 montas vLLM en producción sin pensar mucho, los flags razonables por defecto son:&lt;/p>
&lt;div class="highlight">&lt;pre tabindex="0" class="chroma">&lt;code class="language-yaml" data-lang="yaml">&lt;span class="line">&lt;span class="cl">&lt;span class="nt">args&lt;/span>&lt;span class="p">:&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">model=...&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">tensor-parallel-size=N&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">max-model-len=...&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">kv-cache-dtype=fp8 &lt;/span>&lt;span class="w"> &lt;/span>&lt;span class="c"># cuantización del cache&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">enable-prefix-caching &lt;/span>&lt;span class="w"> &lt;/span>&lt;span class="c"># ahorro fácil en cargas con prompts compartidos&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">enable-chunked-prefill &lt;/span>&lt;span class="w"> &lt;/span>&lt;span class="c"># mejor mezcla prefill/decode&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">gpu-memory-utilization=0.92 &lt;/span>&lt;span class="w"> &lt;/span>&lt;span class="c"># ya cubierto en el post anterior&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">speculative-model=... &lt;/span>&lt;span class="w"> &lt;/span>&lt;span class="c"># SI batch sostenido &amp;gt;4&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">num-speculative-tokens=4 &lt;/span>&lt;span class="w"> &lt;/span>&lt;span class="c"># acompaña al anterior&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">max-num-seqs=128 &lt;/span>&lt;span class="w"> &lt;/span>&lt;span class="c"># admission control para evitar thrashing&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;span class="line">&lt;span class="cl">&lt;span class="w">&lt;/span>- --&lt;span class="l">preemption-mode=recompute &lt;/span>&lt;span class="w"> &lt;/span>&lt;span class="c"># o swap si sesiones largas&lt;/span>&lt;span class="w">
&lt;/span>&lt;/span>&lt;/span>&lt;/code>&lt;/pre>&lt;/div>&lt;p>Para cargas con prefijos masivamente compartidos (agentes), considera &lt;strong>migrar a SGLang&lt;/strong>: el delta de eficiencia compensa la curva de aprendizaje. Para cargas de baja latencia con modelos estables (entrenados in-house, no cambias cada semana), &lt;strong>TensorRT-LLM&lt;/strong> sigue ganando en latencia pura. Para todo lo demás —que es la mayoría—, vLLM con los flags de arriba está dentro del 10% del óptimo en throughput.&lt;/p>
&lt;p>Para arquitecturas grandes (&amp;gt;100 sesiones concurrentes, SLO estricto), &lt;strong>disaggregated serving&lt;/strong> ya no es opcional. NVIDIA Dynamo o llm-d como orquestadores; vLLM o SGLang como motores debajo. La división típica: 1 nodo de prefill por cada 3-4 de decode, ajustando ratios según la longitud media de los prompts.&lt;/p>
&lt;h2 id="trampas-y-mitos-comunes">Trampas y mitos comunes&lt;/h2>
&lt;h3 id="pagedattention-vs-vattention-como-dilema">&amp;ldquo;PagedAttention vs vAttention&amp;rdquo; como dilema&lt;/h3>
&lt;p>No es un dilema. vAttention es una optimización de runtime; el modelo mental sigue siendo paging. La elección es entre dos implementaciones del mismo concepto. Operativamente: si tienes la versión de vLLM que lo soporta y CUDA VMM disponible, vAttention da más throughput; si no, paged va perfectamente.&lt;/p>
&lt;h3 id="cache-compression-sin-probar-calidad">&amp;ldquo;Cache compression sin probar calidad&amp;rdquo;&lt;/h3>
&lt;p>La industria de papers de compresión es prolífica y los benchmarks varían enormemente entre los del autor y los reales en producción. Compresión 8× &lt;em>parece&lt;/em> mágico hasta que mides degradación en tu corpus real. &lt;strong>Siempre evalúa con tus datos antes de activar compresión agresiva.&lt;/strong> Un FP8 cache es seguro casi siempre. Un INT4 cache requiere medir caso por caso.&lt;/p>
&lt;h3 id="prefix-caching-con-prompts-no-determinísticos">&amp;ldquo;Prefix caching con prompts no determinísticos&amp;rdquo;&lt;/h3>
&lt;p>Si tu pipeline inyecta timestamps, IDs únicos o cualquier variabilidad en el system prompt, &lt;strong>el hit rate de prefix caching se cae a cero&lt;/strong>. Es la trampa más común. Para que funcione, los prompts compartidos deben ser &lt;strong>bit-a-bit idénticos&lt;/strong>. Estructura los prompts en capas: parte estática primero, variable al final.&lt;/p>
&lt;h3 id="speculative-decoding-en-cargas-bajas">&amp;ldquo;Speculative decoding en cargas bajas&amp;rdquo;&lt;/h3>
&lt;p>Ya lo mencionamos: por debajo de ~4 sesiones simultáneas, speculative suele ser contraproducente. Si tu carga es batch puro o muy esporádica, &lt;strong>no la actives&lt;/strong>.&lt;/p>
&lt;h3 id="disaggregated-en-cluster-sin-red-rápida">&amp;ldquo;Disaggregated en cluster sin red rápida&amp;rdquo;&lt;/h3>
&lt;p>Si tu inter-nodo es Ethernet 25 GbE o peor, la transferencia del KV cache entre prefill y decode se convierte en cuello de botella. Disaggregation es para clusters con InfiniBand o RoCE 100/200/400 GbE. Sin eso, mejor colocated.&lt;/p>
&lt;h2 id="lo-que-no-hemos-cubierto">Lo que no hemos cubierto&lt;/h2>
&lt;p>Hay terreno suficiente para otra serie:&lt;/p>
&lt;ul>
&lt;li>&lt;strong>Mooncake (Kimi/Moonshot, 2024+)&lt;/strong>: KV cache como &lt;strong>pool compartido entre instancias&lt;/strong>, persistente en RAM/NVMe. Producción real con cientos de millones de queries.&lt;/li>
&lt;li>&lt;strong>LMCache&lt;/strong>: cache de KV persistente en disco entre arranques de vLLM. Reduce el coste de los primeros tokens en cargas con repetición temporal.&lt;/li>
&lt;li>&lt;strong>vLLM Production Stack&lt;/strong>: distribución k8s-native de vLLM con HPA, métricas, multi-modelo, ya probada en producción a escala.&lt;/li>
&lt;li>&lt;strong>Inference scheduling teórico&lt;/strong>: hay literatura aplicando CFS-like algorithms (el scheduler de Linux) al LLM serving. Promete fairness multi-tenant medible. Aún en fase académica.&lt;/li>
&lt;li>&lt;strong>Quantization del modelo combinada con quantization del cache&lt;/strong>: AWQ/GPTQ sobre los pesos + FP8 sobre el cache + INT4 sobre cache evictado. La pirámide completa.&lt;/li>
&lt;/ul>
&lt;h2 id="referencias">Referencias&lt;/h2>
&lt;p>Los papers fundacionales y las extensiones más leídas, en orden cronológico:&lt;/p>
&lt;ul>
&lt;li>Kwon et al., &lt;a href="https://arxiv.org/abs/2309.06180">&lt;em>Efficient Memory Management for Large Language Model Serving with PagedAttention&lt;/em>&lt;/a> (SOSP 2023) — paper original.&lt;/li>
&lt;li>Dao et al., &lt;a href="https://arxiv.org/abs/2307.08691">&lt;em>FlashAttention-2&lt;/em>&lt;/a> (2023) y &lt;em>FlashAttention-3&lt;/em> (2024) — kernels de atención sobre los que vLLM y vAttention apoyan.&lt;/li>
&lt;li>Xiao et al., &lt;a href="https://arxiv.org/abs/2309.17453">&lt;em>Efficient Streaming Language Models with Attention Sinks&lt;/em>&lt;/a> (StreamingLLM, 2024).&lt;/li>
&lt;li>Zhong et al., &lt;a href="https://arxiv.org/abs/2401.09670">&lt;em>DistServe: Disaggregating Prefill and Decoding for Goodput-optimized LLM Serving&lt;/em>&lt;/a> (OSDI 2024).&lt;/li>
&lt;li>Patel et al., &lt;a href="https://www.microsoft.com/en-us/research/publication/splitwise-efficient-generative-llm-inference-using-phase-splitting/">&lt;em>Splitwise: Efficient Generative LLM Inference Using Phase Splitting&lt;/em>&lt;/a> (Microsoft, 2024).&lt;/li>
&lt;li>Li et al., &lt;a href="https://huggingface.co/papers/2401.15077">&lt;em>EAGLE: Speculative Sampling Requires Rethinking Feature Uncertainty&lt;/em>&lt;/a> (2024) y EAGLE-2/3 (2024-2025).&lt;/li>
&lt;li>Prabhu et al., &lt;a href="https://arxiv.org/abs/2405.04437">&lt;em>vAttention: Dynamic Memory Management for Serving LLMs without PagedAttention&lt;/em>&lt;/a> (Microsoft, 2024-2025).&lt;/li>
&lt;li>Zheng et al., &lt;a href="https://arxiv.org/pdf/2312.07104">&lt;em>SGLang: Efficient Execution of Structured Language Model Programs&lt;/em>&lt;/a> (RadixAttention, 2024).&lt;/li>
&lt;li>DeepSeek-AI, &lt;a href="https://arxiv.org/abs/2412.19437">&lt;em>DeepSeek-V3 Technical Report&lt;/em>&lt;/a> (2024) — MTP nativo, base de speculative decoding del estado del arte.&lt;/li>
&lt;li>&lt;a href="https://arxiv.org/pdf/2510.13161">&lt;em>Mirror Speculative Decoding: Breaking the Serial Barrier in LLM Inference&lt;/em>&lt;/a> (2025).&lt;/li>
&lt;li>&lt;a href="https://arxiv.org/abs/2511.01815">&lt;em>KV Cache Transform Coding for Compact Storage in LLM Inference&lt;/em>&lt;/a> (KVTC, 2026).&lt;/li>
&lt;li>&lt;a href="https://arxiv.org/abs/2512.14946">&lt;em>EvicPress: Joint KV-Cache Compression and Eviction for Efficient LLM Serving&lt;/em>&lt;/a> (Microsoft Research, 2026).&lt;/li>
&lt;li>&lt;a href="https://arxiv.org/abs/2605.07234">&lt;em>LaProx: Reformulating KV Cache Eviction Problem for Long-Context LLM Inference&lt;/em>&lt;/a> (2026).&lt;/li>
&lt;li>&lt;a href="https://arxiv.org/pdf/2603.13358">&lt;em>Not All Prefills Are Equal: PPD Disaggregation for Multi-turn LLM Serving&lt;/em>&lt;/a> (2026).&lt;/li>
&lt;/ul>
&lt;p>Operacional:&lt;/p>
&lt;ul>
&lt;li>&lt;a href="https://docs.vllm.ai/en/latest/design/paged_attention/">vLLM Paged Attention design doc&lt;/a> — la propia doc señala que el paper original es ya &amp;ldquo;historical&amp;rdquo;.&lt;/li>
&lt;li>&lt;a href="https://haoailab.com/blogs/distserve-retro/">Disaggregated Inference: 18 Months Later&lt;/a> — Hao AI Lab @ UCSD, retrospectiva de la transición a disaggregated.&lt;/li>
&lt;li>&lt;a href="https://www.marktechpost.com/2026/04/29/top-10-kv-cache-compression-techniques-for-llm-inference-reducing-memory-overhead-across-eviction-quantization-and-low-rank-methods/">Top 10 KV Cache Compression Techniques for LLM Inference&lt;/a> — survey reciente útil como mapa.&lt;/li>
&lt;li>Artículos anteriores en este blog: &lt;a href="https://blog.lo0.es/posts/kv-cache-fundamentos/">KV cache: la memoria de trabajo que sostiene la inferencia LLM&lt;/a> y &lt;a href="https://blog.lo0.es/posts/vllm-kubernetes/">vLLM en Kubernetes: la pieza de inferencia LLM que sí escala&lt;/a>.&lt;/li>
&lt;/ul></description></item><item><title>KV cache: la memoria de trabajo que sostiene la inferencia LLM</title><link>https://blog.lo0.es/posts/kv-cache-fundamentos/</link><pubDate>Mon, 18 May 2026 00:00:00 +0000</pubDate><guid>https://blog.lo0.es/posts/kv-cache-fundamentos/</guid><description>&lt;h2 id="tldr">TL;DR&lt;/h2>
&lt;p>El KV cache es la &lt;strong>memoria de trabajo&lt;/strong> que un modelo de lenguaje mantiene durante una conversación. Sin él, cada token nuevo obligaría a recalcular toda la conversación desde el principio, con un coste &lt;strong>cuadrático&lt;/strong> en la longitud del texto. Con él, el coste es lineal pero a cambio el cache &lt;strong>vive en VRAM y crece con cada token&lt;/strong>. En la práctica, no es el modelo lo que limita cuánto contexto puedes servir: es el KV cache. Para una RTX 4090 con Llama 3 8B, cabe el modelo en 16 GB y queda apenas espacio para ~64 K tokens de cache totales (sumando todas las sesiones simultáneas). Entender este número es la diferencia entre &lt;strong>anunciar&lt;/strong> &amp;ldquo;contexto de 128 K&amp;rdquo; y &lt;strong>servirlo&lt;/strong> de verdad bajo carga.&lt;/p>
&lt;h2 id="la-analogía-el-orador-con-amnesia">La analogía: el orador con amnesia&lt;/h2>
&lt;p>Imagina que asistes a una conferencia técnica de dos horas. El ponente, cada vez que va a decir una frase nueva, &lt;strong>rebobina mentalmente toda la charla desde el inicio&lt;/strong>, recompone el hilo, y solo entonces continúa. Su próxima frase requiere rememorar la anterior; la siguiente, las dos anteriores; al cabo de una hora, cada palabra nueva le cuesta una hora de recapitulación. Una conferencia así sería materialmente imposible.&lt;/p>
&lt;p>Ahora imagina al mismo ponente con un cuaderno donde apunta, mientras habla, las dos o tres ideas clave de cada frase: sujeto, objeto, vínculo con lo anterior. Antes de cada frase nueva, ojea el cuaderno y sigue. Su próxima palabra sólo cuesta una ojeada al cuaderno, no rebobinar la charla entera.&lt;/p>
&lt;p>Ese cuaderno, en un transformer, se llama &lt;strong>KV cache&lt;/strong>. Sin él, los modelos de lenguaje conversacionales serían inviables. Con él, son productos comerciales. Pero el cuaderno &lt;strong>pesa&lt;/strong>: y entender cuánto, dónde y por qué, es lo que separa una infraestructura de inferencia que funciona de una que se cae a la tercera sesión concurrente.&lt;/p>
&lt;h2 id="el-mecanismo-en-sí-en-cristiano">El mecanismo en sí (en cristiano)&lt;/h2>
&lt;p>Un transformer genera texto &lt;strong>un token cada vez&lt;/strong>. Para decidir el siguiente token, el modelo aplica un mecanismo llamado &lt;strong>atención&lt;/strong> sobre todos los tokens previos: pregunta &amp;ldquo;¿qué partes del contexto anterior son relevantes para predecir lo que viene ahora?&amp;rdquo;.&lt;/p>
&lt;p>Internamente, cada token de entrada se proyecta a tres vectores:&lt;/p>
&lt;ul>
&lt;li>&lt;strong>Q&lt;/strong> (Query): &amp;ldquo;qué estoy buscando&amp;rdquo;&lt;/li>
&lt;li>&lt;strong>K&lt;/strong> (Key): &amp;ldquo;qué oferta este token&amp;rdquo;&lt;/li>
&lt;li>&lt;strong>V&lt;/strong> (Value): &amp;ldquo;qué información lleva este token&amp;rdquo;&lt;/li>
&lt;/ul>
&lt;p>La atención del token actual contra el contexto se calcula multiplicando su &lt;strong>Q&lt;/strong> contra las &lt;strong>K&lt;/strong> de todos los tokens previos, normalizando con softmax, y ponderando las &lt;strong>V&lt;/strong> correspondientes. Resultado: una representación contextualizada del token actual.&lt;/p>
&lt;div class="diagram" style="max-width: 720px; margin: 1.5rem auto;">
&lt;svg viewBox="0 0 720 260" xmlns="http://www.w3.org/2000/svg" role="img" aria-label="Diagrama del cálculo de atención con Q, K, V">
&lt;style>
.box { fill: #f4f4f4; stroke: #333; stroke-width: 1.5; }
.box-q { fill: #ffe9d6; }
.box-k { fill: #d6eaff; }
.box-v { fill: #d9f5d6; }
.lbl { font: 600 13px sans-serif; fill: #222; }
.lbl-sm { font: 11px sans-serif; fill: #444; }
.arr { stroke: #444; stroke-width: 1.4; fill: none; marker-end: url(#ah); }
&lt;/style>
&lt;defs>
&lt;marker id="ah" viewBox="0 0 10 10" refX="9" refY="5" markerWidth="6" markerHeight="6" orient="auto">
&lt;path d="M0,0 L10,5 L0,10 z" fill="#444"/>
&lt;/marker>
&lt;/defs>
&lt;text x="360" y="22" text-anchor="middle" class="lbl">Cálculo de atención para el token N&lt;/text>
&lt;rect x="40" y="60" width="120" height="40" rx="6" class="box box-q"/>
&lt;text x="100" y="85" text-anchor="middle" class="lbl">Q (token N)&lt;/text>
&lt;text x="100" y="115" text-anchor="middle" class="lbl-sm">"qué busco"&lt;/text>
&lt;rect x="280" y="60" width="160" height="40" rx="6" class="box box-k"/>
&lt;text x="360" y="85" text-anchor="middle" class="lbl">K (tokens 1..N)&lt;/text>
&lt;text x="360" y="115" text-anchor="middle" class="lbl-sm">del cache&lt;/text>
&lt;rect x="560" y="60" width="120" height="40" rx="6" class="box box-v"/>
&lt;text x="620" y="85" text-anchor="middle" class="lbl">V (tokens 1..N)&lt;/text>
&lt;text x="620" y="115" text-anchor="middle" class="lbl-sm">del cache&lt;/text>
&lt;path class="arr" d="M160,80 L280,80"/>
&lt;path class="arr" d="M440,80 L560,80"/>
&lt;text x="220" y="74" text-anchor="middle" class="lbl-sm">Q·Kᵀ → softmax&lt;/text>
&lt;text x="500" y="74" text-anchor="middle" class="lbl-sm">× V&lt;/text>
&lt;rect x="240" y="170" width="240" height="44" rx="6" class="box"/>
&lt;text x="360" y="197" text-anchor="middle" class="lbl">representación del token N&lt;/text>
&lt;path class="arr" d="M620,100 C620,150 480,150 480,170"/>
&lt;path class="arr" d="M100,100 C100,150 240,150 240,170"/>
&lt;/svg>
&lt;/div>
&lt;p>Aquí está la clave: para predecir el token N, sólo necesito &lt;strong>Q nuevo&lt;/strong> (el del token N) y &lt;strong>K, V de todos los tokens anteriores&lt;/strong>. Las K y V de los tokens 1..N-1 no han cambiado desde la iteración anterior. Recalcularlas sería tirar trabajo.&lt;/p>
&lt;p>&lt;strong>El KV cache es exactamente eso: la memoria que guarda K y V de cada token ya procesado, en cada capa del modelo, para no recalcularlos.&lt;/strong>&lt;/p>
&lt;h2 id="por-qué-existe-el-coste-cuadrático-sin-él">Por qué existe: el coste cuadrático sin él&lt;/h2>
&lt;p>Generar un texto de N tokens implica N pasos. En el paso &lt;code>i&lt;/code>, se calcula la atención sobre &lt;code>i&lt;/code> tokens anteriores. Sin cache, en cada paso recomputas las K, V de los &lt;code>i-1&lt;/code> tokens anteriores &lt;strong>más&lt;/strong> las del nuevo. La cuenta total de cómputos de atención crece como:&lt;/p>
&lt;pre tabindex="0">&lt;code>Σ i (i=1..N) = N·(N+1) / 2 ≈ N² / 2
&lt;/code>&lt;/pre>&lt;p>Con KV cache, sólo procesas el token nuevo en cada paso: coste &lt;strong>lineal en N&lt;/strong>.&lt;/p>
&lt;div class="diagram" style="max-width: 720px; margin: 1.5rem auto;">
&lt;svg viewBox="0 0 720 320" xmlns="http://www.w3.org/2000/svg" role="img" aria-label="Comparativa de coste lineal vs cuadrático">
&lt;style>
.ax { stroke: #333; stroke-width: 1.5; }
.grid { stroke: #ddd; stroke-width: 1; stroke-dasharray: 3,3; }
.lin { stroke: #2a9d8f; stroke-width: 2.5; fill: none; }
.quad { stroke: #e76f51; stroke-width: 2.5; fill: none; }
.lbl { font: 600 13px sans-serif; fill: #222; }
.lbl-sm { font: 11px sans-serif; fill: #555; }
.tag-lin { fill: #2a9d8f; font: 600 12px sans-serif; }
.tag-quad { fill: #e76f51; font: 600 12px sans-serif; }
&lt;/style>
&lt;text x="360" y="20" text-anchor="middle" class="lbl">Cómputo acumulado para generar N tokens&lt;/text>
&lt;line class="ax" x1="80" y1="270" x2="680" y2="270"/>
&lt;line class="ax" x1="80" y1="40" x2="80" y2="270"/>
&lt;text x="380" y="300" text-anchor="middle" class="lbl-sm">tokens generados (N)&lt;/text>
&lt;text x="30" y="155" text-anchor="middle" class="lbl-sm" transform="rotate(-90 30 155)">cómputo relativo&lt;/text>
&lt;line class="grid" x1="80" y1="220" x2="680" y2="220"/>
&lt;line class="grid" x1="80" y1="170" x2="680" y2="170"/>
&lt;line class="grid" x1="80" y1="120" x2="680" y2="120"/>
&lt;line class="grid" x1="80" y1="70" x2="680" y2="70"/>
&lt;text x="75" y="274" text-anchor="end" class="lbl-sm">0&lt;/text>
&lt;text x="75" y="224" text-anchor="end" class="lbl-sm">25%&lt;/text>
&lt;text x="75" y="174" text-anchor="end" class="lbl-sm">50%&lt;/text>
&lt;text x="75" y="124" text-anchor="end" class="lbl-sm">75%&lt;/text>
&lt;text x="75" y="74" text-anchor="end" class="lbl-sm">100%&lt;/text>
&lt;text x="80" y="285" text-anchor="middle" class="lbl-sm">0&lt;/text>
&lt;text x="230" y="285" text-anchor="middle" class="lbl-sm">1K&lt;/text>
&lt;text x="380" y="285" text-anchor="middle" class="lbl-sm">2K&lt;/text>
&lt;text x="530" y="285" text-anchor="middle" class="lbl-sm">3K&lt;/text>
&lt;text x="680" y="285" text-anchor="middle" class="lbl-sm">4K&lt;/text>
&lt;!-- Lineal: pendiente suave -->
&lt;path class="lin" d="M80,270 L680,265"/>
&lt;!-- Cuadrática: parábola -->
&lt;path class="quad" d="M80,270 Q380,270 680,70"/>
&lt;text x="640" y="258" class="tag-lin">con KV cache (lineal)&lt;/text>
&lt;text x="500" y="100" class="tag-quad">sin KV cache (cuadrático)&lt;/text>
&lt;/svg>
&lt;/div>
&lt;p>Los números concretos son demoledores:&lt;/p>
&lt;table>
&lt;thead>
&lt;tr>
&lt;th style="text-align:right">Tokens generados&lt;/th>
&lt;th style="text-align:right">Sin KV cache (operaciones)&lt;/th>
&lt;th style="text-align:right">Con KV cache&lt;/th>
&lt;th style="text-align:right">Ratio&lt;/th>
&lt;/tr>
&lt;/thead>
&lt;tbody>
&lt;tr>
&lt;td style="text-align:right">128&lt;/td>
&lt;td style="text-align:right">8 256&lt;/td>
&lt;td style="text-align:right">128&lt;/td>
&lt;td style="text-align:right">64×&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td style="text-align:right">1 024&lt;/td>
&lt;td style="text-align:right">524 800&lt;/td>
&lt;td style="text-align:right">1 024&lt;/td>
&lt;td style="text-align:right">512×&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td style="text-align:right">4 096&lt;/td>
&lt;td style="text-align:right">8 390 656&lt;/td>
&lt;td style="text-align:right">4 096&lt;/td>
&lt;td style="text-align:right">2 048×&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td style="text-align:right">32 768&lt;/td>
&lt;td style="text-align:right">536 887 296&lt;/td>
&lt;td style="text-align:right">32 768&lt;/td>
&lt;td style="text-align:right">16 384×&lt;/td>
&lt;/tr>
&lt;/tbody>
&lt;/table>
&lt;p>A los 32 K tokens, &lt;strong>el cache te ahorra cuatro órdenes de magnitud&lt;/strong> de cómputo. No es una optimización: es lo que hace que la inferencia conversacional sea posible.&lt;/p>
&lt;h2 id="el-precio-cuánto-pesa-la-mochila">El precio: cuánto pesa la mochila&lt;/h2>
&lt;p>El KV cache se paga en VRAM. La fórmula, por &lt;strong>secuencia&lt;/strong>, es:&lt;/p>
&lt;pre tabindex="0">&lt;code>KV_size = 2 · n_layers · n_kv_heads · head_dim · context_len · bytes_per_param
↑
K y V
&lt;/code>&lt;/pre>&lt;p>Por &lt;strong>token&lt;/strong> (sin el &lt;code>context_len&lt;/code>), es una constante propia del modelo. Veamos números reales:&lt;/p>
&lt;table>
&lt;thead>
&lt;tr>
&lt;th>Modelo&lt;/th>
&lt;th style="text-align:right">n_layers&lt;/th>
&lt;th style="text-align:right">n_kv_heads&lt;/th>
&lt;th style="text-align:right">head_dim&lt;/th>
&lt;th style="text-align:right">Bytes/token (BF16)&lt;/th>
&lt;th style="text-align:right">GB a 8 K ctx&lt;/th>
&lt;th style="text-align:right">GB a 32 K&lt;/th>
&lt;th style="text-align:right">GB a 128 K&lt;/th>
&lt;/tr>
&lt;/thead>
&lt;tbody>
&lt;tr>
&lt;td>Llama 3 8B (MHA hipotético)&lt;/td>
&lt;td style="text-align:right">32&lt;/td>
&lt;td style="text-align:right">32&lt;/td>
&lt;td style="text-align:right">128&lt;/td>
&lt;td style="text-align:right">524 288&lt;/td>
&lt;td style="text-align:right">4.00&lt;/td>
&lt;td style="text-align:right">16.00&lt;/td>
&lt;td style="text-align:right">64.00&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td>&lt;strong>Llama 3 8B (GQA real)&lt;/strong>&lt;/td>
&lt;td style="text-align:right">32&lt;/td>
&lt;td style="text-align:right">8&lt;/td>
&lt;td style="text-align:right">128&lt;/td>
&lt;td style="text-align:right">131 072&lt;/td>
&lt;td style="text-align:right">&lt;strong>1.00&lt;/strong>&lt;/td>
&lt;td style="text-align:right">&lt;strong>4.00&lt;/strong>&lt;/td>
&lt;td style="text-align:right">&lt;strong>16.00&lt;/strong>&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td>Llama 3 70B (GQA)&lt;/td>
&lt;td style="text-align:right">80&lt;/td>
&lt;td style="text-align:right">8&lt;/td>
&lt;td style="text-align:right">128&lt;/td>
&lt;td style="text-align:right">327 680&lt;/td>
&lt;td style="text-align:right">2.50&lt;/td>
&lt;td style="text-align:right">10.00&lt;/td>
&lt;td style="text-align:right">40.00&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td>Qwen3 8B (GQA)&lt;/td>
&lt;td style="text-align:right">36&lt;/td>
&lt;td style="text-align:right">8&lt;/td>
&lt;td style="text-align:right">128&lt;/td>
&lt;td style="text-align:right">147 456&lt;/td>
&lt;td style="text-align:right">1.12&lt;/td>
&lt;td style="text-align:right">4.50&lt;/td>
&lt;td style="text-align:right">18.00&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td>Mistral 7B (GQA)&lt;/td>
&lt;td style="text-align:right">32&lt;/td>
&lt;td style="text-align:right">8&lt;/td>
&lt;td style="text-align:right">128&lt;/td>
&lt;td style="text-align:right">131 072&lt;/td>
&lt;td style="text-align:right">1.00&lt;/td>
&lt;td style="text-align:right">4.00&lt;/td>
&lt;td style="text-align:right">16.00&lt;/td>
&lt;/tr>
&lt;/tbody>
&lt;/table>
&lt;p>Dos lecturas inmediatas:&lt;/p>
&lt;ol>
&lt;li>&lt;strong>Sin GQA, no hay 128 K que valga.&lt;/strong> Un Llama 3 8B con atención multi-head clásica necesitaría 64 GB sólo de KV cache para una única secuencia con 128 K tokens. Es decir, &lt;strong>no cabe en ninguna GPU consumer&lt;/strong>. Por eso Meta, Mistral y compañía adoptaron Grouped Query Attention.&lt;/li>
&lt;li>&lt;strong>El KV cache puede ser mayor que el modelo.&lt;/strong> Llama 3 8B BF16 ocupa ~16 GB. Con 128 K de contexto, su cache son otros 16 GB. Una sola sesión empata al modelo en VRAM.&lt;/li>
&lt;/ol>
&lt;div class="diagram" style="max-width: 720px; margin: 1.5rem auto;">
&lt;svg viewBox="0 0 720 280" xmlns="http://www.w3.org/2000/svg" role="img" aria-label="Crecimiento del KV cache con la longitud de contexto">
&lt;style>
.ax { stroke: #333; stroke-width: 1.5; }
.grid { stroke: #ddd; stroke-width: 1; stroke-dasharray: 3,3; }
.l8b { stroke: #2a9d8f; stroke-width: 2.5; fill: none; }
.l70b { stroke: #e76f51; stroke-width: 2.5; fill: none; }
.lq8 { stroke: #6a4c93; stroke-width: 2.5; fill: none; stroke-dasharray: 5,3; }
.lim { stroke: #c1121f; stroke-width: 1.5; stroke-dasharray: 4,4; }
.lbl { font: 600 13px sans-serif; fill: #222; }
.lbl-sm { font: 11px sans-serif; fill: #555; }
.tag { font: 600 11px sans-serif; }
&lt;/style>
&lt;text x="360" y="20" text-anchor="middle" class="lbl">KV cache (GB) vs longitud de contexto (1 secuencia, BF16)&lt;/text>
&lt;line class="ax" x1="80" y1="240" x2="680" y2="240"/>
&lt;line class="ax" x1="80" y1="40" x2="80" y2="240"/>
&lt;line class="grid" x1="80" y1="190" x2="680" y2="190"/>
&lt;line class="grid" x1="80" y1="140" x2="680" y2="140"/>
&lt;line class="grid" x1="80" y1="90" x2="680" y2="90"/>
&lt;text x="75" y="244" text-anchor="end" class="lbl-sm">0&lt;/text>
&lt;text x="75" y="194" text-anchor="end" class="lbl-sm">10&lt;/text>
&lt;text x="75" y="144" text-anchor="end" class="lbl-sm">20&lt;/text>
&lt;text x="75" y="94" text-anchor="end" class="lbl-sm">30&lt;/text>
&lt;text x="75" y="44" text-anchor="end" class="lbl-sm">40 GB&lt;/text>
&lt;text x="80" y="258" text-anchor="middle" class="lbl-sm">0&lt;/text>
&lt;text x="180" y="258" text-anchor="middle" class="lbl-sm">8K&lt;/text>
&lt;text x="305" y="258" text-anchor="middle" class="lbl-sm">32K&lt;/text>
&lt;text x="430" y="258" text-anchor="middle" class="lbl-sm">64K&lt;/text>
&lt;text x="680" y="258" text-anchor="middle" class="lbl-sm">128K&lt;/text>
&lt;!-- Limite VRAM disponible RTX 4090 (~8 GB libres tras modelo) -->
&lt;line class="lim" x1="80" y1="200" x2="680" y2="200"/>
&lt;text x="680" y="196" text-anchor="end" class="tag" fill="#c1121f">≈ VRAM libre tras cargar 8B en una 4090&lt;/text>
&lt;!-- Llama 3 8B GQA: lineal, 1 GB @8K, 16 GB @128K -->
&lt;path class="l8b" d="M80,240 L180,235 L305,220 L430,200 L680,160"/>
&lt;!-- Qwen3 8B GQA -->
&lt;path class="lq8" d="M80,240 L180,234 L305,217 L430,194 L680,150"/>
&lt;!-- Llama 3 70B GQA -->
&lt;path class="l70b" d="M80,240 L180,228 L305,190 L430,140 L680,40"/>
&lt;text x="690" y="160" class="tag" fill="#2a9d8f">Llama 3 8B&lt;/text>
&lt;text x="690" y="148" class="tag" fill="#6a4c93">Qwen3 8B&lt;/text>
&lt;text x="690" y="42" class="tag" fill="#e76f51">Llama 3 70B&lt;/text>
&lt;/svg>
&lt;/div>
&lt;p>La línea roja punteada marca la VRAM realista disponible en una RTX 4090 después de cargar el modelo. &lt;strong>Cualquier modelo cuya curva cruza esa línea no podrá servir ese contexto&lt;/strong> sin estrategias adicionales (cuantización del cache, offload, particionado).&lt;/p>
&lt;h2 id="la-inferencia-es-memory-bound-no-compute-bound">La inferencia es memory-bound, no compute-bound&lt;/h2>
&lt;p>Hay un equívoco común: pensar que &amp;ldquo;GPU rápida = inferencia rápida&amp;rdquo;. En el régimen donde realmente operan los servicios de inferencia con KV cache, &lt;strong>lo que se mide es el ancho de banda de memoria&lt;/strong>. Cada token nuevo exige leer las K y V de todos los tokens anteriores desde HBM. El cómputo es modesto; el movimiento de datos, masivo.&lt;/p>
&lt;p>Por eso, una H100 SXM (3.35 TB/s de HBM3) puede ser 2–3× más rápida que una A100 (1.55–2 TB/s) &lt;strong>sin que la frecuencia ni el número de cores expliquen del todo la diferencia&lt;/strong>. Lo explica el ancho de banda.&lt;/p>
&lt;p>Y por eso, también, las ofertas de &amp;ldquo;GPU baratas con mucha VRAM pero HBM lenta&amp;rdquo; (algunas variantes con GDDR6 o LPDDR5) decepcionan en inferencia con contextos largos: tienen sitio para guardar el cache pero les cuesta una eternidad releerlo.&lt;/p>
&lt;h2 id="trucos-para-que-el-cuaderno-sea-más-fino">Trucos para que el cuaderno sea más fino&lt;/h2>
&lt;p>Tres técnicas, en orden cronológico, han ido aplanando el tamaño del KV cache:&lt;/p>
&lt;p>&lt;strong>Multi-Head Attention (MHA).&lt;/strong> El planteamiento original del transformer (Vaswani et al., 2017). Cada cabeza de atención tiene su propia K y V. Caro en cache pero teóricamente máximo en expresividad. Es lo que tenían los modelos hasta ~2023.&lt;/p>
&lt;p>&lt;strong>Multi-Query Attention (MQA).&lt;/strong> Una sola K y V compartida por todas las cabezas. Reduce el cache &lt;code>n_heads&lt;/code> veces. Funciona razonablemente pero degrada calidad de generación en algunos benchmarks.&lt;/p>
&lt;p>&lt;strong>Grouped Query Attention (GQA).&lt;/strong> El término medio que ha ganado. Las cabezas se agrupan: en Llama 3 8B, 32 cabezas de query comparten K, V en grupos de 4 → 8 grupos de KV. Reduce el cache 4× respecto a MHA con casi idéntica calidad. Es el estándar de facto desde 2024.&lt;/p>
&lt;p>&lt;strong>Multi-Head Latent Attention (MLA).&lt;/strong> La innovación de DeepSeek-V2/V3: en vez de almacenar K, V por cabeza, comprime el estado en un vector latente más pequeño y proyecta a K, V en el momento. El cache puede llegar a 70 bytes/token, dos órdenes de magnitud menos que GQA. Es la razón principal por la que DeepSeek-V3 (671 B parámetros, 37 B activos) es servible en infraestructura abordable.&lt;/p>
&lt;div class="diagram" style="max-width: 640px; margin: 1.5rem auto;">
&lt;svg viewBox="0 0 640 200" xmlns="http://www.w3.org/2000/svg" role="img" aria-label="Reducción del KV cache por técnica">
&lt;style>
.bar { stroke: #333; stroke-width: 1; }
.b-mha { fill: #e76f51; }
.b-gqa { fill: #f4a261; }
.b-mqa { fill: #e9c46a; }
.b-mla { fill: #2a9d8f; }
.lbl { font: 600 13px sans-serif; fill: #222; }
.lbl-sm{ font: 11px sans-serif; fill: #444; }
&lt;/style>
&lt;text x="320" y="20" text-anchor="middle" class="lbl">KB de cache por token (Llama 3 8B equivalente, BF16)&lt;/text>
&lt;rect x="200" y="40" width="380" height="22" class="bar b-mha"/>
&lt;text x="170" y="56" text-anchor="end" class="lbl-sm">MHA (32 KV heads)&lt;/text>
&lt;text x="595" y="56" class="lbl-sm">512 KB&lt;/text>
&lt;rect x="200" y="76" width="95" height="22" class="bar b-gqa"/>
&lt;text x="170" y="92" text-anchor="end" class="lbl-sm">GQA (8 KV heads)&lt;/text>
&lt;text x="310" y="92" class="lbl-sm">128 KB&lt;/text>
&lt;rect x="200" y="112" width="12" height="22" class="bar b-mqa"/>
&lt;text x="170" y="128" text-anchor="end" class="lbl-sm">MQA (1 KV head)&lt;/text>
&lt;text x="225" y="128" class="lbl-sm">16 KB&lt;/text>
&lt;rect x="200" y="148" width="3" height="22" class="bar b-mla"/>
&lt;text x="170" y="164" text-anchor="end" class="lbl-sm">MLA (DeepSeek-V3)&lt;/text>
&lt;text x="215" y="164" class="lbl-sm">~0.5 KB (real V3)&lt;/text>
&lt;/svg>
&lt;/div>
&lt;blockquote>
&lt;p>&lt;strong>Nota:&lt;/strong> la barra de MLA es ilustrativa con valores típicos publicados por DeepSeek; la implementación exacta depende del tamaño latente. Lo importante es el orden de magnitud.&lt;/p>
&lt;/blockquote>
&lt;p>A esto se suma una cuarta técnica ortogonal: &lt;strong>cuantizar el cache&lt;/strong> a FP8, INT8 o incluso INT4. vLLM y TensorRT-LLM ya lo soportan en producción. Pasar de BF16 (2 bytes) a FP8 (1 byte) &lt;strong>divide el cache por dos&lt;/strong> con coste pequeño en calidad. Pasar a INT4, por cuatro, con coste algo mayor.&lt;/p>
&lt;h2 id="el-siguiente-dragón-la-fragmentación">El siguiente dragón: la fragmentación&lt;/h2>
&lt;p>Hasta aquí hemos hablado del cache como si fuera un bloque contiguo. En la práctica, un servidor de inferencia atiende &lt;strong>decenas de sesiones simultáneas&lt;/strong>, cada una con su propio cache que crece a un ritmo distinto. La asignación naïve —reservar el máximo posible por sesión— &lt;strong>desperdicia entre el 60 % y el 80 % de la VRAM&lt;/strong> según el paper original de PagedAttention.&lt;/p>
&lt;div class="diagram" style="max-width: 720px; margin: 1.5rem auto;">
&lt;svg viewBox="0 0 720 240" xmlns="http://www.w3.org/2000/svg" role="img" aria-label="Fragmentación del KV cache: naïve vs PagedAttention">
&lt;style>
.used { fill: #2a9d8f; stroke: #1a6e63; stroke-width: 1; }
.free { fill: #f0e7d8; stroke: #aaa; stroke-width: 1; }
.blk { stroke: #555; stroke-width: 0.5; }
.lbl { font: 600 13px sans-serif; fill: #222; }
.lbl-sm { font: 11px sans-serif; fill: #444; }
&lt;/style>
&lt;text x="180" y="22" text-anchor="middle" class="lbl">Asignación naïve (contigua)&lt;/text>
&lt;text x="540" y="22" text-anchor="middle" class="lbl">PagedAttention (bloques)&lt;/text>
&lt;!-- Naive: 4 sesiones reservan el máximo, usan poco -->
&lt;text x="30" y="60" class="lbl-sm">sesión A&lt;/text>
&lt;rect x="90" y="48" width="50" height="18" class="used"/>
&lt;rect x="140" y="48" width="180" height="18" class="free"/>
&lt;text x="30" y="92" class="lbl-sm">sesión B&lt;/text>
&lt;rect x="90" y="80" width="25" height="18" class="used"/>
&lt;rect x="115" y="80" width="205" height="18" class="free"/>
&lt;text x="30" y="124" class="lbl-sm">sesión C&lt;/text>
&lt;rect x="90" y="112" width="100" height="18" class="used"/>
&lt;rect x="190" y="112" width="130" height="18" class="free"/>
&lt;text x="30" y="156" class="lbl-sm">sesión D&lt;/text>
&lt;rect x="90" y="144" width="35" height="18" class="used"/>
&lt;rect x="125" y="144" width="195" height="18" class="free"/>
&lt;text x="180" y="190" text-anchor="middle" class="lbl-sm">→ ~70 % de VRAM reservada y vacía&lt;/text>
&lt;!-- PagedAttention: bloques pequeños, ocupación densa -->
&lt;g transform="translate(400,40)">
&lt;!-- 8 bloques x 5 filas -->
&lt;g>
&lt;!-- fila 1 -->
&lt;rect x="0" y="0" width="30" height="20" class="used blk"/>
&lt;rect x="30" y="0" width="30" height="20" class="used blk"/>
&lt;rect x="60" y="0" width="30" height="20" class="used blk"/>
&lt;rect x="90" y="0" width="30" height="20" class="used blk"/>
&lt;rect x="120" y="0" width="30" height="20" class="used blk"/>
&lt;rect x="150" y="0" width="30" height="20" class="used blk"/>
&lt;rect x="180" y="0" width="30" height="20" class="used blk"/>
&lt;rect x="210" y="0" width="30" height="20" class="used blk"/>
&lt;rect x="0" y="22" width="30" height="20" class="used blk"/>
&lt;rect x="30" y="22" width="30" height="20" class="used blk"/>
&lt;rect x="60" y="22" width="30" height="20" class="used blk"/>
&lt;rect x="90" y="22" width="30" height="20" class="used blk"/>
&lt;rect x="120" y="22" width="30" height="20" class="used blk"/>
&lt;rect x="150" y="22" width="30" height="20" class="used blk"/>
&lt;rect x="180" y="22" width="30" height="20" class="free blk"/>
&lt;rect x="210" y="22" width="30" height="20" class="free blk"/>
&lt;rect x="0" y="44" width="30" height="20" class="used blk"/>
&lt;rect x="30" y="44" width="30" height="20" class="used blk"/>
&lt;rect x="60" y="44" width="30" height="20" class="used blk"/>
&lt;rect x="90" y="44" width="30" height="20" class="free blk"/>
&lt;rect x="120" y="44" width="30" height="20" class="free blk"/>
&lt;rect x="150" y="44" width="30" height="20" class="free blk"/>
&lt;rect x="180" y="44" width="30" height="20" class="free blk"/>
&lt;rect x="210" y="44" width="30" height="20" class="free blk"/>
&lt;/g>
&lt;/g>
&lt;text x="540" y="190" text-anchor="middle" class="lbl-sm">→ &amp;lt; 4 % desperdicio (paper vLLM)&lt;/text>
&lt;/svg>
&lt;/div>
&lt;p>&lt;strong>PagedAttention&lt;/strong> —la idea de Kwon et al. (2023) que dio origen a vLLM— resuelve esto pidiendo prestada una técnica de los sistemas operativos: dividir la VRAM en &lt;strong>bloques&lt;/strong> pequeños (típicamente de 16 tokens) y mantener una &lt;strong>tabla de páginas&lt;/strong> lógicas → físicas por sesión. Una sesión ya no reserva un bloque contiguo enorme: crece un bloque cada vez, y los bloques pueden estar dispersos por la VRAM. Resultado: ocupación efectiva del 90 % en lugar del 30 %, y por tanto &lt;strong>2–4× más throughput agregado&lt;/strong> en el mismo hardware.&lt;/p>
&lt;p>PagedAttention merece artículo propio. Lo dejo apuntado para el siguiente.&lt;/p>
&lt;h2 id="dos-escenarios-workstation-vs-cluster">Dos escenarios: workstation vs cluster&lt;/h2>
&lt;p>Bajemos a casos concretos y comparemos las dos puntas del espectro de hardware on-premise: una &lt;strong>workstation con una GPU consumer&lt;/strong> frente a un &lt;strong>cluster de GPUs de datacenter&lt;/strong>. Mismas matemáticas, dos órdenes de magnitud de diferencia en lo que pueden servir.&lt;/p>
&lt;h3 id="escenario-a--una-rtx-4090-24-gb-ada-lovelace">Escenario A — Una RTX 4090 (24 GB, Ada Lovelace)&lt;/h3>
&lt;p>Configuración típica con Qwen3-8B BF16:&lt;/p>
&lt;pre tabindex="0">&lt;code>Modelo BF16: ~16 GB
Activations + overhead: ~2 GB
VRAM disponible para KV cache: ~6 GB (con margen)
&lt;/code>&lt;/pre>&lt;p>Con 144 KB/token (Qwen3-8B GQA), eso son &lt;strong>~43 K tokens totales de cache&lt;/strong> distribuidos entre todas las sesiones simultáneas. En la práctica:&lt;/p>
&lt;table>
&lt;thead>
&lt;tr>
&lt;th style="text-align:right">Concurrencia&lt;/th>
&lt;th style="text-align:right">Contexto máximo por sesión&lt;/th>
&lt;/tr>
&lt;/thead>
&lt;tbody>
&lt;tr>
&lt;td style="text-align:right">1&lt;/td>
&lt;td style="text-align:right">32 768&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td style="text-align:right">4&lt;/td>
&lt;td style="text-align:right">8 192&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td style="text-align:right">8&lt;/td>
&lt;td style="text-align:right">4 096&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td style="text-align:right">16&lt;/td>
&lt;td style="text-align:right">2 048&lt;/td>
&lt;/tr>
&lt;/tbody>
&lt;/table>
&lt;p>Si necesitas anunciar &amp;ldquo;soportamos 32 K de contexto&amp;rdquo; con concurrencia 4+, hay que &lt;strong>cuantizar el cache&lt;/strong> (FP8 baja a 72 KB/token, duplica capacidad) o &lt;strong>subir el modelo de gama&lt;/strong> (un 4B con GQA y cache cuantizado holgaría).&lt;/p>
&lt;h3 id="escenario-b--cluster-de-5h100-sxm-400-gb-agregados-nvlink">Escenario B — Cluster de 5×H100 SXM (400 GB agregados, NVLink)&lt;/h3>
&lt;p>Con tensor parallel = 5 y Llama 3 70B BF16:&lt;/p>
&lt;pre tabindex="0">&lt;code>Modelo BF16: ~140 GB (28 GB/GPU)
Overhead vLLM por GPU: ~2 GB
VRAM libre para KV por GPU: ~50 GB → ~250 GB agregados
&lt;/code>&lt;/pre>&lt;p>Con 320 KB/token (Llama 3 70B GQA), eso son &lt;strong>~800 K tokens totales de cache&lt;/strong>. Mucho margen para servir contextos largos con concurrencia alta:&lt;/p>
&lt;table>
&lt;thead>
&lt;tr>
&lt;th style="text-align:right">Concurrencia&lt;/th>
&lt;th style="text-align:right">Contexto máximo por sesión&lt;/th>
&lt;/tr>
&lt;/thead>
&lt;tbody>
&lt;tr>
&lt;td style="text-align:right">4&lt;/td>
&lt;td style="text-align:right">200 000&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td style="text-align:right">16&lt;/td>
&lt;td style="text-align:right">50 000&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td style="text-align:right">64&lt;/td>
&lt;td style="text-align:right">12 500&lt;/td>
&lt;/tr>
&lt;/tbody>
&lt;/table>
&lt;p>Para DeepSeek-V3 671 B con MLA: la economía cambia radicalmente porque el cache es ~100× más fino. Lo que limita ya no es el cache sino la VRAM del propio modelo (cuantizado FP8 son ~671 GB → no cabe en 5×H100, hace falta cluster mayor o FP4).&lt;/p>
&lt;h3 id="a-y-b-lado-a-lado">A y B, lado a lado&lt;/h3>
&lt;table>
&lt;thead>
&lt;tr>
&lt;th>Métrica&lt;/th>
&lt;th style="text-align:right">Escenario A (1×4090)&lt;/th>
&lt;th style="text-align:right">Escenario B (5×H100 SXM)&lt;/th>
&lt;/tr>
&lt;/thead>
&lt;tbody>
&lt;tr>
&lt;td>VRAM total&lt;/td>
&lt;td style="text-align:right">24 GB&lt;/td>
&lt;td style="text-align:right">400 GB (≈ 17×)&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td>Modelo servible (BF16)&lt;/td>
&lt;td style="text-align:right">Hasta ~8 B&lt;/td>
&lt;td style="text-align:right">Hasta ~70 B&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td>KV cache disponible&lt;/td>
&lt;td style="text-align:right">~6 GB&lt;/td>
&lt;td style="text-align:right">~250 GB (≈ 42×)&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td>Tokens totales de cache&lt;/td>
&lt;td style="text-align:right">~43 K&lt;/td>
&lt;td style="text-align:right">~800 K (≈ 19×)&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td>Ancho de banda HBM&lt;/td>
&lt;td style="text-align:right">1.0 TB/s (GDDR6X)&lt;/td>
&lt;td style="text-align:right">3.35 TB/s/GPU&lt;/td>
&lt;/tr>
&lt;tr>
&lt;td>Coste indicativo&lt;/td>
&lt;td style="text-align:right">~2 K €&lt;/td>
&lt;td style="text-align:right">~250 K € (≈ 125×)&lt;/td>
&lt;/tr>
&lt;/tbody>
&lt;/table>
&lt;p>La asimetría interesante: pasar de A a B cuesta unas 125× más, pero sólo da ~19× más cache total. Lo que el cluster compra de verdad no es proporcionalmente más contexto: es &lt;strong>modelos un orden de magnitud más grandes&lt;/strong> servibles a la vez (8 B → 70 B), &lt;strong>ancho de banda HBM 3× mayor por GPU&lt;/strong> y &lt;strong>concurrencia muy alta sin degradar latencia&lt;/strong>. Si tu carga real es &amp;ldquo;modelos pequeños con mucho contexto y baja concurrencia&amp;rdquo;, la 4090 cuantizada rinde mucho más cerca del cluster de lo que el precio sugiere. Si es &amp;ldquo;modelo grande, baja latencia, muchos usuarios&amp;rdquo;, el salto a HBM3 + NVLink no se sustituye con más 4090s.&lt;/p>
&lt;h3 id="implicaciones-operativas">Implicaciones operativas&lt;/h3>
&lt;p>Tres observaciones para llevarse:&lt;/p>
&lt;p>Primero, &lt;strong>el contexto máximo anunciado por un modelo no es el que puedes servir en tu hardware&lt;/strong>. Llama 3 8B &amp;ldquo;soporta&amp;rdquo; 128 K, pero en una 4090 con 4 sesiones simultáneas tu contexto efectivo son ~8 K. Es trivial comprobarlo antes de comprometerse a un SLA.&lt;/p>
&lt;p>Segundo, &lt;strong>cuantizar el KV cache es de las optimizaciones con mejor relación coste/beneficio&lt;/strong>. No toca los pesos, no afecta a la reproducibilidad de las salidas y duplica capacidad. vLLM lo soporta vía &lt;code>--kv-cache-dtype fp8&lt;/code>.&lt;/p>
&lt;p>Tercero, &lt;strong>si los SLA dictan contextos largos con muchos usuarios concurrentes, GQA es necesario pero no suficiente&lt;/strong>. A medio plazo, hay que mirar modelos con MLA o variantes de attention con compresión.&lt;/p>
&lt;h2 id="lo-que-no-hemos-cubierto-próximos-artículos">Lo que no hemos cubierto (próximos artículos)&lt;/h2>
&lt;ul>
&lt;li>&lt;strong>PagedAttention&lt;/strong> y su implementación en vLLM: bloques, tabla de páginas, evicción.&lt;/li>
&lt;li>&lt;strong>Prefix caching&lt;/strong>: cuando varias peticiones comparten el system prompt, no hace falta recomputar las K, V de la parte común.&lt;/li>
&lt;li>&lt;strong>Speculative decoding&lt;/strong> y su interacción con el cache.&lt;/li>
&lt;li>&lt;strong>Cache offloading&lt;/strong>: mover bloques fríos a RAM o a NVMe, técnica clave para contextos &amp;gt; 1 M.&lt;/li>
&lt;/ul>
&lt;h2 id="referencias">Referencias&lt;/h2>
&lt;ul>
&lt;li>Vaswani et al., &lt;em>Attention Is All You Need&lt;/em> (NeurIPS 2017) — paper fundacional del transformer.&lt;/li>
&lt;li>Ainslie et al., &lt;em>GQA: Training Generalized Multi-Query Transformer Models from Multi-Head Checkpoints&lt;/em> (EMNLP 2023).&lt;/li>
&lt;li>Kwon et al., &lt;em>Efficient Memory Management for Large Language Model Serving with PagedAttention&lt;/em> (SOSP 2023) — paper original de vLLM.&lt;/li>
&lt;li>DeepSeek-AI, &lt;em>DeepSeek-V2 Technical Report&lt;/em> (2024) — introducción de Multi-Head Latent Attention.&lt;/li>
&lt;li>Documentación oficial de vLLM: &lt;a href="https://docs.vllm.ai/">https://docs.vllm.ai/&lt;/a>.&lt;/li>
&lt;li>Llama 3 model card (Meta): especificaciones GQA, n_layers, n_kv_heads.&lt;/li>
&lt;/ul></description></item></channel></rss>