{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "tags": []
   },
   "source": [
    "# Spike Accumulation Forwarding for Effective Training of Spiking Neural Networks\n",
    "\n",
    "## 1. Setup\n",
    "---\n",
    "\n",
    "### 1.1. Import"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "import math\n",
    "import os\n",
    "import random\n",
    "from typing import Any\n",
    "\n",
    "import matplotlib.pyplot as plt  # matplotlib==3.6.1\n",
    "import numpy as np  # numpy==1.23.3\n",
    "import pandas as pd  # pandas==1.5.0\n",
    "import torch  # torch==1.10.0+cu113\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torchvision  # torchvision==0.11.1+cu113\n",
    "import torchvision.transforms as transforms\n",
    "from torch import Tensor\n",
    "\n",
    "cwd = \"./\"\n",
    "# data_dir\n",
    "data_dir = os.path.join(cwd, \"dataset\")\n",
    "# dataset = cifar10\n",
    "out_dir = os.path.join(cwd, \"logs\")\n",
    "# number of data loading workers\n",
    "num_workers = 4\n",
    "# device\n",
    "device = torch.device(\"cuda\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### 1.2. Hyperparameters\n",
    "\n",
    " - `batch_size` : Batch Size\n",
    " - `epochs` : Number of Epochs\n",
    " - `lr` : Learning Rate\n",
    " - `momentum` : momentum for SGD\n",
    " - `T_max` : T_max for CosineAnnealingLR\n",
    " - `weight_decay` : Weight Decay\n",
    " - `loss_alpha` : α * MSE + (1 - α) * CE\n",
    " - `t_step` : simulating time-steps\n",
    " - `lif_lambda` : λ = 1 - (1 / τ)\n",
    " - `cfg` : Model Architecture(VGG)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "# batch size\n",
    "batch_size = 128\n",
    "# number of total epochs to run\n",
    "epochs = 300\n",
    "# learning rate\n",
    "lr = 0.1\n",
    "# momentum for SGD\n",
    "momentum = 0.9\n",
    "# lr_scheduler = CosALR\n",
    "# T_max for CosineAnnealingLR\n",
    "T_max = 300\n",
    "# unused\n",
    "weight_decay = 0.0\n",
    "# loss_alpha\n",
    "loss_alpha = 0.05\n",
    "# simulating time-steps\n",
    "t_step = 6\n",
    "# λ = 1 - (1 / τ)\n",
    "# Equivalent to IF when λ is 1.0\n",
    "lif_lambda = 0.5\n",
    "# Model Architecture(VGG11) -> 64C3-128C3-AP2-256C3-256C3-AP2-512C3-512C3-AP2-512C3-512C3-GAP-FC\n",
    "cfg = [64, 128, \"A\", 256, 256, \"A\", 512, 512, \"A\", 512, 512]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### 1.3. Dataset\n",
    "#### 1.3.1. Data Augmentation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class Cutout(object):\n",
    "    def __init__(self, p=0.5, scale=(0.02, 0.33), ratio=(0.3, 3.3)):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            p: probability that the cutout operation will be performed.\n",
    "            scale: range of proportion of cutout area against input.\n",
    "            ratio: range of aspect ratio of cutout area.\n",
    "        \"\"\"\n",
    "\n",
    "        self.p = p\n",
    "        self.scale = scale\n",
    "        self.ratio = ratio\n",
    "\n",
    "    def __call__(self, x):\n",
    "        \"\"\"perform a cutout transform on the image.\n",
    "\n",
    "        Args:\n",
    "            x: input image\n",
    "        Returns:\n",
    "            Tensor: transformd image\n",
    "        \"\"\"\n",
    "        if random.random() < self.p:\n",
    "            _, img_h, img_w = x.shape\n",
    "            area = img_h * img_w\n",
    "            cutout_area = area * random.uniform(*self.scale)\n",
    "            aspect_ratio = random.uniform(*self.ratio)\n",
    "            h = int(round(math.sqrt(cutout_area * aspect_ratio)))\n",
    "            w = int(round(math.sqrt(cutout_area / aspect_ratio)))\n",
    "            # top left\n",
    "            i = random.randint(0, img_h - h)\n",
    "            j = random.randint(0, img_w - w)\n",
    "            return transforms.functional.erase(x, i, j, h, w, random.random())\n",
    "        return x\n",
    "\n",
    "\n",
    "transform_train = transforms.Compose(\n",
    "    [\n",
    "        transforms.RandomCrop(32, padding=4),\n",
    "        transforms.RandomHorizontalFlip(),\n",
    "        transforms.ToTensor(),\n",
    "        Cutout(scale=(0.02, 0.4), ratio=(0.4, 2.5)),\n",
    "        transforms.Normalize((0.4914, 0.4822, 0.4465), (0.2023, 0.1994, 0.2010)),\n",
    "    ]\n",
    ")\n",
    "\n",
    "\n",
    "transform_test = transforms.Compose(\n",
    "    [\n",
    "        transforms.ToTensor(),\n",
    "        transforms.Normalize((0.4914, 0.4822, 0.4465), (0.2023, 0.1994, 0.2010)),\n",
    "    ]\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 1.3.2. CIFAR10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "num_classes = 10\n",
    "\n",
    "trainset = torchvision.datasets.CIFAR10(\n",
    "    root=data_dir, train=True, download=True, transform=transform_train\n",
    ")\n",
    "train_loader = torch.utils.data.DataLoader(\n",
    "    trainset, batch_size=batch_size, shuffle=True, num_workers=num_workers\n",
    ")\n",
    "\n",
    "testset = torchvision.datasets.CIFAR10(\n",
    "    root=data_dir, train=False, download=False, transform=transform_test\n",
    ")\n",
    "test_loader = torch.utils.data.DataLoader(\n",
    "    testset, batch_size=batch_size, shuffle=False, num_workers=num_workers\n",
    ")\n",
    "\n",
    "x, _ = train_loader.__iter__().next()\n",
    "mean = np.array([0.4914, 0.4822, 0.4465]).reshape(1, 1, -1)\n",
    "std = np.array([0.2023, 0.1994, 0.2010]).reshape(1, 1, -1)\n",
    "plt.figure(figsize=(10, 10))\n",
    "for i, xi in enumerate(x[:25]):\n",
    "    plt.subplot(5, 5, i + 1)\n",
    "    img = np.array(xi).transpose(1, 2, 0)\n",
    "    img = np.clip(img * std + mean, 0, 1)\n",
    "    plt.imshow(img)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 2. Model Architecture\n",
    "\n",
    "In this section, we will define the model architecture in the following four steps:\n",
    "\n",
    " - **Neuron** : Define Online-LIF(OLIF) and Spike Accumulation Forwarding(SAF)\n",
    " - **Custom Module** : Define a custom module\n",
    " - **VGG** : Define the model structure of VGG\n",
    " - **Feadback-VGG** : Add Functionality of feedback connection\n",
    "\n",
    "### 2.1. Neuron\n",
    "\n",
    "For simplicity, each neuron is implemented with $V_{th} = 1.0$. In addition, conversion to a Spike signal is done using the Heaviside function, which can be defined as follows:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def heaviside(x):\n",
    "    \"\"\"Heaviside\n",
    "\n",
    "    Args:\n",
    "        x: input\n",
    "\n",
    "    Returns:\n",
    "        Tensor: Spike\n",
    "    \"\"\"\n",
    "\n",
    "    return (x >= 0).to(x)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.1.2. Online Leaky-Integrate-and-Fire\n",
    "\n",
    "Since OTTT requires spikes $s[t]$ and spike accumulation $\\hat{a}[t]$, we define the OLIF that propagates these two.　OLIF can be defined as follows:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class OLIF(nn.Module):\n",
    "    \"\"\"Online Leaky-Integrate-and-Fire\"\"\"\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        l: float = 0.5,\n",
    "        alpha: float = 4.0,\n",
    "    ):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "            alpha: grad tuning param\n",
    "        \"\"\"\n",
    "\n",
    "        super(OLIF, self).__init__()\n",
    "        self.l = l\n",
    "        self.alpha = torch.tensor(alpha)\n",
    "        self.func = self.Function.apply\n",
    "        self.reset()\n",
    "\n",
    "    def forward(self, input: Tensor) -> Tensor:\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            input: input(intensity of current)\n",
    "\n",
    "        Returns:\n",
    "            Tensor: [spike, accumulated spike]\n",
    "        \"\"\"\n",
    "\n",
    "        # IF(I_t, u_t-1)\n",
    "        self.u = self.l * self.u\n",
    "        st = self.func(input, self.u, self.alpha)\n",
    "        at = self.l * self.a + st\n",
    "        self.a = at.clone().detach()\n",
    "        # u_t-1 -> u_t\n",
    "        self.u = (self.u + input - st).clone().detach()\n",
    "        return torch.cat((st, at))\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset u and a\"\"\"\n",
    "\n",
    "        # membrane potential\n",
    "        self.u = 0\n",
    "        # accumulated spike\n",
    "        self.a = 0\n",
    "\n",
    "    class Function(torch.autograd.Function):\n",
    "        \"\"\"Autograd（Forward/Backward）\"\"\"\n",
    "\n",
    "        @staticmethod\n",
    "        def forward(ctx, i_t: Tensor, u: Tensor, alpha: Tensor) -> Tensor:\n",
    "            \"\"\"Forward process\n",
    "\n",
    "            Args:\n",
    "                i_t: intensity of current\n",
    "                u: membrane potential\n",
    "                alpha: grad tuning param\n",
    "\n",
    "            Returns:\n",
    "                Tensor: spike\n",
    "            \"\"\"\n",
    "            ut = u + i_t\n",
    "            x = ut - 1.0\n",
    "            st = heaviside(x)\n",
    "            if i_t.requires_grad:\n",
    "                ctx.save_for_backward(x)\n",
    "                ctx.alpha = alpha\n",
    "            return st\n",
    "\n",
    "        @staticmethod\n",
    "        def backward(ctx: Any, grad_output: Tensor) -> Tensor:\n",
    "            \"\"\"Backward process\n",
    "\n",
    "            Args:\n",
    "                ctx: saved parameters\n",
    "                grad_output: grad from output\n",
    "\n",
    "            Returns:\n",
    "                Tensor: grad to input\n",
    "            \"\"\"\n",
    "\n",
    "            grad_x = None\n",
    "            if ctx.needs_input_grad[0]:\n",
    "                sgax = (ctx.saved_tensors[0] * ctx.alpha).sigmoid_()\n",
    "                grad_x = grad_output * (1.0 - sgax) * sgax * ctx.alpha\n",
    "            return grad_x, None, None"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.1.2. Spike Accumulation Forwarding\n",
    "\n",
    "Our proposed SAF can be defined as follows:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class SAF(nn.Module):\n",
    "    \"\"\"Spike Accumulation Forwarding\"\"\"\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        l: float = 0.5,\n",
    "        alpha: float = 4.0,\n",
    "        spike: bool = False,\n",
    "    ):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "            alpha: grad tuning param\n",
    "            spike: True -> spike and accumulated spike, False -> accumulated spike\n",
    "        \"\"\"\n",
    "\n",
    "        super(SAF, self).__init__()\n",
    "        self.l = l\n",
    "        self.alpha = torch.tensor(alpha)\n",
    "        self.func = self.Function.apply\n",
    "        self.spike = spike\n",
    "        self.reset()\n",
    "\n",
    "    def forward(self, V: Tensor) -> Tensor:\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            V: accumulated potential\n",
    "\n",
    "        Returns:\n",
    "            Tensor: (spike and) accumulated spike\n",
    "        \"\"\"\n",
    "\n",
    "        # accumulated spike\n",
    "        la = self.l * self.a\n",
    "        # spike\n",
    "        st = self.func(V, la, self.alpha)\n",
    "        # a_t-1 -> a_t\n",
    "        at = la + st\n",
    "        self.a = at.clone().detach()\n",
    "        if self.spike:\n",
    "            out = torch.cat((st, at))\n",
    "        else:\n",
    "            out = at\n",
    "        return out\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset u and a\"\"\"\n",
    "        # accumulated spike\n",
    "        self.a = 0\n",
    "\n",
    "    class Function(torch.autograd.Function):\n",
    "        \"\"\"Autograd（Forward/Backward）\"\"\"\n",
    "\n",
    "        @staticmethod\n",
    "        def forward(ctx, V: Tensor, la: Tensor, alpha: Tensor) -> Tensor:\n",
    "            \"\"\"Forward process\n",
    "\n",
    "            Args:\n",
    "                V: accumulated potential\n",
    "                la: accumulated spike\n",
    "                alpha: grad tuning param\n",
    "\n",
    "            Returns:\n",
    "                Tensor: spike\n",
    "            \"\"\"\n",
    "\n",
    "            x = V - la - 1.0\n",
    "            st = heaviside(x)\n",
    "            if V.requires_grad:\n",
    "                ctx.save_for_backward(x)\n",
    "                ctx.alpha = alpha\n",
    "            return st\n",
    "\n",
    "        @staticmethod\n",
    "        def backward(ctx: Any, grad_output: Tensor) -> Tensor:\n",
    "            \"\"\"Backward process\n",
    "\n",
    "            Args:\n",
    "                ctx: saved parameters\n",
    "                grad_output: grad from output\n",
    "\n",
    "            Returns:\n",
    "                Tensor: grad to input\n",
    "            \"\"\"\n",
    "\n",
    "            grad_x = None\n",
    "            if ctx.needs_input_grad[0]:\n",
    "                sgax = (ctx.saved_tensors[0] * ctx.alpha).sigmoid_()\n",
    "                grad_x = grad_output * (1.0 - sgax) * sgax * ctx.alpha\n",
    "            return grad_x, None, None, None"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### 2.2. Custom Module\n",
    "\n",
    " - **OutputSwap** : Process while swapping spike and rate\n",
    " - **LSUM** : Compute a weighted sum considering λ\n",
    " - **Leaky-Bias** : Since bias can be considered as input of each time, it is added using LSUM\n",
    " - **sWSConv2d** : Unlike BN, which standardizes the dataset, sWS standardizes weights\n",
    " - **Combine CE and MSE** : Define combine cross-entropy (CE) loss and mean-square-error (MSE) loss\n",
    "\n",
    "#### 2.2.1. OutputSwap"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class OutputSwap(nn.Module):\n",
    "    \"\"\"Process while swapping spike and rate\"\"\"\n",
    "\n",
    "    def __init__(self, f):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            f: function (ex. Conv2D, Linear, etc)\n",
    "        \"\"\"\n",
    "\n",
    "        super(OutputSwap, self).__init__()\n",
    "        self.f = f\n",
    "        self.func = self.Function.apply\n",
    "\n",
    "    def forward(self, x):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: [spike, rate]\n",
    "\n",
    "        Returns:\n",
    "            Tensor: Forward -> op(spike), Backward -> grad of rate\n",
    "        \"\"\"\n",
    "\n",
    "        spike, rate = torch.chunk(x, 2, dim=0)\n",
    "        if self.training:\n",
    "            with torch.no_grad():\n",
    "                y1 = self.f(spike).detach()\n",
    "            y2 = self.f(rate)\n",
    "            y = self.func(y2, y1)\n",
    "        else:\n",
    "            y = self.f(spike)\n",
    "        return y\n",
    "\n",
    "    class Function(torch.autograd.Function):\n",
    "        \"\"\"Autograd（Forward/Backward）\"\"\"\n",
    "\n",
    "        @staticmethod\n",
    "        def forward(ctx, x1: Tensor, x2: Tensor) -> Tensor:\n",
    "            \"\"\"Forward process\n",
    "\n",
    "            Args:\n",
    "                x1: used grad\n",
    "                x2: used value\n",
    "\n",
    "            Returns:\n",
    "                Tensor: x2\n",
    "            \"\"\"\n",
    "\n",
    "            return x2\n",
    "\n",
    "        @staticmethod\n",
    "        def backward(ctx: Any, grad_output: Tensor) -> Tensor:\n",
    "            \"\"\"Backward process\n",
    "\n",
    "            Args:\n",
    "                ctx: saved parameters\n",
    "                grad_output: grad from x1\n",
    "\n",
    "            Returns:\n",
    "                Tensor: grad_output\n",
    "            \"\"\"\n",
    "\n",
    "            return grad_output, None"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.2.2. Leaky-SUM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class LSUM(nn.Module):\n",
    "    \"\"\"Leaky-SUM\"\"\"\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        l: float = 0.5,\n",
    "    ):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(LSUM, self).__init__()\n",
    "        # λ = 1 - (1 / τ)\n",
    "        self.l = l\n",
    "        self.lx = 0\n",
    "        self.reset()\n",
    "\n",
    "    def forward(self, x: Tensor) -> Tensor:\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input data\n",
    "\n",
    "        Returns:\n",
    "            Tensor: accumulated x\n",
    "        \"\"\"\n",
    "\n",
    "        y = self.l * self.lx + x\n",
    "        self.lx = y.clone().detach()\n",
    "        return y\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset accumulated x\"\"\"\n",
    "\n",
    "        self.lx = 0"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.2.3. Leaky-Bias"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class LBias2d(nn.Module):\n",
    "    \"\"\"Leaky-Bias2d\"\"\"\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        channels: int,\n",
    "        l: float = 0.5,\n",
    "    ):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            channels: input channels\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "        super(LBias2d, self).__init__()\n",
    "        self.l = l\n",
    "        self.bias = nn.Parameter(torch.zeros(channels))\n",
    "        self.b = 0\n",
    "\n",
    "    def forward(self, x):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "\n",
    "        Returns:\n",
    "            Tensor: accumulated bias added to x\n",
    "        \"\"\"\n",
    "\n",
    "        b = self.l * self.b + self.bias\n",
    "        self.b = b.clone().detach()\n",
    "        return x + b.reshape(1, -1, 1, 1)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset accumulated bias\"\"\"\n",
    "\n",
    "        self.b = 0\n",
    "\n",
    "\n",
    "class LBias(nn.Module):\n",
    "    \"\"\"Leaky-Bias\"\"\"\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        channels: int,\n",
    "        l: float = 0.5,\n",
    "    ):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            channels: input channels\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "        super(LBias, self).__init__()\n",
    "        self.l = l\n",
    "        self.bias = nn.Parameter(torch.zeros(channels))\n",
    "        self.b = 0\n",
    "\n",
    "    def forward(self, x):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "\n",
    "        Returns:\n",
    "            Tensor: accumulated bias added to x\n",
    "        \"\"\"\n",
    "\n",
    "        b = self.l * self.b + self.bias\n",
    "        self.b = b.clone().detach()\n",
    "        return x + b.reshape(1, -1)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset accumulated bias\"\"\"\n",
    "\n",
    "        self.b = 0"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.2.4. sWSConv2d"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class ScaledWeightStandardization(nn.Module):\n",
    "    \"\"\"Scaled weight standardization\"\"\"\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        n,\n",
    "        dim,\n",
    "        eps=1e-4,\n",
    "    ):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            n: number of inputs\n",
    "            dim: dimension to normalize\n",
    "            eps: a value added to the denominator for numerical stability\n",
    "        \"\"\"\n",
    "\n",
    "        super(ScaledWeightStandardization, self).__init__()\n",
    "        self.n = n\n",
    "        self.dim = dim\n",
    "        self.eps = eps\n",
    "\n",
    "    def forward(self, x):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: weight\n",
    "\n",
    "        Returns:\n",
    "            Tensor: normalized weight\n",
    "        \"\"\"\n",
    "\n",
    "        mean = torch.mean(x, dim=self.dim, keepdims=True)\n",
    "        var = torch.var(x, dim=self.dim, keepdims=True)\n",
    "        return (x - mean) / torch.sqrt(var * self.n + self.eps)\n",
    "\n",
    "\n",
    "class sWSConv2d(nn.Conv2d):\n",
    "    \"\"\"Convolution2D with scaled weight standardization\"\"\"\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        in_channels,\n",
    "        out_channels,\n",
    "        kernel_size,\n",
    "        stride=1,\n",
    "        padding=0,\n",
    "        dilation=1,\n",
    "        groups=1,\n",
    "        bias=True,\n",
    "        eps=1e-4,\n",
    "    ):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            in_channels: input channels\n",
    "            out_channels: output channels\n",
    "            kernel_size: kernel size\n",
    "            stride: stride\n",
    "            padding: padding\n",
    "            dilation: spacing between kernel elements\n",
    "            groups: num of blocked connections from in to out\n",
    "            bias: if True, adds a learnable bias\n",
    "            eps: a value added to the denominator for numerical stability\n",
    "        \"\"\"\n",
    "\n",
    "        super(sWSConv2d, self).__init__(\n",
    "            in_channels,\n",
    "            out_channels,\n",
    "            kernel_size,\n",
    "            stride,\n",
    "            padding,\n",
    "            dilation,\n",
    "            groups,\n",
    "            bias,\n",
    "        )\n",
    "        # scaled weight standardization\n",
    "        n = np.prod(self.weight.shape[1:])\n",
    "        self.sws = ScaledWeightStandardization(n, (1, 2, 3))\n",
    "        # learnable scaling factor for the weights\n",
    "        self.gamma = nn.Parameter(torch.ones(self.out_channels, 1, 1, 1))\n",
    "\n",
    "    def forward(self, x):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "\n",
    "        Returns:\n",
    "            Tensor: conv2d(x)\n",
    "        \"\"\"\n",
    "\n",
    "        weight = self.gamma * self.sws(self.weight)\n",
    "        return F.conv2d(\n",
    "            x,\n",
    "            weight,\n",
    "            self.bias,\n",
    "            self.stride,\n",
    "            self.padding,\n",
    "            self.dilation,\n",
    "            self.groups,\n",
    "        )\n",
    "\n",
    "\n",
    "class Gamma(nn.Module):\n",
    "    \"\"\"γ = 1/σH ≈ 2.74\"\"\"\n",
    "\n",
    "    def __init__(self, gamma=2.74):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            gamma: constant\n",
    "        \"\"\"\n",
    "\n",
    "        super(Gamma, self).__init__()\n",
    "        self.gamma = gamma\n",
    "\n",
    "    def forward(self, x):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "\n",
    "        Returns:\n",
    "            Tensor: gamma * x\n",
    "        \"\"\"\n",
    "\n",
    "        return self.gamma * x"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.2.5. Combine CE and MSE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class CombineCEandMSE(nn.Module):\n",
    "    \"\"\"combine cross-entropy (CE) loss and mean-square-error (MSE) loss.\"\"\"\n",
    "\n",
    "    def __init__(self, alpha, num_classes):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            alpha: α * MSE + (1 - α) * CE\n",
    "            num_classes: number of classes\n",
    "        \"\"\"\n",
    "        super(CombineCEandMSE, self).__init__()\n",
    "        self.mse = nn.MSELoss(reduction=\"mean\")\n",
    "        self.a = alpha\n",
    "        self.n = num_classes\n",
    "\n",
    "    def forward(self, y, t):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            y: model output\n",
    "            t: target label\n",
    "\n",
    "        Returns:\n",
    "            Tensor: α * MSE + (1 - α) * CE\n",
    "        \"\"\"\n",
    "\n",
    "        one_hot = F.one_hot(t, self.n).float()\n",
    "        loss1 = self.mse(y, one_hot)\n",
    "        loss2 = F.cross_entropy(y, t)\n",
    "        return self.a * loss1 + (1 - self.a) * loss2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### 2.3. VGG\n",
    "\n",
    "We leverage the VGG network architecture (64C3-128C3-AP2-256C3-256C3-AP2-512C3-512C3-AP2-512C3-512C3-GAP-FC) for experiments on CIFAR-10. We then define three models (**OTTT-based VGG**, **SAF-based VGG** and **Output Leaky-FR model**) with equivalent forward propagation.\n",
    "\n",
    "#### 2.3.1. OTTT-based VGG\n",
    "\n",
    "Combining this model with **Training_E** and **Training_A** defined in 3.1 results in $OTTT_O$ and $OTTT_A$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class OTTTVGG(nn.Module):\n",
    "    \"\"\"OTTT-based VGG model\"\"\"\n",
    "\n",
    "    def __init__(self, l):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(OTTTVGG, self).__init__()\n",
    "        layers = []\n",
    "        in_ch = 3\n",
    "        for i, x in enumerate(cfg):\n",
    "            if x == \"A\":\n",
    "                layers += [nn.AvgPool2d(kernel_size=2, stride=2)]\n",
    "            elif x == \"M\":\n",
    "                layers += [nn.MaxPool2d(kernel_size=2, stride=2)]\n",
    "            else:\n",
    "                conv = sWSConv2d(in_ch, x, 3, 1, 1)\n",
    "                if i != 0:\n",
    "                    conv = OutputSwap(conv)\n",
    "                layers += [conv, OLIF(l=l), Gamma()]\n",
    "                in_ch = x\n",
    "        layers += [\n",
    "            nn.AdaptiveAvgPool2d((1, 1)),\n",
    "            nn.Flatten(),\n",
    "            OutputSwap(nn.Linear(cfg[-1], 10)),\n",
    "        ]\n",
    "        self.features = nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x, init):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "            init: if True, reset all u and a in OLIF\n",
    "\n",
    "        Returns:\n",
    "            Tensor: output\n",
    "        \"\"\"\n",
    "\n",
    "        if init:\n",
    "            self.reset()\n",
    "        return self.features(x)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset all u, b and a in Leaky-XXX Layer\"\"\"\n",
    "\n",
    "        for f in self.modules():\n",
    "            if (\n",
    "                isinstance(f, LSUM)\n",
    "                or isinstance(f, OLIF)\n",
    "                or isinstance(f, SAF)\n",
    "                or isinstance(f, LBias2d)\n",
    "                or isinstance(f, LBias)\n",
    "            ):\n",
    "                f.reset()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.3.2. SAF-based VGG\n",
    "\n",
    "Combining this model with **Training_E** defined in 3.1 results in SAF-E."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class SAFVGG(nn.Module):\n",
    "    \"\"\"SAF-based VGG model\"\"\"\n",
    "\n",
    "    def __init__(self, l):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(SAFVGG, self).__init__()\n",
    "        layers = []\n",
    "        in_ch = 3\n",
    "        for i, x in enumerate(cfg):\n",
    "            if x == \"A\":\n",
    "                layers += [nn.AvgPool2d(kernel_size=2, stride=2)]\n",
    "            elif x == \"M\":\n",
    "                layers += [nn.MaxPool2d(kernel_size=2, stride=2)]\n",
    "            else:\n",
    "                if i == 0:\n",
    "                    layers += [\n",
    "                        sWSConv2d(in_ch, x, 3, 1, 1),\n",
    "                        LSUM(l=l),\n",
    "                        SAF(l=l),\n",
    "                        Gamma(),\n",
    "                    ]\n",
    "                else:\n",
    "                    if i == (len(cfg) - 1):\n",
    "                        act = SAF(l=l, spike=True)\n",
    "                    else:\n",
    "                        act = SAF(l=l)\n",
    "                    layers += [\n",
    "                        sWSConv2d(in_ch, x, 3, 1, 1, bias=False),\n",
    "                        LBias2d(x, l=l),\n",
    "                        act,\n",
    "                        Gamma(),\n",
    "                    ]\n",
    "                in_ch = x\n",
    "        layers += [\n",
    "            nn.AdaptiveAvgPool2d((1, 1)),\n",
    "            nn.Flatten(),\n",
    "            OutputSwap(nn.Linear(cfg[-1], 10)),\n",
    "        ]\n",
    "        self.features = nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x, init):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "            init: if True, reset all u, b and a in Leaky-XXX Layer\n",
    "\n",
    "        Returns:\n",
    "            Tensor: output\n",
    "        \"\"\"\n",
    "\n",
    "        if init:\n",
    "            self.reset()\n",
    "        return self.features(x)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset all u, b and a in Leaky-XXX Layer\"\"\"\n",
    "\n",
    "        for f in self.modules():\n",
    "            if (\n",
    "                isinstance(f, LSUM)\n",
    "                or isinstance(f, OLIF)\n",
    "                or isinstance(f, SAF)\n",
    "                or isinstance(f, LBias2d)\n",
    "                or isinstance(f, LBias)\n",
    "            ):\n",
    "                f.reset()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.3.3. Output Leaky-FR\n",
    "\n",
    "Combining this model with **Training_F** defined in 3.1 results in SAF-F."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class SAFVGG_FR(nn.Module):\n",
    "    \"\"\"Our VGG Model using SAF\"\"\"\n",
    "\n",
    "    def __init__(self, l):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(SAFVGG_FR, self).__init__()\n",
    "        layers = [LSUM(l=l)]\n",
    "        in_ch = 3\n",
    "        for i, x in enumerate(cfg):\n",
    "            if x == \"A\":\n",
    "                layers += [nn.AvgPool2d(kernel_size=2, stride=2)]\n",
    "            elif x == \"M\":\n",
    "                layers += [nn.MaxPool2d(kernel_size=2, stride=2)]\n",
    "            else:\n",
    "                layers += [\n",
    "                    sWSConv2d(in_ch, x, 3, 1, 1, bias=False),\n",
    "                    LBias2d(x, l=l),\n",
    "                    SAF(l=l),\n",
    "                    Gamma(),\n",
    "                ]\n",
    "                in_ch = x\n",
    "        layers += [\n",
    "            nn.AdaptiveAvgPool2d((1, 1)),\n",
    "            nn.Flatten(),\n",
    "            nn.Linear(cfg[-1], 10, bias=False),\n",
    "            LBias(10, l=l),\n",
    "        ]\n",
    "        self.features = nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x, init):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "            init: if True, reset all u, b and a in Leaky-XXX Layer\n",
    "\n",
    "        Returns:\n",
    "            Tensor: output\n",
    "        \"\"\"\n",
    "\n",
    "        if init:\n",
    "            self.reset()\n",
    "        return self.features(x)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset all u, b and a in Leaky-XXX Layer\"\"\"\n",
    "\n",
    "        for f in self.modules():\n",
    "            if (\n",
    "                isinstance(f, LSUM)\n",
    "                or isinstance(f, OLIF)\n",
    "                or isinstance(f, SAF)\n",
    "                or isinstance(f, LBias2d)\n",
    "                or isinstance(f, LBias)\n",
    "            ):\n",
    "                f.reset()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### 2.4. Feadback-VGG\n",
    "\n",
    "Add feedback connection to **2.3.VGG**.\n",
    "\n",
    "#### 2.4.1. OTTT-based Feadback-VGG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class FOTTTVGG(nn.Module):\n",
    "    \"\"\"OTTT-based Feadback-VGG model\"\"\"\n",
    "\n",
    "    def __init__(self, l):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(FOTTTVGG, self).__init__()\n",
    "        scale_factor = 2 ** cfg.count(\"A\") * 2 ** cfg.count(\"M\")\n",
    "        self.up = nn.Upsample(scale_factor=scale_factor, mode=\"nearest\")\n",
    "        self.fb = OutputSwap(\n",
    "            nn.Conv2d(cfg[-1], cfg[0], kernel_size=3, padding=1, stride=1, bias=False)\n",
    "        )\n",
    "        in_ch = 3\n",
    "        layers = []\n",
    "        for i, x in enumerate(cfg):\n",
    "            if x == \"A\":\n",
    "                layers += [nn.AvgPool2d(kernel_size=2, stride=2)]\n",
    "            elif x == \"M\":\n",
    "                layers += [nn.MaxPool2d(kernel_size=2, stride=2)]\n",
    "            else:\n",
    "                conv = sWSConv2d(in_ch, x, 3, 1, 1)\n",
    "                if i != 0:\n",
    "                    conv = OutputSwap(conv)\n",
    "                layers += [conv, OLIF(l=l), Gamma()]\n",
    "                in_ch = x\n",
    "        layers += [\n",
    "            nn.AdaptiveAvgPool2d((1, 1)),\n",
    "            nn.Flatten(),\n",
    "            OutputSwap(nn.Linear(cfg[-1], 10)),\n",
    "        ]\n",
    "        self.input_layer = nn.Sequential(*layers[:1])\n",
    "        self.features = nn.Sequential(*layers[1:-3])\n",
    "        self.output_layer = nn.Sequential(*layers[-3:])\n",
    "\n",
    "    def forward(self, x, init):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "            init: if True, reset all u and a in OLIF\n",
    "\n",
    "        Returns:\n",
    "            Tensor: output\n",
    "        \"\"\"\n",
    "\n",
    "        if init:\n",
    "            self.reset()\n",
    "            fb = 0\n",
    "        else:\n",
    "            fb = self.fb(self.up(self.fb_features))\n",
    "        h = self.input_layer(x) + fb\n",
    "        h = self.features(h)\n",
    "        self.fb_features = h.clone().detach()\n",
    "        return self.output_layer(h)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset all u, b and a in Leaky-XXX Layer\"\"\"\n",
    "\n",
    "        for f in self.modules():\n",
    "            if (\n",
    "                isinstance(f, LSUM)\n",
    "                or isinstance(f, OLIF)\n",
    "                or isinstance(f, SAF)\n",
    "                or isinstance(f, LBias2d)\n",
    "                or isinstance(f, LBias)\n",
    "            ):\n",
    "                f.reset()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.4.2. SAF-based Feadback-VGG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class FSAFVGG(nn.Module):\n",
    "    \"\"\"SAF-based Feadback-VGG model\"\"\"\n",
    "\n",
    "    def __init__(self, l):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(FSAFVGG, self).__init__()\n",
    "        scale_factor = 2 ** cfg.count(\"A\") * 2 ** cfg.count(\"M\")\n",
    "        self.up = nn.Upsample(scale_factor=scale_factor, mode=\"nearest\")\n",
    "        self.fb = nn.Conv2d(\n",
    "            cfg[-1], cfg[0], kernel_size=3, padding=1, stride=1, bias=False\n",
    "        )\n",
    "        in_ch = 3\n",
    "        layers = []\n",
    "        for i, x in enumerate(cfg):\n",
    "            if x == \"A\":\n",
    "                layers += [nn.AvgPool2d(kernel_size=2, stride=2)]\n",
    "            elif x == \"M\":\n",
    "                layers += [nn.MaxPool2d(kernel_size=2, stride=2)]\n",
    "            else:\n",
    "                if i == 0:\n",
    "                    conv = sWSConv2d(in_ch, x, 3, 1, 1)\n",
    "                    layers += [conv, LSUM(l=l), SAF(l=l), Gamma()]\n",
    "                else:\n",
    "                    conv = sWSConv2d(in_ch, x, 3, 1, 1, bias=False)\n",
    "                    bias = LBias2d(x, l=l)\n",
    "                    if i != (len(cfg) - 1):\n",
    "                        layers += [conv, bias, SAF(l=l), Gamma()]\n",
    "                    else:\n",
    "                        layers += [conv, bias, SAF(l=l, spike=True), Gamma()]\n",
    "                in_ch = x\n",
    "        layers += [\n",
    "            nn.AdaptiveAvgPool2d((1, 1)),\n",
    "            nn.Flatten(),\n",
    "            OutputSwap(nn.Linear(cfg[-1], 10)),\n",
    "        ]\n",
    "        self.input_layer = nn.Sequential(*layers[:2])\n",
    "        self.features = nn.Sequential(*layers[2:-3])\n",
    "        self.output_layer = nn.Sequential(*layers[-3:])\n",
    "\n",
    "    def forward(self, x, init):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "            init: if True, reset all u, b and a in Leaky-XXX Layer\n",
    "\n",
    "        Returns:\n",
    "            Tensor: output\n",
    "        \"\"\"\n",
    "\n",
    "        if init:\n",
    "            self.reset()\n",
    "            fb = 0\n",
    "        else:\n",
    "            fb = self.fb(self.up(self.fb_features))\n",
    "        h = self.input_layer(x) + fb\n",
    "        h = self.features(h)\n",
    "        s, r = torch.chunk(h, 2, dim=0)\n",
    "        self.fb_features = r.clone().detach()\n",
    "        return self.output_layer(h)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset all u, b and a in Leaky-XXX Layer\"\"\"\n",
    "\n",
    "        for f in self.modules():\n",
    "            if (\n",
    "                isinstance(f, LSUM)\n",
    "                or isinstance(f, OLIF)\n",
    "                or isinstance(f, SAF)\n",
    "                or isinstance(f, LBias2d)\n",
    "                or isinstance(f, LBias)\n",
    "            ):\n",
    "                f.reset()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 2.4.3. Output Leaky-FR"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class FSAFVGG_FR(nn.Module):\n",
    "    \"\"\"Our VGG Model using SAF\"\"\"\n",
    "\n",
    "    def __init__(self, l):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(FSAFVGG_FR, self).__init__()\n",
    "        scale_factor = 2 ** cfg.count(\"A\") * 2 ** cfg.count(\"M\")\n",
    "        self.up = nn.Upsample(scale_factor=scale_factor, mode=\"nearest\")\n",
    "        self.fb = nn.Conv2d(\n",
    "            cfg[-1], cfg[0], kernel_size=3, padding=1, stride=1, bias=False\n",
    "        )\n",
    "        in_ch = 3\n",
    "        layers = [LSUM(l=l)]\n",
    "        for i, x in enumerate(cfg):\n",
    "            if x == \"A\":\n",
    "                layers += [nn.AvgPool2d(kernel_size=2, stride=2)]\n",
    "            elif x == \"M\":\n",
    "                layers += [nn.MaxPool2d(kernel_size=2, stride=2)]\n",
    "            else:\n",
    "                conv = sWSConv2d(in_ch, x, 3, 1, 1, bias=False)\n",
    "                bias = LBias2d(x, l=l)\n",
    "                layers += [conv, bias, SAF(l=l), Gamma()]\n",
    "                in_ch = x\n",
    "        layers += [\n",
    "            nn.AdaptiveAvgPool2d((1, 1)),\n",
    "            nn.Flatten(),\n",
    "            nn.Linear(cfg[-1], 10),\n",
    "            LBias(10, l=l),\n",
    "        ]\n",
    "        self.input_layer = nn.Sequential(*layers[:3])\n",
    "        self.features = nn.Sequential(*layers[3:-4])\n",
    "        self.output_layer = nn.Sequential(*layers[-4:])\n",
    "\n",
    "    def forward(self, x, init):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "            init: if True, reset all u, b and a in Leaky-XXX Layer\n",
    "\n",
    "        Returns:\n",
    "            Tensor: output\n",
    "        \"\"\"\n",
    "\n",
    "        if init:\n",
    "            self.reset()\n",
    "            fb = 0\n",
    "        else:\n",
    "            fb = self.fb(self.up(self.fb_features))\n",
    "        h = self.input_layer(x) + fb\n",
    "        h = self.features(h)\n",
    "        self.fb_features = h.clone().detach()\n",
    "        return self.output_layer(h)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset all u, b and a in Leaky-XXX Layer\"\"\"\n",
    "\n",
    "        for f in self.modules():\n",
    "            if (\n",
    "                isinstance(f, LSUM)\n",
    "                or isinstance(f, OLIF)\n",
    "                or isinstance(f, SAF)\n",
    "                or isinstance(f, LBias2d)\n",
    "                or isinstance(f, LBias)\n",
    "            ):\n",
    "                f.reset()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 3. Training and Validation\n",
    "\n",
    "In this section, we will define the model training and validation process.\n",
    "\n",
    "### 3.1. Training Process\n",
    "\n",
    "We define three types of training methods.\n",
    "\n",
    " - **Training_E** : Update the parameters with the gradients at each time\n",
    " - **Training_A** : Update the parameters with the sum of the gradients calculated at each time\n",
    " - **Training_F** : Update the parameters with the gradients at final time\n",
    "\n",
    "#### 3.1.1. Training_E"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def train_e(loader, model, criterion, optimizer, t_step, epoch):\n",
    "    \"\"\"Train-E\n",
    "\n",
    "    The training process computes and updates the gradients each time.\n",
    "\n",
    "    Args:\n",
    "        loader: DataLoder\n",
    "        model: training model\n",
    "        criterion: loss function\n",
    "        optimizer: optimizer\n",
    "        t_step: simulating time-steps\n",
    "        epoch: current epoch\n",
    "    \"\"\"\n",
    "\n",
    "    model.train()\n",
    "    total_loss = 0\n",
    "    total_acc = 0\n",
    "    total_samples = 0\n",
    "    for i, (x, t) in enumerate(loader):\n",
    "        x = x.cuda()\n",
    "        t = t.cuda()\n",
    "        Y = 0\n",
    "        for j in range(t_step):\n",
    "            optimizer.zero_grad()\n",
    "            y = model(x, j == 0)\n",
    "            Y = Y + y\n",
    "            loss = criterion(y, t) / t_step\n",
    "            loss.backward()\n",
    "            total_loss += loss.item() * t.numel()\n",
    "            optimizer.step()\n",
    "        total_samples += t.numel()\n",
    "        total_acc += (Y.argmax(1) == t).float().sum().item()\n",
    "    total_loss /= total_samples\n",
    "    total_acc /= total_samples\n",
    "    print(f\"[Train] Accuracy: {(100*total_acc):>0.1f}%, Loss: {total_loss:>8f}\")\n",
    "    return total_loss, total_acc"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 3.1.2. Training_A"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def train_a(loader, model, criterion, optimizer, t_step, epoch):\n",
    "    \"\"\"Train-A\n",
    "\n",
    "    The training process compute and update the sum of gradients at each time.\n",
    "\n",
    "    Args:\n",
    "        loader: DataLoder\n",
    "        model: training model\n",
    "        criterion: loss function\n",
    "        optimizer: optimizer\n",
    "        t_step: simulating time-steps\n",
    "        epoch: current epoch\n",
    "    \"\"\"\n",
    "\n",
    "    model.train()\n",
    "    total_loss = 0\n",
    "    total_acc = 0\n",
    "    total_samples = 0\n",
    "    for i, (x, t) in enumerate(loader):\n",
    "        x = x.cuda()\n",
    "        t = t.cuda()\n",
    "        Y = 0\n",
    "        optimizer.zero_grad()\n",
    "        for j in range(t_step):\n",
    "            y = model(x, j == 0)\n",
    "            Y = Y + y.clone().detach()\n",
    "            loss = criterion(y, t) / t_step\n",
    "            loss.backward()\n",
    "            total_loss += loss.item() * t.numel()\n",
    "        optimizer.step()\n",
    "        total_samples += t.numel()\n",
    "        total_acc += (Y.argmax(1) == t).float().sum().item()\n",
    "    total_loss /= total_samples\n",
    "    total_acc /= total_samples\n",
    "    print(f\"[Train] Accuracy: {(100*total_acc):>0.1f}%, Loss: {total_loss:>8f}\")\n",
    "    return total_loss, total_acc"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 3.1.3. Training_F"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def train_f(loader, model, criterion, optimizer, t_step, epoch, l=lif_lambda):\n",
    "    \"\"\"Train-F\n",
    "\n",
    "    The training process computes and updates the gradient at final time step.\n",
    "\n",
    "    Args:\n",
    "        loader: DataLoder\n",
    "        model: training model\n",
    "        criterion: loss function\n",
    "        optimizer: optimizer\n",
    "        t_step: simulating time-steps\n",
    "        epoch: current epoch\n",
    "        l: λ = 1 - (1 / τ)\n",
    "    \"\"\"\n",
    "\n",
    "    model.train()\n",
    "    total_loss = 0\n",
    "    total_acc = 0\n",
    "    total_samples = 0\n",
    "    for i, (x, t) in enumerate(loader):\n",
    "        x = x.cuda()\n",
    "        t = t.cuda()\n",
    "        a = 0\n",
    "        optimizer.zero_grad()\n",
    "        for j in range(t_step):\n",
    "            a = 1 + a * l\n",
    "            y = model(x, j == 0) / a\n",
    "            if (j + 1) == t_step:\n",
    "                loss = criterion(y, t)\n",
    "        loss.backward()\n",
    "        total_loss += loss.item() * t.numel()\n",
    "        optimizer.step()\n",
    "        total_samples += t.numel()\n",
    "        total_acc += (y.argmax(1) == t).float().sum().item()\n",
    "    total_loss /= total_samples\n",
    "    total_acc /= total_samples\n",
    "    print(f\"[Train] Accuracy: {(100*total_acc):>0.1f}%, Loss: {total_loss:>8f}\")\n",
    "    return total_loss, total_acc"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### 3.2. Validation Process"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def validation(loader, model, criterion, t_step, epoch):\n",
    "    \"\"\"Validation Process\n",
    "\n",
    "    Args:\n",
    "        loader: DataLoder\n",
    "        model: training model\n",
    "        criterion: loss function\n",
    "        t_step: simulating time-steps\n",
    "        epoch: current epoch\n",
    "    \"\"\"\n",
    "\n",
    "    model.eval()\n",
    "    total_loss = 0\n",
    "    total_acc = 0\n",
    "    total_samples = 0\n",
    "    with torch.no_grad():\n",
    "        for i, (x, t) in enumerate(loader):\n",
    "            x = x.cuda()\n",
    "            t = t.cuda()\n",
    "            Y = 0\n",
    "            for j in range(t_step):\n",
    "                y = model(x, j == 0)\n",
    "                Y = Y + y.clone().detach()\n",
    "            loss = criterion(y, t)\n",
    "            total_loss += loss.item() * t.numel()\n",
    "            total_samples += t.numel()\n",
    "            total_acc += (Y.argmax(1) == t).float().sum().item()\n",
    "    total_loss /= total_samples\n",
    "    total_acc /= total_samples\n",
    "    print(f\"[ Test] Accuracy: {(100*total_acc):>0.1f}%, Loss: {total_loss:>8f}\")\n",
    "    return total_loss, total_acc"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### 3.3. Optimization process"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def optimize_model(model, out_path, train):\n",
    "    \"\"\"Optimization of model parameters\n",
    "\n",
    "    Args:\n",
    "        model: training model\n",
    "        out_path: model output path\n",
    "        train: training process (E, A or F)\n",
    "    \"\"\"\n",
    "\n",
    "    last_model = os.path.join(out_path, \"epoch_\" + str(epochs).zfill(4) + \".pth\")\n",
    "    if os.path.exists(last_model):\n",
    "        print(last_model, \"is exists.\")\n",
    "    else:\n",
    "        os.makedirs(out_path, exist_ok=True)\n",
    "        criterion = CombineCEandMSE(loss_alpha, num_classes)\n",
    "        optimizer = torch.optim.SGD(\n",
    "            model.parameters(), lr=lr, momentum=momentum, weight_decay=weight_decay\n",
    "        )\n",
    "        scheduler = torch.optim.lr_scheduler.CosineAnnealingLR(optimizer, T_max=T_max)\n",
    "        columns = [\n",
    "            \"Epoch\",\n",
    "            \"Train Loss\",\n",
    "            \"Train ACC\",\n",
    "            \"Test Loss\",\n",
    "            \"Test ACC\",\n",
    "        ]\n",
    "        log = []\n",
    "        for epoch in range(1, epochs + 1):\n",
    "            print(f\"Epoch {epoch}\\n-------------------------------\")\n",
    "            train_loss, train_acc = train(\n",
    "                train_loader,\n",
    "                model,\n",
    "                criterion,\n",
    "                optimizer,\n",
    "                t_step,\n",
    "                epoch,\n",
    "            )\n",
    "            test_loss, test_acc = validation(\n",
    "                test_loader,\n",
    "                model,\n",
    "                criterion,\n",
    "                t_step,\n",
    "                epoch,\n",
    "            )\n",
    "            log += [\n",
    "                [\n",
    "                    epoch,\n",
    "                    train_loss,\n",
    "                    train_acc,\n",
    "                    test_loss,\n",
    "                    test_acc,\n",
    "                ]\n",
    "            ]\n",
    "            pd.DataFrame(log, columns=columns).to_csv(os.path.join(out_path, \"log.csv\"))\n",
    "            scheduler.step()\n",
    "            if (epoch % 100) == 0:\n",
    "                torch.save(\n",
    "                    {\"state_dict\": model.state_dict()},\n",
    "                    os.path.join(out_path, \"epoch_\" + str(epoch).zfill(4) + \".pth\"),\n",
    "                )"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "tags": []
   },
   "source": [
    "## 4. Experiment\n",
    "\n",
    "In the previous sections, we have defined the functions necessary for the experiment. In this section, we experimentally compare these methods. We trained SAF on the CIFAR-10 dataset (Krizhevsky and Hinton, 2009) and inferred with SNN composed of LIF neurons. \n",
    "\n",
    "### 4.1. Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def initialize_weights(model):\n",
    "    \"\"\"Initialize weight parameters\n",
    "\n",
    "    Args:\n",
    "        model: model\n",
    "    \"\"\"\n",
    "\n",
    "    for m in model.modules():\n",
    "        if isinstance(m, nn.Conv2d) or isinstance(m, sWSConv2d):\n",
    "            nn.init.kaiming_normal_(m.weight, mode=\"fan_out\", nonlinearity=\"relu\")\n",
    "            if m.bias is not None:\n",
    "                nn.init.constant_(m.bias, 0)\n",
    "        elif isinstance(m, LBias2d) or isinstance(m, LBias):\n",
    "            nn.init.constant_(m.bias, 0)\n",
    "        elif isinstance(m, nn.Linear):\n",
    "            nn.init.normal_(m.weight, 0, 0.01)\n",
    "            if m.bias is not None:\n",
    "                nn.init.constant_(m.bias, 0)\n",
    "\n",
    "\n",
    "def get_training_list(models, process, num):\n",
    "    \"\"\"Get a list of training settings\n",
    "\n",
    "    Args:\n",
    "        models: models\n",
    "        process: training process\n",
    "        num: number of models\n",
    "\n",
    "    Returns:\n",
    "        list: training settings\n",
    "    \"\"\"\n",
    "\n",
    "    out = np.meshgrid(models, process, np.arange(1, num + 1))\n",
    "    return np.array(out).reshape(3, -1).T\n",
    "\n",
    "\n",
    "# Number of models to create\n",
    "N = 1\n",
    "training_list = []\n",
    "# Training_E\n",
    "models = [SAFVGG, OTTTVGG, FSAFVGG, FOTTTVGG]\n",
    "updates = [train_e]\n",
    "training_list.append(get_training_list(models, updates, N))\n",
    "# Training_a\n",
    "models = [OTTTVGG]\n",
    "updates = [train_a]\n",
    "# Training_F\n",
    "training_list.append(get_training_list(models, updates, N))\n",
    "models = [SAFVGG_FR, FSAFVGG_FR]\n",
    "updates = [train_f]\n",
    "training_list.append(get_training_list(models, updates, N))\n",
    "# Concatenate\n",
    "training_list = np.concatenate(training_list, axis=0)\n",
    "for param in training_list:\n",
    "    model, train, num = param\n",
    "    tag = train.__name__.upper() + \"-\" + model.__name__\n",
    "    num = str(num).zfill(2)\n",
    "    out_path = os.path.join(out_dir, tag, num)\n",
    "    model = model(lif_lambda)\n",
    "    initialize_weights(model)\n",
    "    model.cuda()\n",
    "    optimize_model(model, out_path, train)\n",
    "    del model\n",
    "    torch.cuda.empty_cache()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### 4.2. Test\n",
    "#### 4.2.1. LIF VGG\n",
    "\n",
    "A simple SNN composed of LIF neurons can be defined as:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "class LIF(nn.Module):\n",
    "    \"\"\"Leaky-Integrate-and-Fire\"\"\"\n",
    "\n",
    "    def __init__(self, l: float = 0.0):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(LIF, self).__init__()\n",
    "        self.l = l\n",
    "        self.reset()\n",
    "\n",
    "    def forward(self, input: Tensor) -> Tensor:\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            input: input(intensity of current)\n",
    "\n",
    "        Returns:\n",
    "            Tensor: [spike, accumulated spike]\n",
    "        \"\"\"\n",
    "\n",
    "        # IF(I_t, u_t-1)\n",
    "        self.u = self.l * self.u + input\n",
    "        self.st = heaviside(self.u - 1.0)\n",
    "        # u_t-1 -> u_t\n",
    "        self.u = self.u - self.st\n",
    "        return self.st\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset u and a\"\"\"\n",
    "        # membrane potential\n",
    "        self.u = 0\n",
    "        self.st = None\n",
    "\n",
    "\n",
    "class VGG(nn.Module):\n",
    "    \"\"\"VGG Model\"\"\"\n",
    "\n",
    "    def __init__(self, l):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(VGG, self).__init__()\n",
    "        in_ch = 3\n",
    "        layers = []\n",
    "        for i, x in enumerate(cfg):\n",
    "            if x == \"A\":\n",
    "                layers += [nn.AvgPool2d(kernel_size=2, stride=2)]\n",
    "            elif x == \"M\":\n",
    "                layers += [nn.MaxPool2d(kernel_size=2, stride=2)]\n",
    "            else:\n",
    "                conv = sWSConv2d(in_ch, x, 3, 1, 1)\n",
    "                layers += [conv, LIF(l=l), Gamma()]\n",
    "                in_ch = x\n",
    "        layers += [\n",
    "            nn.AdaptiveAvgPool2d((1, 1)),\n",
    "            nn.Flatten(),\n",
    "            nn.Linear(cfg[-1], 10),\n",
    "        ]\n",
    "        self.features = nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x, init):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "            init: if True, reset all u and a in LIF\n",
    "\n",
    "        Returns:\n",
    "            Tensor: output\n",
    "        \"\"\"\n",
    "\n",
    "        if init:\n",
    "            self.reset()\n",
    "        return self.features(x)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset all u, b and a in Leaky-XXX Layer\"\"\"\n",
    "\n",
    "        for f in self.modules():\n",
    "            if isinstance(f, LIF):\n",
    "                f.reset()\n",
    "\n",
    "\n",
    "class FVGG(nn.Module):\n",
    "    \"\"\"Feadback-VGG\"\"\"\n",
    "\n",
    "    def __init__(self, l):\n",
    "        \"\"\"Initial setting\n",
    "\n",
    "        Args:\n",
    "            l: λ = 1 - (1 / τ)\n",
    "        \"\"\"\n",
    "\n",
    "        super(FVGG, self).__init__()\n",
    "        scale_factor = 2 ** cfg.count(\"A\") * 2 ** cfg.count(\"M\")\n",
    "        self.up = nn.Upsample(scale_factor=scale_factor, mode=\"nearest\")\n",
    "        self.fb = nn.Conv2d(\n",
    "            cfg[-1], cfg[0], kernel_size=3, padding=1, stride=1, bias=False\n",
    "        )\n",
    "        in_ch = 3\n",
    "        layers = []\n",
    "        for i, x in enumerate(cfg):\n",
    "            if x == \"A\":\n",
    "                layers += [nn.AvgPool2d(kernel_size=2, stride=2)]\n",
    "            elif x == \"M\":\n",
    "                layers += [nn.MaxPool2d(kernel_size=2, stride=2)]\n",
    "            else:\n",
    "                conv = sWSConv2d(in_ch, x, 3, 1, 1)\n",
    "                layers += [conv, LIF(l=l), Gamma()]\n",
    "                in_ch = x\n",
    "        layers += [\n",
    "            nn.AdaptiveAvgPool2d((1, 1)),\n",
    "            nn.Flatten(),\n",
    "            nn.Linear(cfg[-1], 10),\n",
    "        ]\n",
    "        self.input_layer = nn.Sequential(*layers[:1])\n",
    "        self.features = nn.Sequential(*layers[1:-3])\n",
    "        self.output_layer = nn.Sequential(*layers[-3:])\n",
    "\n",
    "    def forward(self, x, init):\n",
    "        \"\"\"Forward process\n",
    "\n",
    "        Args:\n",
    "            x: input\n",
    "            init: if True, reset all u and a in OLIF\n",
    "\n",
    "        Returns:\n",
    "            Tensor: output\n",
    "        \"\"\"\n",
    "\n",
    "        if init:\n",
    "            self.reset()\n",
    "            fb = 0\n",
    "        else:\n",
    "            fb = self.fb(self.up(self.fb_features))\n",
    "        h = self.input_layer(x) + fb\n",
    "        h = self.features(h)\n",
    "        self.fb_features = h.clone().detach()\n",
    "        return self.output_layer(h)\n",
    "\n",
    "    def reset(self):\n",
    "        \"\"\"Reset all u, b and a in Leaky-XXX Layer\"\"\"\n",
    "\n",
    "        for f in self.modules():\n",
    "            if isinstance(f, LIF):\n",
    "                f.reset()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 4.2.2. Conversion Process"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def convert(base_model, lif_lambda, target):\n",
    "    \"\"\"Convert OTTT or SAF model to simple LIF model\n",
    "\n",
    "    Args:\n",
    "        base_model: OTTT or SAF model\n",
    "        lif_lambda:\n",
    "        target:\n",
    "\n",
    "    Returns:\n",
    "        nn.Module: target_model\n",
    "    \"\"\"\n",
    "\n",
    "    model = target(lif_lambda)\n",
    "    params = get_params(model)\n",
    "    base_params = get_params(base_model)\n",
    "    for base_param, param in zip(base_params, params):\n",
    "        param.data = base_param.data.clone().detach()\n",
    "    return model\n",
    "\n",
    "\n",
    "def get_params(model):\n",
    "    \"\"\"Get list of model parameters\n",
    "\n",
    "    Args:\n",
    "        model: model\n",
    "\n",
    "    Returns:\n",
    "        list: model parameters\n",
    "    \"\"\"\n",
    "\n",
    "    params = []\n",
    "    for m in model.modules():\n",
    "        if (\n",
    "            isinstance(m, sWSConv2d)\n",
    "            or isinstance(m, nn.Linear)\n",
    "            or isinstance(m, nn.Conv2d)\n",
    "        ):\n",
    "            params.append(m.weight)\n",
    "            if isinstance(m, sWSConv2d):\n",
    "                if m.gamma is not None:\n",
    "                    params.append(m.gamma)\n",
    "            if m.bias is not None:\n",
    "                params.append(m.bias)\n",
    "        if isinstance(m, LBias2d) or isinstance(m, LBias):\n",
    "            params.append(m.bias)\n",
    "    return params"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 4.2.3. Test Process"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def get_fr(model):\n",
    "    \"\"\"get firing rate\n",
    "\n",
    "    Args:\n",
    "        models: models\n",
    "\n",
    "    Returns:\n",
    "        list: firing rate\n",
    "    \"\"\"\n",
    "\n",
    "    fr = []\n",
    "    for f in model.modules():\n",
    "        if isinstance(f, LIF):\n",
    "            fr.append(torch.mean(f.st).item())\n",
    "    return np.array(fr)\n",
    "\n",
    "\n",
    "def test(loader, model, criterion, t_step, epoch):\n",
    "    \"\"\"Test Process\n",
    "\n",
    "    Args:\n",
    "        loader: DataLoder\n",
    "        model: training model\n",
    "        criterion: loss function\n",
    "        t_step: simulating time-steps\n",
    "        epoch: current epoch\n",
    "    \"\"\"\n",
    "\n",
    "    model.eval()\n",
    "    total_loss = 0\n",
    "    total_acc = 0\n",
    "    total_samples = 0\n",
    "    total_fr = 0\n",
    "    for i, (x, t) in enumerate(loader):\n",
    "        x = x.float().cuda()\n",
    "        t = t.cuda()\n",
    "        Y = 0\n",
    "        for j in range(t_step):\n",
    "            y = model(x, j == 0)\n",
    "            Y = Y + y.clone().detach()\n",
    "            total_fr = total_fr + get_fr(model) * t.numel() / t_step\n",
    "        loss = criterion(y, t)\n",
    "        total_loss += loss.item() * t.numel()\n",
    "        total_samples += t.numel()\n",
    "        total_acc += (Y.argmax(1) == t).float().sum().item()\n",
    "    total_loss /= total_samples\n",
    "    total_acc /= total_samples\n",
    "    total_fr /= total_samples\n",
    "    print(f\"[ Test] Accuracy: {(100*total_acc):>0.1f}%, Loss: {total_loss:>8f}\")\n",
    "    return total_loss, total_acc, total_fr"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### 4.2.4. Run"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "TRAIN_E-OTTTVGG 01\n",
      "[ Test] Accuracy: 93.5%, Loss: 0.343495\n",
      "TRAIN_A-OTTTVGG 01\n",
      "[ Test] Accuracy: 93.3%, Loss: 0.347952\n",
      "TRAIN_E-SAFVGG 01\n",
      "[ Test] Accuracy: 93.6%, Loss: 0.340172\n",
      "TRAIN_F-SAFVGG_FR 01\n",
      "[ Test] Accuracy: 93.1%, Loss: 0.368815\n",
      "TRAIN_E-FOTTTVGG 01\n",
      "[ Test] Accuracy: 93.0%, Loss: 0.411566\n",
      "TRAIN_E-FSAFVGG 01\n",
      "[ Test] Accuracy: 93.2%, Loss: 0.409269\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABRkAAAMzCAYAAAAiYRXsAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/av/WaAAAACXBIWXMAAA9hAAAPYQGoP6dpAABnPUlEQVR4nOzde5RXZb0/8Pdw15BBwQFRFBVEUfOCqWhmJcrJW3pSkCJvZFLkDfkpagfJTO14yS6g59hwsZQBQq2VZipmoIkelTlpEWBCcEzwQoKCF4T5/eHym9+4yLBHRuT1WutZa2bv53n2Z2/3uPS9nr13RV1dXV0AAAAAADZQk8YuAAAAAADYtAkZAQAAAIBChIwAAAAAQCFCRgAAAACgECEjAAAAAFCIkBEAAAAAKETICAAAAAAUImQEAAAAAAoRMgIAAAAAhQgZAQAAAIBC6h0yTp06Nccdd1w6deqUioqK3HXXXR845qGHHsr++++fli1bpmvXrhk7duwGlAoAAAAAfBTVO2RctmxZ9tlnn4wcOXK9+s+dOzfHHHNMPve5z6W2tjbnn39+vva1r+W3v/1tvYsFAAAAAD56Kurq6uo2eHBFRe68886ccMIJa+1z8cUX5+67784zzzxT2nbKKafk1Vdfzb333ruhhwYAAAAAPiKafdgHePTRR9O7d++ybX369Mn555+/1jFvvfVW3nrrrdLvq1atyuLFi9OuXbtUVFR8WKUCAAAAAO9TV1eX1157LZ06dUqTJmt/KPpDDxkXLlyYDh06lG3r0KFDli5dmjfeeCNbbLHFamOuvvrqfOc73/mwSwMAAAAA1sOCBQuyww47rHX/hx4ybohLLrkkQ4YMKf2+ZMmS7LjjjlmwYEHatGnTiJUBAAAAwOZj6dKl6dy5c7baaqt19vvQQ8aOHTtm0aJFZdsWLVqUNm3arHEVY5K0bNkyLVu2XG17mzZthIwAAAAAsJF90CsM6/116frq1atXpkyZUrbt/vvvT69evT7sQwMAAAAAG0G9Q8bXX389tbW1qa2tTZLMnTs3tbW1mT9/fpJ3H3U+9dRTS/0HDRqU5557LhdddFH+8pe/ZNSoUZk4cWIuuOCChjkDAAAAAKBR1TtkfOKJJ7Lffvtlv/32S5IMGTIk++23X4YPH54keeGFF0qBY5LsvPPOufvuu3P//fdnn332yfXXX5+f/vSn6dOnTwOdAgAAAADQmCrq6urqGruID7J06dJUVlZmyZIl3skIAAAAfChWrlyZFStWNHYZsFE1b948TZs2Xev+9c3lPpJflwYAAADYWOrq6rJw4cK8+uqrjV0KNIq2bdumY8eOH/hxl3URMgIAAACbtfcCxqqqqmy55ZaFghbYlNTV1WX58uV58cUXkyTbbbfdBs8lZAQAAAA2WytXriwFjO3atWvscmCj22KLLZIkL774Yqqqqtb56PS61PvDLwAAAAAfF++9g3HLLbds5Eqg8bx3/xd5J6mQEQAAANjseUSazVlD3P9CRgAAAACgECEjAAAAAFCID78AAAAArEGXYXdv1OPNu+aY9e77QY+3Xn755Tn99NOz8847l7ZtvfXW2XvvvXPllVfmsMMOW23M2WefnZ/+9KepqanJySefXLZvxIgRueuuu1JbW1v6/Tvf+U7OPvvs3HzzzaV+tbW12W+//TJ37tx06dJlvc+nT58+eeCBBzJ9+vR86lOfWq8xCxYsyOWXX5577703L7/8crbbbruccMIJGT58eNq1a5d58+aVnf+a7LTTTvnb3/621v2nnXZaxo0bt87xW221VQ499NCy6/Cen/3sZ/na176W559/Pu3bt09dXV1++tOfZvTo0fnTn/6UVatWZaeddkrv3r1zzjnnpGvXrqWxS5cuzbXXXps77rgjzz33XLbccsvssssuOfnkk3PWWWdl6623Xo+rtPFYyQgAAACwiXnhhRdK7cYbb0ybNm3Ktg0dOrTU94EHHsgLL7yQqVOnplOnTjn22GOzaNGisvmWL1+empqaXHTRRRk9evR61dCqVatUV1dnzpw5hc5l/vz5+cMf/pBvfetb633s5557LgcccEDmzJmT8ePH59lnn83NN9+cKVOmpFevXlm8eHE6d+5cdk0uvPDC7LnnnmXbpk2bVvp58uTJSZJZs2aVtv3whz8s658kY8aMKf3+P//zPxk4cGBqamryxhtvrFbnmDFjcvzxx5cCxi9/+cs599xzc/TRR+e+++7Ln//851RXV6dVq1a58sorS+MWL16cgw8+OGPGjMnQoUPz2GOP5amnnsr3vve9zJgxI7fffnuha/5hsJIRAAAAYBPTsWPH0s+VlZWpqKgo25YkL7/8cpKkXbt26dixYzp27JhLL700NTU1eeyxx3L88ceX+k6aNCk9evTIsGHD0qlTpyxYsCCdO3deZw3du3dPVVVVLrvsskycOHGDz2XMmDE59thj841vfCMHH3xwbrjhhmyxxRbrHDN48OC0aNEi9913X6nvjjvumP322y+77rprLrvsstx0001l16R169Zp1qzZatfpPdtss02SpKqqKm3bti1tr6ysLOvXtm3bsjkGDBiQiy++OJMnT86AAQNK2+fOnZuHHnoo99xzT5JkwoQJqampyS9/+cuya7/jjjvm4IMPTl1dXWnbpZdemvnz52f27Nnp1KlTaftOO+2Uo446qqzvR4WVjAAAAACbgTfeeCO33nprkqRFixZl+6qrqzNgwIBUVlbmC1/4QsaOHbtec15zzTWZPHlynnjiiQ2qqa6uLmPGjMmAAQOy++67p2vXrvnFL36xzjGLFy/Ob3/723zzm99cLYzs2LFjvvKVr2TChAkbLYhr3759vvjFL662CnPs2LHZYYcdctRRRyVJxo8fn+7du5cFjO/33iPwq1atyoQJEzJgwICygHFNfT9KhIwAAAAAH2OHHHJIWrdunU984hO57rrr0rNnzxxxxBGl/XPmzMn06dPTr1+/JO+uzBszZsx6hXT7779/+vbtm4svvniDanvggQeyfPny9OnTp3Ts6urqdY6ZM2dO6urqsscee6xx/x577JF//OMfeemllzaopg0xcODAPPTQQ5k7d26Sd8PTcePG5bTTTkuTJu/Gb7Nnz0737t3Lxp1//vlp3bp1WrdunR122CFJ8tJLL+XVV19drW/Pnj1Lffv3778Rzqp+hIwAAAAAH2MTJkzIjBkzMnny5HTt2jVjx45N8+bNS/tHjx6dPn36pH379kmSo48+OkuWLMmDDz64XvNfeeWVmTZtWu6777561zZ69Oj069cvzZq9+0a//v3755FHHslf//rXJMmgQYNKwVrr1q3Lxn6UHhk+8sgjs8MOO2TMmDFJkilTpmT+/Pk544wz1jnusssuS21tbYYPH57XX399nX3vvPPO1NbWpk+fPmt8/2NjEzICAAAAfIx17tw53bp1y4knnpirrroqJ554Yt56660kycqVKzNu3LjcfffdadasWZo1a5Ytt9wyixcvXu+PsOy6664566yzMmzYsHoFf4sXL86dd96ZUaNGlY69/fbb55133ikd+4orrkhtbW2pJUnXrl1TUVGRmTNnrnHemTNnZuutt86222673rUU1aRJk5x++ukZN25cVq1alTFjxuRzn/tcdtlll1Kfbt26ZdasWWXjtt1223Tt2jVVVVVl29q2bbta3x133DFdu3bNVltt9eGezAYSMgIAAABsJk466aQ0a9Yso0aNSpLcc889ee211zJjxoyyMG/8+PG544478uqrr67XvMOHD8/s2bNTU1Oz3rXcdttt2WGHHfK///u/Zce+/vrrM3bs2KxcuTJVVVXp2rVrqSXvfsjmyCOPzKhRo1Zb0bdw4cLcdttt6dev30Z/b+EZZ5yRBQsW5I477sidd96ZgQMHlu3v379/Zs2alV/+8pfrnKdJkybp27dvfv7zn+fvf//7h1lygxIyAgAAAGwmKioqcu655+aaa67J8uXLU11dnWOOOSb77LNP9tprr1Lr27dv2rZtm9tuu2295u3QoUOGDBmSH/3oR+tdS3V1dU466aSy4+61114ZOHBgXn755dx7771rHfuTn/wkb731Vvr06ZOpU6dmwYIFuffee3PkkUdm++23z/e+9731rqOh7Lzzzvn85z+fr3/962nZsmX+/d//vWz/KaeckpNOOimnnHJKrrjiijz22GOZN29efv/732fChAlp2rRpqe9VV12V7bffPgceeGBGjx6dP/7xj/nrX/+aO++8M48++mhZ348KISMAAADAZuS0007LihUr8uMf/zh33313vvSlL63Wp0mTJjnxxBM/8CMs7zd06NDV3pu4Nk8++WT+93//d43HrqyszBFHHLHOY3fr1i1PPPFEdtlll/Tt2ze77rprvv71r+dzn/tcHn300WyzzTbrXXdDGjhwYP7xj3/ky1/+clq1alW2r6KiIhMmTMiNN96Ye+65J0cccUS6d++eM888M507d87DDz9c6tuuXbs8/vjjOfXUU3PttdfmwAMPzN57750RI0akX79+ueWWWzb2qX2girqP0lsy12Lp0qWprKzMkiVL0qZNm8YuBwAAAPiYePPNNzN37tzsvPPOq4VCsLlY19/B+uZyVjICAAAAAIUIGQEAAABocIMGDUrr1q3X2AYNGtTY5dHAmjV2AQAAAAB8/FxxxRUZOnToGvd5Hd7Hj5ARAAAAgAZXVVWVqqqqxi6DjcTj0gAAAABAIUJGAAAAAKAQISMAAAAAUIiQEQAAAAAoRMgIAAAAABQiZAQAAAAAChEyAgAAAKzJiMqN2+qhoqJinW3EiBGZN29e2bZtttkmhx9+eKZNm7bGOc8+++w0bdo0kyZNWv1SjBiRfffdt+z3ioqKDBo0qKxfbW1tKioqMm/evA88h3+t7/1t+vTpax333rH/tT3wwAOlPosXL87555+fnXbaKS1atEinTp1y5plnZv78+et9DT/72c8W2l9RUZHjjjsu//Zv/7bG85g2bVoqKiryxz/+sbRt8uTJ+fznP5+tt946W2yxRbp3754zzzwzM2bMKBv79ttv59prr83++++fT3ziE6msrMw+++yTb3/72/n73//+gdf+wyBkBAAAANjEvPDCC6V24403pk2bNmXbhg4dWur7wAMP5IUXXsjUqVPTqVOnHHvssVm0aFHZfMuXL09NTU0uuuiijB49er1qaNWqVaqrqzNnzpxC5/Jefe9vPXv2XOeYPffcc7Uxn/nMZ5K8GzAefPDBeeCBB3LzzTfn2WefTU1NTZ599tl86lOfynPPPZfkg6/hrbfeWvr58ccfX63W9+9/4YUXssMOO+SKK64o2zZw4MDcf//9+b//+7/VzmHMmDE54IAD8slPfjJJcvHFF6dfv37Zd99986tf/SqzZs3K7bffnl122SWXXHJJadxbb72VI488MldddVVOP/30TJ06NU8//XR+9KMf5eWXX86Pf/zjQv88NlSzRjkqAAAAABusY8eOpZ8rKytTUVFRti1JXn755SRJu3bt0rFjx3Ts2DGXXnppampq8thjj+X4448v9Z00aVJ69OiRYcOGpVOnTlmwYEE6d+68zhq6d++eqqqqXHbZZZk4ceIGn8t79dVHs2bN1jrmsssuy9///vc8++yzpT477rhjfvvb36Zbt24ZPHhwfvOb36zXNXzPm2+++YG1Nm3aNFtttVXZ/mOPPTbbbrttxo4dm29/+9ul7a+//nomTZqUa6+9Nkkyffr0/Od//md++MMf5txzzy3123HHHdOzZ8/U1dWVtv3gBz/Iww8/nCeeeCL77bdfWd/DDz+8rO/GZCUjAAAAwGbgjTfeyK233pokadGiRdm+6urqDBgwIJWVlfnCF76QsWPHrtec11xzTSZPnpwnnniiocvdIKtWrUpNTU2+8pWvrBYGbrHFFvnmN7+Z3/72t1m8ePFGqadZs2Y59dRTM3bs2LLwb9KkSVm5cmX69++fJBk/fnxat26db37zm2ucp6KiovTz+PHjc+SRR5YFjGvruzEJGQEAAAA+xg455JC0bt06n/jEJ3LdddelZ8+eOeKII0r758yZk+nTp6dfv35JkgEDBmTMmDHrtSJu//33T9++fXPxxRcXru/97YM8/fTTZf0PPPDAJMlLL72UV199NXvssccax+2xxx6pq6vLs88+u8H11teZZ56Zv/71r/n9739f2jZmzJh86UtfSmXlu+/inD17dnbZZZc0a/bPh45vuOGGsnNcsmRJqW/37t3LjnHiiSeW+h1yyCEb4axWJ2QEAAAA+BibMGFCZsyYkcmTJ6dr164ZO3ZsmjdvXto/evTo9OnTJ+3bt0+SHH300VmyZEkefPDB9Zr/yiuvzLRp03LfffdtcH21tbVlLUnmz59fFrJdddVVpTHdu3cv6z958uSyORvrkeE12X333XPIIYeU3nX57LPPZtq0aRk4cOA6x5155pmpra3Nf/3Xf2XZsmXrPKdRo0altrY2Z555ZpYvX96g9a8v72QEAAAA+Bjr3LlzunXrlm7duuWdd97JiSeemGeeeSYtW7bMypUrM27cuCxcuLBsFd3KlSszevToshWPa7PrrrvmrLPOyrBhw1JdXb1B9XXt2nW17Z06dSoFjkmyzTbblH5u0aLFGsdsu+22adu2bWbOnLnGY82cOTMVFRVrHPthGjhwYM4555yMHDkyY8aMya677prDDz+8tL9bt255+OGHs2LFilIA3LZt27Rt23a1j8Z069Yts2bNKtu23XbbJSm/RhublYwAAAAAm4mTTjopzZo1y6hRo5Ik99xzT1577bXMmDGjbGXg+PHjc8cdd+TVV19dr3mHDx+e2bNnp6ampsFqbdasWbp27Vpq6xOgNWnSJH379s3tt9+ehQsXlu174403MmrUqPTp02ejh3F9+/ZNkyZNcvvtt+fWW2/NmWeeWfbuxP79++f1118v/XNZl/79++f+++/PjBkzPsyS603ICAAAALCZqKioyLnnnptrrrkmy5cvT3V1dY455pjss88+2WuvvUqtb9++adu2bW677bb1mrdDhw4ZMmRIfvSjH9W7pldeeSULFy4sa+99zXlDXHXVVenYsWOOPPLI/OY3v8mCBQsyderU9OnTJytWrMjIkSM3eO4N1bp16/Tr1y+XXHJJXnjhhZx++ull+3v16pULL7wwF154YYYMGZKHH344f/vb3zJ9+vRUV1enoqIiTZq8G+NdcMEF6dWrV4444oj88Ic/zFNPPZW5c+fmt7/9bX7zm9+kadOmG/38EiEjAAAAwGbltNNOy4oVK/LjH/84d999d770pS+t1qdJkyY58cQT6/X489ChQ9froy3/qnfv3tluu+3K2l133VXved7Trl27TJ8+PZ/73Ody9tlnZ9ddd03fvn2z66675n/+53+yyy67bPDcRQwcODD/+Mc/0qdPn3Tq1Gm1/dddd11uv/32zJgxI8cee2y6deuWk08+OatWrcqjjz6aNm3aJElatWqVKVOm5OKLL86YMWPy6U9/OnvssUfOP//8HHrooYWuXREVdR+lN2GuxdKlS1NZWZklS5aULigAAABAUW+++Wbmzp2bnXfeOa1atWrscqBRrOvvYH1zOSsZAQAAAIBChIwAAAAANLhBgwaldevWa2yDBg1q7PJoYM0+uAsAAAAA1M8VV1yRoUOHrnGf1+F9/AgZAQAAAGhwVVVVqaqqauwy2Eg8Lg0AAAAAFCJkBAAAAAAKETICAAAAAIUIGQEAAACAQoSMAAAAAEAhQkYAAAAAoBAhIwAAAMAa7D1u743a6qOiomKdbcSIEZk3b17Ztm222SaHH354pk2btsY5zz777DRt2jSTJk1abd+IESOy7777lv1eUVGRQYMGlfWrra1NRUVF5s2b94Hn8K/1vb9Nnz59rePeO/a/tgceeCBJMnfu3Hz5y19Op06d0qpVq+ywww754he/mL/85S+rzTV+/Pg0bdo0gwcPXm3fQw89tMbjfPvb387kyZPTtGnTPP/882ussVu3bhkyZEjp92effTZnnnlmdtxxx7Rs2TLbb799jjjiiNx222155513ysb+7ne/y7HHHpttt902rVq1yq677pp+/fpl6tSpH3hNG5OQEQAAAGAT88ILL5TajTfemDZt2pRtGzp0aKnvAw88kBdeeCFTp05Np06dcuyxx2bRokVl8y1fvjw1NTW56KKLMnr06PWqoVWrVqmurs6cOXMKnct79b2/9ezZc51j9txzz9XGfOYzn8mKFSty5JFHZsmSJbnjjjsya9asTJgwIXvvvXdeffXV1eaprq7ORRddlPHjx+fNN99c47FmzZpVdpxhw4bl+OOPT7t27TJu3LjV+k+dOjXPPvtsBg4cmCR5/PHHs//++2fmzJkZOXJknnnmmTz00EP52te+lptuuil/+tOfSmNHjRqVI444Iu3atcuECRMya9as3HnnnTnkkENywQUX1OOqbnzNGrsAAAAAAOqnY8eOpZ8rKytTUVFRti1JXn755SRJu3bt0rFjx3Ts2DGXXnppampq8thjj+X4448v9Z00aVJ69OiRYcOGpVOnTlmwYEE6d+68zhq6d++eqqqqXHbZZZk4ceIGn8t79dVHs2bN1jimtrY2f/3rXzNlypTstNNOSZKddtophx566Gp9586dmz/84Q+ZPHlyfve73+WOO+7Il7/85dX6VVVVpW3btqtt/+pXv5qxY8fm0ksvLds+evToHHTQQdlzzz1TV1eX008/PbvttlseeeSRNGnyz/V+3bp1S//+/VNXV5ckmT9/fs4///ycf/75ueGGG8rm/OQnP5lzzz33gy9MI7KSEQAAAGAz8MYbb+TWW29NkrRo0aJsX3V1dQYMGJDKysp84QtfyNixY9drzmuuuSaTJ0/OE0880dDlbpBtt902TZo0yS9+8YusXLlynX3HjBmTY445JpWVlRkwYECqq6vrdayBAwdmzpw5ZY8xv/766/nFL35RWsVYW1ubmTNnZujQoWUB4/tVVFQkSSZPnpwVK1bkoosuWme/jyohIwAAAMDH2CGHHJLWrVvnE5/4RK677rr07NkzRxxxRGn/nDlzMn369PTr1y9JMmDAgIwZM6a0wm5d9t9///Tt2zcXX3xx4fre3z7I008/Xdb/wAMPTJJsv/32+dGPfpThw4dn6623zuc///l897vfzXPPPVc2ftWqVRk7dmwGDBiQJDnllFPy8MMPZ+7cuasda4cddig71iuvvJIk6dGjRw4++OCyx8snTpyYurq6nHLKKUmS2bNnJ3l31ed7XnzxxbL5Ro0aVerbpk2bshWakydPLuv79NNPf/AFbSRCRgAAAICPsQkTJmTGjBmZPHlyunbtmrFjx6Z58+al/aNHj06fPn3Svn37JMnRRx+dJUuW5MEHH1yv+a+88spMmzYt99133wbXV1tbW9aSdx8ffn/AdtVVV5XGdO/evaz/5MmTS/sGDx6chQsX5rbbbkuvXr0yadKk7Lnnnrn//vtLfe6///4sW7YsRx99dJKkffv2OfLII9f4Pspp06aVHWvrrbcu7TvzzDPzi1/8Iq+99lqSd6/lySefnK222mqt59uuXbvSXG3bts3bb79d2vevqxX79OmT2tra3H333Vm2bNkHrs5sTN7JCAAAAPAx1rlz53Tr1i3dunXLO++8kxNPPDHPPPNMWrZsmZUrV2bcuHFZuHBhmjX7Z0y0cuXKjB49umzF49rsuuuuOeusszJs2LB6P3L8Xn1du3ZdbXunTp1KgWOSbLPNNqWfW7RoscYx79lqq61y3HHH5bjjjsuVV16ZPn365Morr8yRRx6Z5N3HwxcvXpwtttiiNGbVqlX54x//mO985ztljzbvvPPOa3wnY/LuCsgLLrggEydOzGc+85k88sgjufrqq0v7u3XrluTdj8fst99+SZKmTZuWan//Ne/WrVuWLFmShQsXllYztm7dOl27di3r91FlJSMAAADAZuKkk05Ks2bNSo/o3nPPPXnttdcyY8aMstV648ePzx133LHGLzKvyfDhwzN79uzU1NQ0WK3NmjVL165dS+39IWN9VFRUZPfdd8+yZcuSJK+88kp++ctfpqampuycZ8yYkX/84x/1WpG51VZb5eSTT87o0aMzZsyY7LbbbjnssMNK+/fbb7/svvvuue6667Jq1ap1znXSSSelefPm+f73v79B59nYPvoxKAAAAAANoqKiIueee25GjBiRs88+O9XV1TnmmGOyzz77lPXr0aNHLrjggtx2220ZPHjwB87boUOHDBkyJNdee229a3rllVeycOHCsm1t27ZNq1at6j1XbW1tLr/88nz1q19Njx490qJFi/z+97/P6NGjS++N/NnPfpZ27dqlb9++qz2efPTRR6e6ujr/9m//tt7HHDhwYA477LDMnDlztXdTVlRUZMyYMTnyyCNz6KGH5pJLLskee+yRFStWZOrUqXnppZfStGnTJMmOO+6Y66+/Puedd14WL16c008/PTvvvHMWL16cn//850lS6vtRZCUjAAAAwGbktNNOy4oVK/LjH/84d999d770pS+t1qdJkyY58cQT6/X489ChQ9froy3/qnfv3tluu+3K2l133VXveZJ3P9LSpUuXfOc738lBBx2U/fffPz/84Q/zne98J5dddlmSd9+beOKJJ67xa81f+tKX8qtf/Sovv/zyeh/z05/+dLp3756lS5fm1FNPXW3/wQcfnCeffDLdu3fP4MGD06NHjxxyyCEZP358fvCDH+Qb3/hGqe8555yT++67Ly+99FJOOumkdOvWLUcffXTmzp2be++9N3vvvfcGXJWNo6JufT4V1MiWLl2aysrKLFmyJG3atGnscgAAAICPiTfffDNz587NzjvvvEEr5+DjYF1/B+uby1nJCAAAAAAUImQEAAAAoMENGjQorVu3XmMbNGhQY5dHA/PhFwAAAAAa3BVXXJGhQ4eucZ/X4X38CBkBAAAAaHBVVVWpqqpq7DLYSDwuDQAAAAAUImQEAAAAAAoRMgIAAAAAhQgZAQAAAIBChIwAAAAAQCFCRgAAAACgECEjAAAAwBrM3H2Pjdrqo6KiYp1txIgRmTdvXtm2bbbZJocffnimTZu2xjnPPvvsNG3aNJMmTVpt34gRI7LvvvuW/V5RUZFBgwaV9autrU1FRUXmzZv3gefwr/W9v02fPv0Dx48bNy6f+tSnsuWWW2arrbbK4Ycfnl//+tel/aeffvo6r1GXLl0+8Dp+ULv++uvTtGnTPP/882ussVu3bhkyZEjp92effTZnnnlmdtxxx7Rs2TLbb799jjjiiNx222155513ysb+7ne/y7HHHpttt902rVq1yq677pp+/fpl6tSpH3htGoOQEQAAAGAT88ILL5TajTfemDZt2pRtGzp0aKnvAw88kBdeeCFTp05Np06dcuyxx2bRokVl8y1fvjw1NTW56KKLMnr06PWqoVWrVqmurs6cOXMKnct79b2/9ezZc51jhg4dmrPPPjv9+vXLH//4xzz++OP59Kc/nS9+8Yv5yU9+kiT54Q9/WDZnkowZM6b0+7Rp08r29+rVK2eddVbp9//7v//L//3f/5V+v/DCC7PnnnuWjfn617+edu3aZdy4cavVOHXq1Dz77LMZOHBgkuTxxx/P/vvvn5kzZ2bkyJF55pln8tBDD+VrX/tabrrppvzpT38qjR01alSOOOKItGvXLhMmTMisWbNy55135pBDDskFF1xQ6Hp/WJo1dgEAAAAA1E/Hjh1LP1dWVqaioqJsW5K8/PLLSZJ27dqlY8eO6dixYy699NLU1NTksccey/HHH1/qO2nSpPTo0SPDhg1Lp06dsmDBgnTu3HmdNXTv3j1VVVW57LLLMnHixA0+l/fqW1/Tp0/P9ddfnx/96Ec555xzStu/973v5c0338yQIUPyxS9+MZ07d05lZWXZ2LZt2671WC1atMiWW2651v2tW7dOs2bNVtv/1a9+NWPHjs2ll15atn306NE56KCDsueee6auri6nn356dttttzzyyCNp0uSf6/66deuW/v37p66uLkkyf/78nH/++Tn//PNzww03lM35yU9+Mueee+4HXKHGYSUjAAAAwGbgjTfeyK233prk3UDt/aqrqzNgwIBUVlbmC1/4QsaOHbtec15zzTWZPHlynnjiiYYud63Gjx+f1q1b5+yzz15t34UXXpgVK1Zk8uTJG62egQMHZs6cOWWPMb/++uv5xS9+UVrFWFtbm5kzZ2bo0KFlAeP7VVRUJEkmT56cFStW5KKLLlpnv48aISMAAADAx9ghhxyS1q1b5xOf+ESuu+669OzZM0cccURp/5w5czJ9+vT069cvSTJgwICMGTOmtLJuXfbff//07ds3F198ceH63t/WZfbs2dl1111XC0qTpFOnTmnTpk1mz569wfXUV48ePXLwwQeXPWY+ceLE1NXV5ZRTTinVnLy7+vM9L774Ytk5jxo1qtS3TZs2ZSsmJ0+eXNb36aef3hinVi9CRgAAAICPsQkTJmTGjBmZPHlyunbtmrFjx6Z58+al/aNHj06fPn3Svn37JMnRRx+dJUuW5MEHH1yv+a+88spMmzYt99133wbXV1tbW9aSdx8bfn+wdtVVV5XGrE8AujGdeeaZ+cUvfpHXXnstybvX9OSTT85WW2211jHt2rUrnW/btm3z9ttvl/b962rFPn36pLa2NnfffXeWLVuWlStXfjgnUoB3MgIAAAB8jHXu3DndunVLt27d8s477+TEE0/MM888k5YtW2blypUZN25cFi5cmGbN/hkTrVy5MqNHjy5b8bg2u+66a84666wMGzYs1dXVG1Rf165dV9veqVOnUuCYJNtss02SZLfddsvDDz+ct99+e7XVjH//+9+zdOnS7LbbbvWuo4hTTjklF1xwQSZOnJjPfOYzeeSRR3L11VeX9nfr1i1JMmvWrOy3335JkqZNm5bO+/3Xvlu3blmyZEkWLlxYWs3YunXrdO3atazfR42VjAAAAACbiZNOOinNmjUrPZp7zz335LXXXsuMGTPKVhKOHz8+d9xxR1599dX1mnf48OGZPXt2ampqGqzWZs2apWvXrqX2Xsh4yimn5PXXX89//dd/rTbmuuuuS/PmzfOlL32pwepYH1tttVVOPvnkjB49OmPGjMluu+2Www47rLR/v/32y+67757rrrsuq1atWudcJ510Upo3b57vf//7H3bZDeqjG38CAAAA0KAqKipy7rnnZsSIETn77LNTXV2dY445Jvvss09Zvx49euSCCy7IbbfdlsGDB3/gvB06dMiQIUNy7bXX1rumV155JQsXLizb1rZt27Rq1WqN/Xv16pXzzjsv/+///b+8/fbbOeGEE7JixYr8/Oc/zw9/+MPceOONH/hl7A/DwIEDc9hhh2XmzJmrvaOyoqIiY8aMyZFHHplDDz00l1xySfbYY4+sWLEiU6dOzUsvvZSmTZsmSXbcccdcf/31Oe+887J48eKcfvrp2XnnnbN48eL8/Oc/T5JS348SKxkBAAAANiOnnXZaVqxYkR//+Me5++6717jqr0mTJjnxxBPr9fjz0KFDP/CjLWvSu3fvbLfddmXtrrvuWueYG2+8MaNGjcr48eOz11575YADDsjUqVNz11135Zxzzql3DQ3h05/+dLp3756lS5fm1FNPXW3/wQcfnCeffDLdu3fP4MGD06NHjxxyyCEZP358fvCDH+Qb3/hGqe8555yT++67Ly+99FJOOumkdOvWLUcffXTmzp2be++9N3vvvffGPLX1UlH3UXtT5hosXbo0lZWVWbJkSdq0adPY5QAAAAAfE2+++Wbmzp2bnXfeea0r5+Djbl1/B+uby1nJCAAAAAAUImQEAAAAoMENGjQorVu3XmMbNGhQY5dHA/PhFwAAAAAa3BVXXJGhQ4eucZ/X4X38CBkBAAAAaHBVVVWpqqpq7DLYSDwuDQAAAAAUImQEAAAAAAoRMgIAAAAAhQgZAQAAAIBChIwAAAAAQCFCRgAAAACgkGaNXQAAAADAR9HIQQ9u1OMNvvnz6923oqJinfsvv/zynH766dl5551L27beeuvsvffeufLKK3PYYYetNubss8/OT3/609TU1OTkk08u2zdixIjcddddqa2tLf3+ne98J2effXZuvvnmUr/a2trst99+mTt3brp06bLOGufNm1dW3/s9+uijOfjgg9e4b/ny5fnud7+biRMn5vnnn89WW22VHj16ZMiQIfniF79Y1vf//u//sssuu2S33XbLM888s9pca7qOhx56aCZPnpwddtghP/vZz3LKKaes1mfgwIGZMWNGnnrqqSTJ0qVLc+211+aOO+7Ic889ly233DK77LJLTj755Jx11lnZeuutS2OfffbZXHXVVXnggQeyaNGitG/fPrvvvnvOPPPM9OvXL82abZpxnZWMAAAAAJuYF154odRuvPHGtGnTpmzb0KFDS30feOCBvPDCC5k6dWo6deqUY489NosWLSqbb/ny5ampqclFF12U0aNHr1cNrVq1SnV1debMmVPoXN6r7/2tZ8+ea+0/aNCg3HHHHfnxj3+cv/zlL7n33ntz0kkn5ZVXXlmt79ixY9O3b98sXbo0jz322BrnGzNmTNmxf/WrX6VDhw455phj1ngtli1blokTJ2bgwIFJksWLF+fggw/OmDFjMnTo0Dz22GN56qmn8r3vfS8zZszI7bffXhr7+OOPZ//998/MmTMzcuTIPPPMM3nooYfyta99LTfddFP+9Kc/1ffyfWRsmtEoAAAAwGasY8eOpZ8rKytTUVFRti1JXn755SRJu3bt0rFjx3Ts2DGXXnppampq8thjj+X4448v9Z00aVJ69OiRYcOGpVOnTlmwYEE6d+68zhq6d++eqqqqXHbZZZk4ceIGn8t79a2vX/3qV/nhD3+Yo48+OknSpUuXNYaSdXV1GTNmTEaNGpUddtgh1dXVOeigg1br17Zt2zUef+DAgTnhhBMyf/787LjjjqXtkyZNyjvvvJOvfOUrSZJLL7008+fPz+zZs9OpU6dSv5122ilHHXVU6urqSvWcfvrp2W233fLII4+kSZN/rv3r1q1b+vfvX+q7KbKSEQAAAGAz8MYbb+TWW29NkrRo0aJsX3V1dQYMGJDKysp84QtfyNixY9drzmuuuSaTJ0/OE0880dDlrlXHjh1zzz335LXXXltnv9/97ndZvnx5evfunQEDBqSmpibLli1b7+McffTR6dChw2rXYsyYMfn3f//3tG3bNqtWrcqECRMyYMCAsoDx/d57JLu2tjYzZ87M0KFDywLGNfXdFAkZAQAAAD7GDjnkkLRu3Tqf+MQnct1116Vnz5454ogjSvvnzJmT6dOnp1+/fkmSAQMGZMyYMeu1qm7//fdP3759c/HFFxeu7/1tXf77v/87f/jDH9KuXbt86lOfygUXXJBHHnlktX7V1dU55ZRT0rRp0+y1117ZZZddMmnSpNX69e/fv+zYd911V5KkadOmOe200zJ27NjStfjrX/+aadOm5cwzz0ySvPTSS3n11VfTvXv3sjl79uxZmq9///5JktmzZydJWd8XX3yx7NijRo1az6v20SNkBAAAAPgYmzBhQmbMmJHJkyena9euGTt2bJo3b17aP3r06PTp0yft27dP8u4KviVLluTBB9fvwzdXXnllpk2blvvuu2+D66utrS1rSTJ//vyyAO6qq65KknzmM5/Jc889lylTpuSkk07Kn/70pxx22GH57ne/W5rz1VdfzR133JEBAwaUtg0YMCDV1dWrHf8HP/hB2bGPPPLI0r4zzzwzc+fOze9+97sk765i7NKlSz7/+XV/pOfOO+9MbW1t+vTpkzfeeGOt/dq1a1c6btu2bfP2229/8AX7iPJORgAAAICPsc6dO6dbt27p1q1b3nnnnZx44ol55pln0rJly6xcuTLjxo3LwoULy75qvHLlyowePbpsxePa7LrrrjnrrLMybNiwNYZ461Nf165dV9veqVOnUuCYJNtss03p5+bNm+ewww7LYYcdlosvvjhXXnllrrjiilx88cVp0aJFbr/99rz55ptl72Csq6vLqlWrMnv27Oy2226l7R07dlzj8ZN335V42GGHZcyYMfnsZz+bW2+9NWeddVbpseZtt902bdu2zaxZs8rGvfcOx6222iqvvvpqaa4kmTVrVvbbb78k766WfO/Ym+pXpd9jJSMAAADAZuKkk05Ks2bNSo/lvvduwxkzZpSt5hs/fnzuuOOOUkD2QYYPH57Zs2enpqamwWpt1qxZunbtWmrvDxn/VY8ePfLOO+/kzTffTPLuo9IXXnhh2Tn97//+bw477LD1/nr2ewYOHJjJkydn8uTJef7553P66aeX9jVp0iR9+/bNz3/+8/z9739f5zz77bdfdt9991x33XVZtWpVvWrYFAgZAQAAADYTFRUVOffcc3PNNddk+fLlqa6uzjHHHJN99tkne+21V6n17ds3bdu2zW233bZe83bo0CFDhgzJj370o3rX9Morr2ThwoVl7b2wcE0++9nP5r/+67/y5JNPZt68ebnnnnty6aWX5nOf+1zatGmT2traPPXUU/na175Wdk577bVX+vfvn3HjxuWdd95Z7/pOPvnkNG/ePGeffXaOOuqo1b66fdVVV2X77bfPgQcemNGjR+ePf/xj/vrXv+bOO+/Mo48+mqZNmyZ599qPGTMms2bNyqGHHppf/epXmTNnTv785z/n5ptvzksvvVTquykSMgIAAABsRk477bSsWLEiP/7xj3P33XfnS1/60mp9mjRpkhNPPLFejz8PHTr0Az/asia9e/fOdtttV9be+/jKmvTp0yfjxo3LUUcdlT322CPnnHNO+vTpk4kTJyZ5dxVjjx49svvuu6829sQTT8yLL76Ye+65Z73r23LLLXPKKafkH//4R+mDL+/Xrl27PP744zn11FNz7bXX5sADD8zee++dESNGpF+/frnllltKfQ8++OA8+eST6d69ewYPHpwePXrkkEMOyfjx4/ODH/wg3/jGN9a7ro+airr1+VRQI1u6dGkqKyuzZMmStGnTprHLAQAAAD4m3nzzzcydOzc777xzWrVq1djlQKNY19/B+uZyVjICAAAAAIUIGQEAAABocIMGDUrr1q3X2AYNGtTY5dHANu1vYwMAAADwkXTFFVdk6NCha9zndXgfP0JGAAAAABpcVVVVqqqqGrsMNhKPSwMAAAAAhQgZAQAAAIBChIwAAAAAQCFCRgAAAACgECEjAAAAAFCIkBEAAAAAKKRZYxcAAAAA8FF0fb9jN+rxLpzw6416PGhIVjICAAAAbGIqKirW2UaMGJF58+aVbdtmm21y+OGHZ9q0aWuc8+yzz07Tpk0zadKk1faNGDEi++67b9nvFRUVGTRoUFm/2traVFRUZN68eR94Dv9a33ttwIAB6xx3yy23ZJ999knr1q3Ttm3b7Lfffrn66qvX2Hf33XdPy5Yts3DhwtX2ffazn13j8d95553svffeq53be372s5+lZcuWefnll5MkdXV1ueWWW9KrV6+0adMmrVu3zp577pnzzjsvzz77bNnYpUuX5j/+4z+y5557Zosttki7du3yqU99Kv/5n/+Zf/zjHx94zT6o7n/d36pVq+y22265+uqrU1dXt17zbyghIwAAAMAm5oUXXii1G2+8MW3atCnbNnTo0FLfBx54IC+88EKmTp2aTp065dhjj82iRYvK5lu+fHlqampy0UUXZfTo0etVQ6tWrVJdXZ05c+YUOpf36nuvjRw5cq19R48enfPPPz/nnntuamtr88gjj+Siiy7K66+/vlrfhx9+OG+88UZOOumkjBs3bo3znXXWWWXHfuGFF9KsWbMMHDgwNTU1eeONN1YbM2bMmBx//PFp37596urq8uUvfznnnntujj766Nx3333585//nOrq6rRq1SpXXnlladzixYtz8MEHZ8yYMRk6dGgee+yxPPXUU/ne976XGTNm5Pbbb1/va7a2uv91/6xZs3LJJZdk+PDhufnmm9d7/g3hcWkAAACATUzHjh1LP1dWVqaioqJsW5LSSrt27dqlY8eO6dixYy699NLU1NTksccey/HHH1/qO2nSpPTo0SPDhg1Lp06dsmDBgnTu3HmdNXTv3j1VVVW57LLLMnHixA0+l/fqWx+/+tWv0rdv3wwcOLC0bc8991xj3+rq6nz5y1/O4YcfnvPOOy8XX3zxan223HLLNR57wIABufjiizN58uSylZVz587NQw89lHvuuSdJMmHChNTU1OSXv/xl2fXccccdc/DBB5etHrz00kszf/78zJ49O506dSpt32mnnXLUUUfVa6Xh2upe0/4zzjgjP/nJT3L//ffnG9/4xnofo76sZAQAAADYDLzxxhu59dZbkyQtWrQo21ddXZ0BAwaksrIyX/jCFzJ27Nj1mvOaa67J5MmT88QTTzR0uWvUsWPHTJ8+PX/729/W2e+1117LpEmTMmDAgBx55JFZsmTJWh8TX5P27dvni1/84mqrOseOHZsddtghRx11VJJk/Pjx6d69e1nA+H4VFRVJklWrVmXChAkZMGBAWcC4pr4Nqa6uLtOmTctf/vKX1f6ZNzQhIwAAAMDH2CGHHJLWrVvnE5/4RK677rr07NkzRxxxRGn/nDlzMn369PTr1y/Ju6v4xowZs14r6/bff//07dt3jasE61vfe23GjBlr7Xv55Zenbdu26dKlS7p3757TTz89EydOzKpVq8r61dTUpFu3btlzzz3TtGnTnHLKKamurl5tvlGjRpUd+8ILLyztGzhwYB566KHMnTs3ybuB3bhx43LaaaelSZN3I7XZs2ene/fuZXOef/75pfl22GGHJMlLL72UV199dbW+PXv2LPXt37//el+zddX9/v0tW7bMZz7zmaxatSrnnnvues+/IYSMAAAAAB9jEyZMyIwZMzJ58uR07do1Y8eOTfPmzUv7R48enT59+qR9+/ZJkqOPPjpLlizJgw8+uF7zX3nllZk2bVruu+++Da6vtra21Hr06JHk3ceg3wvRvvCFLyRJtttuuzz66KN5+umnc9555+Wdd97Jaaedln/7t38rCxpHjx5d9pjzgAEDMmnSpLz22mtlx/7KV75SduxLLrmktO/II4/MDjvskDFjxiRJpkyZkvnz5+eMM85Y5/lcdtllqa2tzfDhw9f4rsj3u/POO1NbW5s+ffqs8f2Pa7Ouut+//5FHHskXvvCFXHbZZTnkkEPWe/4N4Z2MAAAAAB9jnTt3Trdu3dKtW7e88847OfHEE/PMM8+kZcuWWblyZcaNG5eFCxeWfThk5cqVGT16dNmKx7XZddddc9ZZZ2XYsGFrXC24PvV17dp1te333HNPVqxYkSTZYostyvbttdde2WuvvfLNb34zgwYNymGHHZbf//73+dznPpc///nPmT59eh5//PGyFZYrV65MTU1NzjrrrNK2ysrKNR47SZo0aZLTTz8948aNy4gRIzJmzJh87nOfyy677FLq061bt8yaNats3Lbbbpttt902VVVVZdvatm27Wt8dd9wxSbLVVlvl1VdfXddlKrOuuv91/8SJE9O1a9ccfPDB6d2793ofo76sZAQAAADYTJx00klp1qxZRo0aleTdIO+1117LjBkzylbGjR8/Pnfcccd6B1/Dhw/P7NmzU1NT02C17rTTTunatWu6du2a7bfffq393lv5uGzZsiTvvl/yM5/5TP73f/+37JyGDBlS7xD0jDPOyIIFC3LHHXfkzjvvLPvgTJL0798/s2bNyi9/+ct1ztOkSZP07ds3P//5z/P3v/+9XjUU1bp165x33nkZOnRovT4uU19CRgAAAIDNREVFRc4999xcc801Wb58eaqrq3PMMcdkn332Ka0O3GuvvdK3b9+0bds2t91223rN26FDhwwZMiQ/+tGPPtT6v/GNb+S73/1uHnnkkfztb3/L9OnTc+qpp2bbbbdNr169smLFivzsZz9L//79y85nr732yte+9rU89thj+dOf/rTex9t5553z+c9/Pl//+tfTsmXL/Pu//3vZ/lNOOSUnnXRSTjnllFxxxRV57LHHMm/evPz+97/PhAkT0rRp01Lfq666Kttvv30OPPDAjB49On/84x/z17/+NXfeeWceffTRsr4N7eyzz87s2bMzefLkD+0YHpcGAAAAWIMLJ/y6sUv4UJx22mm57LLL8uMf/zh33313br/99tX6NGnSJCeeeGKqq6szePDg9Zp36NChuemmm/Lmm282dMklvXv3zujRo3PTTTfllVdeSfv27dOrV69MmTIl7dq1y+TJk/PKK6/kxBNPXG3sHnvskT322CPV1dW54YYb1vuYAwcOzJQpU/LNb34zrVq1KttXUVGRCRMm5JZbbsmYMWPyn//5n1mxYkV22GGHHHHEEWXHadeuXR5//PF8//vfz7XXXpu5c+emSZMm6datW/r165fzzz9/g6/LB9lmm21y6qmnZsSIEfn3f//30odrGlJF3Ye5TrKBLF26NJWVlVmyZEnatGnT2OUAAAAAHxNvvvlm5s6dm5133nm1AAk2F+v6O1jfXM7j0gAAAABAIUJGAAAAABrcoEGD0rp16zW2QYMGNXZ5H0nTpk1b6zVr3bp1Y5e3Tt7JCAAAAECDu+KKKzJ06NA17vM6vDU74IADUltb29hlbBAhIwAAAAANrqqqKlVVVY1dxiZliy22SNeuXRu7jA3icWkAAABgs7cJfBcXPjQNcf8LGQEAAIDNVvPmzZMky5cvb+RKoPG8d/+/9/ewITwuDQAAAGy2mjZtmrZt2+bFF19Mkmy55ZapqKho5Kpg46irq8vy5cvz4osvpm3btmnatOkGzyVkBAAAADZrHTt2TJJS0Aibm7Zt25b+DjaUkBEAAADYrFVUVGS77bZLVVVVVqxY0djlwEbVvHnzQisY3yNkBAAAAMi7j043RNgCmyMffgEAAAAAChEyAgAAAACFCBkBAAAAgEKEjAAAAABAIUJGAAAAAKAQISMAAAAAUIiQEQAAAAAoRMgIAAAAABQiZAQAAAAACtmgkHHkyJHp0qVLWrVqlYMOOiiPP/74OvvfeOON6d69e7bYYot07tw5F1xwQd58880NKhgAAAAA+Gipd8g4YcKEDBkyJJdffnmeeuqp7LPPPunTp09efPHFNfa//fbbM2zYsFx++eWZOXNmqqurM2HChFx66aWFiwcAAAAAGl+9Q8YbbrghZ511Vs4444z06NEjN998c7bccsuMHj16jf3/8Ic/5NBDD82Xv/zldOnSJUcddVT69+//gasfAQAAAIBNQ71CxrfffjtPPvlkevfu/c8JmjRJ79698+ijj65xzCGHHJInn3yyFCo+99xzueeee3L00Uev9ThvvfVWli5dWtYAAAAAgI+mZvXp/PLLL2flypXp0KFD2fYOHTrkL3/5yxrHfPnLX87LL7+cT3/606mrq8s777yTQYMGrfNx6auvvjrf+c536lMaAAAAANBIPvSvSz/00EO56qqrMmrUqDz11FO54447cvfdd+e73/3uWsdccsklWbJk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V67MPffck//4j//I9ddfnyuvvHKtx7n66qtTWVlZap07d65PmQAAAADARvShf1161apVqaqqyn//93+nZ8+e6devXy677LLcfPPNax1zySWXZMmSJaW2YMGCD7tMAAAAAGAD1evDL+3bt0/Tpk2zaNGisu2LFi1Kx44d1zhmu+22S/PmzdO0adPStj322CMLFy7M22+/nRYtWqw2pmXLlmnZsmV9SgMAAAAAGkm9VjK2aNEiPXv2zJQpU0rbVq1alSlTpqRXr15rHHPooYfm2WefzapVq0rbZs+ene22226NASMAAAAAsGmp9+PSQ4YMyS233JJx48Zl5syZ+cY3vpFly5bljDPOSJKceuqpueSSS0r9v/GNb2Tx4sU577zzMnv27Nx999256qqrMnjw4IY7CwAAAACg0dTrcekk6devX1566aUMHz48CxcuzL777pt777239DGY+fPnp0mTf2aXnTt3zm9/+9tccMEF+eQnP5ntt98+5513Xi6++OKGOwsAAAAAoNFU1NXV1TV2ER9k6dKlqayszJIlS9KmTZvGLgcAAAAANgvrm8t96F+XBgAAAAA+3oSMAAAAAEAhQkYAAAAAoBAhIwAAAABQiJARAAAAAChEyAgAAAAAFCJkBAAAAAAKETICAAAAAIUIGQEAAACAQoSMAAAAAEAhQkYAAAAAoBAhIwAAAABQiJARAAAAAChEyAgAAAAAFCJkBAAAAAAKETICAAAAAIUIGQEAAACAQoSMAAAAAEAhQkYAAAAAoBAhIwAAAABQiJARAAAAAChEyAgAAAAAFCJkBAAAAAAKETICAAAAAIUIGQEAAACAQoSMAAAAAEAhQkYAAAAAoBAhIwAAAABQiJARAAAAAChEyAgAAAAAFCJkBAAAAAAKETICAAAAAIUIGQEAAACAQoSMAAAAAEAhQkYAAAAAoBAhIwAAAABQiJARAAAAAChEyAgAAAAAFCJkBAAAAAAKETICAAAAAIUIGQEAAACAQoSMAAAAAEAhQkYAAAAAoBAhIwAAAABQiJARAAAAAChEyAgAAAAAFCJkBAAAAAAKETICAAAAAIUIGQEAAACAQoSMAAAAAEAhQkYAAAAAoBAhIwAAAABQiJARAAAAAChEyAgAAAAAFCJkBAAAAAAKETICAAAAAIUIGQEAAACAQoSMAAAAAEAhQkYAAAAAoBAhIwAA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uMYxHTt2rFf/JGnZsmVatmxZtq1t27b1KZWPuDZt2vgXDYW5j2gI7iMainuJhuA+oiG4j2gI7iMagvvo46OysvID+9Trwy8tWrRIz549M2XKlNK2VatWZcqUKenVq9cax/Tq1ausf5Lcf//9a+0PAAAAAGxa6v249JAhQ3LaaaflgAMOyIEHHpgbb7wxy5YtyxlnnJEkOfXUU7P99tvn6quvTpKcd955Ofzww3P99dfnmGOOSU1NTZ544on893//d8OeCQAAAADQKOodMvbr1y8vvfRShg8fnoULF2bffffNvffemw4dOiRJ5s+fnyZN/rlA8pBDDsntt9+eb3/727n00kvTrVu33HXXXdlrr70a7izYZLRs2TKXX375ao/DQ324j2gI7iMainuJhuA+oiG4j2gI7iMagvto81RRV/dB358GAAAAAFi7er2TEQAAAADgXwkZAQAAAIBChIwAAAAAQCFCRgAAAACgECEjG9XIkSPTpUuXtGrVKgcddFAef/zxxi6JTcjUqVNz3HHHpVOnTqmoqMhdd93V2CWxCbr66qvzqU99KltttVWqqqpywgknZNasWY1dFpuYm266KZ/85CfTpk2btGnTJr169cpvfvObxi6LTdw111yTioqKnH/++Y1dCpuYESNGpKKioqztvvvujV0Wm6Dnn38+AwYMSLt27bLFFltk7733zhNPPNHYZbEJ6dKly2r/PqqoqMjgwYMbuzQ2AiEjG82ECRMyZMiQXH755Xnqqaeyzz77pE+fPnnxxRcbuzQ2EcuWLcs+++yTkSNHNnYpbMJ+//vfZ/DgwZk+fXruv//+rFixIkcddVSWLVvW2KWxCdlhhx1yzTXX5Mknn8wTTzyRz3/+8/niF7+YP/3pT41dGpuo//mf/8l//dd/5ZOf/GRjl8Imas8998wLL7xQag8//HBjl8Qm5h//+EcOPfTQNG/ePL/5zW/y5z//Oddff3223nrrxi6NTcj//M//lP276P7770+SnHzyyY1cGRtDRV1dXV1jF8Hm4aCDDsqnPvWp/OQnP0mSrFq1Kp07d84555yTYcOGNXJ1bGoqKipy55135oQTTmjsUtjEvfTSS6mqqsrvf//7fOYzn2nsctiEbbPNNrn22mszcODAxi6FTczrr7+e/fffP6NGjcqVV16ZfffdNzfeeGNjl8UmZMSIEbnrrrtSW1vb2KWwCRs2bFgeeeSRTJs2rbFL4WPk/PPPz69//evMmTMnFRUVjV0OHzIrGdko3n777Tz55JPp3bt3aVuTJk3Su3fvPProo41YGbC5W7JkSZJ3AyLYECtXrkxNTU2WLVuWXr16NXY5bIIGDx6cY445puy/k6C+5syZk06dOmWXXXbJV77ylcyfP7+xS2IT86tf/SoHHHBATj755FRVVWW//fbLLbfc0thlsQl7++238/Of/zxnnnmmgHEzIWRko3j55ZezcuXKdOjQoWx7hw4dsnDhwkaqCtjcrVq1Kueff34OPfTQ7LXXXo1dDpuYp59+Oq1bt07Lli0zaNCg3HnnnenRo0djl8UmpqamJk899VSuvvrqxi6FTdhBBx2UsWPH5t57781NN92UuXPn5rDDDstrr73W2KWxCXnuuedy003/v707CGnyj+M4/vmzMZIYC2uiEhsDU7yY6SwwD4Z0GNLVGAMfEy+yoTS6eCnoMK/qxRjE8uIhBAs6ONqs3UQKBnopiqjEmiEozYOHtm6CxB/898B+PH/fL3hg+53esNv32fN95nXp0iVls1mNj49rYmJCCwsLptPgUM+ePdPe3p5GRkZMp6BG3KYDAAAwJR6Pa3Nzk71V+CttbW0qFova39/X0tKSLMtSoVBg0IgT+/r1qyYnJ/Xy5UudOXPGdA4cLBKJHH3u6OjQtWvXFAwG9fTpU1Y44MQqlYrC4bBSqZQk6cqVK9rc3NSjR49kWZbhOjjR48ePFYlE1NzcbDoFNcI/GVETFy5ckMvlUqlUOnZeKpXU2NhoqArAaZZIJPTixQu9evVKFy9eNJ0DB/J4PGppaVF3d7emp6d1+fJlzc7Oms6Cg7x9+1Y7Ozvq6uqS2+2W2+1WoVDQ3Nyc3G63fv36ZToRDnXu3Dm1trbqw4cPplPgIE1NTX/cKGtvb+fRe/yVz58/K5fLaWxszHQKaoghI2rC4/Gou7tb+Xz+6KxSqSifz7O/CkBNVatVJRIJLS8va3V1VaFQyHQS/icqlYoODw9NZ8BBBgYGtLGxoWKxeHSFw2HFYjEVi0W5XC7TiXCocrmsjx8/qqmpyXQKHOT69et69+7dsbP3798rGAwaKoKTZTIZNTQ0aHBw0HQKaojHpVEzyWRSlmUpHA7r6tWrmpmZ0cHBge7cuWM6DQ5RLpeP3ZH/9OmTisWi6uvrFQgEDJbBSeLxuBYXF/X8+XN5vd6jvbA+n091dXWG6+AUU1NTikQiCgQC+vnzpxYXF/X69Wtls1nTaXAQr9f7xz7Ys2fP6vz58+yJxX9y79493bp1S8FgUNvb23rw4IFcLpei0ajpNDjI3bt31dvbq1QqpaGhIa2vryudTiudTptOg8NUKhVlMhlZliW3m7HTacKvjZq5ffu2fvz4ofv37+v79+/q7OzUysrKHy+DAf7NmzdvdOPGjaPvyWRSkmRZlp48eWKoCk4zPz8vServ7z92nslkWEqNE9vZ2dHw8LC+ffsmn8+njo4OZbNZ3bx503QagFNoa2tL0WhUu7u78vv96uvr09ramvx+v+k0OEhPT4+Wl5c1NTWlhw8fKhQKaWZmRrFYzHQaHCaXy+nLly8aHR01nYIa+6darVZNRwAAAAAAAABwLnYyAgAAAAAAALCFISMAAAAAAAAAWxgyAgAAAAAAALCFISMAAAAAAAAAWxgyAgAAAAAAALCFISMAAAAAAAAAWxgyAgAAAAAAALCFISMAAAAAAAAAWxgyAgAAAAAAALCFISMAAAAAAAAAWxgyAgAAAAAAALCFISMAAAAAAAAAW34DlxaGgq5zNQcAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 1618x1000 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>model</th>\n",
       "      <th>num</th>\n",
       "      <th>Test ACC</th>\n",
       "      <th>Test Loss</th>\n",
       "      <th>FR layer 01</th>\n",
       "      <th>FR layer 02</th>\n",
       "      <th>FR layer 03</th>\n",
       "      <th>FR layer 04</th>\n",
       "      <th>FR layer 05</th>\n",
       "      <th>FR layer 06</th>\n",
       "      <th>FR layer 07</th>\n",
       "      <th>FR layer 08</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>TRAIN_E-OTTTVGG</td>\n",
       "      <td>01</td>\n",
       "      <td>0.9346</td>\n",
       "      <td>0.343495</td>\n",
       "      <td>0.389482</td>\n",
       "      <td>0.202409</td>\n",
       "      <td>0.152466</td>\n",
       "      <td>0.144464</td>\n",
       "      <td>0.125427</td>\n",
       "      <td>0.096682</td>\n",
       "      <td>0.091834</td>\n",
       "      <td>0.026525</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>TRAIN_A-OTTTVGG</td>\n",
       "      <td>01</td>\n",
       "      <td>0.9333</td>\n",
       "      <td>0.347952</td>\n",
       "      <td>0.404665</td>\n",
       "      <td>0.219787</td>\n",
       "      <td>0.161158</td>\n",
       "      <td>0.128598</td>\n",
       "      <td>0.121864</td>\n",
       "      <td>0.083828</td>\n",
       "      <td>0.097096</td>\n",
       "      <td>0.020313</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>TRAIN_E-SAFVGG</td>\n",
       "      <td>01</td>\n",
       "      <td>0.9356</td>\n",
       "      <td>0.340172</td>\n",
       "      <td>0.360862</td>\n",
       "      <td>0.189482</td>\n",
       "      <td>0.154232</td>\n",
       "      <td>0.133830</td>\n",
       "      <td>0.121846</td>\n",
       "      <td>0.096202</td>\n",
       "      <td>0.081688</td>\n",
       "      <td>0.030561</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>TRAIN_F-SAFVGG_FR</td>\n",
       "      <td>01</td>\n",
       "      <td>0.9308</td>\n",
       "      <td>0.368815</td>\n",
       "      <td>0.185963</td>\n",
       "      <td>0.208857</td>\n",
       "      <td>0.136231</td>\n",
       "      <td>0.106258</td>\n",
       "      <td>0.083702</td>\n",
       "      <td>0.060907</td>\n",
       "      <td>0.061718</td>\n",
       "      <td>0.020019</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>TRAIN_E-FOTTTVGG</td>\n",
       "      <td>01</td>\n",
       "      <td>0.9301</td>\n",
       "      <td>0.411566</td>\n",
       "      <td>0.381306</td>\n",
       "      <td>0.190010</td>\n",
       "      <td>0.164387</td>\n",
       "      <td>0.116202</td>\n",
       "      <td>0.111823</td>\n",
       "      <td>0.090724</td>\n",
       "      <td>0.104366</td>\n",
       "      <td>0.018215</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>TRAIN_E-FSAFVGG</td>\n",
       "      <td>01</td>\n",
       "      <td>0.9325</td>\n",
       "      <td>0.409269</td>\n",
       "      <td>0.394179</td>\n",
       "      <td>0.183170</td>\n",
       "      <td>0.147951</td>\n",
       "      <td>0.112670</td>\n",
       "      <td>0.108876</td>\n",
       "      <td>0.092394</td>\n",
       "      <td>0.101724</td>\n",
       "      <td>0.021112</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "               model num  Test ACC  Test Loss  FR layer 01  FR layer 02  \\\n",
       "0    TRAIN_E-OTTTVGG  01    0.9346   0.343495     0.389482     0.202409   \n",
       "1    TRAIN_A-OTTTVGG  01    0.9333   0.347952     0.404665     0.219787   \n",
       "2     TRAIN_E-SAFVGG  01    0.9356   0.340172     0.360862     0.189482   \n",
       "3  TRAIN_F-SAFVGG_FR  01    0.9308   0.368815     0.185963     0.208857   \n",
       "4   TRAIN_E-FOTTTVGG  01    0.9301   0.411566     0.381306     0.190010   \n",
       "5    TRAIN_E-FSAFVGG  01    0.9325   0.409269     0.394179     0.183170   \n",
       "\n",
       "   FR layer 03  FR layer 04  FR layer 05  FR layer 06  FR layer 07  \\\n",
       "0     0.152466     0.144464     0.125427     0.096682     0.091834   \n",
       "1     0.161158     0.128598     0.121864     0.083828     0.097096   \n",
       "2     0.154232     0.133830     0.121846     0.096202     0.081688   \n",
       "3     0.136231     0.106258     0.083702     0.060907     0.061718   \n",
       "4     0.164387     0.116202     0.111823     0.090724     0.104366   \n",
       "5     0.147951     0.112670     0.108876     0.092394     0.101724   \n",
       "\n",
       "   FR layer 08  \n",
       "0     0.026525  \n",
       "1     0.020313  \n",
       "2     0.030561  \n",
       "3     0.020019  \n",
       "4     0.018215  \n",
       "5     0.021112  "
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "criterion = CombineCEandMSE(loss_alpha, num_classes)\n",
    "pth = \"epoch_0300.pth\"\n",
    "nums = [str(x + 1).zfill(2) for x in range(N)]\n",
    "path_list = [\n",
    "    \"./logs/TRAIN_E-OTTTVGG\",\n",
    "    \"./logs/TRAIN_A-OTTTVGG\",\n",
    "    \"./logs/TRAIN_E-SAFVGG\",\n",
    "    \"./logs/TRAIN_F-SAFVGG_FR\",\n",
    "    \"./logs/TRAIN_E-FOTTTVGG\",\n",
    "    \"./logs/TRAIN_E-FSAFVGG\",\n",
    "]\n",
    "structure_list = [\n",
    "    OTTTVGG,\n",
    "    OTTTVGG,\n",
    "    SAFVGG,\n",
    "    SAFVGG_FR,\n",
    "    FOTTTVGG,\n",
    "    FSAFVGG,\n",
    "]\n",
    "logs = []\n",
    "for path, structure in zip(path_list, structure_list):\n",
    "    # creat base (SAF or OTTT) model\n",
    "    base_model = structure(lif_lambda)\n",
    "    name = path.split(\"/\")[-1]\n",
    "    for num in nums:\n",
    "        print(name, num)\n",
    "        # load state to base model\n",
    "        model_path = os.path.join(path, num, pth)\n",
    "        base_model.load_state_dict(torch.load(model_path)[\"state_dict\"])\n",
    "        # base model state to LIF model\n",
    "        if structure.__name__[0] == \"F\":\n",
    "            model = convert(base_model, lif_lambda, FVGG).cuda()\n",
    "        else:\n",
    "            model = convert(base_model, lif_lambda, VGG).cuda()\n",
    "        loss, acc, fr = test(test_loader, model, criterion, t_step, 300)\n",
    "        logs.append([name, num, acc, loss, *fr])\n",
    "        del model\n",
    "        torch.cuda.empty_cache()\n",
    "    del base_model\n",
    "    torch.cuda.empty_cache()\n",
    "layer_num = [\"FR layer \" + str(x + 1).zfill(2) for x in range(len(fr))]\n",
    "columns = [\"model\", \"num\", \"Test ACC\", \"Test Loss\", *layer_num]\n",
    "df = pd.DataFrame(logs, columns=columns)\n",
    "models = sorted(list(set(df[\"model\"])))\n",
    "fr_list = []\n",
    "for model in models:\n",
    "    a = df[df[\"model\"] == model]\n",
    "    fr = np.mean(np.array(a)[:, 4:], axis=0)\n",
    "    fr_list.append(fr)\n",
    "plt.figure(figsize=(16.18, 10))\n",
    "n = len(fr_list)\n",
    "width = 0.8 / n\n",
    "sp = -0.4 + 0.4 / n\n",
    "for i in range(n):\n",
    "    fr = fr_list[i]\n",
    "    label = models[i]\n",
    "    plt.bar(np.arange(len(fr)) + sp + width * i, fr, width=width, label=label)\n",
    "plt.ylim(0.0, 1.0)\n",
    "plt.legend()\n",
    "plt.show()\n",
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "tprolibs:g5-4xlarge",
   "language": "python",
   "name": "391820500782.dkr.ecr.ap-northeast-1.amazonaws.com_aisw_tpro_g5-4xlarge"
  },
  "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.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
