{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from math import sqrt\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import torch as th\n",
    "\n",
    "plt.rc('font', family='serif')\n",
    "plt.rc('text', usetex=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def modular_class(x, m):\n",
    "    return x % m\n",
    "\n",
    "\n",
    "def squared_modular_class(x, m):\n",
    "    return (x**2) % m\n",
    "\n",
    "\n",
    "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"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Scaling with respect to T and d jointly"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# number of input tokens\n",
    "n = 1000\n",
    "# number of output classes\n",
    "m = 5\n",
    "# memory dimension\n",
    "ds = [10, 20, 50, 100, 200, 1000]\n",
    "dmax = np.max(ds)\n",
    "\n",
    "# number of data\n",
    "ts = [10, 100, 1000, 10000]\n",
    "tmax = np.max(ts)\n",
    "\n",
    "# Zipf parameter\n",
    "alpha = 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Population data\n",
    "all_x = th.arange(n)\n",
    "proba = (all_x + 1.) ** (-alpha)\n",
    "proba /= proba.sum()\n",
    "all_y = modular_class(all_x, m)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "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": [
    "P = None\n",
    "rho = 0\n",
    "\n",
    "nb_trials = 100\n",
    "\n",
    "errors = th.zeros(nb_trials, len(ts), len(ds))\n",
    "errors[:] = -1\n",
    "\n",
    "for i_n in range(nb_trials):\n",
    "    # Embeddings\n",
    "    E = get_embeddings(n, dmax, norm=False)\n",
    "    U = get_embeddings(m, dmax, norm=True)\n",
    "    x = th.multinomial(proba, tmax, replacement=True)\n",
    "    y = all_y[x]\n",
    "\n",
    "    for i_t, t in enumerate(ts):\n",
    "\n",
    "        counts = th.bincount(x[:t])\n",
    "        _, order = th.sort(counts, descending=True)\n",
    "        P = th.sum(counts!=0).item()\n",
    "        idx = order[:P]\n",
    "        q = (counts[idx] / counts[idx].sum())**rho\n",
    "\n",
    "        for i_d, d in enumerate(ds):\n",
    "            P = int(d / 12)\n",
    "            W = (E[idx[:P]].T * q[:P]) @ U[all_y[idx[:P]]]\n",
    "\n",
    "            scores = E[:, :d] @ W[:d,:d] @ U[:,:d].T\n",
    "            preds = scores.argmax(dim=1)\n",
    "            errors[i_n, i_t, i_d] = proba[preds != all_y].sum()\n",
    "\n",
    "        # print(len(idx), end=',')\n",
    "    print(i_n, end=',')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "errors_mean = errors.mean(dim=0)\n",
    "errors_std = errors.std(dim=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 100x100 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "X, Y = np.meshgrid(ds, ts)\n",
    "fig, ax = plt.subplots(figsize=(1, 1))\n",
    "c = ax.contourf(X, Y, errors_mean.numpy(), levels=20, cmap='RdBu_r')\n",
    "ax.set_xscale('log')\n",
    "ax.set_yscale('log')\n",
    "ax.set_xticks([10, 100, 1000])\n",
    "ax.set_xticklabels([\"10\", \"$10^2$\", r\"$10^3$\"], fontsize=6)\n",
    "ax.set_yticks([10, 100, 1000])\n",
    "ax.set_yticklabels([\"10\", r\"$10^2$\", r\"$10^3$\"], fontsize=6)\n",
    "ax.set_xlabel('$d$', fontsize=8)\n",
    "ax.set_ylabel('$T$', fontsize=8)\n",
    "bar = fig.colorbar(c, ax=ax, ticks=[0, .3, .6, .9])\n",
    "bar.set_ticklabels([r\"$0$\", r\"$0.3$\", r\"$0.6$\", r\"$0.9$\"], fontsize=6)\n",
    "fig.savefig('thres_storage_error.pdf', bbox_inches='tight')\n",
    "# fig.savefig('fill_storage_error.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAOkAAACvCAYAAAAYNvmSAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAsaUlEQVR4nO2deXxU1d3/33cm+75AIAkQSAAxLEJIwCIoy6AWfcSFBEShbpBSRKlKAlQrtCIQUNxofwno02JtkYTHulSQDNQFRcjCpizBDAFCFiD7nkzm/v6YzJAhmZBMtknmvF+v+8rcc86998zJ/czZvud7JFmWZQQCgdWi6O4MCASClhEiFQisHCFSgcDKESIVCKwcIVKBwMoRIhUIrBwhUoHAyhEiFQisHLvuzoA1oNPpyMnJwd3dHUmSujs7gl6MLMuUlZUREBCAQtG6OlKIFMjJyWHgwIHdnQ2BDXHp0iUGDBjQqrQ2LdKtW7eydetWtFotoC84Dw+Pbs6VoDdTWlrKwIEDcXd3b/U1krDd1Recp6cnJSUlQqSCTsWSd00MHAkEVo4Q6U04e/Yss2bNIjs7u7uzIrBRbFqkW7duJTQ0lIiICLNpFi1axJ49exg7dixffvllF+ZOINAj+qS03E/45ZdfmDt3Lunp6QCsWLGCdevWYW9v3x1ZFfRwRJ+0Exg6dCg//PADy5YtA2DTpk3cddddXLx4sZtzJrAVhEhbgaOjI++88w67d+/G09OTQ4cOMXbsWE6ePNndWRPYADY9T9pWHn74YcaNG8e8efOws7NjxIgR3Z0lgQ1g0yI1GDPU19e3+pohQ4bw3XffUVJSYuyX1tbWkpubS1BQUGdlVWDDiIEjzHfmP9n4W5z3fUuVi5JaV0e07i7g6Y3Cpz/2/YbgGBCKx6Db2PbWJj7b9SGvvfEe0+59gBptPbVaHTVaHTXa+oa/OmNYXb2OAC9nRgd6MtjXRdgL2xCWDBzZdE16M2ouZTLisgxoG44K4CqQAXyrT4PMyeyLlFVU8fyihdw9xIdHwoOoc3GhytGVcidvKhx9KXX0p97BAxc7CSc7uDb8DjKvlOPuZMfIAE9GBXrg7iRGjAVNETUp5n/d0vYncv77z6kvuIJUUoKyvBqH8lqcK3W4VYB7JShkqJNl3r12le2FhQDc6ujImwGBBDk4mDynxh7KXKDCReLcoD4kj1lN2LCBeDrbo5AkBvdxYVSgJ0N8XVEoRO3aG7GkJhUixXLbXW1NFdfOH+fi2TTyMo7y/cFU/vrVz5TVaHG2U/DcLf35tbsHbhXgUtv0+uOh9vx56GoGBAQQHuRNHzdHANyd7Aj192BkoCeezqJ27U0IkVpIRxrYX758mbnz5vH9wYO4uLrxdkICUmk2iiunUBZfxL68EEV5NaOO6mvhc0MkNo9axkXlIIb0cSVisDf+ns4ASBIE+bowOtCT4D5uonbtBQiRWkhHr4LRarWsXbuWkSNHMm/ePGRZpqiyjrySavJKq8i9fAHdrmcZ830ZDlq41F/m/YgoDisnAhDo5UzEYG8G+VwfVHJ1VBLqr++7erk4tPR4gRUjRNpGGk/BZGRkdOpStQMHDpCTk8Pjjz8OwPn8Er7dNJeR6gu4VsNVb0i96zbekxaga/iP+Lk7Eh7kTYifG4oGsUoSDPTW912H+rmhFLVrj0KI1EI6ez3p1atXGT16NPn5+Tz11FO8++67uLi4oNPJfPLGEvrv+gafMihxhcIZbvzvgA0czatB26BWLxd7woO8GdHfw0SULg5KbvX3YHSgJ96uonbtCQiRWkhni7S+vp5169axdu1adDodoaGhJCYmEhoaCsD3u9+ieks8AdegygGuTtOhmbKF/xZ4c/xSMTVaHQBujnaEDfJiVKAn9kpTi84B3s6MCvRkmJ8bdkph7WmtCJFaSFd5Zvj666+ZP38+ubm5ODs7s3XrVp544gkkSSLjx/9w7uUVBGfLaBVwaUotNVNXkeE1mZ8ul5B+sYiKWr1llJO9grEDvLhtoBdO9kqTZzjZK7nV353RgZ74NowWC6wHIVIL6Ur3KVeuXGHBggXs27cPgAULFvD+++9jb29PvuYkPzw/nxHn9D6Xzk2sw2PKHI4OehKtLHM6t4y0C0WUVNUBYK+UGB3oybhB3rg5NrVLCfByYlSgJ8P7uTepeQXdgxCphXS1jyOdTsfGjRt55ZVXmDdvHh9++KFxFLe0MJ+9S+9j9NEKADJGaxk4OYyDI16mXuGITidz7ko5qRcKuVaun3xVShK3+rszPsi72ZFfR3sFXs4O2CslHOwU2CsVOCgV2Nsp9GFKhTHcEOfQEGdv13CuVPSYKSDDK214s+Ubw03SGtKYXtOaeHP3dVAqmrRwDAiRWkh3OSI7dOgQo0aNMnqOq66uxtHRkbqaKj55/n5GfZOLAvglRMfQKb58PfpNqh18AP2LkVVQSWpWITkl1QBIwLB+boQH+dDXveObunYKqUHYehErJQkZGVlueEFl/assy6YvriG+sXhMrrshrDHmBdcojRmBdRcRg32YPKxPs3FCpBZiDd4CZVnmwQcfxMXFhfj4eNxcXfnklUcZ9ulJ7OvhQqDM0LskDo3eyDU30yVyl4uqSLlQyIWCSmPYYF8Xwgf7EOjl3NVfxebpaJEKA3srIT09nS+//BKtVktqaiq7du3ikdd3saff7/H7YC9BlyUu7ZWZWLeM86NXcsp7hvHaQG9nAr0DuVpWQ2pWIeeulJNVUElWQSUBXk5EBPkQJFbb9FhsuibtSmOG1nDo0CHmzZvHxYsXcXBw4M033+R3v/sdP+x6B2nT/8O7HIrcwHV6Ka4RT/KF1+PIzTjXKK6sJe1CEadzy6hv+PcqJLBXKrBTStgrGv42c274bIxXNErXKN5eqUCpkDDI3qB/STINk5Aa/gIN59fjrodJje7RuMl8vbksmzSbzX5u6Vpzn9t6bUvpZZmBPi7cN8afSSFNa1PR3LUQa2juGigsLOTJJ5/ks88+A+CRRx5h+/btZB//L1dW/wH/AplKB6ieVsb421V8MXgVl8qbryHLq7UcvVTEycsl1NXb/L+5S3li0mDWPDCySbgQqYVYk0hB/2v8zjvvsGLFCurq6pg2bRoHDhzg0qkUflr2FIMva9EqIGdqJfeMG8q5qX/h61x7ymu0zd5PW6+jqq4erU6mrl6Htr7h7w3ndToZbb2OuvqGvzc5r5dvHPxpPDDUuRhqXmNNbailJb3jLkPTXiFd/yxJmJhXKhqubbjMNE5qXOObPsPQCjC0GiTJ9HMfN0ceCRvAI+Ob7vUiRGoh1iZSAykpKTz++OPs2LGDiRP1xvfF+Rf5+pmHuOWcfpAo41c1/M8YF7QPvU+KNoT0C0VGc8LupukIr169N47oGsIMzUiF8YVvXhCAVfevO3rgSMxwWzERERH8/PPPRoECHDl+hol/+YRT430BGH7IkX//WIP8zzncUbaPBb8KIriva3dl2QS9yCQUkoRSIWGnUGCnbDQfa6fA0U6Jo70SJ3slzvZKXBzscLLXhxnmbu0a+r8KhWS8py0hRGrl2NldH4A/ceIEs2fP5vbJU+j32/WcVoUAcOtxO/b86EL5p8/j9f06Zo/ux0PjAvERRve9AiHSHsaAAQO4ePEi06ZPJzPoLs7Nm4RWAbdkKPj2oC9Xvn0XEhcy2E3H47cHcefwvjjai39zT0b0SbHePmlzlJaWEh0dzc6dOwGYNWsWi6aHMujvX+BcCzl9YMhd1wgeOAKi/gE+QVTWarlWVku9LFOv01GvA61Oh67hb71Ovn7IMlqdTH293JC+6aFtSFdfr6NexnhPw1+dlb9SSoVkHFDSN8X1A0X6c3284bNCklA29I0VkoSiIe31AxSKRp8liSBfF4L7ujX7bDFwZCE9SaSgH1zZvn07zz33HNXV1QQGBrJm6WOMSvoczwqZQndwm1bEbf284KEECL6zS/On05kKXKuT0Rn+yqbnN/4A6AwDRwaBmAjKIDDTeKOADIIyCglTMUlSt9sf27xIi4uLWb9+PXPnziUsLKzV1/U0kRo4ceIEUVFRnD17ljVr1vDYPZO4/Pvl+BXpqHCEqhllTOkjg2otjH8ClMLArLux+dHd1NRUiouLuzsbXcaYMWNITU1l48aNvPzyywy9fSahHyVyKcAe1xrw+MqdPTl2sCcGPn4MTiZB4XnQ6bo764I20C0iTU9PZ/z48U3CNRoNcXFxJCUlERcX12bBqVQqvLy8OiaTPQQ3NzdiYmJQKvVLo9wDgnlD6ccn3joc6mGQ2oVPLnkhZ+yFT34LuxbCnljI2Acll7s594LW0OXtn6SkJIKDg437fTYmMjKStLQ0QC/YRYsWkZiY2NVZ7NHExcWxT70ftULBTyMHsrrGiREHHdk1rj/3Ds/HM+8E5J2AU59A/zEQdAcMCId+I8G1+Ql4QffS5SKdM2dOs+EajcbkPDg4GLVabTxPSkpqksZwv+Dg4I7NZA/mpZde4sKFC3zwwQf86+QFfh7Ql3cdPRlz1I5DOf44qxy4S86GiquQuR80X0Of4eB/m16s/UeD363g5NndX0XQgNWMJKjVanx8fEzCfHx8SE9PJywszKy4Baa4uLjw/vvvM3XqVJYsWcKJ7Ks86FbBn/x8mZHvSs3Htfx12lAei/DH48opKM+Hq6f1R8Zefe3qPwb6joB+odD3VnBw6e6vZdNYjUjN9T8LG/ZXaQ1qtdqkGW1uhLempoaamhrjeWlpaauf0VNYsGABERERREVFcfLkSZ6rqGL5rQNZpHVhanIpezQV9HnqfmbY10Peccg/BdXFkPUtZH0HviHQ/zbwHQZ9QsAvVF/j2gkrpq7GakRqjrYMHqlUKlQq1U3TrV+/nrVr17YjVz2DESNGcPjwYZYvX05iYiLLPt2PJn4tfb9MZUxmPcXr9vDW3FuJnrUU56sZkP8z5B6H0mwo+EV/OLjpm8D9x4BbX71o/UL1IlY078dH0LFYPE8aERHBqlWrePjhhy17sCTR+NEJCQnEx8cbB44AvL29SUxMbJXw2kJzNenAgQN73DxpW8jNzcXf3x+AC/s/4+CKFYRV6gf3vx3vyOiX1zHJwQVyj+lHfXOPQ/5PUHfdJQteQfq+a5/h4OimF6v/WHDv1/VfqIfSpfOkixcvbiLQAwcOWHo7s0IMDw+3+J7mcHR0xMPDgw8//JDbb7+dGTNm3PyiHo5BoAA/VStZcOwcf7Kvo06WuTOthopnXmLLwV3UTYiGsfNhdCTcvhRCHwTvIfoLiy/A6c/g0Htw+gvI+ApSP4DU/4Wco6BtZus4QbuxuLkrSRJLliwhJCSE4OBgCgsLSUxMZPr06a2+R3FxsXFe88YRWo1GQ3h4eKfOey5dupSlS5caf91shR9//BFZltn5UyYZw4JZY69kaIGS/hu/ZsuRadz7x3jGTFgEBZmQnaIfRKou0U/d5J2EmlK4nKo/PAdC4Hh97fvLfn3tGjAWPAK6+2v2Gixu7g4dOrSJ8cD+/ftJSUlp8Tq1Wk1ycjJxcXHExMQQERFhHLnVaDTEx8cTERFBSkoKq1at6hLjhJ5qFtgedu/ezdNPP01JSQnenp6suGUAD5boPeT/PEgi7/lH+N2stSgkBZRf0Ys1/xTU1+qtlvKOw7VzGJdwO7hBwDh9c9jBDdz89GL1Gwn2Tt32Pa2NLrXd3b9/f5NmYnNh1oy1OSLras6fP8/cuXONP6wLJ09kWUExrjqJcif47IG+/Ob3HzDUe6j+gppyfbM2Jx1qK/U1as4xfT/W0HeVFNDnFn3t6hEIdvb6aZyAseDZ1J2IrdHlBvalpaXs2rULgKioqB77gttiTWqgtraWlStXsmXLFgCS3nmHgP/7EK/cMgC+G6VA+v0inpn0/HWPCPVauPIzXDoCFddAp4WrZ/XiLW1kaujqpxerXygo7fUWTf5jof8osLdNf8BdKtLz588TGRlp7EsePXqUxMRExo4da8ntuhVbFqmBzz77jIMHDxIXF4eupoafY59D8dW3KGTI94Iv5g3h909vI9A90PRCQ7+18Lz+vCxPL9Yrp/TiBbBz1E/hBISBszco7KDvLfqmsXdQl37P7qZLRbp582Zeeuklk7BVq1axfv16S27XLdh6c7cl8vPzWf/8c0T+cgqfci06CT6bZMeg5St4bPTCpheUX4FLh+HKadDVQ12VfqAp56jeSMKAT7BerD7B+qaxi69erP1H24RlU5eKdPfu3TzyyCM3DesJiJrUFFmWuffee9m3bx9jx4zhT0H9GZpxEYCMAPj6N2P4Q9RW+jg3Y5BfUwbZqQ1TMjV6l4CFGshJ0/814OSpF2v/Mfqmr0IJfYbpjSUUSr2ADQeYnkuKBj+aCsvSdaMjsy7dZqI5Y/fz589bejuBFSFJEs8//zxpaWkcO3GCx86fZ/P8R5n4w0GG5+gYtPkEG0/OZHL0q8we9qDpxY7uEDJNv7om74S+KSyF6C2Uqor04s07rp/S0fxXb4LoF6oXrK4erpzpii+I3tluYxE3J/rm4lqRru8t+lVFHZVdS2vSo0ePEhsba1wXqlar2bhxY5vmSbsb0dxtmezsbObPn893330HwDOPPsrTVy/jeekKAIdvkTiyMIw/3ruZ/q79m7+JTgfXMvRN4dIcfVh9nb7PmpOmbyYbcA/QN3vtHBvVeDeKw0yYiVhaSi/dIK5OqFUH3a7/oWqGLh/dPX/+PPHx8QDMnTuXcePGWXqrbkU0d82j1WpZs2YNr7/+OrIsM2bMGP46Yzoee/ei1EGBOyQ84MSU2b/jmdHPtOwTt/gSZB/Rz68aNlgpvawfaLp6BuTu8BhhRrg3ir3JD4AhvJkfBBcfGPcYRDzT5GldKtL22u5aE7Yi0qSkJD7++GOLFtInJyfz+OOP4+npSVpaGopTp9C88Bx2BaXogM8nSqQ+MJzXpm1ghM+Ilm9WWajvt+Yd10/nANSW6+2FS7JBrm8Qsa5h01Hd9c/GcB161/c60zBZbhTe6LyrmRANs+KaBHepSLdt28aiRYtMwg4cONCjmrsGbEWkoPd+Yam3i9zcXIqKiggNDQVAW15O5gu/R/ftQQA0/eG92fZM/tVcVkSswEF5k2VtdVVwOR0up0FthUV5ahXNCdcgbHSmPwg3/QFoJv2NYT7BEPqAfhH9DXTpwFFH2O52N437pJ1FZGQkM2fOpLi4mI8//phVq1ah0WgoKChg48aN7bq3Oe+ICQkJRhc1Hem5wt/f38RQf+v775Pw/Q9sX/JbPP/2vwTn1fD6B3X8/eI/mXXxAGvuWMvkwMnmb2jvDIPvgIETTY0jOprGA0VdwaDbmxWopVgs0g0bNqBSqbh27RrXrukLti0LtK2B1hjYy7JMVZ15ETvbK832w5KSkti2bRteXl4kJCSgUqmMdspJSUntzn9z3hE1Gg2ZmZksXrwYlUrVrpqzJaqrq9myZQsXLlxgxqpVvL1uHdO//RZOnyF6j47DmbnEFCxhwogZrJ20Fk/HFhYwKO30c6X+t0H51QYjiBtrK9m0FkQ2E2curbn4Tkjfwa5nLBZpfHx8s7a7vY2qunpC//iV2fhTf7oHF4fmi7HxAoS0tDQiIyNN4tqLSqUiOTnZJEytVhMSEmI8bzxVplar0Wg0qNXqdj/fycmJw4cPs2DBApKTk1n8wgssePxx1j71FNV/+xsTM3QMy9Gy9f5kZuWl8GL4izw8rBXjF25925Wv3ojFIo2NjWX16tUmA0c9ybi+K2i8gketVhtHwm+MA33TNSEhodn7hIWFtVpUjZf/Gc4NqFQqk0X17aVfv37s3buXDRs28Morr/DhP/5ByogR7HjtNbzefQ+fvDxe2anjs4lF/Lnyj/z7l3/z+uTXGeAuDO3bgsUijY6ObnbRd0/qk7YGZ3slp/50T4vxN6M1LmC8vLyIiYlpS9bM3qcrHYQrFApWr17NlClTePTRRzlz5gwznnwSzenT1GzZQukX/+GBwzKjs+p5e3Y6DxY8yNOjnua3t/3W5rYwtBQxcHSTgSNJksw2Z1tLa5qXHVWThoeH8/HHH5tc2xVMmTKFY8eOsXDhQqZPn06fwEDYvBk3lYq8P7zMkPwK4j6o52+qKv6i3cqerD2su2Mdo/uO7pL89WQ6dNG3Wq0mNTW1o/LWZXTmFExSUhLx8fEEBwcTHR3doaIxWHmFhYWZjPAaRnc1Gg1RUVFd6tVf17CFhUKhH0n9+eefqcjLw2/7dqqOHgMgdbiCv/5aotLVjgeHPsjKCStxsrONheFi0beF2NI8aVdSUVFBeHg4Go2GN954g7n29lx7512or6fUTcHb98PJIQr8XPx4eeLLTBvUvCldb6JTHZEdOHCA7du3s337dkpLS41i3L9/P9u2bWPJkiUcPXrUspwLeiV1dXUMHz6c2tpali1bxu/Uarw/eB/7AQPwKNfxyk4dTx6QKCzN57n/Psey/csoqCro7mxbHa2uSX19fdm/f7/ZRd3FxcWEhIRQUNDzClnUpJ2HLMu8/fbbxMTEUFdXx5AhQ/jX3//OwL17Kdn9fwDk+zux4f46LveRcLV35ba+t3HngDtRDVLRz7V3uQvt1ObuypUr2bBhAwBZWVkmcYMHD26SpichRNr5pKSkEBUVRVZWFvb29mzatIknRo0ib+UqdGVl6OyU/PNuRz4bU2NcmeKgcGCI5xAm9p/I3YPvJtQ3FHulfTd/k/bRqc1dX19f4+eioiIiIyObWM00nkQXCBoTERHB0aNHefjhh6mrq+Pf//43blOnEvyfL3CZMAGFtp7Hv6xk695AJtoPx8nOiVpdLWeLzrLj9A4W7FnAfZ/cx0tfv8QXmi8oqirq7q/UZbR6bqHxCOG4ceOIiopq4j6lp817dYXtrjWRlJREYWEhxcXFHTIn21a8vLyMppL3338/SqUSpZ8fg/72vxT+fQdX3niDvscu8lKGC07PR3Mo3I1vcr7lTOEZCqoLyK3IJbcil68ufIW7gzvDvIYxKWASqkEqgjyDsFNY/a4pFtHq5u6qVatYvXq1cWuIhIQEoqOjTbaKWL9+fY/ycWTAFpq7ho2swsLCmDlzZhNzwu5k+fLlBAYGsvT++8lbtZqaM3rvDE6jRxPw+jqqB/mRmp/K/gv7OX7tOJfLLlMvX/9htVPYMch9EGF+YcwImsGYvmPwcLDO/2On9kkVCoVJTSnLcrPnPbFW6kyRWtsqGI1Gw8aNG01MFLuTQ4cOMWnSJABmzZrF3z74AMXer7i6ZQtydTUolfg+8wx9frcEhaMj1dpqzhaeRX1RTWpeKpoSDZXaSpN7+rn4McJnBHcG3smkgEkEuAWgtJLNpSx61+RWEh0dLWs0Grm4uLjZIzMzU46Ojm7t7ayKkpISGZBLSkqaRup0slxTbv7Q6czeNzExUS4qKpJlWZbj4+PlmJgYk7j2kpycLC9evFhOS0szhmVmZpo8Z86cOSZx8fHxxjxZAzqdTo6Pj5cdHR1lQA4MDJS/++47uTYnR77w9DPyqVtGyKduGSGfm6GSy388bHJtXX2dfL74vLzjpx3y4n2L5WkfT5NH/W2UyTHxo4nyo188Kr+Z8qZ8NP+oXF5b3k3fVE+L75oZWl2THj169KbuUVqTxhpp8dettgJeb2Ffk9U54ODabFRjY/fo6GgiIyON5n03GsJbSmxsbBNrI9BvqAUwfvx40tLS0Gg0Rj/JxcXFVtXcBThx4gSRkZFkZGSgVCr585//TExMDBXq/eStWUN9kX6gyHP2A/RbtQrlDWUnyzJXKq9w7MoxDlw6wKmCU2SXZaOVtcY0dpIdge6BjPIdxV0D72Js37H0c+2n30qji+hyH0e9hc4SaWNCQkLIzMw0G2+p7e6NIo2Li8PLy8so0ps915ooLy9nyZIl/OMf/wBg3rx5/Otf/6K+tJT8uDhKknYDoPD0pP8rL+Nx331mBytLako4W3iWr7O/Jj0/nQulFyivKzdJ08e5DyGeIdzufzt3BN5BkEcQLvad6/u3Sz0z2Az2LnohthR/E3rzKpiOxM3NjR07djB9+nSeffZZ5s6dC4DSw4OA117D6+GHyVm5irqLF8l5aQXFiUn4r3sNhwFNl755OnoywX8CE/wnUKWt4kLJBX7I+YEfc39EU6IhvzKfa1XXuFZ1jcN5h9l2chtBHkGM6TuGqQOmMsx7GH4uflYxYyFEejMkqVU1ZUvYwiqYjkKSJJ588klmzZpFv37XrY3Onj3L0NtuI+SLz7mWkMC1/xdP5eHDaGbdR59ly/B98gkku+ZfZ2c7Z0b4jmCE7wgWjlxIdlk2J6+d5JtL33Cu+ByXyi5Rqa3kdOFpTheeJikjiQC3AIZ5DWNy4GTG+o1lkMcgHJWOXVUMJojmLmIVjLWTnZ3N2LFjue222/joo4/o378/NefPk7tqNVXHjgHgEBJCwIYNOI8e1er7yrJMfmU+GUUZHLx8kJ+u/URWaRZltWUm6XycfBjsMZiwfmFMCpjEEM8hzXvvbwWiT9pGhHPsnsFXX33FI488QkVFBX5+fnz00UeoVCpkWaZ49/9xZf16dBUVIEl4P/oofi++gMK17a2f4upizpec50jeEdLy08gqzSKvIg+5kUtQJ6UTgzwGcYv3LUwOnMwInxEMdB/YanNFIVILsQVjhp7OmTNniIqK4uTJk0iSxOrVq1mzZg12dnZoCwrIW/snyvbtA0DZpw/+f/4T7tMsX/pWpa3iQukFfrr2E4dyDqEp0XCx9CK1ulpjGoWkwN/VnyGeQ4joF0FYvzAGeQzCx8nH7H2FSC1EiLRnUFVVxfLly4199ylTpvDPf/6TAQ0DR+XfHST3D39Ae0W/dYXb9On0X/Mq9n5+7Xpuna6O7LJsMosy+SH3BzKKMsgqzaKkpsQknZejF4M9BnPP4HuYN2Jes2aKQqQWIkTas9i5cyeLFi2ivLyc559/nrfeessYp6us5Mpbb1H0j49Ap0NycaHfipfwmjsXSdH++VBDP1ZToiEtL42fCn7iQukFcstz0XF9m4zpA6fz9vS3m1wvRGohtiLSxuaC3WFg35GcO3eONWvWsG3bNlxcmk6DVZ8+TU5MLDXnzgHgNHIkARs34Dh0aIfmo6i6iKzSLE4VnCI9P52s0iwul19m2bhlPHbrY03SC5FaiK2I1OCD15psdzsKnU7HypUrefbZZxk0aBAAcn09hX/fwdW330auqdHbAT/9FH2WLkXh2PHTKZV1lWSVZqGUlAz2HNzslE2nricVWEZkZCQJCQnExcUxfvx4kpKSiIuLIzY2tt33Li4uJjY21rjCxUBCQgJqtZq4uDgT59g+Pj6kp6f3ynW/mzdvZtOmTYwdO5bPP/8cAEmpxPepJwnZ8yUuv/oV1NdTkLANzaz7qDhypMPz4GLvQqhvKLf43NKxc6odYjXcw7HE6Lk1WJOBfXJyspyZmSnLsiyrVKp2P9vayMzMlMPDww1bqMkvvPCCXFNTY4zX6XRy8X++lM9OmGg02s9+aYWs7eLFBpa8a6ImvQmyLFNZV2n2kFvoLdy4zcTMmTNN4trLjS5Vwfw2E8HBwajVahISEky2u+gtBAcH8/3337N8+XIA3nzzTaZMmWLcfV6SJDxn/ZqQ5H14PPggAKWff84vd99Dyeeft/h/7G56VZ/U4M4lJSWFmTNntloILfUTKusqmfjPiWavPTz/cKuMsoWBfdfx6aef8sQTT1BcXIynpyc7d+7k3nvvNUlTkZpK7spV1GVnA+AyIQL/19fjMCCwU/Nm0wb2hs2IYmJiCAsLIzIyskP3PWkPwsC+a5k9ezbHjh1j3rx5pKenm9gAG3ANDyf4y/9w7S9/pWDbNiqPpKCZNYs+zy7F96mnzNoBdwfdkpP09HQWLVrUREQajYakpCSj3enixYtbbXeqUqmMtY1GoyE8vGP2h3S2c+bw/MMtxt8MYWDf9QQFBfHtt9+SkpJissa5oqIC1waTQYWDA37Ln8fzgQfIiY2l+uRJrr65hZJ/f0pAXBzOo0Z2V/ZN6ZzusXkSExPltLQ0ublHh4WFGT9nZmaaeBVoC3PmzDEOkrSGzho4kmX991WpVE0GeDqC5ORkWaVSyTExMSb3jo+Pl5OTk63OC0N3k5KSIvfp00feuXNnkzidTicX7twpnxkXph9YGnGrnLNmjVxf3rGeHDrVM0NHI0mSSWfd4Dmgce3q7e1NUcOK/KSkJJPpBAM3+vCJi4tDpVK1qQaxlXlSW2fhwoV8+OGHgN5TxpYtW3B2Nm0Jaa9dI/fVVynffwAApa+v3g64gzYi61QfRx3NjY+Oj49vMjUQHBzcptonOTnZmL4tUxydWZMKrIe6ujp59erVsiRJMiCPGTNGPnPmTLNpS//7tXz2jsnG6ZqL0b+V665caXceevQUjLnBjsLCwlZdb6iJFy1aREhISIs+fGpqaigtLTU5BL0fOzs71q1bx969e+nbty8nTpxg/PjxRnctjXGfehdD932F1/z5oFBQ/vXX/HL3PRR+9BGyTtfM3TsPqxGpOVo7UhkcHExRURFpaWlkZma2aPa2fv16PD09jcfAgQM7KLeCnsDdd9/NsWPHmDp1KhUVFSxYsIA9e/Y0SadwccH/j68wOHEXDsHByFVV5P/5Nc7PiaSmC6e1rEakXl5eTWrNwsLCTvEqsGrVKkpKSozHpUuXOvwZAusmICAAtVrNq6++yv33388997Swm/vIkQR/9il9X3gBycGBmlOn0Dwwm/zNb6CrrTV7XYfR7ka2hdz46MzMTJPRXVmWZS8vr04dnXzvvffkW2+9VR4+fLjN9EmTk5M7xCSxN1FfX2/8XF5eLu/atcts2trsbDnr8QXGvmrG1Gly+eHDZtPfSI/rkzZuyjYeoYXrc52d6Z9n6dKlnDp1ipSUlE57hjVRXFxMenp6q/v5toKi0TrTZcuWERUVxW9+8xvKy8ubpLUPDGTQjr/jvykOhYcH2txcLi78DZdffIn6kpIm6Tskf51y1xZQq9XGFSDr16832ZktMTGR2NhYo/OuxMTErs5eh2NNq2BaY1Rhy8iyzNChQ1EoFOzYsYOIiAhOnjzZJJ0kSXj9z/8wNHkfHvffD0Dpf/7DLzPvpvjTTzveDrjVdW4vpDXNXZ1OJ9dXVJg9dD1km4nExEQ5MTFR3rhxo7x48WJh5NAC33zzjRwQECADspOTk5yQkNDi/7n88BE5Y9p0YxM4a8FCueZSdrNpe1xzt7tpTXNXrqribNh4s4dcVWX2WmtaBTNnzhzmzJnTI3di72ruvPNOjh07xq9//Wuqq6tZvHgx8+fPNztV5zohgpC9e/B55hlQKqk8coTcl1/usPxYjxVxL6SxgNRqtcm00I3i6ijb3Rv3mLlxCqu9O7nZCn379uWLL75g8+bNrF69mv3791NeXm7WSkjh4EC/l17E86EHyV+zhn6rVnZYXmxapK3ZRFhyduaWdPOraSTnmxvYi1UwPROFQkFMTAyTJ0+murqagIAW9gRqwCkkhKAG08OOwqZFunTpUpYuXWq0p2wOSZKQmnF01RbEKpiejWH/VAO7d+9m586dbN++3ex705HYtEi7gsbbTKSnp5sVjSU1qVqtNhnZDQsLIywsjNTUVOP62m3btrUr/wJTKisrWbJkCVevXiUtLY2PP/6YiIiITn1mr/LMYCliFYygLRw5coS5c+eSlZWFvb09mzZt4rnnnmvVDmzCW2Ab2bp1K6GhoZ3+SyjoXUyYMIGjR4/y0EMPUVdXx/Lly3nooYc6zUhE1KSImlRgGbIss3XrVl588UVqa2sJCgri+PHjLfZTRU0qEHQhkiTx7LPP8sMPPxASEsLs2bM7ZSBJDBwJBO1k/PjxpKWl4eTk1Cn3t+maVPRJBR2Fp6cnjp2wdQWIPikg+qSCrkP0SQWCXogQqUBg5YiBIzCu/xMOyQSdjeEda0sv06ZFajCwr23wUyMckgm6irKyslZP14iBI/Qb0A4fPpy0tDQkSSIiIsK4xrS0tJSBAwdy6dKlDh1UavyMjkrfUprm4loTZu7cWsqlNde0tVzMhbdUNq19Z2RZpqysjICAABO3LS1h0zWpAYVCgYODg/GXTalUNilcDw+PDn0Zm3tGe9O3lKa5uNaE3ey8u8ulNde0tVzMhbdUFm15Z9pq8CAGjhpYunRps5+74nkdlb6lNM3FtSbsZucdjSX3v9k1bS0Xc+EtlUVnloto7t4EMYfaPKJczNPRZSNq0pvg6OjIq6++2mnWJD0VUS7m6eiyETWpQGDliJpUILByhEgtwJxTalsnKSmJpKQkYmNjUavV3Z0dqyIpKcnoGL65fXZbQojUAlJTU4VHvhsw+FSaM2cO0dHRHeKhv7dQXFxMSkoKKpWKiIiINrtVtXmRpqenM378+CbhGo2GuLg447YQjUXZnFPq3kZby0WlUhkdqRn28emttLVsvLy8jMJMTk4mOjq6bQ9sta/7XkhiYqKclpbWZIc3WZZNdnjLzMw0btdgICYmpk27kPck2lMusqzf2iIzM7NT89hdtKdskpOT5ZiYmDaXjU2L1MCNBW5uG8bG9GaRGrCkXDZu3Njry0WWLSsbWZbltLQ0WaVStelZNt/cbQ61Wo2Pj49JmI+Pj80PFN2sXAxOwMPCwkx2y7MFWiobw656oG/6tnXgSNjuNoO5QSGDy8bmnFLbAi2Vi0ajITIykuDgYIqLi1GpVMyZM6drM9iNtFQ2UVFRqNVq1Go1ycnJbd7SU4i0DTQeJBH7fF6nuLiY4OBgioqKujsrVodhAy3DD5Yl741o7jaDl5dXE0fHhYWFvX5E92aIcjFPZ5aNEGkzmPu1683TCq1BlIt5OrNshEgbaNynCA4ONokzzPvZYo0hysU8XVU2Nt0nNXTkAdavX09ERISx75CYmEhsbKxxxX1bO/s9GVEu5umOshGrYAQCK0c0dwUCK0eIVCCwcoRIBQIrR4hUILByhEgFAitHiFQgsHKESAUCK0eIVNAu4uLikCSpu7PRqxHGDIJ2UVxczIwZM0hLS+vurPRaRE0qaBepqanCwL6TESIVtIvk5GRmzpzZ3dno1YjmrqDNxMXFERwcTGFhIfHx8ezfv99mV8J0BaImFbSJ2NhYo6eBqKgoo+cBQedh00vVBG1Do9GQkJBgdJOSmpoq3Mh0AaImFbSa9PR0k0Ei0R/tGoRIBW2isQeCpKQkm3Tf2dWIgSNBm4iOjjbWnhqNBi8vL8LDw23GrWl3IEQqEFg5orkrEFg5QqQCgZUjRCoQWDlCpAKBlSNEKhBYOUKkAoGVI0QqEFg5QqQCgZUjRCoQWDlCpAKBlSNEKhBYOf8fKQo2JiWcIl0AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 200x150 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "ds = np.asanyarray(ds)\n",
    "ts = np.asanyarray(ts)\n",
    "fig, ax = plt.subplots(figsize=(2, 1.5))\n",
    "for i in range(len(ts)):\n",
    "    ax.plot(ds, errors_mean[i].numpy(), label=f\"$T=10^{int(np.log10(ts[i]))}$\")\n",
    "    ax.fill_between(ds, errors_mean[i].numpy() - .5*errors_std[i].numpy(),\n",
    "                    errors_mean[i].numpy() + .5*errors_std[i].numpy(), alpha=.5)\n",
    "ax.plot(ds, 5*ds.astype(float) ** (-alpha + 1), linestyle='--', c=\"k\")\n",
    "ax.set_xscale('log')\n",
    "ax.set_xticks([10, 100, 1000], fontsize=6)\n",
    "ax.set_yscale('log')\n",
    "# ax.set_yticks([2e-1, 3e-1, 4e-1, 6e-1])\n",
    "# ax.set_yticklabels([2e-1, 3e-1, 4e-1, 6e-1], fontsize=8)\n",
    "ax.set_xlabel('$d$', fontsize=10)\n",
    "ax.set_ylabel('Error', fontsize=10)\n",
    "plt.legend(loc=\"best\", frameon=False, handlelength=.5, ncol=1, fontsize=7)\n",
    "fig.savefig('thres_storage_error_d.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x150 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(2, 1.5))\n",
    "for i in range(2, len(ds), 1):\n",
    "    ax.plot(ts, errors_mean[:, i].numpy(), label=f\"$d={ds[i]}$\")\n",
    "    ax.fill_between(ts, errors_mean[:, i].numpy() - .5*errors_std[:, i].numpy(), errors_mean[:, i].numpy() + .5*errors_std[:, i].numpy(), alpha=.5)\n",
    "ax.plot(ts, .5*ts.astype(float) ** (-1 + 1/alpha), linestyle='--', c=\"k\")\n",
    "ax.set_xscale('log')\n",
    "ax.set_xticks([10, 100, 1000, 1e4], fontsize=6)\n",
    "ax.set_yscale('log')\n",
    "# ax.set_yticks([1e-4, 1e-2, 1], fontsize=6)\n",
    "ax.set_xlabel('$T$', fontsize=10)\n",
    "ax.set_ylabel('Error', fontsize=10)\n",
    "plt.legend(loc=\"best\", frameon=False, handlelength=.5, ncol=1, fontsize=7)\n",
    "fig.savefig('thres_storage_error_T.pdf', bbox_inches='tight')"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Scaling with respect to d"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "# number of input tokens\n",
    "n = 1000\n",
    "# number of output classes\n",
    "m = 5\n",
    "# memory dimension\n",
    "ds = np.logspace(1, 3, num=20, dtype=int)\n",
    "dmax = np.max(ds)\n",
    "\n",
    "# Zipf parameter\n",
    "alpha = 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Population data\n",
    "all_x = th.arange(n)\n",
    "proba = (all_x + 1.) ** (-alpha)\n",
    "proba /= proba.sum()\n",
    "all_y = modular_class(all_x, m)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "nb_trials = 100\n",
    "errors = th.zeros(3, nb_trials, len(ds))\n",
    "errors[:] = -1\n",
    "\n",
    "for i_t in range(nb_trials):\n",
    "    # Embeddings\n",
    "    E = get_embeddings(n, dmax, norm=False)\n",
    "    U = get_embeddings(m, dmax, norm=True)\n",
    "\n",
    "    W_fill = E.T @ U[all_y]\n",
    "    W_weight = E.T * proba @ U[all_y]\n",
    "\n",
    "    for i_d, d in enumerate(ds):\n",
    "        pred_fill = (E[:, :d] @ W_fill[:d,:d] @ U[:,:d].T).argmax(dim=1)\n",
    "        pred_weight = (E[:, :d] @ W_weight[:d,:d] @ U[:,:d].T).argmax(dim=1)\n",
    "\n",
    "        # P = d\n",
    "        # W = E[:P].T @ U[all_y[:P]]\n",
    "        # pred_thres1 = (E[:, :d] @ W[:d,:d] @ U[:,:d].T).argmax(dim=1)\n",
    "\n",
    "        P = int(d / 8)\n",
    "        W = E[:P].T @ U[all_y[:P]]\n",
    "        pred_thres2 = (E[:, :d] @ W[:d,:d] @ U[:,:d].T).argmax(dim=1)\n",
    "        \n",
    "        errors[0, i_t, i_d] = proba[pred_fill != all_y].sum()\n",
    "        errors[1, i_t, i_d] = proba[pred_weight != all_y].sum()\n",
    "        # errors[2, i_t, i_d] = proba[pred_thres1 != all_y].sum()\n",
    "        errors[2, i_t, i_d] = proba[pred_thres2 != all_y].sum()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "errors_mean = errors.mean(dim=1)\n",
    "errors_std = errors.std(dim=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x200 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(figsize=(3, 2))\n",
    "# fig, ax = plt.subplots(figsize=(1.5, 1))\n",
    "leg = []\n",
    "for i in range(3):\n",
    "    a, = ax.plot(ds, errors_mean[i], color='C' + str(i), linewidth=1)\n",
    "    ax.fill_between(ds, errors_mean[i] - .5*errors_std[i], errors_mean[i] + .5*errors_std[i], alpha=.3)\n",
    "    leg.append(a)\n",
    "\n",
    "ax.plot(ds, .35*ds**(-1/4), color='C1', linestyle='--')\n",
    "ax.plot(ds, 3.5*ds**(-1.), color='C2', linestyle='--')\n",
    "ax.set_xscale('log')\n",
    "ax.set_yscale('log')\n",
    "ax.set_xticks([10, 100, 1000])\n",
    "ax.set_xticklabels([\"10\", r\"$10^2$\", r\"$10^3$\"], fontsize=6)\n",
    "# ax.set_yticks([1e-1, 1e-3, 1e-5])\n",
    "# ax.set_yticklabels([\"$10^{-1}$\", r\"$10^{-3}$\", r\"$10^{-5}$\"], fontsize=6)\n",
    "ax.set_xlabel('d', fontsize=8)\n",
    "ax.set_ylabel(r'Error', fontsize=8)\n",
    "ax.legend(leg, ['$q=1$', '$q=p$', r'$q = 1_{x\\leq d / 8}$'], fontsize=8)\n",
    "fig.savefig('gen_error_bigN.pdf', bbox_inches='tight')\n",
    "# fig.savefig('gen_error.pdf', bbox_inches='tight')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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": 2
}
