[Attention] FlashAttention ViT support, make default backend (#28763)

Signed-off-by: Matthew Bonanni <mbonanni@redhat.com>
This commit is contained in:
Matthew Bonanni
2025-11-18 23:06:21 -05:00
committed by GitHub
parent 814843e021
commit 4c23690f43
5 changed files with 15 additions and 46 deletions

View File

@@ -38,7 +38,7 @@ else()
FetchContent_Declare( FetchContent_Declare(
vllm-flash-attn vllm-flash-attn
GIT_REPOSITORY https://github.com/vllm-project/flash-attention.git GIT_REPOSITORY https://github.com/vllm-project/flash-attention.git
GIT_TAG 58e0626a692f09241182582659e3bf8f16472659 GIT_TAG 71bb26f6295449be880344b93b51791cc009237d
GIT_PROGRESS TRUE GIT_PROGRESS TRUE
# Don't share the vllm-flash-attn build between build types # Don't share the vllm-flash-attn build between build types
BINARY_DIR ${CMAKE_BINARY_DIR}/vllm-flash-attn BINARY_DIR ${CMAKE_BINARY_DIR}/vllm-flash-attn

View File

@@ -13,14 +13,14 @@ from vllm.vllm_flash_attn import (
) )
NUM_HEADS = [(4, 4), (8, 2)] NUM_HEADS = [(4, 4), (8, 2)]
HEAD_SIZES = [128, 256] HEAD_SIZES = [40, 72, 80, 128, 256]
BLOCK_SIZES = [16] BLOCK_SIZES = [16]
DTYPES = [torch.bfloat16] DTYPES = [torch.bfloat16]
QDTYPES = [None, torch.float8_e4m3fn] QDTYPES = [None, torch.float8_e4m3fn]
# one value large enough to test overflow in index calculation. # one value large enough to test overflow in index calculation.
# one value small enough to test the schema op check # one value small enough to test the schema op check
NUM_BLOCKS = [32768, 2048] NUM_BLOCKS = [32768, 2048]
SOFT_CAPS = [None, 50.0] SOFT_CAPS = [None]
SLIDING_WINDOWS = [None, 256] SLIDING_WINDOWS = [None, 256]

View File

@@ -62,38 +62,10 @@ def test_mha_attn_platform(device: str):
assert attn.attn_backend == AttentionBackendEnum.FLASH_ATTN assert attn.attn_backend == AttentionBackendEnum.FLASH_ATTN
# Test CUDA with head_size=72 (not divisible by 32) # Test CUDA with head_size=72 (not divisible by 32)
# - with upstream FA not available # - should use vLLM's FlashAttention
# - should use xformers
with ( with (
patch("vllm.attention.layer.current_platform", CudaPlatform()), patch("vllm.attention.layer.current_platform", CudaPlatform()),
patch("vllm.model_executor.models.vision.current_platform", CudaPlatform()), patch("vllm.model_executor.models.vision.current_platform", CudaPlatform()),
patch(
"vllm.attention.layer.check_upstream_fa_availability",
return_value=False,
),
):
attn = MultiHeadAttention(16, 72, scale=1)
assert attn.attn_backend == AttentionBackendEnum.XFORMERS
# Test CUDA with head_size=72 (not divisible by 32)
# - with upstream FA available
# - should use upstream FA
with (
patch("vllm.attention.layer.current_platform", CudaPlatform()),
patch("vllm.model_executor.models.vision.current_platform", CudaPlatform()),
patch(
"vllm.attention.layer.check_upstream_fa_availability", return_value=True
),
patch.dict(
"sys.modules",
{
"flash_attn": type(
"MockFlashAttn",
(),
{"flash_attn_varlen_func": lambda *args, **kwargs: None},
)()
},
),
): ):
attn = MultiHeadAttention(16, 72, scale=1) attn = MultiHeadAttention(16, 72, scale=1)
assert attn.attn_backend == AttentionBackendEnum.FLASH_ATTN assert attn.attn_backend == AttentionBackendEnum.FLASH_ATTN

View File

@@ -267,24 +267,21 @@ class CudaPlatformBase(Platform):
) -> "AttentionBackendEnum": ) -> "AttentionBackendEnum":
from vllm.attention.backends.registry import AttentionBackendEnum from vllm.attention.backends.registry import AttentionBackendEnum
# For Blackwell GPUs, force TORCH_SDPA for now. # Try FlashAttention first
# See https://github.com/facebookresearch/xformers/issues/1317#issuecomment-3199392579 # noqa: E501 try:
if cls.has_device_capability(100):
return AttentionBackendEnum.TORCH_SDPA
if dtype not in (torch.float16, torch.bfloat16):
return AttentionBackendEnum.XFORMERS
if cls.has_device_capability(80):
backend_class = AttentionBackendEnum.FLASH_ATTN.get_class() backend_class = AttentionBackendEnum.FLASH_ATTN.get_class()
if backend_class.supports_head_size( if backend_class.supports_head_size(
head_size head_size
) and backend_class.supports_dtype(dtype): ) and backend_class.supports_dtype(dtype):
return AttentionBackendEnum.FLASH_ATTN return AttentionBackendEnum.FLASH_ATTN
else: except ImportError:
return AttentionBackendEnum.XFORMERS pass
if cls.has_device_capability(100):
# xFormers doesn't support Blackwell, fall back to SDPA
# See https://github.com/facebookresearch/xformers/issues/1317#issuecomment-3199392579 # noqa: E501
return AttentionBackendEnum.TORCH_SDPA
else: else:
# Fallback for Volta/Turing GPUs or FA not supported
return AttentionBackendEnum.XFORMERS return AttentionBackendEnum.XFORMERS
@classmethod @classmethod

View File

@@ -119,8 +119,8 @@ class FlashAttentionBackend(AttentionBackend):
raise ValueError(f"Unrecognized FP8 dtype: {kv_cache_dtype}") raise ValueError(f"Unrecognized FP8 dtype: {kv_cache_dtype}")
@classmethod @classmethod
def get_supported_head_sizes(cls) -> list[int]: def supports_head_size(cls, head_size: int) -> bool:
return [32, 64, 96, 128, 160, 192, 224, 256] return head_size % 8 == 0 and head_size <= 256
@classmethod @classmethod
def supports_kv_cache_dtype(cls, kv_cache_dtype: CacheDType | None) -> bool: def supports_kv_cache_dtype(cls, kv_cache_dtype: CacheDType | None) -> bool: