大型说话模型 (LLM) 越来越多地用于需要多个链式生成挪用、高级 prompt 技术、控制流以及与外部环境交互的复杂任务。然而,用于编程和执行这些应用程序的现有高效体系存在着明显的缺陷。
现在,开源社区的钻研者们面向 LLM 提出了一种结构化生成说话(Structured Generation Language)——SGLang。SGLang 能够增强与 LLM 的交互,通过联合设计后端运转时体系和前端说话,使 LLM 更快、更可控。机器学习领域知名学者、CMU 助理教授陈天奇还转发了这项钻研。
总的来说,SGLang 的贡献主要包括:
在后端,钻研团队提出了 RadixAttention,这是一种跨多个 LLM 生成挪用的 KV 缓存(KV cache)复用技术,主动且高效。
在前端,钻研团队开发了一种嵌入 Python 的、灵活的域指定(domain-specific)说话来控制生成过程。该说话可以在解释器模式或编译器模式下执行。
后端前端组件协同工作,可提高复杂 LLM 程序的执行和编程效率。
该钻研应用 SGLang 实现了常见的 LLM 工作负载,包括智能体、推理、提取、对话和小样本学习任务,并在 NVIDIA A10G GPU 上采用 Llama-7B 和 Mixtral-8x7B 模型。如下图 1 、图 2 表明,与现有体系(即 Guidance 和 vLLM)相比,SGLang 的吞吐量提高了 5 倍。
图 1:不同体系在 LLM 任务上的吞吐量(A10G、FP16 上的 Llama-7B、张量并行度 = 1)
图 2:不同体系在 LLM 任务上的吞吐量(A10G、FP16 上的 Mixtral-8x7B,张量并行度 = 8)
后端:应用 RadixAttention 主动 KV 缓存复用
在 SGLang 运转时的开发过程中,该钻研发现了复杂 LLM 程序的优化关键 ——KV 缓存复用,当前体系对此处理不佳。KV 缓存复用意味着具有相同前缀的不同 prompt 可以同享中间 KV 缓存,避免冗余的内存和计算。在涉及多个 LLM 挪用的复杂程序中,可能存在各种 KV 缓存复用模式。下图 3 说明了 LLM 工作负载中常见的四种此类模式。虽然某些体系能够在某些场景下处理 KV 缓存复用,但通常需要手动配置和临时调整。此外,由于可能的复用模式的多样性,即使通过手动配置,现有体系也无法主动适应所有场景。
图 3:KV 缓存同享示例。蓝色框是可同享的 prompt 部分,绿色框是不可同享的部分,黄色框是不可同享的模型输出。可同享的部分包括小样本学习示例、自洽(self-consistency)问题、多轮对话中的对话历史以及思维树(tree-of-thought)中的搜索历史。
为了体系地利用这些复用机会,该钻研提出了一种在运转时主动 KV 缓存复用的新方法 —— RadixAttention。该方法不是在完成生成请求后丢弃 KV 缓存,而是在基数树(radix tree)中保留 prompt 和生成结果的 KV 缓存。这种数据结构可以实现高效的前缀搜索、插入和遣散。该钻研实现了最近最少应用(LRU)遣散策略,并辅以缓存感知调度策略,以提高缓存命中率。
基数树可作为 trie(前缀树)节省空间的替代方案。与典型的树不同,基数树的边缘不仅可以用单个元素来标记,还可以用不同长度的元素序列来标记,这提高了基数树的效率。
该钻研利用基数树来管理映射,这种映射是在充当键的 token 序列和充当值的相应 KV 缓存张量之间进行的。这些 KV 缓存张量以分页布局存储在 GPU 上,其中每个页的大小相当于一个 token。
考虑到 GPU 内存容量有限,无法重新训练无限的 KV 缓存张量,这就需要遣散策略。该钻研采用 LRU 遣散策略,递归地遣散叶节点。此外,RadixAttention 与连续批处理和分页注意力等现有技术兼容。对于多模态模型,RadixAttention 可以轻松扩展以处理图像 token。
下图说明了在处理多个传入请求时如何维护基数树。前端总是向运转时发送完整的 prompt,运转时会主动进行前缀匹配、复用和缓存。树形结构存储在 CPU 上,维护开销较小。
图 4. 采用 LRU 遣散策略的 RadixAttention 操作示例,分九个步骤进行说明。
图 4 演示了基数树响应各种请求的动态演变。这些请求包括两个聊天会话、一批小样本学习查询和自洽性抽样。每个树边缘都带有一个标签,表示子字符串或 token 序列。节点采用颜色编码以反映不同的状态:绿色表示新添加的节点,蓝色表示在该时间点访问的缓存节点,红色表示已被遣散的节点。
前端:应用 SGLang 轻松进行 LLM 编程
在前端,该钻研提出了 SGLang,一种嵌入在 Python 中的特定于领域的说话,允许表达高级 prompt 技术、控制流、多模态、解码约束和外部交互。SGLang 函数可以通过各种后端运转,例如 OpenAI、Anthropic、Gemini 和本地模型。
图 5. 用 SGLang 实现多维文章评分。
图 5 显示了一个具体示例。它利用分支 – 解决 – 合并 prompt 技术实现多维文章评分。该函数应用 LLM 从多个维度评估文章的质量,合并判断,生成摘要,并分配最终等级。突出显示的区域说明了 SGLang API 的应用。(1) fork 创建 prompt 的多个并行副本。(2) gen 挪用 LLM 生成并将结果存储在变量中。该挪用是非阻塞的,因此它允许多个生成挪用在后台同时运转。(3) [variable_name] 检索生成的结果。(4) 选择对生成施加约束。(5) run 应用其参数执行 SGLang 函数。
给定这样一个 SGLang 程序,我们可以通过解释器执行它,也可以将其跟踪为数据流图并应用图执行器运转它。后一种情况为一些潜在的编译器优化开辟了空间,例如代码移动、指令选择和主动调整。
SGLang 的语法很大程度上受到 Guidance 的启发,并引入了新的原语,还处理程序内并行性和批处理。所有这些新功能都有助于 SGLang 的出色性能。
基准测试
钻研团队在常见的 LLM 工作负载上测试了其体系,并报告了所实现的吞吐量。
具体来说,该钻研在 1 个 NVIDIA A10G GPU (24GB) 上测试了 Llama-7B,在 8 个具有张量并行性的 NVIDIA A10G GPU 上应用 FP16 精度测试了 Mixtral-8x7B,并应用 vllm v0.2.5、指导 v0.1.8 和 Hugging Face TGI v1.3.0 作为基准体系。
如图 1 和图 2 所示,SGLang 在所有基准测试中均优于基准体系,吞吐量提高了 5 倍。它在延迟方面也表现出色,特别是对于第一个 token 延迟,其中前缀缓存命中可以带来显著的好处。这些改进归功于 RadixAttention 的主动 KV 缓存复用、解释器实现的程序内并行性以及前端和后端体系的协同设计。此外,消融钻研表明,即使没有缓存命中,也没有明显的开销,这会导致在运转时始终启用 RadixAttention。
参考链接:https://lmsys.org/blog/2024-01-17-sglang/