import math
import types
from typing import Optional, Tuple

import torch
import torch.nn as nn
from transformers.models.llama.modeling_llama import apply_rotary_pos_emb

def llama_new_forward(
    self,
    hidden_states: torch.Tensor,
    attention_mask: Optional[torch.Tensor] = None,
    position_ids: Optional[torch.LongTensor] = None,
    past_key_value: Optional[Tuple[torch.Tensor]] = None,
    output_attentions: bool = False,
    use_cache: bool = False,
    **kwargs,
) -> Tuple[torch.Tensor, Optional[torch.Tensor], Optional[Tuple[torch.Tensor]]]:
    bsz, q_len, hidden_size = hidden_states.size()

    # Get number of heads and head dimension (compatible with most implementations)
    num_heads = getattr(self, "num_attention_heads", None)
    if num_heads is None:
        num_heads = getattr(self.config, "num_attention_heads", None)
    head_dim = getattr(self, "head_dim", None)
    if head_dim is None:
        head_dim = getattr(self.config, "head_dim", None)
    assert num_heads is not None and head_dim is not None, "Cannot determine attention head parameters."

    # Project query/key/value
    query_states = (
        self.q_proj(hidden_states)
        .view(bsz, q_len, num_heads, head_dim)
        .transpose(1, 2)
    )
    key_states = (
        self.k_proj(hidden_states)
        .view(bsz, q_len, num_heads, head_dim)
        .transpose(1, 2)
    )
    value_states = (
        self.v_proj(hidden_states)
        .view(bsz, q_len, num_heads, head_dim)
        .transpose(1, 2)
    )

    kv_seq_len = key_states.shape[-2]
    if past_key_value is not None:
        if getattr(self, "layer_idx", None) is None:
            raise ValueError(
                f"The cache structure has changed since version v4.36. If you are using {self.__class__.__name__} "
                "for auto-regressive decoding with k/v caching, please make sure to initialize the attention class "
                "with a layer index."
            )
        kv_seq_len += past_key_value.get_usable_length(kv_seq_len, self.layer_idx)

    cos, sin = self.rotary_emb(value_states, position_ids)

    query_states, key_states = apply_rotary_pos_emb(
        query_states, key_states, cos, sin, position_ids
    )

    if past_key_value is not None:
        cache_kwargs = {"sin": sin, "cos": cos}  # Specific to RoPE models
        key_states, value_states = past_key_value.update(
            key_states, value_states, self.layer_idx, cache_kwargs
        )

    attn_weights = torch.matmul(query_states, key_states.transpose(2, 3)) / math.sqrt(head_dim)

    if attn_weights.size() != (bsz, num_heads, q_len, kv_seq_len):
        raise ValueError(
            f"Attention weights should be of size {(bsz, num_heads, q_len, kv_seq_len)}, but got"
            f" {attn_weights.size()}"
        )

    if attention_mask is not None:
        if attention_mask.size() != (bsz, 1, q_len, kv_seq_len):
            raise ValueError(
                f"Attention mask should be of size {(bsz, 1, q_len, kv_seq_len)}, but got {attention_mask.size()}"
            )
        attn_weights = attn_weights + attention_mask
        attn_weights = torch.max(
            attn_weights, torch.tensor(torch.finfo(attn_weights.dtype).min, device=attn_weights.device)
        )

    # === Optional: vision attention modification (custom logic) ===
    if hasattr(self, "aggregation"):
        img_start_idx = getattr(self, "img_start_idx", None)
        img_end_idx = getattr(self, "img_end_idx", None)
        aggregation = self.aggregation
        alpha = getattr(self, "alpha", 0.5)
        if aggregation == "mean" and img_start_idx is not None and img_end_idx is not None:
            attn_weights[:, :, -1, img_start_idx:img_end_idx] = (
                attn_weights[:, :, -1, img_start_idx:img_end_idx]
                + alpha * attn_weights[:, :, -1, img_start_idx:img_end_idx].abs().mean(dim=1, keepdim=True)
            )
    # === vision attention modification END ===

    attn_weights = nn.functional.softmax(attn_weights, dim=-1, dtype=torch.float32).to(query_states.dtype)
    attn_output = torch.matmul(attn_weights, value_states)
    if attn_output.size() != (bsz, num_heads, q_len, head_dim):
        raise ValueError(
            f"`attn_output` should be of size {(bsz, num_heads, q_len, head_dim)}, but got"
            f" {attn_output.size()}"
        )

    attn_output = attn_output.transpose(1, 2).reshape(bsz, q_len, num_heads * head_dim)
    attn_output = self.o_proj(attn_output)

    if output_attentions:
        return attn_output, attn_weights

    return attn_output, None

def llama_head_guide(
    lm_model,                # Pass in llama_for_conditional.model.language_model here
    guided_layer_range,
    aggregation,
    alpha,
    img_start_idx,
    img_end_idx
):
    """
    Patch the attention layers in the specified range to use custom logic for vision attention modification.
    """
    # Build list of layers to patch
    if len(guided_layer_range) == 1:
        layer_list = guided_layer_range
    else:
        layer_list = list(range(guided_layer_range[0], guided_layer_range[1]))

    for i in layer_list:
        # Note: lm_model is LlamaModel, and layers are stored in lm_model.layers
        attn = lm_model.layers[i].self_attn
        attn.img_start_idx = img_start_idx
        attn.img_end_idx   = img_end_idx
        attn.aggregation   = aggregation
        attn.alpha         = alpha
        attn.forward = types.MethodType(llama_new_forward, attn)
