{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "a432f9d9",
   "metadata": {},
   "outputs": [
    {
     "ename": "ImportError",
     "evalue": "cannot import name 'Transformer' from 'End2End.models.transformer' (/opt/tiger/kinwai/jointist/End2End/models/transformer.py)",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mImportError\u001b[0m                               Traceback (most recent call last)",
      "\u001b[0;32m/tmp/ipykernel_2777363/2978061229.py\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      1\u001b[0m \u001b[0;32mfrom\u001b[0m \u001b[0mEnd2End\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mmodels\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mutils\u001b[0m \u001b[0;32mimport\u001b[0m \u001b[0minit_layer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0minit_bn\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 2\u001b[0;31m \u001b[0;32mfrom\u001b[0m \u001b[0mEnd2End\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mmodels\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtransformer\u001b[0m \u001b[0;32mimport\u001b[0m \u001b[0mTransformer\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m      3\u001b[0m \u001b[0;32mfrom\u001b[0m \u001b[0mEnd2End\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mmodels\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mposition_encoding\u001b[0m \u001b[0;32mimport\u001b[0m \u001b[0mPositionEmbeddingSine\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      4\u001b[0m \u001b[0;32mimport\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      5\u001b[0m \u001b[0;32mimport\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mnn\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0mnn\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mImportError\u001b[0m: cannot import name 'Transformer' from 'End2End.models.transformer' (/opt/tiger/kinwai/jointist/End2End/models/transformer.py)"
     ]
    }
   ],
   "source": [
    "from End2End.models.utils import init_layer, init_bn\n",
    "from End2End.models.transformer import Transformer\n",
    "from End2End.models.position_encoding import PositionEmbeddingSine\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import matplotlib.pyplot as plt\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5be2dec9",
   "metadata": {},
   "outputs": [],
   "source": [
    "sequential = nn.Sequential()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "878e0176",
   "metadata": {},
   "outputs": [
    {
     "ename": "AttributeError",
     "evalue": "'Sequential' object has no attribute 'add'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mAttributeError\u001b[0m                            Traceback (most recent call last)",
      "\u001b[0;32m/tmp/ipykernel_2073252/3121341305.py\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0msequential\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0madd\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mnn\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mLinear\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m10\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;36m5\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[0m",
      "\u001b[0;32m~/anaconda3/envs/jointist/lib/python3.8/site-packages/torch/nn/modules/module.py\u001b[0m in \u001b[0;36m__getattr__\u001b[0;34m(self, name)\u001b[0m\n\u001b[1;32m   1175\u001b[0m             \u001b[0;32mif\u001b[0m \u001b[0mname\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mmodules\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1176\u001b[0m                 \u001b[0;32mreturn\u001b[0m \u001b[0mmodules\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mname\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1177\u001b[0;31m         raise AttributeError(\"'{}' object has no attribute '{}'\".format(\n\u001b[0m\u001b[1;32m   1178\u001b[0m             type(self).__name__, name))\n\u001b[1;32m   1179\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mAttributeError\u001b[0m: 'Sequential' object has no attribute 'add'"
     ]
    }
   ],
   "source": [
    "sequential.add(nn.Linear(10,5))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "0681bd5d",
   "metadata": {},
   "outputs": [],
   "source": [
    "hidden_dim = 256\n",
    "dropout = 0.1\n",
    "nheads = 8\n",
    "dim_feedforward = 2048\n",
    "enc_layers = 1\n",
    "dec_layers = 6\n",
    "pre_norm = False\n",
    "return_intermdiate_dec = True\n",
    "position_embedding = 'sine'\n",
    "\n",
    "num_queries=100"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "759a2dba",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "dim_feedforward=2048\n"
     ]
    }
   ],
   "source": [
    "model = Transformer(\n",
    "            d_model=hidden_dim,\n",
    "            dropout=dropout,\n",
    "            nhead=nheads,\n",
    "            dim_feedforward=dim_feedforward,\n",
    "            num_encoder_layers=enc_layers,\n",
    "            num_decoder_layers=dec_layers,\n",
    "            normalize_before=pre_norm,\n",
    "            return_intermediate_dec=True\n",
    "            )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "4e826a76",
   "metadata": {},
   "outputs": [],
   "source": [
    "x = torch.randn(4,256,10,7)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "563e30e9",
   "metadata": {},
   "outputs": [],
   "source": [
    "pos_encoder = PositionEmbeddingSine()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "0e2a7d99",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "PositionEmbeddingSine()"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pos_encoder"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "dcb19203",
   "metadata": {},
   "outputs": [],
   "source": [
    "pos = pos_encoder(x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "a263fca1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([4, 256, 10, 7])"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pos.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "bf9bc39a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7fec635978e0>]"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot(pos[0,0,:,5])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "d5d75c5c",
   "metadata": {},
   "outputs": [],
   "source": [
    "query_embed = nn.Embedding(num_queries, hidden_dim)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "e1093949",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([100, 256])"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "query_embed.weight.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "e97bc93f",
   "metadata": {},
   "outputs": [],
   "source": [
    "hs = model(x, None, query_embed.weight, pos)[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "cd04a4b4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([6, 4, 100, 256])"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "hs.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "93aeb9e1",
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'args' 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/tmp/ipykernel_2058094/3754298389.py\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      1\u001b[0m \u001b[0mnum_queries\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;32m----> 2\u001b[0;31m \u001b[0mhidden_dim\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0margs\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mhidden_dim\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m      3\u001b[0m \u001b[0msrc\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrandn\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m4\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;36m256\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;36m10\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;36m3\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      4\u001b[0m \u001b[0mquery_embed\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnn\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mEmbedding\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mnum_queries\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mhidden_dim\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m      5\u001b[0m \u001b[0mpos_embed\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrandn\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m4\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;36m10\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;36m256\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mNameError\u001b[0m: name 'args' is not defined"
     ]
    }
   ],
   "source": [
    "num_queries = 5\n",
    "hidden_dim = args.hidden_dim\n",
    "src = torch.randn(4,256,10,3)\n",
    "query_embed = nn.Embedding(num_queries, hidden_dim)\n",
    "pos_embed = torch.randn(4,10,256)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8d790a15",
   "metadata": {},
   "outputs": [],
   "source": [
    "query_embed"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d1e99e9c",
   "metadata": {},
   "outputs": [],
   "source": [
    "transformer(src, None, query_embed, pos_embed)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "335df4df",
   "metadata": {},
   "source": [
    "# Backbone test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ee5e02a7",
   "metadata": {},
   "outputs": [],
   "source": [
    "\"\"\"\n",
    "Backbone modules.\n",
    "\"\"\"\n",
    "from collections import OrderedDict\n",
    "\n",
    "import torch\n",
    "import torch.nn.functional as F\n",
    "import torchvision\n",
    "from torch import nn\n",
    "from torchvision.models._utils import IntermediateLayerGetter\n",
    "from typing import Dict, List\n",
    "\n",
    "from End2End.util.misc import NestedTensor, is_main_process\n",
    "\n",
    "from position_encoding import build_position_encoding\n",
    "\n",
    "\n",
    "class FrozenBatchNorm2d(torch.nn.Module):\n",
    "    \"\"\"\n",
    "    BatchNorm2d where the batch statistics and the affine parameters are fixed.\n",
    "    Copy-paste from torchvision.misc.ops with added eps before rqsrt,\n",
    "    without which any other models than torchvision.models.resnet[18,34,50,101]\n",
    "    produce nans.\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self, n):\n",
    "        super(FrozenBatchNorm2d, self).__init__()\n",
    "        self.register_buffer(\"weight\", torch.ones(n))\n",
    "        self.register_buffer(\"bias\", torch.zeros(n))\n",
    "        self.register_buffer(\"running_mean\", torch.zeros(n))\n",
    "        self.register_buffer(\"running_var\", torch.ones(n))\n",
    "\n",
    "    def _load_from_state_dict(self, state_dict, prefix, local_metadata, strict,\n",
    "                              missing_keys, unexpected_keys, error_msgs):\n",
    "        num_batches_tracked_key = prefix + 'num_batches_tracked'\n",
    "        if num_batches_tracked_key in state_dict:\n",
    "            del state_dict[num_batches_tracked_key]\n",
    "\n",
    "        super(FrozenBatchNorm2d, self)._load_from_state_dict(\n",
    "            state_dict, prefix, local_metadata, strict,\n",
    "            missing_keys, unexpected_keys, error_msgs)\n",
    "\n",
    "    def forward(self, x):\n",
    "        # move reshapes to the beginning\n",
    "        # to make it fuser-friendly\n",
    "        w = self.weight.reshape(1, -1, 1, 1)\n",
    "        b = self.bias.reshape(1, -1, 1, 1)\n",
    "        rv = self.running_var.reshape(1, -1, 1, 1)\n",
    "        rm = self.running_mean.reshape(1, -1, 1, 1)\n",
    "        eps = 1e-5\n",
    "        scale = w * (rv + eps).rsqrt()\n",
    "        bias = b - rm * scale\n",
    "        return x * scale + bias\n",
    "\n",
    "\n",
    "class BackboneBase(nn.Module):\n",
    "\n",
    "    def __init__(self, backbone: nn.Module, train_backbone: bool, num_channels: int, return_interm_layers: bool):\n",
    "        super().__init__()\n",
    "        for name, parameter in backbone.named_parameters():\n",
    "            if not train_backbone or 'layer2' not in name and 'layer3' not in name and 'layer4' not in name:\n",
    "                parameter.requires_grad_(False)\n",
    "        if return_interm_layers:\n",
    "            return_layers = {\"layer1\": \"0\", \"layer2\": \"1\", \"layer3\": \"2\", \"layer4\": \"3\"}\n",
    "        else:\n",
    "            return_layers = {'layer4': \"0\"}\n",
    "        self.body = IntermediateLayerGetter(backbone, return_layers=return_layers)\n",
    "        self.num_channels = num_channels\n",
    "\n",
    "    def forward(self, tensor_list: NestedTensor):\n",
    "        xs = self.body(tensor_list.tensors)\n",
    "        out: Dict[str, NestedTensor] = {}\n",
    "        for name, x in xs.items():\n",
    "            m = tensor_list.mask\n",
    "            assert m is not None\n",
    "            mask = F.interpolate(m[None].float(), size=x.shape[-2:]).to(torch.bool)[0]\n",
    "            out[name] = NestedTensor(x, mask)\n",
    "        return out\n",
    "    \n",
    "class Backbone(BackboneBase):\n",
    "    \"\"\"ResNet backbone with frozen BatchNorm.\"\"\"\n",
    "    def __init__(self, name: str,\n",
    "                 train_backbone: bool,\n",
    "                 return_interm_layers: bool,\n",
    "                 dilation: bool):\n",
    "        backbone = getattr(torchvision.models, name)(\n",
    "            replace_stride_with_dilation=[False, False, dilation],\n",
    "            pretrained=is_main_process(), norm_layer=FrozenBatchNorm2d)\n",
    "        num_channels = 512 if name in ('resnet18', 'resnet34') else 2048\n",
    "        super().__init__(backbone, train_backbone, num_channels, return_interm_layers)\n",
    "\n",
    "\n",
    "class Joiner(nn.Sequential):\n",
    "    def __init__(self, backbone, position_embedding):\n",
    "        super().__init__(backbone, position_embedding)\n",
    "\n",
    "    def forward(self, tensor_list: NestedTensor):\n",
    "        xs = self[0](tensor_list)\n",
    "        out: List[NestedTensor] = []\n",
    "        pos = []\n",
    "        for name, x in xs.items():\n",
    "            out.append(x)\n",
    "            # position encoding\n",
    "            pos.append(self[1](x).to(x.tensors.dtype))\n",
    "\n",
    "        return out, pos\n",
    "\n",
    "\n",
    "def build_backbone(args):\n",
    "    position_embedding = build_position_encoding(args)\n",
    "    train_backbone = args.lr_backbone > 0\n",
    "    return_interm_layers = args.masks\n",
    "    backbone = Backbone(args.backbone, train_backbone, return_interm_layers, args.dilation)\n",
    "    model = Joiner(backbone, position_embedding)\n",
    "    model.num_channels = backbone.num_channels\n",
    "    return model    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "ea23de06",
   "metadata": {},
   "outputs": [],
   "source": [
    "model = build_backbone(args)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "616adc4f",
   "metadata": {},
   "outputs": [
    {
     "ename": "AttributeError",
     "evalue": "'Tensor' object has no attribute 'tensors'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mAttributeError\u001b[0m                            Traceback (most recent call last)",
      "\u001b[0;32m/tmp/ipykernel_440529/15704970.py\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      1\u001b[0m \u001b[0mx\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrandn\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m4\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;36m3\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;36m15\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;36m15\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m----> 2\u001b[0;31m \u001b[0mmodel\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mx\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
      "\u001b[0;32m~/anaconda3/envs/jointist/lib/python3.8/site-packages/torch/nn/modules/module.py\u001b[0m in \u001b[0;36m_call_impl\u001b[0;34m(self, *input, **kwargs)\u001b[0m\n\u001b[1;32m   1100\u001b[0m         if not (self._backward_hooks or self._forward_hooks or self._forward_pre_hooks or _global_backward_hooks\n\u001b[1;32m   1101\u001b[0m                 or _global_forward_hooks or _global_forward_pre_hooks):\n\u001b[0;32m-> 1102\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mforward_call\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0minput\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\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   1103\u001b[0m         \u001b[0;31m# Do not call functions when jit is used\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1104\u001b[0m         \u001b[0mfull_backward_hooks\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mnon_full_backward_hooks\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m,\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[0;32m/tmp/ipykernel_440529/4092894891.py\u001b[0m in \u001b[0;36mforward\u001b[0;34m(self, tensor_list)\u001b[0m\n\u001b[1;32m     69\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     70\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mforward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtensor_list\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mNestedTensor\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[0;32m---> 71\u001b[0;31m         \u001b[0mxs\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbody\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mtensor_list\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtensors\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     72\u001b[0m         \u001b[0mout\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mDict\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0mstr\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mNestedTensor\u001b[0m\u001b[0;34m]\u001b[0m \u001b[0;34m=\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     73\u001b[0m         \u001b[0;32mfor\u001b[0m \u001b[0mname\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mx\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mxs\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mitems\u001b[0m\u001b[0;34m(\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[0;31mAttributeError\u001b[0m: 'Tensor' object has no attribute 'tensors'"
     ]
    }
   ],
   "source": [
    "x = torch.randn(4,3,15,15)\n",
    "model[0](x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "ac8bf1db",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Args():\n",
    "    def __init__(self):\n",
    "        self.hidden_dim = 256\n",
    "        self.dropout = 0.1\n",
    "        self.nheads = 8\n",
    "        self.dim_feedforward = 2048\n",
    "        self.enc_layers = 6\n",
    "        self.dec_layers = 6\n",
    "        self.pre_norm = False\n",
    "        self.return_intermdiate_dec = True\n",
    "        self.position_embedding = 'sine'\n",
    "        self.lr_backbone = 1e-5\n",
    "        self.masks = False\n",
    "        self.backbone = 'resnet50'\n",
    "        self.dilation = True\n",
    "args = Args()        "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "39ac79b1",
   "metadata": {},
   "outputs": [],
   "source": [
    "position_embedding = build_position_encoding(args)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "084ba42a",
   "metadata": {},
   "outputs": [],
   "source": [
    "x = NestedTensor(torch.zeros(8,3,10,10), torch.zeros(8,10,10).bool())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "d10a8252",
   "metadata": {},
   "outputs": [],
   "source": [
    "mask = torch.zeros(10,10).bool()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "8068a378",
   "metadata": {},
   "outputs": [],
   "source": [
    "pos_mask_ones = position_embedding(x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "0b8788c9",
   "metadata": {},
   "outputs": [],
   "source": [
    "pos_mask_zeros = position_embedding(x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "9e4f12f2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([8, 256, 10, 10])"
      ]
     },
     "execution_count": 42,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pos_mask_ones.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "id": "8ce553cd",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([8, 256, 10, 10])"
      ]
     },
     "execution_count": 43,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "pos_mask_zeros.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "4a34440e",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "1ad2dc76",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x7ff943041730>"
      ]
     },
     "execution_count": 47,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.imshow(pos_mask_ones[0].flatten(1), aspect='auto')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "92ead62e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x7ff943311b50>"
      ]
     },
     "execution_count": 51,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.imshow(pos_mask_zeros[0].flatten(1), aspect='auto')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0ae24022",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "jointist",
   "language": "python",
   "name": "jointist"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
