{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from math import sqrt\n",
    "from random import seed\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import torch as th\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "\n",
    "th.manual_seed(0)\n",
    "seed(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_embeddings(n, d, norm=True):\n",
    "    emb = th.randn(n, d)\n",
    "    if norm:\n",
    "        emb /= emb.norm(dim=1, keepdim=True)\n",
    "    else:\n",
    "        emb /= sqrt(d)\n",
    "    return emb\n",
    "\n",
    "\n",
    "class AssMem(nn.Module):\n",
    "    def __init__(self, E, U):\n",
    "        \"\"\"\n",
    "        E: torch.Tensor\n",
    "            Input embedding matrix of size $n \\times d$,\n",
    "            where $n$ is the number of tokens and $d$ is the embedding dimension.\n",
    "        U: torch.Tensor\n",
    "            Output unembedding matrix of size $d \\times m$,\n",
    "            where $m$ is the number of classes and $d$ is the embedding dimension.\n",
    "        \"\"\"\n",
    "        super().__init__()\n",
    "        # self.W = nn.Parameter(th.randn(d, d))\n",
    "        d = E.shape[1]\n",
    "        self.W = nn.Parameter(th.zeros(d, d))\n",
    "        self.E = E\n",
    "        self.U = U\n",
    "\n",
    "    def forward(self, x):\n",
    "        # TODO: add noise on the embeddings at inference time\n",
    "        out = self.E[x] @ self.W\n",
    "        out = out @ self.U\n",
    "        # out = F.softmax(out, dim=-1)\n",
    "        return out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# number of input tokens\n",
    "n = 10\n",
    "# number of output classes\n",
    "m = 5\n",
    "# memory dimension\n",
    "d = 5\n",
    "\n",
    "alpha = 1.5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "all_x = th.arange(n)\n",
    "proba = (all_x + 1.) ** (-alpha)\n",
    "proba /= proba.sum()\n",
    "all_y = all_x % m"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "# number of data\n",
    "batch_size = 1\n",
    "nb_epoch = 1000\n",
    "T = nb_epoch * batch_size\n",
    "lr = 1e-1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0,100,200,300,400,500,600,700,800,900,"
     ]
    }
   ],
   "source": [
    "# Embeddings\n",
    "E = get_embeddings(n, d, norm=False)\n",
    "U = get_embeddings(m, d, norm=True).T \n",
    "\n",
    "lrs = [1e1, 1e-1, 1e1]\n",
    "num_models = len(lrs)\n",
    "\n",
    "# models\n",
    "assoc = []\n",
    "opti = []\n",
    "for i in range(num_models):\n",
    "    assoc.append(AssMem(E, U))\n",
    "    lr = lrs[i]\n",
    "    opti.append(th.optim.SGD(assoc[-1].parameters(), lr=lr, momentum=0))\n",
    "scheduler = th.optim.lr_scheduler.StepLR(opti[-1], nb_epoch // 10, gamma=0.1)\n",
    "\n",
    "train_loss = [[] for _ in range(num_models)]\n",
    "test_loss = [[] for _ in range(num_models)]\n",
    "\n",
    "for i in range(nb_epoch):\n",
    "    x = th.multinomial(proba, batch_size, replacement=True)\n",
    "    y = x % m\n",
    "\n",
    "    for j in range(num_models):\n",
    "        out = assoc[j](x)\n",
    "        loss = F.cross_entropy(out, y)\n",
    "        train_loss[j].append(loss.item())\n",
    "\n",
    "        with th.no_grad():\n",
    "            pred = assoc[j](all_x).argmax(dim=-1)\n",
    "            test_loss[j].append(proba[pred != all_y].sum().item())\n",
    "\n",
    "        opti[j].zero_grad()\n",
    "        loss.backward()\n",
    "        opti[j].step()\n",
    "\n",
    "    scheduler.step()\n",
    "\n",
    "    if i % 100 == 0:\n",
    "        print(i, end=',')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x150 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "plt.rc('font', family='serif')\n",
    "plt.rc('text', usetex=True)\n",
    "\n",
    "fig, ax = plt.subplots(1, 1, figsize=(2, 1.5))\n",
    "leg = []\n",
    "for i in range(num_models):\n",
    "    # ax.plot(np.arange(nb_epoch)[10:], train_loss[i][10:], color='C' + str(i), alpha=.1, linestyle='--')\n",
    "    a, = ax.plot(np.arange(nb_epoch), test_loss[i], color= 'C' + str(i))\n",
    "    leg.append(a)\n",
    "ax.set_xscale('log')\n",
    "ax.set_yscale('log')\n",
    "# ax.set_ylim(3e-2, 1e0)\n",
    "ax.set_xlabel('T', fontsize=10)\n",
    "ax.set_ylabel('Error', fontsize=10)\n",
    "ax.tick_params(axis='both', which='major', labelsize=8)\n",
    "ax.legend(leg, [r\"$\\gamma=10$\", r\"$\\gamma=.1$\", r\"StepLR\"], fontsize=7)\n",
    "fig.savefig('sgd_step_lr.pdf', bbox_inches='tight')"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Averaging over several trials"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61,62,63,64,65,66,67,68,69,70,71,72,73,74,75,76,77,78,79,80,81,82,83,84,85,86,87,88,89,90,91,92,93,94,95,96,97,98,99,"
     ]
    }
   ],
   "source": [
    "lrs = [1e1, 1e-1, 1e1]\n",
    "num_models = len(lrs)\n",
    "\n",
    "nb_trials = 100\n",
    "\n",
    "train_loss = th.zeros(num_models, nb_trials, nb_epoch)\n",
    "test_loss = th.zeros(num_models, nb_trials, nb_epoch)\n",
    "train_loss[:] = -1\n",
    "test_loss[:] = -1\n",
    "\n",
    "for k in range(nb_trials):\n",
    "    # Embeddings\n",
    "    E = get_embeddings(n, d, norm=False)\n",
    "    U = get_embeddings(m, d, norm=True).T \n",
    "\n",
    "    # models\n",
    "    assoc = []\n",
    "    opti = []\n",
    "    for i in range(num_models):\n",
    "        assoc.append(AssMem(E, U))\n",
    "        lr = lrs[i]\n",
    "        opti.append(th.optim.SGD(assoc[-1].parameters(), lr=lr, momentum=0))\n",
    "\n",
    "    for i in range(nb_epoch):\n",
    "        x = th.multinomial(proba, batch_size, replacement=True)\n",
    "        y = x % m\n",
    "\n",
    "        for j in range(num_models):\n",
    "            out = assoc[j](x)\n",
    "            loss = F.cross_entropy(out, y)\n",
    "            train_loss[j, k, i] = loss.item()\n",
    "\n",
    "            with th.no_grad():\n",
    "                pred = assoc[j](all_x).argmax(dim=-1)\n",
    "                test_loss[j, k, i] = proba[pred != all_y].sum().item()\n",
    "\n",
    "            opti[j].zero_grad()\n",
    "            loss.backward()\n",
    "            opti[j].step()\n",
    "\n",
    "        if i == 100:\n",
    "            for g in opti[-1].param_groups:\n",
    "                g['lr'] *= .03\n",
    "\n",
    "    print(k, end=',')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x150 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "plt.rc('font', family='serif')\n",
    "plt.rc('text', usetex=True)\n",
    "\n",
    "fig, ax = plt.subplots(1, 1, figsize=(2, 1.5))\n",
    "leg = []\n",
    "for i in range(num_models):\n",
    "    a, = ax.plot(batch_size * np.arange(nb_epoch), test_loss[i].mean(dim=0), color= 'C' + str(i))\n",
    "    ax.fill_between(batch_size * np.arange(nb_epoch), test_loss[i].mean(dim=0) - .5 * test_loss[i].std(dim=0), test_loss[i].mean(dim=0) + .5 * test_loss[i].std(dim=0), color= 'C' + str(i), alpha=.3)\n",
    "    leg.append(a)\n",
    "ax.set_xscale('log')\n",
    "ax.set_yscale('log')\n",
    "ax.set_yticks([.2, .3, .4, .5, .6, .7, .8])\n",
    "ax.set_yticklabels(['.2', '', '.4', '', '.6', '', ''])\n",
    "ax.set_xlabel('T', fontsize=10)\n",
    "ax.set_ylabel('Error', fontsize=10)\n",
    "ax.tick_params(axis='both', which='major', labelsize=8)\n",
    "ax.legend(leg, [r\"$\\gamma=10$\", r\"$\\gamma=.1$\", r\"StepLR\"], fontsize=7)\n",
    "fig.savefig('sgd_step_lr_mean.pdf', bbox_inches='tight')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
