{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Supermasks in Superposition: Training and Task Inference\n",
    "\n",
    "<img src=\"images/teaser_supsup.png\">"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Packages loaded\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torch.optim as optim\n",
    "import torch.nn as nn\n",
    "import torch.autograd as autograd\n",
    "import torch.nn.functional as F\n",
    "\n",
    "import torchvision\n",
    "import numpy as np\n",
    "import math\n",
    "\n",
    "from tqdm.notebook import tqdm\n",
    "\n",
    "print(\"Packages loaded\")\n",
    "# note at edits"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 316,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style>.container { width:100% !important; }</style>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from IPython.core.display import display, HTML\n",
    "display(HTML(\"<style>.container { width:100% !important; }</style>\"))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Training\n",
    "\n",
    "We train our model using the procedure from \"What's Hidden in a Randomly Weighted Neural Network\" (Ramanujan et al. 2019). Below is a modification of the layers (using signed constant initialization) from this paper for our continual learning setup. As a deviation from this setup, we assume that an edge (u, v) is included in the subnetwork if its score is greater than 0, rather than use it's absolute magnitude to construct a threshold. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Subnetwork forward from hidden networks\n",
    "class GetSubnet(autograd.Function):\n",
    "    @staticmethod\n",
    "    def forward(ctx, scores):\n",
    "        return (scores >= 0).float()\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, g):\n",
    "        # send the gradient g straight-through on the backward pass.\n",
    "        return g\n",
    "    \n",
    "class GetSubnetSparse(autograd.Function):\n",
    "    @staticmethod\n",
    "    def forward(ctx, scores, k):\n",
    "        # Get the supermask by sorting the scores and using the top k%\n",
    "        out = scores.clone()\n",
    "        _, idx = scores.flatten().sort()\n",
    "        j = int((1 - k) * scores.numel())\n",
    "\n",
    "        # flat_out and out access the same memory.\n",
    "        flat_out = out.flatten()\n",
    "        flat_out[idx[:j]] = 0\n",
    "        flat_out[idx[j:]] = 1\n",
    "\n",
    "        return out\n",
    "\n",
    "    @staticmethod\n",
    "    def backward(ctx, g):\n",
    "        # send the gradient g straight-through on the backward pass.\n",
    "        return g, None\n",
    "\n",
    "def mask_init(module):\n",
    "    scores = torch.Tensor(module.weight.size())\n",
    "    nn.init.kaiming_uniform_(scores, a=math.sqrt(5))\n",
    "    return scores\n",
    "\n",
    "\n",
    "def signed_constant(module):\n",
    "    fan = nn.init._calculate_correct_fan(module.weight, 'fan_in')\n",
    "    gain = nn.init.calculate_gain('relu')\n",
    "    std = gain / math.sqrt(fan)\n",
    "    module.weight.data = module.weight.data.sign() * std\n",
    "\n",
    "\n",
    "class MultitaskMaskLinear(nn.Linear):\n",
    "    def __init__(self, *args, num_tasks=1, **kwargs):\n",
    "        super().__init__(*args, **kwargs)\n",
    "        self.num_tasks = num_tasks\n",
    "        self.scores = nn.ParameterList(\n",
    "            [\n",
    "                nn.Parameter(mask_init(self))\n",
    "                for _ in range(num_tasks)\n",
    "            ]\n",
    "        )\n",
    "        \n",
    "        # Keep weights untrained\n",
    "        self.weight.requires_grad = False\n",
    "        signed_constant(self)\n",
    "    \n",
    "    @torch.no_grad()\n",
    "    def cache_masks(self):\n",
    "        self.register_buffer(\n",
    "            \"stacked\",\n",
    "            torch.stack(\n",
    "                [\n",
    "                    GetSubnet.apply(self.scores[j])\n",
    "                    for j in range(self.num_tasks)\n",
    "                ]\n",
    "            ),\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        if self.task < 0:\n",
    "            # Superimposed forward pass\n",
    "            alpha_weights = self.alphas[: self.num_tasks_learned]\n",
    "            idxs = (alpha_weights > 0).squeeze().view(self.num_tasks_learned)\n",
    "            if len(idxs.shape) == 0:\n",
    "                idxs = idxs.view(1)\n",
    "            subnet = (\n",
    "                alpha_weights[idxs]\n",
    "                * self.stacked[: self.num_tasks_learned][idxs]\n",
    "            ).sum(dim=0)\n",
    "        else:\n",
    "            # Subnet forward pass (given task info in self.task)\n",
    "            subnet = GetSubnet.apply(self.scores[self.task])\n",
    "        w = self.weight * subnet\n",
    "        x = F.linear(x, w, self.bias)\n",
    "        return x\n",
    "\n",
    "\n",
    "    def __repr__(self):\n",
    "        return f\"MultitaskMaskLinear({self.size()})\"\n",
    "    \n",
    "class MultitaskMaskLinearSparse(nn.Linear):\n",
    "    def __init__(self, *args, num_tasks=1, sparsity=0.5, **kwargs):\n",
    "        super().__init__(*args, **kwargs)\n",
    "        self.num_tasks = num_tasks\n",
    "        self.sparsity = sparsity\n",
    "        self.scores = nn.ParameterList(\n",
    "            [\n",
    "                nn.Parameter(mask_init(self))\n",
    "                for _ in range(num_tasks)\n",
    "            ]\n",
    "        )\n",
    "        \n",
    "        # Keep weights untrained\n",
    "        self.weight.requires_grad = False\n",
    "        signed_constant(self)\n",
    "    \n",
    "    @torch.no_grad()\n",
    "    def cache_masks(self):\n",
    "        self.register_buffer(\n",
    "            \"stacked\",\n",
    "            torch.stack(\n",
    "                [\n",
    "                    GetSubnetSparse.apply(self.scores[j], self.sparsity)\n",
    "                    for j in range(self.num_tasks)\n",
    "                ]\n",
    "            ),\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        if self.task < 0:\n",
    "            # Superimposed forward pass\n",
    "            alpha_weights = self.alphas[: self.num_tasks_learned]\n",
    "            idxs = (alpha_weights > 0).squeeze().view(self.num_tasks_learned)\n",
    "            if len(idxs.shape) == 0:\n",
    "                idxs = idxs.view(1)\n",
    "            subnet = (\n",
    "                alpha_weights[idxs]\n",
    "                * self.stacked[: self.num_tasks_learned][idxs]\n",
    "            ).sum(dim=0)\n",
    "        else:\n",
    "            # Subnet forward pass (given task info in self.task)\n",
    "            subnet = GetSubnetSparse.apply(self.scores[self.task], self.sparsity)\n",
    "        w = self.weight * subnet\n",
    "        x = F.linear(x, w, self.bias)\n",
    "        return x\n",
    "\n",
    "\n",
    "    def __repr__(self):\n",
    "        return f\"MultitaskMaskLinearSparse({self.size()})\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BasisMultitaskMaskLinear(nn.Linear):\n",
    "    def __init__(self, *args, num_tasks=1, num_seed_tasks_learned=1, start_at_optimal=True, **kwargs):\n",
    "        super().__init__(*args, **kwargs)\n",
    "        assert num_tasks >= num_seed_tasks_learned, \"Seed tasks cannot be more than total tasks!\"\n",
    "        self.num_tasks = num_tasks\n",
    "        self.num_seed_tasks_learned = num_seed_tasks_learned\n",
    "        self.scores = nn.ParameterList(\n",
    "            [\n",
    "                nn.Parameter(mask_init(self))\n",
    "                for _ in range(num_tasks)\n",
    "            ]\n",
    "        )\n",
    "        self.task = -1\n",
    "        \n",
    "        # Keep weights untrained\n",
    "        self.weight.requires_grad = False\n",
    "        for s in self.scores:\n",
    "            s.requires_grad = False\n",
    "        self.scores.requires_grad = False\n",
    "        if start_at_optimal:\n",
    "            self.basis_alphas = nn.ParameterList(\n",
    "                [\n",
    "                    nn.Parameter(torch.eye(self.num_seed_tasks_learned)[i])\n",
    "                    for i in range(self.num_seed_tasks_learned)\n",
    "                ]\n",
    "                +\n",
    "                [\n",
    "                    nn.Parameter(torch.ones(self.num_seed_tasks_learned)/self.num_seed_tasks_learned)\n",
    "                    for _ in range(self.num_seed_tasks_learned, self.num_tasks)\n",
    "                ]\n",
    "            )\n",
    "        else:\n",
    "            self.basis_alphas = nn.ParameterList(\n",
    "                [\n",
    "                    nn.Parameter(torch.ones(self.num_seed_tasks_learned)/self.num_seed_tasks_learned)\n",
    "                    for _ in range(self.num_tasks)\n",
    "                ]\n",
    "            )\n",
    "        \n",
    "        signed_constant(self)\n",
    "    \n",
    "    @torch.no_grad()\n",
    "    def cache_masks(self):\n",
    "        self.register_buffer(\n",
    "            \"stacked\",\n",
    "            torch.stack(\n",
    "                [\n",
    "                    GetSubnet.apply(self.scores[j])\n",
    "                    for j in range(self.num_tasks)\n",
    "                ]\n",
    "            ),\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        if self.task < 0:\n",
    "            raise NotImplemented(\"Need task identity at inference time.\")\n",
    "        else:\n",
    "            # Subnet forward pass (given task info in self.task)\n",
    "            subnet = self.stacked[: self.num_seed_tasks_learned][0]\n",
    "            task_alpha = self.basis_alphas[self.task]\n",
    "            w = self.weight * subnet * task_alpha[0]\n",
    "            for i in range(1, self.num_seed_tasks_learned):\n",
    "                subnet = self.stacked[: self.num_seed_tasks_learned][i]\n",
    "                w += self.weight * subnet * task_alpha[i]\n",
    "        x = F.linear(x, w, self.bias)\n",
    "        return x\n",
    "\n",
    "\n",
    "    def __repr__(self):\n",
    "        return f\"BasisMultitaskMaskLinear({self.size()})\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 64,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BasisMultitaskMaskLinearSparse(nn.Linear):\n",
    "    def __init__(self, *args, num_tasks=1, num_seed_tasks_learned=1, start_at_optimal=True, sparsity=0.5, **kwargs):\n",
    "        super().__init__(*args, **kwargs)\n",
    "        assert num_tasks >= num_seed_tasks_learned, \"Seed tasks cannot be more than total tasks!\"\n",
    "        self.num_tasks = num_tasks\n",
    "        self.num_seed_tasks_learned = num_seed_tasks_learned\n",
    "        self.sparsity = sparsity\n",
    "        self.scores = nn.ParameterList(\n",
    "            [\n",
    "                nn.Parameter(mask_init(self))\n",
    "                for _ in range(num_tasks)\n",
    "            ]\n",
    "        )\n",
    "        self.task = -1\n",
    "        \n",
    "        # Keep weights untrained\n",
    "        self.weight.requires_grad = False\n",
    "        for s in self.scores:\n",
    "            s.requires_grad = False\n",
    "        self.scores.requires_grad = False\n",
    "        if start_at_optimal:\n",
    "            self.basis_alphas = nn.ParameterList(\n",
    "                [\n",
    "                    nn.Parameter(torch.eye(self.num_seed_tasks_learned)[i])\n",
    "                    for i in range(self.num_seed_tasks_learned)\n",
    "                ]\n",
    "                +\n",
    "                [\n",
    "                    nn.Parameter(torch.ones(self.num_seed_tasks_learned)/self.num_seed_tasks_learned)\n",
    "                    for _ in range(self.num_seed_tasks_learned, self.num_tasks)\n",
    "                ]\n",
    "            )\n",
    "        else:\n",
    "            self.basis_alphas = nn.ParameterList(\n",
    "                [\n",
    "                    nn.Parameter(torch.ones(self.num_seed_tasks_learned)/self.num_seed_tasks_learned)\n",
    "                    for _ in range(self.num_tasks)\n",
    "                ]\n",
    "            )\n",
    "        \n",
    "        signed_constant(self)\n",
    "    \n",
    "    @torch.no_grad()\n",
    "    def cache_masks(self):\n",
    "        self.register_buffer(\n",
    "            \"stacked\",\n",
    "            torch.stack(\n",
    "                [\n",
    "                    GetSubnetSparse.apply(self.scores[j], self.sparsity)\n",
    "                    for j in range(self.num_tasks)\n",
    "                ]\n",
    "            ),\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        if self.task < 0:\n",
    "            raise NotImplemented(\"Need task identity at inference time.\")\n",
    "        else:\n",
    "            # Subnet forward pass (given task info in self.task)\n",
    "            subnet = self.stacked[: self.num_seed_tasks_learned][0]\n",
    "            task_alpha = self.basis_alphas[self.task]\n",
    "            w = self.weight * subnet * task_alpha[0]\n",
    "            for i in range(1, self.num_seed_tasks_learned):\n",
    "                subnet = self.stacked[: self.num_seed_tasks_learned][i]\n",
    "                w += self.weight * subnet * task_alpha[i]\n",
    "        x = F.linear(x, w, self.bias)\n",
    "        return x\n",
    "\n",
    "\n",
    "    def __repr__(self):\n",
    "        return f\"BasisMultitaskMaskLinearSparse({self.size()})\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BasisMultitaskMaskLinearFrozen(nn.Linear):\n",
    "    def __init__(self, *args, num_tasks=1, num_seed_tasks_learned=1, start_at_optimal=True, **kwargs):\n",
    "        super().__init__(*args, **kwargs)\n",
    "        assert num_tasks >= num_seed_tasks_learned, \"Seed tasks cannot be more than total tasks!\"\n",
    "        self.num_tasks = num_tasks\n",
    "        self.num_seed_tasks_learned = num_seed_tasks_learned\n",
    "        self.scores = nn.ParameterList(\n",
    "            [\n",
    "                nn.Parameter(mask_init(self))\n",
    "                for _ in range(num_tasks)\n",
    "            ]\n",
    "        )\n",
    "        self.task = -1\n",
    "        \n",
    "        # Keep weights untrained\n",
    "        self.weight.requires_grad = False\n",
    "        for s in self.scores:\n",
    "            s.requires_grad = False\n",
    "        self.scores.requires_grad = False\n",
    "        if start_at_optimal:\n",
    "            self.basis_alphas = nn.ParameterList(\n",
    "                [\n",
    "                    nn.Parameter(torch.eye(self.num_seed_tasks_learned)[i])\n",
    "                    for i in range(self.num_seed_tasks_learned)\n",
    "                ]\n",
    "                +\n",
    "                [\n",
    "                    nn.Parameter(torch.ones(self.num_seed_tasks_learned)/self.num_seed_tasks_learned)\n",
    "                    for _ in range(self.num_seed_tasks_learned, self.num_tasks)\n",
    "                ]\n",
    "            )\n",
    "        else:\n",
    "            self.basis_alphas = nn.ParameterList(\n",
    "                [\n",
    "                    nn.Parameter(torch.ones(self.num_seed_tasks_learned)/self.num_seed_tasks_learned)\n",
    "                    for _ in range(self.num_tasks)\n",
    "                ]\n",
    "            )\n",
    "        for a in self.basis_alphas:\n",
    "            a.requires_grad = False\n",
    "        self.basis_alphas.requires_grad = False\n",
    "        \n",
    "        signed_constant(self)\n",
    "    \n",
    "    @torch.no_grad()\n",
    "    def cache_masks(self):\n",
    "        self.register_buffer(\n",
    "            \"stacked\",\n",
    "            torch.stack(\n",
    "                [\n",
    "                    GetSubnet.apply(self.scores[j])\n",
    "                    for j in range(self.num_tasks)\n",
    "                ]\n",
    "            ),\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        if self.task < 0:\n",
    "            raise NotImplemented(\"Need task identity at inference time.\")\n",
    "        else:\n",
    "            # Subnet forward pass (given task info in self.task)\n",
    "            subnet = self.stacked[: self.num_seed_tasks_learned][0]\n",
    "            task_alpha = self.basis_alphas[self.task]\n",
    "            w = self.weight * subnet * task_alpha[0]\n",
    "            for i in range(1, self.num_seed_tasks_learned):\n",
    "                subnet = self.stacked[: self.num_seed_tasks_learned][i]\n",
    "                w += self.weight * subnet * task_alpha[i]\n",
    "        x = F.linear(x, w, self.bias)\n",
    "        return x\n",
    "\n",
    "\n",
    "    def __repr__(self):\n",
    "        return f\"BasisMultitaskMaskLinearFrozen({self.size()})\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BasisMultitaskMaskLinearFrozenSparse(nn.Linear):\n",
    "    def __init__(self, *args, num_tasks=1, num_seed_tasks_learned=1, start_at_optimal=True, sparsity=0.5, **kwargs):\n",
    "        super().__init__(*args, **kwargs)\n",
    "        assert num_tasks >= num_seed_tasks_learned, \"Seed tasks cannot be more than total tasks!\"\n",
    "        self.num_tasks = num_tasks\n",
    "        self.sparsity = sparsity\n",
    "        self.num_seed_tasks_learned = num_seed_tasks_learned\n",
    "        self.scores = nn.ParameterList(\n",
    "            [\n",
    "                nn.Parameter(mask_init(self))\n",
    "                for _ in range(num_tasks)\n",
    "            ]\n",
    "        )\n",
    "        self.task = -1\n",
    "        \n",
    "        # Keep weights untrained\n",
    "        self.weight.requires_grad = False\n",
    "        for s in self.scores:\n",
    "            s.requires_grad = False\n",
    "        self.scores.requires_grad = False\n",
    "        if start_at_optimal:\n",
    "            self.basis_alphas = nn.ParameterList(\n",
    "                [\n",
    "                    nn.Parameter(torch.eye(self.num_seed_tasks_learned)[i])\n",
    "                    for i in range(self.num_seed_tasks_learned)\n",
    "                ]\n",
    "                +\n",
    "                [\n",
    "                    nn.Parameter(torch.ones(self.num_seed_tasks_learned)/self.num_seed_tasks_learned)\n",
    "                    for _ in range(self.num_seed_tasks_learned, self.num_tasks)\n",
    "                ]\n",
    "            )\n",
    "        else:\n",
    "            self.basis_alphas = nn.ParameterList(\n",
    "                [\n",
    "                    nn.Parameter(torch.ones(self.num_seed_tasks_learned)/self.num_seed_tasks_learned)\n",
    "                    for _ in range(self.num_tasks)\n",
    "                ]\n",
    "            )\n",
    "        for a in self.basis_alphas:\n",
    "            a.requires_grad = False\n",
    "        self.basis_alphas.requires_grad = False\n",
    "        \n",
    "        signed_constant(self)\n",
    "    \n",
    "    @torch.no_grad()\n",
    "    def cache_masks(self):\n",
    "        self.register_buffer(\n",
    "            \"stacked\",\n",
    "            torch.stack(\n",
    "                [\n",
    "                    GetSubnetSparse.apply(self.scores[j], self.sparsity)\n",
    "                    for j in range(self.num_tasks)\n",
    "                ]\n",
    "            ),\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        if self.task < 0:\n",
    "            raise NotImplemented(\"Need task identity at inference time.\")\n",
    "        else:\n",
    "            # Subnet forward pass (given task info in self.task)\n",
    "            subnet = self.stacked[: self.num_seed_tasks_learned][0]\n",
    "            task_alpha = self.basis_alphas[self.task]\n",
    "            w = self.weight * subnet * task_alpha[0]\n",
    "            for i in range(1, self.num_seed_tasks_learned):\n",
    "                subnet = self.stacked[: self.num_seed_tasks_learned][i]\n",
    "                w += self.weight * subnet * task_alpha[i]\n",
    "        x = F.linear(x, w, self.bias)\n",
    "        return x\n",
    "\n",
    "\n",
    "    def __repr__(self):\n",
    "        return f\"BasisMultitaskMaskLinearFrozenSparse({self.size()})\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "metadata": {},
   "outputs": [],
   "source": [
    "ValidConvs = [\n",
    "    MultitaskMaskLinear,\n",
    "    MultitaskMaskLinearSparse,\n",
    "    BasisMultitaskMaskLinear,\n",
    "    BasisMultitaskMaskLinearSparse,\n",
    "    BasisMultitaskMaskLinearFrozen,\n",
    "    BasisMultitaskMaskLinearFrozenSparse\n",
    "]\n",
    "\n",
    "def isoftype(m, cls_lst):\n",
    "    return any([isinstance(m, c) for c in cls_lst])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Utility functions\n",
    "def set_model_task(model, task, verbose=True):\n",
    "    for n, m in model.named_modules():\n",
    "        if isoftype(m, ValidConvs):\n",
    "            if verbose:\n",
    "                print(f\"=> Set task of {n} to {task}\")\n",
    "            m.task = task\n",
    "\n",
    "def cache_masks(model):\n",
    "    for n, m in model.named_modules():\n",
    "        if isoftype(m, ValidConvs):\n",
    "            print(f\"=> Caching mask state for {n}\")\n",
    "            m.cache_masks()\n",
    "\n",
    "def set_num_tasks_learned(model, num_tasks_learned):\n",
    "    for n, m in model.named_modules():\n",
    "        if isoftype(m, ValidConvs):\n",
    "            print(f\"=> Setting learned tasks of {n} to {num_tasks_learned}\")\n",
    "            m.num_tasks_learned = num_tasks_learned\n",
    "\n",
    "def set_alphas(model, alphas, verbose=True):\n",
    "    for n, m in model.named_modules():\n",
    "        if isoftype(m, ValidConvs):\n",
    "            if verbose:\n",
    "                print(f\"=> Setting alphas for {n}\")\n",
    "            m.alphas = alphas"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Model\n",
    "\n",
    "For simplicity we use a simple 3-layer fully connected neural network. Note that you can make any neural by replacing a standard layer with its corresponding mask layer."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Multitask Model, a simple fully connected model in this case\n",
    "class MultitaskFC(nn.Module):\n",
    "    def __init__(self, hidden_size, num_tasks):\n",
    "        super().__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            MultitaskMaskLinear(\n",
    "                784,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            MultitaskMaskLinear(\n",
    "                hidden_size,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            MultitaskMaskLinear(\n",
    "                hidden_size,\n",
    "                100,\n",
    "                num_tasks=num_tasks,\n",
    "                bias=False\n",
    "            )\n",
    "        )\n",
    "    \n",
    "    def forward(self, x):\n",
    "        return self.model(x.flatten(1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 70,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Multitask Model, a simple fully connected model in this case\n",
    "class MultitaskFCSparse(nn.Module):\n",
    "    def __init__(self, hidden_size, num_tasks, sparsity):\n",
    "        super().__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            MultitaskMaskLinearSparse(\n",
    "                784,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            MultitaskMaskLinearSparse(\n",
    "                hidden_size,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            MultitaskMaskLinearSparse(\n",
    "                hidden_size,\n",
    "                100,\n",
    "                num_tasks=num_tasks,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            )\n",
    "        )\n",
    "    \n",
    "    def forward(self, x):\n",
    "        return self.model(x.flatten(1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Multitask Model, a simple fully connected model in this case\n",
    "class BasisMultitaskFC(nn.Module):\n",
    "    def __init__(self, hidden_size, num_tasks, num_seed_tasks_learned, start_at_optimal=True):\n",
    "        super().__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            BasisMultitaskMaskLinear(\n",
    "                784,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            BasisMultitaskMaskLinear(\n",
    "                hidden_size,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            BasisMultitaskMaskLinear(\n",
    "                hidden_size,\n",
    "                100,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                bias=False\n",
    "            )\n",
    "        )\n",
    "    \n",
    "    def forward(self, x):\n",
    "        return self.model(x.flatten(1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BasisHiddenOnlyMultitaskFC(nn.Module):\n",
    "    def __init__(self, hidden_size, num_tasks, num_seed_tasks_learned, start_at_optimal=True):\n",
    "        super().__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            BasisMultitaskMaskLinear(\n",
    "                784,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            BasisMultitaskMaskLinear(\n",
    "                hidden_size,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            MultitaskMaskLinear(\n",
    "                hidden_size,\n",
    "                100,\n",
    "                num_tasks=num_tasks,\n",
    "                bias=False\n",
    "            )\n",
    "        )\n",
    "    \n",
    "    def forward(self, x):\n",
    "        return self.model(x.flatten(1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BasisHiddenOnlyFrozenMultitaskFC(nn.Module):\n",
    "    def __init__(self, hidden_size, num_tasks, num_seed_tasks_learned, start_at_optimal=True):\n",
    "        super().__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            BasisMultitaskMaskLinearFrozen(\n",
    "                784,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            BasisMultitaskMaskLinearFrozen(\n",
    "                hidden_size,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            MultitaskMaskLinear(\n",
    "                hidden_size,\n",
    "                100,\n",
    "                num_tasks=num_tasks,\n",
    "                bias=False\n",
    "            )\n",
    "        )\n",
    "    \n",
    "    def forward(self, x):\n",
    "        return self.model(x.flatten(1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 74,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Multitask Model, a simple fully connected model in this case\n",
    "class BasisSparseMultitaskFC(nn.Module):\n",
    "    def __init__(self, hidden_size, num_tasks, num_seed_tasks_learned, sparsity, start_at_optimal=True):\n",
    "        super().__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            BasisMultitaskMaskLinearSparse(\n",
    "                784,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            BasisMultitaskMaskLinearSparse(\n",
    "                hidden_size,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            BasisMultitaskMaskLinearSparse(\n",
    "                hidden_size,\n",
    "                100,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            )\n",
    "        )\n",
    "    \n",
    "    def forward(self, x):\n",
    "        return self.model(x.flatten(1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 116,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BasisHiddenOnlySparseMultitaskFC(nn.Module):\n",
    "    def __init__(self, hidden_size, num_tasks, num_seed_tasks_learned, sparsity, start_at_optimal=True):\n",
    "        super().__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            BasisMultitaskMaskLinearSparse(\n",
    "                784,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            BasisMultitaskMaskLinearSparse(\n",
    "                hidden_size,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            MultitaskMaskLinearSparse(\n",
    "                hidden_size,\n",
    "                100,\n",
    "                num_tasks=num_tasks,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            )\n",
    "        )\n",
    "    \n",
    "    def forward(self, x):\n",
    "        return self.model(x.flatten(1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 147,
   "metadata": {},
   "outputs": [],
   "source": [
    "class BasisHiddenOnlySparseFrozenMultitaskFC(nn.Module):\n",
    "    def __init__(self, hidden_size, num_tasks, num_seed_tasks_learned, sparsity, start_at_optimal=True):\n",
    "        super().__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            BasisMultitaskMaskLinearFrozenSparse(\n",
    "                784,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            BasisMultitaskMaskLinearFrozenSparse(\n",
    "                hidden_size,\n",
    "                hidden_size,\n",
    "                num_tasks=num_tasks,\n",
    "                num_seed_tasks_learned=num_seed_tasks_learned,\n",
    "                start_at_optimal=start_at_optimal,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            ),\n",
    "            nn.ReLU(),\n",
    "            MultitaskMaskLinearSparse(\n",
    "                hidden_size,\n",
    "                100,\n",
    "                num_tasks=num_tasks,\n",
    "                sparsity=sparsity,\n",
    "                bias=False\n",
    "            )\n",
    "        )\n",
    "    \n",
    "    def forward(self, x):\n",
    "        return self.model(x.flatten(1))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Dataset \n",
    "\n",
    "Our \"base\" dataset is MNIST. Each task is a random permutation of the pixels of MNIST. Namely, given a dataset $\\{(x_i, y_i)\\}_i$ for each task $t$, we take a random permutation $S^t\\in S_{784}$ and construct a dataset $\\{(S^t(x_i), y_i)\\}_i$. \n",
    "\n",
    "Below we show this process and an example of a particular task. `MNISTPerm` uses an update task method to generate a particular sub-task. To assure these splits are reproducible, `MNISTPerm` takes a `seed` argument. In our paper we evaluate on seeds 0 through 4."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "metadata": {
    "scrolled": true,
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<PIL.Image.Image image mode=RGB size=493x269 at 0x7EFF52B85B10>"
      ]
     },
     "execution_count": 77,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "class MNISTPerm:\n",
    "    class permute(object):\n",
    "        def __call__(self, tensor):\n",
    "            out = tensor.flatten()\n",
    "            out = out[self.perm]\n",
    "            return out.view(1, 28, 28)\n",
    "\n",
    "        def __repr__(self):\n",
    "            return self.__class__.__name__\n",
    "\n",
    "    def __init__(self, seed=0):\n",
    "        super(MNISTPerm, self).__init__()\n",
    "        \n",
    "        data_root = \"mnist\"\n",
    "        self.permuter = self.permute()\n",
    "        self.seed = seed\n",
    "        train_dataset = torchvision.datasets.MNIST(\n",
    "            data_root,\n",
    "            train=True,\n",
    "            download=True,\n",
    "            transform=torchvision.transforms.Compose(\n",
    "                [\n",
    "                    torchvision.transforms.ToTensor(),\n",
    "                    torchvision.transforms.Normalize((0.1307,), (0.3081,)),\n",
    "                    self.permuter,\n",
    "                ]\n",
    "            ),\n",
    "        )\n",
    "\n",
    "        # Data loading code\n",
    "        self.train_loader = torch.utils.data.DataLoader(\n",
    "            train_dataset, batch_size=128, shuffle=True\n",
    "        )\n",
    "        self.val_loader = torch.utils.data.DataLoader(\n",
    "            torchvision.datasets.MNIST(\n",
    "                data_root,\n",
    "                train=False,\n",
    "                transform=torchvision.transforms.Compose(\n",
    "                    [\n",
    "                        torchvision.transforms.ToTensor(),\n",
    "                        torchvision.transforms.Normalize((0.1307,), (0.3081,)),\n",
    "                        self.permuter,\n",
    "                    ]\n",
    "                ),\n",
    "            ),\n",
    "            batch_size=128,\n",
    "            shuffle=False,\n",
    "        )\n",
    "\n",
    "    def update_task(self, i):\n",
    "        np.random.seed(i + self.seed)\n",
    "        self.permuter.__setattr__(\"perm\", np.random.permutation(784))\n",
    "    \n",
    "    def unpermute(self):\n",
    "        self.permuter.__setattr__(\"perm\", np.arange(784))\n",
    "\n",
    "mnist = MNISTPerm()\n",
    "\n",
    "# Showing some example images from MNISTPerm\n",
    "mnist.unpermute()\n",
    "batch, labels = next(iter(mnist.val_loader))\n",
    "\n",
    "mnist.update_task(0)\n",
    "task0, labels = next(iter(mnist.val_loader))\n",
    "\n",
    "torchvision.transforms.ToPILImage()(\n",
    "    torchvision.utils.make_grid(\n",
    "        torch.cat([batch, task0], dim=-1)[:64],\n",
    "        normalize=True,\n",
    "        padding=5,\n",
    "        pad_value=0.2\n",
    "    )\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<PIL.Image.Image image mode=RGB size=493x269 at 0x7EFF5264FF10>"
      ]
     },
     "execution_count": 87,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import random\n",
    "\n",
    "class MNISTRot:\n",
    "    class rot(object):\n",
    "        def __call__(self, img):\n",
    "            if self.rotate:\n",
    "                return torchvision.transforms.RandomRotation(270)(img)\n",
    "            return img\n",
    "\n",
    "        def __repr__(self):\n",
    "            return self.__class__.__name__\n",
    "\n",
    "    def __init__(self, seed=0):\n",
    "        super(MNISTRot, self).__init__()\n",
    "        \n",
    "        data_root = \"mnist\"\n",
    "        self.seed = seed\n",
    "        self.rotator = self.rot()\n",
    "        train_dataset = torchvision.datasets.MNIST(\n",
    "            data_root,\n",
    "            train=True,\n",
    "            download=True,\n",
    "            transform=torchvision.transforms.Compose(\n",
    "                [\n",
    "                    self.rotator,\n",
    "                    torchvision.transforms.ToTensor(),\n",
    "                    torchvision.transforms.Normalize((0.1307,), (0.3081,)),\n",
    "                ]\n",
    "            ),\n",
    "        )\n",
    "\n",
    "        # Data loading code\n",
    "        self.train_loader = torch.utils.data.DataLoader(\n",
    "            train_dataset, batch_size=128, shuffle=True\n",
    "        )\n",
    "        self.val_loader = torch.utils.data.DataLoader(\n",
    "            torchvision.datasets.MNIST(\n",
    "                data_root,\n",
    "                train=False,\n",
    "                transform=torchvision.transforms.Compose(\n",
    "                     [\n",
    "                         self.rotator,\n",
    "                         torchvision.transforms.ToTensor(),\n",
    "                         torchvision.transforms.Normalize((0.1307,), (0.3081,))\n",
    "                     ]\n",
    "                 ),\n",
    "            ),\n",
    "            batch_size=128,\n",
    "            shuffle=False,\n",
    "        )\n",
    "\n",
    "    def update_task(self, i):\n",
    "        random.seed(i + self.seed)\n",
    "        self.rotator.__setattr__(\"rotate\", True)\n",
    "    \n",
    "    def unrotate(self):\n",
    "        self.rotator.__setattr__(\"rotate\", False)\n",
    "\n",
    "mnist_rot = MNISTRot()\n",
    "\n",
    "# Showing some example images from MNISTPerm\n",
    "mnist_rot.unrotate()\n",
    "batch, labels = next(iter(mnist_rot.val_loader))\n",
    "\n",
    "mnist_rot.update_task(0)\n",
    "task0, labels = next(iter(mnist_rot.val_loader))\n",
    "\n",
    "torchvision.transforms.ToPILImage()(\n",
    "    torchvision.utils.make_grid(\n",
    "        torch.cat([batch, task0], dim=-1)[:64],\n",
    "        normalize=True,\n",
    "        padding=5,\n",
    "        pad_value=0.2\n",
    "    )\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Simple train and evaluation loops for classification"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Finding supermasks per task\n",
    "\n",
    "def train(model, trainloader, optimizer, epoch):\n",
    "    model.train()\n",
    "\n",
    "    criterion = nn.CrossEntropyLoss()\n",
    "    num_correct = 0\n",
    "    total_seen = 0\n",
    "    for i, (batch, labels) in tqdm(\n",
    "        enumerate(trainloader),\n",
    "        ascii=True,\n",
    "        total=len(trainloader)\n",
    "    ):\n",
    "        logits = model(batch)\n",
    "        loss = criterion(logits, labels)\n",
    "\n",
    "        optimizer.zero_grad()\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "\n",
    "        if i % 20 == 0:\n",
    "            predictions = logits.argmax(dim=-1)\n",
    "            num_correct += (predictions == labels).float().sum()\n",
    "            total_seen += logits.size(0) \n",
    "            tqdm.write(\n",
    "                (f\"e{epoch} {i+1}/{len(trainloader)}\"\n",
    "                f\" => Loss {loss.item():0.4f}, \"\n",
    "                f\"Acc@1 {(num_correct / total_seen):0.4f}\"),\n",
    "                end=\"\\r\"\n",
    "            )\n",
    "\n",
    "\n",
    "@torch.no_grad()\n",
    "def evaluate(model, val_loader, epoch):\n",
    "    model.eval()\n",
    "    num_correct = 0\n",
    "    total_seen = 0\n",
    "    for batch, labels in tqdm(\n",
    "        val_loader,\n",
    "        ascii=True,\n",
    "        total=len(val_loader)\n",
    "    ):\n",
    "        logits = model(batch)\n",
    "        predictions = logits.argmax(dim=-1)\n",
    "        num_correct += (predictions == labels).float().sum()\n",
    "        total_seen += logits.size(0) \n",
    "    \n",
    "\n",
    "    tqdm.write(\n",
    "        f\"Val Perf after {epoch + 1} epochs \"\n",
    "        f\"Acc@1 {(num_correct / total_seen):0.4f}\", \n",
    "    )\n",
    "    return num_correct / total_seen\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "For any scenario where task ID is given during train time, we simply train each supermask independently, one per task. We make use of our utility function `set_model_task` to do this. We use 5 tasks for demonstration purposes. Feel free to increase `num_tasks` in the following cell. In our paper we use as many as 2500 tasks with no drop in performance for either task inference or validation accuracy."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map = {}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2f01b79e8a214d20b9f6c699bdc899e4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3408, Acc@1 0.8434\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "473858bd1178400e8554fbdd2f0a83ba",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9295\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "dc6b4abb08cf40c9b6af445807c4435b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3102, Acc@1 0.8564\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e59258e7c04e4dc39f70c62b5e765049",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9266\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "\n",
      "Training for task 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1a6c611288ac4162a8d442cbccb30fa3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2391, Acc@1 0.8480\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "4b38798b30af47f6a8263d3f7e989b68",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9264\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "\n",
      "Training for task 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "47e5818d8e394381995248b7509db6e9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2603, Acc@1 0.8561\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a6f9b8761e164af2b88e4a3efbf84ba4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9247\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "\n",
      "Training for task 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f3ce972a61d64a68a939f5b035b28440",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3563, Acc@1 0.8522\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fd9cbcea00614ed987cbdb9be271e8f6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9281\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "\n",
      "Training for task 5\n",
      "=> Set task of model.0 to 5\n",
      "=> Set task of model.2 to 5\n",
      "=> Set task of model.4 to 5\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ee6c3806f3794d2abf157008cba9794f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2611, Acc@1 0.8594\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6332c02e6e47433f8ac5d560b1ede2c9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9229\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 6\n",
      "=> Setting learned tasks of model.2 to 6\n",
      "=> Setting learned tasks of model.4 to 6\n",
      "\n",
      "Training for task 6\n",
      "=> Set task of model.0 to 6\n",
      "=> Set task of model.2 to 6\n",
      "=> Set task of model.4 to 6\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9ddb079ccad947e79ee2ce3ba5318f64",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2415, Acc@1 0.8659\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "91eec7deb30a4d1eac1a2c76fdde98f2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9262\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 7\n",
      "=> Setting learned tasks of model.2 to 7\n",
      "=> Setting learned tasks of model.4 to 7\n",
      "\n",
      "Training for task 7\n",
      "=> Set task of model.0 to 7\n",
      "=> Set task of model.2 to 7\n",
      "=> Set task of model.4 to 7\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cbb92a4954c6431099ce84bfad7a4bd8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2871, Acc@1 0.8623\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d72033a9ffe6474aaa307a263a4d7ebc",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9242\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 8\n",
      "=> Setting learned tasks of model.2 to 8\n",
      "=> Setting learned tasks of model.4 to 8\n",
      "\n",
      "Training for task 8\n",
      "=> Set task of model.0 to 8\n",
      "=> Set task of model.2 to 8\n",
      "=> Set task of model.4 to 8\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "520722f6d43a4b64926870cd1ac64bd8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3398, Acc@1 0.8477\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "121f191c62b14154b62962190e7ad318",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9263\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 9\n",
      "=> Setting learned tasks of model.2 to 9\n",
      "=> Setting learned tasks of model.4 to 9\n",
      "\n",
      "Training for task 9\n",
      "=> Set task of model.0 to 9\n",
      "=> Set task of model.2 to 9\n",
      "=> Set task of model.4 to 9\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "904555278a8a4943a123de1fe19ba4f9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.1636, Acc@1 0.8590\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e99fd88f65134a47b46ecd888f3a1cc4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9262\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 10\n",
      "=> Setting learned tasks of model.2 to 10\n",
      "=> Setting learned tasks of model.4 to 10\n",
      "\n",
      "Training for task 10\n",
      "=> Set task of model.0 to 10\n",
      "=> Set task of model.2 to 10\n",
      "=> Set task of model.4 to 10\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a79ebca2614d4e91849349ef51744351",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9255778\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 13\n",
      "=> Setting learned tasks of model.2 to 13\n",
      "=> Setting learned tasks of model.4 to 13\n",
      "\n",
      "Training for task 13\n",
      "=> Set task of model.0 to 13\n",
      "=> Set task of model.2 to 13\n",
      "=> Set task of model.4 to 13\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a8dcdcc981ab4f8e88c7faaa7886c76b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2556, Acc@1 0.8464\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e4e3b2c8e6324a74b383b1fb1d38e60e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9287\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 14\n",
      "=> Setting learned tasks of model.2 to 14\n",
      "=> Setting learned tasks of model.4 to 14\n",
      "\n",
      "Training for task 14\n",
      "=> Set task of model.0 to 14\n",
      "=> Set task of model.2 to 14\n",
      "=> Set task of model.4 to 14\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "4279e74273fc43d193e2f041cef1f09b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3549, Acc@1 0.8532\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fef5312785144ae780f5266bef8fa5d4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9242\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 15\n",
      "=> Setting learned tasks of model.2 to 15\n",
      "=> Setting learned tasks of model.4 to 15\n",
      "\n",
      "Training for task 15\n",
      "=> Set task of model.0 to 15\n",
      "=> Set task of model.2 to 15\n",
      "=> Set task of model.4 to 15\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "88e98e2e3dfd40918b4877011751834c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2233, Acc@1 0.8438\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cf8e16404c0e438e8b2f196de65ec538",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9277\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 16\n",
      "=> Setting learned tasks of model.2 to 16\n",
      "=> Setting learned tasks of model.4 to 16\n",
      "\n",
      "Training for task 16\n",
      "=> Set task of model.0 to 16\n",
      "=> Set task of model.2 to 16\n",
      "=> Set task of model.4 to 16\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e6a0cf1c801e4ada921db00ae68f96f1",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.4212, Acc@1 0.8535\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "50643b6014b14a4a9602cab8b9729546",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9236\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 17\n",
      "=> Setting learned tasks of model.2 to 17\n",
      "=> Setting learned tasks of model.4 to 17\n",
      "\n",
      "Training for task 17\n",
      "=> Set task of model.0 to 17\n",
      "=> Set task of model.2 to 17\n",
      "=> Set task of model.4 to 17\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a7226cbe2c0f4cea9da0615574190ae7",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 261/469 => Loss 0.3730, Acc@1 0.8052\r"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "IOPub message rate exceeded.\n",
      "The notebook server will temporarily stop sending output\n",
      "to the client in order to avoid crashing it.\n",
      "To change this limit, set the config variable\n",
      "`--NotebookApp.iopub_msg_rate_limit`.\n",
      "\n",
      "Current values:\n",
      "NotebookApp.iopub_msg_rate_limit=1000.0 (msgs/sec)\n",
      "NotebookApp.rate_limit_window=3.0 (secs)\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# Task ID given during train time\n",
    "from IPython.display import clear_output\n",
    "\n",
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "model = MultitaskFC(hidden_size=300, num_tasks=num_tasks)\n",
    "\n",
    "for task in range(num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(model, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in model.parameters() if p.requires_grad], lr=1e-4)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(1):\n",
    "        train(model, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(model, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(model)\n",
    "    print()\n",
    "    set_num_tasks_learned(model, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.9260\n",
      "Per task performance\n",
      "Task 0: 0.9295\n",
      "Task 1: 0.9266\n",
      "Task 2: 0.9264\n",
      "Task 3: 0.9247\n",
      "Task 4: 0.9281\n",
      "Task 5: 0.9229\n",
      "Task 6: 0.9262\n",
      "Task 7: 0.9242\n",
      "Task 8: 0.9263\n",
      "Task 9: 0.9262\n",
      "Task 10: 0.9243\n",
      "Task 11: 0.9255\n",
      "Task 12: 0.9255\n",
      "Task 13: 0.9287\n",
      "Task 14: 0.9242\n",
      "Task 15: 0.9277\n",
      "Task 16: 0.9236\n",
      "Task 17: 0.9266\n",
      "Task 18: 0.9224\n",
      "Task 19: 0.9300\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(model, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(model, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 83,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['supsup_mnist'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### MNIST Perm with Sparsity"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 84,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0cf4c62434694fdba141e047faf7965a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2830, Acc@1 0.8398\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "db83f980f1e04db780830302dc4d2b2b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9116\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "dc583f3b7214409685038cccbc25fbe2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3703, Acc@1 0.8431\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "94974cf2d35b4840952c909cb0fdc6ec",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9166\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "\n",
      "Training for task 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ed7ff08c5adc45558e9a7731cbf5cee2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2802, Acc@1 0.8385\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "336efc44c00746d1a4604ee847fc6cf4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9111\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "\n",
      "Training for task 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ae27508ba1834542bf3c630e2aeb246e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3283, Acc@1 0.8421\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "48b1d1fc9b77434683ad6af546fb6514",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9166\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "\n",
      "Training for task 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "43483a823b6a4a6db2c41aa79f9582de",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 41/469 => Loss 0.9449, Acc@1 0.5182\r"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "IOPub message rate exceeded.\n",
      "The notebook server will temporarily stop sending output\n",
      "to the client in order to avoid crashing it.\n",
      "To change this limit, set the config variable\n",
      "`--NotebookApp.iopub_msg_rate_limit`.\n",
      "\n",
      "Current values:\n",
      "NotebookApp.iopub_msg_rate_limit=1000.0 (msgs/sec)\n",
      "NotebookApp.rate_limit_window=3.0 (secs)\n",
      "\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9108\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 12\n",
      "=> Setting learned tasks of model.2 to 12\n",
      "=> Setting learned tasks of model.4 to 12\n",
      "\n",
      "Training for task 12\n",
      "=> Set task of model.0 to 12\n",
      "=> Set task of model.2 to 12\n",
      "=> Set task of model.4 to 12\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9b353a6e519b408bbc66c82f22274ca3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2399, Acc@1 0.8405\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "94f70775222840d48c62e9cec3ae4069",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9127\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 13\n",
      "=> Setting learned tasks of model.2 to 13\n",
      "=> Setting learned tasks of model.4 to 13\n",
      "\n",
      "Training for task 13\n",
      "=> Set task of model.0 to 13\n",
      "=> Set task of model.2 to 13\n",
      "=> Set task of model.4 to 13\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "dab970d7469b412ca3a0d65e57d3c1cb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3334, Acc@1 0.8350\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "013077f7d2af4bd7ac360f94f1858b93",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9135\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 14\n",
      "=> Setting learned tasks of model.2 to 14\n",
      "=> Setting learned tasks of model.4 to 14\n",
      "\n",
      "Training for task 14\n",
      "=> Set task of model.0 to 14\n",
      "=> Set task of model.2 to 14\n",
      "=> Set task of model.4 to 14\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9885abb912064732ba4b35ec9a46d52f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 221/469 => Loss 0.4793, Acc@1 0.7832\r"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "IOPub message rate exceeded.\n",
      "The notebook server will temporarily stop sending output\n",
      "to the client in order to avoid crashing it.\n",
      "To change this limit, set the config variable\n",
      "`--NotebookApp.iopub_msg_rate_limit`.\n",
      "\n",
      "Current values:\n",
      "NotebookApp.iopub_msg_rate_limit=1000.0 (msgs/sec)\n",
      "NotebookApp.rate_limit_window=3.0 (secs)\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# Task ID given during train time\n",
    "from IPython.display import clear_output\n",
    "\n",
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "models = MultitaskFCSparse(hidden_size=300, num_tasks=num_tasks, sparsity=0.25)\n",
    "\n",
    "for task in range(num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(models, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in models.parameters() if p.requires_grad], lr=1e-4)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(1):\n",
    "        train(models, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(models, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(models)\n",
    "    print()\n",
    "    set_num_tasks_learned(models, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.9123\n",
      "Per task performance\n",
      "Task 0: 0.9116\n",
      "Task 1: 0.9166\n",
      "Task 2: 0.9111\n",
      "Task 3: 0.9166\n",
      "Task 4: 0.9125\n",
      "Task 5: 0.9125\n",
      "Task 6: 0.9131\n",
      "Task 7: 0.9119\n",
      "Task 8: 0.9126\n",
      "Task 9: 0.9058\n",
      "Task 10: 0.9093\n",
      "Task 11: 0.9108\n",
      "Task 12: 0.9127\n",
      "Task 13: 0.9135\n",
      "Task 14: 0.9118\n",
      "Task 15: 0.9123\n",
      "Task 16: 0.9130\n",
      "Task 17: 0.9120\n",
      "Task 18: 0.9134\n",
      "Task 19: 0.9134\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(models, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(models, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['supsup_mnist_sparsity_25'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### MNIST Rotated"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "03aa94ab836a49e69d694e8a737f5da2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1026, Acc@1 0.5319\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "bf046f83ad294711ac971b97985ed6c0",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.6489\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "01bfd843094842b387df304dfe37d198",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2014, Acc@1 0.5283\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "24d494facdd149818949cf2e1fc1d0fa",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.6528\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "\n",
      "Training for task 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0ed15679da3043c4a8fa37f825656b67",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1272, Acc@1 0.5371\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d492b1943d054e198a3e35fdde5cfa4c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.6497\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "\n",
      "Training for task 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ab88be1220064ece8976867185d12219",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1070, Acc@1 0.5426\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "da527d0902f54dfc9dbb05ef0c4395f1",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.6420\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "\n",
      "Training for task 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fd8d198745e94803b076db543fa09d15",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2219, Acc@1 0.5469\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2be7ed019d14405691add8cafef0b933",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.6487\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "\n",
      "Training for task 5\n",
      "=> Set task of model.0 to 5\n",
      "=> Set task of model.2 to 5\n",
      "=> Set task of model.4 to 5\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "33ebf324b0d24e1e839800bee0e40b2e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2437, Acc@1 0.5423\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9836f1fc121b4525b89f5d07c518948d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.6494\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 6\n",
      "=> Setting learned tasks of model.2 to 6\n",
      "=> Setting learned tasks of model.4 to 6\n",
      "\n",
      "Training for task 6\n",
      "=> Set task of model.0 to 6\n",
      "=> Set task of model.2 to 6\n",
      "=> Set task of model.4 to 6\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6a0f035f3fcd49a6a54772ba3b577139",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 161/469 => Loss 1.2736, Acc@1 0.4349\r"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "IOPub message rate exceeded.\n",
      "The notebook server will temporarily stop sending output\n",
      "to the client in order to avoid crashing it.\n",
      "To change this limit, set the config variable\n",
      "`--NotebookApp.iopub_msg_rate_limit`.\n",
      "\n",
      "Current values:\n",
      "NotebookApp.iopub_msg_rate_limit=1000.0 (msgs/sec)\n",
      "NotebookApp.rate_limit_window=3.0 (secs)\n",
      "\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 341/469 => Loss 1.1898, Acc@1 0.5095\r"
     ]
    }
   ],
   "source": [
    "# Task ID given during train time\n",
    "from IPython.display import clear_output\n",
    "\n",
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "model_rot = MultitaskFC(hidden_size=300, num_tasks=num_tasks)\n",
    "\n",
    "for task in range(num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(model_rot, task)\n",
    "    mnist_rot.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in model_rot.parameters() if p.requires_grad], lr=1e-4)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(1):\n",
    "        train(model_rot, mnist_rot.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(model_rot, mnist_rot.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(model_rot)\n",
    "    print()\n",
    "    set_num_tasks_learned(model_rot, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(model_rot, task)\n",
    "    mnist_rot.update_task(task)\n",
    "    acc1 = evaluate(model_rot, mnist_rot.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['supsup_mnist_rot'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Basis Masks "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Only New Tasks)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model = BasisMultitaskFC(hidden_size=300, num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "seed_dict = model.state_dict()\n",
    "basis_dict = basis_model.state_dict()\n",
    "load_dict = {k: seed_dict[k] for k in basis_model.state_dict().keys() if k in seed_dict.keys()}\n",
    "basis_dict.update(load_dict)\n",
    "basis_model.load_state_dict(basis_dict, False)\n",
    "cache_masks(basis_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "for task in range(num_seed_tasks_learned, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model.parameters() if p.requires_grad], lr=1e-2)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(2):\n",
    "        train(basis_model, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": false
   },
   "outputs": [],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_model, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_mnist'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Only New Tasks) Sparsity Enabled"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_models = BasisSparseMultitaskFC(hidden_size=300, num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, sparsity=0.25, start_at_optimal=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "seed_dict = models.state_dict()\n",
    "basis_dict = basis_models.state_dict()\n",
    "load_dict = {k: seed_dict[k] for k in basis_models.state_dict().keys() if k in seed_dict.keys()}\n",
    "basis_dict.update(load_dict)\n",
    "basis_models.load_state_dict(basis_dict, False)\n",
    "cache_masks(basis_models)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "for task in range(num_seed_tasks_learned, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_models, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_models.parameters() if p.requires_grad], lr=1e-2)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(2):\n",
    "        train(basis_models, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_models, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_models)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_models, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": false
   },
   "outputs": [],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_models, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_models, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_mnist_sparsity_25'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Only New Tasks, Rotation MNIST)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_rot = BasisMultitaskFC(hidden_size=300, num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "seed_dictr = model_rot.state_dict()\n",
    "basis_dictr = basis_model_rot.state_dict()\n",
    "load_dictr = {k: seed_dictr[k] for k in basis_model_rot.state_dict().keys() if k in seed_dictr.keys()}\n",
    "basis_dictr.update(load_dictr)\n",
    "basis_model_rot.load_state_dict(basis_dictr, False)\n",
    "cache_masks(basis_model_rot)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "for task in range(num_seed_tasks_learned, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model_rot, task)\n",
    "    mnist_rot.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model_rot.parameters() if p.requires_grad], lr=1e-2)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(2):\n",
    "        train(basis_model_rot, mnist_rot.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model_rot, mnist_rot.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model_rot)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model_rot, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_rot, task)\n",
    "    mnist_rot.update_task(task)\n",
    "    acc1 = evaluate(basis_model_rot, mnist_rot.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_mnist_rot'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Except Final Layer"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Only New Tasks)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_hid = BasisHiddenOnlyMultitaskFC(hidden_size=300, num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "seed_dict_hid = model.state_dict()\n",
    "basis_dict_hid = basis_model_hid.state_dict()\n",
    "load_dict_hid = {k: seed_dict_hid[k] for k in basis_model_hid.state_dict().keys() if k in seed_dict_hid.keys()}\n",
    "basis_dict_hid.update(load_dict_hid)\n",
    "basis_model_hid.load_state_dict(basis_dict_hid, False)\n",
    "cache_masks(basis_model_hid)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "for task in range(num_seed_tasks_learned, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model_hid, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model_hid.parameters() if p.requires_grad], lr=1e-2)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(3):\n",
    "        train(basis_model_hid, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model_hid, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model_hid)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model_hid, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_hid, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_model_hid, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_hybrid_mnist'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Only New Tasks) Sparsity Enabled"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 117,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_hids = BasisHiddenOnlySparseMultitaskFC(hidden_size=300, num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=True, sparsity=0.25)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 118,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n"
     ]
    }
   ],
   "source": [
    "seed_dict_hid = models.state_dict()\n",
    "basis_dict_hid = basis_model_hids.state_dict()\n",
    "load_dict_hid = {k: seed_dict_hid[k] for k in basis_model_hids.state_dict().keys() if k in seed_dict_hid.keys()}\n",
    "basis_dict_hid.update(load_dict_hid)\n",
    "basis_model_hids.load_state_dict(basis_dict_hid, False)\n",
    "cache_masks(basis_model_hids)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 15\n",
      "=> Set task of model.0 to 15\n",
      "=> Set task of model.2 to 15\n",
      "=> Set task of model.4 to 15\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6468a3948ddf43daad1428602b31ac6c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.9486, Acc@1 0.6442\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6153081470f04b16876f9e477f51b044",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7806\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8077d317ea4f4938af9987ab6c7e67eb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.6577, Acc@1 0.7383\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e46342eb38174b1cad9ea5ceca83b4ee",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.7257\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 16\n",
      "=> Setting learned tasks of model.2 to 16\n",
      "=> Setting learned tasks of model.4 to 16\n",
      "\n",
      "Training for task 16\n",
      "=> Set task of model.0 to 16\n",
      "=> Set task of model.2 to 16\n",
      "=> Set task of model.4 to 16\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "21323626720d436db6e4f9c38535783b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2357, Acc@1 0.6048\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fffa81eae57c4891993e20347b25e049",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7068\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "30e6bd1649d245b9bcf562765e745198",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.9058, Acc@1 0.6950\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7994cdea6e5641188a22a14966ef7fdb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.6946\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 17\n",
      "=> Setting learned tasks of model.2 to 17\n",
      "=> Setting learned tasks of model.4 to 17\n",
      "\n",
      "Training for task 17\n",
      "=> Set task of model.0 to 17\n",
      "=> Set task of model.2 to 17\n",
      "=> Set task of model.4 to 17\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f067c9787ac34ed499c2658517067930",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.7452, Acc@1 0.6458\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ccd22bc06b654896a886ab07f1f7fa23",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7472\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "18f7daa26eca45ae8a369e9b1f7a0e47",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.6034, Acc@1 0.7477\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d762f949913743a2a90ffd0cb4b2e4a7",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.7418\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 18\n",
      "=> Setting learned tasks of model.2 to 18\n",
      "=> Setting learned tasks of model.4 to 18\n",
      "\n",
      "Training for task 18\n",
      "=> Set task of model.0 to 18\n",
      "=> Set task of model.2 to 18\n",
      "=> Set task of model.4 to 18\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6c07e651962d4af9bd97d106eb23cd75",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.0417, Acc@1 0.6214\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "10eccf54f97a46aba89b42e34ffaaa11",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.6675\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f2eb31a0005f439683cbbc73e0ad056a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1832, Acc@1 0.6865\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "471870f90ad24f48ab0b14de3d9ac0cc",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.7656\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 19\n",
      "=> Setting learned tasks of model.2 to 19\n",
      "=> Setting learned tasks of model.4 to 19\n",
      "\n",
      "Training for task 19\n",
      "=> Set task of model.0 to 19\n",
      "=> Set task of model.2 to 19\n",
      "=> Set task of model.4 to 19\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "4f78f7d702e9441dac84908ae91dd4d3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.9743, Acc@1 0.6370\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9aebb38cf8674641a1847fb62ff5a081",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7769\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cb7fb43c6e9d44d7aa2c85c23eeca20e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.9557, Acc@1 0.7396\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9a44cc4e556640dd9fb7194ec9ad7823",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.7597\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 20\n",
      "=> Setting learned tasks of model.2 to 20\n",
      "=> Setting learned tasks of model.4 to 20\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(num_seed_tasks_learned, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model_hids, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model_hids.parameters() if p.requires_grad], lr=1e-2)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(2):\n",
    "        train(basis_model_hids, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model_hids, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model_hids)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model_hids, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 120,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.8685\n",
      "Per task performance\n",
      "Task 0: 0.9116\n",
      "Task 1: 0.9166\n",
      "Task 2: 0.9111\n",
      "Task 3: 0.9166\n",
      "Task 4: 0.9125\n",
      "Task 5: 0.9125\n",
      "Task 6: 0.9131\n",
      "Task 7: 0.9119\n",
      "Task 8: 0.9126\n",
      "Task 9: 0.9058\n",
      "Task 10: 0.9093\n",
      "Task 11: 0.9108\n",
      "Task 12: 0.9127\n",
      "Task 13: 0.9135\n",
      "Task 14: 0.9118\n",
      "Task 15: 0.7257\n",
      "Task 16: 0.6946\n",
      "Task 17: 0.7418\n",
      "Task 18: 0.7656\n",
      "Task 19: 0.7597\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_hids, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_model_hids, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_hybrid_mnist_sparsity_25'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Only New Tasks, Rotation MNIST)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 122,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_roth = BasisHiddenOnlyMultitaskFC(hidden_size=300, num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 123,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n"
     ]
    }
   ],
   "source": [
    "seed_dictr = model_rot.state_dict()\n",
    "basis_dictrh = basis_model_roth.state_dict()\n",
    "load_dictrh = {k: seed_dictr[k] for k in basis_model_roth.state_dict().keys() if k in seed_dictr.keys()}\n",
    "basis_dictrh.update(load_dictrh)\n",
    "basis_model_roth.load_state_dict(basis_dictrh, False)\n",
    "cache_masks(basis_model_roth)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 124,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 15\n",
      "=> Set task of model.0 to 15\n",
      "=> Set task of model.2 to 15\n",
      "=> Set task of model.4 to 15\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0fcc7f896b524c78afb003debce528c4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.5047, Acc@1 0.4681\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6de979e22ef9415594cf306c6554d20d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.5270\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fc89eab30aef4db0a87845b3683463a6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1133, Acc@1 0.5400\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "92d2a58593e54f2e98fd6b8c26267bd7",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.5896\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a4d8a834e4d54727b335acefc8c3cd2e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1454, Acc@1 0.5791\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c7eae9baba604336bd3c5fe3855bd153",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 3 epochs Acc@1 0.6195\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 16\n",
      "=> Setting learned tasks of model.2 to 16\n",
      "=> Setting learned tasks of model.4 to 16\n",
      "\n",
      "Training for task 16\n",
      "=> Set task of model.0 to 16\n",
      "=> Set task of model.2 to 16\n",
      "=> Set task of model.4 to 16\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "83222d19a38e48e49bd655da48354c53",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1746, Acc@1 0.4743\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7947e506bd87411ba0bbe29b526f35b5",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.5304\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0632672dd623469eb594ff6865c03270",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1109, Acc@1 0.5924\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2cdaad061d284e38a5c501d8366f3421",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.5690\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f39116cbeeea45b588ac2d29aaf17292",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2276, Acc@1 0.5807\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cfd01b8120784bd8b8354c803d74b66d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 3 epochs Acc@1 0.6131\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 17\n",
      "=> Setting learned tasks of model.2 to 17\n",
      "=> Setting learned tasks of model.4 to 17\n",
      "\n",
      "Training for task 17\n",
      "=> Set task of model.0 to 17\n",
      "=> Set task of model.2 to 17\n",
      "=> Set task of model.4 to 17\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c2a8a91923554028b1a64fbd3df2e8e8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.5711, Acc@1 0.4574\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0b54ef1139624810afa1baa9c6c579bc",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.5098\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9ccc57efadae4d988e83e29108ec09c6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3492, Acc@1 0.5127\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5ae7587d33394a8a838f5d5bd37df331",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.4823\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2f5121f2f41a4c42ab088ac47a8d8499",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2600, Acc@1 0.5137\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8dab6d2161de48bcb80b947114d61c16",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 3 epochs Acc@1 0.4934\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 18\n",
      "=> Setting learned tasks of model.2 to 18\n",
      "=> Setting learned tasks of model.4 to 18\n",
      "\n",
      "Training for task 18\n",
      "=> Set task of model.0 to 18\n",
      "=> Set task of model.2 to 18\n",
      "=> Set task of model.4 to 18\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cd687d455a3e431dba9f958fb677c4da",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.6396, Acc@1 0.4854\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f8c517229d634badb828006b4e54d0eb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.5021\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "108cc75e02f9410b811cd1f8a6363e22",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3288, Acc@1 0.5072\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d717da00c5c04456ab0b92ccbd504620",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.5642\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9716a757034d4c6ead4af557df133562",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3338, Acc@1 0.5475\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "edd0ad0310d347e59285111d96032bdc",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 3 epochs Acc@1 0.5563\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 19\n",
      "=> Setting learned tasks of model.2 to 19\n",
      "=> Setting learned tasks of model.4 to 19\n",
      "\n",
      "Training for task 19\n",
      "=> Set task of model.0 to 19\n",
      "=> Set task of model.2 to 19\n",
      "=> Set task of model.4 to 19\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "902ca3c196264f86bde0ef53ce88dfd3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3448, Acc@1 0.4518\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ec6f461670304fe1b3c77b16add617d4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.5545\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "bb421bd3a6664e5c8b7488c21a1c43ec",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3816, Acc@1 0.5358\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e0806d70bfbd491e9ce156e94d53849d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.5599\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "abf5643864f64aa3abc4394d7f6eda72",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2830, Acc@1 0.5358\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f819ad2944b84f9e9407a8c41a1a242e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 3 epochs Acc@1 0.5938\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 20\n",
      "=> Setting learned tasks of model.2 to 20\n",
      "=> Setting learned tasks of model.4 to 20\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(num_seed_tasks_learned, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model_roth, task)\n",
    "    mnist_rot.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model_roth.parameters() if p.requires_grad], lr=1e-2)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(3):\n",
    "        train(basis_model_roth, mnist_rot.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model_roth, mnist_rot.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model_roth)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model_roth, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 125,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.6299\n",
      "Per task performance\n",
      "Task 0: 0.6406\n",
      "Task 1: 0.6466\n",
      "Task 2: 0.6469\n",
      "Task 3: 0.6428\n",
      "Task 4: 0.6543\n",
      "Task 5: 0.6528\n",
      "Task 6: 0.6538\n",
      "Task 7: 0.6475\n",
      "Task 8: 0.6346\n",
      "Task 9: 0.6585\n",
      "Task 10: 0.6547\n",
      "Task 11: 0.6553\n",
      "Task 12: 0.6446\n",
      "Task 13: 0.6526\n",
      "Task 14: 0.6456\n",
      "Task 15: 0.6185\n",
      "Task 16: 0.6123\n",
      "Task 17: 0.4920\n",
      "Task 18: 0.5514\n",
      "Task 19: 0.5917\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_roth, task)\n",
    "    mnist_rot.update_task(task)\n",
    "    acc1 = evaluate(basis_model_roth, mnist_rot.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 126,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_hybrid_mnist_rot'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 128,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "dict_keys(['supsup_mnist', 'supsup_mnist_sparsity_25', 'supsup_mnist_rot', 'basis_mnist', 'basis_mnist_sparsity_25', 'basis_mnist_rot', 'basis_hybrid_mnist', 'basis_hybrid_mnist_sparsity_25', 'basis_hybrid_mnist_rot'])"
      ]
     },
     "execution_count": 128,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "performance_map.keys()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 129,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 317,
   "metadata": {},
   "outputs": [],
   "source": [
    "df = pd.DataFrame(performance_map)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 318,
   "metadata": {},
   "outputs": [],
   "source": [
    "df.loc[:14, 'Seen'] = True\n",
    "df.loc[15:, 'Seen'] = False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 319,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style  type=\"text/css\" >\n",
       "#T_b3428_row0_col0,#T_b3428_row0_col2,#T_b3428_row0_col4,#T_b3428_row1_col0,#T_b3428_row1_col2,#T_b3428_row1_col4,#T_b3428_row2_col0,#T_b3428_row2_col2,#T_b3428_row2_col4,#T_b3428_row3_col0,#T_b3428_row3_col2,#T_b3428_row3_col4,#T_b3428_row4_col0,#T_b3428_row4_col2,#T_b3428_row4_col4,#T_b3428_row5_col0,#T_b3428_row5_col2,#T_b3428_row5_col4,#T_b3428_row6_col0,#T_b3428_row6_col2,#T_b3428_row6_col4,#T_b3428_row7_col0,#T_b3428_row7_col2,#T_b3428_row7_col4,#T_b3428_row8_col0,#T_b3428_row8_col2,#T_b3428_row8_col4,#T_b3428_row9_col0,#T_b3428_row9_col2,#T_b3428_row9_col4,#T_b3428_row10_col0,#T_b3428_row10_col2,#T_b3428_row10_col4,#T_b3428_row11_col0,#T_b3428_row11_col2,#T_b3428_row11_col4,#T_b3428_row12_col0,#T_b3428_row12_col2,#T_b3428_row12_col4,#T_b3428_row13_col0,#T_b3428_row13_col2,#T_b3428_row13_col4,#T_b3428_row14_col0,#T_b3428_row14_col2,#T_b3428_row14_col4,#T_b3428_row15_col0,#T_b3428_row16_col0,#T_b3428_row17_col0,#T_b3428_row18_col0,#T_b3428_row19_col0{\n",
       "            background-color:  #73a9cf;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row0_col1,#T_b3428_row0_col3,#T_b3428_row0_col5,#T_b3428_row1_col7,#T_b3428_row2_col1,#T_b3428_row2_col3,#T_b3428_row2_col5,#T_b3428_row4_col1,#T_b3428_row4_col3,#T_b3428_row4_col5,#T_b3428_row9_col1,#T_b3428_row9_col3,#T_b3428_row9_col5,#T_b3428_row10_col1,#T_b3428_row10_col3,#T_b3428_row10_col5,#T_b3428_row11_col1,#T_b3428_row11_col3,#T_b3428_row11_col5,#T_b3428_row13_col1,#T_b3428_row13_col3,#T_b3428_row13_col5,#T_b3428_row15_col1,#T_b3428_row17_col1,#T_b3428_row19_col1{\n",
       "            background-color:  #78abd0;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row0_col6,#T_b3428_row18_col5{\n",
       "            background-color:  #9fbad9;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row0_col7{\n",
       "            background-color:  #f5eef6;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row0_col8,#T_b3428_row0_col9,#T_b3428_row0_col10,#T_b3428_row1_col8,#T_b3428_row1_col9,#T_b3428_row2_col8,#T_b3428_row3_col9,#T_b3428_row3_col10,#T_b3428_row4_col8,#T_b3428_row4_col9,#T_b3428_row4_col10,#T_b3428_row5_col8,#T_b3428_row5_col9,#T_b3428_row6_col9,#T_b3428_row7_col8,#T_b3428_row7_col9,#T_b3428_row7_col10,#T_b3428_row8_col10,#T_b3428_row9_col8,#T_b3428_row9_col10,#T_b3428_row10_col9,#T_b3428_row10_col10,#T_b3428_row11_col8,#T_b3428_row11_col9,#T_b3428_row12_col9,#T_b3428_row12_col10,#T_b3428_row13_col9,#T_b3428_row13_col10,#T_b3428_row14_col8,#T_b3428_row14_col9,#T_b3428_row15_col7,#T_b3428_row16_col7,#T_b3428_row17_col7,#T_b3428_row18_col7,#T_b3428_row19_col7{\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row1_col1,#T_b3428_row1_col3,#T_b3428_row1_col5,#T_b3428_row3_col1,#T_b3428_row3_col3,#T_b3428_row3_col5,#T_b3428_row5_col1,#T_b3428_row5_col3,#T_b3428_row5_col5,#T_b3428_row6_col1,#T_b3428_row6_col3,#T_b3428_row6_col5,#T_b3428_row7_col1,#T_b3428_row7_col3,#T_b3428_row7_col5,#T_b3428_row8_col1,#T_b3428_row8_col3,#T_b3428_row8_col5,#T_b3428_row12_col1,#T_b3428_row12_col3,#T_b3428_row12_col5,#T_b3428_row14_col1,#T_b3428_row14_col3,#T_b3428_row14_col5,#T_b3428_row16_col1,#T_b3428_row18_col1{\n",
       "            background-color:  #76aad0;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row1_col6{\n",
       "            background-color:  #75a9cf;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row1_col10,#T_b3428_row6_col8,#T_b3428_row13_col8{\n",
       "            background-color:  #fdf5fa;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row2_col6,#T_b3428_row7_col6,#T_b3428_row9_col6{\n",
       "            background-color:  #9ebad9;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row2_col7{\n",
       "            background-color:  #f5eff6;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row2_col9,#T_b3428_row2_col10,#T_b3428_row5_col10,#T_b3428_row6_col10,#T_b3428_row8_col8,#T_b3428_row14_col10{\n",
       "            background-color:  #fef6fb;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row3_col6,#T_b3428_row5_col6,#T_b3428_row19_col6{\n",
       "            background-color:  #a1bbda;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row3_col7,#T_b3428_row11_col7{\n",
       "            background-color:  #f3edf5;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row3_col8{\n",
       "            background-color:  #faf3f9;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row4_col6,#T_b3428_row12_col6,#T_b3428_row14_col6{\n",
       "            background-color:  #9cb9d9;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row4_col7{\n",
       "            background-color:  #f2ecf5;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row5_col7,#T_b3428_row7_col7{\n",
       "            background-color:  #f4edf6;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row6_col6{\n",
       "            background-color:  #99b8d8;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row6_col7,#T_b3428_row10_col7{\n",
       "            background-color:  #f0eaf4;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row8_col6,#T_b3428_row10_col6,#T_b3428_row11_col6{\n",
       "            background-color:  #9ab8d8;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row8_col7{\n",
       "            background-color:  #eee9f3;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row8_col9,#T_b3428_row9_col9,#T_b3428_row10_col8,#T_b3428_row11_col10,#T_b3428_row12_col8{\n",
       "            background-color:  #fef6fa;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row9_col7,#T_b3428_row12_col7,#T_b3428_row18_col9{\n",
       "            background-color:  #f4eef6;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row13_col6,#T_b3428_row17_col4{\n",
       "            background-color:  #a4bcda;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row13_col7,#T_b3428_row16_col2{\n",
       "            background-color:  #ece7f2;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row14_col7,#T_b3428_row16_col9,#T_b3428_row19_col9{\n",
       "            background-color:  #f1ebf4;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row15_col2{\n",
       "            background-color:  #e3e0ee;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row15_col3{\n",
       "            background-color:  #d9d8ea;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row15_col4,#T_b3428_row16_col4{\n",
       "            background-color:  #93b5d6;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row15_col5{\n",
       "            background-color:  #a9bfdc;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row15_col6,#T_b3428_row18_col6{\n",
       "            background-color:  #abbfdc;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row15_col8{\n",
       "            background-color:  #fbf4f9;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row15_col9,#T_b3428_row17_col8,#T_b3428_row18_col8{\n",
       "            background-color:  #f8f1f8;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row15_col10{\n",
       "            background-color:  #adc1dd;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row16_col3,#T_b3428_row17_col2,#T_b3428_row19_col2{\n",
       "            background-color:  #e7e3f0;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row16_col5,#T_b3428_row16_col10{\n",
       "            background-color:  #b0c2de;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row16_col6,#T_b3428_row17_col6,#T_b3428_row19_col5{\n",
       "            background-color:  #a2bcda;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row16_col8,#T_b3428_row17_col9{\n",
       "            background-color:  #f7f0f7;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row17_col3{\n",
       "            background-color:  #e9e5f1;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row17_col5{\n",
       "            background-color:  #a5bddb;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row17_col10{\n",
       "            background-color:  #b1c2de;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row18_col2,#T_b3428_row18_col3{\n",
       "            background-color:  #dbdaeb;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row18_col4{\n",
       "            background-color:  #97b7d7;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row18_col10{\n",
       "            background-color:  #b8c6e0;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row19_col3{\n",
       "            background-color:  #dddbec;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row19_col4{\n",
       "            background-color:  #91b5d6;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row19_col8{\n",
       "            background-color:  #fcf4fa;\n",
       "            color:  #000000;\n",
       "        }#T_b3428_row19_col10{\n",
       "            background-color:  #b4c4df;\n",
       "            color:  #000000;\n",
       "        }</style><table id=\"T_b3428_\" ><thead>    <tr>        <th class=\"blank level0\" ></th>        <th class=\"col_heading level0 col0\" >supsup_mnist</th>        <th class=\"col_heading level0 col1\" >supsup_mnist_sparsity_25</th>        <th class=\"col_heading level0 col2\" >basis_mnist</th>        <th class=\"col_heading level0 col3\" >basis_mnist_sparsity_25</th>        <th class=\"col_heading level0 col4\" >basis_hybrid_mnist</th>        <th class=\"col_heading level0 col5\" >basis_hybrid_mnist_sparsity_25</th>        <th class=\"col_heading level0 col6\" >basis_hybrid_mnist_frozen</th>        <th class=\"col_heading level0 col7\" >basis_hybrid_mnist_frozen_sparsity25</th>        <th class=\"col_heading level0 col8\" >basis_mnist_task_zero_only</th>        <th class=\"col_heading level0 col9\" >basis_mnist_task_zero_only_sparsity_25</th>        <th class=\"col_heading level0 col10\" >basis_mnist_task_zero_only_hidden_sparsity_25</th>        <th class=\"col_heading level0 col11\" >Seen</th>    </tr>    <tr>        <th class=\"index_name level0\" >Task</th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>    </tr></thead><tbody>\n",
       "                <tr>\n",
       "                        <th id=\"T_b3428_level0_row0\" class=\"row_heading level0 row0\" >0</th>\n",
       "                        <td id=\"T_b3428_row0_col0\" class=\"data row0 col0\" >0.929500</td>\n",
       "                        <td id=\"T_b3428_row0_col1\" class=\"data row0 col1\" >0.911600</td>\n",
       "                        <td id=\"T_b3428_row0_col2\" class=\"data row0 col2\" >0.929500</td>\n",
       "                        <td id=\"T_b3428_row0_col3\" class=\"data row0 col3\" >0.911600</td>\n",
       "                        <td id=\"T_b3428_row0_col4\" class=\"data row0 col4\" >0.929500</td>\n",
       "                        <td id=\"T_b3428_row0_col5\" class=\"data row0 col5\" >0.911600</td>\n",
       "                        <td id=\"T_b3428_row0_col6\" class=\"data row0 col6\" >0.727900</td>\n",
       "                        <td id=\"T_b3428_row0_col7\" class=\"data row0 col7\" >0.127700</td>\n",
       "                        <td id=\"T_b3428_row0_col8\" class=\"data row0 col8\" >0.000000</td>\n",
       "                        <td id=\"T_b3428_row0_col9\" class=\"data row0 col9\" >0.002200</td>\n",
       "                        <td id=\"T_b3428_row0_col10\" class=\"data row0 col10\" >0.002700</td>\n",
       "                        <td id=\"T_b3428_row0_col11\" class=\"data row0 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row1\" class=\"row_heading level0 row1\" >1</th>\n",
       "                        <td id=\"T_b3428_row1_col0\" class=\"data row1 col0\" >0.926600</td>\n",
       "                        <td id=\"T_b3428_row1_col1\" class=\"data row1 col1\" >0.916600</td>\n",
       "                        <td id=\"T_b3428_row1_col2\" class=\"data row1 col2\" >0.926600</td>\n",
       "                        <td id=\"T_b3428_row1_col3\" class=\"data row1 col3\" >0.916600</td>\n",
       "                        <td id=\"T_b3428_row1_col4\" class=\"data row1 col4\" >0.926600</td>\n",
       "                        <td id=\"T_b3428_row1_col5\" class=\"data row1 col5\" >0.916600</td>\n",
       "                        <td id=\"T_b3428_row1_col6\" class=\"data row1 col6\" >0.923200</td>\n",
       "                        <td id=\"T_b3428_row1_col7\" class=\"data row1 col7\" >0.910100</td>\n",
       "                        <td id=\"T_b3428_row1_col8\" class=\"data row1 col8\" >0.001400</td>\n",
       "                        <td id=\"T_b3428_row1_col9\" class=\"data row1 col9\" >0.003700</td>\n",
       "                        <td id=\"T_b3428_row1_col10\" class=\"data row1 col10\" >0.027000</td>\n",
       "                        <td id=\"T_b3428_row1_col11\" class=\"data row1 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row2\" class=\"row_heading level0 row2\" >2</th>\n",
       "                        <td id=\"T_b3428_row2_col0\" class=\"data row2 col0\" >0.926400</td>\n",
       "                        <td id=\"T_b3428_row2_col1\" class=\"data row2 col1\" >0.911100</td>\n",
       "                        <td id=\"T_b3428_row2_col2\" class=\"data row2 col2\" >0.926400</td>\n",
       "                        <td id=\"T_b3428_row2_col3\" class=\"data row2 col3\" >0.911100</td>\n",
       "                        <td id=\"T_b3428_row2_col4\" class=\"data row2 col4\" >0.926400</td>\n",
       "                        <td id=\"T_b3428_row2_col5\" class=\"data row2 col5\" >0.911100</td>\n",
       "                        <td id=\"T_b3428_row2_col6\" class=\"data row2 col6\" >0.735400</td>\n",
       "                        <td id=\"T_b3428_row2_col7\" class=\"data row2 col7\" >0.126300</td>\n",
       "                        <td id=\"T_b3428_row2_col8\" class=\"data row2 col8\" >0.005600</td>\n",
       "                        <td id=\"T_b3428_row2_col9\" class=\"data row2 col9\" >0.014300</td>\n",
       "                        <td id=\"T_b3428_row2_col10\" class=\"data row2 col10\" >0.014300</td>\n",
       "                        <td id=\"T_b3428_row2_col11\" class=\"data row2 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row3\" class=\"row_heading level0 row3\" >3</th>\n",
       "                        <td id=\"T_b3428_row3_col0\" class=\"data row3 col0\" >0.924700</td>\n",
       "                        <td id=\"T_b3428_row3_col1\" class=\"data row3 col1\" >0.916600</td>\n",
       "                        <td id=\"T_b3428_row3_col2\" class=\"data row3 col2\" >0.924700</td>\n",
       "                        <td id=\"T_b3428_row3_col3\" class=\"data row3 col3\" >0.916600</td>\n",
       "                        <td id=\"T_b3428_row3_col4\" class=\"data row3 col4\" >0.924700</td>\n",
       "                        <td id=\"T_b3428_row3_col5\" class=\"data row3 col5\" >0.916600</td>\n",
       "                        <td id=\"T_b3428_row3_col6\" class=\"data row3 col6\" >0.715700</td>\n",
       "                        <td id=\"T_b3428_row3_col7\" class=\"data row3 col7\" >0.146200</td>\n",
       "                        <td id=\"T_b3428_row3_col8\" class=\"data row3 col8\" >0.064800</td>\n",
       "                        <td id=\"T_b3428_row3_col9\" class=\"data row3 col9\" >0.001600</td>\n",
       "                        <td id=\"T_b3428_row3_col10\" class=\"data row3 col10\" >0.000100</td>\n",
       "                        <td id=\"T_b3428_row3_col11\" class=\"data row3 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row4\" class=\"row_heading level0 row4\" >4</th>\n",
       "                        <td id=\"T_b3428_row4_col0\" class=\"data row4 col0\" >0.928100</td>\n",
       "                        <td id=\"T_b3428_row4_col1\" class=\"data row4 col1\" >0.912500</td>\n",
       "                        <td id=\"T_b3428_row4_col2\" class=\"data row4 col2\" >0.928100</td>\n",
       "                        <td id=\"T_b3428_row4_col3\" class=\"data row4 col3\" >0.912500</td>\n",
       "                        <td id=\"T_b3428_row4_col4\" class=\"data row4 col4\" >0.928100</td>\n",
       "                        <td id=\"T_b3428_row4_col5\" class=\"data row4 col5\" >0.912500</td>\n",
       "                        <td id=\"T_b3428_row4_col6\" class=\"data row4 col6\" >0.747100</td>\n",
       "                        <td id=\"T_b3428_row4_col7\" class=\"data row4 col7\" >0.167300</td>\n",
       "                        <td id=\"T_b3428_row4_col8\" class=\"data row4 col8\" >0.001700</td>\n",
       "                        <td id=\"T_b3428_row4_col9\" class=\"data row4 col9\" >0.008400</td>\n",
       "                        <td id=\"T_b3428_row4_col10\" class=\"data row4 col10\" >0.005400</td>\n",
       "                        <td id=\"T_b3428_row4_col11\" class=\"data row4 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row5\" class=\"row_heading level0 row5\" >5</th>\n",
       "                        <td id=\"T_b3428_row5_col0\" class=\"data row5 col0\" >0.922900</td>\n",
       "                        <td id=\"T_b3428_row5_col1\" class=\"data row5 col1\" >0.912500</td>\n",
       "                        <td id=\"T_b3428_row5_col2\" class=\"data row5 col2\" >0.922900</td>\n",
       "                        <td id=\"T_b3428_row5_col3\" class=\"data row5 col3\" >0.912500</td>\n",
       "                        <td id=\"T_b3428_row5_col4\" class=\"data row5 col4\" >0.922900</td>\n",
       "                        <td id=\"T_b3428_row5_col5\" class=\"data row5 col5\" >0.912500</td>\n",
       "                        <td id=\"T_b3428_row5_col6\" class=\"data row5 col6\" >0.720000</td>\n",
       "                        <td id=\"T_b3428_row5_col7\" class=\"data row5 col7\" >0.139900</td>\n",
       "                        <td id=\"T_b3428_row5_col8\" class=\"data row5 col8\" >0.002300</td>\n",
       "                        <td id=\"T_b3428_row5_col9\" class=\"data row5 col9\" >0.000100</td>\n",
       "                        <td id=\"T_b3428_row5_col10\" class=\"data row5 col10\" >0.009800</td>\n",
       "                        <td id=\"T_b3428_row5_col11\" class=\"data row5 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row6\" class=\"row_heading level0 row6\" >6</th>\n",
       "                        <td id=\"T_b3428_row6_col0\" class=\"data row6 col0\" >0.926200</td>\n",
       "                        <td id=\"T_b3428_row6_col1\" class=\"data row6 col1\" >0.913100</td>\n",
       "                        <td id=\"T_b3428_row6_col2\" class=\"data row6 col2\" >0.926200</td>\n",
       "                        <td id=\"T_b3428_row6_col3\" class=\"data row6 col3\" >0.913100</td>\n",
       "                        <td id=\"T_b3428_row6_col4\" class=\"data row6 col4\" >0.926200</td>\n",
       "                        <td id=\"T_b3428_row6_col5\" class=\"data row6 col5\" >0.913100</td>\n",
       "                        <td id=\"T_b3428_row6_col6\" class=\"data row6 col6\" >0.754300</td>\n",
       "                        <td id=\"T_b3428_row6_col7\" class=\"data row6 col7\" >0.186500</td>\n",
       "                        <td id=\"T_b3428_row6_col8\" class=\"data row6 col8\" >0.024500</td>\n",
       "                        <td id=\"T_b3428_row6_col9\" class=\"data row6 col9\" >0.000700</td>\n",
       "                        <td id=\"T_b3428_row6_col10\" class=\"data row6 col10\" >0.010800</td>\n",
       "                        <td id=\"T_b3428_row6_col11\" class=\"data row6 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row7\" class=\"row_heading level0 row7\" >7</th>\n",
       "                        <td id=\"T_b3428_row7_col0\" class=\"data row7 col0\" >0.924200</td>\n",
       "                        <td id=\"T_b3428_row7_col1\" class=\"data row7 col1\" >0.911900</td>\n",
       "                        <td id=\"T_b3428_row7_col2\" class=\"data row7 col2\" >0.924200</td>\n",
       "                        <td id=\"T_b3428_row7_col3\" class=\"data row7 col3\" >0.911900</td>\n",
       "                        <td id=\"T_b3428_row7_col4\" class=\"data row7 col4\" >0.924200</td>\n",
       "                        <td id=\"T_b3428_row7_col5\" class=\"data row7 col5\" >0.911900</td>\n",
       "                        <td id=\"T_b3428_row7_col6\" class=\"data row7 col6\" >0.729800</td>\n",
       "                        <td id=\"T_b3428_row7_col7\" class=\"data row7 col7\" >0.139900</td>\n",
       "                        <td id=\"T_b3428_row7_col8\" class=\"data row7 col8\" >0.005600</td>\n",
       "                        <td id=\"T_b3428_row7_col9\" class=\"data row7 col9\" >0.003000</td>\n",
       "                        <td id=\"T_b3428_row7_col10\" class=\"data row7 col10\" >0.000500</td>\n",
       "                        <td id=\"T_b3428_row7_col11\" class=\"data row7 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row8\" class=\"row_heading level0 row8\" >8</th>\n",
       "                        <td id=\"T_b3428_row8_col0\" class=\"data row8 col0\" >0.926300</td>\n",
       "                        <td id=\"T_b3428_row8_col1\" class=\"data row8 col1\" >0.912600</td>\n",
       "                        <td id=\"T_b3428_row8_col2\" class=\"data row8 col2\" >0.926300</td>\n",
       "                        <td id=\"T_b3428_row8_col3\" class=\"data row8 col3\" >0.912600</td>\n",
       "                        <td id=\"T_b3428_row8_col4\" class=\"data row8 col4\" >0.926300</td>\n",
       "                        <td id=\"T_b3428_row8_col5\" class=\"data row8 col5\" >0.912600</td>\n",
       "                        <td id=\"T_b3428_row8_col6\" class=\"data row8 col6\" >0.747000</td>\n",
       "                        <td id=\"T_b3428_row8_col7\" class=\"data row8 col7\" >0.207500</td>\n",
       "                        <td id=\"T_b3428_row8_col8\" class=\"data row8 col8\" >0.012100</td>\n",
       "                        <td id=\"T_b3428_row8_col9\" class=\"data row8 col9\" >0.018400</td>\n",
       "                        <td id=\"T_b3428_row8_col10\" class=\"data row8 col10\" >0.002900</td>\n",
       "                        <td id=\"T_b3428_row8_col11\" class=\"data row8 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row9\" class=\"row_heading level0 row9\" >9</th>\n",
       "                        <td id=\"T_b3428_row9_col0\" class=\"data row9 col0\" >0.926200</td>\n",
       "                        <td id=\"T_b3428_row9_col1\" class=\"data row9 col1\" >0.905800</td>\n",
       "                        <td id=\"T_b3428_row9_col2\" class=\"data row9 col2\" >0.926200</td>\n",
       "                        <td id=\"T_b3428_row9_col3\" class=\"data row9 col3\" >0.905800</td>\n",
       "                        <td id=\"T_b3428_row9_col4\" class=\"data row9 col4\" >0.926200</td>\n",
       "                        <td id=\"T_b3428_row9_col5\" class=\"data row9 col5\" >0.905800</td>\n",
       "                        <td id=\"T_b3428_row9_col6\" class=\"data row9 col6\" >0.737800</td>\n",
       "                        <td id=\"T_b3428_row9_col7\" class=\"data row9 col7\" >0.131200</td>\n",
       "                        <td id=\"T_b3428_row9_col8\" class=\"data row9 col8\" >0.001100</td>\n",
       "                        <td id=\"T_b3428_row9_col9\" class=\"data row9 col9\" >0.019300</td>\n",
       "                        <td id=\"T_b3428_row9_col10\" class=\"data row9 col10\" >0.002300</td>\n",
       "                        <td id=\"T_b3428_row9_col11\" class=\"data row9 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row10\" class=\"row_heading level0 row10\" >10</th>\n",
       "                        <td id=\"T_b3428_row10_col0\" class=\"data row10 col0\" >0.924300</td>\n",
       "                        <td id=\"T_b3428_row10_col1\" class=\"data row10 col1\" >0.909300</td>\n",
       "                        <td id=\"T_b3428_row10_col2\" class=\"data row10 col2\" >0.924300</td>\n",
       "                        <td id=\"T_b3428_row10_col3\" class=\"data row10 col3\" >0.909300</td>\n",
       "                        <td id=\"T_b3428_row10_col4\" class=\"data row10 col4\" >0.924300</td>\n",
       "                        <td id=\"T_b3428_row10_col5\" class=\"data row10 col5\" >0.909300</td>\n",
       "                        <td id=\"T_b3428_row10_col6\" class=\"data row10 col6\" >0.747800</td>\n",
       "                        <td id=\"T_b3428_row10_col7\" class=\"data row10 col7\" >0.191600</td>\n",
       "                        <td id=\"T_b3428_row10_col8\" class=\"data row10 col8\" >0.018100</td>\n",
       "                        <td id=\"T_b3428_row10_col9\" class=\"data row10 col9\" >0.001000</td>\n",
       "                        <td id=\"T_b3428_row10_col10\" class=\"data row10 col10\" >0.002000</td>\n",
       "                        <td id=\"T_b3428_row10_col11\" class=\"data row10 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row11\" class=\"row_heading level0 row11\" >11</th>\n",
       "                        <td id=\"T_b3428_row11_col0\" class=\"data row11 col0\" >0.925500</td>\n",
       "                        <td id=\"T_b3428_row11_col1\" class=\"data row11 col1\" >0.910800</td>\n",
       "                        <td id=\"T_b3428_row11_col2\" class=\"data row11 col2\" >0.925500</td>\n",
       "                        <td id=\"T_b3428_row11_col3\" class=\"data row11 col3\" >0.910800</td>\n",
       "                        <td id=\"T_b3428_row11_col4\" class=\"data row11 col4\" >0.925500</td>\n",
       "                        <td id=\"T_b3428_row11_col5\" class=\"data row11 col5\" >0.910800</td>\n",
       "                        <td id=\"T_b3428_row11_col6\" class=\"data row11 col6\" >0.747100</td>\n",
       "                        <td id=\"T_b3428_row11_col7\" class=\"data row11 col7\" >0.151500</td>\n",
       "                        <td id=\"T_b3428_row11_col8\" class=\"data row11 col8\" >0.001200</td>\n",
       "                        <td id=\"T_b3428_row11_col9\" class=\"data row11 col9\" >0.006700</td>\n",
       "                        <td id=\"T_b3428_row11_col10\" class=\"data row11 col10\" >0.017300</td>\n",
       "                        <td id=\"T_b3428_row11_col11\" class=\"data row11 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row12\" class=\"row_heading level0 row12\" >12</th>\n",
       "                        <td id=\"T_b3428_row12_col0\" class=\"data row12 col0\" >0.925500</td>\n",
       "                        <td id=\"T_b3428_row12_col1\" class=\"data row12 col1\" >0.912700</td>\n",
       "                        <td id=\"T_b3428_row12_col2\" class=\"data row12 col2\" >0.925500</td>\n",
       "                        <td id=\"T_b3428_row12_col3\" class=\"data row12 col3\" >0.912700</td>\n",
       "                        <td id=\"T_b3428_row12_col4\" class=\"data row12 col4\" >0.925500</td>\n",
       "                        <td id=\"T_b3428_row12_col5\" class=\"data row12 col5\" >0.912700</td>\n",
       "                        <td id=\"T_b3428_row12_col6\" class=\"data row12 col6\" >0.740400</td>\n",
       "                        <td id=\"T_b3428_row12_col7\" class=\"data row12 col7\" >0.131400</td>\n",
       "                        <td id=\"T_b3428_row12_col8\" class=\"data row12 col8\" >0.019300</td>\n",
       "                        <td id=\"T_b3428_row12_col9\" class=\"data row12 col9\" >0.000900</td>\n",
       "                        <td id=\"T_b3428_row12_col10\" class=\"data row12 col10\" >0.003100</td>\n",
       "                        <td id=\"T_b3428_row12_col11\" class=\"data row12 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row13\" class=\"row_heading level0 row13\" >13</th>\n",
       "                        <td id=\"T_b3428_row13_col0\" class=\"data row13 col0\" >0.928700</td>\n",
       "                        <td id=\"T_b3428_row13_col1\" class=\"data row13 col1\" >0.913500</td>\n",
       "                        <td id=\"T_b3428_row13_col2\" class=\"data row13 col2\" >0.928700</td>\n",
       "                        <td id=\"T_b3428_row13_col3\" class=\"data row13 col3\" >0.913500</td>\n",
       "                        <td id=\"T_b3428_row13_col4\" class=\"data row13 col4\" >0.928700</td>\n",
       "                        <td id=\"T_b3428_row13_col5\" class=\"data row13 col5\" >0.913500</td>\n",
       "                        <td id=\"T_b3428_row13_col6\" class=\"data row13 col6\" >0.707500</td>\n",
       "                        <td id=\"T_b3428_row13_col7\" class=\"data row13 col7\" >0.235900</td>\n",
       "                        <td id=\"T_b3428_row13_col8\" class=\"data row13 col8\" >0.029200</td>\n",
       "                        <td id=\"T_b3428_row13_col9\" class=\"data row13 col9\" >0.002400</td>\n",
       "                        <td id=\"T_b3428_row13_col10\" class=\"data row13 col10\" >0.001100</td>\n",
       "                        <td id=\"T_b3428_row13_col11\" class=\"data row13 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row14\" class=\"row_heading level0 row14\" >14</th>\n",
       "                        <td id=\"T_b3428_row14_col0\" class=\"data row14 col0\" >0.924200</td>\n",
       "                        <td id=\"T_b3428_row14_col1\" class=\"data row14 col1\" >0.911800</td>\n",
       "                        <td id=\"T_b3428_row14_col2\" class=\"data row14 col2\" >0.924200</td>\n",
       "                        <td id=\"T_b3428_row14_col3\" class=\"data row14 col3\" >0.911800</td>\n",
       "                        <td id=\"T_b3428_row14_col4\" class=\"data row14 col4\" >0.924200</td>\n",
       "                        <td id=\"T_b3428_row14_col5\" class=\"data row14 col5\" >0.911800</td>\n",
       "                        <td id=\"T_b3428_row14_col6\" class=\"data row14 col6\" >0.741000</td>\n",
       "                        <td id=\"T_b3428_row14_col7\" class=\"data row14 col7\" >0.189400</td>\n",
       "                        <td id=\"T_b3428_row14_col8\" class=\"data row14 col8\" >0.011700</td>\n",
       "                        <td id=\"T_b3428_row14_col9\" class=\"data row14 col9\" >0.011100</td>\n",
       "                        <td id=\"T_b3428_row14_col10\" class=\"data row14 col10\" >0.021500</td>\n",
       "                        <td id=\"T_b3428_row14_col11\" class=\"data row14 col11\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row15\" class=\"row_heading level0 row15\" >15</th>\n",
       "                        <td id=\"T_b3428_row15_col0\" class=\"data row15 col0\" >0.927700</td>\n",
       "                        <td id=\"T_b3428_row15_col1\" class=\"data row15 col1\" >0.912300</td>\n",
       "                        <td id=\"T_b3428_row15_col2\" class=\"data row15 col2\" >0.421100</td>\n",
       "                        <td id=\"T_b3428_row15_col3\" class=\"data row15 col3\" >0.483800</td>\n",
       "                        <td id=\"T_b3428_row15_col4\" class=\"data row15 col4\" >0.811800</td>\n",
       "                        <td id=\"T_b3428_row15_col5\" class=\"data row15 col5\" >0.725700</td>\n",
       "                        <td id=\"T_b3428_row15_col6\" class=\"data row15 col6\" >0.714900</td>\n",
       "                        <td id=\"T_b3428_row15_col7\" class=\"data row15 col7\" >0.168400</td>\n",
       "                        <td id=\"T_b3428_row15_col8\" class=\"data row15 col8\" >0.207300</td>\n",
       "                        <td id=\"T_b3428_row15_col9\" class=\"data row15 col9\" >0.244000</td>\n",
       "                        <td id=\"T_b3428_row15_col10\" class=\"data row15 col10\" >0.702900</td>\n",
       "                        <td id=\"T_b3428_row15_col11\" class=\"data row15 col11\" >False</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row16\" class=\"row_heading level0 row16\" >16</th>\n",
       "                        <td id=\"T_b3428_row16_col0\" class=\"data row16 col0\" >0.923600</td>\n",
       "                        <td id=\"T_b3428_row16_col1\" class=\"data row16 col1\" >0.913000</td>\n",
       "                        <td id=\"T_b3428_row16_col2\" class=\"data row16 col2\" >0.368700</td>\n",
       "                        <td id=\"T_b3428_row16_col3\" class=\"data row16 col3\" >0.402800</td>\n",
       "                        <td id=\"T_b3428_row16_col4\" class=\"data row16 col4\" >0.811800</td>\n",
       "                        <td id=\"T_b3428_row16_col5\" class=\"data row16 col5\" >0.694600</td>\n",
       "                        <td id=\"T_b3428_row16_col6\" class=\"data row16 col6\" >0.751600</td>\n",
       "                        <td id=\"T_b3428_row16_col7\" class=\"data row16 col7\" >0.176100</td>\n",
       "                        <td id=\"T_b3428_row16_col8\" class=\"data row16 col8\" >0.255700</td>\n",
       "                        <td id=\"T_b3428_row16_col9\" class=\"data row16 col9\" >0.316500</td>\n",
       "                        <td id=\"T_b3428_row16_col10\" class=\"data row16 col10\" >0.695500</td>\n",
       "                        <td id=\"T_b3428_row16_col11\" class=\"data row16 col11\" >False</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row17\" class=\"row_heading level0 row17\" >17</th>\n",
       "                        <td id=\"T_b3428_row17_col0\" class=\"data row17 col0\" >0.926600</td>\n",
       "                        <td id=\"T_b3428_row17_col1\" class=\"data row17 col1\" >0.912000</td>\n",
       "                        <td id=\"T_b3428_row17_col2\" class=\"data row17 col2\" >0.397500</td>\n",
       "                        <td id=\"T_b3428_row17_col3\" class=\"data row17 col3\" >0.382800</td>\n",
       "                        <td id=\"T_b3428_row17_col4\" class=\"data row17 col4\" >0.747800</td>\n",
       "                        <td id=\"T_b3428_row17_col5\" class=\"data row17 col5\" >0.741800</td>\n",
       "                        <td id=\"T_b3428_row17_col6\" class=\"data row17 col6\" >0.751600</td>\n",
       "                        <td id=\"T_b3428_row17_col7\" class=\"data row17 col7\" >0.172500</td>\n",
       "                        <td id=\"T_b3428_row17_col8\" class=\"data row17 col8\" >0.244200</td>\n",
       "                        <td id=\"T_b3428_row17_col9\" class=\"data row17 col9\" >0.250800</td>\n",
       "                        <td id=\"T_b3428_row17_col10\" class=\"data row17 col10\" >0.688100</td>\n",
       "                        <td id=\"T_b3428_row17_col11\" class=\"data row17 col11\" >False</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row18\" class=\"row_heading level0 row18\" >18</th>\n",
       "                        <td id=\"T_b3428_row18_col0\" class=\"data row18 col0\" >0.922400</td>\n",
       "                        <td id=\"T_b3428_row18_col1\" class=\"data row18 col1\" >0.913400</td>\n",
       "                        <td id=\"T_b3428_row18_col2\" class=\"data row18 col2\" >0.476400</td>\n",
       "                        <td id=\"T_b3428_row18_col3\" class=\"data row18 col3\" >0.480100</td>\n",
       "                        <td id=\"T_b3428_row18_col4\" class=\"data row18 col4\" >0.791000</td>\n",
       "                        <td id=\"T_b3428_row18_col5\" class=\"data row18 col5\" >0.765600</td>\n",
       "                        <td id=\"T_b3428_row18_col6\" class=\"data row18 col6\" >0.716900</td>\n",
       "                        <td id=\"T_b3428_row18_col7\" class=\"data row18 col7\" >0.180700</td>\n",
       "                        <td id=\"T_b3428_row18_col8\" class=\"data row18 col8\" >0.252700</td>\n",
       "                        <td id=\"T_b3428_row18_col9\" class=\"data row18 col9\" >0.289300</td>\n",
       "                        <td id=\"T_b3428_row18_col10\" class=\"data row18 col10\" >0.657500</td>\n",
       "                        <td id=\"T_b3428_row18_col11\" class=\"data row18 col11\" >False</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_b3428_level0_row19\" class=\"row_heading level0 row19\" >19</th>\n",
       "                        <td id=\"T_b3428_row19_col0\" class=\"data row19 col0\" >0.930000</td>\n",
       "                        <td id=\"T_b3428_row19_col1\" class=\"data row19 col1\" >0.913400</td>\n",
       "                        <td id=\"T_b3428_row19_col2\" class=\"data row19 col2\" >0.406400</td>\n",
       "                        <td id=\"T_b3428_row19_col3\" class=\"data row19 col3\" >0.471700</td>\n",
       "                        <td id=\"T_b3428_row19_col4\" class=\"data row19 col4\" >0.819800</td>\n",
       "                        <td id=\"T_b3428_row19_col5\" class=\"data row19 col5\" >0.759700</td>\n",
       "                        <td id=\"T_b3428_row19_col6\" class=\"data row19 col6\" >0.761300</td>\n",
       "                        <td id=\"T_b3428_row19_col7\" class=\"data row19 col7\" >0.184000</td>\n",
       "                        <td id=\"T_b3428_row19_col8\" class=\"data row19 col8\" >0.217200</td>\n",
       "                        <td id=\"T_b3428_row19_col9\" class=\"data row19 col9\" >0.324200</td>\n",
       "                        <td id=\"T_b3428_row19_col10\" class=\"data row19 col10\" >0.683600</td>\n",
       "                        <td id=\"T_b3428_row19_col11\" class=\"data row19 col11\" >False</td>\n",
       "            </tr>\n",
       "    </tbody></table>"
      ],
      "text/plain": [
       "<pandas.io.formats.style.Styler at 0x7eff288e5810>"
      ]
     },
     "execution_count": 319,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df[[c for c in df.columns if not 'rot' in c]].reset_index().rename(columns={'index': 'Task'}).set_index('Task').style.background_gradient(low=df.min().min(), high=df.max().max(), axis=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 98,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style  type=\"text/css\" >\n",
       "    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row0_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row0_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row0_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row1_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row1_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row1_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row2_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row2_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row2_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row3_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row3_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row3_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row4_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row4_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row4_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row5_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row5_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row5_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row6_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row6_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row6_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row7_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row7_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row7_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row8_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row8_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row8_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row9_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row9_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row9_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row10_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row10_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row10_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row11_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row11_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row11_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row12_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row12_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row12_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row13_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row13_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row13_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row14_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row14_col1 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row14_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row15_col0 {\n",
       "            background-color:  #75a9cf;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row15_col1 {\n",
       "            background-color:  #73a9cf;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row15_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row16_col0 {\n",
       "            background-color:  #7bacd1;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row16_col1 {\n",
       "            background-color:  #73a9cf;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row16_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row17_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row17_col1 {\n",
       "            background-color:  #acc0dd;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row17_col2 {\n",
       "            background-color:  #73a9cf;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row18_col0 {\n",
       "            background-color:  #73a9cf;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row18_col1 {\n",
       "            background-color:  #d1d2e6;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row18_col2 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row19_col0 {\n",
       "            background-color:  #fff7fb;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row19_col1 {\n",
       "            background-color:  #73a9cf;\n",
       "            color:  #000000;\n",
       "        }    #T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row19_col2 {\n",
       "            background-color:  #86b0d3;\n",
       "            color:  #000000;\n",
       "        }</style><table id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4\" ><thead>    <tr>        <th class=\"blank level0\" ></th>        <th class=\"col_heading level0 col0\" >supsup_mnist_rot</th>        <th class=\"col_heading level0 col1\" >basis_mnist_rot</th>        <th class=\"col_heading level0 col2\" >basis_hybrid_mnist_rot</th>        <th class=\"col_heading level0 col3\" >Seen</th>    </tr>    <tr>        <th class=\"index_name level0\" >Task</th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>        <th class=\"blank\" ></th>    </tr></thead><tbody>\n",
       "                <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row0\" class=\"row_heading level0 row0\" >0</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row0_col0\" class=\"data row0 col0\" >0.776300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row0_col1\" class=\"data row0 col1\" >0.776300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row0_col2\" class=\"data row0 col2\" >0.776300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row0_col3\" class=\"data row0 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row1\" class=\"row_heading level0 row1\" >1</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row1_col0\" class=\"data row1 col0\" >0.779800</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row1_col1\" class=\"data row1 col1\" >0.779800</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row1_col2\" class=\"data row1 col2\" >0.779800</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row1_col3\" class=\"data row1 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row2\" class=\"row_heading level0 row2\" >2</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row2_col0\" class=\"data row2 col0\" >0.783900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row2_col1\" class=\"data row2 col1\" >0.783900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row2_col2\" class=\"data row2 col2\" >0.783900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row2_col3\" class=\"data row2 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row3\" class=\"row_heading level0 row3\" >3</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row3_col0\" class=\"data row3 col0\" >0.771300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row3_col1\" class=\"data row3 col1\" >0.771300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row3_col2\" class=\"data row3 col2\" >0.771300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row3_col3\" class=\"data row3 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row4\" class=\"row_heading level0 row4\" >4</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row4_col0\" class=\"data row4 col0\" >0.762600</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row4_col1\" class=\"data row4 col1\" >0.762600</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row4_col2\" class=\"data row4 col2\" >0.762600</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row4_col3\" class=\"data row4 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row5\" class=\"row_heading level0 row5\" >5</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row5_col0\" class=\"data row5 col0\" >0.772700</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row5_col1\" class=\"data row5 col1\" >0.772700</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row5_col2\" class=\"data row5 col2\" >0.772700</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row5_col3\" class=\"data row5 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row6\" class=\"row_heading level0 row6\" >6</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row6_col0\" class=\"data row6 col0\" >0.782200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row6_col1\" class=\"data row6 col1\" >0.782200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row6_col2\" class=\"data row6 col2\" >0.782200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row6_col3\" class=\"data row6 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row7\" class=\"row_heading level0 row7\" >7</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row7_col0\" class=\"data row7 col0\" >0.775200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row7_col1\" class=\"data row7 col1\" >0.775200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row7_col2\" class=\"data row7 col2\" >0.775200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row7_col3\" class=\"data row7 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row8\" class=\"row_heading level0 row8\" >8</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row8_col0\" class=\"data row8 col0\" >0.785200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row8_col1\" class=\"data row8 col1\" >0.785200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row8_col2\" class=\"data row8 col2\" >0.785200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row8_col3\" class=\"data row8 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row9\" class=\"row_heading level0 row9\" >9</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row9_col0\" class=\"data row9 col0\" >0.781300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row9_col1\" class=\"data row9 col1\" >0.781300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row9_col2\" class=\"data row9 col2\" >0.781300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row9_col3\" class=\"data row9 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row10\" class=\"row_heading level0 row10\" >10</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row10_col0\" class=\"data row10 col0\" >0.778700</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row10_col1\" class=\"data row10 col1\" >0.778700</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row10_col2\" class=\"data row10 col2\" >0.778700</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row10_col3\" class=\"data row10 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row11\" class=\"row_heading level0 row11\" >11</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row11_col0\" class=\"data row11 col0\" >0.770900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row11_col1\" class=\"data row11 col1\" >0.770900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row11_col2\" class=\"data row11 col2\" >0.770900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row11_col3\" class=\"data row11 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row12\" class=\"row_heading level0 row12\" >12</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row12_col0\" class=\"data row12 col0\" >0.781500</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row12_col1\" class=\"data row12 col1\" >0.781500</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row12_col2\" class=\"data row12 col2\" >0.781500</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row12_col3\" class=\"data row12 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row13\" class=\"row_heading level0 row13\" >13</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row13_col0\" class=\"data row13 col0\" >0.781900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row13_col1\" class=\"data row13 col1\" >0.781900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row13_col2\" class=\"data row13 col2\" >0.781900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row13_col3\" class=\"data row13 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row14\" class=\"row_heading level0 row14\" >14</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row14_col0\" class=\"data row14 col0\" >0.770200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row14_col1\" class=\"data row14 col1\" >0.770200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row14_col2\" class=\"data row14 col2\" >0.770200</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row14_col3\" class=\"data row14 col3\" >True</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row15\" class=\"row_heading level0 row15\" >15</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row15_col0\" class=\"data row15 col0\" >0.784500</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row15_col1\" class=\"data row15 col1\" >0.785000</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row15_col2\" class=\"data row15 col2\" >0.713700</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row15_col3\" class=\"data row15 col3\" >False</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row16\" class=\"row_heading level0 row16\" >16</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row16_col0\" class=\"data row16 col0\" >0.783600</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row16_col1\" class=\"data row16 col1\" >0.789100</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row16_col2\" class=\"data row16 col2\" >0.617800</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row16_col3\" class=\"data row16 col3\" >False</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row17\" class=\"row_heading level0 row17\" >17</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row17_col0\" class=\"data row17 col0\" >0.770800</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row17_col1\" class=\"data row17 col1\" >0.784000</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row17_col2\" class=\"data row17 col2\" >0.789300</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row17_col3\" class=\"data row17 col3\" >False</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row18\" class=\"row_heading level0 row18\" >18</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row18_col0\" class=\"data row18 col0\" >0.783900</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row18_col1\" class=\"data row18 col1\" >0.779700</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row18_col2\" class=\"data row18 col2\" >0.775500</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row18_col3\" class=\"data row18 col3\" >False</td>\n",
       "            </tr>\n",
       "            <tr>\n",
       "                        <th id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4level0_row19\" class=\"row_heading level0 row19\" >19</th>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row19_col0\" class=\"data row19 col0\" >0.772800</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row19_col1\" class=\"data row19 col1\" >0.782600</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row19_col2\" class=\"data row19 col2\" >0.781700</td>\n",
       "                        <td id=\"T_cd65e4b6_a616_11eb_b07c_8c8590bf39c4row19_col3\" class=\"data row19 col3\" >False</td>\n",
       "            </tr>\n",
       "    </tbody></table>"
      ],
      "text/plain": [
       "<pandas.io.formats.style.Styler at 0x7fd1111a15d0>"
      ]
     },
     "execution_count": 98,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df[[c for c in df.columns if 'rot' in c] + ['Seen']].reset_index().rename(columns={'index': 'Task'}).set_index('Task').style.background_gradient(low=df.min().min(), high=df.max().max(), axis=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Cross task analysts, retraining last layer) -- Using mask from task 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 133,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_hidf = BasisHiddenOnlyFrozenMultitaskFC(hidden_size=300,\n",
    "                            num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 134,
   "metadata": {},
   "outputs": [],
   "source": [
    "custom_state_dict = basis_model_hidf.state_dict().copy()\n",
    "custom_state_dict.update({k:v for k,v in model.state_dict().items() if k in custom_state_dict.keys()})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 135,
   "metadata": {},
   "outputs": [],
   "source": [
    "for k in custom_state_dict.keys():\n",
    "    if k.startswith('model.0.scores.'):\n",
    "        custom_state_dict[k] = model.state_dict()['model.0.scores.1']\n",
    "    elif k.startswith('model.2.scores.'):\n",
    "        custom_state_dict[k] = model.state_dict()['model.2.scores.1']\n",
    "    elif k.startswith('model.0.basis_alphas.'):\n",
    "        custom_state_dict[k] = custom_state_dict['model.0.basis_alphas.1'] \n",
    "    elif k.startswith('model.2.basis_alphas.'):\n",
    "        custom_state_dict[k] = custom_state_dict['model.2.basis_alphas.1']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 137,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n"
     ]
    }
   ],
   "source": [
    "basis_model_hidf.load_state_dict(custom_state_dict, False)\n",
    "cache_masks(basis_model_hidf)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 138,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3659, Acc@1 0.6859\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7279\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3937, Acc@1 0.9176\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9232\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "\n",
      "Training for task 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3747, Acc@1 0.6618\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7354\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "\n",
      "Training for task 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.4363, Acc@1 0.6738\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "41e58b77cd614695b10265bcb7c4f464",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7157\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "\n",
      "Training for task 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "354ca2fb1e02485b8976cf8b36d50b67",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1674, Acc@1 0.6934\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f0c548195de24e04b8f7578010d7207c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7471\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "\n",
      "Training for task 5\n",
      "=> Set task of model.0 to 5\n",
      "=> Set task of model.2 to 5\n",
      "=> Set task of model.4 to 5\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c91895b5db4c4a3081e8d3c7b8142334",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3776, Acc@1 0.6852\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6f06b8e08e1c482987c2f577951d07a3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7200\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 6\n",
      "=> Setting learned tasks of model.2 to 6\n",
      "=> Setting learned tasks of model.4 to 6\n",
      "\n",
      "Training for task 6\n",
      "=> Set task of model.0 to 6\n",
      "=> Set task of model.2 to 6\n",
      "=> Set task of model.4 to 6\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "87e3bfed2aa8462b82086a198fe3f359",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3697, Acc@1 0.7106\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c48e4b1da59e4c42a3d10e87d1b5cafd",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7543\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 7\n",
      "=> Setting learned tasks of model.2 to 7\n",
      "=> Setting learned tasks of model.4 to 7\n",
      "\n",
      "Training for task 7\n",
      "=> Set task of model.0 to 7\n",
      "=> Set task of model.2 to 7\n",
      "=> Set task of model.4 to 7\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "4fbe175e6e2a466d8497d3269d31e32f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3089, Acc@1 0.6768\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8072c12bdc1b4367b4135a73bfe2c9aa",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7298\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 8\n",
      "=> Setting learned tasks of model.2 to 8\n",
      "=> Setting learned tasks of model.4 to 8\n",
      "\n",
      "Training for task 8\n",
      "=> Set task of model.0 to 8\n",
      "=> Set task of model.2 to 8\n",
      "=> Set task of model.4 to 8\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "40f005b206e147ee8877eed9c03b6b62",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2656, Acc@1 0.6865\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9fba375f200a49e69a067c0456d6b2d3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7470\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 9\n",
      "=> Setting learned tasks of model.2 to 9\n",
      "=> Setting learned tasks of model.4 to 9\n",
      "\n",
      "Training for task 9\n",
      "=> Set task of model.0 to 9\n",
      "=> Set task of model.2 to 9\n",
      "=> Set task of model.4 to 9\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "75d48db76258440c962c622cce7e989c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3775, Acc@1 0.6908\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d330b82d8a1a49c3bd038571c5222bbb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7378\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 10\n",
      "=> Setting learned tasks of model.2 to 10\n",
      "=> Setting learned tasks of model.4 to 10\n",
      "\n",
      "Training for task 10\n",
      "=> Set task of model.0 to 10\n",
      "=> Set task of model.2 to 10\n",
      "=> Set task of model.4 to 10\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "71529b1bb47943869485ea12456dafe8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3209, Acc@1 0.7008\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "20297da2d2964f89aa480b8519959733",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7478\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 11\n",
      "=> Setting learned tasks of model.2 to 11\n",
      "=> Setting learned tasks of model.4 to 11\n",
      "\n",
      "Training for task 11\n",
      "=> Set task of model.0 to 11\n",
      "=> Set task of model.2 to 11\n",
      "=> Set task of model.4 to 11\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d2dd93b08af34cd9b23ef6dab460eec0",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2105, Acc@1 0.6953\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "68bd89e562a742c88d5f9b32b6dead74",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7471\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 12\n",
      "=> Setting learned tasks of model.2 to 12\n",
      "=> Setting learned tasks of model.4 to 12\n",
      "\n",
      "Training for task 12\n",
      "=> Set task of model.0 to 12\n",
      "=> Set task of model.2 to 12\n",
      "=> Set task of model.4 to 12\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "71b6f6bfdbad47be8fb1514f941f0f0d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2974, Acc@1 0.6846\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "159cde023fca46e2a6e6b08df01be36d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7404\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 13\n",
      "=> Setting learned tasks of model.2 to 13\n",
      "=> Setting learned tasks of model.4 to 13\n",
      "\n",
      "Training for task 13\n",
      "=> Set task of model.0 to 13\n",
      "=> Set task of model.2 to 13\n",
      "=> Set task of model.4 to 13\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7b4359dcb51649a3b3faf9a78d3a5871",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3349, Acc@1 0.6699\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "900ece2efb3b40b9ac0916b456ea8f50",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7075\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 14\n",
      "=> Setting learned tasks of model.2 to 14\n",
      "=> Setting learned tasks of model.4 to 14\n",
      "\n",
      "Training for task 14\n",
      "=> Set task of model.0 to 14\n",
      "=> Set task of model.2 to 14\n",
      "=> Set task of model.4 to 14\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "05e0aca46aab4f2ebffbd81cc6b97d30",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3001, Acc@1 0.6862\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "db83f073581c48f5b2e8e7d089988ad9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7410\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 15\n",
      "=> Setting learned tasks of model.2 to 15\n",
      "=> Setting learned tasks of model.4 to 15\n",
      "\n",
      "Training for task 15\n",
      "=> Set task of model.0 to 15\n",
      "=> Set task of model.2 to 15\n",
      "=> Set task of model.4 to 15\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5221ba2c4faa4ac0bb06a6e9c86da64f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3483, Acc@1 0.6693\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1f5732a098094b2782396c261d883520",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7149\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 16\n",
      "=> Setting learned tasks of model.2 to 16\n",
      "=> Setting learned tasks of model.4 to 16\n",
      "\n",
      "Training for task 16\n",
      "=> Set task of model.0 to 16\n",
      "=> Set task of model.2 to 16\n",
      "=> Set task of model.4 to 16\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "99f4ede55c0848b992523853e84d63ac",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3433, Acc@1 0.6706\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7d3e5cb9110b490c80be0cdd7685dcff",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7516\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 17\n",
      "=> Setting learned tasks of model.2 to 17\n",
      "=> Setting learned tasks of model.4 to 17\n",
      "\n",
      "Training for task 17\n",
      "=> Set task of model.0 to 17\n",
      "=> Set task of model.2 to 17\n",
      "=> Set task of model.4 to 17\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "15716a92ac7e4c8e93dec0884794fb48",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3107, Acc@1 0.6950\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0551793cd62442469892c3d43a56e96f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7516\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 18\n",
      "=> Setting learned tasks of model.2 to 18\n",
      "=> Setting learned tasks of model.4 to 18\n",
      "\n",
      "Training for task 18\n",
      "=> Set task of model.0 to 18\n",
      "=> Set task of model.2 to 18\n",
      "=> Set task of model.4 to 18\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e575adbce7404135b563e8dcda1f3405",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3609, Acc@1 0.6810\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "3ff2bd99691f4e52a8f52783a954de4a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7169\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 19\n",
      "=> Setting learned tasks of model.2 to 19\n",
      "=> Setting learned tasks of model.4 to 19\n",
      "\n",
      "Training for task 19\n",
      "=> Set task of model.0 to 19\n",
      "=> Set task of model.2 to 19\n",
      "=> Set task of model.4 to 19\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e3ccec0247904dae941ef9832d5e0395",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.3019, Acc@1 0.6917\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "522300471e144845b23aecd76b4cce65",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7613\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 20\n",
      "=> Setting learned tasks of model.2 to 20\n",
      "=> Setting learned tasks of model.4 to 20\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(0, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model_hidf, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model_hidf.parameters() if p.requires_grad], lr=1e-3)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(1):\n",
    "        train(basis_model_hidf, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model_hidf, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model_hidf)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model_hidf, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 139,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.7459\n",
      "Per task performance\n",
      "Task 0: 0.7279\n",
      "Task 1: 0.9232\n",
      "Task 2: 0.7354\n",
      "Task 3: 0.7157\n",
      "Task 4: 0.7471\n",
      "Task 5: 0.7200\n",
      "Task 6: 0.7543\n",
      "Task 7: 0.7298\n",
      "Task 8: 0.7470\n",
      "Task 9: 0.7378\n",
      "Task 10: 0.7478\n",
      "Task 11: 0.7471\n",
      "Task 12: 0.7404\n",
      "Task 13: 0.7075\n",
      "Task 14: 0.7410\n",
      "Task 15: 0.7149\n",
      "Task 16: 0.7516\n",
      "Task 17: 0.7516\n",
      "Task 18: 0.7169\n",
      "Task 19: 0.7613\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_hidf, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_model_hidf, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 140,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_hybrid_mnist_frozen'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Cross task analysis, retraining last layer) -- Using mask from task 1, Sparsity 25"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 148,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_hidfs = BasisHiddenOnlySparseFrozenMultitaskFC(hidden_size=300,\n",
    "                            num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, sparsity=0.25, start_at_optimal=True,)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 149,
   "metadata": {},
   "outputs": [],
   "source": [
    "custom_state_dict = basis_model_hidfs.state_dict().copy()\n",
    "custom_state_dict.update({k:v for k,v in models.state_dict().items() if k in custom_state_dict.keys()})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 150,
   "metadata": {},
   "outputs": [],
   "source": [
    "for k in custom_state_dict.keys():\n",
    "    if k.startswith('model.0.scores.'):\n",
    "        custom_state_dict[k] = models.state_dict()['model.0.scores.1']\n",
    "    elif k.startswith('model.2.scores.'):\n",
    "        custom_state_dict[k] = models.state_dict()['model.2.scores.1']\n",
    "    elif k.startswith('model.0.basis_alphas.'):\n",
    "        custom_state_dict[k] = custom_state_dict['model.0.basis_alphas.1'] \n",
    "    elif k.startswith('model.2.basis_alphas.'):\n",
    "        custom_state_dict[k] = custom_state_dict['model.2.basis_alphas.1']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 151,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n"
     ]
    }
   ],
   "source": [
    "basis_model_hidfs.load_state_dict(custom_state_dict, False)\n",
    "cache_masks(basis_model_hidfs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 152,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f383ae6e41d046c89a0bcc69621190d6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.7949, Acc@1 0.2321\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "45f533fe42914d05973229ab4edfabc9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1277\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "28c41d8cef4d4521b43a566b58e6ef5d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2193, Acc@1 0.9085\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8879c63784734cb293e9cc6e98b78dc4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9101\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "\n",
      "Training for task 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "83ca75b72be2443ca9f5015f45afbbb9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0700, Acc@1 0.2096\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9b33fa1274b14b2184108a950cf32411",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1263\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "\n",
      "Training for task 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "197f461e64ce4d6889a51587df2b2e36",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8840, Acc@1 0.2051\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "28602dbbbac1498b861e487a36316875",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1462\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "\n",
      "Training for task 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7583434cea964199a186baaa5a82524e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.7379, Acc@1 0.2620\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "bc48b13fb8f94603b148d56db7f98e65",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1673\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "\n",
      "Training for task 5\n",
      "=> Set task of model.0 to 5\n",
      "=> Set task of model.2 to 5\n",
      "=> Set task of model.4 to 5\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0015fd992a0442cf91b0108257c64ba6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0438, Acc@1 0.2386\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "37c1e692942a4037a487669d75c46d8b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1399\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 6\n",
      "=> Setting learned tasks of model.2 to 6\n",
      "=> Setting learned tasks of model.4 to 6\n",
      "\n",
      "Training for task 6\n",
      "=> Set task of model.0 to 6\n",
      "=> Set task of model.2 to 6\n",
      "=> Set task of model.4 to 6\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5d4f3334a7b94e93837f00f46b3660a2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8774, Acc@1 0.2490\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c9f032166e7f4e169db05046bd5785df",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1865\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 7\n",
      "=> Setting learned tasks of model.2 to 7\n",
      "=> Setting learned tasks of model.4 to 7\n",
      "\n",
      "Training for task 7\n",
      "=> Set task of model.0 to 7\n",
      "=> Set task of model.2 to 7\n",
      "=> Set task of model.4 to 7\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1c678cc00cec4014b61a49b129bc8b2a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0601, Acc@1 0.2380\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5b581c3452314dc5a1a1e1344649f969",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1399\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 8\n",
      "=> Setting learned tasks of model.2 to 8\n",
      "=> Setting learned tasks of model.4 to 8\n",
      "\n",
      "Training for task 8\n",
      "=> Set task of model.0 to 8\n",
      "=> Set task of model.2 to 8\n",
      "=> Set task of model.4 to 8\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "444c224b07544f62b7e28731fa9f507b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.9645, Acc@1 0.2562\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f01e6ba2ca494c53a292cd827ca3acba",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2075\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 9\n",
      "=> Setting learned tasks of model.2 to 9\n",
      "=> Setting learned tasks of model.4 to 9\n",
      "\n",
      "Training for task 9\n",
      "=> Set task of model.0 to 9\n",
      "=> Set task of model.2 to 9\n",
      "=> Set task of model.4 to 9\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7770c197a7aa45ec96aa22d9bf847dd2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8566, Acc@1 0.2057\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "83a5680bb2ef43c2b4947961bb77609a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1312\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 10\n",
      "=> Setting learned tasks of model.2 to 10\n",
      "=> Setting learned tasks of model.4 to 10\n",
      "\n",
      "Training for task 10\n",
      "=> Set task of model.0 to 10\n",
      "=> Set task of model.2 to 10\n",
      "=> Set task of model.4 to 10\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "4354b9d264a04e099034094dceac8dc8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0359, Acc@1 0.2523\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5aea7e9adc92421ea093e852e7c91df0",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1916\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 11\n",
      "=> Setting learned tasks of model.2 to 11\n",
      "=> Setting learned tasks of model.4 to 11\n",
      "\n",
      "Training for task 11\n",
      "=> Set task of model.0 to 11\n",
      "=> Set task of model.2 to 11\n",
      "=> Set task of model.4 to 11\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "77f1b0d52cd84dab87578f06503163d9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0021, Acc@1 0.2122\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c1954edd97c441d8965b919cb905b603",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1515\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 12\n",
      "=> Setting learned tasks of model.2 to 12\n",
      "=> Setting learned tasks of model.4 to 12\n",
      "\n",
      "Training for task 12\n",
      "=> Set task of model.0 to 12\n",
      "=> Set task of model.2 to 12\n",
      "=> Set task of model.4 to 12\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "285aefe6e593483d951839946cfc3117",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0505, Acc@1 0.2178\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "03c25f1744324361930581513a7e1045",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1314\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 13\n",
      "=> Setting learned tasks of model.2 to 13\n",
      "=> Setting learned tasks of model.4 to 13\n",
      "\n",
      "Training for task 13\n",
      "=> Set task of model.0 to 13\n",
      "=> Set task of model.2 to 13\n",
      "=> Set task of model.4 to 13\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d2615ad554f542928a9c44f5048be7aa",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0472, Acc@1 0.2930\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ccdb0bef27ce4e6ab3df596f5869b90b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2359\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 14\n",
      "=> Setting learned tasks of model.2 to 14\n",
      "=> Setting learned tasks of model.4 to 14\n",
      "\n",
      "Training for task 14\n",
      "=> Set task of model.0 to 14\n",
      "=> Set task of model.2 to 14\n",
      "=> Set task of model.4 to 14\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "bdff9bb6197a4360ac17002b4d2922c1",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0947, Acc@1 0.2178\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "af82d7f0a120404cb6bc432080eaf462",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1894\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 15\n",
      "=> Setting learned tasks of model.2 to 15\n",
      "=> Setting learned tasks of model.4 to 15\n",
      "\n",
      "Training for task 15\n",
      "=> Set task of model.0 to 15\n",
      "=> Set task of model.2 to 15\n",
      "=> Set task of model.4 to 15\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "555bce0c858543549c7b6ff66c1a7b4a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0573, Acc@1 0.2038\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "729020b3a02942e285e5ccc3568bdab4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1684\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 16\n",
      "=> Setting learned tasks of model.2 to 16\n",
      "=> Setting learned tasks of model.4 to 16\n",
      "\n",
      "Training for task 16\n",
      "=> Set task of model.0 to 16\n",
      "=> Set task of model.2 to 16\n",
      "=> Set task of model.4 to 16\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "cbfee3b38fd244d0b3ef2e0351b8be28",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.7659, Acc@1 0.2484\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "49914cd3095a4130bd41d0c2c602f1e2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1761\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 17\n",
      "=> Setting learned tasks of model.2 to 17\n",
      "=> Setting learned tasks of model.4 to 17\n",
      "\n",
      "Training for task 17\n",
      "=> Set task of model.0 to 17\n",
      "=> Set task of model.2 to 17\n",
      "=> Set task of model.4 to 17\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ba8d4b900ca94411ace7171b0b2f5e33",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.9463, Acc@1 0.2057\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1ff1336793504ea3b0184f0ca583ca3c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1725\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 18\n",
      "=> Setting learned tasks of model.2 to 18\n",
      "=> Setting learned tasks of model.4 to 18\n",
      "\n",
      "Training for task 18\n",
      "=> Set task of model.0 to 18\n",
      "=> Set task of model.2 to 18\n",
      "=> Set task of model.4 to 18\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d1c5059ec73746978e721a5c96f22ae7",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0789, Acc@1 0.2331\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "57fb2c81d1ae4ae0a197c40caecd3e29",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1807\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 19\n",
      "=> Setting learned tasks of model.2 to 19\n",
      "=> Setting learned tasks of model.4 to 19\n",
      "\n",
      "Training for task 19\n",
      "=> Set task of model.0 to 19\n",
      "=> Set task of model.2 to 19\n",
      "=> Set task of model.4 to 19\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a42c7146437447feadba316f15de6627",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.9310, Acc@1 0.2454\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1f77e2964a9e402f8e6a2672bd66fa0e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1840\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 20\n",
      "=> Setting learned tasks of model.2 to 20\n",
      "=> Setting learned tasks of model.4 to 20\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(0, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model_hidfs, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model_hidfs.parameters() if p.requires_grad], lr=1e-3)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(1):\n",
    "        train(basis_model_hidfs, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model_hidfs, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model_hidfs)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model_hidfs, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 153,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.2032\n",
      "Per task performance\n",
      "Task 0: 0.1277\n",
      "Task 1: 0.9101\n",
      "Task 2: 0.1263\n",
      "Task 3: 0.1462\n",
      "Task 4: 0.1673\n",
      "Task 5: 0.1399\n",
      "Task 6: 0.1865\n",
      "Task 7: 0.1399\n",
      "Task 8: 0.2075\n",
      "Task 9: 0.1312\n",
      "Task 10: 0.1916\n",
      "Task 11: 0.1515\n",
      "Task 12: 0.1314\n",
      "Task 13: 0.2359\n",
      "Task 14: 0.1894\n",
      "Task 15: 0.1684\n",
      "Task 16: 0.1761\n",
      "Task 17: 0.1725\n",
      "Task 18: 0.1807\n",
      "Task 19: 0.1840\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_hidfs, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_model_hidfs, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 154,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_hybrid_mnist_frozen_sparsity25'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Cross task analysis, no retraining) -- Using mask from task 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 158,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_hidf = BasisHiddenOnlyFrozenMultitaskFC(hidden_size=300,\n",
    "                            num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 159,
   "metadata": {},
   "outputs": [],
   "source": [
    "custom_state_dict = basis_model_hidf.state_dict().copy()\n",
    "custom_state_dict.update({k:v for k,v in model.state_dict().items() if k in custom_state_dict.keys()})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 160,
   "metadata": {},
   "outputs": [],
   "source": [
    "for k in custom_state_dict.keys():\n",
    "    if k.startswith('model.0.scores.'):\n",
    "        custom_state_dict[k] = model.state_dict()['model.0.scores.1']\n",
    "    elif k.startswith('model.2.scores.'):\n",
    "        custom_state_dict[k] = model.state_dict()['model.2.scores.1']\n",
    "    elif k.startswith('model.0.basis_alphas.'):\n",
    "        custom_state_dict[k] = custom_state_dict['model.0.basis_alphas.1'] \n",
    "    elif k.startswith('model.2.basis_alphas.'):\n",
    "        custom_state_dict[k] = custom_state_dict['model.2.basis_alphas.1']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 161,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n"
     ]
    }
   ],
   "source": [
    "basis_model_hidf.load_state_dict(custom_state_dict, False)\n",
    "cache_masks(basis_model_hidf)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 162,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.1522\n",
      "Per task performance\n",
      "Task 0: 0.1224\n",
      "Task 1: 0.9266\n",
      "Task 2: 0.0949\n",
      "Task 3: 0.0670\n",
      "Task 4: 0.1245\n",
      "Task 5: 0.1782\n",
      "Task 6: 0.1502\n",
      "Task 7: 0.1269\n",
      "Task 8: 0.0935\n",
      "Task 9: 0.1140\n",
      "Task 10: 0.0936\n",
      "Task 11: 0.1076\n",
      "Task 12: 0.0997\n",
      "Task 13: 0.1127\n",
      "Task 14: 0.0945\n",
      "Task 15: 0.0943\n",
      "Task 16: 0.1285\n",
      "Task 17: 0.1021\n",
      "Task 18: 0.1017\n",
      "Task 19: 0.1111\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_hidf, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_model_hidf, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Overlap Analysis"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 163,
   "metadata": {},
   "outputs": [],
   "source": [
    "mnist1 = MNISTPerm()\n",
    "mnist2 = MNISTPerm()\n",
    "mnist3 = MNISTPerm()\n",
    "mnist4 = MNISTPerm()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 164,
   "metadata": {},
   "outputs": [],
   "source": [
    "weight_dict = {k: v for k,v in model.state_dict().items() if k.endswith('weight')}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 165,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 165,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Task ID given during train time\n",
    "from IPython.display import clear_output\n",
    "\n",
    "num_tasks = 5 # For demonstration purposes, we go up to 2500 in our paper\n",
    "model1 = MultitaskFC(hidden_size=300, num_tasks=num_tasks)\n",
    "sd1 = model1.state_dict()\n",
    "sd1.update(weight_dict)\n",
    "model1.load_state_dict(sd1)\n",
    "model2 = MultitaskFC(hidden_size=300, num_tasks=num_tasks)\n",
    "sd2 = model2.state_dict()\n",
    "sd2.update(weight_dict)\n",
    "model2.load_state_dict(sd2)\n",
    "model3 = MultitaskFC(hidden_size=300, num_tasks=num_tasks)\n",
    "sd3 = model3.state_dict()\n",
    "sd3.update(weight_dict)\n",
    "model3.load_state_dict(sd3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 166,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c379da7c8e004c8eab0d133ab8957ea4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.2601, Acc@1 0.8555\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "83dcabca7ebb40458938ae8aaf8fdbb8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.1826, Acc@1 0.8415\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "da0b955b69664c9d988cf67ff1d2dd9e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8496\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "593f24a970c443b89f09e201aa3fcda3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3428, Acc@1 0.8548\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "80707e45bc9446ef9782844cfbcf6dc3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.2812, Acc@1 0.8551\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fe4ae314749b4d2195ab365331a188c9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8522\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "\n",
      "Training for task 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "dfa8b336166743a492cc2e8b818fdcad",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3166, Acc@1 0.8516\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "826a2c04b939496e8555a32d2c9035bb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3793, Acc@1 0.8529\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b1da9dcbab50497399f643059c9ca6ff",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8577\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "\n",
      "Training for task 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e37274276500458384d85ada6696b32d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.2475, Acc@1 0.8604\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d4505312f2be44ffbfb82aa3e73976ba",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.2908, Acc@1 0.8564\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "81aa2832dbcc4c42ba75fee61aef587f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8639\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "\n",
      "Training for task 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5436b85627cf408cb0c667691b6aa2ae",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3427, Acc@1 0.8555\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b1bb51fddd174482b2275d7a55e540ff",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.2153, Acc@1 0.8483\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "307c63a55866434fb32b172adf6561a7",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8594\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(model1, task)\n",
    "    set_model_task(model2, task)\n",
    "    set_model_task(model3, task)\n",
    "    mnist1.update_task(task)\n",
    "    mnist2.update_task(task)\n",
    "    mnist3.update_task(task)\n",
    "\n",
    "    optimizer1 = optim.RMSprop([p for p in model1.parameters() if p.requires_grad], lr=1e-4)\n",
    "    optimizer2 = optim.RMSprop([p for p in model2.parameters() if p.requires_grad], lr=1e-4)\n",
    "    optimizer3 = optim.RMSprop([p for p in model3.parameters() if p.requires_grad], lr=1e-4)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(1):\n",
    "        train(model1, mnist1.train_loader, optimizer1, e)\n",
    "        train(model2, mnist2.train_loader, optimizer2, e)\n",
    "        train(model3, mnist3.train_loader, optimizer3, e)\n",
    "\n",
    "#         print(\"Validation\")\n",
    "#         print(\"============\")\n",
    "#         acc1 = evaluate(model, mnist.val_loader, e)\n",
    "\n",
    "\n",
    "    cache_masks(model1)\n",
    "    cache_masks(model2)\n",
    "    cache_masks(model3)\n",
    "    print()\n",
    "    set_num_tasks_learned(model1, task + 1)\n",
    "    set_num_tasks_learned(model2, task + 1)\n",
    "    set_num_tasks_learned(model3, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 167,
   "metadata": {},
   "outputs": [],
   "source": [
    "def overlap(mask1, mask2):\n",
    "    assert mask1.shape == mask2.shape\n",
    "    count_same = (mask1 + mask2 - 1).abs().sum()\n",
    "    return count_same / (1.0*(mask1.shape[0] * mask1.shape[1]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 168,
   "metadata": {},
   "outputs": [],
   "source": [
    "from itertools import product"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 169,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Layer: 0, Task: 0, Overlap: 0.51096510887146\n",
      "Layer: 0, Task: 1, Overlap: 0.5155952572822571\n",
      "Layer: 0, Task: 2, Overlap: 0.5139200687408447\n",
      "Layer: 0, Task: 3, Overlap: 0.5134651064872742\n",
      "Layer: 0, Task: 4, Overlap: 0.5120918154716492\n",
      "Layer: 2, Task: 0, Overlap: 0.503944456577301\n",
      "Layer: 2, Task: 1, Overlap: 0.5013333559036255\n",
      "Layer: 2, Task: 2, Overlap: 0.5013222098350525\n",
      "Layer: 2, Task: 3, Overlap: 0.50445556640625\n",
      "Layer: 2, Task: 4, Overlap: 0.501455545425415\n",
      "Layer: 4, Task: 0, Overlap: 0.5084999799728394\n",
      "Layer: 4, Task: 1, Overlap: 0.5153999924659729\n",
      "Layer: 4, Task: 2, Overlap: 0.5112333297729492\n",
      "Layer: 4, Task: 3, Overlap: 0.5065333247184753\n",
      "Layer: 4, Task: 4, Overlap: 0.5167333483695984\n"
     ]
    }
   ],
   "source": [
    "for layer, task in product([0,2,4], [i for i in range(5)]):\n",
    "    print('Layer: {}, Task: {}, Overlap: {}'.format(layer, task,\n",
    "        overlap(model2.state_dict()['model.{}.stacked'.format(layer)][task],\n",
    "                model3.state_dict()['model.{}.stacked'.format(layer)][task])))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Overlap Analysis with Sparsity (25%)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 170,
   "metadata": {},
   "outputs": [],
   "source": [
    "mnist1 = MNISTPerm()\n",
    "mnist2 = MNISTPerm()\n",
    "mnist3 = MNISTPerm()\n",
    "mnist4 = MNISTPerm()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 171,
   "metadata": {},
   "outputs": [],
   "source": [
    "weight_dict = {k: v for k,v in models.state_dict().items() if k.endswith('weight')}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 172,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 172,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Task ID given during train time\n",
    "from IPython.display import clear_output\n",
    "\n",
    "num_tasks = 5 # For demonstration purposes, we go up to 2500 in our paper\n",
    "model1 = MultitaskFCSparse(hidden_size=300, num_tasks=num_tasks, sparsity=0.25)\n",
    "sd1 = model1.state_dict()\n",
    "sd1.update(weight_dict)\n",
    "model1.load_state_dict(sd1)\n",
    "model2 = MultitaskFCSparse(hidden_size=300, num_tasks=num_tasks, sparsity=0.25)\n",
    "sd2 = model2.state_dict()\n",
    "sd2.update(weight_dict)\n",
    "model2.load_state_dict(sd2)\n",
    "model3 = MultitaskFCSparse(hidden_size=300, num_tasks=num_tasks, sparsity=0.25)\n",
    "sd3 = model3.state_dict()\n",
    "sd3.update(weight_dict)\n",
    "model3.load_state_dict(sd3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 173,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.2722, Acc@1 0.8424\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3783, Acc@1 0.8298\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8356\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.5086, Acc@1 0.8408\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3411, Acc@1 0.8275\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8291\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "\n",
      "Training for task 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3689, Acc@1 0.8395\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.2454, Acc@1 0.8382\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8438\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "\n",
      "Training for task 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3847, Acc@1 0.8372\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3338, Acc@1 0.8398\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8369\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "\n",
      "Training for task 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3435, Acc@1 0.8385\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 0.3908, Acc@1 0.8415\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.8395\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(model1, task)\n",
    "    set_model_task(model2, task)\n",
    "    set_model_task(model3, task)\n",
    "    mnist1.update_task(task)\n",
    "    mnist2.update_task(task)\n",
    "    mnist3.update_task(task)\n",
    "\n",
    "    optimizer1 = optim.RMSprop([p for p in model1.parameters() if p.requires_grad], lr=1e-4)\n",
    "    optimizer2 = optim.RMSprop([p for p in model2.parameters() if p.requires_grad], lr=1e-4)\n",
    "    optimizer3 = optim.RMSprop([p for p in model3.parameters() if p.requires_grad], lr=1e-4)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(1):\n",
    "        train(model1, mnist1.train_loader, optimizer1, e)\n",
    "        train(model2, mnist2.train_loader, optimizer2, e)\n",
    "        train(model3, mnist3.train_loader, optimizer3, e)\n",
    "\n",
    "#         print(\"Validation\")\n",
    "#         print(\"============\")\n",
    "#         acc1 = evaluate(model, mnist.val_loader, e)\n",
    "\n",
    "\n",
    "    cache_masks(model1)\n",
    "    cache_masks(model2)\n",
    "    cache_masks(model3)\n",
    "    print()\n",
    "    set_num_tasks_learned(model1, task + 1)\n",
    "    set_num_tasks_learned(model2, task + 1)\n",
    "    set_num_tasks_learned(model3, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 174,
   "metadata": {},
   "outputs": [],
   "source": [
    "def overlap(mask1, mask2):\n",
    "    assert mask1.shape == mask2.shape\n",
    "    count_same = (mask1 + mask2 - 1).abs().sum()\n",
    "    return count_same / (1.0*(mask1.shape[0] * mask1.shape[1]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 175,
   "metadata": {},
   "outputs": [],
   "source": [
    "from itertools import product"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 176,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Layer: 0, Task: 0, Overlap: 0.6428911685943604\n",
      "Layer: 0, Task: 1, Overlap: 0.6420833468437195\n",
      "Layer: 0, Task: 2, Overlap: 0.6420493125915527\n",
      "Layer: 0, Task: 3, Overlap: 0.6426190733909607\n",
      "Layer: 0, Task: 4, Overlap: 0.6432312726974487\n",
      "Layer: 2, Task: 0, Overlap: 0.6288444399833679\n",
      "Layer: 2, Task: 1, Overlap: 0.6273333430290222\n",
      "Layer: 2, Task: 2, Overlap: 0.6287555694580078\n",
      "Layer: 2, Task: 3, Overlap: 0.6279555559158325\n",
      "Layer: 2, Task: 4, Overlap: 0.627488911151886\n",
      "Layer: 4, Task: 0, Overlap: 0.6443333625793457\n",
      "Layer: 4, Task: 1, Overlap: 0.6440666913986206\n",
      "Layer: 4, Task: 2, Overlap: 0.6439999938011169\n",
      "Layer: 4, Task: 3, Overlap: 0.6449333429336548\n",
      "Layer: 4, Task: 4, Overlap: 0.642799973487854\n"
     ]
    }
   ],
   "source": [
    "for layer, task in product([0,2,4], [i for i in range(5)]):\n",
    "    print('Layer: {}, Task: {}, Overlap: {}'.format(layer, task,\n",
    "        overlap(model2.state_dict()['model.{}.stacked'.format(layer)][task],\n",
    "                model3.state_dict()['model.{}.stacked'.format(layer)][task])))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Overlap Analysis with Sparsity (5%)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 177,
   "metadata": {},
   "outputs": [],
   "source": [
    "mnist1 = MNISTPerm()\n",
    "mnist2 = MNISTPerm()\n",
    "mnist3 = MNISTPerm()\n",
    "mnist4 = MNISTPerm()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 178,
   "metadata": {},
   "outputs": [],
   "source": [
    "weight_dict = {k: v for k,v in models.state_dict().items() if k.endswith('weight')}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 179,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 179,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Task ID given during train time\n",
    "from IPython.display import clear_output\n",
    "\n",
    "num_tasks = 5 # For demonstration purposes, we go up to 2500 in our paper\n",
    "model1 = MultitaskFCSparse(hidden_size=300, num_tasks=num_tasks, sparsity=0.05)\n",
    "sd1 = model1.state_dict()\n",
    "sd1.update(weight_dict)\n",
    "model1.load_state_dict(sd1)\n",
    "model2 = MultitaskFCSparse(hidden_size=300, num_tasks=num_tasks, sparsity=0.05)\n",
    "sd2 = model2.state_dict()\n",
    "sd2.update(weight_dict)\n",
    "model2.load_state_dict(sd2)\n",
    "model3 = MultitaskFCSparse(hidden_size=300, num_tasks=num_tasks, sparsity=0.05)\n",
    "sd3 = model3.state_dict()\n",
    "sd3.update(weight_dict)\n",
    "model3.load_state_dict(sd3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 180,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.6100, Acc@1 0.3216\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.5388, Acc@1 0.3506\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.3001\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.5993, Acc@1 0.3086\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.5760, Acc@1 0.3245\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.3900\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "\n",
      "Training for task 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.5494, Acc@1 0.3434\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.5749, Acc@1 0.2956\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.3307\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "\n",
      "Training for task 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.5912, Acc@1 0.3600\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.5472, Acc@1 0.3096\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.3190\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "\n",
      "Training for task 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.5756, Acc@1 0.2614\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "e0 461/469 => Loss 2.5821, Acc@1 0.2933\r"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.00.3981\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(model1, task)\n",
    "    set_model_task(model2, task)\n",
    "    set_model_task(model3, task)\n",
    "    mnist1.update_task(task)\n",
    "    mnist2.update_task(task)\n",
    "    mnist3.update_task(task)\n",
    "\n",
    "    optimizer1 = optim.RMSprop([p for p in model1.parameters() if p.requires_grad], lr=1e-4)\n",
    "    optimizer2 = optim.RMSprop([p for p in model2.parameters() if p.requires_grad], lr=1e-4)\n",
    "    optimizer3 = optim.RMSprop([p for p in model3.parameters() if p.requires_grad], lr=1e-4)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(1):\n",
    "        train(model1, mnist1.train_loader, optimizer1, e)\n",
    "        train(model2, mnist2.train_loader, optimizer2, e)\n",
    "        train(model3, mnist3.train_loader, optimizer3, e)\n",
    "\n",
    "#         print(\"Validation\")\n",
    "#         print(\"============\")\n",
    "#         acc1 = evaluate(model, mnist.val_loader, e)\n",
    "\n",
    "\n",
    "    cache_masks(model1)\n",
    "    cache_masks(model2)\n",
    "    cache_masks(model3)\n",
    "    print()\n",
    "    set_num_tasks_learned(model1, task + 1)\n",
    "    set_num_tasks_learned(model2, task + 1)\n",
    "    set_num_tasks_learned(model3, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 181,
   "metadata": {},
   "outputs": [],
   "source": [
    "def overlap(mask1, mask2):\n",
    "    assert mask1.shape == mask2.shape\n",
    "    count_same = (mask1 + mask2 - 1).abs().sum()\n",
    "    return count_same / (1.0*(mask1.shape[0] * mask1.shape[1]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 182,
   "metadata": {},
   "outputs": [],
   "source": [
    "from itertools import product"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 183,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Layer: 0, Task: 0, Overlap: 0.912508487701416\n",
      "Layer: 0, Task: 1, Overlap: 0.9120748043060303\n",
      "Layer: 0, Task: 2, Overlap: 0.9122363924980164\n",
      "Layer: 0, Task: 3, Overlap: 0.912295937538147\n",
      "Layer: 0, Task: 4, Overlap: 0.9120323061943054\n",
      "Layer: 2, Task: 0, Overlap: 0.9100444316864014\n",
      "Layer: 2, Task: 1, Overlap: 0.9103778004646301\n",
      "Layer: 2, Task: 2, Overlap: 0.9103555679321289\n",
      "Layer: 2, Task: 3, Overlap: 0.9097777605056763\n",
      "Layer: 2, Task: 4, Overlap: 0.9114000201225281\n",
      "Layer: 4, Task: 0, Overlap: 0.9117333292961121\n",
      "Layer: 4, Task: 1, Overlap: 0.9111999869346619\n",
      "Layer: 4, Task: 2, Overlap: 0.9112666845321655\n",
      "Layer: 4, Task: 3, Overlap: 0.9115999937057495\n",
      "Layer: 4, Task: 4, Overlap: 0.9118666648864746\n"
     ]
    }
   ],
   "source": [
    "for layer, task in product([0,2,4], [i for i in range(5)]):\n",
    "    print('Layer: {}, Task: {}, Overlap: {}'.format(layer, task,\n",
    "        overlap(model2.state_dict()['model.{}.stacked'.format(layer)][task],\n",
    "                model3.state_dict()['model.{}.stacked'.format(layer)][task])))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Multimask Same Task "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 203,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_masks_to_create = 15"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 204,
   "metadata": {},
   "outputs": [],
   "source": [
    "loaders = {}\n",
    "for i in range(num_masks_to_create):\n",
    "    loaders[i] = MNISTPerm()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 205,
   "metadata": {},
   "outputs": [],
   "source": [
    "weight_dict = {k: v for k,v in model.state_dict().items() if k.endswith('weight')}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 206,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Task ID given during train time\n",
    "from IPython.display import clear_output\n",
    "\n",
    "model_map = {}\n",
    "for i in range(num_masks_to_create):\n",
    "    modeli = MultitaskFC(hidden_size=300, num_tasks=1)\n",
    "    sdi = modeli.state_dict()\n",
    "    sdi.update(weight_dict)\n",
    "    modeli.load_state_dict(sdi)\n",
    "    model_map[i] = modeli"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 207,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "3843a32d8dab4d2e8740e3ac11117a92",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2728, Acc@1 0.8610\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "73bc644fa8fe44dd934c51179b917cf6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9274\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6a38b5eda3ca4b059a17654e3e5df8ce",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2012, Acc@1 0.8587\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1377d8ac26ac442aae62d5bd3a8dc689",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9257\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7d67f4aa0e844f6eb23740dc6b274b52",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.4101, Acc@1 0.8499\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6c6ca57174f94bd2849be827f01c2d36",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9225\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "904898b22f5d4ef98270a20afa55a96e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3553, Acc@1 0.8447\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6da4d6df44604aa291d3847639d6f3bc",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9246\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "3c1def4277a74d6cb67b0b69615e50b4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3421, Acc@1 0.8538\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a18978f7466d428a96c59f6b31e4c59d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9259\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "19b0d33b1a884e1d91f8e1ef75e7320f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2036, Acc@1 0.8493\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1719d72499b645128d3c54c6544f3a61",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9262\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5f148b10b5964f4f85cef4fd31347d2e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2799, Acc@1 0.8503\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "85420876074c4f77ac01797472003c76",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9258\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6576d83626bf4a368b59c6b0f5926b8e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3788, Acc@1 0.8477\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "be14eb066ea048adae72a38b516f0269",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9254\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "102e8fd6d61440109bf236b51215b50f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3063, Acc@1 0.8633\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "104302108f8b43cd91b7ceb1c14685ad",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9269\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "03cd8a9aef5f413caea865af8338a724",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2844, Acc@1 0.8607\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "46516ecc91664829b59fb83b16f8f799",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9246\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7705d9d3d13c4ce9beeb7643301060ef",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2708, Acc@1 0.8529\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "12a05ea7fc7143d0b14d7a22bb5f4a23",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9239\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d10a86e5c19549e4902e57a08e7aa532",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2453, Acc@1 0.8525\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c7e0c45dc0ec445abeb52b36369af186",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9219\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "52481f877aa0461987f7a980e3ea336a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3848, Acc@1 0.8483\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "96763cdaeaa243f1b36345dcaaba1403",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9217\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "5ac2392bb0f64d7aa71cbe9f71644543",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3255, Acc@1 0.8555\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8ddc5a191c1a4871a83210d7cc5113fb",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9239\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "363b68c36254448c8130647c2983813c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3825, Acc@1 0.8473\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "aa175e4baaa348bda16ee97b44be0e04",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9248\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for idx, modeli in model_map.items():\n",
    "    for task in range(1):\n",
    "        print(f\"Training for task {task}\")\n",
    "        set_model_task(modeli, task)\n",
    "        mnisti = loaders[idx]\n",
    "        mnisti.update_task(task)\n",
    "\n",
    "        optimizer1 = optim.RMSprop([p for p in modeli.parameters() if p.requires_grad], lr=1e-4)\n",
    "        # Train for 1 epoch\n",
    "        for e in range(1):\n",
    "            train(modeli, mnisti.train_loader, optimizer1, e)\n",
    "            \n",
    "            print(\"Validation\")\n",
    "            print(\"============\")\n",
    "            acc1 = evaluate(modeli, mnisti.val_loader, e)\n",
    "\n",
    "\n",
    "        cache_masks(modeli)\n",
    "        print()\n",
    "        set_num_tasks_learned(modeli, task + 1)\n",
    "        print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 208,
   "metadata": {},
   "outputs": [],
   "source": [
    "def overlap(mask1, mask2):\n",
    "    assert mask1.shape == mask2.shape\n",
    "    count_same = (mask1 + mask2 - 1).abs().sum()\n",
    "    return count_same / (1.0*(mask1.shape[0] * mask1.shape[1]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 209,
   "metadata": {},
   "outputs": [],
   "source": [
    "from itertools import product"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 212,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Layer: 0, Models: (0, 1), Overlap: 0.5137117505073547\n",
      "Layer: 0, Models: (0, 2), Overlap: 0.5123809576034546\n",
      "Layer: 0, Models: (0, 3), Overlap: 0.5121513605117798\n",
      "Layer: 0, Models: (0, 4), Overlap: 0.5120238065719604\n",
      "Layer: 0, Models: (0, 5), Overlap: 0.5122109055519104\n",
      "Layer: 0, Models: (0, 6), Overlap: 0.5125425457954407\n",
      "Layer: 0, Models: (0, 7), Overlap: 0.511126697063446\n",
      "Layer: 0, Models: (0, 8), Overlap: 0.5117729306221008\n",
      "Layer: 0, Models: (0, 9), Overlap: 0.5117644667625427\n",
      "Layer: 0, Models: (0, 10), Overlap: 0.5147448778152466\n",
      "Layer: 0, Models: (0, 11), Overlap: 0.5122236609458923\n",
      "Layer: 0, Models: (0, 12), Overlap: 0.51350337266922\n",
      "Layer: 0, Models: (0, 13), Overlap: 0.5119345188140869\n",
      "Layer: 0, Models: (0, 14), Overlap: 0.5143027305603027\n",
      "Layer: 0, Models: (1, 0), Overlap: 0.5137117505073547\n",
      "Layer: 0, Models: (1, 2), Overlap: 0.5127848386764526\n",
      "Layer: 0, Models: (1, 3), Overlap: 0.5127848386764526\n",
      "Layer: 0, Models: (1, 4), Overlap: 0.5139413475990295\n",
      "Layer: 0, Models: (1, 5), Overlap: 0.5135841965675354\n",
      "Layer: 0, Models: (1, 6), Overlap: 0.5139753222465515\n",
      "Layer: 0, Models: (1, 7), Overlap: 0.5132483243942261\n",
      "Layer: 0, Models: (1, 8), Overlap: 0.513069748878479\n",
      "Layer: 0, Models: (1, 9), Overlap: 0.5143877267837524\n",
      "Layer: 0, Models: (1, 10), Overlap: 0.5102338194847107\n",
      "Layer: 0, Models: (1, 11), Overlap: 0.5123384594917297\n",
      "Layer: 0, Models: (1, 12), Overlap: 0.5123512148857117\n",
      "Layer: 0, Models: (1, 13), Overlap: 0.5134353637695312\n",
      "Layer: 0, Models: (1, 14), Overlap: 0.5111352205276489\n",
      "Layer: 0, Models: (2, 0), Overlap: 0.5123809576034546\n",
      "Layer: 0, Models: (2, 1), Overlap: 0.5127848386764526\n",
      "Layer: 0, Models: (2, 3), Overlap: 0.5136479735374451\n",
      "Layer: 0, Models: (2, 4), Overlap: 0.5133758783340454\n",
      "Layer: 0, Models: (2, 5), Overlap: 0.5108928680419922\n",
      "Layer: 0, Models: (2, 6), Overlap: 0.5130271911621094\n",
      "Layer: 0, Models: (2, 7), Overlap: 0.5125892758369446\n",
      "Layer: 0, Models: (2, 8), Overlap: 0.5131419897079468\n",
      "Layer: 0, Models: (2, 9), Overlap: 0.513694703578949\n",
      "Layer: 0, Models: (2, 10), Overlap: 0.5135884284973145\n",
      "Layer: 0, Models: (2, 11), Overlap: 0.5120790600776672\n",
      "Layer: 0, Models: (2, 12), Overlap: 0.5121343731880188\n",
      "Layer: 0, Models: (2, 13), Overlap: 0.5129294395446777\n",
      "Layer: 0, Models: (2, 14), Overlap: 0.5132483243942261\n",
      "Layer: 0, Models: (3, 0), Overlap: 0.5121513605117798\n",
      "Layer: 0, Models: (3, 1), Overlap: 0.5127848386764526\n",
      "Layer: 0, Models: (3, 2), Overlap: 0.5136479735374451\n",
      "Layer: 0, Models: (3, 4), Overlap: 0.5127891302108765\n",
      "Layer: 0, Models: (3, 5), Overlap: 0.5118622183799744\n",
      "Layer: 0, Models: (3, 6), Overlap: 0.5120152831077576\n",
      "Layer: 0, Models: (3, 7), Overlap: 0.5143324732780457\n",
      "Layer: 0, Models: (3, 8), Overlap: 0.5140944123268127\n",
      "Layer: 0, Models: (3, 9), Overlap: 0.5145110487937927\n",
      "Layer: 0, Models: (3, 10), Overlap: 0.5128911733627319\n",
      "Layer: 0, Models: (3, 11), Overlap: 0.5124617218971252\n",
      "Layer: 0, Models: (3, 12), Overlap: 0.5122448801994324\n",
      "Layer: 0, Models: (3, 13), Overlap: 0.5131845474243164\n",
      "Layer: 0, Models: (3, 14), Overlap: 0.5129761695861816\n",
      "Layer: 0, Models: (4, 0), Overlap: 0.5120238065719604\n",
      "Layer: 0, Models: (4, 1), Overlap: 0.5139413475990295\n",
      "Layer: 0, Models: (4, 2), Overlap: 0.5133758783340454\n",
      "Layer: 0, Models: (4, 3), Overlap: 0.5127891302108765\n",
      "Layer: 0, Models: (4, 5), Overlap: 0.5117006897926331\n",
      "Layer: 0, Models: (4, 6), Overlap: 0.5124574899673462\n",
      "Layer: 0, Models: (4, 7), Overlap: 0.5107525587081909\n",
      "Layer: 0, Models: (4, 8), Overlap: 0.5125042796134949\n",
      "Layer: 0, Models: (4, 9), Overlap: 0.5104039311408997\n",
      "Layer: 0, Models: (4, 10), Overlap: 0.5133928656578064\n",
      "Layer: 0, Models: (4, 11), Overlap: 0.5125722885131836\n",
      "Layer: 0, Models: (4, 12), Overlap: 0.512704074382782\n",
      "Layer: 0, Models: (4, 13), Overlap: 0.511509358882904\n",
      "Layer: 0, Models: (4, 14), Overlap: 0.5099660158157349\n",
      "Layer: 0, Models: (5, 0), Overlap: 0.5122109055519104\n",
      "Layer: 0, Models: (5, 1), Overlap: 0.5135841965675354\n",
      "Layer: 0, Models: (5, 2), Overlap: 0.5108928680419922\n",
      "Layer: 0, Models: (5, 3), Overlap: 0.5118622183799744\n",
      "Layer: 0, Models: (5, 4), Overlap: 0.5117006897926331\n",
      "Layer: 0, Models: (5, 6), Overlap: 0.5149915218353271\n",
      "Layer: 0, Models: (5, 7), Overlap: 0.5127338171005249\n",
      "Layer: 0, Models: (5, 8), Overlap: 0.5124277472496033\n",
      "Layer: 0, Models: (5, 9), Overlap: 0.5119600296020508\n",
      "Layer: 0, Models: (5, 10), Overlap: 0.5107483267784119\n",
      "Layer: 0, Models: (5, 11), Overlap: 0.512130081653595\n",
      "Layer: 0, Models: (5, 12), Overlap: 0.5109098553657532\n",
      "Layer: 0, Models: (5, 13), Overlap: 0.5146386027336121\n",
      "Layer: 0, Models: (5, 14), Overlap: 0.5094302892684937\n",
      "Layer: 0, Models: (6, 0), Overlap: 0.5125425457954407\n",
      "Layer: 0, Models: (6, 1), Overlap: 0.5139753222465515\n",
      "Layer: 0, Models: (6, 2), Overlap: 0.5130271911621094\n",
      "Layer: 0, Models: (6, 3), Overlap: 0.5120152831077576\n",
      "Layer: 0, Models: (6, 4), Overlap: 0.5124574899673462\n",
      "Layer: 0, Models: (6, 5), Overlap: 0.5149915218353271\n",
      "Layer: 0, Models: (6, 7), Overlap: 0.511713445186615\n",
      "Layer: 0, Models: (6, 8), Overlap: 0.5126062631607056\n",
      "Layer: 0, Models: (6, 9), Overlap: 0.5132015347480774\n",
      "Layer: 0, Models: (6, 10), Overlap: 0.5134949088096619\n",
      "Layer: 0, Models: (6, 11), Overlap: 0.5131419897079468\n",
      "Layer: 0, Models: (6, 12), Overlap: 0.5110969543457031\n",
      "Layer: 0, Models: (6, 13), Overlap: 0.5095025300979614\n",
      "Layer: 0, Models: (6, 14), Overlap: 0.5109778642654419\n",
      "Layer: 0, Models: (7, 0), Overlap: 0.511126697063446\n",
      "Layer: 0, Models: (7, 1), Overlap: 0.5132483243942261\n",
      "Layer: 0, Models: (7, 2), Overlap: 0.5125892758369446\n",
      "Layer: 0, Models: (7, 3), Overlap: 0.5143324732780457\n",
      "Layer: 0, Models: (7, 4), Overlap: 0.5107525587081909\n",
      "Layer: 0, Models: (7, 5), Overlap: 0.5127338171005249\n",
      "Layer: 0, Models: (7, 6), Overlap: 0.511713445186615\n",
      "Layer: 0, Models: (7, 8), Overlap: 0.5136479735374451\n",
      "Layer: 0, Models: (7, 9), Overlap: 0.5131887793540955\n",
      "Layer: 0, Models: (7, 10), Overlap: 0.5114668607711792\n",
      "Layer: 0, Models: (7, 11), Overlap: 0.5122959017753601\n",
      "Layer: 0, Models: (7, 12), Overlap: 0.5141113996505737\n",
      "Layer: 0, Models: (7, 13), Overlap: 0.5124319791793823\n",
      "Layer: 0, Models: (7, 14), Overlap: 0.5136181712150574\n",
      "Layer: 0, Models: (8, 0), Overlap: 0.5117729306221008\n",
      "Layer: 0, Models: (8, 1), Overlap: 0.513069748878479\n",
      "Layer: 0, Models: (8, 2), Overlap: 0.5131419897079468\n",
      "Layer: 0, Models: (8, 3), Overlap: 0.5140944123268127\n",
      "Layer: 0, Models: (8, 4), Overlap: 0.5125042796134949\n",
      "Layer: 0, Models: (8, 5), Overlap: 0.5124277472496033\n",
      "Layer: 0, Models: (8, 6), Overlap: 0.5126062631607056\n",
      "Layer: 0, Models: (8, 7), Overlap: 0.5136479735374451\n",
      "Layer: 0, Models: (8, 9), Overlap: 0.5128146409988403\n",
      "Layer: 0, Models: (8, 10), Overlap: 0.5128273963928223\n",
      "Layer: 0, Models: (8, 11), Overlap: 0.5132653117179871\n",
      "Layer: 0, Models: (8, 12), Overlap: 0.5124107003211975\n",
      "Layer: 0, Models: (8, 13), Overlap: 0.5130527019500732\n",
      "Layer: 0, Models: (8, 14), Overlap: 0.5135331749916077\n",
      "Layer: 0, Models: (9, 0), Overlap: 0.5117644667625427\n",
      "Layer: 0, Models: (9, 1), Overlap: 0.5143877267837524\n",
      "Layer: 0, Models: (9, 2), Overlap: 0.513694703578949\n",
      "Layer: 0, Models: (9, 3), Overlap: 0.5145110487937927\n",
      "Layer: 0, Models: (9, 4), Overlap: 0.5104039311408997\n",
      "Layer: 0, Models: (9, 5), Overlap: 0.5119600296020508\n",
      "Layer: 0, Models: (9, 6), Overlap: 0.5132015347480774\n",
      "Layer: 0, Models: (9, 7), Overlap: 0.5131887793540955\n",
      "Layer: 0, Models: (9, 8), Overlap: 0.5128146409988403\n",
      "Layer: 0, Models: (9, 10), Overlap: 0.5113987922668457\n",
      "Layer: 0, Models: (9, 11), Overlap: 0.513681948184967\n",
      "Layer: 0, Models: (9, 12), Overlap: 0.5126317739486694\n",
      "Layer: 0, Models: (9, 13), Overlap: 0.5120493173599243\n",
      "Layer: 0, Models: (9, 14), Overlap: 0.5143834948539734\n",
      "Layer: 0, Models: (10, 0), Overlap: 0.5147448778152466\n",
      "Layer: 0, Models: (10, 1), Overlap: 0.5102338194847107\n",
      "Layer: 0, Models: (10, 2), Overlap: 0.5135884284973145\n",
      "Layer: 0, Models: (10, 3), Overlap: 0.5128911733627319\n",
      "Layer: 0, Models: (10, 4), Overlap: 0.5133928656578064\n",
      "Layer: 0, Models: (10, 5), Overlap: 0.5107483267784119\n",
      "Layer: 0, Models: (10, 6), Overlap: 0.5134949088096619\n",
      "Layer: 0, Models: (10, 7), Overlap: 0.5114668607711792\n",
      "Layer: 0, Models: (10, 8), Overlap: 0.5128273963928223\n",
      "Layer: 0, Models: (10, 9), Overlap: 0.5113987922668457\n",
      "Layer: 0, Models: (10, 11), Overlap: 0.5106505155563354\n",
      "Layer: 0, Models: (10, 12), Overlap: 0.5126615762710571\n",
      "Layer: 0, Models: (10, 13), Overlap: 0.5129209160804749\n",
      "Layer: 0, Models: (10, 14), Overlap: 0.5113435387611389\n",
      "Layer: 0, Models: (11, 0), Overlap: 0.5122236609458923\n",
      "Layer: 0, Models: (11, 1), Overlap: 0.5123384594917297\n",
      "Layer: 0, Models: (11, 2), Overlap: 0.5120790600776672\n",
      "Layer: 0, Models: (11, 3), Overlap: 0.5124617218971252\n",
      "Layer: 0, Models: (11, 4), Overlap: 0.5125722885131836\n",
      "Layer: 0, Models: (11, 5), Overlap: 0.512130081653595\n",
      "Layer: 0, Models: (11, 6), Overlap: 0.5131419897079468\n",
      "Layer: 0, Models: (11, 7), Overlap: 0.5122959017753601\n",
      "Layer: 0, Models: (11, 8), Overlap: 0.5132653117179871\n",
      "Layer: 0, Models: (11, 9), Overlap: 0.513681948184967\n",
      "Layer: 0, Models: (11, 10), Overlap: 0.5106505155563354\n",
      "Layer: 0, Models: (11, 12), Overlap: 0.5133545994758606\n",
      "Layer: 0, Models: (11, 13), Overlap: 0.5122959017753601\n",
      "Layer: 0, Models: (11, 14), Overlap: 0.5113393068313599\n",
      "Layer: 0, Models: (12, 0), Overlap: 0.51350337266922\n",
      "Layer: 0, Models: (12, 1), Overlap: 0.5123512148857117\n",
      "Layer: 0, Models: (12, 2), Overlap: 0.5121343731880188\n",
      "Layer: 0, Models: (12, 3), Overlap: 0.5122448801994324\n",
      "Layer: 0, Models: (12, 4), Overlap: 0.512704074382782\n",
      "Layer: 0, Models: (12, 5), Overlap: 0.5109098553657532\n",
      "Layer: 0, Models: (12, 6), Overlap: 0.5110969543457031\n",
      "Layer: 0, Models: (12, 7), Overlap: 0.5141113996505737\n",
      "Layer: 0, Models: (12, 8), Overlap: 0.5124107003211975\n",
      "Layer: 0, Models: (12, 9), Overlap: 0.5126317739486694\n",
      "Layer: 0, Models: (12, 10), Overlap: 0.5126615762710571\n",
      "Layer: 0, Models: (12, 11), Overlap: 0.5133545994758606\n",
      "Layer: 0, Models: (12, 13), Overlap: 0.513677716255188\n",
      "Layer: 0, Models: (12, 14), Overlap: 0.5116837024688721\n",
      "Layer: 0, Models: (13, 0), Overlap: 0.5119345188140869\n",
      "Layer: 0, Models: (13, 1), Overlap: 0.5134353637695312\n",
      "Layer: 0, Models: (13, 2), Overlap: 0.5129294395446777\n",
      "Layer: 0, Models: (13, 3), Overlap: 0.5131845474243164\n",
      "Layer: 0, Models: (13, 4), Overlap: 0.511509358882904\n",
      "Layer: 0, Models: (13, 5), Overlap: 0.5146386027336121\n",
      "Layer: 0, Models: (13, 6), Overlap: 0.5095025300979614\n",
      "Layer: 0, Models: (13, 7), Overlap: 0.5124319791793823\n",
      "Layer: 0, Models: (13, 8), Overlap: 0.5130527019500732\n",
      "Layer: 0, Models: (13, 9), Overlap: 0.5120493173599243\n",
      "Layer: 0, Models: (13, 10), Overlap: 0.5129209160804749\n",
      "Layer: 0, Models: (13, 11), Overlap: 0.5122959017753601\n",
      "Layer: 0, Models: (13, 12), Overlap: 0.513677716255188\n",
      "Layer: 0, Models: (13, 14), Overlap: 0.5114923715591431\n",
      "Layer: 0, Models: (14, 0), Overlap: 0.5143027305603027\n",
      "Layer: 0, Models: (14, 1), Overlap: 0.5111352205276489\n",
      "Layer: 0, Models: (14, 2), Overlap: 0.5132483243942261\n",
      "Layer: 0, Models: (14, 3), Overlap: 0.5129761695861816\n",
      "Layer: 0, Models: (14, 4), Overlap: 0.5099660158157349\n",
      "Layer: 0, Models: (14, 5), Overlap: 0.5094302892684937\n",
      "Layer: 0, Models: (14, 6), Overlap: 0.5109778642654419\n",
      "Layer: 0, Models: (14, 7), Overlap: 0.5136181712150574\n",
      "Layer: 0, Models: (14, 8), Overlap: 0.5135331749916077\n",
      "Layer: 0, Models: (14, 9), Overlap: 0.5143834948539734\n",
      "Layer: 0, Models: (14, 10), Overlap: 0.5113435387611389\n",
      "Layer: 0, Models: (14, 11), Overlap: 0.5113393068313599\n",
      "Layer: 0, Models: (14, 12), Overlap: 0.5116837024688721\n",
      "Layer: 0, Models: (14, 13), Overlap: 0.5114923715591431\n",
      "Layer: 2, Models: (0, 1), Overlap: 0.5035111308097839\n",
      "Layer: 2, Models: (0, 2), Overlap: 0.5018222332000732\n",
      "Layer: 2, Models: (0, 3), Overlap: 0.501288890838623\n",
      "Layer: 2, Models: (0, 4), Overlap: 0.5050444602966309\n",
      "Layer: 2, Models: (0, 5), Overlap: 0.5036333203315735\n",
      "Layer: 2, Models: (0, 6), Overlap: 0.5012555718421936\n",
      "Layer: 2, Models: (0, 7), Overlap: 0.5016555786132812\n",
      "Layer: 2, Models: (0, 8), Overlap: 0.5023555755615234\n",
      "Layer: 2, Models: (0, 9), Overlap: 0.5033000111579895\n",
      "Layer: 2, Models: (0, 10), Overlap: 0.5029333233833313\n",
      "Layer: 2, Models: (0, 11), Overlap: 0.5023444294929504\n",
      "Layer: 2, Models: (0, 12), Overlap: 0.5037555694580078\n",
      "Layer: 2, Models: (0, 13), Overlap: 0.5035111308097839\n",
      "Layer: 2, Models: (0, 14), Overlap: 0.503777801990509\n",
      "Layer: 2, Models: (1, 0), Overlap: 0.5035111308097839\n",
      "Layer: 2, Models: (1, 2), Overlap: 0.5052666664123535\n",
      "Layer: 2, Models: (1, 3), Overlap: 0.5029555559158325\n",
      "Layer: 2, Models: (1, 4), Overlap: 0.49984443187713623\n",
      "Layer: 2, Models: (1, 5), Overlap: 0.5020555257797241\n",
      "Layer: 2, Models: (1, 6), Overlap: 0.5048999786376953\n",
      "Layer: 2, Models: (1, 7), Overlap: 0.5056777596473694\n",
      "Layer: 2, Models: (1, 8), Overlap: 0.5005333423614502\n",
      "Layer: 2, Models: (1, 9), Overlap: 0.5012111067771912\n",
      "Layer: 2, Models: (1, 10), Overlap: 0.5021555423736572\n",
      "Layer: 2, Models: (1, 11), Overlap: 0.5053666830062866\n",
      "Layer: 2, Models: (1, 12), Overlap: 0.5031333565711975\n",
      "Layer: 2, Models: (1, 13), Overlap: 0.5016444325447083\n",
      "Layer: 2, Models: (1, 14), Overlap: 0.49977776408195496\n",
      "Layer: 2, Models: (2, 0), Overlap: 0.5018222332000732\n",
      "Layer: 2, Models: (2, 1), Overlap: 0.5052666664123535\n",
      "Layer: 2, Models: (2, 3), Overlap: 0.5053777694702148\n",
      "Layer: 2, Models: (2, 4), Overlap: 0.5054888725280762\n",
      "Layer: 2, Models: (2, 5), Overlap: 0.5024555325508118\n",
      "Layer: 2, Models: (2, 6), Overlap: 0.5068777799606323\n",
      "Layer: 2, Models: (2, 7), Overlap: 0.5007444620132446\n",
      "Layer: 2, Models: (2, 8), Overlap: 0.5024666786193848\n",
      "Layer: 2, Models: (2, 9), Overlap: 0.5029444694519043\n",
      "Layer: 2, Models: (2, 10), Overlap: 0.5038889050483704\n",
      "Layer: 2, Models: (2, 11), Overlap: 0.5056333541870117\n",
      "Layer: 2, Models: (2, 12), Overlap: 0.5023333430290222\n",
      "Layer: 2, Models: (2, 13), Overlap: 0.5032888650894165\n",
      "Layer: 2, Models: (2, 14), Overlap: 0.500333309173584\n",
      "Layer: 2, Models: (3, 0), Overlap: 0.501288890838623\n",
      "Layer: 2, Models: (3, 1), Overlap: 0.5029555559158325\n",
      "Layer: 2, Models: (3, 2), Overlap: 0.5053777694702148\n",
      "Layer: 2, Models: (3, 4), Overlap: 0.5028889179229736\n",
      "Layer: 2, Models: (3, 5), Overlap: 0.5024333596229553\n",
      "Layer: 2, Models: (3, 6), Overlap: 0.5004777908325195\n",
      "Layer: 2, Models: (3, 7), Overlap: 0.5015666484832764\n",
      "Layer: 2, Models: (3, 8), Overlap: 0.5039777755737305\n",
      "Layer: 2, Models: (3, 9), Overlap: 0.5019222497940063\n",
      "Layer: 2, Models: (3, 10), Overlap: 0.5009777545928955\n",
      "Layer: 2, Models: (3, 11), Overlap: 0.5015888810157776\n",
      "Layer: 2, Models: (3, 12), Overlap: 0.504800021648407\n",
      "Layer: 2, Models: (3, 13), Overlap: 0.5006666779518127\n",
      "Layer: 2, Models: (3, 14), Overlap: 0.5019111037254333\n",
      "Layer: 2, Models: (4, 0), Overlap: 0.5050444602966309\n",
      "Layer: 2, Models: (4, 1), Overlap: 0.49984443187713623\n",
      "Layer: 2, Models: (4, 2), Overlap: 0.5054888725280762\n",
      "Layer: 2, Models: (4, 3), Overlap: 0.5028889179229736\n",
      "Layer: 2, Models: (4, 5), Overlap: 0.5020111203193665\n",
      "Layer: 2, Models: (4, 6), Overlap: 0.49727776646614075\n",
      "Layer: 2, Models: (4, 7), Overlap: 0.5023000240325928\n",
      "Layer: 2, Models: (4, 8), Overlap: 0.5039555430412292\n",
      "Layer: 2, Models: (4, 9), Overlap: 0.49950000643730164\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Layer: 2, Models: (4, 10), Overlap: 0.505577802658081\n",
      "Layer: 2, Models: (4, 11), Overlap: 0.5021666884422302\n",
      "Layer: 2, Models: (4, 12), Overlap: 0.5025110840797424\n",
      "Layer: 2, Models: (4, 13), Overlap: 0.5019111037254333\n",
      "Layer: 2, Models: (4, 14), Overlap: 0.5006444454193115\n",
      "Layer: 2, Models: (5, 0), Overlap: 0.5036333203315735\n",
      "Layer: 2, Models: (5, 1), Overlap: 0.5020555257797241\n",
      "Layer: 2, Models: (5, 2), Overlap: 0.5024555325508118\n",
      "Layer: 2, Models: (5, 3), Overlap: 0.5024333596229553\n",
      "Layer: 2, Models: (5, 4), Overlap: 0.5020111203193665\n",
      "Layer: 2, Models: (5, 6), Overlap: 0.5037333369255066\n",
      "Layer: 2, Models: (5, 7), Overlap: 0.5029110908508301\n",
      "Layer: 2, Models: (5, 8), Overlap: 0.5046555399894714\n",
      "Layer: 2, Models: (5, 9), Overlap: 0.503155529499054\n",
      "Layer: 2, Models: (5, 10), Overlap: 0.5000777840614319\n",
      "Layer: 2, Models: (5, 11), Overlap: 0.5025333166122437\n",
      "Layer: 2, Models: (5, 12), Overlap: 0.503433346748352\n",
      "Layer: 2, Models: (5, 13), Overlap: 0.5032111406326294\n",
      "Layer: 2, Models: (5, 14), Overlap: 0.503766655921936\n",
      "Layer: 2, Models: (6, 0), Overlap: 0.5012555718421936\n",
      "Layer: 2, Models: (6, 1), Overlap: 0.5048999786376953\n",
      "Layer: 2, Models: (6, 2), Overlap: 0.5068777799606323\n",
      "Layer: 2, Models: (6, 3), Overlap: 0.5004777908325195\n",
      "Layer: 2, Models: (6, 4), Overlap: 0.49727776646614075\n",
      "Layer: 2, Models: (6, 5), Overlap: 0.5037333369255066\n",
      "Layer: 2, Models: (6, 7), Overlap: 0.5034888982772827\n",
      "Layer: 2, Models: (6, 8), Overlap: 0.5033666491508484\n",
      "Layer: 2, Models: (6, 9), Overlap: 0.5056666731834412\n",
      "Layer: 2, Models: (6, 10), Overlap: 0.5020999908447266\n",
      "Layer: 2, Models: (6, 11), Overlap: 0.5022444725036621\n",
      "Layer: 2, Models: (6, 12), Overlap: 0.502144455909729\n",
      "Layer: 2, Models: (6, 13), Overlap: 0.5010777711868286\n",
      "Layer: 2, Models: (6, 14), Overlap: 0.5040777921676636\n",
      "Layer: 2, Models: (7, 0), Overlap: 0.5016555786132812\n",
      "Layer: 2, Models: (7, 1), Overlap: 0.5056777596473694\n",
      "Layer: 2, Models: (7, 2), Overlap: 0.5007444620132446\n",
      "Layer: 2, Models: (7, 3), Overlap: 0.5015666484832764\n",
      "Layer: 2, Models: (7, 4), Overlap: 0.5023000240325928\n",
      "Layer: 2, Models: (7, 5), Overlap: 0.5029110908508301\n",
      "Layer: 2, Models: (7, 6), Overlap: 0.5034888982772827\n",
      "Layer: 2, Models: (7, 8), Overlap: 0.5036110877990723\n",
      "Layer: 2, Models: (7, 9), Overlap: 0.501622200012207\n",
      "Layer: 2, Models: (7, 10), Overlap: 0.5014111399650574\n",
      "Layer: 2, Models: (7, 11), Overlap: 0.5037111043930054\n",
      "Layer: 2, Models: (7, 12), Overlap: 0.5028777718544006\n",
      "Layer: 2, Models: (7, 13), Overlap: 0.5021222233772278\n",
      "Layer: 2, Models: (7, 14), Overlap: 0.504788875579834\n",
      "Layer: 2, Models: (8, 0), Overlap: 0.5023555755615234\n",
      "Layer: 2, Models: (8, 1), Overlap: 0.5005333423614502\n",
      "Layer: 2, Models: (8, 2), Overlap: 0.5024666786193848\n",
      "Layer: 2, Models: (8, 3), Overlap: 0.5039777755737305\n",
      "Layer: 2, Models: (8, 4), Overlap: 0.5039555430412292\n",
      "Layer: 2, Models: (8, 5), Overlap: 0.5046555399894714\n",
      "Layer: 2, Models: (8, 6), Overlap: 0.5033666491508484\n",
      "Layer: 2, Models: (8, 7), Overlap: 0.5036110877990723\n",
      "Layer: 2, Models: (8, 9), Overlap: 0.5045222043991089\n",
      "Layer: 2, Models: (8, 10), Overlap: 0.4997555613517761\n",
      "Layer: 2, Models: (8, 11), Overlap: 0.5022777915000916\n",
      "Layer: 2, Models: (8, 12), Overlap: 0.500511109828949\n",
      "Layer: 2, Models: (8, 13), Overlap: 0.5044222474098206\n",
      "Layer: 2, Models: (8, 14), Overlap: 0.5035333037376404\n",
      "Layer: 2, Models: (9, 0), Overlap: 0.5033000111579895\n",
      "Layer: 2, Models: (9, 1), Overlap: 0.5012111067771912\n",
      "Layer: 2, Models: (9, 2), Overlap: 0.5029444694519043\n",
      "Layer: 2, Models: (9, 3), Overlap: 0.5019222497940063\n",
      "Layer: 2, Models: (9, 4), Overlap: 0.49950000643730164\n",
      "Layer: 2, Models: (9, 5), Overlap: 0.503155529499054\n",
      "Layer: 2, Models: (9, 6), Overlap: 0.5056666731834412\n",
      "Layer: 2, Models: (9, 7), Overlap: 0.501622200012207\n",
      "Layer: 2, Models: (9, 8), Overlap: 0.5045222043991089\n",
      "Layer: 2, Models: (9, 10), Overlap: 0.5012111067771912\n",
      "Layer: 2, Models: (9, 11), Overlap: 0.5027777552604675\n",
      "Layer: 2, Models: (9, 12), Overlap: 0.5012555718421936\n",
      "Layer: 2, Models: (9, 13), Overlap: 0.5040110945701599\n",
      "Layer: 2, Models: (9, 14), Overlap: 0.5015222430229187\n",
      "Layer: 2, Models: (10, 0), Overlap: 0.5029333233833313\n",
      "Layer: 2, Models: (10, 1), Overlap: 0.5021555423736572\n",
      "Layer: 2, Models: (10, 2), Overlap: 0.5038889050483704\n",
      "Layer: 2, Models: (10, 3), Overlap: 0.5009777545928955\n",
      "Layer: 2, Models: (10, 4), Overlap: 0.505577802658081\n",
      "Layer: 2, Models: (10, 5), Overlap: 0.5000777840614319\n",
      "Layer: 2, Models: (10, 6), Overlap: 0.5020999908447266\n",
      "Layer: 2, Models: (10, 7), Overlap: 0.5014111399650574\n",
      "Layer: 2, Models: (10, 8), Overlap: 0.4997555613517761\n",
      "Layer: 2, Models: (10, 9), Overlap: 0.5012111067771912\n",
      "Layer: 2, Models: (10, 11), Overlap: 0.5015444159507751\n",
      "Layer: 2, Models: (10, 12), Overlap: 0.5033555626869202\n",
      "Layer: 2, Models: (10, 13), Overlap: 0.5000444650650024\n",
      "Layer: 2, Models: (10, 14), Overlap: 0.5013555288314819\n",
      "Layer: 2, Models: (11, 0), Overlap: 0.5023444294929504\n",
      "Layer: 2, Models: (11, 1), Overlap: 0.5053666830062866\n",
      "Layer: 2, Models: (11, 2), Overlap: 0.5056333541870117\n",
      "Layer: 2, Models: (11, 3), Overlap: 0.5015888810157776\n",
      "Layer: 2, Models: (11, 4), Overlap: 0.5021666884422302\n",
      "Layer: 2, Models: (11, 5), Overlap: 0.5025333166122437\n",
      "Layer: 2, Models: (11, 6), Overlap: 0.5022444725036621\n",
      "Layer: 2, Models: (11, 7), Overlap: 0.5037111043930054\n",
      "Layer: 2, Models: (11, 8), Overlap: 0.5022777915000916\n",
      "Layer: 2, Models: (11, 9), Overlap: 0.5027777552604675\n",
      "Layer: 2, Models: (11, 10), Overlap: 0.5015444159507751\n",
      "Layer: 2, Models: (11, 12), Overlap: 0.5009889006614685\n",
      "Layer: 2, Models: (11, 13), Overlap: 0.5046555399894714\n",
      "Layer: 2, Models: (11, 14), Overlap: 0.5015000104904175\n",
      "Layer: 2, Models: (12, 0), Overlap: 0.5037555694580078\n",
      "Layer: 2, Models: (12, 1), Overlap: 0.5031333565711975\n",
      "Layer: 2, Models: (12, 2), Overlap: 0.5023333430290222\n",
      "Layer: 2, Models: (12, 3), Overlap: 0.504800021648407\n",
      "Layer: 2, Models: (12, 4), Overlap: 0.5025110840797424\n",
      "Layer: 2, Models: (12, 5), Overlap: 0.503433346748352\n",
      "Layer: 2, Models: (12, 6), Overlap: 0.502144455909729\n",
      "Layer: 2, Models: (12, 7), Overlap: 0.5028777718544006\n",
      "Layer: 2, Models: (12, 8), Overlap: 0.500511109828949\n",
      "Layer: 2, Models: (12, 9), Overlap: 0.5012555718421936\n",
      "Layer: 2, Models: (12, 10), Overlap: 0.5033555626869202\n",
      "Layer: 2, Models: (12, 11), Overlap: 0.5009889006614685\n",
      "Layer: 2, Models: (12, 13), Overlap: 0.5039555430412292\n",
      "Layer: 2, Models: (12, 14), Overlap: 0.5068666934967041\n",
      "Layer: 2, Models: (13, 0), Overlap: 0.5035111308097839\n",
      "Layer: 2, Models: (13, 1), Overlap: 0.5016444325447083\n",
      "Layer: 2, Models: (13, 2), Overlap: 0.5032888650894165\n",
      "Layer: 2, Models: (13, 3), Overlap: 0.5006666779518127\n",
      "Layer: 2, Models: (13, 4), Overlap: 0.5019111037254333\n",
      "Layer: 2, Models: (13, 5), Overlap: 0.5032111406326294\n",
      "Layer: 2, Models: (13, 6), Overlap: 0.5010777711868286\n",
      "Layer: 2, Models: (13, 7), Overlap: 0.5021222233772278\n",
      "Layer: 2, Models: (13, 8), Overlap: 0.5044222474098206\n",
      "Layer: 2, Models: (13, 9), Overlap: 0.5040110945701599\n",
      "Layer: 2, Models: (13, 10), Overlap: 0.5000444650650024\n",
      "Layer: 2, Models: (13, 11), Overlap: 0.5046555399894714\n",
      "Layer: 2, Models: (13, 12), Overlap: 0.5039555430412292\n",
      "Layer: 2, Models: (13, 14), Overlap: 0.5013111233711243\n",
      "Layer: 2, Models: (14, 0), Overlap: 0.503777801990509\n",
      "Layer: 2, Models: (14, 1), Overlap: 0.49977776408195496\n",
      "Layer: 2, Models: (14, 2), Overlap: 0.500333309173584\n",
      "Layer: 2, Models: (14, 3), Overlap: 0.5019111037254333\n",
      "Layer: 2, Models: (14, 4), Overlap: 0.5006444454193115\n",
      "Layer: 2, Models: (14, 5), Overlap: 0.503766655921936\n",
      "Layer: 2, Models: (14, 6), Overlap: 0.5040777921676636\n",
      "Layer: 2, Models: (14, 7), Overlap: 0.504788875579834\n",
      "Layer: 2, Models: (14, 8), Overlap: 0.5035333037376404\n",
      "Layer: 2, Models: (14, 9), Overlap: 0.5015222430229187\n",
      "Layer: 2, Models: (14, 10), Overlap: 0.5013555288314819\n",
      "Layer: 2, Models: (14, 11), Overlap: 0.5015000104904175\n",
      "Layer: 2, Models: (14, 12), Overlap: 0.5068666934967041\n",
      "Layer: 2, Models: (14, 13), Overlap: 0.5013111233711243\n",
      "Layer: 4, Models: (0, 1), Overlap: 0.5085333585739136\n",
      "Layer: 4, Models: (0, 2), Overlap: 0.5146999955177307\n",
      "Layer: 4, Models: (0, 3), Overlap: 0.5140666961669922\n",
      "Layer: 4, Models: (0, 4), Overlap: 0.5147666931152344\n",
      "Layer: 4, Models: (0, 5), Overlap: 0.5136666893959045\n",
      "Layer: 4, Models: (0, 6), Overlap: 0.5113333463668823\n",
      "Layer: 4, Models: (0, 7), Overlap: 0.51419997215271\n",
      "Layer: 4, Models: (0, 8), Overlap: 0.5169333219528198\n",
      "Layer: 4, Models: (0, 9), Overlap: 0.5062666535377502\n",
      "Layer: 4, Models: (0, 10), Overlap: 0.5163000226020813\n",
      "Layer: 4, Models: (0, 11), Overlap: 0.5098999738693237\n",
      "Layer: 4, Models: (0, 12), Overlap: 0.5148333311080933\n",
      "Layer: 4, Models: (0, 13), Overlap: 0.5144333243370056\n",
      "Layer: 4, Models: (0, 14), Overlap: 0.5082333087921143\n",
      "Layer: 4, Models: (1, 0), Overlap: 0.5085333585739136\n",
      "Layer: 4, Models: (1, 2), Overlap: 0.5145666599273682\n",
      "Layer: 4, Models: (1, 3), Overlap: 0.5159333348274231\n",
      "Layer: 4, Models: (1, 4), Overlap: 0.5151666402816772\n",
      "Layer: 4, Models: (1, 5), Overlap: 0.5118666887283325\n",
      "Layer: 4, Models: (1, 6), Overlap: 0.512333333492279\n",
      "Layer: 4, Models: (1, 7), Overlap: 0.510200023651123\n",
      "Layer: 4, Models: (1, 8), Overlap: 0.5121999979019165\n",
      "Layer: 4, Models: (1, 9), Overlap: 0.5109999775886536\n",
      "Layer: 4, Models: (1, 10), Overlap: 0.5120333433151245\n",
      "Layer: 4, Models: (1, 11), Overlap: 0.5152333378791809\n",
      "Layer: 4, Models: (1, 12), Overlap: 0.5162333250045776\n",
      "Layer: 4, Models: (1, 13), Overlap: 0.5091666579246521\n",
      "Layer: 4, Models: (1, 14), Overlap: 0.5116333365440369\n",
      "Layer: 4, Models: (2, 0), Overlap: 0.5146999955177307\n",
      "Layer: 4, Models: (2, 1), Overlap: 0.5145666599273682\n",
      "Layer: 4, Models: (2, 3), Overlap: 0.5163666605949402\n",
      "Layer: 4, Models: (2, 4), Overlap: 0.51173335313797\n",
      "Layer: 4, Models: (2, 5), Overlap: 0.5165666937828064\n",
      "Layer: 4, Models: (2, 6), Overlap: 0.5163000226020813\n",
      "Layer: 4, Models: (2, 7), Overlap: 0.5143666863441467\n",
      "Layer: 4, Models: (2, 8), Overlap: 0.5121666789054871\n",
      "Layer: 4, Models: (2, 9), Overlap: 0.5136333107948303\n",
      "Layer: 4, Models: (2, 10), Overlap: 0.5109999775886536\n",
      "Layer: 4, Models: (2, 11), Overlap: 0.5185999870300293\n",
      "Layer: 4, Models: (2, 12), Overlap: 0.5112666487693787\n",
      "Layer: 4, Models: (2, 13), Overlap: 0.518666684627533\n",
      "Layer: 4, Models: (2, 14), Overlap: 0.51419997215271\n",
      "Layer: 4, Models: (3, 0), Overlap: 0.5140666961669922\n",
      "Layer: 4, Models: (3, 1), Overlap: 0.5159333348274231\n",
      "Layer: 4, Models: (3, 2), Overlap: 0.5163666605949402\n",
      "Layer: 4, Models: (3, 4), Overlap: 0.510699987411499\n",
      "Layer: 4, Models: (3, 5), Overlap: 0.5134000182151794\n",
      "Layer: 4, Models: (3, 6), Overlap: 0.5161333084106445\n",
      "Layer: 4, Models: (3, 7), Overlap: 0.5078666806221008\n",
      "Layer: 4, Models: (3, 8), Overlap: 0.5160666704177856\n",
      "Layer: 4, Models: (3, 9), Overlap: 0.509933352470398\n",
      "Layer: 4, Models: (3, 10), Overlap: 0.5163666605949402\n",
      "Layer: 4, Models: (3, 11), Overlap: 0.5092999935150146\n",
      "Layer: 4, Models: (3, 12), Overlap: 0.5169666409492493\n",
      "Layer: 4, Models: (3, 13), Overlap: 0.5108333230018616\n",
      "Layer: 4, Models: (3, 14), Overlap: 0.5131666660308838\n",
      "Layer: 4, Models: (4, 0), Overlap: 0.5147666931152344\n",
      "Layer: 4, Models: (4, 1), Overlap: 0.5151666402816772\n",
      "Layer: 4, Models: (4, 2), Overlap: 0.51173335313797\n",
      "Layer: 4, Models: (4, 3), Overlap: 0.510699987411499\n",
      "Layer: 4, Models: (4, 5), Overlap: 0.5160333514213562\n",
      "Layer: 4, Models: (4, 6), Overlap: 0.5138999819755554\n",
      "Layer: 4, Models: (4, 7), Overlap: 0.5129666924476624\n",
      "Layer: 4, Models: (4, 8), Overlap: 0.516700029373169\n",
      "Layer: 4, Models: (4, 9), Overlap: 0.51583331823349\n",
      "Layer: 4, Models: (4, 10), Overlap: 0.5145333409309387\n",
      "Layer: 4, Models: (4, 11), Overlap: 0.5103333592414856\n",
      "Layer: 4, Models: (4, 12), Overlap: 0.515333354473114\n",
      "Layer: 4, Models: (4, 13), Overlap: 0.5136666893959045\n",
      "Layer: 4, Models: (4, 14), Overlap: 0.5127333402633667\n",
      "Layer: 4, Models: (5, 0), Overlap: 0.5136666893959045\n",
      "Layer: 4, Models: (5, 1), Overlap: 0.5118666887283325\n",
      "Layer: 4, Models: (5, 2), Overlap: 0.5165666937828064\n",
      "Layer: 4, Models: (5, 3), Overlap: 0.5134000182151794\n",
      "Layer: 4, Models: (5, 4), Overlap: 0.5160333514213562\n",
      "Layer: 4, Models: (5, 6), Overlap: 0.5163333415985107\n",
      "Layer: 4, Models: (5, 7), Overlap: 0.5157333612442017\n",
      "Layer: 4, Models: (5, 8), Overlap: 0.5109999775886536\n",
      "Layer: 4, Models: (5, 9), Overlap: 0.5184000134468079\n",
      "Layer: 4, Models: (5, 10), Overlap: 0.508566677570343\n",
      "Layer: 4, Models: (5, 11), Overlap: 0.512499988079071\n",
      "Layer: 4, Models: (5, 12), Overlap: 0.5132333040237427\n",
      "Layer: 4, Models: (5, 13), Overlap: 0.5109000205993652\n",
      "Layer: 4, Models: (5, 14), Overlap: 0.5097000002861023\n",
      "Layer: 4, Models: (6, 0), Overlap: 0.5113333463668823\n",
      "Layer: 4, Models: (6, 1), Overlap: 0.512333333492279\n",
      "Layer: 4, Models: (6, 2), Overlap: 0.5163000226020813\n",
      "Layer: 4, Models: (6, 3), Overlap: 0.5161333084106445\n",
      "Layer: 4, Models: (6, 4), Overlap: 0.5138999819755554\n",
      "Layer: 4, Models: (6, 5), Overlap: 0.5163333415985107\n",
      "Layer: 4, Models: (6, 7), Overlap: 0.5170000195503235\n",
      "Layer: 4, Models: (6, 8), Overlap: 0.5176666378974915\n",
      "Layer: 4, Models: (6, 9), Overlap: 0.5120000243186951\n",
      "Layer: 4, Models: (6, 10), Overlap: 0.5160999894142151\n",
      "Layer: 4, Models: (6, 11), Overlap: 0.5115000009536743\n",
      "Layer: 4, Models: (6, 12), Overlap: 0.5163666605949402\n",
      "Layer: 4, Models: (6, 13), Overlap: 0.5112333297729492\n",
      "Layer: 4, Models: (6, 14), Overlap: 0.5146333575248718\n",
      "Layer: 4, Models: (7, 0), Overlap: 0.51419997215271\n",
      "Layer: 4, Models: (7, 1), Overlap: 0.510200023651123\n",
      "Layer: 4, Models: (7, 2), Overlap: 0.5143666863441467\n",
      "Layer: 4, Models: (7, 3), Overlap: 0.5078666806221008\n",
      "Layer: 4, Models: (7, 4), Overlap: 0.5129666924476624\n",
      "Layer: 4, Models: (7, 5), Overlap: 0.5157333612442017\n",
      "Layer: 4, Models: (7, 6), Overlap: 0.5170000195503235\n",
      "Layer: 4, Models: (7, 8), Overlap: 0.5151333212852478\n",
      "Layer: 4, Models: (7, 9), Overlap: 0.5166666507720947\n",
      "Layer: 4, Models: (7, 10), Overlap: 0.5129666924476624\n",
      "Layer: 4, Models: (7, 11), Overlap: 0.5136333107948303\n",
      "Layer: 4, Models: (7, 12), Overlap: 0.5172333121299744\n",
      "Layer: 4, Models: (7, 13), Overlap: 0.5098999738693237\n",
      "Layer: 4, Models: (7, 14), Overlap: 0.5182333588600159\n",
      "Layer: 4, Models: (8, 0), Overlap: 0.5169333219528198\n",
      "Layer: 4, Models: (8, 1), Overlap: 0.5121999979019165\n",
      "Layer: 4, Models: (8, 2), Overlap: 0.5121666789054871\n",
      "Layer: 4, Models: (8, 3), Overlap: 0.5160666704177856\n",
      "Layer: 4, Models: (8, 4), Overlap: 0.516700029373169\n",
      "Layer: 4, Models: (8, 5), Overlap: 0.5109999775886536\n",
      "Layer: 4, Models: (8, 6), Overlap: 0.5176666378974915\n",
      "Layer: 4, Models: (8, 7), Overlap: 0.5151333212852478\n",
      "Layer: 4, Models: (8, 9), Overlap: 0.5109999775886536\n",
      "Layer: 4, Models: (8, 10), Overlap: 0.5077000260353088\n",
      "Layer: 4, Models: (8, 11), Overlap: 0.5113000273704529\n",
      "Layer: 4, Models: (8, 12), Overlap: 0.5160999894142151\n",
      "Layer: 4, Models: (8, 13), Overlap: 0.5160333514213562\n",
      "Layer: 4, Models: (8, 14), Overlap: 0.5128999948501587\n",
      "Layer: 4, Models: (9, 0), Overlap: 0.5062666535377502\n",
      "Layer: 4, Models: (9, 1), Overlap: 0.5109999775886536\n",
      "Layer: 4, Models: (9, 2), Overlap: 0.5136333107948303\n",
      "Layer: 4, Models: (9, 3), Overlap: 0.509933352470398\n",
      "Layer: 4, Models: (9, 4), Overlap: 0.51583331823349\n",
      "Layer: 4, Models: (9, 5), Overlap: 0.5184000134468079\n",
      "Layer: 4, Models: (9, 6), Overlap: 0.5120000243186951\n",
      "Layer: 4, Models: (9, 7), Overlap: 0.5166666507720947\n",
      "Layer: 4, Models: (9, 8), Overlap: 0.5109999775886536\n",
      "Layer: 4, Models: (9, 10), Overlap: 0.5089666843414307\n",
      "Layer: 4, Models: (9, 11), Overlap: 0.5160999894142151\n",
      "Layer: 4, Models: (9, 12), Overlap: 0.5210333466529846\n",
      "Layer: 4, Models: (9, 13), Overlap: 0.510366678237915\n",
      "Layer: 4, Models: (9, 14), Overlap: 0.5121666789054871\n",
      "Layer: 4, Models: (10, 0), Overlap: 0.5163000226020813\n",
      "Layer: 4, Models: (10, 1), Overlap: 0.5120333433151245\n",
      "Layer: 4, Models: (10, 2), Overlap: 0.5109999775886536\n",
      "Layer: 4, Models: (10, 3), Overlap: 0.5163666605949402\n",
      "Layer: 4, Models: (10, 4), Overlap: 0.5145333409309387\n",
      "Layer: 4, Models: (10, 5), Overlap: 0.508566677570343\n",
      "Layer: 4, Models: (10, 6), Overlap: 0.5160999894142151\n",
      "Layer: 4, Models: (10, 7), Overlap: 0.5129666924476624\n",
      "Layer: 4, Models: (10, 8), Overlap: 0.5077000260353088\n",
      "Layer: 4, Models: (10, 9), Overlap: 0.5089666843414307\n",
      "Layer: 4, Models: (10, 11), Overlap: 0.5113999843597412\n",
      "Layer: 4, Models: (10, 12), Overlap: 0.5144666433334351\n",
      "Layer: 4, Models: (10, 13), Overlap: 0.517799973487854\n",
      "Layer: 4, Models: (10, 14), Overlap: 0.5095333456993103\n",
      "Layer: 4, Models: (11, 0), Overlap: 0.5098999738693237\n",
      "Layer: 4, Models: (11, 1), Overlap: 0.5152333378791809\n",
      "Layer: 4, Models: (11, 2), Overlap: 0.5185999870300293\n",
      "Layer: 4, Models: (11, 3), Overlap: 0.5092999935150146\n",
      "Layer: 4, Models: (11, 4), Overlap: 0.5103333592414856\n",
      "Layer: 4, Models: (11, 5), Overlap: 0.512499988079071\n",
      "Layer: 4, Models: (11, 6), Overlap: 0.5115000009536743\n",
      "Layer: 4, Models: (11, 7), Overlap: 0.5136333107948303\n",
      "Layer: 4, Models: (11, 8), Overlap: 0.5113000273704529\n",
      "Layer: 4, Models: (11, 9), Overlap: 0.5160999894142151\n",
      "Layer: 4, Models: (11, 10), Overlap: 0.5113999843597412\n",
      "Layer: 4, Models: (11, 12), Overlap: 0.510533332824707\n",
      "Layer: 4, Models: (11, 13), Overlap: 0.5134000182151794\n",
      "Layer: 4, Models: (11, 14), Overlap: 0.5127333402633667\n",
      "Layer: 4, Models: (12, 0), Overlap: 0.5148333311080933\n",
      "Layer: 4, Models: (12, 1), Overlap: 0.5162333250045776\n",
      "Layer: 4, Models: (12, 2), Overlap: 0.5112666487693787\n",
      "Layer: 4, Models: (12, 3), Overlap: 0.5169666409492493\n",
      "Layer: 4, Models: (12, 4), Overlap: 0.515333354473114\n",
      "Layer: 4, Models: (12, 5), Overlap: 0.5132333040237427\n",
      "Layer: 4, Models: (12, 6), Overlap: 0.5163666605949402\n",
      "Layer: 4, Models: (12, 7), Overlap: 0.5172333121299744\n",
      "Layer: 4, Models: (12, 8), Overlap: 0.5160999894142151\n",
      "Layer: 4, Models: (12, 9), Overlap: 0.5210333466529846\n",
      "Layer: 4, Models: (12, 10), Overlap: 0.5144666433334351\n",
      "Layer: 4, Models: (12, 11), Overlap: 0.510533332824707\n",
      "Layer: 4, Models: (12, 13), Overlap: 0.517799973487854\n",
      "Layer: 4, Models: (12, 14), Overlap: 0.5182666778564453\n",
      "Layer: 4, Models: (13, 0), Overlap: 0.5144333243370056\n",
      "Layer: 4, Models: (13, 1), Overlap: 0.5091666579246521\n",
      "Layer: 4, Models: (13, 2), Overlap: 0.518666684627533\n",
      "Layer: 4, Models: (13, 3), Overlap: 0.5108333230018616\n",
      "Layer: 4, Models: (13, 4), Overlap: 0.5136666893959045\n",
      "Layer: 4, Models: (13, 5), Overlap: 0.5109000205993652\n",
      "Layer: 4, Models: (13, 6), Overlap: 0.5112333297729492\n",
      "Layer: 4, Models: (13, 7), Overlap: 0.5098999738693237\n",
      "Layer: 4, Models: (13, 8), Overlap: 0.5160333514213562\n",
      "Layer: 4, Models: (13, 9), Overlap: 0.510366678237915\n",
      "Layer: 4, Models: (13, 10), Overlap: 0.517799973487854\n",
      "Layer: 4, Models: (13, 11), Overlap: 0.5134000182151794\n",
      "Layer: 4, Models: (13, 12), Overlap: 0.517799973487854\n",
      "Layer: 4, Models: (13, 14), Overlap: 0.5124666690826416\n",
      "Layer: 4, Models: (14, 0), Overlap: 0.5082333087921143\n",
      "Layer: 4, Models: (14, 1), Overlap: 0.5116333365440369\n",
      "Layer: 4, Models: (14, 2), Overlap: 0.51419997215271\n",
      "Layer: 4, Models: (14, 3), Overlap: 0.5131666660308838\n",
      "Layer: 4, Models: (14, 4), Overlap: 0.5127333402633667\n",
      "Layer: 4, Models: (14, 5), Overlap: 0.5097000002861023\n",
      "Layer: 4, Models: (14, 6), Overlap: 0.5146333575248718\n",
      "Layer: 4, Models: (14, 7), Overlap: 0.5182333588600159\n",
      "Layer: 4, Models: (14, 8), Overlap: 0.5128999948501587\n",
      "Layer: 4, Models: (14, 9), Overlap: 0.5121666789054871\n",
      "Layer: 4, Models: (14, 10), Overlap: 0.5095333456993103\n",
      "Layer: 4, Models: (14, 11), Overlap: 0.5127333402633667\n",
      "Layer: 4, Models: (14, 12), Overlap: 0.5182666778564453\n",
      "Layer: 4, Models: (14, 13), Overlap: 0.5124666690826416\n"
     ]
    }
   ],
   "source": [
    "task = 0\n",
    "for layer, modelidx in product([0,2,4], product([i for i in range(15)], [i for i in range(15)])):\n",
    "    modeli, modelj = modelidx\n",
    "    if modeli == modelj:\n",
    "        continue\n",
    "    print('Layer: {}, Models: ({}, {}), Overlap: {}'.format(layer, modeli, modelj,\n",
    "        overlap(model_map[modeli].state_dict()['model.{}.stacked'.format(layer)][task],\n",
    "                model_map[modelj].state_dict()['model.{}.stacked'.format(layer)][task])))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Multimask Same Task (Sparsity 25)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 213,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_masks_to_create = 15"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 214,
   "metadata": {},
   "outputs": [],
   "source": [
    "loaders = {}\n",
    "for i in range(num_masks_to_create):\n",
    "    loaders[i] = MNISTPerm()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 215,
   "metadata": {},
   "outputs": [],
   "source": [
    "weight_dict = {k: v for k,v in models.state_dict().items() if k.endswith('weight')}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 216,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Task ID given during train time\n",
    "from IPython.display import clear_output\n",
    "\n",
    "model_maps = {}\n",
    "for i in range(num_masks_to_create):\n",
    "    modeli = MultitaskFCSparse(hidden_size=300, num_tasks=1, sparsity=0.25)\n",
    "    sdi = modeli.state_dict()\n",
    "    sdi.update(weight_dict)\n",
    "    modeli.load_state_dict(sdi)\n",
    "    model_maps[i] = modeli"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 217,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1b4ffe2f6ab84e6c96bd90213f432a0b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2764, Acc@1 0.8298\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "748f56da6e9e4e8b92ec729c426970b7",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9111\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "bcabeef64ad64e798c0f411da2330387",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3107, Acc@1 0.8330\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "16efb555527843d6b97c385397a72105",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9105\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "62afc78613b04fbf9db9a1989eaeb4dc",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2673, Acc@1 0.8320\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f3db17dce59740e0bcad643e25a1d3ae",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9130\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d63597214fef46e3a8e8d5b300af297f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3994, Acc@1 0.8379\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7ecacb60e0404436acbec3557d7855e5",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9136\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6045264adcf9433eba2826b2c93a6353",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3628, Acc@1 0.8438\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "85effc9609954527b9d9f901200b5c6d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9119\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f0d3a5c8e8da45a7aa760ee3374b6236",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3440, Acc@1 0.8298\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "025efc28526a4e70be80fc33a265fe7e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9142\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a574902e8bcc47a895ade27d1b6c45f5",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.4477, Acc@1 0.8454\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "417de54ccd904e779cb7f7d1100666ea",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9139\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "17cf944454c44dcf93ab5b5fe9666888",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.2559, Acc@1 0.8467\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1a282ff878be4dd69aa04eaf65e71ce3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9119\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "29b0d8009a6947ae9fb092fb5b20019d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3870, Acc@1 0.8392\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8b166a47b9b7420d9fb1063908b3b021",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9124\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2528b250166644e08210e4678374d2ac",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3584, Acc@1 0.8411\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0d5c0a1295d5417585c5eae9726cce72",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9079\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "74c764806cf74fe5b86702dd46badbe6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3098, Acc@1 0.8350\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8500b72e2ff74acfbffa1ecb0f0d6210",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9113\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d8ed48a37a6f41c0856aa51e1bbfaa23",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3097, Acc@1 0.8460\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7a4b895d2a044d40ab18b2446dda210d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9160\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fd4556508b2e415194dfeab0875a3062",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3899, Acc@1 0.8424\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1485c7cbc36044d58562251d3002bdf4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9115\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6a9d9b71c185444696e2ed4344fddc1b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.3227, Acc@1 0.8454\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7910f1930f5949aebe929fd6166f7f94",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9101\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "aa352be6b7284271a4c9fdadda6d53a4",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.4783, Acc@1 0.8314\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "a47d893ebbbe45e1bad7119c8ea240e2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.9123\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for idx, modeli in model_maps.items():\n",
    "    for task in range(1):\n",
    "        print(f\"Training for task {task}\")\n",
    "        set_model_task(modeli, task)\n",
    "        mnisti = loaders[idx]\n",
    "        mnisti.update_task(task)\n",
    "\n",
    "        optimizer1 = optim.RMSprop([p for p in modeli.parameters() if p.requires_grad], lr=1e-4)\n",
    "        # Train for 1 epoch\n",
    "        for e in range(1):\n",
    "            train(modeli, mnisti.train_loader, optimizer1, e)\n",
    "            \n",
    "            print(\"Validation\")\n",
    "            print(\"============\")\n",
    "            acc1 = evaluate(modeli, mnisti.val_loader, e)\n",
    "\n",
    "\n",
    "        cache_masks(modeli)\n",
    "        print()\n",
    "        set_num_tasks_learned(modeli, task + 1)\n",
    "        print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 220,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Layer: 0, Models: (0, 1), Overlap: 0.6411479711532593\n",
      "Layer: 0, Models: (0, 2), Overlap: 0.6433843374252319\n",
      "Layer: 0, Models: (0, 3), Overlap: 0.6438350081443787\n",
      "Layer: 0, Models: (0, 4), Overlap: 0.6411394476890564\n",
      "Layer: 0, Models: (0, 5), Overlap: 0.6416326761245728\n",
      "Layer: 0, Models: (0, 6), Overlap: 0.6418962478637695\n",
      "Layer: 0, Models: (0, 7), Overlap: 0.6417006850242615\n",
      "Layer: 0, Models: (0, 8), Overlap: 0.6428826451301575\n",
      "Layer: 0, Models: (0, 9), Overlap: 0.6420578360557556\n",
      "Layer: 0, Models: (0, 10), Overlap: 0.6414455771446228\n",
      "Layer: 0, Models: (0, 11), Overlap: 0.641232967376709\n",
      "Layer: 0, Models: (0, 12), Overlap: 0.6421173214912415\n",
      "Layer: 0, Models: (0, 13), Overlap: 0.643562912940979\n",
      "Layer: 0, Models: (0, 14), Overlap: 0.6414710879325867\n",
      "Layer: 0, Models: (1, 0), Overlap: 0.6411479711532593\n",
      "Layer: 0, Models: (1, 2), Overlap: 0.6411224603652954\n",
      "Layer: 0, Models: (1, 3), Overlap: 0.6421768665313721\n",
      "Layer: 0, Models: (1, 4), Overlap: 0.6426190733909607\n",
      "Layer: 0, Models: (1, 5), Overlap: 0.6410118937492371\n",
      "Layer: 0, Models: (1, 6), Overlap: 0.6428316235542297\n",
      "Layer: 0, Models: (1, 7), Overlap: 0.6403146386146545\n",
      "Layer: 0, Models: (1, 8), Overlap: 0.6446343660354614\n",
      "Layer: 0, Models: (1, 9), Overlap: 0.6426615715026855\n",
      "Layer: 0, Models: (1, 10), Overlap: 0.6415306329727173\n",
      "Layer: 0, Models: (1, 11), Overlap: 0.6421428322792053\n",
      "Layer: 0, Models: (1, 12), Overlap: 0.6403146386146545\n",
      "Layer: 0, Models: (1, 13), Overlap: 0.6426785588264465\n",
      "Layer: 0, Models: (1, 14), Overlap: 0.6424319744110107\n",
      "Layer: 0, Models: (2, 0), Overlap: 0.6433843374252319\n",
      "Layer: 0, Models: (2, 1), Overlap: 0.6411224603652954\n",
      "Layer: 0, Models: (2, 3), Overlap: 0.641615629196167\n",
      "Layer: 0, Models: (2, 4), Overlap: 0.6410714387893677\n",
      "Layer: 0, Models: (2, 5), Overlap: 0.6425340175628662\n",
      "Layer: 0, Models: (2, 6), Overlap: 0.642363965511322\n",
      "Layer: 0, Models: (2, 7), Overlap: 0.6421003341674805\n",
      "Layer: 0, Models: (2, 8), Overlap: 0.64335036277771\n",
      "Layer: 0, Models: (2, 9), Overlap: 0.6415051221847534\n",
      "Layer: 0, Models: (2, 10), Overlap: 0.6427721381187439\n",
      "Layer: 0, Models: (2, 11), Overlap: 0.6422278881072998\n",
      "Layer: 0, Models: (2, 12), Overlap: 0.6438605189323425\n",
      "Layer: 0, Models: (2, 13), Overlap: 0.6423129439353943\n",
      "Layer: 0, Models: (2, 14), Overlap: 0.6419217586517334\n",
      "Layer: 0, Models: (3, 0), Overlap: 0.6438350081443787\n",
      "Layer: 0, Models: (3, 1), Overlap: 0.6421768665313721\n",
      "Layer: 0, Models: (3, 2), Overlap: 0.641615629196167\n",
      "Layer: 0, Models: (3, 4), Overlap: 0.6411564350128174\n",
      "Layer: 0, Models: (3, 5), Overlap: 0.6413010358810425\n",
      "Layer: 0, Models: (3, 6), Overlap: 0.6410629153251648\n",
      "Layer: 0, Models: (3, 7), Overlap: 0.6421173214912415\n",
      "Layer: 0, Models: (3, 8), Overlap: 0.6440306305885315\n",
      "Layer: 0, Models: (3, 9), Overlap: 0.6411479711532593\n",
      "Layer: 0, Models: (3, 10), Overlap: 0.6385799050331116\n",
      "Layer: 0, Models: (3, 11), Overlap: 0.6430357098579407\n",
      "Layer: 0, Models: (3, 12), Overlap: 0.6404251456260681\n",
      "Layer: 0, Models: (3, 13), Overlap: 0.6422619223594666\n",
      "Layer: 0, Models: (3, 14), Overlap: 0.6402720808982849\n",
      "Layer: 0, Models: (4, 0), Overlap: 0.6411394476890564\n",
      "Layer: 0, Models: (4, 1), Overlap: 0.6426190733909607\n",
      "Layer: 0, Models: (4, 2), Overlap: 0.6410714387893677\n",
      "Layer: 0, Models: (4, 3), Overlap: 0.6411564350128174\n",
      "Layer: 0, Models: (4, 5), Overlap: 0.6398809552192688\n",
      "Layer: 0, Models: (4, 6), Overlap: 0.6405782103538513\n",
      "Layer: 0, Models: (4, 7), Overlap: 0.6410884261131287\n",
      "Layer: 0, Models: (4, 8), Overlap: 0.6410459280014038\n",
      "Layer: 0, Models: (4, 9), Overlap: 0.6409863829612732\n",
      "Layer: 0, Models: (4, 10), Overlap: 0.6415646076202393\n",
      "Layer: 0, Models: (4, 11), Overlap: 0.6420238018035889\n",
      "Layer: 0, Models: (4, 12), Overlap: 0.6406547427177429\n",
      "Layer: 0, Models: (4, 13), Overlap: 0.641981303691864\n",
      "Layer: 0, Models: (4, 14), Overlap: 0.6414541006088257\n",
      "Layer: 0, Models: (5, 0), Overlap: 0.6416326761245728\n",
      "Layer: 0, Models: (5, 1), Overlap: 0.6410118937492371\n",
      "Layer: 0, Models: (5, 2), Overlap: 0.6425340175628662\n",
      "Layer: 0, Models: (5, 3), Overlap: 0.6413010358810425\n",
      "Layer: 0, Models: (5, 4), Overlap: 0.6398809552192688\n",
      "Layer: 0, Models: (5, 6), Overlap: 0.6410204172134399\n",
      "Layer: 0, Models: (5, 7), Overlap: 0.6416581869125366\n",
      "Layer: 0, Models: (5, 8), Overlap: 0.6433928608894348\n",
      "Layer: 0, Models: (5, 9), Overlap: 0.6416666507720947\n",
      "Layer: 0, Models: (5, 10), Overlap: 0.640629231929779\n",
      "Layer: 0, Models: (5, 11), Overlap: 0.6414285898208618\n",
      "Layer: 0, Models: (5, 12), Overlap: 0.6429761648178101\n",
      "Layer: 0, Models: (5, 13), Overlap: 0.6416751742362976\n",
      "Layer: 0, Models: (5, 14), Overlap: 0.6436139345169067\n",
      "Layer: 0, Models: (6, 0), Overlap: 0.6418962478637695\n",
      "Layer: 0, Models: (6, 1), Overlap: 0.6428316235542297\n",
      "Layer: 0, Models: (6, 2), Overlap: 0.642363965511322\n",
      "Layer: 0, Models: (6, 3), Overlap: 0.6410629153251648\n",
      "Layer: 0, Models: (6, 4), Overlap: 0.6405782103538513\n",
      "Layer: 0, Models: (6, 5), Overlap: 0.6410204172134399\n",
      "Layer: 0, Models: (6, 7), Overlap: 0.6426785588264465\n",
      "Layer: 0, Models: (6, 8), Overlap: 0.6433588266372681\n",
      "Layer: 0, Models: (6, 9), Overlap: 0.6437329649925232\n",
      "Layer: 0, Models: (6, 10), Overlap: 0.6413010358810425\n",
      "Layer: 0, Models: (6, 11), Overlap: 0.6410544514656067\n",
      "Layer: 0, Models: (6, 12), Overlap: 0.6417347192764282\n",
      "Layer: 0, Models: (6, 13), Overlap: 0.6435034275054932\n",
      "Layer: 0, Models: (6, 14), Overlap: 0.6401955485343933\n",
      "Layer: 0, Models: (7, 0), Overlap: 0.6417006850242615\n",
      "Layer: 0, Models: (7, 1), Overlap: 0.6403146386146545\n",
      "Layer: 0, Models: (7, 2), Overlap: 0.6421003341674805\n",
      "Layer: 0, Models: (7, 3), Overlap: 0.6421173214912415\n",
      "Layer: 0, Models: (7, 4), Overlap: 0.6410884261131287\n",
      "Layer: 0, Models: (7, 5), Overlap: 0.6416581869125366\n",
      "Layer: 0, Models: (7, 6), Overlap: 0.6426785588264465\n",
      "Layer: 0, Models: (7, 8), Overlap: 0.6410884261131287\n",
      "Layer: 0, Models: (7, 9), Overlap: 0.6442092061042786\n",
      "Layer: 0, Models: (7, 10), Overlap: 0.640799343585968\n",
      "Layer: 0, Models: (7, 11), Overlap: 0.6402975916862488\n",
      "Layer: 0, Models: (7, 12), Overlap: 0.6417602300643921\n",
      "Layer: 0, Models: (7, 13), Overlap: 0.6398809552192688\n",
      "Layer: 0, Models: (7, 14), Overlap: 0.6410374045372009\n",
      "Layer: 0, Models: (8, 0), Overlap: 0.6428826451301575\n",
      "Layer: 0, Models: (8, 1), Overlap: 0.6446343660354614\n",
      "Layer: 0, Models: (8, 2), Overlap: 0.64335036277771\n",
      "Layer: 0, Models: (8, 3), Overlap: 0.6440306305885315\n",
      "Layer: 0, Models: (8, 4), Overlap: 0.6410459280014038\n",
      "Layer: 0, Models: (8, 5), Overlap: 0.6433928608894348\n",
      "Layer: 0, Models: (8, 6), Overlap: 0.6433588266372681\n",
      "Layer: 0, Models: (8, 7), Overlap: 0.6410884261131287\n",
      "Layer: 0, Models: (8, 9), Overlap: 0.6413605213165283\n",
      "Layer: 0, Models: (8, 10), Overlap: 0.6416666507720947\n",
      "Layer: 0, Models: (8, 11), Overlap: 0.6435884237289429\n",
      "Layer: 0, Models: (8, 12), Overlap: 0.6428061127662659\n",
      "Layer: 0, Models: (8, 13), Overlap: 0.6437074542045593\n",
      "Layer: 0, Models: (8, 14), Overlap: 0.6434524059295654\n",
      "Layer: 0, Models: (9, 0), Overlap: 0.6420578360557556\n",
      "Layer: 0, Models: (9, 1), Overlap: 0.6426615715026855\n",
      "Layer: 0, Models: (9, 2), Overlap: 0.6415051221847534\n",
      "Layer: 0, Models: (9, 3), Overlap: 0.6411479711532593\n",
      "Layer: 0, Models: (9, 4), Overlap: 0.6409863829612732\n",
      "Layer: 0, Models: (9, 5), Overlap: 0.6416666507720947\n",
      "Layer: 0, Models: (9, 6), Overlap: 0.6437329649925232\n",
      "Layer: 0, Models: (9, 7), Overlap: 0.6442092061042786\n",
      "Layer: 0, Models: (9, 8), Overlap: 0.6413605213165283\n",
      "Layer: 0, Models: (9, 10), Overlap: 0.6411564350128174\n",
      "Layer: 0, Models: (9, 11), Overlap: 0.6426870822906494\n",
      "Layer: 0, Models: (9, 12), Overlap: 0.6399660110473633\n",
      "Layer: 0, Models: (9, 13), Overlap: 0.6413945555686951\n",
      "Layer: 0, Models: (9, 14), Overlap: 0.6437414884567261\n",
      "Layer: 0, Models: (10, 0), Overlap: 0.6414455771446228\n",
      "Layer: 0, Models: (10, 1), Overlap: 0.6415306329727173\n",
      "Layer: 0, Models: (10, 2), Overlap: 0.6427721381187439\n",
      "Layer: 0, Models: (10, 3), Overlap: 0.6385799050331116\n",
      "Layer: 0, Models: (10, 4), Overlap: 0.6415646076202393\n",
      "Layer: 0, Models: (10, 5), Overlap: 0.640629231929779\n",
      "Layer: 0, Models: (10, 6), Overlap: 0.6413010358810425\n",
      "Layer: 0, Models: (10, 7), Overlap: 0.640799343585968\n",
      "Layer: 0, Models: (10, 8), Overlap: 0.6416666507720947\n",
      "Layer: 0, Models: (10, 9), Overlap: 0.6411564350128174\n",
      "Layer: 0, Models: (10, 11), Overlap: 0.6432822942733765\n",
      "Layer: 0, Models: (10, 12), Overlap: 0.6424574851989746\n",
      "Layer: 0, Models: (10, 13), Overlap: 0.6411819458007812\n",
      "Layer: 0, Models: (10, 14), Overlap: 0.6424319744110107\n",
      "Layer: 0, Models: (11, 0), Overlap: 0.641232967376709\n",
      "Layer: 0, Models: (11, 1), Overlap: 0.6421428322792053\n",
      "Layer: 0, Models: (11, 2), Overlap: 0.6422278881072998\n",
      "Layer: 0, Models: (11, 3), Overlap: 0.6430357098579407\n",
      "Layer: 0, Models: (11, 4), Overlap: 0.6420238018035889\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Layer: 0, Models: (11, 5), Overlap: 0.6414285898208618\n",
      "Layer: 0, Models: (11, 6), Overlap: 0.6410544514656067\n",
      "Layer: 0, Models: (11, 7), Overlap: 0.6402975916862488\n",
      "Layer: 0, Models: (11, 8), Overlap: 0.6435884237289429\n",
      "Layer: 0, Models: (11, 9), Overlap: 0.6426870822906494\n",
      "Layer: 0, Models: (11, 10), Overlap: 0.6432822942733765\n",
      "Layer: 0, Models: (11, 12), Overlap: 0.6423469185829163\n",
      "Layer: 0, Models: (11, 13), Overlap: 0.6433843374252319\n",
      "Layer: 0, Models: (11, 14), Overlap: 0.643146276473999\n",
      "Layer: 0, Models: (12, 0), Overlap: 0.6421173214912415\n",
      "Layer: 0, Models: (12, 1), Overlap: 0.6403146386146545\n",
      "Layer: 0, Models: (12, 2), Overlap: 0.6438605189323425\n",
      "Layer: 0, Models: (12, 3), Overlap: 0.6404251456260681\n",
      "Layer: 0, Models: (12, 4), Overlap: 0.6406547427177429\n",
      "Layer: 0, Models: (12, 5), Overlap: 0.6429761648178101\n",
      "Layer: 0, Models: (12, 6), Overlap: 0.6417347192764282\n",
      "Layer: 0, Models: (12, 7), Overlap: 0.6417602300643921\n",
      "Layer: 0, Models: (12, 8), Overlap: 0.6428061127662659\n",
      "Layer: 0, Models: (12, 9), Overlap: 0.6399660110473633\n",
      "Layer: 0, Models: (12, 10), Overlap: 0.6424574851989746\n",
      "Layer: 0, Models: (12, 11), Overlap: 0.6423469185829163\n",
      "Layer: 0, Models: (12, 13), Overlap: 0.6418792605400085\n",
      "Layer: 0, Models: (12, 14), Overlap: 0.6414625644683838\n",
      "Layer: 0, Models: (13, 0), Overlap: 0.643562912940979\n",
      "Layer: 0, Models: (13, 1), Overlap: 0.6426785588264465\n",
      "Layer: 0, Models: (13, 2), Overlap: 0.6423129439353943\n",
      "Layer: 0, Models: (13, 3), Overlap: 0.6422619223594666\n",
      "Layer: 0, Models: (13, 4), Overlap: 0.641981303691864\n",
      "Layer: 0, Models: (13, 5), Overlap: 0.6416751742362976\n",
      "Layer: 0, Models: (13, 6), Overlap: 0.6435034275054932\n",
      "Layer: 0, Models: (13, 7), Overlap: 0.6398809552192688\n",
      "Layer: 0, Models: (13, 8), Overlap: 0.6437074542045593\n",
      "Layer: 0, Models: (13, 9), Overlap: 0.6413945555686951\n",
      "Layer: 0, Models: (13, 10), Overlap: 0.6411819458007812\n",
      "Layer: 0, Models: (13, 11), Overlap: 0.6433843374252319\n",
      "Layer: 0, Models: (13, 12), Overlap: 0.6418792605400085\n",
      "Layer: 0, Models: (13, 14), Overlap: 0.6431717872619629\n",
      "Layer: 0, Models: (14, 0), Overlap: 0.6414710879325867\n",
      "Layer: 0, Models: (14, 1), Overlap: 0.6424319744110107\n",
      "Layer: 0, Models: (14, 2), Overlap: 0.6419217586517334\n",
      "Layer: 0, Models: (14, 3), Overlap: 0.6402720808982849\n",
      "Layer: 0, Models: (14, 4), Overlap: 0.6414541006088257\n",
      "Layer: 0, Models: (14, 5), Overlap: 0.6436139345169067\n",
      "Layer: 0, Models: (14, 6), Overlap: 0.6401955485343933\n",
      "Layer: 0, Models: (14, 7), Overlap: 0.6410374045372009\n",
      "Layer: 0, Models: (14, 8), Overlap: 0.6434524059295654\n",
      "Layer: 0, Models: (14, 9), Overlap: 0.6437414884567261\n",
      "Layer: 0, Models: (14, 10), Overlap: 0.6424319744110107\n",
      "Layer: 0, Models: (14, 11), Overlap: 0.643146276473999\n",
      "Layer: 0, Models: (14, 12), Overlap: 0.6414625644683838\n",
      "Layer: 0, Models: (14, 13), Overlap: 0.6431717872619629\n",
      "Layer: 2, Models: (0, 1), Overlap: 0.6269999742507935\n",
      "Layer: 2, Models: (0, 2), Overlap: 0.6281111240386963\n",
      "Layer: 2, Models: (0, 3), Overlap: 0.6264222264289856\n",
      "Layer: 2, Models: (0, 4), Overlap: 0.6245555281639099\n",
      "Layer: 2, Models: (0, 5), Overlap: 0.6274222135543823\n",
      "Layer: 2, Models: (0, 6), Overlap: 0.6270444393157959\n",
      "Layer: 2, Models: (0, 7), Overlap: 0.6279333233833313\n",
      "Layer: 2, Models: (0, 8), Overlap: 0.6266444325447083\n",
      "Layer: 2, Models: (0, 9), Overlap: 0.6282222270965576\n",
      "Layer: 2, Models: (0, 10), Overlap: 0.6276222467422485\n",
      "Layer: 2, Models: (0, 11), Overlap: 0.6280444264411926\n",
      "Layer: 2, Models: (0, 12), Overlap: 0.6255999803543091\n",
      "Layer: 2, Models: (0, 13), Overlap: 0.6299999952316284\n",
      "Layer: 2, Models: (0, 14), Overlap: 0.6259777545928955\n",
      "Layer: 2, Models: (1, 0), Overlap: 0.6269999742507935\n",
      "Layer: 2, Models: (1, 2), Overlap: 0.6283777952194214\n",
      "Layer: 2, Models: (1, 3), Overlap: 0.626800000667572\n",
      "Layer: 2, Models: (1, 4), Overlap: 0.6275333166122437\n",
      "Layer: 2, Models: (1, 5), Overlap: 0.6323111057281494\n",
      "Layer: 2, Models: (1, 6), Overlap: 0.6284444332122803\n",
      "Layer: 2, Models: (1, 7), Overlap: 0.6271777749061584\n",
      "Layer: 2, Models: (1, 8), Overlap: 0.6274222135543823\n",
      "Layer: 2, Models: (1, 9), Overlap: 0.6290444731712341\n",
      "Layer: 2, Models: (1, 10), Overlap: 0.6286666393280029\n",
      "Layer: 2, Models: (1, 11), Overlap: 0.62782222032547\n",
      "Layer: 2, Models: (1, 12), Overlap: 0.6273777484893799\n",
      "Layer: 2, Models: (1, 13), Overlap: 0.6268666386604309\n",
      "Layer: 2, Models: (1, 14), Overlap: 0.6305999755859375\n",
      "Layer: 2, Models: (2, 0), Overlap: 0.6281111240386963\n",
      "Layer: 2, Models: (2, 1), Overlap: 0.6283777952194214\n",
      "Layer: 2, Models: (2, 3), Overlap: 0.6280666589736938\n",
      "Layer: 2, Models: (2, 4), Overlap: 0.6293777823448181\n",
      "Layer: 2, Models: (2, 5), Overlap: 0.6292444467544556\n",
      "Layer: 2, Models: (2, 6), Overlap: 0.6263555288314819\n",
      "Layer: 2, Models: (2, 7), Overlap: 0.6272222399711609\n",
      "Layer: 2, Models: (2, 8), Overlap: 0.626800000667572\n",
      "Layer: 2, Models: (2, 9), Overlap: 0.6272444725036621\n",
      "Layer: 2, Models: (2, 10), Overlap: 0.6245111227035522\n",
      "Layer: 2, Models: (2, 11), Overlap: 0.628422200679779\n",
      "Layer: 2, Models: (2, 12), Overlap: 0.6285111308097839\n",
      "Layer: 2, Models: (2, 13), Overlap: 0.6290444731712341\n",
      "Layer: 2, Models: (2, 14), Overlap: 0.6278444528579712\n",
      "Layer: 2, Models: (3, 0), Overlap: 0.6264222264289856\n",
      "Layer: 2, Models: (3, 1), Overlap: 0.626800000667572\n",
      "Layer: 2, Models: (3, 2), Overlap: 0.6280666589736938\n",
      "Layer: 2, Models: (3, 4), Overlap: 0.6296444535255432\n",
      "Layer: 2, Models: (3, 5), Overlap: 0.6295777559280396\n",
      "Layer: 2, Models: (3, 6), Overlap: 0.6273333430290222\n",
      "Layer: 2, Models: (3, 7), Overlap: 0.6280221939086914\n",
      "Layer: 2, Models: (3, 8), Overlap: 0.6280888915061951\n",
      "Layer: 2, Models: (3, 9), Overlap: 0.629800021648407\n",
      "Layer: 2, Models: (3, 10), Overlap: 0.6276666522026062\n",
      "Layer: 2, Models: (3, 11), Overlap: 0.6277777552604675\n",
      "Layer: 2, Models: (3, 12), Overlap: 0.6277333498001099\n",
      "Layer: 2, Models: (3, 13), Overlap: 0.6264888644218445\n",
      "Layer: 2, Models: (3, 14), Overlap: 0.628333330154419\n",
      "Layer: 2, Models: (4, 0), Overlap: 0.6245555281639099\n",
      "Layer: 2, Models: (4, 1), Overlap: 0.6275333166122437\n",
      "Layer: 2, Models: (4, 2), Overlap: 0.6293777823448181\n",
      "Layer: 2, Models: (4, 3), Overlap: 0.6296444535255432\n",
      "Layer: 2, Models: (4, 5), Overlap: 0.6286222338676453\n",
      "Layer: 2, Models: (4, 6), Overlap: 0.6271777749061584\n",
      "Layer: 2, Models: (4, 7), Overlap: 0.6277999877929688\n",
      "Layer: 2, Models: (4, 8), Overlap: 0.6260666847229004\n",
      "Layer: 2, Models: (4, 9), Overlap: 0.6272666454315186\n",
      "Layer: 2, Models: (4, 10), Overlap: 0.6291555762290955\n",
      "Layer: 2, Models: (4, 11), Overlap: 0.6265333294868469\n",
      "Layer: 2, Models: (4, 12), Overlap: 0.6290222406387329\n",
      "Layer: 2, Models: (4, 13), Overlap: 0.6290666460990906\n",
      "Layer: 2, Models: (4, 14), Overlap: 0.6269999742507935\n",
      "Layer: 2, Models: (5, 0), Overlap: 0.6274222135543823\n",
      "Layer: 2, Models: (5, 1), Overlap: 0.6323111057281494\n",
      "Layer: 2, Models: (5, 2), Overlap: 0.6292444467544556\n",
      "Layer: 2, Models: (5, 3), Overlap: 0.6295777559280396\n",
      "Layer: 2, Models: (5, 4), Overlap: 0.6286222338676453\n",
      "Layer: 2, Models: (5, 6), Overlap: 0.6263999938964844\n",
      "Layer: 2, Models: (5, 7), Overlap: 0.6269111037254333\n",
      "Layer: 2, Models: (5, 8), Overlap: 0.6290666460990906\n",
      "Layer: 2, Models: (5, 9), Overlap: 0.6278666853904724\n",
      "Layer: 2, Models: (5, 10), Overlap: 0.6255555748939514\n",
      "Layer: 2, Models: (5, 11), Overlap: 0.6259999871253967\n",
      "Layer: 2, Models: (5, 12), Overlap: 0.6278889179229736\n",
      "Layer: 2, Models: (5, 13), Overlap: 0.6291999816894531\n",
      "Layer: 2, Models: (5, 14), Overlap: 0.628333330154419\n",
      "Layer: 2, Models: (6, 0), Overlap: 0.6270444393157959\n",
      "Layer: 2, Models: (6, 1), Overlap: 0.6284444332122803\n",
      "Layer: 2, Models: (6, 2), Overlap: 0.6263555288314819\n",
      "Layer: 2, Models: (6, 3), Overlap: 0.6273333430290222\n",
      "Layer: 2, Models: (6, 4), Overlap: 0.6271777749061584\n",
      "Layer: 2, Models: (6, 5), Overlap: 0.6263999938964844\n",
      "Layer: 2, Models: (6, 7), Overlap: 0.6263111233711243\n",
      "Layer: 2, Models: (6, 8), Overlap: 0.6270889043807983\n",
      "Layer: 2, Models: (6, 9), Overlap: 0.6266666650772095\n",
      "Layer: 2, Models: (6, 10), Overlap: 0.6276666522026062\n",
      "Layer: 2, Models: (6, 11), Overlap: 0.6296666860580444\n",
      "Layer: 2, Models: (6, 12), Overlap: 0.6289555430412292\n",
      "Layer: 2, Models: (6, 13), Overlap: 0.6287999749183655\n",
      "Layer: 2, Models: (6, 14), Overlap: 0.6290222406387329\n",
      "Layer: 2, Models: (7, 0), Overlap: 0.6279333233833313\n",
      "Layer: 2, Models: (7, 1), Overlap: 0.6271777749061584\n",
      "Layer: 2, Models: (7, 2), Overlap: 0.6272222399711609\n",
      "Layer: 2, Models: (7, 3), Overlap: 0.6280221939086914\n",
      "Layer: 2, Models: (7, 4), Overlap: 0.6277999877929688\n",
      "Layer: 2, Models: (7, 5), Overlap: 0.6269111037254333\n",
      "Layer: 2, Models: (7, 6), Overlap: 0.6263111233711243\n",
      "Layer: 2, Models: (7, 8), Overlap: 0.6254444718360901\n",
      "Layer: 2, Models: (7, 9), Overlap: 0.6269999742507935\n",
      "Layer: 2, Models: (7, 10), Overlap: 0.6286888718605042\n",
      "Layer: 2, Models: (7, 11), Overlap: 0.6299777626991272\n",
      "Layer: 2, Models: (7, 12), Overlap: 0.6240000128746033\n",
      "Layer: 2, Models: (7, 13), Overlap: 0.6298444271087646\n",
      "Layer: 2, Models: (7, 14), Overlap: 0.626977801322937\n",
      "Layer: 2, Models: (8, 0), Overlap: 0.6266444325447083\n",
      "Layer: 2, Models: (8, 1), Overlap: 0.6274222135543823\n",
      "Layer: 2, Models: (8, 2), Overlap: 0.626800000667572\n",
      "Layer: 2, Models: (8, 3), Overlap: 0.6280888915061951\n",
      "Layer: 2, Models: (8, 4), Overlap: 0.6260666847229004\n",
      "Layer: 2, Models: (8, 5), Overlap: 0.6290666460990906\n",
      "Layer: 2, Models: (8, 6), Overlap: 0.6270889043807983\n",
      "Layer: 2, Models: (8, 7), Overlap: 0.6254444718360901\n",
      "Layer: 2, Models: (8, 9), Overlap: 0.6281999945640564\n",
      "Layer: 2, Models: (8, 10), Overlap: 0.6285333037376404\n",
      "Layer: 2, Models: (8, 11), Overlap: 0.6259777545928955\n",
      "Layer: 2, Models: (8, 12), Overlap: 0.6277111172676086\n",
      "Layer: 2, Models: (8, 13), Overlap: 0.6267111301422119\n",
      "Layer: 2, Models: (8, 14), Overlap: 0.6270666718482971\n",
      "Layer: 2, Models: (9, 0), Overlap: 0.6282222270965576\n",
      "Layer: 2, Models: (9, 1), Overlap: 0.6290444731712341\n",
      "Layer: 2, Models: (9, 2), Overlap: 0.6272444725036621\n",
      "Layer: 2, Models: (9, 3), Overlap: 0.629800021648407\n",
      "Layer: 2, Models: (9, 4), Overlap: 0.6272666454315186\n",
      "Layer: 2, Models: (9, 5), Overlap: 0.6278666853904724\n",
      "Layer: 2, Models: (9, 6), Overlap: 0.6266666650772095\n",
      "Layer: 2, Models: (9, 7), Overlap: 0.6269999742507935\n",
      "Layer: 2, Models: (9, 8), Overlap: 0.6281999945640564\n",
      "Layer: 2, Models: (9, 10), Overlap: 0.6274444460868835\n",
      "Layer: 2, Models: (9, 11), Overlap: 0.6285777688026428\n",
      "Layer: 2, Models: (9, 12), Overlap: 0.6259777545928955\n",
      "Layer: 2, Models: (9, 13), Overlap: 0.6295333504676819\n",
      "Layer: 2, Models: (9, 14), Overlap: 0.6269999742507935\n",
      "Layer: 2, Models: (10, 0), Overlap: 0.6276222467422485\n",
      "Layer: 2, Models: (10, 1), Overlap: 0.6286666393280029\n",
      "Layer: 2, Models: (10, 2), Overlap: 0.6245111227035522\n",
      "Layer: 2, Models: (10, 3), Overlap: 0.6276666522026062\n",
      "Layer: 2, Models: (10, 4), Overlap: 0.6291555762290955\n",
      "Layer: 2, Models: (10, 5), Overlap: 0.6255555748939514\n",
      "Layer: 2, Models: (10, 6), Overlap: 0.6276666522026062\n",
      "Layer: 2, Models: (10, 7), Overlap: 0.6286888718605042\n",
      "Layer: 2, Models: (10, 8), Overlap: 0.6285333037376404\n",
      "Layer: 2, Models: (10, 9), Overlap: 0.6274444460868835\n",
      "Layer: 2, Models: (10, 11), Overlap: 0.6290000081062317\n",
      "Layer: 2, Models: (10, 12), Overlap: 0.6280666589736938\n",
      "Layer: 2, Models: (10, 13), Overlap: 0.6299111247062683\n",
      "Layer: 2, Models: (10, 14), Overlap: 0.6284888982772827\n",
      "Layer: 2, Models: (11, 0), Overlap: 0.6280444264411926\n",
      "Layer: 2, Models: (11, 1), Overlap: 0.62782222032547\n",
      "Layer: 2, Models: (11, 2), Overlap: 0.628422200679779\n",
      "Layer: 2, Models: (11, 3), Overlap: 0.6277777552604675\n",
      "Layer: 2, Models: (11, 4), Overlap: 0.6265333294868469\n",
      "Layer: 2, Models: (11, 5), Overlap: 0.6259999871253967\n",
      "Layer: 2, Models: (11, 6), Overlap: 0.6296666860580444\n",
      "Layer: 2, Models: (11, 7), Overlap: 0.6299777626991272\n",
      "Layer: 2, Models: (11, 8), Overlap: 0.6259777545928955\n",
      "Layer: 2, Models: (11, 9), Overlap: 0.6285777688026428\n",
      "Layer: 2, Models: (11, 10), Overlap: 0.6290000081062317\n",
      "Layer: 2, Models: (11, 12), Overlap: 0.6268888711929321\n",
      "Layer: 2, Models: (11, 13), Overlap: 0.6262666583061218\n",
      "Layer: 2, Models: (11, 14), Overlap: 0.6255333423614502\n",
      "Layer: 2, Models: (12, 0), Overlap: 0.6255999803543091\n",
      "Layer: 2, Models: (12, 1), Overlap: 0.6273777484893799\n",
      "Layer: 2, Models: (12, 2), Overlap: 0.6285111308097839\n",
      "Layer: 2, Models: (12, 3), Overlap: 0.6277333498001099\n",
      "Layer: 2, Models: (12, 4), Overlap: 0.6290222406387329\n",
      "Layer: 2, Models: (12, 5), Overlap: 0.6278889179229736\n",
      "Layer: 2, Models: (12, 6), Overlap: 0.6289555430412292\n",
      "Layer: 2, Models: (12, 7), Overlap: 0.6240000128746033\n",
      "Layer: 2, Models: (12, 8), Overlap: 0.6277111172676086\n",
      "Layer: 2, Models: (12, 9), Overlap: 0.6259777545928955\n",
      "Layer: 2, Models: (12, 10), Overlap: 0.6280666589736938\n",
      "Layer: 2, Models: (12, 11), Overlap: 0.6268888711929321\n",
      "Layer: 2, Models: (12, 13), Overlap: 0.628777801990509\n",
      "Layer: 2, Models: (12, 14), Overlap: 0.6263999938964844\n",
      "Layer: 2, Models: (13, 0), Overlap: 0.6299999952316284\n",
      "Layer: 2, Models: (13, 1), Overlap: 0.6268666386604309\n",
      "Layer: 2, Models: (13, 2), Overlap: 0.6290444731712341\n",
      "Layer: 2, Models: (13, 3), Overlap: 0.6264888644218445\n",
      "Layer: 2, Models: (13, 4), Overlap: 0.6290666460990906\n",
      "Layer: 2, Models: (13, 5), Overlap: 0.6291999816894531\n",
      "Layer: 2, Models: (13, 6), Overlap: 0.6287999749183655\n",
      "Layer: 2, Models: (13, 7), Overlap: 0.6298444271087646\n",
      "Layer: 2, Models: (13, 8), Overlap: 0.6267111301422119\n",
      "Layer: 2, Models: (13, 9), Overlap: 0.6295333504676819\n",
      "Layer: 2, Models: (13, 10), Overlap: 0.6299111247062683\n",
      "Layer: 2, Models: (13, 11), Overlap: 0.6262666583061218\n",
      "Layer: 2, Models: (13, 12), Overlap: 0.628777801990509\n",
      "Layer: 2, Models: (13, 14), Overlap: 0.6273777484893799\n",
      "Layer: 2, Models: (14, 0), Overlap: 0.6259777545928955\n",
      "Layer: 2, Models: (14, 1), Overlap: 0.6305999755859375\n",
      "Layer: 2, Models: (14, 2), Overlap: 0.6278444528579712\n",
      "Layer: 2, Models: (14, 3), Overlap: 0.628333330154419\n",
      "Layer: 2, Models: (14, 4), Overlap: 0.6269999742507935\n",
      "Layer: 2, Models: (14, 5), Overlap: 0.628333330154419\n",
      "Layer: 2, Models: (14, 6), Overlap: 0.6290222406387329\n",
      "Layer: 2, Models: (14, 7), Overlap: 0.626977801322937\n",
      "Layer: 2, Models: (14, 8), Overlap: 0.6270666718482971\n",
      "Layer: 2, Models: (14, 9), Overlap: 0.6269999742507935\n",
      "Layer: 2, Models: (14, 10), Overlap: 0.6284888982772827\n",
      "Layer: 2, Models: (14, 11), Overlap: 0.6255333423614502\n",
      "Layer: 2, Models: (14, 12), Overlap: 0.6263999938964844\n",
      "Layer: 2, Models: (14, 13), Overlap: 0.6273777484893799\n",
      "Layer: 4, Models: (0, 1), Overlap: 0.6401333212852478\n",
      "Layer: 4, Models: (0, 2), Overlap: 0.6405333280563354\n",
      "Layer: 4, Models: (0, 3), Overlap: 0.6480000019073486\n",
      "Layer: 4, Models: (0, 4), Overlap: 0.6396666765213013\n",
      "Layer: 4, Models: (0, 5), Overlap: 0.6438000202178955\n",
      "Layer: 4, Models: (0, 6), Overlap: 0.6391333341598511\n",
      "Layer: 4, Models: (0, 7), Overlap: 0.6425333619117737\n",
      "Layer: 4, Models: (0, 8), Overlap: 0.6447333097457886\n",
      "Layer: 4, Models: (0, 9), Overlap: 0.6410666704177856\n",
      "Layer: 4, Models: (0, 10), Overlap: 0.6406000256538391\n",
      "Layer: 4, Models: (0, 11), Overlap: 0.6398000121116638\n",
      "Layer: 4, Models: (0, 12), Overlap: 0.6401333212852478\n",
      "Layer: 4, Models: (0, 13), Overlap: 0.6403999924659729\n",
      "Layer: 4, Models: (0, 14), Overlap: 0.6439333558082581\n",
      "Layer: 4, Models: (1, 0), Overlap: 0.6401333212852478\n",
      "Layer: 4, Models: (1, 2), Overlap: 0.6420666575431824\n",
      "Layer: 4, Models: (1, 3), Overlap: 0.6435333490371704\n",
      "Layer: 4, Models: (1, 4), Overlap: 0.6430666446685791\n",
      "Layer: 4, Models: (1, 5), Overlap: 0.6439999938011169\n",
      "Layer: 4, Models: (1, 6), Overlap: 0.6443333625793457\n",
      "Layer: 4, Models: (1, 7), Overlap: 0.6417333483695984\n",
      "Layer: 4, Models: (1, 8), Overlap: 0.6408666372299194\n",
      "Layer: 4, Models: (1, 9), Overlap: 0.6453333497047424\n",
      "Layer: 4, Models: (1, 10), Overlap: 0.6446666717529297\n",
      "Layer: 4, Models: (1, 11), Overlap: 0.6445333361625671\n",
      "Layer: 4, Models: (1, 12), Overlap: 0.6413999795913696\n",
      "Layer: 4, Models: (1, 13), Overlap: 0.6403999924659729\n",
      "Layer: 4, Models: (1, 14), Overlap: 0.643666684627533\n",
      "Layer: 4, Models: (2, 0), Overlap: 0.6405333280563354\n",
      "Layer: 4, Models: (2, 1), Overlap: 0.6420666575431824\n",
      "Layer: 4, Models: (2, 3), Overlap: 0.6444000005722046\n",
      "Layer: 4, Models: (2, 4), Overlap: 0.642133355140686\n",
      "Layer: 4, Models: (2, 5), Overlap: 0.6399999856948853\n",
      "Layer: 4, Models: (2, 6), Overlap: 0.6428666710853577\n",
      "Layer: 4, Models: (2, 7), Overlap: 0.6387333273887634\n",
      "Layer: 4, Models: (2, 8), Overlap: 0.6366666555404663\n",
      "Layer: 4, Models: (2, 9), Overlap: 0.6492666602134705\n",
      "Layer: 4, Models: (2, 10), Overlap: 0.6445333361625671\n",
      "Layer: 4, Models: (2, 11), Overlap: 0.6439999938011169\n",
      "Layer: 4, Models: (2, 12), Overlap: 0.642133355140686\n",
      "Layer: 4, Models: (2, 13), Overlap: 0.6429333090782166\n",
      "Layer: 4, Models: (2, 14), Overlap: 0.6417999863624573\n",
      "Layer: 4, Models: (3, 0), Overlap: 0.6480000019073486\n",
      "Layer: 4, Models: (3, 1), Overlap: 0.6435333490371704\n",
      "Layer: 4, Models: (3, 2), Overlap: 0.6444000005722046\n",
      "Layer: 4, Models: (3, 4), Overlap: 0.6422666907310486\n",
      "Layer: 4, Models: (3, 5), Overlap: 0.6434000134468079\n",
      "Layer: 4, Models: (3, 6), Overlap: 0.6431999802589417\n",
      "Layer: 4, Models: (3, 7), Overlap: 0.6437333226203918\n",
      "Layer: 4, Models: (3, 8), Overlap: 0.6431999802589417\n",
      "Layer: 4, Models: (3, 9), Overlap: 0.6452666521072388\n",
      "Layer: 4, Models: (3, 10), Overlap: 0.6416666507720947\n",
      "Layer: 4, Models: (3, 11), Overlap: 0.6437333226203918\n",
      "Layer: 4, Models: (3, 12), Overlap: 0.6425333619117737\n",
      "Layer: 4, Models: (3, 13), Overlap: 0.6421999931335449\n",
      "Layer: 4, Models: (3, 14), Overlap: 0.6442000269889832\n",
      "Layer: 4, Models: (4, 0), Overlap: 0.6396666765213013\n",
      "Layer: 4, Models: (4, 1), Overlap: 0.6430666446685791\n",
      "Layer: 4, Models: (4, 2), Overlap: 0.642133355140686\n",
      "Layer: 4, Models: (4, 3), Overlap: 0.6422666907310486\n",
      "Layer: 4, Models: (4, 5), Overlap: 0.6435999870300293\n",
      "Layer: 4, Models: (4, 6), Overlap: 0.6463333368301392\n",
      "Layer: 4, Models: (4, 7), Overlap: 0.6424000263214111\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Layer: 4, Models: (4, 8), Overlap: 0.642799973487854\n",
      "Layer: 4, Models: (4, 9), Overlap: 0.646399974822998\n",
      "Layer: 4, Models: (4, 10), Overlap: 0.6416000127792358\n",
      "Layer: 4, Models: (4, 11), Overlap: 0.6425999999046326\n",
      "Layer: 4, Models: (4, 12), Overlap: 0.6443333625793457\n",
      "Layer: 4, Models: (4, 13), Overlap: 0.6402666568756104\n",
      "Layer: 4, Models: (4, 14), Overlap: 0.6410666704177856\n",
      "Layer: 4, Models: (5, 0), Overlap: 0.6438000202178955\n",
      "Layer: 4, Models: (5, 1), Overlap: 0.6439999938011169\n",
      "Layer: 4, Models: (5, 2), Overlap: 0.6399999856948853\n",
      "Layer: 4, Models: (5, 3), Overlap: 0.6434000134468079\n",
      "Layer: 4, Models: (5, 4), Overlap: 0.6435999870300293\n",
      "Layer: 4, Models: (5, 6), Overlap: 0.6475333571434021\n",
      "Layer: 4, Models: (5, 7), Overlap: 0.6420000195503235\n",
      "Layer: 4, Models: (5, 8), Overlap: 0.6407999992370605\n",
      "Layer: 4, Models: (5, 9), Overlap: 0.6434000134468079\n",
      "Layer: 4, Models: (5, 10), Overlap: 0.6443333625793457\n",
      "Layer: 4, Models: (5, 11), Overlap: 0.6435999870300293\n",
      "Layer: 4, Models: (5, 12), Overlap: 0.6387333273887634\n",
      "Layer: 4, Models: (5, 13), Overlap: 0.6412666440010071\n",
      "Layer: 4, Models: (5, 14), Overlap: 0.6416000127792358\n",
      "Layer: 4, Models: (6, 0), Overlap: 0.6391333341598511\n",
      "Layer: 4, Models: (6, 1), Overlap: 0.6443333625793457\n",
      "Layer: 4, Models: (6, 2), Overlap: 0.6428666710853577\n",
      "Layer: 4, Models: (6, 3), Overlap: 0.6431999802589417\n",
      "Layer: 4, Models: (6, 4), Overlap: 0.6463333368301392\n",
      "Layer: 4, Models: (6, 5), Overlap: 0.6475333571434021\n",
      "Layer: 4, Models: (6, 7), Overlap: 0.63919997215271\n",
      "Layer: 4, Models: (6, 8), Overlap: 0.640333354473114\n",
      "Layer: 4, Models: (6, 9), Overlap: 0.6466666460037231\n",
      "Layer: 4, Models: (6, 10), Overlap: 0.6376000046730042\n",
      "Layer: 4, Models: (6, 11), Overlap: 0.6408666372299194\n",
      "Layer: 4, Models: (6, 12), Overlap: 0.6395999789237976\n",
      "Layer: 4, Models: (6, 13), Overlap: 0.6439999938011169\n",
      "Layer: 4, Models: (6, 14), Overlap: 0.6448000073432922\n",
      "Layer: 4, Models: (7, 0), Overlap: 0.6425333619117737\n",
      "Layer: 4, Models: (7, 1), Overlap: 0.6417333483695984\n",
      "Layer: 4, Models: (7, 2), Overlap: 0.6387333273887634\n",
      "Layer: 4, Models: (7, 3), Overlap: 0.6437333226203918\n",
      "Layer: 4, Models: (7, 4), Overlap: 0.6424000263214111\n",
      "Layer: 4, Models: (7, 5), Overlap: 0.6420000195503235\n",
      "Layer: 4, Models: (7, 6), Overlap: 0.63919997215271\n",
      "Layer: 4, Models: (7, 8), Overlap: 0.6423333287239075\n",
      "Layer: 4, Models: (7, 9), Overlap: 0.6419333219528198\n",
      "Layer: 4, Models: (7, 10), Overlap: 0.6429333090782166\n",
      "Layer: 4, Models: (7, 11), Overlap: 0.6442000269889832\n",
      "Layer: 4, Models: (7, 12), Overlap: 0.6444666385650635\n",
      "Layer: 4, Models: (7, 13), Overlap: 0.6407333612442017\n",
      "Layer: 4, Models: (7, 14), Overlap: 0.644599974155426\n",
      "Layer: 4, Models: (8, 0), Overlap: 0.6447333097457886\n",
      "Layer: 4, Models: (8, 1), Overlap: 0.6408666372299194\n",
      "Layer: 4, Models: (8, 2), Overlap: 0.6366666555404663\n",
      "Layer: 4, Models: (8, 3), Overlap: 0.6431999802589417\n",
      "Layer: 4, Models: (8, 4), Overlap: 0.642799973487854\n",
      "Layer: 4, Models: (8, 5), Overlap: 0.6407999992370605\n",
      "Layer: 4, Models: (8, 6), Overlap: 0.640333354473114\n",
      "Layer: 4, Models: (8, 7), Overlap: 0.6423333287239075\n",
      "Layer: 4, Models: (8, 9), Overlap: 0.6421999931335449\n",
      "Layer: 4, Models: (8, 10), Overlap: 0.6412000060081482\n",
      "Layer: 4, Models: (8, 11), Overlap: 0.6437333226203918\n",
      "Layer: 4, Models: (8, 12), Overlap: 0.637333333492279\n",
      "Layer: 4, Models: (8, 13), Overlap: 0.6425999999046326\n",
      "Layer: 4, Models: (8, 14), Overlap: 0.6365333199501038\n",
      "Layer: 4, Models: (9, 0), Overlap: 0.6410666704177856\n",
      "Layer: 4, Models: (9, 1), Overlap: 0.6453333497047424\n",
      "Layer: 4, Models: (9, 2), Overlap: 0.6492666602134705\n",
      "Layer: 4, Models: (9, 3), Overlap: 0.6452666521072388\n",
      "Layer: 4, Models: (9, 4), Overlap: 0.646399974822998\n",
      "Layer: 4, Models: (9, 5), Overlap: 0.6434000134468079\n",
      "Layer: 4, Models: (9, 6), Overlap: 0.6466666460037231\n",
      "Layer: 4, Models: (9, 7), Overlap: 0.6419333219528198\n",
      "Layer: 4, Models: (9, 8), Overlap: 0.6421999931335449\n",
      "Layer: 4, Models: (9, 10), Overlap: 0.6438000202178955\n",
      "Layer: 4, Models: (9, 11), Overlap: 0.6444000005722046\n",
      "Layer: 4, Models: (9, 12), Overlap: 0.6485333442687988\n",
      "Layer: 4, Models: (9, 13), Overlap: 0.6450666785240173\n",
      "Layer: 4, Models: (9, 14), Overlap: 0.6425999999046326\n",
      "Layer: 4, Models: (10, 0), Overlap: 0.6406000256538391\n",
      "Layer: 4, Models: (10, 1), Overlap: 0.6446666717529297\n",
      "Layer: 4, Models: (10, 2), Overlap: 0.6445333361625671\n",
      "Layer: 4, Models: (10, 3), Overlap: 0.6416666507720947\n",
      "Layer: 4, Models: (10, 4), Overlap: 0.6416000127792358\n",
      "Layer: 4, Models: (10, 5), Overlap: 0.6443333625793457\n",
      "Layer: 4, Models: (10, 6), Overlap: 0.6376000046730042\n",
      "Layer: 4, Models: (10, 7), Overlap: 0.6429333090782166\n",
      "Layer: 4, Models: (10, 8), Overlap: 0.6412000060081482\n",
      "Layer: 4, Models: (10, 9), Overlap: 0.6438000202178955\n",
      "Layer: 4, Models: (10, 11), Overlap: 0.6444000005722046\n",
      "Layer: 4, Models: (10, 12), Overlap: 0.6398666501045227\n",
      "Layer: 4, Models: (10, 13), Overlap: 0.6399333477020264\n",
      "Layer: 4, Models: (10, 14), Overlap: 0.647266685962677\n",
      "Layer: 4, Models: (11, 0), Overlap: 0.6398000121116638\n",
      "Layer: 4, Models: (11, 1), Overlap: 0.6445333361625671\n",
      "Layer: 4, Models: (11, 2), Overlap: 0.6439999938011169\n",
      "Layer: 4, Models: (11, 3), Overlap: 0.6437333226203918\n",
      "Layer: 4, Models: (11, 4), Overlap: 0.6425999999046326\n",
      "Layer: 4, Models: (11, 5), Overlap: 0.6435999870300293\n",
      "Layer: 4, Models: (11, 6), Overlap: 0.6408666372299194\n",
      "Layer: 4, Models: (11, 7), Overlap: 0.6442000269889832\n",
      "Layer: 4, Models: (11, 8), Overlap: 0.6437333226203918\n",
      "Layer: 4, Models: (11, 9), Overlap: 0.6444000005722046\n",
      "Layer: 4, Models: (11, 10), Overlap: 0.6444000005722046\n",
      "Layer: 4, Models: (11, 12), Overlap: 0.6441333293914795\n",
      "Layer: 4, Models: (11, 13), Overlap: 0.6420000195503235\n",
      "Layer: 4, Models: (11, 14), Overlap: 0.6434666514396667\n",
      "Layer: 4, Models: (12, 0), Overlap: 0.6401333212852478\n",
      "Layer: 4, Models: (12, 1), Overlap: 0.6413999795913696\n",
      "Layer: 4, Models: (12, 2), Overlap: 0.642133355140686\n",
      "Layer: 4, Models: (12, 3), Overlap: 0.6425333619117737\n",
      "Layer: 4, Models: (12, 4), Overlap: 0.6443333625793457\n",
      "Layer: 4, Models: (12, 5), Overlap: 0.6387333273887634\n",
      "Layer: 4, Models: (12, 6), Overlap: 0.6395999789237976\n",
      "Layer: 4, Models: (12, 7), Overlap: 0.6444666385650635\n",
      "Layer: 4, Models: (12, 8), Overlap: 0.637333333492279\n",
      "Layer: 4, Models: (12, 9), Overlap: 0.6485333442687988\n",
      "Layer: 4, Models: (12, 10), Overlap: 0.6398666501045227\n",
      "Layer: 4, Models: (12, 11), Overlap: 0.6441333293914795\n",
      "Layer: 4, Models: (12, 13), Overlap: 0.6437333226203918\n",
      "Layer: 4, Models: (12, 14), Overlap: 0.6427333354949951\n",
      "Layer: 4, Models: (13, 0), Overlap: 0.6403999924659729\n",
      "Layer: 4, Models: (13, 1), Overlap: 0.6403999924659729\n",
      "Layer: 4, Models: (13, 2), Overlap: 0.6429333090782166\n",
      "Layer: 4, Models: (13, 3), Overlap: 0.6421999931335449\n",
      "Layer: 4, Models: (13, 4), Overlap: 0.6402666568756104\n",
      "Layer: 4, Models: (13, 5), Overlap: 0.6412666440010071\n",
      "Layer: 4, Models: (13, 6), Overlap: 0.6439999938011169\n",
      "Layer: 4, Models: (13, 7), Overlap: 0.6407333612442017\n",
      "Layer: 4, Models: (13, 8), Overlap: 0.6425999999046326\n",
      "Layer: 4, Models: (13, 9), Overlap: 0.6450666785240173\n",
      "Layer: 4, Models: (13, 10), Overlap: 0.6399333477020264\n",
      "Layer: 4, Models: (13, 11), Overlap: 0.6420000195503235\n",
      "Layer: 4, Models: (13, 12), Overlap: 0.6437333226203918\n",
      "Layer: 4, Models: (13, 14), Overlap: 0.642799973487854\n",
      "Layer: 4, Models: (14, 0), Overlap: 0.6439333558082581\n",
      "Layer: 4, Models: (14, 1), Overlap: 0.643666684627533\n",
      "Layer: 4, Models: (14, 2), Overlap: 0.6417999863624573\n",
      "Layer: 4, Models: (14, 3), Overlap: 0.6442000269889832\n",
      "Layer: 4, Models: (14, 4), Overlap: 0.6410666704177856\n",
      "Layer: 4, Models: (14, 5), Overlap: 0.6416000127792358\n",
      "Layer: 4, Models: (14, 6), Overlap: 0.6448000073432922\n",
      "Layer: 4, Models: (14, 7), Overlap: 0.644599974155426\n",
      "Layer: 4, Models: (14, 8), Overlap: 0.6365333199501038\n",
      "Layer: 4, Models: (14, 9), Overlap: 0.6425999999046326\n",
      "Layer: 4, Models: (14, 10), Overlap: 0.647266685962677\n",
      "Layer: 4, Models: (14, 11), Overlap: 0.6434666514396667\n",
      "Layer: 4, Models: (14, 12), Overlap: 0.6427333354949951\n",
      "Layer: 4, Models: (14, 13), Overlap: 0.642799973487854\n"
     ]
    }
   ],
   "source": [
    "task = 0\n",
    "for layer, modelidx in product([0,2,4], product([i for i in range(15)], [i for i in range(15)])):\n",
    "    modeli, modelj = modelidx\n",
    "    if modeli == modelj:\n",
    "        continue\n",
    "    print('Layer: {}, Models: ({}, {}), Overlap: {}'.format(layer, modeli, modelj,\n",
    "        overlap(model_maps[modeli].state_dict()['model.{}.stacked'.format(layer)][task],\n",
    "                model_maps[modelj].state_dict()['model.{}.stacked'.format(layer)][task])))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## New Task performance using task 0 masks only!"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Only New Tasks)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 271,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_zero = BasisMultitaskFC(hidden_size=300, num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 272,
   "metadata": {},
   "outputs": [],
   "source": [
    "seed_dict = basis_model_zero.state_dict()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 273,
   "metadata": {},
   "outputs": [],
   "source": [
    "update_dict = seed_dict.copy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 274,
   "metadata": {},
   "outputs": [],
   "source": [
    "assert all([(model_map[0].state_dict()['model.0.weight'] == model_map[i].state_dict()['model.0.weight']).all() for i in range(1,15)])\n",
    "update_dict['model.0.weight'] = model_map[0].state_dict()['model.0.weight']\n",
    "for task in range(15):\n",
    "    for layer in [0,2,4]:\n",
    "        update_dict['model.{}.scores.{}'.format(layer, task)] = model_map[task].state_dict()['model.{}.scores.0'.format(layer)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 275,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n"
     ]
    }
   ],
   "source": [
    "seed_dict.update(update_dict)\n",
    "basis_model_zero.load_state_dict(seed_dict, False)\n",
    "cache_masks(basis_model_zero)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 276,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(15, 20)"
      ]
     },
     "execution_count": 276,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "num_seed_tasks_learned, num_tasks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 277,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 15\n",
      "=> Set task of model.0 to 15\n",
      "=> Set task of model.2 to 15\n",
      "=> Set task of model.4 to 15\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8492, Acc@1 0.1826\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2111\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.9094, Acc@1 0.1963\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.2073\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 16\n",
      "=> Setting learned tasks of model.2 to 16\n",
      "=> Setting learned tasks of model.4 to 16\n",
      "\n",
      "Training for task 16\n",
      "=> Set task of model.0 to 16\n",
      "=> Set task of model.2 to 16\n",
      "=> Set task of model.4 to 16\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8647, Acc@1 0.2272\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2612\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8692, Acc@1 0.2588\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.2557\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 17\n",
      "=> Setting learned tasks of model.2 to 17\n",
      "=> Setting learned tasks of model.4 to 17\n",
      "\n",
      "Training for task 17\n",
      "=> Set task of model.0 to 17\n",
      "=> Set task of model.2 to 17\n",
      "=> Set task of model.4 to 17\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0088, Acc@1 0.2031\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2468\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8724, Acc@1 0.2604\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.2442\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 18\n",
      "=> Setting learned tasks of model.2 to 18\n",
      "=> Setting learned tasks of model.4 to 18\n",
      "\n",
      "Training for task 18\n",
      "=> Set task of model.0 to 18\n",
      "=> Set task of model.2 to 18\n",
      "=> Set task of model.4 to 18\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8419, Acc@1 0.2070\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2473\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8511, Acc@1 0.2546\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.2527\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 19\n",
      "=> Setting learned tasks of model.2 to 19\n",
      "=> Setting learned tasks of model.4 to 19\n",
      "\n",
      "Training for task 19\n",
      "=> Set task of model.0 to 19\n",
      "=> Set task of model.2 to 19\n",
      "=> Set task of model.4 to 19\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.9444, Acc@1 0.1637\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2081\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.3216, Acc@1 0.2191\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.2172\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 20\n",
      "=> Setting learned tasks of model.2 to 20\n",
      "=> Setting learned tasks of model.4 to 20\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(num_seed_tasks_learned, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model_zero, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model_zero.parameters() if p.requires_grad], lr=1e-2)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(2):\n",
    "        train(basis_model_zero, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model_zero, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model_zero)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model_zero, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 278,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.0688\n",
      "Per task performance\n",
      "Task 0: 0.0000\n",
      "Task 1: 0.0014\n",
      "Task 2: 0.0056\n",
      "Task 3: 0.0648\n",
      "Task 4: 0.0017\n",
      "Task 5: 0.0023\n",
      "Task 6: 0.0245\n",
      "Task 7: 0.0056\n",
      "Task 8: 0.0121\n",
      "Task 9: 0.0011\n",
      "Task 10: 0.0181\n",
      "Task 11: 0.0012\n",
      "Task 12: 0.0193\n",
      "Task 13: 0.0292\n",
      "Task 14: 0.0117\n",
      "Task 15: 0.2073\n",
      "Task 16: 0.2557\n",
      "Task 17: 0.2442\n",
      "Task 18: 0.2527\n",
      "Task 19: 0.2172\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_zero, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_model_zero, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 279,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_mnist_task_zero_only'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Only New Tasks) Sparsity Enabled"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 287,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_zeros = BasisSparseMultitaskFC(hidden_size=300, num_tasks=num_tasks, num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=False, sparsity=0.25)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 288,
   "metadata": {},
   "outputs": [],
   "source": [
    "seed_dict = basis_model_zeros.state_dict()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 289,
   "metadata": {},
   "outputs": [],
   "source": [
    "update_dict = seed_dict.copy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 290,
   "metadata": {},
   "outputs": [],
   "source": [
    "assert all([(model_maps[0].state_dict()['model.0.weight'] == model_maps[i].state_dict()['model.0.weight']).all() for i in range(1,15)])\n",
    "update_dict['model.0.weight'] = model_map[0].state_dict()['model.0.weight']\n",
    "for task in range(15):\n",
    "    for layer in [0,2,4]:\n",
    "        update_dict['model.{}.scores.{}'.format(layer, task)] = model_maps[task].state_dict()['model.{}.scores.0'.format(layer)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 291,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n"
     ]
    }
   ],
   "source": [
    "seed_dict.update(update_dict)\n",
    "basis_model_zeros.load_state_dict(seed_dict, False)\n",
    "cache_masks(basis_model_zeros)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 292,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(15, 20)"
      ]
     },
     "execution_count": 292,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "num_seed_tasks_learned, num_tasks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 293,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 15\n",
      "=> Set task of model.0 to 15\n",
      "=> Set task of model.2 to 15\n",
      "=> Set task of model.4 to 15\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 3.0438, Acc@1 0.2035\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2395\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.9901, Acc@1 0.2288\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.2440\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 16\n",
      "=> Setting learned tasks of model.2 to 16\n",
      "=> Setting learned tasks of model.4 to 16\n",
      "\n",
      "Training for task 16\n",
      "=> Set task of model.0 to 16\n",
      "=> Set task of model.2 to 16\n",
      "=> Set task of model.4 to 16\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.6824, Acc@1 0.2598\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.3140\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8247, Acc@1 0.3197\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.3165\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 17\n",
      "=> Setting learned tasks of model.2 to 17\n",
      "=> Setting learned tasks of model.4 to 17\n",
      "\n",
      "Training for task 17\n",
      "=> Set task of model.0 to 17\n",
      "=> Set task of model.2 to 17\n",
      "=> Set task of model.4 to 17\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.6087, Acc@1 0.2148\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2645\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.6360, Acc@1 0.2676\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.2508\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 18\n",
      "=> Setting learned tasks of model.2 to 18\n",
      "=> Setting learned tasks of model.4 to 18\n",
      "\n",
      "Training for task 18\n",
      "=> Set task of model.0 to 18\n",
      "=> Set task of model.2 to 18\n",
      "=> Set task of model.4 to 18\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.8992, Acc@1 0.2344\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.2567\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.7513, Acc@1 0.2575\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.2893\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 19\n",
      "=> Setting learned tasks of model.2 to 19\n",
      "=> Setting learned tasks of model.4 to 19\n",
      "\n",
      "Training for task 19\n",
      "=> Set task of model.0 to 19\n",
      "=> Set task of model.2 to 19\n",
      "=> Set task of model.4 to 19\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.5227, Acc@1 0.2640\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.3277\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.9361, Acc@1 0.2992\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.3242\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 20\n",
      "=> Setting learned tasks of model.2 to 20\n",
      "=> Setting learned tasks of model.4 to 20\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(num_seed_tasks_learned, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model_zeros, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model_zeros.parameters() if p.requires_grad], lr=1e-2)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(2):\n",
    "        train(basis_model_zeros, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model_zeros, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model_zeros)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model_zeros, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 294,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.0759\n",
      "Per task performance\n",
      "Task 0: 0.0022\n",
      "Task 1: 0.0037\n",
      "Task 2: 0.0143\n",
      "Task 3: 0.0016\n",
      "Task 4: 0.0084\n",
      "Task 5: 0.0001\n",
      "Task 6: 0.0007\n",
      "Task 7: 0.0030\n",
      "Task 8: 0.0184\n",
      "Task 9: 0.0193\n",
      "Task 10: 0.0010\n",
      "Task 11: 0.0067\n",
      "Task 12: 0.0009\n",
      "Task 13: 0.0024\n",
      "Task 14: 0.0111\n",
      "Task 15: 0.2440\n",
      "Task 16: 0.3165\n",
      "Task 17: 0.2508\n",
      "Task 18: 0.2893\n",
      "Task 19: 0.3242\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_zeros, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_model_zeros, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 295,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_mnist_task_zero_only_sparsity_25'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Hybrid (Last layer with sparsity)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (Only New Tasks)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 305,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 20 # For demonstration purposes, we go up to 2500 in our paper\n",
    "num_seed_tasks_learned = 15\n",
    "basis_model_zero_sh = BasisHiddenOnlySparseMultitaskFC(hidden_size=300, num_tasks=num_tasks, \n",
    "                                num_seed_tasks_learned=num_seed_tasks_learned, start_at_optimal=True, sparsity=0.25)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 306,
   "metadata": {},
   "outputs": [],
   "source": [
    "seed_dict = basis_model_zero_sh.state_dict()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 307,
   "metadata": {},
   "outputs": [],
   "source": [
    "update_dict = seed_dict.copy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 308,
   "metadata": {},
   "outputs": [],
   "source": [
    "assert all([(model_maps[0].state_dict()['model.0.weight'] == model_maps[i].state_dict()['model.0.weight']).all() for i in range(1,15)])\n",
    "update_dict['model.0.weight'] = model_maps[0].state_dict()['model.0.weight']\n",
    "for task in range(15):\n",
    "    for layer in [0,2]:\n",
    "        update_dict['model.{}.scores.{}'.format(layer, task)] = model_maps[task].state_dict()['model.{}.scores.0'.format(layer)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 310,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n"
     ]
    }
   ],
   "source": [
    "seed_dict.update(update_dict)\n",
    "basis_model_zero_sh.load_state_dict(seed_dict, False)\n",
    "cache_masks(basis_model_zero_sh)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 311,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(15, 20)"
      ]
     },
     "execution_count": 311,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "num_seed_tasks_learned, num_tasks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 313,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 15\n",
      "=> Set task of model.0 to 15\n",
      "=> Set task of model.2 to 15\n",
      "=> Set task of model.4 to 15\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f8cb2ae8925a45b49554ca6bb147fcd6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.2329, Acc@1 0.6400\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "dabe6822037b4d048fb56153659776d0",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7035\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7268852919fe445a8e1760bfb170c8bd",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.1218, Acc@1 0.6872\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "825fcd9a9c58486f8856445f401379cc",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.7029\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 16\n",
      "=> Setting learned tasks of model.2 to 16\n",
      "=> Setting learned tasks of model.4 to 16\n",
      "\n",
      "Training for task 16\n",
      "=> Set task of model.0 to 16\n",
      "=> Set task of model.2 to 16\n",
      "=> Set task of model.4 to 16\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "2fa60328db77412ca5e2971e9bb4d022",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.8850, Acc@1 0.6598\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "99a8e4c6bbd74954b5250c43a28d4ffa",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.6777\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "89bb5a8e33c44fa9909fd8b75aa48f78",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.9747, Acc@1 0.6602\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "558b21215c89475ea1e8cd00b0dacf1c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.6955\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 17\n",
      "=> Setting learned tasks of model.2 to 17\n",
      "=> Setting learned tasks of model.4 to 17\n",
      "\n",
      "Training for task 17\n",
      "=> Set task of model.0 to 17\n",
      "=> Set task of model.2 to 17\n",
      "=> Set task of model.4 to 17\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "051a63601f2141c0a1c00b91ecd4de91",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.9942, Acc@1 0.6693\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d78be754c5f04ad0995970413c70cf9b",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7069\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e69143ae7b0b4e3aa24093bab782129d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.0425, Acc@1 0.7067\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "41133620132d484ea4e85ae49c34c3aa",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.6881\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 18\n",
      "=> Setting learned tasks of model.2 to 18\n",
      "=> Setting learned tasks of model.4 to 18\n",
      "\n",
      "Training for task 18\n",
      "=> Set task of model.0 to 18\n",
      "=> Set task of model.2 to 18\n",
      "=> Set task of model.4 to 18\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "1bff7ee6fecf417c9b9c884bdfa0d6b5",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.8554, Acc@1 0.6445\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "fa8068ae2ecf410f9061e5a52e5dd28f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.6575\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "71cab350bc05452390c255343c499732",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 1.0503, Acc@1 0.6621\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "9bc0f6bf1e194658893969fe26020567",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.6575\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 19\n",
      "=> Setting learned tasks of model.2 to 19\n",
      "=> Setting learned tasks of model.4 to 19\n",
      "\n",
      "Training for task 19\n",
      "=> Set task of model.0 to 19\n",
      "=> Set task of model.2 to 19\n",
      "=> Set task of model.4 to 19\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "aaea79f5259b499cb9b8cee05ca67489",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.7341, Acc@1 0.7217\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f93f9d60294e404ba6ca7d0ac429d3de",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.7128\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "268f885913b0432ab4bf9fa922f3117a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.9278, Acc@1 0.7334\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e54659e5d92e4d63bbc90fae2e09a35a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 2 epochs Acc@1 0.6836\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 20\n",
      "=> Setting learned tasks of model.2 to 20\n",
      "=> Setting learned tasks of model.4 to 20\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(num_seed_tasks_learned, num_tasks):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model_zero_sh, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model_zero_sh.parameters() if p.requires_grad], lr=1e-2)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(2):\n",
    "        train(basis_model_zero_sh, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model_zero_sh, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model_zero_sh)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model_zero_sh, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 314,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.1774\n",
      "Per task performance\n",
      "Task 0: 0.0027\n",
      "Task 1: 0.0270\n",
      "Task 2: 0.0143\n",
      "Task 3: 0.0001\n",
      "Task 4: 0.0054\n",
      "Task 5: 0.0098\n",
      "Task 6: 0.0108\n",
      "Task 7: 0.0005\n",
      "Task 8: 0.0029\n",
      "Task 9: 0.0023\n",
      "Task 10: 0.0020\n",
      "Task 11: 0.0173\n",
      "Task 12: 0.0031\n",
      "Task 13: 0.0011\n",
      "Task 14: 0.0215\n",
      "Task 15: 0.7029\n",
      "Task 16: 0.6955\n",
      "Task 17: 0.6881\n",
      "Task 18: 0.6575\n",
      "Task 19: 0.6836\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(num_tasks):\n",
    "    set_model_task(basis_model_zero_sh, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(basis_model_zero_sh, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(num_tasks):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 315,
   "metadata": {},
   "outputs": [],
   "source": [
    "performance_map['basis_mnist_task_zero_only_hidden_sparsity_25'] = gg_performance.copy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model Initialization (All Tasks, Zero start)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 320,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_tasks = 15 # For demonstration purposes, we go up to 2500 in our paper\n",
    "basis_model = BasisMultitaskFC(hidden_size=300, num_tasks=num_tasks, num_seed_tasks_learned=5, start_at_optimal=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 323,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n"
     ]
    }
   ],
   "source": [
    "seed_dict = model.state_dict()\n",
    "basis_dict = basis_model.state_dict()\n",
    "load_dict = {k: seed_dict[k] for k in basis_model.state_dict().keys() if k in seed_dict.keys()}\n",
    "basis_dict.update(load_dict)\n",
    "basis_model.load_state_dict(basis_dict, False)\n",
    "cache_masks(basis_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 324,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training for task 0\n",
      "=> Set task of model.0 to 0\n",
      "=> Set task of model.2 to 0\n",
      "=> Set task of model.4 to 0\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d2b6247e69364a20951502c4e6cef812",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.5117, Acc@1 0.6963\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "71d9e26f308540b9955b1c1338175856",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.8642\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 1\n",
      "=> Setting learned tasks of model.2 to 1\n",
      "=> Setting learned tasks of model.4 to 1\n",
      "\n",
      "Training for task 1\n",
      "=> Set task of model.0 to 1\n",
      "=> Set task of model.2 to 1\n",
      "=> Set task of model.4 to 1\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8981c6a903a540619de0dd655f3ce70f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.4671, Acc@1 0.6878\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "357c198fa6534b3ba3d03c505fa2fc65",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.8814\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 2\n",
      "=> Setting learned tasks of model.2 to 2\n",
      "=> Setting learned tasks of model.4 to 2\n",
      "\n",
      "Training for task 2\n",
      "=> Set task of model.0 to 2\n",
      "=> Set task of model.2 to 2\n",
      "=> Set task of model.4 to 2\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "24dc783cceaf4107a7e7a5d02700c044",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.4373, Acc@1 0.7464\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "0e9313c1d7b94410aef36a29db456cdc",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.8866\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 3\n",
      "=> Setting learned tasks of model.2 to 3\n",
      "=> Setting learned tasks of model.4 to 3\n",
      "\n",
      "Training for task 3\n",
      "=> Set task of model.0 to 3\n",
      "=> Set task of model.2 to 3\n",
      "=> Set task of model.4 to 3\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "88caf1cb594b416ab7399195ff023669",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.4166, Acc@1 0.7474\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "75ef164fa9e64f348bba759b579250d9",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.8936\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 4\n",
      "=> Setting learned tasks of model.2 to 4\n",
      "=> Setting learned tasks of model.4 to 4\n",
      "\n",
      "Training for task 4\n",
      "=> Set task of model.0 to 4\n",
      "=> Set task of model.2 to 4\n",
      "=> Set task of model.4 to 4\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8ba7d9f67e3b4875992ecaf227ab7739",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 0.4466, Acc@1 0.7380\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "4026b1cfca36468dbf6a26dfe87112ff",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.8923\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 5\n",
      "=> Setting learned tasks of model.2 to 5\n",
      "=> Setting learned tasks of model.4 to 5\n",
      "\n",
      "Training for task 5\n",
      "=> Set task of model.0 to 5\n",
      "=> Set task of model.2 to 5\n",
      "=> Set task of model.4 to 5\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b9c63d1d209f4ac783f6d305c0348456",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.4617, Acc@1 0.1748\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "dac02d0f98f44646ba41c50e51baccba",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1884\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 6\n",
      "=> Setting learned tasks of model.2 to 6\n",
      "=> Setting learned tasks of model.4 to 6\n",
      "\n",
      "Training for task 6\n",
      "=> Set task of model.0 to 6\n",
      "=> Set task of model.2 to 6\n",
      "=> Set task of model.4 to 6\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "46c9e17ab6054eaea6f73b66658209ca",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/469 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Validation => Loss 2.5350, Acc@1 0.1419\n",
      "============\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "b873fbd03333420586c2637aa5e76664",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/79 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Val Perf after 1 epochs Acc@1 0.1791\n",
      "=> Caching mask state for model.0\n",
      "=> Caching mask state for model.2\n",
      "=> Caching mask state for model.4\n",
      "\n",
      "=> Setting learned tasks of model.0 to 7\n",
      "=> Setting learned tasks of model.2 to 7\n",
      "=> Setting learned tasks of model.4 to 7\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for task in range(7):\n",
    "    print(f\"Training for task {task}\")\n",
    "    set_model_task(basis_model, task)\n",
    "    mnist.update_task(task)\n",
    "\n",
    "    optimizer = optim.RMSprop([p for p in basis_model.parameters() if p.requires_grad], lr=1e-3)\n",
    "    # Train for 1 epoch\n",
    "    for e in range(1):\n",
    "        train(basis_model, mnist.train_loader, optimizer, e)\n",
    "        \n",
    "        print(\"Validation\")\n",
    "        print(\"============\")\n",
    "        acc1 = evaluate(basis_model, mnist.val_loader, e)\n",
    "        \n",
    "    \n",
    "    cache_masks(basis_model)\n",
    "    print()\n",
    "    set_num_tasks_learned(basis_model, task + 1)\n",
    "    print()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Inference"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Scenario GG: Task ID is given at train time and inference time\n",
    "\n",
    "If task ID is given at inference time, we simply evaluate using the mask we have trained for that ID. This procedure is shown below."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average top 1 performance: 0.9253\n",
      "Per task performance\n",
      "Task 0: 0.9255\n",
      "Task 1: 0.9290\n",
      "Task 2: 0.9222\n",
      "Task 3: 0.9253\n",
      "Task 4: 0.9243\n"
     ]
    }
   ],
   "source": [
    "# When task ID we can simply set the mask and evaluate\n",
    "\n",
    "gg_performance = []\n",
    "for task in range(5):\n",
    "    set_model_task(model, task)\n",
    "    mnist.update_task(task)\n",
    "    acc1 = evaluate(model, mnist.val_loader, 0)\n",
    "    gg_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "    \n",
    "print(f\"Average top 1 performance: {(sum(gg_performance) / len(gg_performance)):.4f}\")\n",
    "\n",
    "print(\"Per task performance\")\n",
    "for t in range(5):\n",
    "    print(f\"Task {t}: {gg_performance[t]:.4f}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Scenario GNs: Task ID is given at train time but not inference time\n",
    "\n",
    "We can infer task identity at inference time for arbitrary pieces of data. Here we use the one-shot algorithm described in the paper for simplicity. In short, we assign each learned supermask $M^i$ a weight $\\alpha_i$ such that $\\sum_{i} \\alpha_i = 1$ and $\\alpha_i >= 0$. \n",
    "\n",
    "Given an example data point $x$ to classify, we can compute our loss as $H = \\mathcal{H}\\left(f\\left(x, \\theta \\odot \\left(\\sum_i \\alpha_i M^i\\right)\\right)\\right)$ where $f(x, \\theta)$ is our neural network which outputs logits and $\\mathcal H$ is our entropy function. From here our inferred task is simply \n",
    "$$t = \\text{argmin}_i \\frac{\\partial\\mathcal{H}}{\\partial \\alpha_i}.$$\n",
    "\n",
    "#### Intuition\n",
    "\n",
    "We can see entropy as a measure of confidence in our prediction. High entropy prediction distributions are very uncertain (close to uniform) and lowest entropy is reached when our distribution is very certain (at a one hot vector). So it stands to reason that supermasks which haven't been trained on the exemplar data will have high entropy. Further, instead of showing each individual supermask $x$, which would be computationally intractible for many tasks, we can show $x$ to a _superposition_ of supermasks and optimize for superposition coefficients $\\alpha_i$. Choosing the smallest $\\alpha_i$ for our task is essentially greedily moving in the direction of maximal decrease in entropy. \n",
    "\n",
    "Below is a simple implementation of oneshot task inference."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# When task info is not provided we can infer it\n",
    "# here we use the oneshot task inference alg detailed in the paper\n",
    "\n",
    "def oneshot_task_inference(model, batch, num_tasks):\n",
    "    # Set task < 0 for inference mode\n",
    "    set_model_task(model, -1, verbose=False)\n",
    "    \n",
    "    # Initialize alphas to uniform\n",
    "    alphas = torch.ones(num_tasks, 1, 1) / num_tasks\n",
    "    alphas.requires_grad_(True)\n",
    "    set_alphas(model, alphas, verbose=False)\n",
    "    \n",
    "    logits = model(batch)\n",
    "    \n",
    "    # Entropy of logits\n",
    "    entropy = -(logits.softmax(dim=1) * logits.log_softmax(dim=1)).sum(1).mean()\n",
    "    \n",
    "    # Gradient wrt alphas\n",
    "    g, = autograd.grad(entropy, alphas)\n",
    "    \n",
    "    inferred_task = (-g).squeeze().argmax()\n",
    "\n",
    "    return inferred_task.item()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Task inference accuracy\n",
    "\n",
    "Below we show how accurate task inference is using our oneshot algorithm. If you would like to evaluate on more tasks (we use 5 for demonstration), just modify the `num_tasks` variable in the cell where we declare `MultitaskFC`. \n",
    "\n",
    "Here we evaluate task inference on 50 random images per task (the `trials` variable modifies this number) and show we predict the correct task with 100% accuracy. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ad31d78ed0ba45d8887ceaaff5aa1de8",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/50 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "ename": "NameError",
     "evalue": "name 'model' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-9-59ae2beeb500>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      8\u001b[0m         \u001b[0mmnist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mupdate_task\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtask\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      9\u001b[0m         inferred_task = oneshot_task_inference(\n\u001b[0;32m---> 10\u001b[0;31m             \u001b[0mmodel\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     11\u001b[0m             \u001b[0mbatch\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mnext\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0miter\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mmnist\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mval_loader\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0mnum_examples\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     12\u001b[0m             \u001b[0mnum_tasks\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;36m5\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mNameError\u001b[0m: name 'model' is not defined"
     ]
    }
   ],
   "source": [
    "# How many examples to show for task inference\n",
    "num_examples = 1\n",
    "trials = 50\n",
    "num_correct = 0\n",
    "num_seen = 0\n",
    "for _ in tqdm(range(trials)):\n",
    "    for task in range(5):\n",
    "        mnist.update_task(task)\n",
    "        inferred_task = oneshot_task_inference(\n",
    "            model,\n",
    "            batch=next(iter(mnist.val_loader))[0][:num_examples],\n",
    "            num_tasks=5\n",
    "        )\n",
    "        if inferred_task == task:\n",
    "            num_correct += 1\n",
    "        num_seen += 1\n",
    "\n",
    "print(f\"Task inference accuracy: {100 * num_correct / num_seen}%\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Full inference procedure\n",
    "\n",
    "Below is an example of how this could be used in an inference pipeline. In this scenario, we use one random image from the validation set to infer task identity $t$ and then evaluate with the appropriate mask $M^t$ on all tasks. We see that there is no drop in performance from the GG scenario. \n",
    "\n",
    "Note that for simplicity this is a small deviation from the evaluation setup in our paper. In our paper we use 1 image per batch to determine the task identity of the batch, and evaluate that batch using the inferred task. So task inference needs to be correct more often to match GG performance. We have already shown task inference accuracy is 100% with high probability in the previous cell so we don't need to do that here. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "=> Average top 1 performance: 92.5260%\n",
      "=> Difference from GG performance: 0.0000\n"
     ]
    }
   ],
   "source": [
    "# Evaluation\n",
    "\n",
    "num_examples = 1\n",
    "\n",
    "# After task inference, we just perform the same evaluation procedure as before\n",
    "gns_performance = []\n",
    "for task in range(5):\n",
    "    mnist.update_task(task)\n",
    "    inferred_task = oneshot_task_inference(\n",
    "        model,\n",
    "        batch=next(iter(mnist.val_loader))[0][:num_examples],\n",
    "        num_tasks=5\n",
    "    )\n",
    "    set_model_task(model, inferred_task)\n",
    "    acc1 = evaluate(model, mnist.val_loader, 0)\n",
    "    gns_performance.append(acc1.item())\n",
    "\n",
    "clear_output()\n",
    "avg_acc = (sum(gns_performance) / len(gns_performance))\n",
    "print(f\"=> Average top 1 performance: {100*avg_acc:.4f}%\")\n",
    "print(f\"=> Difference from GG performance: \"\n",
    "      f\"{(sum(gg_performance) / len(gg_performance) - avg_acc):.4f}\")"
   ]
  }
 ],
 "metadata": {
  "@webio": {
   "lastCommId": null,
   "lastKernelId": null
  },
  "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.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
