{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "id": "m55DuHdx51Ek"
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/danielkunin/Library/Python/3.9/lib/python/site-packages/urllib3/__init__.py:34: NotOpenSSLWarning: urllib3 v2 only supports OpenSSL 1.1.1+, currently the 'ssl' module is compiled with 'LibreSSL 2.8.3'. See: https://github.com/urllib3/urllib3/issues/3020\n",
      "  warnings.warn(\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "import torch\n",
    "import torchvision\n",
    "import torch.nn.functional as F\n",
    "import torch.nn as nn\n",
    "\n",
    "from einops import rearrange, reduce, repeat\n",
    "\n",
    "import itertools\n",
    "from tqdm import tqdm\n",
    "\n",
    "import copy"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "5am6z7LW1wCv"
   },
   "source": [
    "## TODO\n",
    "\n",
    "* random init seed\n",
    "* LR-dependent fudge factor for jump times?\n",
    "* longer costmin times in GD; instantaneous in AGF. Wider nets mask this discrepancy"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "PUZP5kp4ouuf"
   },
   "source": [
    "# Library"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "id": "DJDYiUY95_4d"
   },
   "outputs": [],
   "source": [
    "class ImageData():\n",
    "    \"\"\"\n",
    "    Get image datasets as numpy arrays.\n",
    "    \"\"\"\n",
    "\n",
    "    dataset_dict = {\n",
    "        'mnist': torchvision.datasets.MNIST,\n",
    "        'fmnist': torchvision.datasets.FashionMNIST,\n",
    "        'cifar10': torchvision.datasets.CIFAR10,\n",
    "        'cifar100': torchvision.datasets.CIFAR100,\n",
    "        'svhn': torchvision.datasets.SVHN,\n",
    "        'imagenet32': None,\n",
    "        'imagenet64': None,\n",
    "    }\n",
    "\n",
    "    def __init__(self, dataset_name, data_dir, classes=None, onehot=True):\n",
    "        \"\"\"\n",
    "        dataset_name (str): one of  'mnist', 'fmnist', 'cifar10', 'cifar100', 'imagenet32', 'imagenet64'\n",
    "        dataset_dir (str): the directory where the raw dataset is saved\n",
    "        classes (iterable): a list of groupings of old class labels that each constitute a new class.\n",
    "            e.g. [[0,1], [8]] on MNIST would be a binary classification problem where the first class\n",
    "            consists of samples of 0's and 1's and the second class has samples of 8's\n",
    "        onehot (boolean): whether to use one-hot label encodings (typical for MSE loss). Default: True\n",
    "        format (str): specify order of (sample, channel, height, width) dims. 'NCHW' default, or 'NHWC.'\n",
    "            torchvision.dataset('cifar10') uses latter, needs ToTensor transform to reshape; former is ready-to-use.\n",
    "\n",
    "        returns: numpy ndarray with shape (b, c, h, w)\n",
    "        \"\"\"\n",
    "\n",
    "        assert dataset_name in self.dataset_dict\n",
    "        self.name = dataset_name\n",
    "\n",
    "        def format_data(dataset):\n",
    "            if self.name in ['cifar10','cifar100']:\n",
    "                X, y = dataset.data, dataset.targets\n",
    "                X = rearrange(X, 'b h w c -> b c h w')\n",
    "                y = np.array(y)\n",
    "            if self.name in ['mnist', 'fmnist']:\n",
    "                X, y = dataset.data.numpy(), dataset.targets.numpy()\n",
    "                X = rearrange(X, 'b h w -> b 1 h w')\n",
    "            if self.name in ['svhn']:\n",
    "                X, y = dataset.data, dataset.labels\n",
    "            if self.name in ['imagenet32', 'imagenet64']:\n",
    "                X, y = dataset['data'], dataset['labels']\n",
    "                X = rearrange(X, 'b d -> b c h w', c=3, h=32, w=32)\n",
    "                y -= 1\n",
    "\n",
    "            if classes is not None:\n",
    "                # convert old class labels to new\n",
    "                converter = -1 * np.ones(int(max(y)) + 1)\n",
    "                for new_class, group in enumerate(classes):\n",
    "                    group = [group] if type(group) == int else group\n",
    "                    for old_class in group:\n",
    "                        converter[old_class] = new_class\n",
    "                # remove datapoints not in new classes\n",
    "                mask = (converter[y] >= 0)\n",
    "                X = X[mask]\n",
    "                y = converter[y][mask]\n",
    "\n",
    "            # make elements of input O(1)\n",
    "            X = X/255.0\n",
    "            # shape labels (N, nclasses)\n",
    "            y = F.one_hot(torch.Tensor(y).long()).numpy() if onehot else y[:, None]\n",
    "\n",
    "            return X.astype(np.float32), y.astype(np.float32)\n",
    "\n",
    "        if self.name in ['cifar10','cifar100', 'mnist', 'fmnist']:\n",
    "            raw_train = self.dataset_dict[self.name](root=data_dir, train=True, download=True)\n",
    "            raw_test = self.dataset_dict[self.name](root=data_dir, train=False, download=True)\n",
    "        if self.name == 'svhn':\n",
    "            raw_train = self.dataset_dict[self.name](root=data_dir, split='train', download=True)\n",
    "            raw_test = self.dataset_dict[self.name](root=data_dir, split='test', download=True)\n",
    "        if self.name in ['imagenet32', 'imagenet64']:\n",
    "            raw_train = np.load(f\"{data_dir}/{self.name}-val.npz\")\n",
    "            raw_test = np.load(f\"{data_dir}/{self.name}-val.npz\")\n",
    "\n",
    "        # process raw datasets\n",
    "        self.train_X, self.train_y = format_data(raw_train)\n",
    "        self.test_X, self.test_y = format_data(raw_test)\n",
    "\n",
    "    def get_dataset(self, n, get=\"train\", rng=None):\n",
    "        \"\"\"Generate an image dataset.\n",
    "\n",
    "        n (int): the dataset size\n",
    "        rng (numpy RNG): numpy RNG state for random sampling. Default: None\n",
    "        get (str): either \"train\" or \"test.\" Default: \"train\"\n",
    "\n",
    "        Returns: tuple (X, y) such that X.shape = (n, *in_shape), y.shape = (n, *out_shape)\n",
    "        \"\"\"\n",
    "\n",
    "        assert int(n) == n\n",
    "        n = int(n)\n",
    "        assert n > 0\n",
    "        assert get in [\"train\", \"test\"]\n",
    "        full_X, full_y = (self.train_X, self.train_y) if get == \"train\" else (self.test_X, self.test_y)\n",
    "\n",
    "        # get subset\n",
    "        idxs = slice(n) if rng is None else rng.choice(len(full_X), size=n, replace=False)\n",
    "        X, y = full_X[idxs].copy(), full_y[idxs].copy()\n",
    "        assert len(X) == n\n",
    "        return X, y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "id": "mINS0LFL6Bh8"
   },
   "outputs": [],
   "source": [
    "class ExptTrace():\n",
    "\n",
    "    @classmethod\n",
    "    def multi_init(cls, num_init, var_names):\n",
    "        return [cls(var_names) for _ in range(num_init)]\n",
    "\n",
    "    def __init__(self, var_names):\n",
    "        assert \"val\" not in var_names, f\"variable name 'val' disallowed\"\n",
    "        self.var_names = var_names\n",
    "        self.vals = {}\n",
    "        self.valshape = None\n",
    "\n",
    "    def __setitem__(self, key, val):\n",
    "        if self.valshape is None:\n",
    "            self.valshape = np.shape(val)\n",
    "        assert np.shape(val) == self.valshape, f\"value shape {np.shape(val)} != expected {self.valshape}\"\n",
    "        key = tuple((key,)) if not isinstance(key, tuple) else key\n",
    "        assert len(key) == len(self.var_names), f\"num keys {len(key)} != num vars {len(self.var_names)}\"\n",
    "        assert key not in self.vals, f\"key {key} already exists. overwriting not supported\"\n",
    "        self.vals[key] = val\n",
    "\n",
    "    def __getitem__(self, key):\n",
    "        assert self.valshape is not None, \"must add items before getting\"\n",
    "        key = tuple((key,)) if not isinstance(key, tuple) else key\n",
    "        assert len(key) == len(self.var_names), f\"num keys {len(key)} != num vars {len(self.var_names)}\"\n",
    "        key_axes = []\n",
    "        for idx, var_name in enumerate(self.var_names):\n",
    "            key_i = key[idx]\n",
    "            key_idx_extent = [key_i]\n",
    "            if isinstance(key_i, slice):\n",
    "                slice_is_full = all([x==None for x in [key_i.start, key_i.stop, key_i.step]])\n",
    "                assert slice_is_full, f\"slice start/stop/step not supported ({var_name})\"\n",
    "                key_idx_extent = self.get_axis(var_name)\n",
    "            key_axes.append(key_idx_extent)\n",
    "        shape = [len(key_idx_extent) for key_idx_extent in key_axes]\n",
    "        if np.prod(shape) == 1:\n",
    "            assert key in self.vals, f\"key {key} not found\"\n",
    "            return self.vals[key]\n",
    "        vals = np.zeros(shape + list(self.valshape))\n",
    "\n",
    "        idx_maps = []\n",
    "        for axis in key_axes:\n",
    "            idx_maps.append({val: i for i, val in enumerate(axis)})\n",
    "        for key in itertools.product(*key_axes):\n",
    "            shape_idxs = tuple(idx_maps[dim][val] for dim, val in enumerate(key))\n",
    "            assert key in self.vals, f\"key {key} not found\"\n",
    "            vals[shape_idxs] = self.vals[key]\n",
    "\n",
    "        return vals\n",
    "\n",
    "    def get_axis(self, var_name):\n",
    "        assert var_name in self.var_names, f\"var {var_name} not found\"\n",
    "        idx = self.var_names.index(var_name)\n",
    "        key_idx_extent = set()\n",
    "        for keys in self.vals.keys():\n",
    "            key_idx_extent.add(keys[idx])\n",
    "        return sorted(list(key_idx_extent))\n",
    "\n",
    "    def get(self, **kwargs):\n",
    "        key = self._get_key(_mode='get', **kwargs)\n",
    "        return self[key]\n",
    "\n",
    "    def set(self, **kwargs):\n",
    "        assert \"val\" in kwargs, f\"no val given\"\n",
    "        val = kwargs[\"val\"]\n",
    "        key = self._get_key(_mode='set', **kwargs)\n",
    "        self[key] = val\n",
    "\n",
    "    def is_written(self, **kwargs):\n",
    "        key = self._get_key(_mode='set', **kwargs)\n",
    "        return key in self.vals\n",
    "\n",
    "    def _get_key(self, _mode='set', **kwargs):\n",
    "        for var_name in self.var_names:\n",
    "            if _mode == 'set':\n",
    "                assert var_name in kwargs, f\"must specify var {var_name}\"\n",
    "            elif _mode == 'get':\n",
    "                if var_name not in kwargs:\n",
    "                    kwargs[var_name] = slice(None, None, None)\n",
    "            assert kwargs[var_name] is not None, f\"var {var_name} cannot be None\"\n",
    "        key = tuple([kwargs[var_name] for var_name in self.var_names])\n",
    "        return key\n",
    "\n",
    "    def serialize(self):\n",
    "        return {\n",
    "            \"var_names\": self.var_names,\n",
    "            \"vals\": self.vals,\n",
    "            \"valshape\": self.valshape\n",
    "        }\n",
    "\n",
    "    @classmethod\n",
    "    def deserialize(cls, data):\n",
    "        obj = cls(data[\"var_names\"])\n",
    "        obj.vals = data[\"vals\"]\n",
    "        obj.valshape = data[\"valshape\"]\n",
    "        return obj\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "id": "IELHGAUBiYpf"
   },
   "outputs": [],
   "source": [
    "def get_cifar10_dataset(n_train, n_test, device, data_dir = 'data_dir', **kwargs):\n",
    "    classes = kwargs.get(\"classes\", None)\n",
    "    cifar10 = ImageData('cifar10', data_dir, classes=classes)\n",
    "    X_train, y_train = cifar10.get_dataset(n_train, get=\"train\")\n",
    "    X_test, y_test = cifar10.get_dataset(n_test, get=\"test\")\n",
    "    X_train, y_train, X_test, y_test = [torch.Tensor(t).to(device) for t in (X_train, y_train, X_test, y_test)]\n",
    "\n",
    "    X_train = rearrange(X_train, 'Ntrain c h w -> Ntrain (c h w)')\n",
    "    X_test = rearrange(X_test, 'Ntest c h w -> Ntest (c h w)')\n",
    "\n",
    "    if kwargs.get('grayscale', False):\n",
    "        X_train = reduce(X_train, 'N (3 s) -> N s', 'mean')\n",
    "        X_test = reduce(X_test, 'N (3 s) -> N s', 'mean')\n",
    "\n",
    "    if kwargs.get('center', False):\n",
    "        X_mean = reduce(X_train, 'N d -> d', 'mean')\n",
    "        X_train -= X_mean\n",
    "        X_test -= X_mean\n",
    "\n",
    "    if kwargs.get('normalize', False):\n",
    "        X_train /= torch.linalg.norm(X_train, axis=1, keepdims=True)\n",
    "        X_test /= torch.linalg.norm(X_test, axis=1, keepdims=True)\n",
    "        if kwargs.get('center', False):\n",
    "            X_mean = reduce(X_train, 'N d -> d', 'mean')\n",
    "            X_train -= X_mean\n",
    "            X_test -= X_mean\n",
    "\n",
    "    if kwargs.get('whiten', False):\n",
    "        assert not kwargs.get('normalize', False)\n",
    "\n",
    "        X_mean = reduce(X_train, 'N d -> d', 'mean')\n",
    "        X_train -= X_mean\n",
    "        covar = torch.cov(X_train.T)\n",
    "        U, S, _ = torch.linalg.svd(covar)\n",
    "        zca_matrix = U @ torch.diag(1.0 / (torch.sqrt(S) + 1e-5))  @  U.T\n",
    "        X_train = X_train @ zca_matrix\n",
    "        X_test = X_test @ zca_matrix\n",
    "\n",
    "    return X_train, y_train, X_test, y_test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {
    "id": "E1x3SMr_I92L"
   },
   "outputs": [],
   "source": [
    "# plt.rc(\"font\", family='stixgeneral', size=14)\n",
    "# plt.rc(\"mathtext\", fontset='cm')\n",
    "\n",
    "def style_axes(ax, numyticks=5, numxticks=5):\n",
    "    ax.tick_params(axis=\"y\", which=\"both\", bottom=True, top=False,\n",
    "                   labelbottom=True, left=True, right=False,\n",
    "                   labelleft=True, direction='out', length=7, width=1.5, pad=8, labelsize=24)\n",
    "    ax.tick_params(axis=\"x\", which=\"both\", bottom=True, top=False,\n",
    "                   labelbottom=True, left=True, right=False,\n",
    "                   labelleft=True, direction='out', length=7, width=1.5, pad=8, labelsize=24)\n",
    "    # ax.xaxis.offsetText.set_fontsize(20)\n",
    "    ax.grid(alpha=0.3)\n",
    "    # Remove top/right borders and set linewidth\n",
    "    for spine in [\"top\", \"right\"]:\n",
    "        ax.spines[spine].set_visible(False)\n",
    "    for spine in [\"left\", \"bottom\"]:\n",
    "        ax.spines[spine].set_linewidth(1.5)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "id": "Dg6bn4pY6Iik"
   },
   "outputs": [],
   "source": [
    "def full_error(model, dataloader):\n",
    "    model.eval()\n",
    "    test_loss = 0\n",
    "    correct = 0\n",
    "    with torch.no_grad():\n",
    "        for data, target in dataloader:\n",
    "            output = model(data)\n",
    "            d_out = output.shape[1]\n",
    "            test_loss += F.mse_loss(output, target, reduction='sum').item() / d_out\n",
    "            pred = output.argmax(dim=1)\n",
    "            correct += (pred==target.argmax(dim=1)).sum().item()\n",
    "    test_loss /= len(dataloader.dataset)\n",
    "    accuracy = correct / len(dataloader.dataset)\n",
    "    return test_loss, accuracy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "id": "I9Azs2nhJPlY"
   },
   "outputs": [],
   "source": [
    "class ShallowMLP(nn.Module):\n",
    "\n",
    "    def __init__(self, d_in=3072, d_out=10, width=3000, init_scale=1):\n",
    "        super().__init__()\n",
    "        self.readin = nn.Linear(d_in, width, bias=False)\n",
    "        self.readout = nn.Linear(width, d_out, bias=False)\n",
    "\n",
    "        self.init_scale = init_scale\n",
    "        self.width = width\n",
    "        self.d_in = d_in\n",
    "\n",
    "        with torch.no_grad():\n",
    "            self.readin.weight.normal_(0, init_scale)\n",
    "            self.readout.weight.normal_(0, init_scale)\n",
    "\n",
    "    def get_activations(self, x):\n",
    "        h1 = self.readin(x)\n",
    "        h2 = self.readout(F.relu(h1))\n",
    "        return x, h1, h2\n",
    "\n",
    "    def forward(self, x):\n",
    "        _, _, h2 = self.get_activations(x)\n",
    "        return h2\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "id": "SMNN4HhWui8f"
   },
   "outputs": [],
   "source": [
    "def make_plot(et_train_loss, result=None, utilmax_lr=1.0, lr=1.0, mom=0.0, fudge_fac=7):\n",
    "    fig, ax = plt.subplots(nrows=1, ncols=1, figsize=(7, 5.1))\n",
    "\n",
    "    t = et_train_loss.get_axis('nstep')\n",
    "    train_loss = et_train_loss[:].squeeze()\n",
    "\n",
    "    ax.plot(t, train_loss, label=\"relu\", color=\"xkcd:azure\", lw=2)\n",
    "    # if result:\n",
    "    #     f = utilmax_lr / (lr/(1-mom)) * fudge_fac\n",
    "    #     print(f)\n",
    "    #     ax.step(f*np.array(result[\"t\"]), np.array(result[\"losses\"]), color=\"xkcd:azure\", lw=2, ls=\":\",\n",
    "    #             marker='o', where=\"post\")\n",
    "\n",
    "    if result is not None:\n",
    "        f = utilmax_lr / (lr / (1 - mom)) * fudge_fac\n",
    "        print(f)\n",
    "        x_step = f * np.array(result[\"t\"], dtype=float)\n",
    "        y_step = np.array(result[\"losses\"], dtype=float)\n",
    "\n",
    "        # ensure increasing x (just in case)\n",
    "        order = np.argsort(x_step)\n",
    "        x_step, y_step = x_step[order], y_step[order]\n",
    "\n",
    "        # extend the last step to the end of the plot domain (use GD's last step)\n",
    "        end_target = float(np.max(t))\n",
    "        if x_step[-1] < end_target:\n",
    "            x_step_ext = np.concatenate([x_step, [end_target]])\n",
    "            y_step_ext = np.concatenate([y_step, [y_step[-1]]])\n",
    "        else:\n",
    "            # still add a tiny epsilon so where=\"post\" shows a flat cap on log scale\n",
    "            x_step_ext = np.concatenate([x_step, [x_step[-1] * 1.000001]])\n",
    "            y_step_ext = np.concatenate([y_step, [y_step[-1]]])\n",
    "\n",
    "        ax.step(x_step_ext, y_step_ext, color=\"xkcd:azure\", lw=2, ls=\":\",\n",
    "                marker='o', where=\"post\", label=\"agf\")\n",
    "\n",
    "    ax.set_title(\"solid=GD, dashed=AGF\", fontsize=14)\n",
    "\n",
    "    ax.set_xscale(\"log\")\n",
    "    # ax.set_yscale(\"log\")\n",
    "    # ax.set_ylim(0.040, .101)\n",
    "    ax.set_xlabel('gradient steps')\n",
    "    ax.set_ylabel('training loss (MSE)')\n",
    "    ax.legend()\n",
    "    style_axes(ax)\n",
    "    \n",
    "    return ax"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "NBVEuOefDzDl",
    "outputId": "edf830c3-eb55-4244-c6a4-e08577ed9fa7"
   },
   "outputs": [],
   "source": [
    "class Neuron(nn.Module):\n",
    "\n",
    "    def __init__(self, d_in, d_out, nonlin=nn.Identity, init_sz=1):\n",
    "        super().__init__()\n",
    "        self.w = nn.Linear(d_in, 1, bias=False)\n",
    "        self.a = nn.Linear(1, d_out, bias=False)\n",
    "        with torch.no_grad():\n",
    "            self.w.weight.mul_(init_sz)\n",
    "            self.a.weight.mul_(init_sz)\n",
    "        self.nonlin = nonlin()\n",
    "        self.active = False\n",
    "        self.util_acc = 0\n",
    "        self.c_a = np.log(1/self.get_norm().item()) # is this right?\n",
    "\n",
    "        self.normalize()\n",
    "\n",
    "    def forward(self, x):\n",
    "        return self.a(self.nonlin(self.w(x)))\n",
    "\n",
    "    def get_norm(self):\n",
    "        sqnorm = lambda x: torch.linalg.norm(x.weight)**2\n",
    "        norm = torch.sqrt(sqnorm(self.w) + sqnorm(self.a))\n",
    "        return norm\n",
    "\n",
    "    def load_init(self, w, a):\n",
    "        with torch.no_grad():\n",
    "            self.w.weight.copy_(w)\n",
    "            self.a.weight.copy_(a)\n",
    "        self.c_a = np.log(1/self.get_norm().item()) # is this right?\n",
    "\n",
    "    def normalize(self):\n",
    "        norm = self.get_norm()\n",
    "        with torch.no_grad():\n",
    "            self.w.weight.div_(norm)\n",
    "            self.a.weight.div_(norm)\n",
    "\n",
    "    def utility_step(self, x, residual, learning_rate):\n",
    "        f_i = self(x)\n",
    "        util = torch.einsum('nd,nd->n',  f_i, residual).mean()\n",
    "        self.util_acc += 2 * learning_rate * util.item()\n",
    "\n",
    "        util.backward()\n",
    "        with torch.no_grad():\n",
    "            self.w.weight += learning_rate * self.w.weight.grad\n",
    "            self.a.weight += learning_rate * self.a.weight.grad\n",
    "            self.w.weight.grad.zero_()\n",
    "            self.a.weight.grad.zero_()\n",
    "            self.normalize()\n",
    "\n",
    "\n",
    "class Network(nn.Module):\n",
    "\n",
    "    def __init__(self, d_in, d_out, device, nonlin=nn.Identity, init_sz=1, width=100):\n",
    "        super().__init__()\n",
    "        neurons = [Neuron(d_in, d_out, nonlin, init_sz) for _ in range(width)]\n",
    "        self.neurons = nn.ModuleList(neurons)\n",
    "        self.set_mode(\"utilmax\")\n",
    "        self.d_out = d_out\n",
    "        self.device = device\n",
    "\n",
    "    def load_init(self, W, A):\n",
    "        for i, n in enumerate(self.neurons):\n",
    "            w, a = W[i][None, :], A[:, i][:, None]\n",
    "            n.load_init(w, a)\n",
    "\n",
    "    def dormant(self):\n",
    "        return [neuron for neuron in self.neurons if not neuron.active]\n",
    "\n",
    "    def active(self):\n",
    "        return [neuron for neuron in self.neurons if neuron.active]\n",
    "\n",
    "    def set_mode(self, mode):\n",
    "        if mode not in [\"utilmax\", \"costmin\"]:\n",
    "            raise ValueError(\"mode must be utilmax or costmin\")\n",
    "        self.mode = mode\n",
    "        for neuron in self.neurons:\n",
    "            grad_on = (mode==\"utilmax\") ^ neuron.active\n",
    "            for param in neuron.parameters():\n",
    "                param.requires_grad = grad_on\n",
    "\n",
    "    def forward(self, x):\n",
    "        if not np.any([n.active for n in self.neurons]):\n",
    "            return torch.zeros(x.shape[0], self.d_out).to(self.device)\n",
    "        else:\n",
    "            outputs = torch.stack([neuron(x) for neuron in self.neurons if neuron.active], dim=0)\n",
    "            return torch.sum(outputs, dim=0)\n",
    "\n",
    "\n",
    "def train_agf(X_train, Y_train, init_sz=1e-3, agf_steps=5, nonlin=nn.Identity, load_init=None,\n",
    "              utilmax_lr=1, costmin_lr=1, costmin_maxiter=2_000, loss_thresh=1e-4, device='cpu'):\n",
    "    # Initialize\n",
    "    d_in, d_out = X_train.shape[-1], Y_train.shape[-1]\n",
    "    if load_init:\n",
    "        W, A = load_init[\"W\"], load_init[\"A\"]\n",
    "        width = W.shape[0]\n",
    "        net = Network(d_in, d_out, device, nonlin, init_sz, width=width).to(device)\n",
    "        net.load_init(W, A)\n",
    "    else:\n",
    "        net = Network(d_in, d_out, nonlin, init_sz, width=agf_steps).to(device)\n",
    "\n",
    "    def update_results(results, t):\n",
    "        results[\"t\"].append(t)\n",
    "        residual = (Y_train - net(X_train))\n",
    "        residual = residual.detach()\n",
    "        results[\"residuals\"].append(residual)\n",
    "        loss = (residual**2).mean().item()\n",
    "        results[\"losses\"].append(loss)\n",
    "        results[\"models\"].append(net.state_dict())\n",
    "\n",
    "    results = {\n",
    "        \"t\": [],\n",
    "        \"residuals\": [],\n",
    "        \"losses\": [],\n",
    "        \"models\": [],\n",
    "    }\n",
    "    t = 0\n",
    "    update_results(results, t)\n",
    "    for _ in tqdm(range(agf_steps), desc=f\"AGF\"):\n",
    "\n",
    "        # Utility Maximization\n",
    "        residual = (1/d_out) * 2*(Y_train - net(X_train))\n",
    "        residual = residual.detach()\n",
    "        iters = 0\n",
    "        mode = \"utilmax\"\n",
    "        while mode == \"utilmax\":\n",
    "            for n in net.neurons:\n",
    "                if n.active:\n",
    "                    continue\n",
    "                n.utility_step(X_train, residual, utilmax_lr)\n",
    "                if n.util_acc > n.c_a:\n",
    "                    n.active = True\n",
    "                    mode = \"costmin\"\n",
    "            iters += 1\n",
    "        net.set_mode(mode)\n",
    "        t += iters\n",
    "\n",
    "        residual = Y_train - net(X_train)\n",
    "        # print(f\"loss: {(residual ** 2).mean().item():.5f}\")\n",
    "\n",
    "        # Cost Minimization\n",
    "        optimizer = torch.optim.SGD(net.parameters(), lr=costmin_lr, momentum=0.9)\n",
    "        for i in range(int(costmin_maxiter)):\n",
    "            optimizer.zero_grad(set_to_none=False)\n",
    "            residual = Y_train - net(X_train)\n",
    "            loss = (residual ** 2).mean()\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "        net.set_mode(\"utilmax\")\n",
    "        update_results(results, t)\n",
    "\n",
    "        if not net.dormant() or loss.item() < loss_thresh:\n",
    "            print(net.dormant())\n",
    "            break\n",
    "\n",
    "    return results"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "st2oZU8cI1X3"
   },
   "source": [
    "# Main expt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "GUEBqF9bI0ae",
    "outputId": "7fcbd819-19fc-441a-ad7a-96a8eedf0e46"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n"
     ]
    }
   ],
   "source": [
    "classes = None\n",
    "n_train = 10000\n",
    "n_test = 2000\n",
    "grayscale = True\n",
    "\n",
    "# Pick a local device to run on\n",
    "device = (\n",
    "    \"cuda\" if torch.cuda.is_available()\n",
    "    else \"cpu\"\n",
    ")\n",
    "\n",
    "DATA_DIR = \"~/Documents/datasets\" # <- put your dataset dir\n",
    "X_train, y_train, X_test, y_test = get_cifar10_dataset(n_train, n_test, device, data_dir=DATA_DIR, \n",
    "                                                       classes=classes, center=True, normalize=False, grayscale=grayscale)\n",
    "\n",
    "# mean center labels\n",
    "y_train -= torch.ones(y_train.shape).to(device) / 10\n",
    "y_test -= torch.ones(y_test.shape).to(device) / 10\n",
    "\n",
    "startidx = 2000\n",
    "data_size = 5000\n",
    "X_train = X_train[startidx:startidx + data_size]\n",
    "y_train = y_train[startidx:startidx + data_size]\n",
    "\n",
    "# Whitened input such that E[||x||^2] ~ 1\n",
    "with torch.no_grad():\n",
    "    U, S, Vt = torch.linalg.svd(X_train, full_matrices=False)\n",
    "    X_train = U @ Vt * np.sqrt(X_train.shape[0] / X_train.shape[1])\n",
    "\n",
    "# # Scaled input such that E[||x||^2] ~ 1\n",
    "# with torch.no_grad():\n",
    "#     mean_row_sq = X_train.pow(2).sum(dim=1).mean()\n",
    "#     scale = mean_row_sq.sqrt()\n",
    "#     X_train = X_train / scale\n",
    "#     X_test  = X_test  / scale   # use the SAME scale from train\n",
    "\n",
    "train_loader = torch.utils.data.DataLoader(torch.utils.data.TensorDataset(X_train, y_train),\n",
    "                                           batch_size=256, shuffle=True)\n",
    "test_loader = torch.utils.data.DataLoader(torch.utils.data.TensorDataset(X_test, y_test),\n",
    "                                          batch_size=2000, shuffle=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "E[norm(x)] = 0.9703\n",
      "E[Loss(theta_0)] = 0.0900\n"
     ]
    }
   ],
   "source": [
    "print(f'E[norm(x)] = {torch.linalg.norm(X_train, dim=-1).mean().item():.4f}')\n",
    "print(f'E[Loss(theta_0)] = {y_train.pow(2).mean().item():.4f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Optimal linear predictor training MSE): 0.066701\n"
     ]
    }
   ],
   "source": [
    "with torch.no_grad():\n",
    "    # Solve min_W ||X_aug W - Y||_F^2 (handles rank-deficiency, returns min-norm solution)\n",
    "    lstsq = torch.linalg.lstsq(X_train, y_train)\n",
    "    W_opt = lstsq.solution                       # (d [+1], d_out)\n",
    "    Y_hat = X_train @ W_opt\n",
    "\n",
    "    opt_linear_mse = F.mse_loss(Y_hat, y_train, reduction='mean').item()\n",
    "    print(f\"Optimal linear predictor training MSE): {opt_linear_mse:.6f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "1COWtn9kJJ7G",
    "outputId": "79227f39-ce29-473e-c470-7ac1b438a500"
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 1000/1000 [00:25<00:00, 40.00it/s]\n"
     ]
    }
   ],
   "source": [
    "# Change this back later\n",
    "width = 3\n",
    "init_scale = 1e-20\n",
    "lr = 10\n",
    "mom = 0\n",
    "batch_size = 256\n",
    "\n",
    "train_loader = torch.utils.data.DataLoader(torch.utils.data.TensorDataset(X_train, y_train),\n",
    "                                           batch_size=batch_size, shuffle=True)\n",
    "\n",
    "ntrials = 1\n",
    "nepochs = 1000\n",
    "\n",
    "et_train_loss = ExptTrace([\"nstep\"])\n",
    "\n",
    "d_in, d_out = X_train.shape[-1], y_train.shape[-1]\n",
    "\n",
    "model = ShallowMLP(d_in=d_in, d_out=d_out, width=width, init_scale=init_scale).to(device)\n",
    "model_copy = copy.deepcopy(model.state_dict())\n",
    "\n",
    "optimizer = torch.optim.SGD(model.parameters(), lr=lr, momentum=mom)\n",
    "nsteps = 0\n",
    "for epoch in tqdm(range(nepochs)):\n",
    "    et_train_loss[nsteps] = full_error(model, train_loader)[0]\n",
    "    for batch_idx, (data, target) in enumerate(train_loader):\n",
    "        optimizer.zero_grad()\n",
    "        output = model(data)\n",
    "        loss = F.mse_loss(output, target)\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        nsteps += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|                                                    | 0/50 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.08953\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|▉                                           | 1/50 [00:11<09:38, 11.80s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.08539\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  4%|█▊                                          | 2/50 [00:17<06:26,  8.05s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.07929\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  4%|█▊                                          | 2/50 [00:24<09:38, 12.06s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "utilmax_lr = 10\n",
    "init_weights = {\n",
    "    \"W\": model_copy[\"readin.weight\"].detach(),\n",
    "    \"A\": model_copy[\"readout.weight\"].detach(),\n",
    "}\n",
    "result = train_agf(X_train, y_train, init_sz=1e-1, agf_steps=50, nonlin=nn.ReLU, load_init=init_weights,\n",
    "              utilmax_lr=utilmax_lr, costmin_lr=10, costmin_maxiter=4_000, loss_thresh=1e-4, device=device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 534
    },
    "id": "59T62Bhfrbig",
    "outputId": "48c6bb02-527c-465a-9f81-aa82ee7d7e2c"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.0\n"
     ]
    },
    {
     "data": {
      "image/png": "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3r07Fi9eDI1Gg7FjxwIARo0ahaZNm2LBggUAgMmTJ6NPnz6Ii4vDgAEDEB8fj3379uHrr782qjc3Nxdr1qxBXFycyT4TExOxZ88ePPDAA3B3d0diYiKmTp2Kp59+Go0aNaqW4yIiIiKimmV1kH3nnXcwffp0vP322wgLCzMZI+rh4WFVfTExMcjMzMScOXOQlpaG0NBQbNy40XCD1KVLl4zGdPXq1QurVq3CG2+8gddeew2tW7fG2rVr0bFjR6N64+PjIYTA8OHDTfbp5OSE+Ph4zJs3D0VFRQgODsbUqVNNZkMgIiIiorrL6nlky4fK8nOoCiEgSRJ0Ol31te4uoNFo4ObmBkDfS8xZC4iotsmyjIyMDPj4+PBmLyKqVbIs1+h1x+oe2S1bttREO4iIiIiIrGJ1kA0ODkazZs1MnmglhMDly5errWFERERERJWxuq83ODgYmZmZJsuvX79ebTeAERERERHdjtVBtnQs7K3y8vJMHjpARERERFRTLB5aUHpHvyRJmD17NlxcXAzrdDod9uzZg9DQ0GpvIBERERGRORYH2YMHDwLQ98geOXIEKpXKsE6lUiEkJATTp0+v/hYSEREREZlhcZAtna1g7Nix+Pjjj62eL5aIiIiIqDpZPWvB8uXLa6IdRERERERW4czYRERERGSXGGSJiIiIyC4xyBIRERGRXWKQJSIiIiK7xCBLRERERHaJQZaIiIiI7BKDLBERERHZJQZZIiIiIrJLDLJEREREZJcYZImIiIjILjHIEhEREZFdYpAlIiIiIrvEIEtEREREdolBloiIiIjsEoMsEREREdklBlkiIiIisksMskRERERklxhkiYiIiMguMcgSERERkV1ikCUiIiIiu8QgS0RERER2iUGWiIiIiOwSgywRERER2SUGWSIiIiKySwyyRERERGSXGGSJiIiIyC4xyBIRERGRXWKQJSIiIiK7xCBLRERERHaJQZaIiIiI7BKDLBERERHZJQZZIiIiIrJLDLJEREREZJcYZImIiIjILjHIEhEREZFdYpAlIiIiIrvEIEtEREREdolBloiIiIjsEoMsEREREdklBlkiIiIisksMskRERERklxhkiYiIiMguMcgSERERkV1ikCUiIiIiu8QgS0RERER2iUGWiIiIiOwSgywRERER2SUGWSIiIiKySwyyRERERGSXGGSJiIiI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20MLMjMz4e3tzaEFFeA1Qo/XCOvL3g3XCEvL1sWhBQyyNax8kM3Nza3ZcSJERGbIsoyMjAz4+PiYHadPRFRTZFmu0esOr2hEREREZJcYZImIiIjILjHIEhEREZFdYpAlIiIiIrvEIEtEREREdolBloiIiIjsEoMsEREREdklBlkiIiIisksMskRERERklxhkiYiIiMguMcgSERERkV1ikCUiIiIiu8QgS0RERER2ycHWDbjbCSEMv2s0GigU/OxARLVLlmXk5+fzGkREtU6WZSgUCri4uECSpGqvn0G2huXn5xt+b9KkiQ1bQkRERGQbeXl5cHV1rfZ6+dG8hmk0Gls3gYiIiMimaioPsUe2hnl5eRl+T0tLg5ubmw1bc3fRaDTw9fUFAKSnp9fIJ73aUteOpbbbU5P7q866q6OuqtZR29vR7d0t57auHQevPzVX351sX5Vty2/j4uJiVVstxSBbw8qPR3Nzc7P5BeJu5erqetec27p2LLXdnprcX3XWXR11VbWO2t6Obu9uObd17Th4/am5+u5k+6psWxPjYwEOLSAiIiIiO8UgS0RERER2iUGWiIiIiOwSgywRERER2SUGWSIiIiKySwyyRERERGSXGGSJiIiIyC5JQghh60YQEREREVmLPbJEREREZJcYZImIiIjILjHIEhEREZFdYpAlIiIiIrvEIEtEREREdolBloiIiIjsEoMsEREREdklBlkiIiIisksMskRERERklxhkiYiIiMguMcgSERERkV1ikCUiIiIiu8Qga0PZ2dkIDw9HaGgoOnbsiKVLl9q6SURUD+Xn5yMwMBDTp0+3dVOIqJ4JCgpC586dERoaigceeMDq7R1qoE1kIXd3d2zfvh0uLi7QaDTo2LEjnnzySXh6etq6aURUj8yfPx89e/a0dTOIqJ7avXs33NzcqrQte2RtSKlUwsXFBQBQVFQEIQSEEDZuFRHVJ2fOnMHJkyfRv39/WzeFiMhqDLJ3YPv27Rg4cCD8/f0hSRLWrl1rUuazzz5DUFAQ1Go1evTogb179xqtz87ORkhICAICAvDKK6/Ay8urllpPRPauOq5B06dPx4IFC2qpxUR0N6mOa5AkSejTpw+6deuG77//3uo2MMjeAY1Gg5CQEHz22Wdm169evRqxsbGYO3cuDhw4gJCQEERFRSEjI8NQpmHDhjh06BCSk5OxatUqpKen11bzicjO3ek16L///S/atGmDNm3a1GazieguUR05aOfOndi/fz/WrVuHd999F4cPH7auEYKqBQDxyy+/GC3r3r27eOmllwyvdTqd8Pf3FwsWLDBbxwsvvCDWrFlTk80kortUVa5BM2fOFAEBASIwMFB4enoKDw8P8eabb9Zms4noLlEdOWj69Oli+fLlVu2XPbI1pLi4GPv370dkZKRhmUKhQGRkJBITEwEA6enpuHHjBgAgJycH27dvR9u2bW3SXiK6u1hyDVqwYAEuX76MCxcuYOHChXjuuecwZ84cWzWZiO4illyDNBqNIQfl5eVh8+bNuOeee6zaD2ctqCFZWVnQ6XTw9fU1Wu7r64uTJ08CAC5evIgJEyYYbvJ6+eWX0alTJ1s0l4juMpZcg4iIaool16D09HQ88cQTAACdTofnnnsO3bp1s2o/DLI21L17dyQlJdm6GUREGDNmjK2bQET1TIsWLXDo0KE7qoNDC2qIl5cXlEqlyc1b6enp8PPzs1GriKi+4DWIiGyptq5BDLI1RKVSISwsDAkJCYZlsiwjISEBERERNmwZEdUHvAYRkS3V1jWIQwvuQF5eHs6ePWt4nZycjKSkJDRu3BjNmzdHbGwsRo8ejfDwcHTv3h2LFy+GRqPB2LFjbdhqIrpb8BpERLZUJ65B1k2uQOVt2bJFADD5GT16tKHMp59+Kpo3by5UKpXo3r27+Ouvv2zXYCK6q/AaRES2VBeuQZIQfCYqEREREdkfjpElIiIiIrvEIEtEREREdolBloiIiIjsEoMsEREREdklBlkiIiIisksMskRERERklxhkiYiIiMguMcgSERERkV1ikCUiIiIiu8QgS0RUx8ybNw+hoaGG12PGjEF0dLTN2kNEVFcxyBIR1XEff/wxVqxYUa113hqWbVUHEdGdcLB1A4iI7kbFxcVQqVTVUleDBg2qpR4iorsNe2SJiG7jxo0bGDlyJFxdXdGkSRN89NFH6Nu3L6ZMmWIoExQUhLfffhujRo2Ch4cHJkyYAACYMWMG2rRpAxcXF7Ro0QKzZ89GSUmJUf3vvfcefH194e7ujvHjx6OwsNBo/a1DC2RZxoIFCxAcHAxnZ2eEhITgP//5j2H91q1bIUkSEhISEB4eDhcXF/Tq1QunTp0CAKxYsQJvvvkmDh06BEmSIElShT2+W7duRffu3eHq6oqGDRuid+/euHjxYqV1ZGdn49lnn4W3tzc8PDzQr18/HDp0yFBnaU/uV199hWbNmsHFxQVDhw5FTk7ObfdLRFQegywR0W3ExsZi165dWLduHTZt2oQdO3bgwIEDJuUWLlyIkJAQHDx4ELNnzwYAuLu7Y8WKFTh+/Dg+/vhjLF26FB999JFhmx9//BHz5s3Du+++i3379qFJkyb4/PPPK23PggUL8O233+LLL7/EsWPHMHXqVDz99NPYtm2bUbnXX38dcXFx2LdvHxwcHDBu3DgAQExMDKZNm4Z77rkHqampSE1NRUxMjMl+tFotoqOj0adPHxw+fBiJiYmYMGECJEmqtI4hQ4YgIyMDv//+O/bv34+uXbviwQcfxPXr1w11nz17Fj/++CN+/fVXbNy4EQcPHsSLL7542/0SERkRRERUodzcXOHo6CjWrFljWJadnS1cXFzE5MmTDcsCAwNFdHT0bev78MMPRVhYmOF1RESEePHFF43K9OjRQ4SEhBhejx49Wjz++ONCCCEKCwuFi4uL2L17t9E248ePF8OHDxdCCLFlyxYBQPz555+G9Rs2bBAAREFBgRBCiLlz5xrtw5xr164JAGLr1q1m15urY8eOHcLDw0MUFhYaLW/ZsqX46quvDNsplUpx5coVw/rff/9dKBQKkZqaetv9EhGVYo8sEVElzp8/j5KSEnTv3t2wrEGDBmjbtq1J2fDwcJNlq1evRu/eveHn5wc3Nze88cYbuHTpkmH9iRMn0KNHD6NtIiIiKmzP2bNnkZ+fj4ceeghubm6Gn2+//Rbnzp0zKtu5c2fD702aNAEAZGRk3OaIyzRu3BhjxoxBVFQUBg4ciI8//hipqamVbnPo0CHk5eXB09PTqH3JyclG7WvevDmaNm1qdMyyLOPUqVNV2i8R1U8MskRE1cTV1dXodWJiIkaOHIlHH30U69evx8GDB/H666+juLi4yvvIy8sDAGzYsAFJSUmGn+PHjxuNkwUAR0dHw++lX8vLsmzV/pYvX47ExET06tULq1evRps2bfDXX39V2r4mTZoYtS0pKQmnTp3CK6+8UmP7JaL6iUGWiKgSLVq0gKOjI/7++2/DspycHJw+ffq22+7evRuBgYF4/fXXER4ejtatW5vcsNS+fXvs2bPHaFllga1Dhw5wcnLCpUuX0KpVK6OfZs2aWXxcKpUKOp3OorJdunTBrFmzsHv3bnTs2BGrVq2qsI6uXbsiLS0NDg4OJu3z8vIylLt06RJSUlKMjlmhUBj1dFe0XyKiUpx+i4ioEu7u7hg9ejReeeUVNG7cGD4+Ppg7dy4UCsVtbz5q3bo1Ll26hPj4eHTr1g0bNmzAL7/8YlRm8uTJGDNmDMLDw9G7d298//33OHbsGFq0aFFhe6ZPn46pU6dClmXce++9yMnJwa5du+Dh4YHRo0dbdFxBQUFITk5GUlISAgIC4O7uDicnJ6MyycnJ+PrrrzFo0CD4+/vj1KlTOHPmDEaNGlVhHZGRkYiIiEB0dDQ++OADtGnTBikpKdiwYQOeeOIJw/ALtVqN0aNHY+HChcjNzcWkSZMwdOhQ+Pn53Xa/RESl2CNLRHQbixYtQkREBB577DFERkaid+/eaN++PdRqdaXbDRo0CFOnTsXEiRMRGhqK3bt3G2YzKBUTE4PZs2fj1VdfRVhYGC5evIgXXnih0nrffvttzJ49GwsWLED79u3xyCOPYMOGDQgODrb4mAYPHoxHHnkEDzzwALy9vfHDDz+YlHFxccHJkycxePBgtGnTBhMmTMBLL72Ef/3rXxXWIUkSfvvtN9x///0YO3Ys2rRpg2HDhuHixYvw9fU11N2qVSs8+eSTePTRR/Hwww+jc+fOhtkabrdfIqJSkhBC2LoRRET2RKPRoGnTpoiLi8P48eNt3Ry7M2/ePKxduxZJSUm2bgoR2TkOLSAiuo2DBw/i5MmT6N69O3JycvDWW28BAB5//HEbt4yIqH5jkCUissDChQtx6tQpqFQqhIWFYceOHUY3LxERUe3j0AIiIiIisku82YuIiIiI7BKDLBERERHZJQZZIiIiIrJLDLJEREREZJcYZImIiIjILjHIEhEREZFdYpAlIiIiIrvEIEtEREREdun/AcjLPuTfD6jCAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 700x510 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "ax = make_plot(et_train_loss, result, utilmax_lr=utilmax_lr, lr=lr, mom=mom, fudge_fac=1.0)\n",
    "ax.axhline(opt_linear_mse, ls='--', c='k')\n",
    "ax.text(1000, opt_linear_mse + 0.001, 'Optimal Linear Predictor', ha='left', va='center', color='black', fontsize=10)\n",
    "ax.set_xlim(1e3, 1e5)\n",
    "plt.tight_layout()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running GD and AGF for width = 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|                                                    | 0/50 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.08960\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|                                                    | 0/50 [00:03<?, ?it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[]\n",
      "Running GD and AGF for width = 2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|                                                    | 0/50 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.08958\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|▉                                           | 1/50 [00:04<03:52,  4.75s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.08644\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|▉                                           | 1/50 [00:07<06:09,  7.55s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[]\n",
      "Running GD and AGF for width = 4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "GD W=4:  51%|████████████████▍               | 772/1500 [00:19<00:19, 38.06it/s]"
     ]
    }
   ],
   "source": [
    "# ---------- experiment settings ----------\n",
    "width_list   = [1, 2, 4, 8, 16, 32]\n",
    "init_scale   = 1e-20\n",
    "lr           = 10.0\n",
    "mom          = 0.0\n",
    "batch_size   = 256\n",
    "nepochs      = 1500\n",
    "\n",
    "# AGF settings\n",
    "utilmax_lr   = 20.0\n",
    "costmin_lr   = 10.0\n",
    "costmin_maxiter = 2000\n",
    "agf_steps    = 50\n",
    "loss_thresh  = 1e-4\n",
    "fudge_fac    = 1.0   # time-axis fudge for AGF overlay\n",
    "\n",
    "# ---------- dataloaders (use your existing X_train/y_train) ----------\n",
    "train_loader = torch.utils.data.DataLoader(\n",
    "    torch.utils.data.TensorDataset(X_train, y_train),\n",
    "    batch_size=batch_size, shuffle=True\n",
    ")\n",
    "\n",
    "# ---------- fallback: compute optimal linear predictor MSE if not defined ----------\n",
    "try:\n",
    "    _ = opt_linear_mse\n",
    "except NameError:\n",
    "    X = X_train\n",
    "    Y = y_train\n",
    "    with torch.no_grad():\n",
    "        # (labels already mean-centered; intercept not strictly needed)\n",
    "        lstsq = torch.linalg.lstsq(X, Y)\n",
    "        W_opt = lstsq.solution\n",
    "        mse = F.mse_loss(X @ W_opt, Y, reduction='mean').item()\n",
    "    opt_linear_mse = mse\n",
    "    print(f\"[info] computed opt_linear_mse = {opt_linear_mse:.6f}\")\n",
    "\n",
    "# ---------- helpers ----------\n",
    "def run_gd(width):\n",
    "    \"\"\"Train GD once for a given width. Returns (steps, losses, init_weights_for_agf).\"\"\"\n",
    "    d_in, d_out = X_train.shape[-1], y_train.shape[-1]\n",
    "    model = ShallowMLP(d_in=d_in, d_out=d_out, width=width, init_scale=init_scale).to(device)\n",
    "    init_state = copy.deepcopy(model.state_dict())\n",
    "\n",
    "    opt = torch.optim.SGD(model.parameters(), lr=lr, momentum=mom)\n",
    "\n",
    "    steps, losses = [], []\n",
    "    nsteps = 0\n",
    "    for epoch in tqdm(range(nepochs), leave=False, desc=f\"GD\"):\n",
    "        # log train MSE at epoch boundary\n",
    "        with torch.no_grad():\n",
    "            loss_epoch = full_error(model, train_loader)[0]\n",
    "            losses.append(float(loss_epoch))\n",
    "            steps.append(nsteps if nsteps > 0 else 1)  # avoid 0 on log-x plots\n",
    "\n",
    "        for data, target in train_loader:\n",
    "            opt.zero_grad(set_to_none=True)\n",
    "            out = model(data)\n",
    "            loss = F.mse_loss(out, target)\n",
    "            loss.backward()\n",
    "            opt.step()\n",
    "            nsteps += 1\n",
    "\n",
    "    init_weights = {\n",
    "        \"W\": init_state[\"readin.weight\"].detach(),\n",
    "        \"A\": init_state[\"readout.weight\"].detach(),\n",
    "    }\n",
    "    return np.array(steps, dtype=float), np.array(losses, dtype=float), init_weights\n",
    "\n",
    "def run_agf(init_weights, width):\n",
    "    \"\"\"Run AGF from the same init (shapes are inferred from init_weights).\"\"\"\n",
    "    res = train_agf(\n",
    "        X_train, y_train,\n",
    "        init_sz=1e-1, agf_steps=agf_steps, nonlin=nn.ReLU,\n",
    "        load_init=init_weights,\n",
    "        utilmax_lr=utilmax_lr, costmin_lr=costmin_lr,\n",
    "        costmin_maxiter=costmin_maxiter, loss_thresh=loss_thresh,\n",
    "        device=device,\n",
    "    )\n",
    "    # Time scaling to align AGF's utility-time with GD steps\n",
    "    f = utilmax_lr / (lr / (1 - mom)) * fudge_fac\n",
    "    x = f * np.array(res[\"t\"], dtype=float)\n",
    "    y = np.array(res[\"losses\"], dtype=float)\n",
    "    return x, y\n",
    "\n",
    "# ---------- run all widths ----------\n",
    "all_gd, all_agf = {}, {}\n",
    "max_step = 0.0\n",
    "\n",
    "for W in width_list:\n",
    "    print(\"Running GD and AGF for width = {}\".format(W))\n",
    "    t_gd, l_gd, init_w = run_gd(W)\n",
    "    all_gd[W] = (t_gd, l_gd)\n",
    "    max_step = max(max_step, float(t_gd.max()))\n",
    "\n",
    "    x_agf, y_agf = run_agf(init_w, W)\n",
    "    all_agf[W] = (x_agf, y_agf)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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iMgxKQS/C5gsoNFo4OnpCQAoKyuDl5eXjTO6swghoNFooFKp7PIrzi1h78dmq/xaYtzmGMOafVral7nttVotSkpK0L59ezg52cUlbHcMe58PLGHvx2bLuc4e3w9HwRmKiIiIiEhGWOATEREREckIC3wiIiIiIhlhgU9EREREJCMs8ImIiIiIZIQFPhERERGRjLDAJyIiIiKSERb4REREREQywgKfiIiIiEhGWOATEREREckIC3wiIiIiIhlhgU9EREREJCMs8ImIiIiIZIQFPhERERGRjLDAJyIiIiKSERb4REREREQywgKfiIiIiEhGWOATEREREckIC3wiIiIiIhlhgU9EREREJCMs8ImIiIiIZIQFPhERERGRjLDAJyIiIiKSERb4RERERC1o+fLlCA8Pt/txli9fjoiICOsl5MBufS+nT5+O+Ph4m+XTFBb4RERERA38/vvvePzxx9GhQwe4uroiODgY8+bNw5UrV0zuS6FQYMeOHTrPLVq0COnp6VbK1nx5eXlQKBTIyspq9PVFixZh7969LZuUCZYvXw6FQgGFQgEXFxeEhIRgwYIFUKvVzT72+vXrkZKSYlRsU+9zc2CBT0RERPSX3377DZGRkfj111+xZcsWnD9/Hu+//z7S09MRHR2Nq1evWjyGp6cnfHx8rJBt87KXPKurq/W+dvfdd6OoqAh5eXl4/fXX8cEHH2DhwoUm92Oq1q1bo02bNlbrz1g1NTVGxbHAJyIiIvrLU089BVdXV3z33XeIjY1Fp06d8OCDD2Lv3r34448/8MILL0ixISEhePnllzFp0iSoVCp07NgR7777rs7rADB27FgoFApp/9blHrNnz8bYsWOxatUq+Pv7o02bNlixYgVqa2uxePFitGvXDkFBQUhOTtbJdcmSJQgLC4OHhwdCQ0ORlJRkdAFojFuX6NQvS1mzZg0CAwPh4+ODp556SmfMqqoqLFq0CB07doRKpcKAAQNw4MAB6fUrV65g0qRJ6NixIzw8PNC7d29s2bJFZ9whQ4Zg7ty5mD9/Pnx9fTF8+HC9Obq4uCAgIABBQUGYMGECpkyZgp07d0r5h4eH46OPPkLnzp2hVCoBANeuXcMTTzwBPz8/eHt7Y+jQoThx4oROv6+99hr8/f3h5eWFmTNnorKyUuf1W5foaLVarF69Gl27doWbmxs6deqElStXAgA6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RGRVI0aMwO7du6X93NxchISE2C4hInJ4PINPdiEhIQHfffcdysrKUFBQgJ9++gnnz59HXl4exo4di+rqajz++OO4ePGirVMlIrKa5ORk7N69G2PHjrV1KkQkIzyDT3avsrISgYGBuHbtGv71r3/hySeftHVKREQWKyoqQq9evdCmTRt8/fXXuOeeewDwDD4RWY5n8MnuKZVKhIaGAgA0Go2NsyEiso45c+bg2rVr2LBhA1Qqla3TISIZYYF/h6irq8PJkyfx8ccfY86cOYiMjISrq6t0UdeQIUPM7ru6uhqbNm3CyJEjERwcDKVSicDAQAwcOBBr1qzB5cuXLcr98uXLOHPmDAAgKirKor6ISN4cZa7bsmULvvrqK0ydOhUPPPCA2TkRETVKkOxt375deHh4CAB6H7GxsWb1nZOTI8LDww323b59e5GWlmZy3yUlJeKbb74RERERAoCYPHmyWTkS0Z3BUea6kpIS4evrK3x9fUVpaakQQojc3Fypn9zcXLNyJCKq59ICf0OQjV27dg3l5eVW7/fixYuIi4tDYWEhAEChUGDw4MHo0qULSktLsXfvXlRUVKCkpATx8fHYtWsXhg4darDPHTt23HaxWefOnbFhwwb8/e9/t/oxEJF8OMpcN3fuXFy+fBn//ve/4evra/V8iYhY4N9B/P39ERUVJT12796N9evXm93f5MmTpf/wgoOD8dVXX6FPnz7S65cvX8bEiRORnp6OmpoajB8/HhcuXECbNm309unj44OYmBhotVoUFhbi4sWLyMvLw6efforBgwejR48eZudLRHcGe57rduzYgdTUVIwYMQJTpkwxOyciIoNs/RECNb+ioiKRn59/2/PLli0z+2PrtLQ0qa2rq6s4efJko3FqtVqEhoZKsc8//7xJ4xQWForHH39cABDe3t4iLy/PpPZEdOew97nu6tWrIiAgQKhUqtvmMi7RISJr4kW2d4CAgAB06tTJqn2+++670nZCQgJ69+7daJxKpcKKFSuk/Q0bNqC2ttbocQIDA/Hxxx/jgQceQFlZGVauXGl+0kQka/Y+1y1evBiXLl3CK6+8guDgYKvmSUTUEAt8MplarUZ6erq0P2PGDIPx48aNg6enJwDg6tWr+OGHH0we8+GHHwYAZGZmmtyWiMgc1p7r6uevVatWISAgQOfR8A5hUVFRCAgIwLx586x1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",
      "text/plain": [
       "<Figure size 760x530 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ---------- plot ----------\n",
    "fig, ax = plt.subplots(1, 1, figsize=(7.6, 5.3))\n",
    "cmap = plt.get_cmap(\"tab10\")\n",
    "\n",
    "for i, W in enumerate(width_list):\n",
    "    color = cmap(i % 10)\n",
    "\n",
    "    # GD curve (solid)\n",
    "    t_gd, l_gd = all_gd[W]\n",
    "    ax.plot(t_gd, l_gd, lw=2, color=color, label=f\"Width = {W}\")\n",
    "\n",
    "    # AGF curve (dashed, extended flat to plot end)\n",
    "    x_agf, y_agf = all_agf[W]\n",
    "    order = np.argsort(x_agf)\n",
    "    x_agf, y_agf = x_agf[order], y_agf[order]\n",
    "\n",
    "    if x_agf[-1] < max_step:\n",
    "        x_agf_ext = np.concatenate([x_agf, [max_step]])\n",
    "        y_agf_ext = np.concatenate([y_agf, [y_agf[-1]]])\n",
    "    else:\n",
    "        x_agf_ext = np.concatenate([x_agf, [x_agf[-1] * 1.000001]])\n",
    "        y_agf_ext = np.concatenate([y_agf, [y_agf[-1]]])\n",
    "\n",
    "    ax.step(x_agf_ext, y_agf_ext, where=\"post\", lw=2, ls=\":\", marker='.', color=color)\n",
    "\n",
    "# Optimal linear predictor line\n",
    "ax.axhline(opt_linear_mse, ls='--', c='k', lw=1.5)\n",
    "ax.text(max_step * 0.8, opt_linear_mse * (1 + 3e-3), 'Optimal Linear Predictor',\n",
    "        ha='left', va='bottom', color='black', fontsize=10)\n",
    "\n",
    "ax.set_xlim(1e3, 5e4)\n",
    "ax.set_xscale(\"log\")\n",
    "ax.set_xlabel(\"gradient steps\")\n",
    "ax.set_ylabel(\"training loss (MSE)\")\n",
    "ax.legend(ncol=1, fontsize=9)\n",
    "ax.grid(True, which=\"both\", alpha=0.15)\n",
    "try:\n",
    "    style_axes(ax)  # optional, if you defined this elsewhere\n",
    "except Exception:\n",
    "    pass\n",
    "plt.tight_layout()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "accelerator": "GPU",
  "colab": {
   "gpuType": "T4",
   "provenance": []
  },
  "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.9.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
