import math
import time

import torch
import torch.nn as nn
import transformers

from quant import *
from MagR import *


DEBUG = False

torch.backends.cuda.matmul.allow_tf32 = False
torch.backends.cudnn.allow_tf32 = False


class GPTQ:

    def __init__(self, layer):
        self.layer = layer
        self.dev = self.layer.weight.device
        W = layer.weight.data.clone()
        if isinstance(self.layer, nn.Conv2d):
            W = W.flatten(1)
        if isinstance(self.layer, transformers.Conv1D):
            W = W.t()
        self.rows = W.shape[0]
        self.columns = W.shape[1]
        self.H = torch.zeros((self.columns, self.columns), device=self.dev) #H:[col, col]
        self.XtX = torch.zeros((self.columns, self.columns), device=self.dev)
        self.nsamples = 0

    def add_batch(self, inp, out):
        if DEBUG:
            self.inp1 = inp
            self.out1 = out
        if len(inp.shape) == 2:
            inp = inp.unsqueeze(0)
        tmp = inp.shape[0]

        if isinstance(self.layer, nn.Linear) or isinstance(self.layer, transformers.Conv1D):
            if len(inp.shape) == 3:
                inp = inp.reshape((-1, inp.shape[-1]))
            inp = inp.t()
        if isinstance(self.layer, nn.Conv2d):
            unfold = nn.Unfold(
                self.layer.kernel_size,
                dilation=self.layer.dilation,
                padding=self.layer.padding,
                stride=self.layer.stride
            )
            inp = unfold(inp)
            inp = inp.permute([1, 0, 2])
            inp = inp.flatten(1)

        self.X = inp.t().float()
        self.XtX += inp.float().matmul(inp.float().t())
        self.H *= self.nsamples / (self.nsamples + tmp)
        self.nsamples += tmp

        inp = math.sqrt(2 / self.nsamples) * inp.float()

        self.H += inp.matmul(inp.t())

    def fasterquant(
        self, blocksize=128, percdamp=.01, groupsize=-1, actorder=False, static_groups=False, magr=False, CD_iter=1,
    ):
        W = self.layer.weight.data.clone()
        if isinstance(self.layer, nn.Conv2d):
            W = W.flatten(1)
        if isinstance(self.layer, transformers.Conv1D):
            W = W.t()

        W = W.float()
        W_orig = W

        if magr:

            if groupsize != -1:

                print('per group!')

                W = W_proximal_preprocess_groupwise(W, self.X, self.dev, group_size=groupsize)
            else:

                print('per layer!')
                W = W_proximal_preprocess(W, self.X, self.dev)

            self.quantizer.find_params(W, weight=True)

        else:
            pass

        tick = time.time()

        if not self.quantizer.ready():
            self.quantizer.find_params(W, weight=True)

        H = self.H

        dead = torch.diag(H) == 0

        H[dead, dead] = 1

        W[:, dead] = 0


        if static_groups:
            import copy
            groups = []
            for i in range(0, self.columns, groupsize):
                quantizer = copy.deepcopy(self.quantizer)
                quantizer.find_params(W[:, i:(i + groupsize)], weight=True)
                groups.append(quantizer)

        if actorder:
            perm = torch.argsort(torch.diag(H), descending=True)
            W = W[:, perm]
            H = H[perm][:, perm]
            invperm = torch.argsort(perm)

        Losses = torch.zeros_like(W)
        Q = torch.zeros_like(W)

        damp = percdamp * torch.mean(torch.diag(H))

        diag = torch.arange(self.columns, device=self.dev)

        H[diag, diag] += damp
        H = torch.linalg.cholesky(H)
        H = torch.cholesky_inverse(H)
        H = torch.linalg.cholesky(H, upper=True)
        Hinv = H

        for i1 in range(0, self.columns, blocksize):
            i2 = min(i1 + blocksize, self.columns)
            count = i2 - i1

            W1 = W[:, i1:i2].clone()

            Q1 = torch.zeros_like(W1)
            Err1 = torch.zeros_like(W1)
            Losses1 = torch.zeros_like(W1)
            Hinv1 = Hinv[i1:i2, i1:i2]

            for i in range(count):
                w = W1[:, i]
                d = Hinv1[i, i]

                if groupsize != -1:
                    if not static_groups:
                        if (i1 + i) % groupsize == 0:
                            self.quantizer.find_params(W[:, (i1 + i):(i1 + i + groupsize)], weight=True)
                    else:
                        idx = i1 + i
                        if actorder:
                            idx = perm[idx]
                        self.quantizer = groups[idx // groupsize]

                q = quantize(
                    w.unsqueeze(1), self.quantizer.scale, self.quantizer.zero, self.quantizer.maxq
                ).flatten()
                Q1[:, i] = q
                Losses1[:, i] = (w - q) ** 2 / d ** 2

                err1 = (w - q) / d
                W1[:, i:] -= err1.unsqueeze(1).matmul(Hinv1[i, i:].unsqueeze(0))
                Err1[:, i] = err1

            Q[:, i1:i2] = Q1
            Losses[:, i1:i2] = Losses1 / 2

            W[:, i2:] -= Err1.matmul(Hinv[i1:i2, i2:])

            if DEBUG:
                self.layer.weight.data[:, :i2] = Q[:, :i2]
                self.layer.weight.data[:, i2:] = W[:, i2:]
                print(torch.sum((self.layer(self.inp1) - self.out1) ** 2))
                print(torch.sum(Losses))

        torch.cuda.synchronize()
        print('time %.2f' % (time.time() - tick))
        print('error', torch.sum(Losses).item())

        if actorder:
            Q = Q[:, invperm]

        del H, Hinv, W1, Q1, Err1, Losses1, Hinv1


        print('comq', CD_iter)
        Q_CD = Q.reshape(self.layer.weight.shape).clone()

        self.XtX = self.XtX.to(self.dev)
        diag_Sigma = torch.diagonal(self.XtX, 0)
        diag_Sigma += 0.75*torch.mean(torch.diagonal(self.XtX))
        norm_Sigma = torch.div(self.XtX, diag_Sigma+0.1)

        P = torch.matmul(W_orig, norm_Sigma)

        norm_Sigma.fill_diagonal_(0)
        norm_Sigma = norm_Sigma.t()

        for _ in range(CD_iter):

            delta_Q = Q_CD.clone().t()

            P_hat = torch.matmul(norm_Sigma, delta_Q).t()

            for j in range(self.columns):

                u = P[:, j] - P_hat[:, j]

                if j > 0:

                    u += torch.matmul(norm_Sigma[j, :j], delta_Q[:j, :])

                if groupsize != -1:

                    if not static_groups:

                        if j % groupsize == 0:

                            self.quantizer.find_params(Q[:, j:(j + groupsize)], weight=True)

                    else:

                        idx = j

                        if actorder:
                            idx = perm[idx]

                        self.quantizer = groups[idx // groupsize]

                u = quantize(u.unsqueeze(1), self.quantizer.scale, self.quantizer.zero, self.quantizer.maxq).flatten()

                Q_CD[:, j] = u

                delta_Q[j, :] -= u

        if isinstance(self.layer, transformers.Conv1D):
            Q = Q.t()
        self.layer.weight.data = Q_CD.reshape(self.layer.weight.shape).to(self.layer.weight.data.dtype)

        if static_groups:
            self.quantizer.scale = torch.cat([quant.scale for quant in groups], dim=1)
            self.quantizer.zero = torch.cat([quant.zero for quant in groups], dim=1)

        if DEBUG:
            print(torch.sum((self.layer(self.inp1) - self.out1) ** 2))

    def free(self):
        if DEBUG:
            self.inp1 = None
            self.out1 = None
        self.H = None
        self.Losses = None
        self.Trace = None
        torch.cuda.empty_cache()
