{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ac4565f1",
   "metadata": {},
   "outputs": [],
   "source": [
    "import gc\n",
    "import itertools\n",
    "import math\n",
    "import os\n",
    "import random\n",
    "import sys\n",
    "from collections import Counter, defaultdict\n",
    "from copy import deepcopy\n",
    "from dataclasses import dataclass\n",
    "from functools import partial\n",
    "from pathlib import Path\n",
    "from typing import Any, Callable, Literal, TypeAlias\n",
    "\n",
    "import einops\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import torch as t\n",
    "from datasets import load_dataset\n",
    "from IPython.display import clear_output, display\n",
    "from jaxtyping import Float, Int\n",
    "from rich import print as rprint\n",
    "from rich.table import Table\n",
    "\n",
    "from transformer_lens import HookedTransformer, HookedTransformerConfig\n",
    "from tabulate import tabulate\n",
    "from torch import Tensor, nn\n",
    "from torch.nn import functional as F\n",
    "from tqdm.auto import tqdm\n",
    "from transformer_lens import ActivationCache, loading_from_pretrained\n",
    "from transformer_lens.hook_points import HookPoint\n",
    "from transformer_lens.utils import get_act_name, to_numpy\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import torch\n",
    "\n",
    "from scipy.sparse import csr_array\n",
    "from scipy.sparse.csgraph import maximum_bipartite_matching, min_weight_full_bipartite_matching\n",
    "\n",
    "device = t.device(\"mps\" if t.backends.mps.is_available() else \"cuda\" if t.cuda.is_available() else \"cpu\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "46e26310",
   "metadata": {},
   "source": [
    "## Adam vs AdamW - Performance Parity"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ae8d938b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import importlib\n",
    "import pandas as pd\n",
    "import torch\n",
    "\n",
    "# Set architecture (without \"_wd\")\"\n",
    "arch = \"gpt2\"\n",
    "arch_list = [f\"{arch}\", f\"{arch}_wd\"]\n",
    "\n",
    "# Constants\n",
    "chkpt_file = \"final.pt\"\n",
    "shard = 9\n",
    "epoch = 1\n",
    "print(f\"Using checkpoint: {chkpt_file}\")\n",
    "\n",
    "SCRATCH = \"Path to root directory\"\n",
    "\n",
    "SEEDS = [i for i in range(1,51)]\n",
    "if \"gpt2\" in arch:\n",
    "    SEEDS = [i for i in range(1,6)]  # only 5 seeds for gpt2\n",
    "\n",
    "# Validation dataset\n",
    "split = \"test\"\n",
    "dataset_variant = \"wikitext-2-raw-v1\"\n",
    "ds = load_dataset(\"wikitext\", dataset_variant, split=split)\n",
    "\n",
    "def load_named_config(module_name: str, config_name: str) -> dict:\n",
    "    try:\n",
    "        mod = importlib.import_module(module_name)\n",
    "    except Exception as e:\n",
    "        raise ImportError(f\"Could not import config module '{module_name}': {e}\") from e\n",
    "\n",
    "    if not hasattr(mod, \"CONFIGS\"):\n",
    "        raise AttributeError(f\"Module '{module_name}' does not define CONFIGS.\")\n",
    "\n",
    "    CONFIGS = getattr(mod, \"CONFIGS\")\n",
    "    if config_name not in CONFIGS:\n",
    "        available = \", \".join(sorted(CONFIGS.keys()))\n",
    "        raise KeyError(f\"Config '{config_name}' not found in {module_name}. Available: {available}\")\n",
    "\n",
    "    return dict(CONFIGS[config_name])  # copy so we can tweak\n",
    "\n",
    "# (wikitext_perplexity function expected to be defined below in this same cell)\n",
    "from typing import Optional\n",
    "import math\n",
    "import torch\n",
    "import torch.nn.functional as F\n",
    "from datasets import load_dataset\n",
    "from transformer_lens import HookedTransformer\n",
    "\n",
    "def _pick_device(device: Optional[str]) -> str:\n",
    "    return device if device is not None else (\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "\n",
    "@torch.no_grad()\n",
    "def wikitext_perplexity(\n",
    "    model_name: Optional[str] = None,\n",
    "    model: Optional[HookedTransformer] = None,\n",
    "    dataset_variant: str = \"wikitext-2-raw-v1\",\n",
    "    split: str = \"test\",\n",
    "    device: Optional[str] = None,\n",
    "    prepend_bos: bool = True,\n",
    "    block_size: Optional[int] = None,\n",
    "    max_examples: Optional[int] = None,\n",
    "    ds: Optional[object] = None\n",
    ") -> float:\n",
    "    assert (model_name is None) ^ (model is None), \"Provide exactly one of `model_name` or `model`.\"\n",
    "\n",
    "    #ds = load_dataset(\"wikitext\", dataset_variant, split=split)\n",
    "    texts = ds[\"text\"]\n",
    "    if max_examples is not None:\n",
    "        texts = texts[:max_examples]\n",
    "    raw_text = \"\\n\\n\".join(texts)\n",
    "\n",
    "    #if model is None:\n",
    "    #    model = HookedTransformer.from_pretrained(model_name)\n",
    "    model.eval()\n",
    "    device = _pick_device(device)\n",
    "    model.to(device)\n",
    "\n",
    "    toks = model.to_tokens(raw_text, prepend_bos=prepend_bos).squeeze(0).to(\"cpu\")\n",
    "    N = toks.numel()\n",
    "    if N < 2:\n",
    "        raise ValueError(\"Tokenized corpus is too short to compute perplexity.\")\n",
    "\n",
    "    if block_size is None:\n",
    "        block_size = model.cfg.n_ctx\n",
    "    block_size = int(block_size)\n",
    "    assert block_size >= 2, \"block_size (context length) must be at least 2.\"\n",
    "\n",
    "    stride = block_size - 1\n",
    "    total_nll = 0.0\n",
    "    total_tokens = 0\n",
    "\n",
    "    for start in range(0, N - 1, stride):\n",
    "        end = min(start + block_size, N)\n",
    "        window = toks[start:end]\n",
    "        window = window.to(device)\n",
    "        logits = model(window.unsqueeze(0))\n",
    "        targets = window[1:]\n",
    "        pred_logits = logits[0, :-1, :]\n",
    "\n",
    "        nll = F.cross_entropy(pred_logits, targets, reduction=\"sum\")\n",
    "        total_nll += float(nll.item())\n",
    "        total_tokens += targets.numel()\n",
    "\n",
    "    mean_nll = total_nll / max(total_tokens, 1)\n",
    "    ppl = float(math.exp(mean_nll))\n",
    "    return ppl\n",
    "\n",
    "# Run: build/load models for each arch and compute perplexity, storing results in a DataFrame\n",
    "results = []\n",
    "for arch in arch_list:\n",
    "    print(f\"Processing arch: {arch}\")\n",
    "    cfg_dict = load_named_config(\"model_configs\", arch)\n",
    "\n",
    "    # Build HookedTransformerConfig for this arch\n",
    "    cfg = HookedTransformerConfig(\n",
    "        n_layers=cfg_dict[\"n_layers\"],\n",
    "        d_model=cfg_dict[\"d_model\"],\n",
    "        n_heads=cfg_dict[\"n_heads\"],\n",
    "        d_head=cfg_dict[\"d_head\"],\n",
    "        d_mlp=cfg_dict.get(\"d_mlp\", None),\n",
    "        n_ctx=cfg_dict[\"n_ctx\"],\n",
    "        act_fn=cfg_dict.get(\"act_fn\", \"gelu\"),\n",
    "        d_vocab=cfg_dict[\"d_vocab\"],\n",
    "        init_weights=True,\n",
    "        tokenizer_name=cfg_dict[\"tokenizer_name\"],\n",
    "        model_name=cfg_dict.get(\"model_name\", arch),\n",
    "        attn_only=cfg_dict.get(\"attn_only\", False),\n",
    "    )\n",
    "\n",
    "    # Constants\n",
    "    SCRATCH = \"Path to root directory\"\n",
    "    chkpt_dir = SCRATCH + \"chkpts/\" + arch\n",
    "    NUM_LAYERS = cfg.n_layers\n",
    "    NUM_HEADS = cfg.n_heads\n",
    "    ATTN_ONLY = cfg.attn_only\n",
    "    chkpt_file = \"final.pt\"\n",
    "    shard = 9\n",
    "    epoch = 1\n",
    "\n",
    "    # instantiate models (one per seed)\n",
    "    models = []\n",
    "    for SEED in SEEDS:\n",
    "        cfg.seed = SEED\n",
    "        cfg.init_weights = True\n",
    "        model = HookedTransformer(cfg)\n",
    "        models.append(model)\n",
    "\n",
    "    for ind, SEED in enumerate(SEEDS):\n",
    "        if (arch == \"gpt2\") or (arch == \"gpt2_wd\"):\n",
    "            model_state_dict = t.load(chkpt_dir + f\"/gpt2_seed{SEED}_shard{shard}_epoch{epoch}_owt/{chkpt_file}\")\n",
    "            models[ind].load_and_process_state_dict(model_state_dict, fold_ln=False)\n",
    "        else:\n",
    "            if ATTN_ONLY:\n",
    "                model_state_dict = t.load(\n",
    "                    chkpt_dir + f\"/causal_attn_only_l{NUM_LAYERS}_h{NUM_HEADS}_seed{SEED}_epoch{epoch}_c4_gelu/{chkpt_file}\"\n",
    "                )\n",
    "                models[ind].load_and_process_state_dict(model_state_dict, fold_ln=False)\n",
    "\n",
    "            else:\n",
    "                model_state_dict = t.load(\n",
    "                    chkpt_dir + f\"/causal_attn_l{NUM_LAYERS}_h{NUM_HEADS}_seed{SEED}_epoch{epoch}_c4_gelu/{chkpt_file}\"\n",
    "                )\n",
    "                print(f\"Loading model for seed {SEED} from {chkpt_dir}/causal_attn_l{NUM_LAYERS}_h{NUM_HEADS}_seed{SEED}_epoch{epoch}_c4_gelu/{chkpt_file}\")\n",
    "                models[ind].load_and_process_state_dict(model_state_dict, fold_ln=False)\n",
    "\n",
    "        # compute perplexity (uses device selection inside the function)\n",
    "        ppl = wikitext_perplexity(model=models[ind], ds=ds)\n",
    "        results.append({\n",
    "            \"arch\": arch,\n",
    "            \"seed\": SEED,\n",
    "            \"epoch\": epoch,\n",
    "            \"chkpt_file\": chkpt_file,\n",
    "            \"perplexity\": ppl\n",
    "        })\n",
    "        print(f\"  -> seed {SEED} PPL: {ppl:.3f}\")\n",
    "\n",
    "df = pd.DataFrame(results)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "541c80c8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot mean and overlay scatter of individual seed points\n",
    "stats = df.groupby(\"arch\")[\"perplexity\"].agg([\"mean\", \"std\"]).reset_index()\n",
    "x = np.arange(len(stats))\n",
    "means = stats[\"mean\"].values\n",
    "\n",
    "plt.figure(figsize=(4, 4))\n",
    "plt.plot(x, means, 'o', color='C0', markersize=10, label='mean')\n",
    "\n",
    "# scatter individual points with a small horizontal jitter\n",
    "for i, arch in enumerate(stats[\"arch\"]):\n",
    "    ys = df.loc[df[\"arch\"] == arch, \"perplexity\"].values\n",
    "    xs = np.random.normal(loc=i, scale=0.07, size=len(ys))\n",
    "    plt.scatter(xs, ys, color='C1', zorder=1, s=10, label='_nolegend_' if i > 0 else 'refits')\n",
    "\n",
    "plt.xticks(x, [\"Adam\",\"AdamW\"])\n",
    "plt.xlabel(\"Optimizer variant\", fontsize=18)\n",
    "plt.ylabel(\"Perplexity\", fontsize=18)\n",
    "#plt.title(f\"Perplexity for {NUM_LAYERS}-layer {NUM_HEADS}-head {ATTN_ONLY and 'Attention-Only' or 'MLP'} instances trained with Adam vs AdamW\")\n",
    "plt.legend()\n",
    "plt.ylim(bottom=0, top=50)\n",
    "plt.grid(axis='y', alpha=0.3)\n",
    "plt.show()\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "py3.10.4",
   "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.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
