{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "767fe42d",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "import torch\n",
    "import torch.nn.functional as F\n",
    "from torchvision.datasets import MNIST\n",
    "from torchvision.transforms import v2\n",
    "from tqdm import tqdm"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "85f3e0a5",
   "metadata": {},
   "source": [
    "## Utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5cae103a",
   "metadata": {},
   "outputs": [],
   "source": [
    "def rectangle(x):\n",
    "    # Returns 1 where x is in [-0.5, 0.5], 0 elsewhere\n",
    "    return ((x >= -0.5) & (x <= 0.5)).float()\n",
    "\n",
    "class JumpReLU(torch.autograd.Function):\n",
    "    \"\"\"\n",
    "    Custom autograd function for JumpReLU:\n",
    "    Forward: x * 1{x > threshold}\n",
    "    Backward: Learnable threshold with soft gradient approximation\n",
    "    \"\"\"\n",
    "\n",
    "    @staticmethod\n",
    "    def forward(ctx, x, threshold, bandwidth):\n",
    "        # Save inputs for backward pass\n",
    "        if not isinstance(bandwidth, torch.Tensor):\n",
    "            bandwidth = torch.tensor(bandwidth, dtype=x.dtype, device=x.device)\n",
    "        ctx.save_for_backward(x, threshold, bandwidth)\n",
    "        return x * (x > threshold)\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, grad_output):\n",
    "        # Compute gradients w.r.t. x and threshold\n",
    "        x, threshold, bandwidth = ctx.saved_tensors\n",
    "\n",
    "        x_grad = (x > threshold).float() * grad_output\n",
    "        threshold_grad = (\n",
    "            -(threshold / bandwidth)\n",
    "            * rectangle((x - threshold) / bandwidth)\n",
    "            * grad_output\n",
    "        ).sum(dim=0, keepdim=True)  # Sum over batch\n",
    "\n",
    "        return x_grad, threshold_grad, None  # No gradient for fixed bandwidth\n",
    "\n",
    "def jumprelu(x, threshold, bandwidth=1e-1):\n",
    "    # Wrapper for JumpReLU\n",
    "    return JumpReLU.apply(x, threshold, bandwidth)\n",
    "\n",
    "\n",
    "class StepFunction(torch.autograd.Function):\n",
    "    \"\"\"\n",
    "    Step function with differentiable surrogate gradient for threshold learning.\n",
    "    Forward: binary step at threshold.\n",
    "    Backward: gradient only for threshold (not input).\n",
    "    \"\"\"\n",
    "\n",
    "    @staticmethod\n",
    "    def forward(ctx, input, threshold, bandwidth):\n",
    "        if not isinstance(threshold, torch.Tensor):\n",
    "            threshold = torch.tensor(threshold, dtype=input.dtype, device=input.device)\n",
    "        if not isinstance(bandwidth, torch.Tensor):\n",
    "            bandwidth = torch.tensor(bandwidth, dtype=input.dtype, device=input.device)\n",
    "        ctx.save_for_backward(input, threshold, bandwidth)\n",
    "        return (input > threshold).type(input.dtype)\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, grad_output):\n",
    "        # Gradient only for threshold\n",
    "        x, threshold, bandwidth = ctx.saved_tensors\n",
    "        grad_input = 0.0 * grad_output  # no gradient w.r.t. input\n",
    "        grad_threshold = (\n",
    "            -(1.0 / bandwidth)\n",
    "            * rectangle((x - threshold) / bandwidth)\n",
    "            * grad_output\n",
    "        ).sum(dim=0, keepdim=True)\n",
    "        return grad_input, grad_threshold, None\n",
    "\n",
    "def step(input, threshold, bandwidth=1e-1):\n",
    "    # Wrapper for StepFunction\n",
    "    return StepFunction.apply(input, threshold, bandwidth)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "60ded903",
   "metadata": {},
   "source": [
    "## SAEs Architectures"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "60bef3a5",
   "metadata": {},
   "outputs": [],
   "source": [
    "class MnistMP(torch.nn.Module):\n",
    "    def __init__(self, params, device, D=None, b_pre=None):\n",
    "        super(MnistMP, self).__init__()\n",
    "\n",
    "        self.device = device\n",
    "        self.p = params[\"p\"]\n",
    "        self.k = params[\"k\"]\n",
    "        self.dropout = params[\"dropout\"]\n",
    "        self.tied = params[\"tied\"]\n",
    "\n",
    "        if D is None:\n",
    "            D = torch.randn((self.p, 28, 28), device=self.device)\n",
    "        if b_pre is None:\n",
    "            b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        self.D = torch.nn.Parameter(D)\n",
    "\n",
    "        if params[\"b_pre\"]:\n",
    "            self.b_pre = torch.nn.Parameter(b_pre)\n",
    "        else:\n",
    "            self.b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        if params[\"normalize\"]:\n",
    "            self.normalize(params[\"normalize\"])\n",
    "\n",
    "        if self.tied:\n",
    "            self.W = self.D\n",
    "        else:\n",
    "            self.W = torch.nn.Parameter(self.D.data.clone())\n",
    "\n",
    "    def normalize(self, normalization):\n",
    "        if normalization == \"on\":\n",
    "            self.D.data = F.normalize(self.D.data, p=2, dim=[1, 2])\n",
    "        elif normalization == \"in\":\n",
    "            self.D.data = self.D.data / torch.maximum(\n",
    "                torch.ones(1, device=self.device),\n",
    "                torch.norm(self.D.data, p=2, dim=[1, 2], keepdim=True),\n",
    "            )\n",
    "        else:\n",
    "            raise ValueError(\n",
    "                'normalize should be \"in\"/\"on\" the hypersphere or set to False'\n",
    "            )\n",
    "\n",
    "    def encode(self, x):\n",
    "        if self.dropout and self.training:\n",
    "            keep_mask = torch.bernoulli(\n",
    "                torch.full((self.p, 1, 1), 1 - self.dropout, device=self.device)\n",
    "            )\n",
    "            W = keep_mask * self.W\n",
    "        else:\n",
    "            W = self.W\n",
    "\n",
    "        batch_size = x.shape[0]\n",
    "        zhat = torch.zeros(batch_size, self.p, device=self.device)\n",
    "        r = x.clone() - self.b_pre\n",
    "        for _ in range(self.k):\n",
    "            WTr = (W * r).sum(axis=[-1, -2])\n",
    "            values, indices = torch.max(torch.relu(WTr), dim=1, keepdim=True)\n",
    "            zhat_ = torch.zeros_like(zhat).scatter_(1, indices, values)\n",
    "            zhat += zhat_\n",
    "            r = r - torch.einsum(\"ijk,li->ljk\", self.D, zhat_).unsqueeze(1)\n",
    "        return zhat\n",
    "\n",
    "    def decode(self, z):\n",
    "        return (\n",
    "            torch.einsum(\n",
    "                \"ijk,li->ljk\",\n",
    "                self.D,\n",
    "                z,\n",
    "            ).unsqueeze(1)\n",
    "            + self.b_pre\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        zhat = self.encode(x)\n",
    "        xhat = self.decode(zhat)\n",
    "        return xhat, zhat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "6c59fe72",
   "metadata": {},
   "outputs": [],
   "source": [
    "class MnistReLU(torch.nn.Module):\n",
    "    def __init__(self, params, device, D=None, b_pre=None):\n",
    "        super(MnistReLU, self).__init__()\n",
    "\n",
    "        self.device = device\n",
    "        self.p = params[\"p\"]\n",
    "        self.dropout = params[\"dropout\"]\n",
    "        self.tied = params[\"tied\"]\n",
    "\n",
    "        if D is None:\n",
    "            D = torch.randn((self.p, 28, 28), device=device)\n",
    "        if b_pre is None:\n",
    "            b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        self.D = torch.nn.Parameter(D)\n",
    "\n",
    "        if params[\"b_pre\"]:\n",
    "            self.b_pre = torch.nn.Parameter(b_pre)\n",
    "        else:\n",
    "            self.b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        if params[\"normalize\"]:\n",
    "            self.normalize(params[\"normalize\"])\n",
    "\n",
    "        if self.tied:\n",
    "            self.W = self.D\n",
    "        else:\n",
    "            self.W = torch.nn.Parameter(self.D.data.clone())\n",
    "\n",
    "        self.b_enc = torch.nn.Parameter(torch.zeros(self.p, device=device))\n",
    "\n",
    "    def normalize(self, normalization):\n",
    "        if normalization == \"on\":\n",
    "            self.D.data = F.normalize(self.D.data, p=2, dim=[1, 2])\n",
    "        elif normalization == \"in\":\n",
    "            self.D.data = self.D.data / torch.maximum(\n",
    "                torch.ones(1, device=self.device),\n",
    "                torch.norm(self.D.data, p=2, dim=[1, 2], keepdim=True),\n",
    "            )\n",
    "        else:\n",
    "            raise ValueError(\n",
    "                'normalize should be \"in\"/\"on\" the hypersphere or set to False'\n",
    "            )\n",
    "\n",
    "    def encode(self, x):\n",
    "        if self.dropout and self.training:\n",
    "            keep_mask = torch.bernoulli(\n",
    "                torch.full((self.p, 1, 1), 1 - self.dropout, device=self.device)\n",
    "            )\n",
    "            W = keep_mask * self.W\n",
    "            b_enc = keep_mask.squeeze() * self.b_enc\n",
    "\n",
    "        else:\n",
    "            W = self.W\n",
    "            b_enc = self.b_enc\n",
    "\n",
    "        WTr_b = (W * (x - self.b_pre)).sum(axis=[-1, -2]) + b_enc\n",
    "        zhat = torch.relu(WTr_b)\n",
    "        return zhat\n",
    "\n",
    "    def decode(self, z):\n",
    "        return (\n",
    "            torch.einsum(\n",
    "                \"ijk,li->ljk\",\n",
    "                self.D,\n",
    "                z,\n",
    "            ).unsqueeze(1)\n",
    "            + self.b_pre\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        zhat = self.encode(x)\n",
    "        xhat = self.decode(zhat)\n",
    "        return xhat, zhat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "68300188",
   "metadata": {},
   "outputs": [],
   "source": [
    "class MnistJumpReLU(torch.nn.Module):\n",
    "    def __init__(self, params, device, D=None, b_pre=None):\n",
    "        super(MnistJumpReLU, self).__init__()\n",
    "\n",
    "        self.device = device\n",
    "        self.p = params[\"p\"]\n",
    "        self.dropout = params[\"dropout\"]\n",
    "        self.tied = params[\"tied\"]\n",
    "\n",
    "        if D is None:\n",
    "            D = torch.randn((self.p, 28, 28), device=device)\n",
    "        if b_pre is None:\n",
    "            b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        self.D = torch.nn.Parameter(D)\n",
    "\n",
    "        if params[\"b_pre\"]:\n",
    "            self.b_pre = torch.nn.Parameter(b_pre)\n",
    "        else:\n",
    "            self.b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        if params[\"normalize\"]:\n",
    "            self.normalize(params[\"normalize\"])\n",
    "\n",
    "        if self.tied:\n",
    "            self.W = self.D\n",
    "        else:\n",
    "            self.W = torch.nn.Parameter(self.D.data.clone())\n",
    "\n",
    "        self.b_enc = torch.nn.Parameter(torch.zeros(self.p, device=device))\n",
    "        self.log_thresholds = torch.nn.Parameter(\n",
    "            torch.log(0.1 * torch.ones(self.p, device=device))\n",
    "        )\n",
    "\n",
    "    def normalize(self, normalization):\n",
    "        if normalization == \"on\":\n",
    "            self.D.data = F.normalize(self.D.data, p=2, dim=[1, 2])\n",
    "        elif normalization == \"in\":\n",
    "            self.D.data = self.D.data / torch.maximum(\n",
    "                torch.ones(1, device=self.device),\n",
    "                torch.norm(self.D.data, p=2, dim=[1, 2], keepdim=True),\n",
    "            )\n",
    "        else:\n",
    "            raise ValueError(\n",
    "                'normalize should be \"in\"/\"on\" the hypersphere or set to False'\n",
    "            )\n",
    "\n",
    "    def encode(self, x):\n",
    "        if self.dropout and self.training:\n",
    "            keep_mask = torch.bernoulli(\n",
    "                torch.full((self.p, 1, 1), 1 - self.dropout, device=self.device)\n",
    "            )\n",
    "            W = keep_mask * self.W\n",
    "            b_enc = keep_mask.squeeze() * self.b_enc\n",
    "\n",
    "        else:\n",
    "            W = self.W\n",
    "            b_enc = self.b_enc\n",
    "\n",
    "        WTr_b = (W * (x - self.b_pre)).sum(axis=[-1, -2]) + b_enc\n",
    "\n",
    "        zhat = jumprelu(\n",
    "            torch.relu(WTr_b),\n",
    "            threshold=torch.exp(self.log_thresholds),\n",
    "        )\n",
    "        return zhat\n",
    "\n",
    "    def decode(self, z):\n",
    "        return (\n",
    "            torch.einsum(\n",
    "                \"ijk,li->ljk\",\n",
    "                self.D,\n",
    "                z,\n",
    "            ).unsqueeze(1)\n",
    "            + self.b_pre\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        zhat = self.encode(x)\n",
    "        xhat = self.decode(zhat)\n",
    "        return xhat, zhat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "0bac9b9d",
   "metadata": {},
   "outputs": [],
   "source": [
    "class MnistTopK(torch.nn.Module):\n",
    "    def __init__(self, params, device, D=None, b_pre=None):\n",
    "        super(MnistTopK, self).__init__()\n",
    "\n",
    "        self.device = device\n",
    "        self.p = params[\"p\"]\n",
    "        self.k = params[\"k\"]\n",
    "        self.dropout = params[\"dropout\"]\n",
    "        self.tied = params[\"tied\"]\n",
    "\n",
    "        if D is None:\n",
    "            D = torch.randn((self.p, 28, 28), device=self.device)\n",
    "        if b_pre is None:\n",
    "            b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        self.D = torch.nn.Parameter(D)\n",
    "\n",
    "        if params[\"b_pre\"]:\n",
    "            self.b_pre = torch.nn.Parameter(b_pre)\n",
    "        else:\n",
    "            self.b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        if params[\"normalize\"]:\n",
    "            self.normalize(params[\"normalize\"])\n",
    "\n",
    "        if self.tied:\n",
    "            self.W = self.D\n",
    "        else:\n",
    "            self.W = torch.nn.Parameter(self.D.data.clone())\n",
    "\n",
    "        self.b_enc = torch.nn.Parameter(torch.zeros(self.p, device=device))\n",
    "\n",
    "    def normalize(self, normalization):\n",
    "        if normalization == \"on\":\n",
    "            self.D.data = F.normalize(self.D.data, p=2, dim=[1, 2])\n",
    "        elif normalization == \"in\":\n",
    "            self.D.data = self.D.data / torch.maximum(\n",
    "                torch.ones(1, device=self.device),\n",
    "                torch.norm(self.D.data, p=2, dim=[1, 2], keepdim=True),\n",
    "            )\n",
    "        else:\n",
    "            raise ValueError(\n",
    "                'normalize should be \"in\"/\"on\" the hypersphere or set to False'\n",
    "            )\n",
    "\n",
    "    def encode(self, x):\n",
    "        if self.dropout and self.training:\n",
    "            keep_mask = torch.bernoulli(\n",
    "                torch.full((self.p, 1, 1), 1 - self.dropout, device=self.device)\n",
    "            )\n",
    "\n",
    "            W = keep_mask * self.W\n",
    "            b_enc = keep_mask.squeeze() * self.b_enc\n",
    "\n",
    "        else:\n",
    "            W = self.W\n",
    "            b_enc = self.b_enc\n",
    "\n",
    "        batch_size = x.shape[0]\n",
    "        WTr_b = (W * (x - self.b_pre)).sum(axis=[-1, -2]) + b_enc\n",
    "\n",
    "        values, indices = torch.topk(\n",
    "            torch.relu(WTr_b),\n",
    "            k=self.k,\n",
    "            dim=1,\n",
    "        )\n",
    "        zhat = torch.zeros(batch_size, self.p, device=self.device).scatter_(\n",
    "            1, indices, values\n",
    "        )\n",
    "        return zhat\n",
    "\n",
    "    def decode(self, z):\n",
    "        return (\n",
    "            torch.einsum(\n",
    "                \"ijk,li->ljk\",\n",
    "                self.D,\n",
    "                z,\n",
    "            ).unsqueeze(1)\n",
    "            + self.b_pre\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        zhat = self.encode(x)\n",
    "        xhat = self.decode(zhat)\n",
    "        return xhat, zhat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "e57b9400",
   "metadata": {},
   "outputs": [],
   "source": [
    "class MnistBatchTopK(torch.nn.Module):\n",
    "    def __init__(self, params, device, D=None, b_pre=None):\n",
    "        super(MnistBatchTopK, self).__init__()\n",
    "\n",
    "        self.device = device\n",
    "        self.p = params[\"p\"]\n",
    "        self.k = params[\"k\"]\n",
    "        self.dropout = params[\"dropout\"]\n",
    "        self.tied = params[\"tied\"]\n",
    "        self.register_buffer(\n",
    "            \"threshold\", torch.tensor(float(\"nan\"), device=self.device)\n",
    "        )\n",
    "        self.threshold_momentum = 0.9\n",
    "\n",
    "        if D is None:\n",
    "            D = torch.randn((self.p, 28, 28), device=self.device)\n",
    "        if b_pre is None:\n",
    "            b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        self.D = torch.nn.Parameter(D)\n",
    "\n",
    "        if params[\"b_pre\"]:\n",
    "            self.b_pre = torch.nn.Parameter(b_pre)\n",
    "        else:\n",
    "            self.b_pre = torch.zeros((1, 28, 28), device=device)\n",
    "\n",
    "        if params[\"normalize\"]:\n",
    "            self.normalize(params[\"normalize\"])\n",
    "\n",
    "        if self.tied:\n",
    "            self.W = self.D\n",
    "        else:\n",
    "            self.W = torch.nn.Parameter(self.D.data.clone())\n",
    "\n",
    "        self.b_enc = torch.nn.Parameter(torch.zeros(self.p, device=device))\n",
    "\n",
    "    def normalize(self, normalization):\n",
    "        if normalization == \"on\":\n",
    "            self.D.data = F.normalize(self.D.data, p=2, dim=[1, 2])\n",
    "        elif normalization == \"in\":\n",
    "            self.D.data = self.D.data / torch.maximum(\n",
    "                torch.ones(1, device=self.device),\n",
    "                torch.norm(self.D.data, p=2, dim=[1, 2], keepdim=True),\n",
    "            )\n",
    "        else:\n",
    "            raise ValueError(\n",
    "                'normalize should be \"in\"/\"on\" the hypersphere or set to False'\n",
    "            )\n",
    "\n",
    "    def encode(self, x):\n",
    "        if self.dropout and self.training:\n",
    "            keep_mask = torch.bernoulli(\n",
    "                torch.full((self.p, 1, 1), 1 - self.dropout, device=self.device)\n",
    "            )\n",
    "\n",
    "            W = keep_mask * self.W\n",
    "            b_enc = keep_mask.squeeze() * self.b_enc\n",
    "\n",
    "        else:\n",
    "            W = self.W\n",
    "            b_enc = self.b_enc\n",
    "\n",
    "        batch_size = x.shape[0]\n",
    "        WTr_b = (W * (x - self.b_pre)).sum(axis=[-1, -2]) + b_enc\n",
    "        if self.training:\n",
    "            values, indices = torch.topk(\n",
    "                torch.relu(WTr_b.flatten()),\n",
    "                k=self.k * batch_size,\n",
    "                dim=0,\n",
    "            )\n",
    "            zhat = (\n",
    "                torch.zeros(batch_size * self.p, device=self.device)\n",
    "                .scatter(0, indices, values)\n",
    "                .reshape(batch_size, self.p)\n",
    "            )\n",
    "            if torch.isnan(self.threshold):\n",
    "                self.threshold.data = values[-1].detach()\n",
    "            else:\n",
    "                self.threshold.data = (\n",
    "                    self.threshold_momentum * self.threshold\n",
    "                    + (1 - self.threshold_momentum) * values[-1].detach()\n",
    "                )\n",
    "        else:\n",
    "            mask = (WTr_b >= self.threshold).float()\n",
    "            zhat = WTr_b * mask\n",
    "        return zhat\n",
    "\n",
    "    def decode(self, z):\n",
    "        return (\n",
    "            torch.einsum(\n",
    "                \"ijk,li->ljk\",\n",
    "                self.D,\n",
    "                z,\n",
    "            ).unsqueeze(1)\n",
    "            + self.b_pre\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        zhat = self.encode(x)\n",
    "        xhat = self.decode(zhat)\n",
    "        return xhat, zhat"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b19a226e",
   "metadata": {},
   "source": [
    "## Train"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b584a08e",
   "metadata": {},
   "source": [
    "### Config"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "40d4cf91",
   "metadata": {},
   "outputs": [],
   "source": [
    "config = {\n",
    "    \"device\": \"cuda\",  # Device to use for training (e.g., \"cuda\" or \"cpu\")\n",
    "    \"num_epochs\": 50,  # Number of training epochs\n",
    "    \"network\": \"ReLU\",  # SAE class to use\n",
    "    \"lr\": 1.0e-3,  # Learning rate\n",
    "}\n",
    "\n",
    "config[\"net_params\"] = {  # Parameters passed to the model\n",
    "    \"p\": 1000,  # Dictionary size (number of atoms)\n",
    "    \"k\": 10,  # Sparsity level (number of active atoms)\n",
    "    \"normalize\": \"on\",  # Dictionary normalization method (\"on\" = unit norm)\n",
    "    \"dropout\": (\n",
    "        0.75 if config[\"network\"] == \"MP\" else 0\n",
    "    ),  # Dropout probability during training\n",
    "    \"tied\": False,  # Whether encoder and decoder share weights\n",
    "    \"b_pre\": True,  # Learnable input bias\n",
    "}\n",
    "\n",
    "config[\"dataloader\"] = {  # DataLoader settings for training\n",
    "    \"shuffle\": True,  # Shuffle dataset each epoch\n",
    "    \"batch_size\": 50,  # Training batch size\n",
    "    \"num_workers\": 0,  # Number of parallel data loading workers\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "65dc353c",
   "metadata": {},
   "outputs": [],
   "source": [
    "network_classes = {\n",
    "    \"MP\": MnistMP,\n",
    "    \"TopK\": MnistTopK,\n",
    "    \"ReLU\": MnistReLU,\n",
    "    \"JumpReLU\": MnistJumpReLU,\n",
    "    \"BatchTopK\": MnistBatchTopK,\n",
    "}\n",
    "\n",
    "regularization_penalty = {\n",
    "    \"MP\": 0,\n",
    "    \"TopK\": 0.1,\n",
    "    \"BatchTopK\": 0.1,\n",
    "    \"ReLU\": 0.1,\n",
    "    \"JumpReLU\": 10,\n",
    "}"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0e12919a",
   "metadata": {},
   "source": [
    "### SAE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "0c7f373d",
   "metadata": {},
   "outputs": [],
   "source": [
    "net = network_classes[config[\"network\"]](\n",
    "    config[\"net_params\"],\n",
    "    device=config[\"device\"],\n",
    "    b_pre = torch.load(\"b_pre.pt\").to(config[\"device\"]),  # Pre-bias initialized as the geometric mean of the training data\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8f20b7ad",
   "metadata": {},
   "source": [
    "### Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "7212e03f",
   "metadata": {},
   "outputs": [],
   "source": [
    "transform = v2.Compose([v2.ToImage(), v2.ToDtype(torch.float32, scale=True)])\n",
    "dataset = MNIST(\"data\", train=True, transform=transform, download=True)\n",
    "train_dataset = torch.utils.data.Subset(dataset, range(50000))\n",
    "train_loader = torch.utils.data.DataLoader(train_dataset, **config[\"dataloader\"])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "78b6ad94",
   "metadata": {},
   "source": [
    "### Training "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3ef55011",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 2/2 [00:39<00:00, 19.79s/it]\n"
     ]
    }
   ],
   "source": [
    "optimizer = torch.optim.Adam(net.parameters(), lr=config[\"lr\"])\n",
    "reg_penalty = regularization_penalty[config[\"network\"]]\n",
    "\n",
    "for epoch in tqdm(range(config[\"num_epochs\"])):\n",
    "    net.train()\n",
    "\n",
    "    if epoch > 0:\n",
    "        net.dropout = 0\n",
    "\n",
    "    for idx, (x, _) in tqdm(enumerate(train_loader), disable=True):\n",
    "        optimizer.zero_grad()\n",
    "\n",
    "        x = x.to(config[\"device\"])\n",
    "\n",
    "        zhat = net.encode(x)\n",
    "        xhat = net.decode(zhat)\n",
    "\n",
    "        loss = torch.sum((x - xhat) ** 2, dim=[-1, -2]).mean()\n",
    "\n",
    "        # ReLU L1 loss\n",
    "        if config[\"network\"] == \"MnistReLU\":\n",
    "            sparsity = torch.sum(zhat != 0, dim=1).float().mean()\n",
    "            if sparsity > config[\"net_params\"][\"k\"]:\n",
    "                loss += reg_penalty * zhat.abs().sum(dim=1).mean()\n",
    "            if sparsity > 1.5 * config[\"net_params\"][\"k\"]:\n",
    "                reg_penalty *= 1.001\n",
    "\n",
    "        # JumpReLU L0 target loss\n",
    "        if config[\"network\"] == \"MnistJumpReLU\":\n",
    "            loss += (\n",
    "                reg_penalty\n",
    "                * (\n",
    "                    (\n",
    "                        step(\n",
    "                            (net.W * (x - net.b_pre)).sum(axis=[-1, -2]) + net.b_enc,\n",
    "                            threshold=torch.exp(net.log_thresholds),\n",
    "                        ).sum(dim=1)\n",
    "                        / config[\"net_params\"][\"k\"]\n",
    "                    )\n",
    "                    - 1\n",
    "                )\n",
    "                ** 2\n",
    "            ).mean()\n",
    "\n",
    "        # TopK / BatchTopK auxiliary loss\n",
    "        if config[\"network\"] == \"MnistTopK\" or config[\"network\"] == \"MnistBatchTopK\":\n",
    "            projection = (net.W * (x - net.b_pre)).sum(axis=[-1, -2]) + net.b_enc\n",
    "            unused_codes = torch.relu(projection) - zhat\n",
    "            auxiliary_topk = torch.topk(unused_codes, k=net.p // 2, dim=1)\n",
    "            top_unused_codes = torch.zeros_like(zhat).scatter(\n",
    "                1, auxiliary_topk.indices, auxiliary_topk.values\n",
    "            )\n",
    "\n",
    "            residual_hat = net.decode(top_unused_codes)\n",
    "            auxilary_mse = torch.sum(\n",
    "                (x - (xhat + residual_hat)) ** 2, dim=[-1, -2]\n",
    "            ).mean()\n",
    "            loss += reg_penalty * auxilary_mse\n",
    "\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        if config[\"net_params\"][\"normalize\"]:\n",
    "            net.normalize(config[\"net_params\"][\"normalize\"])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "282de3a3",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(net.state_dict(), f'{config[\"network\"]}_p{net.p}_k{config[\"net_params\"][\"k\"]}.pt')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7a9af8c6",
   "metadata": {},
   "source": [
    "## Evaluation "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "7de96020",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 32,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Remove the SAEs_trained/ folder if you want to use the newly trained network instead of loading a saved one\n",
    "net.load_state_dict(torch.load(f'SAEs_trained/{config[\"network\"]}_p{net.p}_k{config[\"net_params\"][\"k\"]}.pt', weights_only=True))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "438c3c66",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "MnistReLU()"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "net.eval()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "db7eb198",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset = MNIST(\"data\", train=True, transform=transform)\n",
    "val_dataset = torch.utils.data.Subset(dataset, range(50000,len(dataset)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "b17ac05a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "index = 18\n",
    "x_gt, _ = val_dataset[index] \n",
    "\n",
    "plt.imshow(x_gt[0], cmap=\"gray\")\n",
    "plt.axis(\"off\");"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "ddb664de",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    x = x_gt.to(config[\"device\"]).unsqueeze(0)\n",
    "    z_hat = net.encode(x)\n",
    "    x_hat = net.decode(z_hat)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "0f1b5aaa",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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/LJOOOEEtWzZUk2bNtVRRx2l559/fp/npfx6+NOf/qTvfve76ty5s5o2baoNGzYUtNzFixfrwgsvVKdOndSoUSP17NlTX/ziF7Vjx46K1xRyfT7wwAO64YYb1KVLFzVu3FjHHnusZs+eXdBxLyoq0ubNm3X33XdXjBfl2SV7O+9HH310pXGlnMsb2rNnj37+859ryJAhaty4scrKynTCCSfo1Vdf3ec2xDIh7rjjDg0aNEiNGjVSp06ddMkllwR/LXD00Udr8ODBmjp1qkaPHq2mTZuqc+fO+vGPfxxst/O3v/1NPXr0UO/eve1/nzt3rsaMGaOSkhJ16tRJ1157bXCNpVyLsf2/+uqr9Y1vfEOS1LNnz4r/Vn4sUsZS6e17wsknn6xnnnlGw4YNU5MmTTRkyJCKvwD8y1/+UnFuhg4dqtdff32fx6b8vDz77LP60pe+pHbt2qlLly4V/31f5+fWW29VvXr1KtV++tOfqqioSF/96lcrart371bz5s31rW99q9L6P/KRj+if//xncLwBAMCHE38JAQDAe6B79+568cUXNWXKFA0ePHivr73hhhv0ve99T2eddZYuuugirVy5UrfddpuOPPJIvf766/ZBRrnPf/7z+t3vfqcLLrhAl112mebNm6fbb79dr7/+up5//nk1aNBA0ttfPnz2s5/VoEGDdMUVV6hVq1Z6/fXX9dhjj+ncc8/VlVdeqfXr12vRokW6+eabJakixHPPnj069dRT9Z///EcXX3yxBgwYoDfeeEM333yzZs6cWennKC666CLdc889Ovfcc3X44Yfrqaee0tixY5OO2YYNG/Sb3/xGn/jEJ/S5z31OGzdu1G9/+1uNGTNGEyZM0IEHHqiysjL94he/0Be/+EV97GMf0+mnny5J2n///SVJ48aN04knnqhevXrp6quv1tatW3XbbbfpiCOO0GuvvRZ80Xb22WdrwIAB+uEPf6hHHnlE119/vUpLS3XnnXfqmGOO0Y9+9CPde++9+vrXv67hw4fryCOPlCT9+te/1mWXXaYzzjhD/+///T9t27ZN//3vf/Xyyy/r3HPPje7j4sWL9dZbb+nggw+uVL/lllt06aWXqlmzZrryyislSe3bt5f09pf2hx9+uLZs2aLLLrtMbdq00d13361TTz1VDz30kD72sY9VWtYNN9ygoqIifetb39KKFSt0yy236LjjjtOkSZPUpEmTfZ6HM888U3379tWNN95Y8WXRRRddpLvvvltnnHGGvva1r+nll1/WD37wA02bNk1//etfJUmjRo3SzTffrDfffLOi58ePH6/i4mKNHz9el112WUVNUsWxLDd06FD9/e9/14YNG9SiRYvo9r3yyit64YUXdM4556hLly6aP3++fvGLX+joo4/W1KlT1bRp00qvv/TSS9W6dWtdddVVmj9/vm655RZ9+ctf1v3331/xmu9///u6/vrrddJJJ+mkk07Sa6+9puOPP77SF8378vvf/14bN27UJZdcom3btunnP/+5jjnmGL3xxhsV51J6+0vIMWPGaOTIkfrJT36ipk2bKssynXrqqXr66ad14YUX6sADD9Tjjz+ub3zjG1q8eHHFNSm9/YXxAw88oE9/+tM69NBD9eyzz+71GnPnM2XM2bFjh8aMGaPt27fr0ksvVYcOHbR48WI9/PDDWrdunVq2bKk333xTJ598svbff39de+21atSokWbPnr3XhwRnnXWWvvnNb+qBBx6o+LK23AMPPKDjjz9erVu3lvT2A8UTTzxRQ4cO1VVXXaXi4mLdddddOuaYYzR+/HiNGDFin+fluuuuU8OGDfX1r39d27dvV8OGDZOXu2TJEo0YMULr1q3TxRdfrP79+2vx4sV66KGHtGXLFjVs2LDg6/OHP/yhiouL9fWvf13r16/Xj3/8Y33yk5/Uyy+/LElJx/0Pf/iDLrroIo0YMUIXX3yxJAVfurvzXogLL7xQv/vd73TiiSfqoosu0q5duzR+/Hi99NJLGjZsWNI2vNPVV1+ta665Rscdd5y++MUvasaMGfrFL36hV155pdK9SpLWrl2rE044QaeffrrOOussPfTQQ/rWt76lIUOG6MQTT9zrdr/wwgvB+Fpu9+7dOuGEE3TooYfqxz/+sR577DFdddVV2rVrV8VfYKVei7H9Lykp0cyZM3Xffffp5ptvVtu2bSWp4uF5ylhabvbs2Tr33HP1+c9/Xp/61Kf0k5/8RKeccop++ctf6jvf+Y6+9KUvSZJ+8IMf6KyzztKMGTOS/srnS1/6ksrKyvT9739fmzdvTj4/o0aN0p49e/Sf//xHJ598sqTKY3y5119/XZs2bbJjfNV7BAAA+BDLAABAtT3xxBNZvXr1snr16mWHHXZY9s1vfjN7/PHHsx07dlR63fz587N69eplN9xwQ6X6G2+8kdWvX79S/bzzzsu6d+9e8b/Hjx+fScruvffeSu997LHHKtXXrVuXNW/ePDvkkEOyrVu3Vnrtnj17Kv7/sWPHVlp+uT/84Q9ZcXFxNn78+Er1X/7yl5mk7Pnnn8+yLMsmTZqUScq+9KUvVXrdueeem0nKrrrqKnOk/s+uXbuy7du3V6qtXbs2a9++ffbZz362orZy5cro8g488MCsXbt22erVqytqkydPzoqLi7PPfOYzFbWrrroqk5RdfPHFldbfpUuXrKioKPvhD39YaRuaNGmSnXfeeRW10047LRs0aNBe98cZN25cJin75z//Gfy3QYMGZUcddVRQv/zyyzNJlY7/xo0bs549e2Y9evTIdu/enWVZlj399NOZpKxz587Zhg0bKl77wAMPZJKyn//853vdtvJj8olPfKJSvfy8XnTRRZXqX//61zNJ2VNPPZVlWZatWLEik5TdcccdWZa93XfFxcXZmWeembVv377ifZdddllWWlpaqfeyLMv++Mc/ZpKyl19+ea/buWXLlqD24osvZpKy3//+9xW1u+66K5OUHXfccZXW9ZWvfCWrV69etm7duortbtiwYTZ27NhKr/vOd76TSap03p158+ZlkrImTZpkixYtqqi//PLLmaTsK1/5SkXtvPPOyyRl3/72tyst429/+1smKbv++usr1c8444ysqKgomz17dpZlWTZx4sRMUnb55ZdXet35558fXBOx85k65rz++uuZpOzBBx+M7vvNN9+cScpWrlwZfU358bnrrrsqaocddlg2dOjQSq+bMGFCpXO4Z8+erG/fvtmYMWMqnZctW7ZkPXv2zD7ykY9E15ll/3c99OrVq1LPFLLcz3zmM1lxcXH2yiuvBMsvf2+h1+eAAQMqjXM///nPM0nZG2+8kWVZ2nHPsiwrKSmxvRk771mWZUcddZQdY6reW5566qlMUnbZZZdF93tv21B+7c2bNy/Lsv+7xo4//viK45FlWXb77bdnkrL//d//rbSNVa/l7du3Zx06dMg+/vGPB+t6p507d2ZFRUXZ1772NbuPkrJLL7200r6MHTs2a9iwYUUPp16Le9v/m266qdL+l0sdS7Msy7p3755Jyl544YWK2uOPP14x1ixYsKCifuedd2aSsqeffjp+cLL/Oy8jR47Mdu3aVVFPPT+7d+/OWrRokX3zm9/Msuzt49emTZvszDPPzOrVq5dt3Lgxy7Is+9nPfpYVFxdna9eurbT+F154IZOU3X///XvdTgAA8OHAzzEBAPAe+MhHPqIXX3xRp556qiZPnqwf//jHGjNmjDp37qx//OMfFa/7y1/+oj179uiss87SqlWrKv6vQ4cO6tu3b6WfIqrqwQcfVMuWLfWRj3yk0nuHDh2qZs2aVbz33//+tzZu3Khvf/vbwW+zv/Mngfa2ngEDBqh///6V1lP+u87l6/nXv/4lSRX/4r3c5Zdfvu8Dprd/R7xhw4aS3v7rizVr1mjXrl0aNmyYXnvttX2+f+nSpZo0aZLOP/98lZaWVtT3339/feQjH6nYvne66KKLKq1/2LBhyrJMF154YUW9VatW2m+//TR37txKtUWLFumVV15J2rdyq1evlqSKf+md4l//+pdGjBhR6ad0mjVrposvvljz58/X1KlTK73+M5/5jJo3b17xv8844wx17NjR7r9TNTC7/H3v/LkNSfra174mSRU/O1NWVqb+/fvrueeekyQ9//zzqlevnr7xjW9o+fLlmjVrlqS3/+XsyJEjg94rPyarVq3a6/a98685du7cqdWrV6tPnz5q1aqV7ZOLL7640rpGjRql3bt3V/wc0Lhx47Rjx46Kn44ql9q35T760Y+qc+fOFf97xIgROuSQQ+xx/+IXv1jpf//rX/9SvXr1gmvna1/7mrIs06OPPipJeuyxxySp4l9Al7v00kuj21X1fKaOOS1btpQkPf7449qyZYtddvlfaf3973/Xnj17ottQ1dlnn62JEydWCiK///771ahRI5122mmS3g5wnzVrls4991ytXr26Yjs3b96sY489Vs8991zSOs8777xKPZO63D179uhvf/ubTjnlFA0bNixYbnmvFHp9XnDBBRXjnPR2P0qqGF9SjnuKque9EH/+859VVFSkq666KvhvKfeMqsqvscsvv7zSv9T/3Oc+pxYtWgQ/XdWsWTN96lOfqvjfDRs21IgRIyqNwc6aNWuUZdlex9cvf/nLFf9/UVGRvvzlL2vHjh0aN26cpPRr8d1IHUvLDRw4UIcddljF/z7kkEMkScccc4y6desW1Pd1fMp97nOfq5TZkXp+iouLdfjhh1eM8dOmTdPq1av17W9/W1mW6cUXX5T09hg/ePDg4K84U8d4AADw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7PHHH6/mzZvrBz/4gbZt2xaso1xJSYn9yY+zzjpLixcv1q9//evgv23durXiJ45OPPFESdKtt95a6TW33HJLdB/eqfxfd75zm15++eWK35ku17RpU0kKvnTu2LGjDjzwQN19992V/tuUKVP0xBNP6KSTTkrajhRVz0vDhg01cOBAZVmmnTt3Rt/XuXNnde3aVa+++mrw30pKSuwX6SeddJImTJhQ6Ths3rxZv/rVr9SjRw8NHDiw0ut///vfa+PGjRX/+6GHHtLSpUsrzk+hyo9b1fNY/pcxY8eOraj17NlTnTt31s0336ydO3fqiCOOkPT2w4k5c+booYce0qGHHqr69cN/9zJx4kQNGjSo4qdFYurVqxdcK7fddtu7/tfcxx13nBo0aKDbbrut0nJT+7bc3/72Ny1evLjif0+YMEEvv/xy0nE/6aSTtHv3bt1+++2V6jfffLOKiooqljFmzBhJ0h133FHpdbfddlvydqaOORs2bNCuXbsq/fchQ4aouLhY27dvlyT71wLlf11Q/pqYj3/846pXr57uu+8+Pfjggzr55JMrPSQcOnSoevfurZ/85CfatGlT8P6VK1fue2eN1OUWFxfrox/9qP75z3/a67X82BV6fe5LynGX4uPF3vTu3bviX6qXmzx5cvBTch//+MeVZZn9q66q94yUbTjuuOPUsGFD3XrrrZXe/9vf/lbr16+vNIZU12GHHWbPV7l3XmNZlun2229XgwYNdOyxx0pKvxal+P6X93HV/1bIWFqTCjk/jRs31vDhw3XffffprbfeqvSXEFu3btWtt96q3r17q2PHjsF6Jk6cqJYtW2rQoEH57xQAAKjz+EsIAADeA5deeqm2bNmij33sY+rfv7927NihF154Qffff7969OihCy64QNLbXwpdf/31uuKKKzR//nx99KMfVfPmzTVv3jz99a9/1cUXX6yvf/3rdh1HHXWUPv/5z+sHP/iBJk2apOOPP14NGjTQrFmz9OCDD+rnP/+5zjjjDLVo0UI333yzLrroIg0fPlznnnuuWrdurcmTJ2vLli26++67Jb395dz999+vr371qxo+fLiaNWumU045RZ/+9Kf1wAMP6Atf+IKefvppHXHEEdq9e7emT5+uBx54QI8//riGDRumAw88UJ/4xCd0xx13aP369Tr88MP15JNPJv8r7ZNPPll/+ctf9LGPfUxjx47VvHnz9Mtf/lIDBw6s9GVhkyZNNHDgQN1///3q16+fSktLNXjwYA0ePFg33XSTTjzxRB122GG68MILtXXrVt12221q2bKlrr766uqd1Hc4/vjj1aFDBx1xxBFq3769pk2bpttvv11jx44Nfku8qtNOO01//etflWVZpX8xO3ToUP3iF7/Q9ddfrz59+qhdu3Y65phj9O1vf1v33XefTjzxRF122WUqLS3V3XffrXnz5unPf/5z8NM+paWlGjlypC644AItX75ct9xyi/r06aPPfe5z72pfDzjgAJ133nn61a9+pXXr1umoo47ShAkTdPfdd+ujH/2oRo8eXen1o0aN0p/+9CcNGTKk4vfuDz74YJWUlGjmzJk699xzg3Xs3LlTzz77bBC47Jx88sn6wx/+oJYtW2rgwIF68cUXNW7cuIqfuipUWVmZvv71r+sHP/iBTj75ZJ100kl6/fXX9eijjxb0syF9+vTRyJEj9cUvflHbt2/XLbfcojZt2uib3/zmPt97yimnaPTo0bryyis1f/58HXDAAXriiSf097//XZdffnnFv94fOnSoPv7xj+uWW27R6tWrdeihh+rZZ5+t+NfcKf8yPXXMeeqpp/TlL39ZZ555pvr166ddu3bpD3/4g+rVq6ePf/zjkqRrr71Wzz33nMaOHavu3btrxYoVuuOOO9SlS5dKQc1Ou3btNHr0aP3sZz/Txo0bdfbZZ1f678XFxfrNb36jE088UYMGDdIFF1ygzp07a/HixXr66afVokUL/fOf/9zn/lZVyHJvvPFGPfHEEzrqqKN08cUXa8CAAVq6dKkefPBB/ec//1GrVq0Kvj73JeW4S2/3wrhx4/Szn/1MnTp1Us+ePSsCimM++9nP6mc/+5nGjBmjCy+8UCtWrNAvf/lLDRo0SBs2bKh43ejRo/XpT39at956q2bNmqUTTjhBe/bs0fjx4zV69OiKcOfUbSgrK9MVV1yha665RieccIJOPfVUzZgxQ3fccYeGDx9eKYS6uk477TT94Q9/0MyZM4O/9GncuLEee+wxnXfeeTrkkEP06KOP6pFHHtF3vvOdip8vTL0W97b/5X/pdeWVV+qcc85RgwYNdMoppxQ8ltaUQs/PqFGj9MMf/lAtW7bUkCFDJL19Pe+3336aMWOGzj//fLuef//73zrllFPIhAAAAG/LAABAtT366KPZZz/72ax///5Zs2bNsoYNG2Z9+vTJLr300mz58uXB6//85z9nI0eOzEpKSrKSkpKsf//+2SWXXJLNmDGj4jXnnXde1r179+C9v/rVr7KhQ4dmTZo0yZo3b54NGTIk++Y3v5ktWbKk0uv+8Y9/ZIcffnjWpEmTrEWLFtmIESOy++67r+K/b9q0KTv33HOzVq1aZZIqrWvHjh3Zj370o2zQoEFZo0aNstatW2dDhw7Nrrnmmmz9+vUVr9u6dWt22WWXZW3atMlKSkqyU045JVu4cGEmKbvqqqv2esz27NmT3XjjjVn37t2zRo0aZQcddFD28MMP2/1+4YUXsqFDh2YNGzYMlj1u3LjsiCOOqNjPU045JZs6dWql91911VWZpGzlypWV6uedd15WUlISbNtRRx2VDRo0qOJ/33nnndmRRx6ZtWnTJmvUqFHWu3fv7Bvf+EalYxHz2muvZZKy8ePHV6ovW7YsGzt2bNa8efNMUnbUUUdV/Lc5c+ZkZ5xxRtaqVauscePG2YgRI7KHH3640vuffvrpTFJ23333ZVdccUXWrl27rEmTJtnYsWOzBQsW7HO7Yscky7Js586d2TXXXJP17Nkza9CgQda1a9fsiiuuyLZt2xa89n/+538ySdkXv/jFSvXjjjsuk5Q9+eSTwXseffTRTFI2a9asfW7n2rVrswsuuCBr27Zt1qxZs2zMmDHZ9OnTs+7du2fnnXdexevuuuuuTFL2yiuvVHp/+XF6+umnK2q7d+/Orrnmmqxjx45ZkyZNsqOPPjqbMmVKsExn3rx5maTspptuyn76059mXbt2zRo1apSNGjUqmzx5cqXXxvory7Js48aN2Ve+8pWsU6dOWYMGDbK+fftmN910U7Znz55Kr9u8eXN2ySWXZKWlpVmzZs2yj370o9mMGTMySdkPf/jDitft7Xxm2b7HnLlz52af/exns969e2eNGzfOSktLs9GjR2fjxo2rWMaTTz6ZnXbaaVmnTp2yhg0bZp06dco+8YlPZDNnzgyOz1133RVsw69//etMUta8efNs69atdjtff/317PTTT6+41rp3756dddZZto/eqfw8P/jgg9Va7oIFC7LPfOYzWVlZWdaoUaOsV69e2SWXXJJt37694jWFXJ9Vt6fq8Uk57lmWZdOnT8+OPPLIrEmTJpmkij7d13m/5557sl69emUNGzbMDjzwwOzxxx+3Y+yuXbuym266Kevfv3/WsGHDrKysLDvxxBOziRMn7nMbyq+9efPmVVrm7bffnvXv3z9r0KBB1r59++yLX/xitnbt2kqvqTrWlovd/6ravn171rZt2+y6664L3l9SUpLNmTMnO/7447OmTZtm7du3z6666qps9+7dlV6bei3G9j/Lsuy6667LOnfunBUXF1c6Fqljaffu3bOxY8cG+ycpu+SSSyrV3jkG7U1sTCyXcn6yLMseeeSRTFJ24oknVqpfdNFFmaTst7/9bfCeadOmZZKCPgYAAB9eRVlWSyl9AAAAHxLHHnusOnXqpD/84Q/v2TKfeeYZjR49Wg8++KDOOOOM92y5NeGjH/2oioqK9Ne//rW2N6Vg8+fPV8+ePXXTTTdF/2opb5MmTdJBBx2ke+65R5/85CdrZRuAuuK6667TXXfdpVmzZkVDnFGzLr/8cj333HOaOHEifwkBAAAkkQkBAACQuxtvvFH3339/EAj7YTRt2jQ9/PDDNtsEoa1btwa1W265RcXFxTYMFviw+cpXvqJNmzbpT3/6U21vCvR2htJvfvMbXX/99TyAAAAAFciEAAAAyNkhhxyiHTt21PZm1AkDBgwIgngR9+Mf/1gTJ07U6NGjVb9+fT366KN69NFHdfHFF6tr1661vXlArWvWrJlWrFhR25uB/1+bNm1sCDwAAPhw4yEEAAAAUEcdfvjh+ve//63rrrtOmzZtUrdu3XT11VfryiuvrO1NAwAAAIAkZEIAAAAAAAAAAIBckAkBAAAAAAAAAABywUMIAAAAAAAAAACQCx5CAAAAAAAAAACAXPAQAgAAAAAAAAAA5IKHEAAAAAAAAAAAIBc8hAAAAAAAAAAAALngIQQAAAAAAAAAAMgFDyEAAAAAAAAAAEAueAgBAAAAAAAAAABywUMIAAAAAAAAAACQCx5CAAAAAAAAAACAXPAQAgAAAAAAAAAA5IKHEAAAAAAAAAAAIBc8hAAAAAAAAAAAALngIQQAAAAAAAAAAMgFDyEAAAAAAAAAAEAueAgBAAAAAAAAAABywUMIAAAAAAAAAACQCx5CAAAAAAAAAACAXPAQAgAAAAAAAAAA5IKHEAAAAAAAAAAAIBc8hAAAAAAAAAAAALngIQQAAAAAAAAAAMgFDyEAAAAAAAAAAEAueAhRTdu3b9e3vvUtderUSU2aNNEhhxyif//737W9WcA+bdq0SVdddZVOOOEElZaWqqioSL/73e9qe7OAfXrllVf05S9/WYMGDVJJSYm6deums846SzNnzqztTQP26c0339SZZ56pXr16qWnTpmrbtq2OPPJI/fOf/6ztTQMKcsMNN6ioqEiDBw+u7U0B9uqZZ55RUVGR/b+XXnqptjcPSPLaa6/p1FNPVWlpqZo2barBgwfr1ltvre3NAvbq/PPPj46/RUVFWrx4cW1vIhA1a9YsnXPOOerSpYuaNm2q/v3769prr9WWLVtqe9Pet+rX9ga8351//vl66KGHdPnll6tv37763e9+p5NOOklPP/20Ro4cWdubB0StWrVK1157rbp166YDDjhAzzzzTG1vEpDkRz/6kZ5//nmdeeaZ2n///bVs2TLdfvvtOvjgg/XSSy/xhRjqtAULFmjjxo0677zz1KlTJ23ZskV//vOfdeqpp+rOO+/UxRdfXNubCOzTokWLdOONN6qkpKS2NwVIdtlll2n48OGVan369KmlrQHSPfHEEzrllFN00EEH6Xvf+56aNWumOXPmaNGiRbW9acBeff7zn9dxxx1XqZZlmb7whS+oR48e6ty5cy1tGbB3Cxcu1IgRI9SyZUt9+ctfVmlpqV588UVdddVVmjhxov7+97/X9ia+LxVlWZbV9ka8X02YMEGHHHKIbrrpJn3961+XJG3btk2DBw9Wu3bt9MILL9TyFgJx27dv19q1a9WhQwe9+uqrGj58uO666y6df/75tb1pwF698MILGjZsmBo2bFhRmzVrloYMGaIzzjhD99xzTy1uHVC43bt3a+jQodq2bZumT59e25sD7NM555yjlStXavfu3Vq1apWmTJlS25sERD3zzDMaPXq0HnzwQZ1xxhm1vTlAQTZs2KB+/frp8MMP10MPPaTiYn7MAu9v//nPfzRq1CjdcMMN+s53vlPbmwNYN954o6688kpNmTJFgwYNqqifd955+v3vf681a9aodevWtbiF70/cwarhoYceUr169Sr9q8XGjRvrwgsv1IsvvqiFCxfW4tYBe9eoUSN16NChtjcDKNjhhx9e6QGEJPXt21eDBg3StGnTammrgHevXr166tq1q9atW1fbmwLs03PPPaeHHnpIt9xyS21vClCwjRs3ateuXbW9GUCyP/7xj1q+fLluuOEGFRcXa/PmzdqzZ09tbxbwrv3xj39UUVGRzj333NreFCBqw4YNkqT27dtXqnfs2FHFxcXB9xFIw0OIanj99dfVr18/tWjRolJ9xIgRkqRJkybVwlYBwIdPlmVavny52rZtW9ubAiTZvHmzVq1apTlz5ujmm2/Wo48+qmOPPba2NwvYq927d+vSSy/VRRddpCFDhtT25gAFueCCC9SiRQs1btxYo0eP1quvvlrbmwTs07hx49SiRQstXrxY++23n5o1a6YWLVroi1/8orZt21bbmwcUZOfOnXrggQd0+OGHq0ePHrW9OUDU0UcfLUm68MILNWnSJC1cuFD333+/fvGLX+iyyy7jJ0nfJTIhqmHp0qXq2LFjUC+vLVmypKY3CQA+lO69914tXrxY1157bW1vCpDka1/7mu68805JUnFxsU4//XTdfvvttbxVwN798pe/1IIFCzRu3Lja3hQgWcOGDfXxj39cJ510ktq2baupU6fqJz/5iUaNGqUXXnhBBx10UG1vIhA1a9Ys7dq1S6eddpouvPBC/eAHP9Azzzyj2267TevWrdN9991X25sIJHv88ce1evVqffKTn6ztTQH26oQTTtB1112nG2+8Uf/4xz8q6ldeeaWuv/76Wtyy9zceQlTD1q1b1ahRo6DeuHHjiv8OAMjX9OnTdckll+iwww7TeeedV9ubAyS5/PLLdcYZZ2jJkiV64IEHtHv3bu3YsaO2NwuIWr16tb7//e/re9/7nsrKymp7c4Bkhx9+uA4//PCK/33qqafqjDPO0P77768rrrhCjz32WC1uHbB3mzZt0pYtW/SFL3xBt956qyTp9NNP144dO3TnnXfq2muvVd++fWt5K4E0f/zjH9WgQQOdddZZtb0pwD716NFDRx55pD7+8Y+rTZs2euSRR3TjjTeqQ4cO+vKXv1zbm/e+xEOIamjSpIm2b98e1Mv/LLJJkyY1vUkA8KGybNkyjR07Vi1btqzI6QHeD/r376/+/ftLkj7zmc/o+OOP1ymnnKKXX35ZRUVFtbx1QOi73/2uSktLdemll9b2pgDV1qdPH5122mn6y1/+ot27dzN/QJ1V/p3CJz7xiUr1c889V3feeadefPFFHkLgfWHTpk36+9//rjFjxqhNmza1vTnAXv3pT3/SxRdfrJkzZ6pLly6S3n4AvGfPHn3rW9/SJz7xCfr4XSAToho6duyopUuXBvXyWqdOnWp6kwDgQ2P9+vU68cQTtW7dOj322GOMuXhfO+OMM/TKK69o5syZtb0pQGDWrFn61a9+pcsuu0xLlizR/PnzNX/+fG3btk07d+7U/PnztWbNmtreTKAgXbt21Y4dO7R58+ba3hQgqnx+WzUctV27dpKktWvX1vg2Ae/G3/72N23ZsoWfYsL7wh133KGDDjqo4gFEuVNPPVVbtmzR66+/Xktb9v7GQ4hqOPDAAzVz5syK1PRyL7/8csV/BwC897Zt26ZTTjlFM2fO1MMPP6yBAwfW9iYB1VL+E47r16+v5S0BQosXL9aePXt02WWXqWfPnhX/9/LLL2vmzJnq2bMnmTx435k7d64aN26sZs2a1famAFFDhw6V9PY4/E7l+ZP8PB7eL+699141a9ZMp556am1vCrBPy5cv1+7du4P6zp07JUm7du2q6U36QOAhRDWcccYZ2r17t371q19V1LZv36677rpLhxxyiLp27VqLWwcAH0y7d+/W2WefrRdffFEPPvigDjvssNreJCDZihUrgtrOnTv1+9//Xk2aNOGBGuqkwYMH669//Wvwf4MGDVK3bt3017/+VRdeeGFtbyZgrVy5MqhNnjxZ//jHP3T88ceruJiPxKi7yn87/7e//W2l+m9+8xvVr19fRx99dC1sFVCYlStXaty4cfrYxz6mpk2b1vbmAPvUr18/vf7668Ffqd93330qLi7W/vvvX0tb9v5GJkQ1HHLIITrzzDN1xRVXaMWKFerTp4/uvvtuzZ8/P5gkAHXR7bffrnXr1lX8S5p//vOfWrRokSTp0ksvVcuWLWtz8wDra1/7mv7xj3/olFNO0Zo1a3TPPfdU+u+f+tSnamnLgH37/Oc/rw0bNujII49U586dtWzZMt17772aPn26fvrTn/IvclEntW3bVh/96EeD+i233CJJ9r8BdcXZZ5+tJk2a6PDDD1e7du00depU/epXv1LTpk31wx/+sLY3D9irgw46SJ/97Gf1v//7v9q1a5eOOuooPfPMM3rwwQd1xRVX8HOkeF+4//77tWvXLn6KCe8b3/jGN/Too49q1KhR+vKXv6w2bdro4Ycf1qOPPqqLLrqIsfddKsqyLKvtjXg/27Ztm773ve/pnnvu0dq1a7X//vvruuuu05gxY2p704B96tGjhxYsWGD/27x589Sj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",
      "text/plain": [
       "<Figure size 2000x500 with 18 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "z_hat_L0 = int((z_hat != 0).sum())\n",
    "atoms_indices_sorted = torch.argsort(z_hat.squeeze(0), descending=True)[:z_hat_L0]\n",
    "atoms = net.D[atoms_indices_sorted].detach().cpu()\n",
    "\n",
    "fig, axes = plt.subplots(2, z_hat_L0 + 1, figsize=(20,5))\n",
    "fig.suptitle(\"Selected atoms (top row) and progressive reconstruction (bottom row)\")\n",
    "for i in range(len(atoms_indices_sorted)+1):\n",
    "    if i == 0:\n",
    "        axes[0, i].imshow(torch.zeros(28, 28), cmap=\"gray\")\n",
    "        axes[0, i].axis(\"off\")\n",
    "    else:\n",
    "        axes[0, i].imshow(atoms[i-1], cmap=\"gray\")\n",
    "        axes[0, i].axis(\"off\")\n",
    "    axes[0, i].set_title(i)\n",
    "\n",
    "    if i == 0:\n",
    "        axes[1, i].imshow(net.b_pre.detach().cpu()[0], cmap=\"gray\")\n",
    "        axes[1, i].axis(\"off\")\n",
    "    else:\n",
    "        z_ = torch.zeros_like(z_hat)\n",
    "        z_[:, atoms_indices_sorted[:i]] = z_hat[:, atoms_indices_sorted[:i]]\n",
    "        axes[1, i].imshow(net.decode(z_)[0,0].detach().cpu(), cmap=\"gray\")\n",
    "        axes[1, i].axis(\"off\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.11.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
