{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "d233a2f9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "env: CUDA_VISIBLE_DEVICES=8\n"
     ]
    }
   ],
   "source": [
    "%env CUDA_VISIBLE_DEVICES=8\n",
    "\n",
    "import sys\n",
    "sys.path.append(\"../src/\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "be028e19",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "\n",
    "torch.backends.fp32_precision = \"tf32\"\n",
    "torch.backends.cuda.matmul.allow_tf32 = True\n",
    "torch.backends.cudnn.allow_tf32 = True\n",
    "\n",
    "\n",
    "from transformers import AutoModelForCausalLM, AutoTokenizer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "94391451",
   "metadata": {},
   "outputs": [],
   "source": [
    "model = AutoModelForCausalLM.from_pretrained(\n",
    "    \"meta-llama/Llama-3.2-1B\",\n",
    "    device_map=\"cuda\",\n",
    "    torch_dtype=torch.float32,\n",
    ")\n",
    "\n",
    "tokenizer = AutoTokenizer.from_pretrained(\"meta-llama/Llama-3.2-1B\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "cd6b03a4",
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch import nn\n",
    "\n",
    "def find_linear_layers(module: nn.Module, name='') -> dict[str, nn.Module]:\n",
    "    \"\"\"\n",
    "    Returns a dict of all nn.Linear submodules in a module with paths as keys\n",
    "    \"\"\"\n",
    "    if type(module) == nn.Linear:\n",
    "        return {name: module}\n",
    "    res = {}\n",
    "    for name1, child in module.named_children():\n",
    "        res.update(find_linear_layers(\n",
    "            child, name=name + '.' + name1 if name != '' else name1\n",
    "        ))\n",
    "    return res\n",
    "\n",
    "\n",
    "def replace_submodule(module, submodule_path, new_submodule):\n",
    "    \"\"\"\n",
    "    Replaces a submodule specified by a path with a new submodule\n",
    "    \"\"\"\n",
    "    submodule_names = submodule_path.split(\".\")\n",
    "    for submodule in submodule_names[:-1]:\n",
    "        module = getattr(module, submodule)\n",
    "    setattr(module, submodule_names[-1], new_submodule)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "c6e18295",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a418b09222c24ed588b6a33a874ab778",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "Replacing nn.Linear submodules:   0%|          | 0/112 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from tqdm.auto import tqdm, trange\n",
    "\n",
    "from models.quantization.schemes import Quartet_II_Linear, TetraJetV2Linear, NvidiaLinear\n",
    "\n",
    "linear_submodules = find_linear_layers(model.model)\n",
    "for name, linear in tqdm(linear_submodules.items(), desc=\"Replacing nn.Linear submodules\"):\n",
    "    qlinear = NvidiaLinear(\n",
    "        linear.in_features, linear.out_features,\n",
    "        device=linear.weight.device, dtype=linear.weight.dtype,\n",
    "        bias=linear.bias is not None,\n",
    "        disable_forward_quant=True,\n",
    "    )\n",
    "    \n",
    "    # qlinear = Quartet_II_Linear(\n",
    "    #     linear.in_features, linear.out_features,\n",
    "    #     device=linear.weight.device, dtype=linear.weight.dtype,\n",
    "    #     bias=linear.bias is not None,\n",
    "    #     disable_forward_quant=True,\n",
    "    #     four_over_six=True, hadamard_dim=128,\n",
    "    # )\n",
    "    \n",
    "    # qlinear = TetraJetV2Linear(\n",
    "    #     linear.in_features, linear.out_features,\n",
    "    #     device=linear.weight.device, dtype=linear.weight.dtype,\n",
    "    #     bias=linear.bias is not None,\n",
    "    #     disable_forward_quant=True,\n",
    "    # )\n",
    "    \n",
    "    qlinear.weight = linear.weight\n",
    "    replace_submodule(model.model, name, qlinear)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "02c90880",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "LlamaForCausalLM(\n",
       "  (model): LlamaModel(\n",
       "    (embed_tokens): Embedding(128256, 2048)\n",
       "    (layers): ModuleList(\n",
       "      (0-15): 16 x LlamaDecoderLayer(\n",
       "        (self_attn): LlamaAttention(\n",
       "          (q_proj): NvidiaLinear(in_features=2048, out_features=2048, bias=False)\n",
       "          (k_proj): NvidiaLinear(in_features=2048, out_features=512, bias=False)\n",
       "          (v_proj): NvidiaLinear(in_features=2048, out_features=512, bias=False)\n",
       "          (o_proj): NvidiaLinear(in_features=2048, out_features=2048, bias=False)\n",
       "        )\n",
       "        (mlp): LlamaMLP(\n",
       "          (gate_proj): NvidiaLinear(in_features=2048, out_features=8192, bias=False)\n",
       "          (up_proj): NvidiaLinear(in_features=2048, out_features=8192, bias=False)\n",
       "          (down_proj): NvidiaLinear(in_features=8192, out_features=2048, bias=False)\n",
       "          (act_fn): SiLU()\n",
       "        )\n",
       "        (input_layernorm): LlamaRMSNorm((2048,), eps=1e-05)\n",
       "        (post_attention_layernorm): LlamaRMSNorm((2048,), eps=1e-05)\n",
       "      )\n",
       "    )\n",
       "    (norm): LlamaRMSNorm((2048,), eps=1e-05)\n",
       "    (rotary_emb): LlamaRotaryEmbedding()\n",
       "  )\n",
       "  (lm_head): Linear(in_features=2048, out_features=128256, bias=False)\n",
       ")"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ab370488",
   "metadata": {},
   "outputs": [],
   "source": [
    "for name, p in model.named_parameters():\n",
    "    p.requires_grad_(True)\n",
    "\n",
    "for p in model.parameters():\n",
    "    p.grad = None\n",
    "\n",
    "for module in model.modules():\n",
    "    if isinstance(module, nn.Linear):\n",
    "        module.disable_backward_quant = True\n",
    "\n",
    "def get_loss(model):\n",
    "    return -nn.functional.log_softmax(model(\n",
    "        **tokenizer(\n",
    "            \"\"\"Lorem ipsum dolor sit amet, consectetur adipiscing elit. Aliquam fringilla neque eget efficitur bibendum. Proin vitae interdum dolor. Phasellus porttitor augue eu ullamcorper finibus. Donec molestie velit nec dolor gravida dignissim. Suspendisse laoreet erat quis dui pretium lacinia. Mauris interdum est eget tellus ullamcorper, vitae pellentesque lacus accumsan. Donec euismod quam nisl, ac euismod lorem rutrum a. Aenean eget porta nunc. Es\"\"\",\n",
    "            return_tensors='pt',\n",
    "        ).to(model.device),\n",
    "        use_cache=False,\n",
    "    ).logits, dim=-1)[0,-1, 325]\n",
    "\n",
    "get_loss(model).backward()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "35bc5612",
   "metadata": {},
   "outputs": [],
   "source": [
    "ref_grads = {}\n",
    "\n",
    "for name, p in model.named_parameters():\n",
    "    if p.requires_grad:\n",
    "        if p.grad is None:\n",
    "            raise Exception(f\"Parameter {name} has no grad!\")\n",
    "        ref_grads[name] = p.grad.clone().detach()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "348ffe15",
   "metadata": {},
   "outputs": [],
   "source": [
    "for module in model.modules():\n",
    "    if isinstance(module, nn.Linear):\n",
    "        module.disable_backward_quant = False\n",
    "\n",
    "grad_err = {k: {} for k in ref_grads}\n",
    "\n",
    "for acc_steps in tqdm([1, 4, 16, 64, 256, 1024]):\n",
    "    for p in model.parameters():\n",
    "        p.grad = None\n",
    "    \n",
    "    for step in trange(acc_steps, leave=False):\n",
    "        get_loss(model).backward()\n",
    "    \n",
    "    for name, p in model.named_parameters():\n",
    "        if p.requires_grad:\n",
    "            if p.grad is None:\n",
    "                raise Exception(f\"Parameter {name} has no grad!\")\n",
    "            p.grad /= acc_steps\n",
    "            \n",
    "            ref = ref_grads[name]\n",
    "            grad = p.grad\n",
    "            \n",
    "            quad_err = (grad - ref).pow(2).mean() / ref.pow(2).mean()\n",
    "            eff_bitwidth = (-torch.log2(quad_err) / 2).item()\n",
    "            grad_err[name][acc_steps] = quad_err.item()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "780a7b30",
   "metadata": {},
   "outputs": [],
   "source": [
    "import json\n",
    "\n",
    "with open(\"../data/46_concentration.json\", \"w\") as f:\n",
    "    json.dump(grad_err, f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "b75a8841",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0, 0.5, '$\\\\|\\\\frac{1}{N}\\\\sum\\\\widehat{G} - G\\\\|_2^2$')"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "key_color = {\n",
    "    \"model.layers.0.self_attn.q_proj.weight\": \"red\",\n",
    "    \"model.layers.0.self_attn.k_proj.weight\": \"orange\",\n",
    "    \"model.layers.0.self_attn.v_proj.weight\": \"green\",\n",
    "}\n",
    "\n",
    "for key, color in key_color.items():\n",
    "    data = grad_err[key]\n",
    "    \n",
    "    plt.scatter(data.keys(), data.values(), label=key, color=color)\n",
    "\n",
    "\n",
    "x = np.logspace(0, 5, num=50, base=4)\n",
    "y = 0.8 / x\n",
    "\n",
    "plt.plot(x, y, color=\"black\", linestyle=\"--\", label=\"1/N\")\n",
    "\n",
    "    \n",
    "plt.xscale('log')\n",
    "plt.yscale('log')\n",
    "\n",
    "plt.xlabel(\"N\")\n",
    "\n",
    "plt.ylabel(r\"$\\|\\frac{1}{N}\\sum\\widehat{G} - G\\|_2^2$\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "771743d7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([8.00000000e-01, 6.94472885e-01, 6.02865736e-01, 5.23342383e-01,\n",
       "       4.54308869e-01, 3.94381489e-01, 3.42359063e-01, 2.97198858e-01,\n",
       "       2.57995685e-01, 2.23963760e-01, 1.94420948e-01, 1.68775096e-01,\n",
       "       1.46512160e-01, 1.27185903e-01, 1.10408951e-01, 9.58450288e-02,\n",
       "       8.32022171e-02, 7.22271047e-02, 6.26997073e-02, 5.44290583e-02,\n",
       "       4.72493814e-02, 4.10167678e-02, 3.56062914e-02, 3.09095049e-02,\n",
       "       2.68322663e-02, 2.32928517e-02, 2.02203174e-02, 1.75530777e-02,\n",
       "       1.52376707e-02, 1.32276864e-02, 1.14828369e-02, 9.96814861e-03,\n",
       "       8.65326116e-03, 7.51181906e-03, 6.52094332e-03, 5.66077290e-03,\n",
       "       4.91406661e-03, 4.26585752e-03, 3.70315298e-03, 3.21467417e-03,\n",
       "       2.79063006e-03, 2.42252113e-03, 2.10296905e-03, 1.82556873e-03,\n",
       "       1.58475998e-03, 1.37571604e-03, 1.19424686e-03, 1.03671508e-03,\n",
       "       8.99963142e-04, 7.81250000e-04])"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d82114ab",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
