{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Demonstrating Artifacts in an RL Model"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Pre-trained model for playing Atari games by Sam Greydanus downloaded from https://github.com/greydanus/visualize_atari"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Task and model details in the following paper:\n",
    "Greydanus, et al. Visualizing and understanding atari agents. (ICML 2018) https://arxiv.org/abs/1711.00138"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Loading the Model (all layers use zero padding)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "from torch.autograd import Variable\n",
    "import torch.nn.functional as F\n",
    "import torch.nn as nn\n",
    "\n",
    "import glob\n",
    "import numpy as np\n",
    "\n",
    "class NNPolicy(torch.nn.Module): # an actor-critic neural network\n",
    "    def __init__(self, channels, num_actions):\n",
    "        super(NNPolicy, self).__init__()\n",
    "        self.conv1 = nn.Conv2d(channels, 32, 3, stride=2, padding=1)\n",
    "        self.conv2 = nn.Conv2d(32, 32, 3, stride=2, padding=1)\n",
    "        self.conv3 = nn.Conv2d(32, 32, 3, stride=2, padding=1)\n",
    "        self.conv4 = nn.Conv2d(32, 32, 3, stride=2, padding=1)\n",
    "        self.lstm = nn.LSTMCell(32 * 5 * 5, 256)\n",
    "        self.critic_linear, self.actor_linear = nn.Linear(256, 1), nn.Linear(256, num_actions)\n",
    "\n",
    "    def forward(self, inputs):\n",
    "        inputs = inputs\n",
    "        x = F.elu(self.conv1(inputs))\n",
    "        x = F.elu(self.conv2(x))\n",
    "        x = F.elu(self.conv3(x))\n",
    "        x = F.elu(self.conv4(x))\n",
    "        x = x.view(-1, 32 * 5 * 5)\n",
    "       # hx, cx = self.lstm(x, (hx, cx))\n",
    "        return\n",
    "\n",
    "    def try_load(self, save_dir, checkpoint='*.tar'):\n",
    "        paths = glob.glob(save_dir + checkpoint) ; step = 0\n",
    "        if len(paths) > 0:\n",
    "            ckpts = [int(s.split('.')[-2]) for s in paths]\n",
    "            ix = np.argmax(ckpts) ; step = ckpts[ix]\n",
    "            self.load_state_dict(torch.load(paths[ix]))\n",
    "        print(\"\\tno saved models\") if step is 0 else print(\"\\tloaded model: {}\".format(paths[ix]))\n",
    "        return step\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model = NNPolicy(channels=1, num_actions=6)\n",
    "ckpt = torch.load(\"/work/data/RL/greydanus/overfit-pong-v0/expert/model.40.tar\")\n",
    "model.load_state_dict(ckpt)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "NNPolicy(\n",
       "  (conv1): Conv2d(1, 32, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1))\n",
       "  (conv2): Conv2d(32, 32, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1))\n",
       "  (conv3): Conv2d(32, 32, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1))\n",
       "  (conv4): Conv2d(32, 32, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1))\n",
       "  (lstm): LSTMCell(800, 256)\n",
       "  (critic_linear): Linear(in_features=256, out_features=1, bias=True)\n",
       "  (actor_linear): Linear(in_features=256, out_features=6, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.eval()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Average Feature Maps for 0-valued Input"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "activations = {}\n",
    "def save_activation(name):\n",
    "    def hook(mod, inp, out):\n",
    "        activations[name] = out.detach().numpy()\n",
    "    return hook\n",
    "    \n",
    "for name, mod in model.named_modules():\n",
    "    mod.register_forward_hook(save_activation(name))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "model.forward(torch.zeros([1,1, 64, 64]))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1050x450 with 8 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "num_of_maps = len(activations)\n",
    "fig = plt.figure(figsize=(7,3), dpi=150)\n",
    "plt.subplots_adjust(left=0, bottom=0.1, right=1, top=0.85, wspace=0.1, hspace=0.4)\n",
    "ind = 0\n",
    "for key in activations:\n",
    "    avg_feat_map = np.mean(activations[key], axis=(0, 1))\n",
    "    if len(avg_feat_map.shape) == 2: # fc layer\n",
    "        ax = fig.add_subplot(int(num_of_maps / 4) + 1, 4, ind + 1)\n",
    "        ax.set_title(\"{} - {}\".format(key, activations[key].shape), fontsize=6)\n",
    "        plt.imshow(avg_feat_map)\n",
    "        cbar = plt.colorbar()\n",
    "        cbar.ax.tick_params(labelsize=6) \n",
    "        ax.axis('off')\n",
    "        ax.get_xaxis().set_visible(False)\n",
    "        ax.get_yaxis().set_visible(False)\n",
    "        ind += 1\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Notice the propagation of spatial bias at the top-left corner due to one-sided padding."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Visualizing Average Weight Kernels"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(32, 1, 3, 3) - conv1 - <class 'torch.nn.modules.conv.Conv2d'>\n",
      "(32, 32, 3, 3) - conv2 - <class 'torch.nn.modules.conv.Conv2d'>\n",
      "(32, 32, 3, 3) - conv3 - <class 'torch.nn.modules.conv.Conv2d'>\n",
      "(32, 32, 3, 3) - conv4 - <class 'torch.nn.modules.conv.Conv2d'>\n"
     ]
    }
   ],
   "source": [
    "weight_mats = {}\n",
    "total_sum_3x3 = np.zeros([3, 3])\n",
    "total_sum_1x1 = 0\n",
    "for name, mod in model.named_modules():\n",
    "    if hasattr(mod, \"weight\"):\n",
    "        weight_mat = mod.weight.detach().numpy()\n",
    "        if (len(weight_mat.shape)) == 4:\n",
    "            print(\"{} - {} - {}\".format(weight_mat.shape, name, type(mod)))\n",
    "            weight_mats[name] = np.sum(weight_mat, (0, 1))\n",
    "            if len(weight_mats[name]) == 3:\n",
    "                total_sum_3x3 += weight_mats[name]\n",
    "            else:\n",
    "                if len(weight_mats[name]) == 1:\n",
    "                    total_sum_1x1 += np.sum(weight_mats[name])  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1200x300 with 8 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "fig = plt.figure(figsize=(8, 2), dpi=150)\n",
    "# fig.suptitle(\"classifier out map per class\", fontsize=12)\n",
    "plt.subplots_adjust(left=0, bottom=0.1, right=1, top=0.85, wspace=0.1, hspace=0.2)\n",
    "ind = 0\n",
    "for key in weight_mats:\n",
    "    if len(weight_mats[key])== 1:\n",
    "        continue\n",
    "    ax = fig.add_subplot(len(weight_mats) / 5 + 1, 5, ind + 1)\n",
    "    ax.set_title(\"{}\".format(key), fontsize=6)\n",
    "    \n",
    "    vmin = np.min(weight_mats[key])\n",
    "    vmax = np.max(weight_mats[key])\n",
    "    max_abs = max(np.abs(vmin), np.abs(vmax))\n",
    "\n",
    "    plt.imshow(weight_mats[key],vmin = -max_abs, vmax = max_abs, cmap=plt.get_cmap('PiYG'))\n",
    "    cbar = plt.colorbar()\n",
    "    cbar.ax.tick_params(labelsize=4) \n",
    "    ax.grid(b=False)\n",
    "    ax.get_xaxis().set_visible(False)\n",
    "    ax.get_yaxis().set_visible(False)\n",
    "    ind += 1\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
