{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "6d0d9db2",
   "metadata": {},
   "source": [
    "# Self-supervised denoiser impulse response\n",
    "\n",
    "**Authors: Nick Luiken, Matteo Ravasi - KAUST**"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a0969d84",
   "metadata": {},
   "source": [
    "In this notebook, we display the impulse response (aka receptive field) of both the original blind-spot denoiser of [Laine et al.](https://arxiv.org/abs/1901.10277) and our modified version for structured noise.\n",
    "\n",
    "In order to obtain the impulse response we feed an zero image with a single unitary pixel in the middle in the blind-spot networks with constant weights and zero biases."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "e72a617c",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "%matplotlib inline\n",
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "\n",
    "from ssdeblend.model.ssdenoise import NoiseNetwork\n",
    "from ssdeblend.model.ssstructdenoise import NoiseNetwork as StructNoiseNetwork\n",
    "from ssdeblend.model.utils import display_receptive"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4b335354",
   "metadata": {},
   "source": [
    "Let's set the device to use our GPU of choice"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "6ddc72a6",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "NVIDIA GeForce RTX 3090\n"
     ]
    }
   ],
   "source": [
    "device = 'cuda:0'\n",
    "print(torch.cuda.get_device_name(device))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4829dc9d",
   "metadata": {},
   "source": [
    "## Blind-spot network for random noise"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3205905c",
   "metadata": {},
   "source": [
    "Let's first create the network"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ca07a528",
   "metadata": {
    "scrolled": false
   },
   "outputs": [],
   "source": [
    "network = NoiseNetwork(1, 1, blindspot=True).to(device)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b3adca93",
   "metadata": {},
   "source": [
    "And now we can display its receptive field"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "efce4dfb",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "n=32\n",
    "network_rec, out = display_receptive(network, n, weight_fill=1e-3, device=device)\n",
    "\n",
    "plt.figure()\n",
    "plt.imshow(out.detach().cpu().squeeze(), cmap='Greens', vmin=0, vmax=out.detach().cpu().max())\n",
    "plt.scatter(n//2, n//2, c='#FFA500', s=50, marker='s')\n",
    "plt.xlim(n//2-8,n//2+8)\n",
    "plt.ylim(n//2+8,n//2-8)\n",
    "plt.axis('off');"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2360a4d7",
   "metadata": {},
   "source": [
    "## Blind-spot network for coherent noise"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7276a8c8",
   "metadata": {},
   "source": [
    "We can now do the same for our new network"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "e6222ed9",
   "metadata": {},
   "outputs": [],
   "source": [
    "network = StructNoiseNetwork(1, 1, blindspot=True).to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "1264100e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAOcAAADnCAYAAADl9EEgAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAAsTAAALEwEAmpwYAAAFT0lEQVR4nO3dQWsdVRzG4RkTmpA22rSCSKhJBAWLrfYjdNO1xaX5eLqUutaFrhTdVKj7pGpaXARCq5e0GMdlKCTB+Wc69536PNvL6Zkh/Dhw7zk9bdd1DZDntXk/AHAycUIocUIocUIocUKoxbM+PDyaDfpV7tjfDHdN//kqY5rmPO+W/m15O+Kopmnb8daLtvpubfXtTra8sHLiP2jlhFDihFDihFDihFDihFDihFBn/pQytOpX0GP+BFP9er3820H8LynVn1KG/bnhZcw19E8iQ7NyQihxQihxQihxQihxQihxQihxQihxQihxQihxQihxQihxQqhRN76PrbIhuvp/CNU3zE/gGUc0hWcci5UTQokTQokTQokTQokTQokTQokTQokTQokTQokTQokTQokTQokTQk3iVMqY1zhUT0UcPN8vjXs0+733mJ0nu6W5Nlc3eo9Zv3itNNflC1dL4yrSr1WosnJCKHFCKHFCKHFCKHFCKHFCKHFCKHFCKHFCKHFCKHFCKHFCKHFCqEmcSqka87TC49leadw3D7/rPebeg/ulue7euNV7zJ2N26W51pbeLI3jmJUTQokTQokTQokTQokTQokTQokTQokTQokTQokTQokTQokTQr3SG9/HtPv0YWncV7/83HvMT1/8UJqr3e5/EOD9K++W5rq+9lFpHMesnBBKnBBKnBBKnBBKnBBKnBBKnBBKnBBKnBBKnBBKnBBKnBBKnBDKqZSBbK1ulsZVrkhotktTleaqvhfnZ+WEUOKEUOKEUOKEUOKEUOKEUOKEUOKEUOKEUOKEUOKEUOKEUOKEUG3Xdad+eHg0O/1DXnDwbL807vFsr/eYnae7pbkqJ0zeXlkvzXV56Wpp3P/R8sLKiZfYWDkhlDghlDghlDghlDghlDghlDghlDghlDghlDghlDghlDghlI3vMGc2vsPEiBNCuWVs4pa+fKtp//7z1M+7xUvNs0//GPGJGIqVc+LOCvO/fE4ucUIocUIocUIocUIocUIocUIocU5ct3jpXJ+Ty95amDN7a2FibN8biOsYXuQ6hvOzckIocUIocUIocUIocUIocUIocUIocUIocUIocUIocUIocUIoG98HUtnA3jRN8/Wv3/Yec+/B/dJcd2/c6j3mzju3S3PZ+H5+Vk4IJU4IJU4IJU4IJU4IJU4IJU4IJU4IJU4IJU4IJU4IJU4IJU4I5VTKQKpXJFROmPz4+feluZrt/kPeW9sqTfXB2s3SOI5ZOSGUOCGUOCGUOCGUOCGUOCGUOCGUOCGUOCGUOCGUOCGUOCGUOCGUUykD2VzdKI375MOPe4/pPutGm6v6Xl1Xe8a2bUvjXkVWTgglTgglTgglTgglTgglTgglTgglTgglTgglTgglTgglTgjVnrVB+fBoVtu9HKK6+bri4Pl+adzeX7/1HrPzZKc019br/a9WWL94rTTXGxeulMa1Tf+N71PfLL+8sHLiC1g5IZQ4IZQ4IZQ4IZQ4IZQ4IZQ4IZQ4IZQ4IZQ4IZQ4IZQ4IZQ4IdQkTqWMebqka8Z95bHn66tySmRs1WdMOc3iVApMjDghlDghlDghlDghlDghlDghlDghlDghlDghlDghlDghlDgh1OK8H+BlGvPER9f9Uxs38HMMr/iExRMflRMm5b9z+dXGOc1i5YRQ4oRQ4oRQ4oRQ4oRQ4oRQ4oRQ4oRQ4oRQ4oRQ4oRQ4oRQo258r16rMOoG9uJc9SfM3vreVa86qF6hEXJFQgIrJ4QSJ4QSJ4QSJ4QSJ4QSJ4Q682ZrYH6snBBKnBBKnBBKnBBKnBBKnBDqX0EBwQg3IlKvAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "n=32\n",
    "network_rec, out = display_receptive(network, n, weight_fill=1e-3, device=device)\n",
    "\n",
    "plt.figure()\n",
    "plt.imshow(out.detach().cpu().squeeze(), cmap='Greens', vmin=0, vmax=out.detach().cpu().max())\n",
    "plt.scatter(n//2, n//2, c='#FFA500', s=50, marker='s')\n",
    "plt.xlim(n//2-8,n//2+8)\n",
    "plt.ylim(n//2+8,n//2-8)\n",
    "plt.axis('off');"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2561ef93",
   "metadata": {},
   "source": [
    "Nice! By modifying the way we handle the input image (only one flip over the y-axis instead of 4 rotations), our network is now blind not only to the central pixels but to the entire middle row in the x-axis."
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.12"
  },
  "toc": {
   "base_numbering": 1,
   "nav_menu": {},
   "number_sections": true,
   "sideBar": true,
   "skip_h1_title": false,
   "title_cell": "Table of Contents",
   "title_sidebar": "Contents",
   "toc_cell": false,
   "toc_position": {
    "height": "calc(100% - 180px)",
    "left": "10px",
    "top": "150px",
    "width": "613.253px"
   },
   "toc_section_display": true,
   "toc_window_display": true
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
