{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Minimal Bidirectional Predictive Coding (bPC) Model Demo\n",
    "\n",
    "This notebook demonstrates a minimal implementation and training of a bidirectional predictive coding (bPC) model on MNIST, using JAX and Optax. No external project utilities are used."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import jax\n",
    "import jax.numpy as jnp\n",
    "import optax\n",
    "import numpy as np\n",
    "from functools import partial\n",
    "from tqdm import trange\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from torchvision import datasets, transforms\n",
    "from torch.utils.data import DataLoader, Subset"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Load and preprocess MNIST data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "batch_size = 256\n",
    "data_shape = (784,)\n",
    "\n",
    "transform = transforms.Compose([transforms.ToTensor(), transforms.Normalize((0.5,), (0.5,)),])\n",
    "train_dataset = datasets.MNIST(\n",
    "    \"./data\", download=True, train=True, transform=transform\n",
    ")\n",
    "val_dataset = datasets.MNIST(\n",
    "    \"./data\", download=True, train=False, transform=transform\n",
    ")\n",
    "\n",
    "def convert_dataset_to_numpy(dataset):\n",
    "    data_loader = DataLoader(dataset, batch_size=len(dataset), shuffle=False)\n",
    "    data, label = list(data_loader)[0]\n",
    "    nm_elements = len(data)\n",
    "    X = (jax.nn.one_hot(label.numpy(), 10))\n",
    "    X = X[: batch_size * (nm_elements // batch_size)]\n",
    "    y = data.numpy()[: batch_size * (nm_elements // batch_size)]\n",
    "\n",
    "    return list(\n",
    "        zip(\n",
    "            X.reshape(-1, batch_size, 10),\n",
    "            y.reshape(-1, batch_size, *data_shape),\n",
    "        )\n",
    "    )\n",
    "\n",
    "# Process the datasets into numpy arrays\n",
    "train_dl = convert_dataset_to_numpy(train_dataset)\n",
    "val_dl = convert_dataset_to_numpy(val_dataset)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Define a minimal MLP for bPC"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pcax as px\n",
    "import pcax.predictive_coding as pxc \n",
    "import pcax.nn as pxnn\n",
    "\n",
    "class Model(pxc.EnergyModule):\n",
    "    def __init__(\n",
    "        self,\n",
    "        hidden_dim: int,\n",
    "        nm_layers: int,\n",
    "        activation: str,\n",
    "        alpha_up=1.0,\n",
    "        alpha_down=1.0,\n",
    "    ) -> None:\n",
    "        super().__init__()\n",
    "\n",
    "        assert nm_layers >= 3, \"Number of layers must be at least 3\"\n",
    "\n",
    "        self.key = px.RKG()\n",
    "\n",
    "        self.alpha_up = alpha_up\n",
    "        self.alpha_down = alpha_down\n",
    "\n",
    "        activation = getattr(jax.nn, activation)\n",
    "        self.activation = px.static(activation)\n",
    "\n",
    "        # set initialisation of neural activity\n",
    "        # here we set it with a feedforward sweep\n",
    "        ruleset = {\"pxc.STATUS.INIT\": (\"h, u <- u\",),}\n",
    "        tforms = {}\n",
    "\n",
    "        self.vodes = [\n",
    "                pxc.Vode(\n",
    "                    (10,),  # number of classes\n",
    "                    ruleset=ruleset,\n",
    "                    tforms=tforms,\n",
    "                )\n",
    "            ] + [\n",
    "                pxc.Vode(\n",
    "                    (hidden_dim,),\n",
    "                    ruleset=ruleset,\n",
    "                    tforms=tforms,\n",
    "                )\n",
    "                for _ in range(nm_layers - 2)\n",
    "            ] + [\n",
    "                pxc.Vode(\n",
    "                    (784,), # image shape\n",
    "                    ruleset=ruleset,\n",
    "                    tforms=tforms,\n",
    "                )\n",
    "            ]\n",
    "        self.vodes[0].h.frozen = True  # fixed latent state\n",
    "        self.vodes[-1].h.frozen = True  # fixed data\n",
    "\n",
    "        # setup up down layers\n",
    "        self.layers_down = [\n",
    "                    pxnn.Linear(10, hidden_dim)\n",
    "            ] + [\n",
    "                    pxnn.Linear(hidden_dim, hidden_dim)\n",
    "                    for _ in range(nm_layers - 3)\n",
    "            ] + [\n",
    "                    pxnn.Linear(hidden_dim, 784)\n",
    "            ]\n",
    "        \n",
    "        # setup up layers\n",
    "        self.layers_up = [\n",
    "                    pxnn.Linear(784, hidden_dim)\n",
    "            ] + [\n",
    "                        pxnn.Linear(hidden_dim, hidden_dim)\n",
    "                        for _ in range(nm_layers - 3)\n",
    "            ] + [pxnn.Linear(hidden_dim, 10)]\n",
    "\n",
    "\n",
    "        # setup the down model as sequence of layers\n",
    "        self.down = [self.vodes[0]]\n",
    "        for idx, (v, l) in enumerate(zip(self.vodes[1:], self.layers_down)):\n",
    "            if idx == len(self.layers_down) - 1:\n",
    "                act_fn = px.static(jax.nn.tanh) # final layer activation\n",
    "            else:\n",
    "                act_fn = self.activation\n",
    "            self.down += [l, act_fn, v]\n",
    "\n",
    "        # setup up layers\n",
    "        self.up = [self.vodes[-1]]\n",
    "        for idx, (v, l) in enumerate(zip(self.vodes[-2::-1], self.layers_up)):\n",
    "            if idx == len(self.layers_up) - 1:\n",
    "                act_fn = px.static(lambda x: x) # final layer activation\n",
    "            else:\n",
    "                act_fn = self.activation\n",
    "            self.up += [l, act_fn, v]\n",
    "\n",
    "    def model_down(self, x):\n",
    "        # input is not used be needed for parallelism using vmap\n",
    "        input = self.down[0].get(\"h\")   # activity of label layer\n",
    "        for l in self.down:\n",
    "            input = l(input)\n",
    "        return self.down[-1].get(\"u\")   # return prediction of data layer\n",
    "\n",
    "    def model_up(self, y):\n",
    "        # input is not used be needed for parallelism using vmap\n",
    "        input = self.up[0].get(\"h\")     # activity of data layer\n",
    "        for l in self.up:\n",
    "            input = l(input)\n",
    "        return self.up[-1].get(\"u\")     # return prediction of label layer\n",
    "\n",
    "    def model_down_fp(self, input):\n",
    "        # equivalent of model_down but without the Vode layers\n",
    "        for l in self.down[1:-1]:\n",
    "            if isinstance(l, pxc.Vode):\n",
    "                pass\n",
    "            else:\n",
    "                input = l(input)\n",
    "        return input\n",
    "\n",
    "    def model_up_fp(self, input):\n",
    "        # equivalent of model_up but without the Vode layers\n",
    "        for l in self.up[1:-1]:\n",
    "            if isinstance(l, pxc.Vode):\n",
    "                pass\n",
    "            else:\n",
    "                input = l(input)\n",
    "        return input\n",
    "    \n",
    "    def __call__(self, x, y, is_up_initialisation: bool = True):\n",
    "        # this function is used to initialise the activity of neurons to the label x or data y\n",
    "        # and then run the initialisation sweeo either down or up model depending on is_up_initialisation\n",
    "        if x is not None:\n",
    "            self.down[0].set(\"h\", x)\n",
    "        if y is not None:\n",
    "            self.up[0].set(\"h\", y)\n",
    "\n",
    "        if is_up_initialisation:\n",
    "            output = self.model_up(y)\n",
    "        else:\n",
    "            output = self.model_down(x)\n",
    "\n",
    "        if x is not None:  # need reset if initialisation has overwritten the values\n",
    "            self.down[0].set(\"h\", x)\n",
    "        if y is not None:\n",
    "            self.up[0].set(\"h\", y)\n",
    "        return output\n",
    "    \n",
    "model = Model(\n",
    "    hidden_dim=256,\n",
    "    nm_layers=4,\n",
    "    activation=\"gelu\",\n",
    "    alpha_up=1.0,\n",
    "    alpha_down=0.0001,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Define the optimisers for the bPC model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pcax.utils as pxu\n",
    "import pcax.functional as pxf\n",
    "\n",
    "lr_activity = 0.01\n",
    "lr_param = 0.0004\n",
    "\n",
    "@pxf.vmap(pxu.Mask(pxc.VodeParam | pxc.VodeParam.Cache, (None, 0)), in_axes=(0, 0), out_axes=0)\n",
    "def initialisation(x, y, *, model: Model, is_up_initialisation: bool = True):\n",
    "    return model(x, y, is_up_initialisation=is_up_initialisation)\n",
    "\n",
    "\n",
    "optim_h = pxu.Optim(optax.sgd(lr_activity))\n",
    "with pxu.step(model, pxc.STATUS.INIT, clear_params=pxc.VodeParam.Cache):\n",
    "        initialisation(jax.numpy.zeros((batch_size, 10)), jax.numpy.zeros((batch_size, 784)),\n",
    "            model=model,\n",
    "            is_up_initialisation=True,\n",
    "        )\n",
    "        optim_w = pxu.Optim(optax.adamw(lr_param), pxu.Mask(pxnn.LayerParam)(model),)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Define bPC energy and inference"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "@pxf.vmap(\n",
    "    pxu.Mask(pxc.VodeParam | pxc.VodeParam.Cache, (None, 0)),\n",
    "    in_axes=(0, 0),\n",
    "    out_axes=(None, 0, 0),\n",
    "    axis_name=\"batch\",\n",
    ")\n",
    "def energy(x, y, *, model: Model):\n",
    "    # scaled energy of up predictions\n",
    "    with pxu.step(model, clear_params=pxc.VodeParam.Cache):\n",
    "        x_up = model.model_up(y)\n",
    "        energy_up = model.energy() * model.alpha_up\n",
    "    # scaled energy of down predictions\n",
    "    with pxu.step(model, clear_params=pxc.VodeParam.Cache):\n",
    "        y_down = model.model_down(x)\n",
    "        energy_down = model.energy() * model.alpha_down\n",
    "    return jax.lax.pmean(energy_up + energy_down, \"batch\"), y_down, x_up\n",
    "\n",
    "\n",
    "@pxf.jit(static_argnums=(0, 3, 4))\n",
    "def infer_on_batch(\n",
    "    T: int,\n",
    "    x: jax.Array,\n",
    "    y: jax.Array,\n",
    "    is_up_initialisation: bool,\n",
    "    mode: int,\n",
    "    *,\n",
    "    model: Model,\n",
    "    optim_h: pxu.Optim,\n",
    "):\n",
    "    model.train()\n",
    "\n",
    "    # set the mode of inference - what neurons are frozen/what data is provided\n",
    "    mode_mapping = {\n",
    "        0: \"constrained\",\n",
    "        1: \"label-only\",\n",
    "        2: \"data-only\",\n",
    "        3: \"unconstrained\",\n",
    "    }\n",
    "    mode = mode_mapping.get(mode, mode)\n",
    "\n",
    "    if mode == \"constrained\":\n",
    "        model.vodes[0].h.frozen = True\n",
    "        model.vodes[-1].h.frozen = True\n",
    "    elif mode == \"label-only\":\n",
    "        model.vodes[0].h.frozen = True\n",
    "        model.vodes[-1].h.frozen = False\n",
    "        if not is_up_initialisation:\n",
    "            y = None\n",
    "        else:\n",
    "            assert y is not None\n",
    "        # is_up_initialisation = False\n",
    "    elif mode == \"data-only\":\n",
    "        model.vodes[0].h.frozen = False\n",
    "        model.vodes[-1].h.frozen = True\n",
    "        if is_up_initialisation:\n",
    "            x = None\n",
    "        else:\n",
    "            assert x is not None\n",
    "        # is_up_initialisation = True\n",
    "    elif mode == \"unconstrained\":\n",
    "        model.vodes[0].h.frozen = False\n",
    "        model.vodes[-1].h.frozen = False\n",
    "        x = None\n",
    "        y = None\n",
    "\n",
    "    # Init neural activity\n",
    "    with pxu.step(model, pxc.STATUS.INIT, clear_params=pxc.VodeParam.Cache):\n",
    "        initialisation(x, y, model=model, is_up_initialisation=is_up_initialisation)\n",
    "\n",
    "    # Initialise the activity optimiser\n",
    "    optim_h.init(pxu.Mask(pxu.m(pxc.VodeParam).has_not(frozen=True))(model))\n",
    "\n",
    "    # Inference steps\n",
    "    for _ in range(T):\n",
    "        # run the inference step\n",
    "        # this will update the model parameters and the activity of neurons\n",
    "        # depending on the mode of inference\n",
    "        # and return the energy, predictions and labels\n",
    "        with pxu.step(model, clear_params=pxc.VodeParam.Cache):\n",
    "            (e, pred), g = pxf.value_and_grad(\n",
    "                pxu.Mask(pxu.m(pxc.VodeParam).has_not(frozen=True), [False, True]),\n",
    "                has_aux=True,\n",
    "            )(energy)(x, y, model=model)\n",
    "        optim_h.step(model, g[\"model\"], True)  #\n",
    "\n",
    "    # clear the activity optimiser\n",
    "    optim_h.clear()\n",
    "\n",
    "    # reset default frozen states\n",
    "    model.vodes[0].h.frozen = True\n",
    "    model.vodes[-1].h.frozen = True\n",
    "    return model.vodes[0].get(\"h\"), model.vodes[-1].get(\"h\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Training loop for bPC (supervised)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "@pxf.jit(static_argnums=(0, 3))\n",
    "def train_on_batch(\n",
    "    T: int,\n",
    "    x: jax.Array,\n",
    "    y: jax.Array,\n",
    "    is_up_initialisation: bool,\n",
    "    *,\n",
    "    model: Model,\n",
    "    optim_w: pxu.Optim,\n",
    "    optim_h: pxu.Optim,\n",
    "):\n",
    "\n",
    "    model.train()\n",
    "\n",
    "    infer_on_batch(T, x, y, is_up_initialisation, 0, model=model, optim_h=optim_h)\n",
    "\n",
    "    # Learning step\n",
    "    with pxu.step(model, clear_params=pxc.VodeParam.Cache):\n",
    "        (e, y_), g = pxf.value_and_grad(\n",
    "            pxu.Mask(pxnn.LayerParam, [False, True]), has_aux=True\n",
    "        )(energy)(x, y, model=model)\n",
    "    optim_w.step(model, g[\"model\"])\n",
    "    return e\n",
    "\n",
    "def train(\n",
    "    dl,\n",
    "    T,\n",
    "    is_up_initialisation,\n",
    "    *,\n",
    "    model: Model,\n",
    "    optim_w: pxu.Optim,\n",
    "    optim_h: pxu.Optim,\n",
    "):\n",
    "    for x, y in dl:\n",
    "        e = train_on_batch(\n",
    "            T,\n",
    "            x,\n",
    "            y,\n",
    "            is_up_initialisation,\n",
    "            model=model,\n",
    "            optim_w=optim_w,\n",
    "            optim_h=optim_h,\n",
    "        )\n",
    "    return e"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/26 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epochs 25/25: Energy 0.014: 100%|██████████| 26/26 [00:25<00:00,  1.01it/s]\n"
     ]
    }
   ],
   "source": [
    "from tqdm import tqdm\n",
    "import random\n",
    "\n",
    "nm_epochs = 25\n",
    "T = 8\n",
    "is_up_initialisation = True  # whether to initialise the activity of neurons with bottom-up or top-down sweep\n",
    "\n",
    "epoch_dl = tqdm(np.arange(-1, nm_epochs))\n",
    "for e in epoch_dl:\n",
    "    random.shuffle(train_dl)\n",
    "    energy_batch = train(\n",
    "        train_dl,\n",
    "        T=T,\n",
    "        is_up_initialisation=is_up_initialisation,\n",
    "        model=model,\n",
    "        optim_w=optim_w,\n",
    "        optim_h=optim_h,\n",
    "    )\n",
    "\n",
    "    epoch_dl.set_description_str(\n",
    "        f\"Epochs {e+1}/{nm_epochs}: Energy {energy_batch:.3f}\"\n",
    "    )\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Evaluate the model\n",
    "### classification accuracy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: 0.982\n"
     ]
    }
   ],
   "source": [
    "# forward pass from data to labels skipping the Vode layers == bottom-up sweep\n",
    "@pxf.vmap(pxu.Mask(pxc.VodeParam | pxc.VodeParam.Cache, (None, 0)), in_axes=(0,), out_axes=0)\n",
    "def fp_up(y, *, model: Model):\n",
    "    return model.model_up_fp(y)\n",
    "\n",
    "# assess classification accuracy\n",
    "correct_count = 0\n",
    "total_count = 0\n",
    "for x, y in val_dl:\n",
    "    pred_up = fp_up(y, model=model)\n",
    "    correct_count += np.sum(np.argmax(pred_up, axis=1) == np.argmax(x, axis=1))\n",
    "    total_count += len(x)\n",
    "\n",
    "accuracy = correct_count / total_count\n",
    "print(f\"Accuracy: {accuracy:.3f}\")\n",
    "\n",
    "# this can also be evaluated using iterative inference - see main code"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Generate images from labels"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 10 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# forward pass from labels to data skipping the Vode layers == top-down sweep\n",
    "@pxf.vmap(pxu.Mask(pxc.VodeParam | pxc.VodeParam.Cache, (None, 0)), in_axes=(0,), out_axes=0)\n",
    "def fp_down(x, *, model: Model):\n",
    "    return model.model_down_fp(x)\n",
    "\n",
    "\n",
    "# one hot encoding of each class\n",
    "X_test = jax.nn.one_hot(np.arange(batch_size) % 10, 10)\n",
    "\n",
    "pred_down = fp_down(X_test, model=model)\n",
    "\n",
    "pred_down = (pred_down.reshape(-1, 28, 28))\n",
    "\n",
    "fig, axs = plt.subplots(1, 10)\n",
    "img = pred_down / 2 + 0.5\n",
    "if len(img.shape) == 4:\n",
    "    img = np.transpose(img, (0, 2, 3, 1))\n",
    "for i in range(10):\n",
    "    axs[i].imshow(img[i], cmap=\"gray\", vmin=0, vmax=1)\n",
    "    axs[i].axis(\"off\")\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "pcax_fixx",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
