{
 "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": "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": 4,
   "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": 5,
   "metadata": {
    "id": "E1x3SMr_I92L"
   },
   "outputs": [],
   "source": [
    "def style_axes(ax, numyticks=5, numxticks=5, xlabels=True, ylabels=True):\n",
    "    \n",
    "    # plt.rc(\"font\", family='stixgeneral', size=14)\n",
    "    # plt.rc(\"mathtext\", fontset='cm')\n",
    "    \n",
    "    ax.tick_params(axis=\"y\", which=\"both\", bottom=True, top=False,\n",
    "                   labelbottom=ylabels, left=True, right=False,\n",
    "                   labelleft=True, direction='out', length=7, width=1.5, pad=8, labelsize=24)\n",
    "    # ax.yaxis.set_major_locator(plt.MaxNLocator(numyticks))\n",
    "    ax.tick_params(axis=\"x\", which=\"both\", bottom=True, top=False,\n",
    "                   labelbottom=xlabels, left=True, right=False,\n",
    "                   labelleft=True, direction='out', length=7, width=1.5, pad=8, labelsize=24)\n",
    "    # ax.xaxis.set_major_locator(plt.MaxNLocator(numxticks))\n",
    "    ax.xaxis.offsetText.set_fontsize(20)\n",
    "\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(3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "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": 7,
   "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": 8,
   "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": 9,
   "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",
    "            break\n",
    "\n",
    "    return results"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "st2oZU8cI1X3"
   },
   "source": [
    "# Main expt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "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": 11,
   "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": 12,
   "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": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running GD and AGF for width = 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "GD: 100%|███████████████████████████████████| 2000/2000 [00:49<00:00, 40.67it/s]\n",
      "AGF:   0%|                                                | 0/1 [00:03<?, ?it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running GD and AGF for width = 2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "GD: 100%|███████████████████████████████████| 2000/2000 [00:50<00:00, 39.59it/s]\n",
      "AGF:  50%|████████████████████                    | 1/2 [00:07<00:07,  7.05s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running GD and AGF for width = 3\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "GD: 100%|███████████████████████████████████| 2000/2000 [00:50<00:00, 39.55it/s]\n",
      "AGF:  67%|██████████████████████████▋             | 2/3 [00:12<00:06,  6.49s/it]\n"
     ]
    }
   ],
   "source": [
    "# ---------- experiment settings ----------\n",
    "width_list   = [1, 2, 3]\n",
    "init_scale   = 1e-20\n",
    "lr           = 10.0\n",
    "mom          = 0.0\n",
    "batch_size   = 256\n",
    "nepochs      = 2000\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",
    "# ---------- 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), 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=width, 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": 16,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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l112mfTve++9h/z8fCnIb0lg8fbbb8NsNuO2226TggoAUKvVePnll/Hxxx+3aTjj/fffLwUVgKsH84EHHsD27duxZcuWoAMLdyDVEiNHjmzxNd5qa2ul9t1///3IyMhoU3kUftKj02WBxem6053YGiKi8BLSwGLevHn461//ikOHDuGnn37C1KlTffI8+uijfq/duHEjGhoapOCDiFouJycHzz77rGyoU15eHgYOHIiUlBQAngAjLy8PkyZNwrFjx3Du3DkkJSXhggsuaLYOd6Ayb948n8eSkpIwffp0rFu3rtW/w/Tp033SBg4cCAA4d+5c0OUsXrwYixcvbnU7WsrhcOCWW27BkSNHMGbMGDz11FMdVjd1nL5xfbGvzLNPybHqY53YGiKi8BLSydsqlQrr1q1Djx49IIqi7AcAfve730nfojb27rvvSseTJ08OZbOIuo3x48dDoVBg9+7daGhoQFlZGQoKCmTf3g8bNgyxsbFS8OEOFC6//PKgNsNzT0jOysry+3igCdnB8tejGRMTAwCwWCxtKrs93Xnnnfj8888xaNAgfPHFF9BoNJ3dJGoHfePke8QcrzneSS0hIgo/Ie2xAFzfLP700094++23sXXrVtTW1qJXr174xS9+4febSACoqKjA999/j6ysLMTExGDs2LGhbhZRtxAXF4fhw4dj79692LlzJ6qrqwHIhwUpFAqMHTsWO3bsgMPhaPeN8VpK0co5Jo299dZbATcDDGT27NmYPXt2i+t69NFH8eabb6JXr1749ttv/U5Kp8jQL14+zPJ4zXFp41ciou4u5IEFAERHR+Oee+7BPffcE1T+pKQkv8tUErVKVKJrrkJXFJXY5iJycnKwd+9e5OXl+Q0sANdwqG+++Qb79u1r0fwKAMjIyMChQ4dQWFiIIUOG+DzuXla2s+Xl5bV48nZ2dnaLA4tly5bh+eefR2pqKr799lv06tWrRddT19K4x6LWWosKcwWSoxhMEhG1S2BB1KkUijZNgO7qcnJy8Pe//x15eXmoqqpCWloaBgwYIMvjnmexdu1aHD58GAaDAaNGjQq6/NzcXKxZswZXXXWV7LHKykp88803fq9zDw2y2+1Qqdr/rWfFihXNrkDVVm+++SYeeeQRxMfH4+uvv+bCE11Yg70BqwtWy9LmDZ6HKJV8eePM6EzolDrZrtzHqo8xsCAiQojnWBBR53P3PGzfvh179uyRgghvl156KZRKJV599VUAwNixY4P+sL9w4UJotVqsXLkSGzZskNJtNhvuv/9+GI1Gv9e5l5E+dOhQi36fcPXhhx/iN7/5DaKjo7F+/fo2ryhFnavB3oCXdr0k+2mwN/jkUwgK9InrI0s7Wn20o5pJRBTWOqTH4vDhw9i9ezfKy8tRV1eHmJgYJCcnY9SoUdJqL0QUGunp6ejfvz+OHnV92PG37Gp0dDRGjBiB3bt3A2jZ/Io+ffrgxRdfxG9/+1vMmDEDEyZMQHp6Onbu3ImqqirccsstWLlypc911157LTZv3owpU6Zg8uTJMBgMSE5OxnPPPdfK37TzlJaW4pZbboHT6USfPn3wxhtv4I033vDJ19o5GxTeBiQMwM+Vno1gfyz7EbdccEsntoiIKDy0W2BRW1uLV155Ba+//nqTS0RmZmbiN7/5DX73u98hNja2vZpD1K3k5OQ0GVgAruFQrQksAODuu+9Gjx498NxzzyE/Px86nQ4TJkzAc889h9WrV/u95p577kFVVRVWrVqFjz76CDabDVlZWV0ysDCZTLBarQCA/fv3Y//+/X7ztWbOBoW/kakj8emxT6XzvaV7OYGbiAiAILrXgg2hHTt2YP78+Th9+jSCKV4QBPTu3RurVq3iilDdVFlZGVJTU2VppaWl0t4L/pjNZpw4cQJ9+vSBTqdr7yZSK4iiCKfTCYVC0e0/dDV1L7rCa1kURRiNRhgMhrB5LkPZpjprHR7Z8ogs7fkJzyNGE+OT90jVEcz5dI4s7cNrPsSgxNDPsWmv+x7KckNVVji+xqj1+Hx6dKd7EfIei127dmHGjBmycdYKhQIDBw5EdnY2DAYDjEYjTp48icOHD8PpdAJwrSQzffp05ObmBj2JlIiIKBRiNDF4beprQeXtH98fGYYMnDN6euO/KfymXQILIqKuJKSTt+12O26++WbU19dDFEXExsbi+eefR3FxMX766SesX78eH3zwAdavX4+ffvoJJSUlWLZsGeLi4iAIAurr63HzzTfD4XCEsllEREQhIwgCpmVNk6V9c/KboHroiYgiWUgDi5UrV+LIkSMQBAH9+vXD3r178fDDDwfcLCopKQkPPfQQ9uzZg379XJsOHTlyxO/ETyIionDROLA4WXsSO8/t7KTWEBGFh5AGFp988ol0/P777yMrKyuo67KysrBq1Spp3NnHH38cymYRERGF1LDkYehh6CFL+9e+f3VSa4iIwkNIA4vdu3dDEARceumlLZ4ncfHFF+PSSy+FKIrYs2dPKJtFREQUUgpBgVsH3ipL+6HkB2w+vbmTWkRE1PlCGliUlpYCAIYMGdKq693XucshIiIKVzOzZiJVL1/N7vnvn4fFYemkFhERda6QBhZqtRoAYLG07k3VvS68uxwiIqKOYLabsapglezHbDc3eY1GqcE9F90jSztddxr/3v/v9mwqEVHYCulys+np6Thy5Ajy8/Nbdb37uvT09FA2i4iIqEkmuwl/zf+rLG1G9gzoVE3vK3JN32vw0ZGPsKfUM4T3X/v+hTHpY3BJ+iXt0lYionAV0h4L9w6/x44dwwcffNCiaz/88ENpRalAOwUTERGFE0EQ8MdL/wiF4Plz6hAd+O2m32JXya5ObBkRUccLaWAxd+5c6fhXv/oVvvrqq6Cu+/bbb7Fo0SK/5RAREYWzwYmD8ZsRv5GlGW1G/Obb32B70fZOahURUccLaWAxffp0XHHFFRBFEfX19Zg1axbmzJmDzz77DBUVFbK8lZWV+Pzzz3HjjTfiyiuvRH19PQRBwBVXXIHp06eHsllERERNUilUuCTtEtmPShH8aOElw5Ygp0eOLM3sMOPujXdj+YHlcIrOUDeZiCjsCGKItwotKyvDuHHjcPz4cVcF5/emAICoqCgYDAYYjUY0NDRI6e4m9O/fH9u2bUNKSkoom0RdQFlZGVJT5aurlJaWNvlaMJvNOHHiBPr06QOdrulx0NQ5RFGE0+mEQqGQvRd0R03di67wWhZFEUajEQaDIWyey85sk7+6rQ4rHtr8EL47/Z1P/tHpo/HXy/+KdEPL5hC21+8YynJDVVY4vsao9fh8enSnexHSHgsASElJwfbt23HllVcCcN1M94/JZEJZWRlMJpMsHQCuuuoq5OXlMaggIqIuSaPU4MVJL+KqPlf5PPZ98feY88kcvPfze82uNkVE1FWFPLAAgNTUVKxfvx4bN27E3Llzfb6J9s43d+5cbNq0CV988UXAfETUtD179kAQBPTo0cPv406nE/Hx8RAEAUuWLPGbZ8uWLRAEAUOHDpXSsrOzW/ztyhNPPAFBELBixYoWXTdp0iQIgoCTJ0+26LrOsmXLFtxxxx0YNWoU0tLSoNFokJiYiMmTJ+Pdd99FiDuDqYtQK9R49vJnsXjYYgiQ/9+ps9Xh2f89ixkfzcBb+99CrbW2k1pJRNQ+QrrcbGOTJ0/G5MmTAQBFRUUoKytDfX09oqOjkZKSgszMTFn+zz//HJWVlQCA2267rT2bRhRRRowYgdjYWBQVFeH48ePo27ev7PH9+/ejpqYGAJCXl+e3jK1btwIAcnJy/D7eVtnZ2SgsLIyYD9yffvop3nrrLQwcOBAXXXQREhIScPbsWWzduhW5ubn48ssv8d5773V2M6kTKBVK3DvqXozPHI8/5f0J54znZI9Xmivx8u6X8a99/8KsvrMwf/B8DEwY2EmtJSIKnXYNLLxlZmb6BBKNPf7449i3bx8ABhZELaFQKDB+/Hh89dVXyMvL8wks3EHDiBEjsG/fPlRUVCApKclvHu/AYuPGjbDZbO3c+q5p0aJFeOCBB3ze144ePYoJEyZg1apVuPnmm3H11Vd3Ugups41OH40Pr/0Qf9n5F6w/sd7n8QZ7Az48/CE+PPwhBiQMwKSekzCp1yRcmHyhbPlaIqKuIuzeuSLl20yijuYOCPz1SOTl5UGtVuO+++6DKIrYtm2b7HGn04kdO3bIygGAfv36YfDgwe3Y6q5ryJAhfr8s6d+/P+666y4AwKZNmzq6WRRmYjWxeH7C83h7xtsYnzk+YL4jVUfw5v43ccv6W3DFmivw5+1/xqZTm2CymTqwtUREbRN2gQURtY47IHD3PHjbunUrRo0ahWnTpvnN8+OPP6K2tha9e/dG7969pfSm5lh8+umnGDduHPR6PZKSknDDDTfg8OHDPvlyc3OhUChQWFgIwLVSnPsnOzvbb9nr1q3D2LFjYTAYkJiYiPnz5+PMmTPN34QwoVarAQAajaaTW0LBsjgsWHd0nezH4rCErPzR6aPxxrQ3sPrq1ZiRPQNKQRkwb4W5AmuPrMW9392LiWsm4sFtD+LDwx+i2FjML9+IKKx12FAoImpfY8aMgVarxaFDh1BeXo7k5GQAwPHjx1FUVIT58+ejR48eyMrK8unVaOn8itdffx133nknBEFATk4OMjIysHPnTowZMwbXXHONLG96ejoWLFiADz/8EEajEQsWLJAec7fR22uvvYaXXnoJOTk5mDlzJvLz87F69Wrs2rULP/74I6Kiolp0Xzra6dOn8frrrwMAZs6c2cmtoWAZbUY8vu1xWdqEnhOgVWpDWs/QpKF4YeILKDGW4MMjrmFQ5Q3lAfNbHBZsL96O7cWujfYSdYnoF98PgxMH48KkCzE4aTB6RfeCWqkOaTuJiFqDgQVFHKfoRLWlurOb0Srx2vhWj63WarUYPXo08vLysG3bNlx33XUAPEOjLr/8cgDAZZddhg8//BANDQ3Sh3R3nmACi8LCQtx///1Qq9X47LPPMGPGDACAzWbDwoUL8d///leWf/DgwVi+fDlyc3NhNBqbXS3q1VdfxdatWzFu3DgAgMlkwrRp07B9+3asWrUKixYtCup+3H777fjPf/4TVF63P//5z3jiiSdadM2OHTvwxhtvwOFwoKioCHl5ebDb7XjmmWcwYcKEFpVF3UeaIQ13j7wbvxn+G+wr34fvTn+Hzac343jN8SavqzRXorK4Et8Xfy8vT5+GdEM6hiUPw4jUERiZMrLFe2YQEbUVAwuKONWWakx8f2JnN6NVNs/djERdYquvz8nJQV5eHrZu3SoFFu7eiMsuu0z697333kN+fj4mTZoEoGWBxdtvvw2z2YzbbrtNCioA1/Cfl19+GR9//DFMptaPC7///vuloAIA9Ho9HnjgAWzfvh1btmwJOrBwB1ItMXLkyBZfc+zYMVkAo1Qq8dRTT+Ghhx5qcVnU/SgVSlyUehEuSr0ID1z8AE7VnkLu6VzknsnF7pLdcIiOoMopMZWgxFSCH8t+xH9/dgX36YZ0XJh0IXrG9ETP6J7oEdMDPaJ7IDM6ExoFh+kRUegxsCCKIDk5OXj22WdlQ53y8vIwcOBAafNJd4CRl5eHSZMm4dixYzh37hySkpJwwQUXNFuHO1CZN2+ez2NJSUmYPn061q1b1+rfYfr06T5pAwe6luI8d+6cz2OBLF68GIsXL251O4J166234tZbb4XVasXJkyfxzjvv4KmnnsJnn32GL7/8EgkJCe3eBmo7paDE0KShPmkdrXdsb9w29DbcNvQ21FhqsPXMVmw6uQk7S3aizlbXorKKjcUoNhb7fSxJl4R0fTp6x/ZGZnQmsmKzkGZIQ9+4vkiOSoZKwY8HRNRyfOcgiiDjx4+HQqHA7t270dDQgPr6ehQUFMi+5R82bBhiY2Ol4MMdKFx++eVBbYZXVFQEAMjKyvL7eKAJ2cHq2bOnT1pMTAwAwGIJ3WTaUNNoNBg4cCCeeeYZJCYm4sEHH8TSpUvx97//vbObRkGI08Zh9dWrO7sZMnHaOMzqOwuT0iZBG6XF8ZrjKKwtxLHqYzhQcQAHyw+iwlzRqrIrzBWoMFfgYOVBn8dUChUMagNi1DFIN6RDqVAiUZeIrNgsxGvjoVaokRmdieSoZFgdVsRr4xEnxLX11yWiCMDAgiiCxMXFYfjw4di7dy927tyJ6upqAPJhQQqFAmPHjsWOHTvgcDjafWO8llIoQrNY3VtvvRVwM8BAZs+ejdmzZ7e57l/+8pd48MEH8cknnzCwoJBQKVQYlDgIgxIHydKrzFUoqCxAjbUGhTWF2Fu2Fz+W/Yg6a8t6N7zZnXbUWGpQY6nBmfrgVmNTK9RIiUpBtCYaepUeGqUGOpUOafo0JEUlIV4bj6zYLOhVekRropESlYJYTSyUio7vFSKi9sPAgiJOvDYem+du7uxmtEq8Nr7NZeTk5GDv3r3Iy8vzG1gAruFQ33zzDfbt29ei+RUAkJGRgUOHDqGwsBBDhgzxedy9rGxny8vLa/Hk7ezs7JAEFomJiVAoFCgrK2tzWURNSdAlYFzmOFmaU3TiRM0J/Fj2I07WnsTZurM4W+/6aa+FLWxOG4qMRYCxZdcpBAV6x/RGj+geiNXEIkYZg9SYVERromFQG6BSqJAclYwEbQIMagMMagP0aj00Ck1QPaxE1LEYWFDEUQiKNk2A7upycnLw97//HXl5eaiqqkJaWhoGDBggy+OeZ7F27VocPnwYBoMBo0aNCrr83NxcrFmzBldddZXsscrKSnzzzTd+r3Pv6WC326FStf9bz4oVK5pdgaq9bN26FU6nE/369euU+ql7UwgK9Ivvh37xvq+/ems9ztafxTnjORSUFqDUWopTdadQY6lBYW0hGuwNHdpWp+jEydqTOFl7skXXKQUlolRR0o9erUdSVBKqzFVI0CUgKyYLKfoU2Jw2AMCA+AEQISJaHY1EXSLitHGI08ZBp9QxQCEKoVb9dX/qqadC3Q4AQHGx/0lmRBQ8d8/D9u3bYTabce211/rkufTSS6FUKvHqq68CAMaOHRv0h/2FCxdi2bJlWLlyJW655RZMnToVgGu52fvvvx9Go/+vLDMzM3HkyBEcOnQIQ4cO9ZunK/m///s/LF682Gdy9vfff4877rgDgOteEYWTaE00BiUOwsCEgbgk4RIYDAbpg7VTdKK8oRwlxhLUWetw1ngWRqsRVRbXcCu1Qg2T3YQqcxVO150O6QaCLeUQHai31aPeVu/38W3YFlQ5UaooqBVq1FprpbRRqaOgVChRb62HRqlBclQyesf0hsVhQZw2Dkm6JGiUGticNmTHZiNVn+paaUvJlbaIWhVYPPHEE4zwicJUeno6+vfvj6NHjwLwv+xqdHQ0RowYgd27dwNo2fyKPn364MUXX8Rvf/tbzJgxAxMmTEB6ejp27tyJqqoq3HLLLVi5cqXPdddeey02b96MKVOmYPLkyTAYDEhOTsZzzz3Xyt+0c/3+97/HY489hosuugjZ2dmwWq04fvw4fvzxRwDAL37xC9x7772d3Eqi4CkEBVL1qUjVpwaV370LeJW5CgWlBTALZpjsJtRZ63DOeA61FteH9bKGMpypO4M6ax0a7A0w2owQER47iDfYG9AAeS/N7tLdrS4vWh0tBTspUSnQq/XoGd0T2XHZOFp1FIm6RDhEB+K18UiOSkaqPhXxunhEq6NxQdIFMKgMsIv2kG/MSNRRWj0ewf2GQkThJycnp8nAAnANh2pNYAEAd999N3r06IHnnnsO+fn50Ol0mDBhAp577jmsXu1/ZZ177rkHVVVVWLVqFT766CPYbDZkZWV12cDi73//O7777jvs3bsXBw4cgM1mQ0pKCq677jrcfvvtIZmrQR3H6rDiu9PfydIm95rMb6Gb4P6CMUGXgOFJw2W9H02xO+0oMZWg3lqP8oZyHK85jlO1p1BtqYbRYkS9vR4QXB/6Kxoqwi4YaYp3D0pZQxnQABTWFmJbUXA9KMESIODK7CsRq43FwfKDECHi4rSLkR2XjUOVh3BB4gW4JP0SpES5lhlXCAooBAWsDivqbfWI0cTAoDZAFEXU2eoQq4kNafuo+xLEVkQIkyZNavcei++++675TBQxysrKkJoq/5astLRU2nvBH7PZjBMnTqBPnz7Q6XTt3URqBVEU4XQ6oVAoun0vZ1P3oiu8lkVRhNFoDPrDY0cIZZsqzZU+G2s2tWFlR92P9qonlOWGqqymynGKTpjtZhhtRpjsJldPg70BDbYGGO1GFNUXodpSjWpLNQ5XHoZKoUK0JhpO0YlqczUqzBUw282otdYGvekg+eoV0wtjM8ZiQs8JqDRXIlGXiEEJg3Cy9iT6x/eHRqmRBSn19fXYVb0Luadz8ZsRv2n1HimiKAb92mpJ3o4Sju+f7aVVPRa5ubkhbgYRERGRfwpBAb1aD71a3+ayzHYzyhrKYLKZXPM0rPU4VXcKhbWFSNOnwWQ3QSkooVVqcaT6CL468RUSdYlQCkqUNpRCrVBLk8K7m9N1p3G67jQ+OPxBi6/96MhHIWtH37i+mDNgDsZnjkdyVDLitfF4Zc8reGv/WwCAd696FyNTRzZbjlN0YmfRTnx58ktsObMFAPDomEdxVZ+rYLKZUG2pRrohXert+fz45xAg4Op+V0OtULeq7TaHDXbRjihVVKuuD3et6rEgCjX2WEQm9lh4sMci9NhjER7ldkSPRbgy2ow4VXsKJ2tPos5ahyhVFM4Zz+FkzUnsLt2NBnsDJvacCLPDjJ8qfkJhbXgsx01tNyx5GAYkDEC8Nh5RqihpVTK9So9pWdOQqk/FmfozSNen45zxHPYW7YVRNOKv//srACBJl4T3r34faYa0gHU4RSc+OPQBdpzbgUszLsXcQXMhiiJ+LPsRibpEZMdlB7zW4XRg69mtUApKXN4j8Aa4oiji68Kvcaz6GGb2mYk+cX3adF+43CwREXV7SkGJfnH9fNKImmJQG3BB0gW4IOmCVl3vcDpgc9pQb6tHmakMZQ1lOFlzEr1iemHzmc04U3cGu0p3YWLPiRBFEZtObwIApOnTUGut7fDlgcljf/l+7C/f7/ex579/vtnrK8wVmPrh1KDr23hqI/6a/1ef9ChVFOK0cbiu33X4+OjHKDWV+uS5rMdlmD9oPmqttfjyxJfYW7bX7yaar//4Ot656h0kaBOQFZvVqgCfPRYUFthjEZnYY+HBHovQ68w2scci9GWF42usq3DPK3APs3GKThTWFkIhKHC46jDO1p1FQWUB4rRxOF13Gin6FOSezmVgQkHbv8B/ENUYeyyIiIiIujB3IKZWqqGGa+z/kKQhAIDBiYNbXa47YBFFEQ7RAZVChQZ7A07VujZVrLHWwCk6sfXMVhSbihGniUOaIQ3v/vRut56L0p0xsCAiIjqv2FiMU7WnkGHIgEFjkNIVUCBeF+/3GpvDhjqbZ1iBAAEJugT/eZ02nyEIgeZx2J121FhqYLKYYFFaIAhCwLwOpwM11hpZWoI2we83/07RiSpzlazceG08FILCJ68oiqiyVMnS4jRxUCr8DxOrNFfKzmM1sQFXAWqcN1od7Tcf4Norw3u52Rh1DNRK/5Nnq83VcMIpKzfQssE1lhrZKlEtyWtQGwLuN1FrrYXdaZfO9So9dCr/vZF11jrZB3D3buLB5NUpdQEntNdb62F1WoPKa7QZZRseapVaGNSeniNBEKASXM+jKIpI0acgRe8aUaBRaDAje4asvN+P/j1EUUR5TTkUWk8vrVqhRowmxm8bGuwNqLPWQSkopSFiWpUWtZZapBvSUWoqRbGpGJtPb8a2s9uQFZuFImMRzhnP+R3WQ6H1v3P/w5iMMc3mY2BBREQEYO2RtXhyx5Nwik4IEGQfZDMNmfj6xq/9Xre3bC8Wfb1IOk/QJmDLvC1+8x6qPIT5X8yXzqNUUfjfLf/zm/dkzUlc/+n1srRAwxGKTcW48qMrZWn5N+f7/SBZbanGpDWTZGmBJqo32Bt8JrV/dcNX6BHdw287Guf9+NqP0T+hv9+8V350pWwoznsz30OfKP8TR69bd50swHl7xtsYnT7ab965n89FkbFIOn91yquY0HOC37wLvlyAYzXHpPMXJr7g8yHZ7dff/hoHKw5K509f9jRm95/tN++9m+7FDyU/SOd/vPSPmD94vt+8j2x5BFvPbpXOH7j4ASy8cKHfvH/e/md8W/itdH7niDtx18i7/OZ99n/P4tNjn0rntw+9HQ9e8qDfvP9v1//D+4fel87nDpqLx8Y+5jfvP3/8J1YcXCGdX9vvWvzl8r/4zbvy8Er8++d/S+fTsqbhpUkv+c27umA1XtrleSynRw5em/oakqOSAQDZcdnIjsvG2IyxWFWwSjbf4JK0S7D8yuV+y113dB0e3/a4dD40aShWX+2735LVYcWHhz/Es/97VkrrEd0Dr015DRaHBbmnc7GrdBfO1J1BlCoK9dZ6FJuK/dYZiX71za+CGg7FwIK6PE4Toq6Or+HOV2wsloIKAF1iMzYiCh2NUoOkqCRZmk6pQ9/4vgDgM0F/y5ktuHvj3dJ5U18+/O/c//Crb34lncdr4/HOVe8gXhuP3NO5iNfGI0GXgB9KfsC5+nNYc3iNTxneu7qHMwYW1GUpFK5ue4eDmx1R1+Z+Dbtf09TxTtWekoIKIqL2JECQlnW9foCnV3Jk6kgcKD8gCyya6tU8WnU0qF5Nm9OGEmMJrlp7lSy9ca+mKIrYVbILZrsZd268s+W/GBhYUBemVquhVCrR0NCA6OjAY3OJwl1DQwOUSiXU6tZtuERt1zu2NxSCgsEFEUUctUId1EprgiDgkvRLfOY/tQSXm6Ww0JrlZgHgzJkzMJvN6NOnD5RKrjkfbrjcrEege+FwOHDixAnodDr07NmzE1vYtHBcCjTUbfKeY6EQFHjokocwq+8sAL6Tt73rtjvt7Tt522SCXq8P/eRtr3LbMnnb+140ztvSydvWBqvf55OTt/3nbe/J2/6YbCaYHWbpXKPQIFrj++VeayZve8+5aSqv2W6GyW6SzlUKFWI1sX7zWhwWGG1G6VwpKBGnjfOb1+qwyoYbtSRvUws8WO1WlNSUSP/fQvkeUWutDSpvS98jqi3VAIAvjn+BF354AU7RGdQcCwYWFBZaG1hYrVacPHkSKpUKiYmJ0Gq1YfOhhxhYeGt8L0RRhMViQWVlJex2O7Kzs6HR+P9AEw66Q2ABuOZanK47jV4xvZBuSO/QujuyHu5jQe2Nz6dHV78X7vfFQAsmeONQKOrSNBoNevbsifLycpw7d66zm0N+uD9Mk/97YTAYkJ6eHtZBRXeSbkhvMqAgIupuWvK+GNGBhdVqxfvvv49Vq1bh4MGDKCkpQUJCAvr06YM5c+bg9ttvR3JycljX3ZbIdsGCBVixYoXfx7Kzs1FYWNii8k6fPh2WQzX0ej169+4Nu90Ou93e/AXUYURRlA236M783QuVSgWVKqLfhomIqBuJ2L9oBQUFmD9/Pvbu3StLLy4uRnFxMXbs2IH/+7//w/LlyzFz5syIqdtbenr3+taNH9LCjyiKcDgc0Ol0DCx4L4iIKMJF5KewM2fOYMqUKSgqcm2QIwgCJkyYgH79+qGsrAwbNmxAQ0MDSktLMXv2bHz11Ve44oorwrLuu+++O+BjjR08eBC5ubnS+a233hrUdbfddhtiYvxPkPLGlZeIiIiIKJCIDCxuvvlm6YN9VlYWPvnkE4wYMUJ6vLy8HPPmzcPGjRths9lw00034dixY4iPjw+7uv/xj38EXfdNN90kHY8aNQoXXnhhUNc9+eSTyM7ODroeIiIiIqLGIm5G5fr167F161YArom9n332meyDPQAkJyfjk08+Qd++rt0UKysrsWzZsi5dd1VVFT777DPpfMGCBW0uk4iIiIgoWBEXWLz66qvS8YIFCzBs2DC/+QwGA5566inp/I033mjzxN/OrHv16tWwWFxrUKvVatx8881tKo+IiIiIqCUiKrCor6/Hxo0bpfOFCxc2mf+GG26Q5g1UVlZiy5YtXbJuAPjPf/4jHc+aNavdVrsiIiIiIvInogKL7du3S9/aGwwGjB7d9EYeOp0O48aNk843bdrUJes+fPgw8vPzpXMOgyIiIiKijhZRk7d//vln6XjYsGFBLT06atQofPvttz7Xd6W6vXsrkpOTMWvWrBZdv2vXLnzyySc4e/YsACApKQlDhgzB5ZdfjoQE/1vOExERERF5i6jA4tChQ9JxVlZWUNf07t1bOi4oKOhydTudTrz77rvS+c033wy1Wt2iMm688Ua/6Wq1GvPmzcMTTzwhTTYnIiIiIvInooZCVVRUSMdpaWlBXeO9iVxlZWWXq/u7777D6dOnpfNQDoOy2Wx49913cdFFF8lWnGoto9HY5A8RERERdV0R1WNRX18vHUdFRQV1jXc+7+u7St3vvPOOdDxs2DCMGjUqqOtUKhVmzZqFa665BmPGjEHfvn1hMBhQU1ODPXv2YPXq1XjnnXdgs9lQW1uLm266CRs3bsRll13WqnYCLd9gTxRFiKLY6vqo87mfQz6PXf9ehGP7O7NNHVV3e9UTynJDVVY4vsao9fh8ekTCvRAEIah8ERVYmM1m6Vij0QR1jVarlY4bGhq6VN319fX46KOPpPOW9Fbk5+cjKSnJJz0pKQlTp07F1KlTcccdd2DWrFmoqKiAxWLBr371Kxw8eBBKpbLFbW0No9EIvV7fIXVR+xBFUXptB/umFKm6+r0Ix/Z3Zps6qu72qieU5YaqrHB8jVHr8fn0iIR7EeyXwxEVWOh0OunYarUGdY17JScg+J6GcKn7o48+koYQqVQq3HLLLUFf6y+oaOzSSy/FqlWrMH36dACueSTr1q3DDTfc0OK2Ak33ypSVlaFPnz6yNIPBAIPB0Kq6KDy4v50xGAxd9s00VLr6vQjH9ndmmzqq7vaqJ5TlhqqscHyNUevx+fToTvciogIL72gq2B4A73wtHarT2XV7D4OaMWOGbM5GqEybNg2XX3458vLyAABffvllqwOLpoIEk8nkkyYIQsT/B+wO3M8jn8uufy/Csf2d2aaOqru96glluaEqKxxfY9R6fD49usu9iKjJ297fwpeUlAR1TXFxsXScmJjYZeo+deoUvvvuO+m8PfeumDp1qnTclmVxiYiIiChyRVRgMWjQIOm4sLAwqGtOnTolHQ8ePLjL1P3uu+9KXWsJCQm49tprW3R9S2RkZEjH5eXl7VYPEREREXVdERVYXHDBBdLx/v37Ybfbm71m9+7dfq8P97q9h0HNmzdPNhE81LyXguWcByIiIiLyJ6ICi/Hjx0sfsI1GI3744Ycm81ssFuzcuVM6v+KKK7pE3Tt37sThw4el8/YcBgUAe/bskY4zMzPbtS4iIiIi6poiKrCIjo7GlClTpPMVK1Y0mX/t2rWoq6sD4JrjMGHChC5R93/+8x/pePDgwbj00ktb1tgWqKiowCeffCKdT5o0qd3qIiIiIqKuK6ICCwC46667pOMVK1bg4MGDfvOZTCYsXbpUOl+yZAlUqrYtktURdVssFrz//vvSeWt6K4LdjM/hcGDx4sWora0F4Nqf46abbmpxfUREREQU+SIusJg1axZycnIAuD6EX3311di3b58sT0VFBWbPno2jR48CcPUYPPLII37LO3nypGyJsKZ6IkJdtz+ffvopqqqqAAAKhQK33npr0Ne6jR07Fvfddx927doVMM/+/fsxdepUrFu3Tkq7//77kZWV1eL6iIiIiCjyRdQ+Fm7vvfcexowZg3PnzuHkyZMYOXIkJk6ciH79+qGsrAwbNmyQ9k1QqVRYs2YN4uPju0Td3pO2p0yZgp49e7a4jfX19Xj55Zfx8ssvIzk5GSNHjkRGRgb0ej1qa2vx448/4qeffpJdc+211+Ivf/lLi+siIiIiou4hIgOLnj17YtOmTZg/fz727t0LURSRm5uL3NxcWb6UlBQsX75cNjcinOsuLS3FV199JZ3ffvvtbW5veXk5NmzYEPBxvV6PP/3pT3j00UehUERcBxcRERERhUhEBhaAa1Jzfn4+Vq9ejVWrVuHgwYMoKSlBfHw8+vbtizlz5mDhwoVITk7uMnWvXLlSWsY2NjYW119/favat3XrVmzfvh07duzA7t27UVJSgoqKCtTU1ECv10u9GJMmTcIvf/nLkPXmEBEREVHkEkT3LmtEnaisrAypqamytNLSUqSkpHRSiygURFGE0WiEwWCAIAid3ZxO1dXvRTi2vzPb1FF1t1c9oSw3VGWF42uMWo/Pp0d3uhcc20JERERERG3GwIKIiIiIiNqMgQUREREREbUZAwsiIiIiImozBhZERERERNRmDCyIiIjo/7d33+FRlen/xz8T0ifUhEAQkgAKASmigPRIV1SKoAtYgrpfRUBld0F0VRDWysKirrrAShMFRFEQWZGOq0hoosICsSQUgRRCSyEJyfn9wS/HDClMciZMMnm/rmuu65T7PM99zkxO5p7TAMAyCgsAAAAAllFYAAAAALCMwgIAAACAZRQWAAAAACyjsAAAAABgGYUFAAAAAMsoLAAAAABYRmEBAAAAwDIKCwAAAACWUVgAAAAAsIzCAgAAAIBlFBYAAAAALKOwAAAAAGAZhQUAAAAAyygsAAAAAFhGYQEAAADAMgoLAAAAAJZRWAAAAACwjMICAAAAgGUUFgAAAAAso7AAAAAAYBmFBQAAAADLKCwAAAAAWEZhAQAAAMAyCgsAAAAAllFYAAAAALCMwgIAAACAZRQWAAAAACyjsAAAAABgGYUFAAAAAMsoLAAAAABYRmEBAAAAwDIKCwAAAACWUVgAAAAAsIzCAgAAAIBlFBYAAAAALKOwAAAAAGAZhQUAAAAAyygsAAAAAFhGYQEAAADAMgoLAAAAAJZRWAAAAACwjMICAAAAgGUUFgAAAAAso7AAAAAAYBmFBQAAAADLKCwAAAAAWEZhAQAAAMAyCgsAAAAAllFYAAAAALCMwgIAAACAZRQWAAAAACyjsAAAAABgGYUFAAAAAMsoLAAAAABYRmEBAAAAwDIKCwAAAACWUVgAAAAAsIzCAgAAAIBlFBYAAAAALKOwAAAAAGCZRxcW2dnZWrx4sQYMGKCIiAj5+/srLCxMXbp00YwZM5SSklLh+7bZbGV+jRo1yqk+Dhw4oIkTJ6pNmzaqU6eO7Ha7mjVrppiYGG3cuNHCVgAAAEBVYTMMw3B3EuXh4MGDGjFihPbu3VtsTGhoqBYsWKABAwZU2L5tNluZ85g0aZJeffXVEmNeeuklTZ06VTk5OcXGjBgxQnPmzFH16tXLnMuVJCcnKzQ01GFaUlKS6tatW259ovwZhqH09HTZ7XZLn2VPUNm3RUXM3505Xa2+y6sfV7brqrYq4mcMZcf7+buqtC283Z1AeTh27Jh69+6t48ePS7r05bxHjx5q2rSpkpOTtWHDBmVmZiopKUmDBw/W2rVr1atXrwrZ99ixY53ue//+/dqyZYs5ft9995UYP3nyZP3tb38zx8PCwtS9e3f5+/tr9+7d2r9/vyRp6dKlOnXqlNasWSNvb4/8yAAAAMAqwwN1797dkGRIMiIiIoy9e/c6zE9OTjZ69+5txtSpU8c4ffp0pe972LBhZrs33nhjibEbNmwwYyUZEydONLKyshxilixZYvj7+5sxU6dOdUmeRUlKSnLIR5KRlJRUbv3h6sjLyzPOnz9v5OXluTsVt6vs26Ii5u/OnK5W3+XVjyvbdVVbFfEzhrLj/fxdVdoWHneNxX/+8x/997//lST5+vpq9erVatu2rUNMSEiIVq1apSZNmkiSUlNTNX369Erd9+nTp7V69WpzPCYmpsT4Z555xhwePny4pk+fLl9fX4eYESNGaNasWeZ4eV+XAgAAgMrL4wqLt99+2xyOiYlR69ati4yz2+2aNm2aOT5nzhxdvHix0va9bNkyZWVlSZJ8fHw0cuTIYmN37typnTt3SpK8vLxKLGweffRRXXfddZKk8+fPa/HixZbyBAAAgGfyqMIiLS3N4S5GDz74YInxQ4cOVVBQkKRLRw6++uqrStm3JC1atMgcvv322xUSElJs7MqVK83hPn36qFGjRsXG2mw2h6Mfn376qaU8AQAA4Jk8qrDYtm2b+au93W5Xhw4dSoz39/dX586dzfFNmzZVyr7j4uIUGxtrjl/pNKjNmzebw7fccssV2+/Zs6c5XHA9AQAAgHweVVgcOHDAHG7durVTdzC68cYbi1y+MvVd8GhFSEiIbr/99hLjC/ZVMIfitGvXzhzOzc1VXFxcGbIEAACAJ/OowuLQoUPmcEREhFPLhIeHm8MHDx6sdH3n5eU5XPcwcuRI+fj4FBuflJSkM2fOmOPO5BoQEODwPAkr2wkAAACeyaMeSnDq1ClzuF69ek4tU79+fXM4NTW10vW9efNmHT161By/0mlQBfOUSpdrcnKypLLnmp6eXqZ5AAAAqPg8qrBIS0szhwMCApxapmBcweUrS9/vvfeeOdy6desrntp0eT9XM9f8i9WdZRiGDM98MHyVkf8e8j5W/m1REfN3Z05Xq+/y6seV7bqqrYr4GUPZ8X7+zhO2hbNPDPeowuLChQvm8OXPZCiOn5+fOZyZmVmp+k5LS9OKFSvM8SsdrZAc85Su/nYqjfT0dAUGBl6VvlA+DMMwPy/O7pQ8VWXfFhUxf3fmdLX6Lq9+XNmuq9qqiJ8xlB3v5+88YVs4++OwRxUW/v7+5nB2drZTyxS8w5Gzv95XlL5XrFhhnkLk7e2te++994rLFMxTupTr5dOK4ortVNKRjuTkZDVu3Nhhmt1ul91uL1NfqBjyf52x2+2VdmfqKpV9W1TE/N2Z09Xqu7z6cWW7rmqrIn7GUHa8n7+rStvCowqLgtWUs7+qF4wr7ak67u674GlQ/fv3d7hmoziX95OZmelUYeGK7VRSkZCRkVFoms1m8/g/wKog/33kvaz826Ii5u/OnK5W3+XVjyvbdVVbFfEzhrLj/fxdVdkWHnVXqODgYHM4MTHRqWVOnjxpDtepU6fS9H3kyBGH51E4cxqU5JindPW3EwAAADyTRxUWzZs3N4cPHz7s1DJHjhwxh6OioipN34sXLzYPrdWuXVsDBw50arnQ0FDVqlXLHHcm1wsXLph3hCpLrgAAAPB8HlVYtGjRwhz+8ccfdfHixSsus2fPniKXr+h9FzwNavjw4Q4XV19Jwb6+++67K8YXzLNatWpq1qyZ030BAACgavCowqJLly7mF+z09HTt2rWrxPisrCxt377dHO/Vq1el6Hv79u0OT7929jSofD179jSHt2zZcsX4rVu3msMF1xMAAADI51GFRVBQkHr37m2OL1y4sMT4Tz75ROfPn5d06bqBHj16VIq+Fy1aZA5HRUXp5ptvLlWugwcPNoc3bNigY8eOlRhfcF0KLgsAAADk86jCQpLGjBljDi9cuFD79+8vMi4jI0OTJ082xx955BF5e1u7SdbV6DsrK0sffvihOV7aoxWS1KFDB3Xo0EGSlJubq6effrrY2Llz55pHR6pXr64HHnig1P0BAADA83lcYXH77bere/fuki59Cb/jjjv0ww8/OMScOnVKgwcP1s8//yzp0hGDSZMmFdleQkKCwy3CSjoS4eq+i/LZZ5/p9OnTkiQvLy/dd999Ti9b0CuvvGIOf/DBB3r66aeVk5PjELN8+XKNHz/eHJ8wYYJCQkLK1B8AAAA8m0c9xyLfkiVL1LFjR504cUIJCQm64YYbFB0draZNmyo5OVkbNmwwn5vg7e2t5cuXO9wpqSL3XfCi7d69e6thw4ZlyrN379567rnn9OKLL0qSXnvtNS1evFjdu3eXv7+/du/erX379pnxffv21V//+tcy9QUAAADP55GFRcOGDbVp0yaNGDFCe/fulWEY2rJlS6ELlevWrasFCxY4XBtRkftOSkrS2rVrzfFRo0ZZynXatGny8/PTtGnTlJOTo+PHjzucZpVv+PDhmjNnjuVTxQAAAOC5PPabYlRUlGJjY7Vs2TItXbpU+/fvV2JiomrVqqUmTZrorrvu0oMPPlgup/aUV98ffPCBeRvbGjVqaMiQIZbytNlseu655zR06FC9++67WrdunY4ePaqcnByFhYWpc+fOiomJUZ8+fSz1AwAAAM9nM/Kfsga4UXJyskJDQx2mJSUlqW7dum7KCK5gGIbS09Nlt9tls9ncnY5bVfZtURHzd2dOV6vv8urHle26qq2K+BlD2fF+/q4qbQuPu3gbAAAAwNVHYQEAAADAMgoLAAAAAJZRWAAAAACwjMICAAAAgGUUFgAAAAAso7AAAAAAYBmFBQAAAADLKCwAAAAAWEZhAQAAAMAyCgsAAAAAllFYAAAAALCMwgIAAACAZRQWAAAAACyjsAAAAABgGYUFAAAAAMsoLAAAAABYRmEBAAAAwDIKCwAAAACWUVgAAAAAsIzCAgAAAIBlFBYAAAAALKOwAAAAAGAZhQUqvdTUVL3wwgtq3769ateurYCAADVu3FgxMTH69ttvr2ouW7Zskc1m06hRo65qv85YuHChbDabXnjhBaeXiYyMlM1mU0JCwhVjK/K6l4f87Vnw5ePjo4YNG2rEiBHas2ePu1PULbfcUuT7Z7PZFBkZ6ZacAACei8ICldrGjRt17bXXaurUqUpISFD37t01aNAg1ahRQ++99566dOmi8ePHKy8vzyX9vfDCC7LZbFq4cKFL2kPl17RpU8XExCgmJkaDBg2St7e3li1bpk6dOumzzz5zd3rlpriiBQBQdXm7OwGgrHbu3KkBAwYoJydH06ZN09NPPy0fHx9z/tdff60RI0bojTfeULVq1TRz5sxyz6ljx446cOCAatasWe59VTRVdd27devmUGjm5ORozJgxevfddzV69Gjdeuut8vX1dV+CRThw4IDD3woAAK7AEQtUSoZhKCYmRtnZ2ZoyZYqef/75Ql+UunXrpnXr1snf31+zZs3S9u3byz2vwMBARUVFKSwsrNz7qmiq8roX5OPjo9dff11BQUE6ceKEduzY4e6UComKilLTpk3dnQYAwMNQWKBS+uKLL3TgwAE1aNBAf/3rX4uNa9GihcaOHSvDMPSPf/zDYV7BUznef/993XTTTQoMDFRoaKhiYmL022+/OcRHRkZq6tSpkqQHH3zQ4dz6LVu2SCr+OoOCp1Dt3r1bt912m2rVqqU6deronnvu0bFjxyRJ6enpeuqppxQZGSl/f3+1atVKH3/8caH1MgxDS5cu1fDhw9WsWTPZ7XZVr15dHTt21DvvvOOyU79Ko7h1f/nll+Xl5aWFCxfqxx9/1MCBA1W7dm3Z7XZFR0dr27ZtxbYZGxuru+++W2FhYfL19VXDhg31xz/+UUeOHCkUe+bMGf3zn/9U//79FRERIT8/PwUHB+vWW2/V+vXri2y/4GdgyZIl6tSpk6pXr65atWpZ2RSy2+1q1qyZJOno0aOSLl2TUb16db3wwguKi4vT8OHDVa9ePXl5eWnlypXmsgcOHNCoUaPUqFEj+fn5qV69eho+fLj2799fZF+5ubmaMWOGoqKi5O/vr0aNGunJJ5/UuXPnis2vpGssDhw4oIcffliRkZHy8/NTaGiounbtqhkzZujixYtKSEiQzWbT1q1bJUmNGzd2+FsoKCMjQ3/729/UqlUrBQQEqGbNmurRo4eWLVtWZN/51/QYhqF//vOfatu2rQIDA3XDDTcUuy4AgIqDwgKV0po1ayRJd9999xVP6bj33nslSevWrSvyC/eMGTP0wAMPKCgoSIMGDZLdbtd7772nTp06mV/4JWnYsGFq27atJKlr167mefUxMTGqX7++U3nHxsaqa9euSk5OVv/+/RUcHKyPPvpIvXv31tmzZ9WzZ08tWrRIHTp0UOfOnfW///1P99xzj7788kuHdrKysjRy5Eht2LBB9evX15133qlOnTpp//79Gjt2rB566CGn8rmadu3apU6dOikhIUH9+/fXddddp6+++kq9e/fWvn37CsW/88476tKliz755BNFRERo8ODBCg4O1rx589S+fXsdOHDAIX779u164oknFBcXp+bNm2vIkCFq3ry51q1bp/79+2v+/PnF5vbKK6/o/vvvl6+vr+644w61atXK8vqeP39ekuTn5+cwPS4uTh06dNCOHTvUs2dP9e3b1/wMr1y5Uu3atdOiRYsUEhKigQMHqnHjxlq+fLk6duyor776qlA/9913nyZOnKijR4+qX79+6tChgxYtWqRevXopKyurVDl/9NFHateunebPn6/AwEANGTJEN910k44ePaqnnnpKaWlpCgoKUkxMjOrVqydJGjp0qMPfQsH179GjhyZPnqykpCTdcccd6tq1q3bs2KERI0boySefLDaP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0+n4mjVrllhYDBo0SH5+fnryySd17Ngxh2sf8tWqVUvz58/XgAEDSuxrwoQJql27tv785z/r3LlzysrK0tq1awvFVatWTePGjdPMmTNLfK6Et7e3Vq5cqeeff14zZ85UVlaWsrOztXXrVm3durXIdv39/UvM0RUK3g43MzNTsbGxio2NLTK2cePGev/999WmTZtyzwtA5UFhAQDwSKNHj1b37t01e/ZsbdiwwXxidGRkpO688049/vjjCgsLc6qthx9+WIMGDdK///1vffHFF4qLi1NqaqqqV6+uRo0aqU+fPnrooYccnvdQEi8vL7300ksaPXq0Fi5cqPXr1+vnn39WSkqKvL29FRoaquuvv169e/fWH/7wB11zzTVl3g7O2rt3r7Zv367Nmzdrx44dOnTokI4fP66MjAwFBgaqfv36uuGGGzRw4EDdc8898vPzK/ecAFQuNqMy3pICAIDLREZG6vDhw5Kk+Ph4h1u7AgDKHxdvAwAAALCMwgIAAACAZRQWAAAAACyjsAAAAABgGYUFAAAAAMsoLAAAAABYxu1mAQAAAFjGEQsAAAAAllFYAAAAALCMwgIAAACAZRQWAAAAACyjsAAAAABgGYUFAAAAAMsoLAAAAABYRmEBAAAAwDIKCwAAAACW/T8folTZBiZt3gAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 800x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ---------- plot ----------\n",
    "fig, ax = plt.subplots(1, 1, figsize=(8, 6))\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=3, 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=3, 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(1e3 + 50, opt_linear_mse * (1 + 3e-3), 'Optimal Linear Predictor',\n",
    "        ha='left', va='bottom', color='black', fontsize=15)\n",
    "\n",
    "ax.set_xlim(1e3, max_step)\n",
    "ax.set_xscale(\"log\")\n",
    "ax.set_xlabel(\"Epochs\", fontsize=24)\n",
    "ax.set_ylabel(\"Loss\", fontsize=24)\n",
    "ax.legend(ncol=1, fontsize=15)\n",
    "ax.grid(True, which=\"both\", alpha=0.15)\n",
    "style_axes(ax)\n",
    "plt.tight_layout()\n",
    "plt.savefig(\"relu.pdf\")"
   ]
  },
  {
   "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
}
