From 832979f85040018b6f101d794f4794a35888bc0c Mon Sep 17 00:00:00 2001 From: "T.J. Alumbaugh" Date: Mon, 16 Dec 2024 12:53:13 -0800 Subject: [PATCH] Build RoPE cos, sin tensors on demand PiperOrigin-RevId: 706805762 --- .../generative/examples/gemma/gemma2.py | 29 +++++---- .../test_models/toy_model_with_kv_cache.py | 6 +- ai_edge_torch/generative/layers/attention.py | 33 ++-------- .../layers/rotary_position_embedding.py | 63 ++++++++++++++++--- .../generative/utilities/model_builder.py | 23 ++++--- 5 files changed, 86 insertions(+), 68 deletions(-) diff --git a/ai_edge_torch/generative/examples/gemma/gemma2.py b/ai_edge_torch/generative/examples/gemma/gemma2.py index a934c835..c921ea57 100644 --- a/ai_edge_torch/generative/examples/gemma/gemma2.py +++ b/ai_edge_torch/generative/examples/gemma/gemma2.py @@ -22,6 +22,7 @@ from ai_edge_torch.generative.layers import kv_cache as kv_utils import ai_edge_torch.generative.layers.attention_utils as attn_utils import ai_edge_torch.generative.layers.model_config as cfg +import ai_edge_torch.generative.layers.rotary_position_embedding as rotary_pos_emb from ai_edge_torch.generative.utilities import model_builder import ai_edge_torch.generative.utilities.loader as loading_utils import torch @@ -103,17 +104,12 @@ def __init__(self, config: cfg.ModelConfig): config.embedding_dim, config.final_norm_config, ) - # Gemma2 has same hyper parameters for each layer except for attention - # types. Use the first layer. - attn_config = config.block_config(0).attn_config - self.rope_cache = attn_utils.build_rope_cache( - size=config.kv_cache_max, - dim=int(attn_config.rotary_percentage * attn_config.head_dim), - base=attn_config.rotary_base, - ) self.mask_cache = attn_utils.build_causal_mask_cache( size=config.kv_cache_max, ) + # Gemma2 has same hyper parameters for each layer except for attention + # types. Use the first layer. + attn_config = config.block_config(0).attn_config self.sliding_window_mask_cache = attn_utils.build_sliding_window_mask_cache( size=config.kv_cache_max, window_size=attn_config.sliding_window_size, @@ -145,24 +141,27 @@ def forward( " must be the same." ) - cos, sin = self.rope_cache - cos = cos.index_select(0, input_pos) - sin = sin.index_select(0, input_pos) + # RoPE parameters are the same for all blocks. Use the first layer. + attn_config = self.config.block_config(0).attn_config + n_elem = int(attn_config.rotary_percentage * attn_config.head_dim) + rope = rotary_pos_emb.build_rope( + input_pos, n_elem, attn_config.head_dim, attn_config.rotary_base + ) # token embeddings of shape (b, t, n_embd) x = self.tok_embedding(tokens) x = x * (self.config.embedding_dim**0.5) - updated_kv_entires = [] + updated_kv_entries = [] for i, block in enumerate(self.transformer_blocks): mask = self.get_attention_mask( block.config.attn_config.attn_type, input_pos ) kv_entry = kv_cache.caches[i] if kv_cache else None - x, kv_entry = block(x, (cos, sin), mask, input_pos, kv_entry) + x, kv_entry = block(x, rope, mask, input_pos, kv_entry) if kv_entry: - updated_kv_entires.append(kv_entry) - updated_kv_cache = kv_utils.KVCache(tuple(updated_kv_entires)) + updated_kv_entries.append(kv_entry) + updated_kv_cache = kv_utils.KVCache(tuple(updated_kv_entries)) if export_config is not None: if ( diff --git a/ai_edge_torch/generative/examples/test_models/toy_model_with_kv_cache.py b/ai_edge_torch/generative/examples/test_models/toy_model_with_kv_cache.py index d4d44845..aeb8431e 100644 --- a/ai_edge_torch/generative/examples/test_models/toy_model_with_kv_cache.py +++ b/ai_edge_torch/generative/examples/test_models/toy_model_with_kv_cache.py @@ -72,14 +72,14 @@ def forward( mask = self.mask_cache.index_select(2, input_pos) mask = mask[:, :, :, : self.config.max_seq_len] - updated_kv_entires = [] + updated_kv_entries = [] for i, block in enumerate(self.transformer_blocks): kv_entry = kv_cache.caches[i] if kv_cache else None x, kv_entry = block(x, (cos, sin), mask, input_pos, kv_entry) if kv_entry: - updated_kv_entires.append(kv_entry) + updated_kv_entries.append(kv_entry) - updated_kv_cache = kv_utils.KVCache(tuple(updated_kv_entires)) + updated_kv_cache = kv_utils.KVCache(tuple(updated_kv_entries)) if export_config is not None: if ( diff --git a/ai_edge_torch/generative/layers/attention.py b/ai_edge_torch/generative/layers/attention.py index f4c43e40..b52e04b4 100644 --- a/ai_edge_torch/generative/layers/attention.py +++ b/ai_edge_torch/generative/layers/attention.py @@ -26,33 +26,6 @@ from torch import nn -def _embed_rope( - q: torch.Tensor, - k: torch.Tensor, - n_elem: int, - rope: Tuple[torch.Tensor, torch.Tensor], -) -> Tuple[torch.Tensor, torch.Tensor]: - """Embed rotary positional embedding for query and key. - - Args: - q (torch.Tensor): query tensor. - k (torch.Tensor): key tensor. - n_elem (int): number of elements to embed rotarty positional embedding. - rope (Tuple[torch.Tensor, torch.Tensor]): the input rope tensor. - """ - if n_elem > 0: - cos, sin = rope - q_roped = rotary_pos_emb.apply_rope( - q[..., :n_elem], cos.repeat(1, 2), sin.repeat(1, 2) - ) - k_roped = rotary_pos_emb.apply_rope( - k[..., :n_elem], cos.repeat(1, 2), sin.repeat(1, 2) - ) - q = torch.cat((q_roped, q[..., n_elem:]), dim=-1) - k = torch.cat((k_roped, k[..., n_elem:]), dim=-1) - return q, k - - class TransformerBlock(nn.Module): def __init__( @@ -238,7 +211,8 @@ def forward( if rope is not None: # Compute rotary positional embedding for query and key. n_elem = int(self.config.rotary_percentage * self.config.head_dim) - q, k = _embed_rope(q, k, n_elem, rope) + cos, sin = rope + q, k = rotary_pos_emb.apply_rope_inline(q, k, cos, sin) if kv_cache is not None: kv_cache = kv_utils.update(kv_cache, input_pos, k, v) @@ -374,7 +348,8 @@ def forward( if rope is not None: # Compute rotary positional embedding for query and key. n_elem = int(self.config.rotary_percentage * self.config.head_dim) - q, k = _embed_rope(q, k, n_elem, rope) + cos, sin = rope + q, k = rotary_pos_emb.apply_rope_inline(q, k, cos, sin) if kv_cache is not None: kv_cache = kv_utils.update(kv_cache, input_pos, k, v) diff --git a/ai_edge_torch/generative/layers/rotary_position_embedding.py b/ai_edge_torch/generative/layers/rotary_position_embedding.py index c06dc818..f9b8fb05 100644 --- a/ai_edge_torch/generative/layers/rotary_position_embedding.py +++ b/ai_edge_torch/generative/layers/rotary_position_embedding.py @@ -39,38 +39,83 @@ def apply_rope( return roped.transpose(1, 2).type_as(x) -def apply_rope_inline( +def _embed_rope( q: torch.Tensor, k: torch.Tensor, + n_elem: int, + rope: Tuple[torch.Tensor, torch.Tensor], +) -> Tuple[torch.Tensor, torch.Tensor]: + """Embed rotary positional embedding for query and key. + + Args: + q (torch.Tensor): query tensor. + k (torch.Tensor): key tensor. + n_elem (int): number of elements to embed rotarty positional embedding. + rope (Tuple[torch.Tensor, torch.Tensor]): the input rope tensor. + """ + if n_elem > 0: + cos, sin = rope + q_roped = rotary_pos_emb.apply_rope( + q[..., :n_elem], cos.repeat(1, 2), sin.repeat(1, 2) + ) + k_roped = rotary_pos_emb.apply_rope( + k[..., :n_elem], cos.repeat(1, 2), sin.repeat(1, 2) + ) + q = torch.cat((q_roped, q[..., n_elem:]), dim=-1) + k = torch.cat((k_roped, k[..., n_elem:]), dim=-1) + return q, k + + +def build_rope( input_pos: torch.Tensor, n_elem: int, + head_dim: int, base: int = 10_000, ) -> Tuple[torch.Tensor, torch.Tensor]: - """Computes rotary positional embedding inline for a query and key. + """Computes rotary positional embedding cosine and sine tensors. Args: - q: the query tensor. - k: the key tensor. input_pos: the sequence indices for the query and key n_elem: number of elements of the head dimension for RoPE computation + base: the base of the exponentiated value for RoPE. Returns: - output the RoPE'd query and key. + cos, sin tensors """ if n_elem <= 0: - return q, k + return None, None theta = 1.0 / (base ** (torch.arange(0, n_elem, 2).float() / n_elem)) freq_exponents = (2.0 / n_elem) * torch.arange( - q.shape[-1] // 2, dtype=torch.float32 + head_dim // 2, dtype=torch.float32 ) timescale = float(base) ** freq_exponents radians = input_pos.clone().unsqueeze(0).unsqueeze(-1) / timescale.unsqueeze( 0 ).unsqueeze(0) - cos = torch.cos(radians).type_as(q) - sin = torch.sin(radians).type_as(q) + cos = torch.cos(radians) + sin = torch.sin(radians) + return cos, sin + + +def apply_rope_inline( + q: torch.Tensor, + k: torch.Tensor, + cos: torch.Tensor, + sin: torch.Tensor, +) -> Tuple[torch.Tensor, torch.Tensor]: + """Computes rotary positional embedding inline for a query and key. + + Args: + q: the query tensor. + k: the key tensor. + cos: the cosine tensor. + sin: the sine tensor. + + Returns: + output the RoPE'd query and key. + """ def apply(x, sin, cos): x = x.transpose(1, 2) diff --git a/ai_edge_torch/generative/utilities/model_builder.py b/ai_edge_torch/generative/utilities/model_builder.py index 2714eeaa..d259a5be 100644 --- a/ai_edge_torch/generative/utilities/model_builder.py +++ b/ai_edge_torch/generative/utilities/model_builder.py @@ -24,6 +24,7 @@ from ai_edge_torch.generative.layers import kv_cache as kv_utils import ai_edge_torch.generative.layers.attention_utils as attn_utils import ai_edge_torch.generative.layers.model_config as cfg +import ai_edge_torch.generative.layers.rotary_position_embedding as rotary_pos_emb import ai_edge_torch.generative.utilities.loader as loading_utils import torch from torch import nn @@ -85,13 +86,6 @@ def __init__(self, config: cfg.ModelConfig): config.embedding_dim, config.final_norm_config, ) - # ROPE parameters for all attn_configs are the same. Take the first one. - attn_config = config.block_config(0).attn_config - self.rope_cache = attn_utils.build_rope_cache( - size=config.kv_cache_max, - dim=int(attn_config.rotary_percentage * attn_config.head_dim), - base=attn_config.rotary_base, - ) self.mask_cache = attn_utils.build_causal_mask_cache( size=config.kv_cache_max, ) @@ -113,11 +107,16 @@ def forward( # token embeddings of shape (b, t, n_embd) input_embeds = self.tok_embedding(tokens) - cos, sin = self.rope_cache - rope = (cos.index_select(0, input_pos), sin.index_select(0, input_pos)) mask = self.mask_cache.index_select(2, input_pos) mask = mask[:, :, :, : self.config.kv_cache_max] + # ROPE parameters for all attn_configs are the same. Take the first one. + attn_config = self.config.block_config(0).attn_config + n_elem = int(attn_config.rotary_percentage * attn_config.head_dim) + rope = rotary_pos_emb.build_rope( + input_pos, n_elem, attn_config.head_dim, attn_config.rotary_base + ) + return self.forward_with_embeds( input_embeds, rope, mask, input_pos, kv_cache, export_config ) @@ -141,13 +140,13 @@ def forward_with_embeds( if self.config.embedding_scale is not None: x = x * self.config.embedding_scale - updated_kv_entires = [] + updated_kv_entries = [] for i, block in enumerate(self.transformer_blocks): kv_entry = kv_cache.caches[i] if kv_cache else None x, kv_entry = block(x, rope, mask, input_pos, kv_entry) if kv_entry: - updated_kv_entires.append(kv_entry) - updated_kv_cache = kv_utils.KVCache(tuple(updated_kv_entires)) + updated_kv_entries.append(kv_entry) + updated_kv_cache = kv_utils.KVCache(tuple(updated_kv_entries)) if export_config is not None: if (