123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304 |
- import os
- from typing import List, Optional, Tuple, Union
- import torch
- import torch_xla.core.xla_model as xm
- import torch_xla.runtime as xr
- import aphrodite.common.envs as envs
- from aphrodite.common.config import (CacheConfig, DeviceConfig, LoadConfig,
- ModelConfig, ParallelConfig,
- SchedulerConfig)
- from aphrodite.common.sequence import ExecuteModelRequest
- from aphrodite.common.utils import STR_DTYPE_TO_TORCH_DTYPE, get_dtype_size
- from aphrodite.distributed import (ensure_model_parallel_initialized,
- init_distributed_environment)
- from aphrodite.modeling import set_random_seed
- from aphrodite.task_handler.tpu_model_runner import TPUModelRunner
- from aphrodite.task_handler.worker_base import (LocalOrDistributedWorkerBase,
- LoraNotSupportedWorkerBase,
- WorkerInput)
- class TPUWorker(LoraNotSupportedWorkerBase, LocalOrDistributedWorkerBase):
- def __init__(
- self,
- model_config: ModelConfig,
- parallel_config: ParallelConfig,
- scheduler_config: SchedulerConfig,
- device_config: DeviceConfig,
- cache_config: CacheConfig,
- load_config: LoadConfig,
- local_rank: int,
- rank: int,
- distributed_init_method: str,
- is_driver_worker: bool,
- ) -> None:
- self.model_config = model_config
- self.parallel_config = parallel_config
- self.parallel_config.rank = rank
- self.scheduler_config = scheduler_config
- self.device_config = device_config
- self.cache_config = cache_config
- self.load_config = load_config
- self.local_rank = local_rank
- self.rank = rank
- self.distributed_init_method = distributed_init_method
- self.is_driver_worker = is_driver_worker
- assert self.device_config.device_type == "tpu"
- if self.cache_config.cache_dtype == "auto":
- self.cache_dtype = self.model_config.dtype
- else:
- self.cache_dtype = STR_DTYPE_TO_TORCH_DTYPE[
- self.cache_config.cache_dtype]
- self.model_runner: TPUModelRunner = TPUModelRunner(
- model_config,
- parallel_config,
- scheduler_config,
- device_config,
- cache_config,
- load_config,
- is_driver_worker=is_driver_worker)
- def init_device(self) -> None:
- os.environ["PJRT_DEVICE"] = "TPU"
- torch.set_grad_enabled(False)
- torch.set_default_dtype(self.model_config.dtype)
- # NOTE: This is just to initialize the TP group and broadcast
- # the input objects on CPU. The all-reduce and all-gather ops on TPU
- # are invoked by `xm.all_reduce` and `xm.all_gather` which use their
- # own context.
- init_distributed_environment(
- world_size=self.parallel_config.world_size,
- rank=self.rank,
- local_rank=self.local_rank,
- distributed_init_method=self.distributed_init_method,
- backend="gloo",
- )
- ensure_model_parallel_initialized(
- self.parallel_config.tensor_parallel_size,
- self.parallel_config.pipeline_parallel_size)
- # Device initialization should happen after initializing the distributed
- # runtime.
- self.device = xm.xla_device()
- self.device_config.device = self.device
- # Set random seed.
- set_random_seed(self.model_config.seed)
- xm.set_rng_state(self.model_config.seed, self.device)
- # Increase the cache size limit, which is the maximum number of
- # dynamo graphs that can be compiled.
- # NOTE: Usually, we compile 10-15 graphs for prefill and
- # 30-40 graphs for decode. 128 is an arbitrary safe number.
- torch._dynamo.config.cache_size_limit = 128
- # Use persistent cache to avoid XLA recompilation.
- # NOTE: Set per-rank cache path since different ranks
- # can have slightly different XLA graphs.
- APHRODITE_XLA_CACHE_PATH = envs.APHRODITE_XLA_CACHE_PATH
- world_size = self.parallel_config.world_size
- per_rank_path = os.path.join(APHRODITE_XLA_CACHE_PATH,
- f"tp{world_size}_rank{self.rank}")
- xr.initialize_cache(per_rank_path, readonly=False)
- def load_model(self):
- self.model_runner.load_model()
- def determine_num_available_blocks(self) -> Tuple[int, int]:
- num_layers = self.model_config.get_num_layers(self.parallel_config)
- head_size = self.model_config.get_head_size()
- num_kv_heads = self.model_config.get_num_kv_heads(self.parallel_config)
- kv_caches = [(None, None) for _ in range(num_layers)]
- self.model_runner._dummy_run(
- batch_size=1,
- seq_len=self.scheduler_config.max_num_batched_tokens,
- kv_caches=kv_caches,
- is_prompt=True,
- )
- # Synchronize before measuring the memory usage.
- xm.wait_device_ops()
- dtype_btyes = get_dtype_size(self.cache_dtype)
- block_size = self.cache_config.block_size
- block_size_bytes = (dtype_btyes * block_size * num_layers * 2 *
- head_size * num_kv_heads)
- # Calculate the TPU KV cache size based on profiling.
- m = xm.get_memory_info(self.device)
- total_memory_size = m["bytes_limit"]
- usable_memory_size = int(total_memory_size *
- self.cache_config.gpu_memory_utilization)
- profiled = m["bytes_used"] # Weights + intermediate activations.
- tpu_kv_cache_bytes = max(usable_memory_size - profiled, 0)
- num_tpu_blocks = tpu_kv_cache_bytes // block_size_bytes
- num_tpu_blocks = (num_tpu_blocks // 8) * 8 # Round down to 8.
- # Calculate the CPU KV cache size based on the config.
- num_cpu_blocks = int(self.cache_config.swap_space_bytes //
- block_size_bytes)
- num_cpu_blocks = (num_cpu_blocks // 8) * 8 # Round down to 8.
- # reset and discard the guard and compiled bytecode for profiling runs
- torch._dynamo.reset()
- return num_tpu_blocks, num_cpu_blocks
- def initialize_cache(
- self,
- num_gpu_blocks: int,
- num_cpu_blocks: int,
- ) -> None:
- self.cache_config.num_gpu_blocks = num_gpu_blocks
- self.cache_config.num_cpu_blocks = num_cpu_blocks
- self.block_size = self.cache_config.block_size
- dtype = self.cache_dtype
- num_layers = self.model_config.get_num_layers(self.parallel_config)
- num_kv_heads = self.model_config.get_num_kv_heads(self.parallel_config)
- head_size = self.model_config.get_head_size()
- self.cpu_cache: List[Tuple[torch.Tensor, torch.Tensor]] = []
- self.tpu_cache: List[Tuple[torch.Tensor, torch.Tensor]] = []
- tpu_cache_shape = self.model_runner.attn_backend.get_kv_cache_shape(
- num_gpu_blocks, self.block_size, num_kv_heads, head_size)
- cpu_cache_shape = self.model_runner.attn_backend.get_kv_cache_shape(
- num_cpu_blocks, self.block_size, num_kv_heads, head_size)
- for _ in range(num_layers):
- tpu_k_cache = torch.zeros(tpu_cache_shape,
- dtype=dtype,
- device=self.device)
- tpu_v_cache = torch.zeros_like(tpu_k_cache)
- self.tpu_cache.append((tpu_k_cache, tpu_v_cache))
- cpu_k_cache = torch.zeros(cpu_cache_shape,
- dtype=dtype,
- device="cpu")
- cpu_v_cache = torch.zeros_like(cpu_k_cache)
- self.cpu_cache.append((cpu_k_cache, cpu_v_cache))
- self._warmup_model()
- def _warmup_model(self) -> None:
- # FIXME: Here we are abusing `enforce_eager` which is defined
- # for CUDA graphs. We should refactor this part.
- if not self.model_config.enforce_eager:
- # Warm up the model with all possible input shapes so that
- # compilation never happens during the actual execution.
- # This may take ~30 mins for the first run and ~20 mins for the
- # subsequent runs.
- # If `enforce_eager` is True, the ahead-of-time compilation is
- # skipped and the compilation happens during the actual execution,
- # which is bad for performance but useful for development.
- self.model_runner.warmup_model(self.tpu_cache)
- def get_cache_block_size_bytes(self) -> int:
- head_size = self.model_config.get_head_size()
- num_heads = self.model_config.get_num_kv_heads(self.parallel_config)
- num_layers = self.model_config.get_num_layers(self.parallel_config)
- key_cache_block = self.cache_config.block_size * num_heads * head_size
- value_cache_block = key_cache_block
- total = num_layers * (key_cache_block + value_cache_block)
- dtype_size = get_dtype_size(self.cache_dtype)
- return dtype_size * total
- @property
- def do_metadata_broadcast(self) -> bool:
- return self.parallel_config.tensor_parallel_size > 1
- @property
- def kv_cache(self) -> Optional[List[List[torch.Tensor]]]:
- # NOTE: This assumes virtual_engine == 0, i.e., no pipeline
- # parallelism.
- return [self.tpu_cache]
- def prepare_worker_input(
- self,
- execute_model_req: ExecuteModelRequest,
- ) -> WorkerInput:
- virtual_engine = execute_model_req.virtual_engine
- num_seq_groups = len(execute_model_req.seq_group_metadata_list)
- blocks_to_swap_in = _make_src_to_dst(
- execute_model_req.blocks_to_swap_in, "cpu", self.device)
- blocks_to_swap_out = _make_src_to_dst(
- execute_model_req.blocks_to_swap_out, self.device, "cpu")
- blocks_to_copy = _make_src_to_dst(execute_model_req.blocks_to_copy,
- self.device, self.device)
- return WorkerInput(
- num_seq_groups=num_seq_groups,
- blocks_to_swap_in=blocks_to_swap_in,
- blocks_to_swap_out=blocks_to_swap_out,
- blocks_to_copy=blocks_to_copy,
- virtual_engine=virtual_engine,
- )
- def execute_worker(self, worker_input: WorkerInput) -> None:
- virtual_engine = worker_input.virtual_engine
- assert virtual_engine == 0
- attn_backend = self.model_runner.attn_backend
- num_layers = self.model_config.get_num_layers(self.parallel_config)
- # Issue cache operations.
- if worker_input.blocks_to_swap_in is not None:
- src_indices, dst_indices = worker_input.blocks_to_swap_in
- if src_indices.numel() > 0:
- # Swap from CPU to TPU.
- for i in range(num_layers):
- tpu_k_cache, tpu_v_cache = self.tpu_cache[i]
- cpu_k_cache, cpu_v_cache = self.cpu_cache[i]
- k = cpu_k_cache[:, src_indices].to(self.device)
- v = cpu_v_cache[:, src_indices].to(self.device)
- _insert_kv(k, v, dst_indices, tpu_k_cache, tpu_v_cache)
- if worker_input.blocks_to_swap_out is not None:
- src_indices, dst_indices = worker_input.blocks_to_swap_out
- if src_indices.numel() > 0:
- # Swap from TPU to CPU.
- for i in range(num_layers):
- tpu_k_cache, tpu_v_cache = self.tpu_cache[i]
- cpu_k_cache, cpu_v_cache = self.cpu_cache[i]
- cpu_k_cache[:, dst_indices] = tpu_k_cache[:, src_indices]
- cpu_v_cache[:, dst_indices] = tpu_v_cache[:, src_indices]
- if worker_input.blocks_to_copy is not None:
- src_indices, dst_indices = worker_input.blocks_to_copy
- if src_indices.numel() > 0:
- attn_backend.copy_blocks(self.tpu_cache,
- (src_indices, dst_indices))
- def _make_src_to_dst(
- mapping: List[Tuple[int, int]],
- src_device: Union[torch.device, str],
- dst_device: Union[torch.device, str],
- ) -> Optional[Tuple[torch.Tensor, torch.Tensor]]:
- if not mapping:
- return None
- src_indices = [i for i, _ in mapping]
- dst_indices = [i for _, i in mapping]
- src_indices = torch.tensor(src_indices,
- device=src_device,
- dtype=torch.int64)
- dst_indices = torch.tensor(dst_indices,
- device=dst_device,
- dtype=torch.int64)
- return src_indices, dst_indices
- @torch.compile(backend="openxla")
- def _insert_kv(
- k: torch.Tensor,
- v: torch.Tensor,
- indices: torch.Tensor,
- tpu_k_cache: torch.Tensor,
- tpu_v_cache: torch.Tensor,
- ) -> None:
- torch.ops.xla.dynamo_set_buffer_donor_(tpu_k_cache, True)
- torch.ops.xla.dynamo_set_buffer_donor_(tpu_v_cache, True)
- tpu_k_cache[:, indices] = k
- tpu_v_cache[:, indices] = v
|