from functools import partial
import torch
import torch.nn as nn
import torch.nn.functional as F
from timm.models.layers import trunc_normal_
from timm.models.registry import register_model
from timm.models.vision_transformer import _cfg
from spikingjelly.clock_driven.neuron import (
    MultiStepLIFNode,
    MultiStepParametricLIFNode,
)
from module import *
from module import mlif

class SpikeDrivenTransformer(nn.Module):
    def __init__(
        self,
        img_size_h=128,
        img_size_w=128,
        patch_size=16,
        in_channels=2,
        num_classes=11,
        embed_dims=512,
        num_heads=8,
        mlp_ratios=4,
        qkv_bias=False,
        qk_scale=None,
        drop_rate=0.0,
        attn_drop_rate=0.0,
        drop_path_rate=0.0,
        norm_layer=nn.LayerNorm,
        depths=[6, 8, 6],
        sr_ratios=[8, 4, 2],
        T=4,
        pooling_stat="1111",
        attn_mode="direct_xor",
        spike_mode="lif",
        mlif_channels=1,
        scaled=False,
        get_embed=False,
        dvs_mode=False,
        TET=False,
        cml=False,
        pretrained=False,
        pretrained_cfg=None,
    ):
        super().__init__()
        self.num_classes = num_classes
        self.depths = depths

        self.T = T
        self.TET = TET
        self.dvs = dvs_mode

        dpr = [
            x.item() for x in torch.linspace(0, drop_path_rate, depths)
        ]  # stochastic depth decay rule

        patch_embed = MS_SPS(
            img_size_h=img_size_h,
            img_size_w=img_size_w,
            patch_size=patch_size,
            in_channels=in_channels,
            embed_dims=embed_dims,
            pooling_stat=pooling_stat,
            spike_mode=spike_mode,
            mlif_channels=mlif_channels,
            scaled=scaled,
        )

        blocks = nn.ModuleList(
            [
                MS_Block_Conv(
                    dim=embed_dims,
                    num_heads=num_heads,
                    mlp_ratio=mlp_ratios,
                    qkv_bias=qkv_bias,
                    qk_scale=qk_scale,
                    drop=drop_rate,
                    attn_drop=attn_drop_rate,
                    drop_path=dpr[j],
                    norm_layer=norm_layer,
                    sr_ratio=sr_ratios,
                    attn_mode=attn_mode,
                    spike_mode=spike_mode,
                    mlif_channels=mlif_channels,
                    scaled=scaled,
                    dvs=dvs_mode,
                    layer=j,
                )
                for j in range(depths)
            ]
        )

        setattr(self, f"patch_embed", patch_embed)
        setattr(self, f"block", blocks)

        # classification head
        if spike_mode in ["lif", "alif", "blif"]:
            self.head_lif = MultiStepLIFNode(tau=2.0, detach_reset=True, backend="cupy")
        elif spike_mode == "mlif":
            self.head_lif = mlif.MultiStepMLIFNode(tau=2.0, detach_reset=True, channels=mlif_channels, backend="torch", scaled=scaled)
        elif spike_mode == "pmlif":
            self.head_lif = mlif.MultiStepParametricMLIFNode(init_tau=2.0, detach_reset=True, channels=mlif_channels, backend="torch", scaled=scaled)
        elif spike_mode == "fast_mlif":
            # self.head_lif = mlif.FastMultiStepMLIFNode(tau=2.0, detach_reset=True, channels=mlif_channels, backend="torch", scaled=scaled)
            self.head_lif = mlif.FastMultiStepMLIFNode(tau=2.0, detach_reset=True, channels=mlif_channels, backend="cupy", scaled=scaled)
        elif spike_mode == "fast_pmlif":
            self.head_lif = mlif.FastMultiStepParametricMLIFNode(init_tau=2.0, detach_reset=True, channels=mlif_channels, backend="torch", scaled=scaled)
        elif spike_mode == "plif":
            self.head_lif = MultiStepParametricLIFNode(
                init_tau=2.0, detach_reset=True, backend="cupy"
            )
        self.head = (
            nn.Linear(embed_dims, num_classes) if num_classes > 0 else nn.Identity()
        )
        self.apply(self._init_weights)

    def _init_weights(self, m):
        if isinstance(m, nn.Conv2d):
            trunc_normal_(m.weight, std=0.02)
            if m.bias is not None:
                nn.init.constant_(m.bias, 0)
        elif isinstance(m, nn.BatchNorm2d):
            nn.init.constant_(m.bias, 0)
            nn.init.constant_(m.weight, 1.0)

    def forward_features(self, x, hook=None):
        block = getattr(self, f"block")
        patch_embed = getattr(self, f"patch_embed")
    
        # print("@spikeformer.py: Before patch_embed input shape ", x.shape)
        # print("@spikeformer.py: Before patch_embed hook ", hook)
        x, _, hook = patch_embed(x, hook=hook)
        for blk in block:
            x, _, hook = blk(x, hook=hook)

        x = x.flatten(3).mean(3)
        return x, hook

    def forward(self, x, hook=None):
        # print("@spikeformer.py: Before unsqueeze input shape ", x.shape)
        if len(x.shape) < 5:
            x = (x.unsqueeze(0)).repeat(self.T, 1, 1, 1, 1)
        else:
            x = x.transpose(0, 1).contiguous()
        # print("@spikeformer.py: Before forward_features input shape ", x.shape)
        # print("@spikeformer.py: Before forward_features hook ", hook) 
        x, hook = self.forward_features(x, hook=hook)
        # print("@spikeformer.py: Before head_lif input shape ", x.shape)
        # print("@spikeformer.py: Before head_lif hook ", hook) 
        x = self.head_lif(x)
        if hook is not None:
            hook["head"] = {"tensor" : x.detach()}
        if hook is not None:
            hook["head"]["flops"] = torch.prod(torch.tensor(list(self.head.weight.shape)))
        
        x = self.head(x)
        
        if not self.TET:
            x = x.mean(0)
        return x, hook


@register_model
def sdt(**kwargs):
    model = SpikeDrivenTransformer(
        **kwargs,
    )
    model.default_cfg = _cfg()
    return model
