{
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "367UV2Kz15PW",
        "outputId": "441637e6-3213-4aae-b3f5-e481654b567b"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Requirement already satisfied: pyscnn in /usr/local/lib/python3.10/dist-packages (0.0.9b0)\n",
            "Requirement already satisfied: numpy>=1.21.3 in /usr/local/lib/python3.10/dist-packages (from pyscnn) (1.25.2)\n",
            "Requirement already satisfied: torch>=1.10.0 in /usr/local/lib/python3.10/dist-packages (from pyscnn) (2.3.0+cu121)\n",
            "Requirement already satisfied: cvxpy>=1.2.1 in /usr/local/lib/python3.10/dist-packages (from pyscnn) (1.3.4)\n",
            "Requirement already satisfied: scikit-learn>=1.0.0 in /usr/local/lib/python3.10/dist-packages (from pyscnn) (1.2.2)\n",
            "Requirement already satisfied: scipy>=1.7.2 in /usr/local/lib/python3.10/dist-packages (from pyscnn) (1.11.4)\n",
            "Requirement already satisfied: typing-extensions in /usr/local/lib/python3.10/dist-packages (from pyscnn) (4.11.0)\n",
            "Requirement already satisfied: tqdm in /usr/local/lib/python3.10/dist-packages (from pyscnn) (4.66.4)\n",
            "Requirement already satisfied: opt-einsum in /usr/local/lib/python3.10/dist-packages (from pyscnn) (3.3.0)\n",
            "Requirement already satisfied: linalg-backends in /usr/local/lib/python3.10/dist-packages (from pyscnn) (0.0.1b0)\n",
            "Requirement already satisfied: osqp>=0.4.1 in /usr/local/lib/python3.10/dist-packages (from cvxpy>=1.2.1->pyscnn) (0.6.2.post8)\n",
            "Requirement already satisfied: ecos>=2 in /usr/local/lib/python3.10/dist-packages (from cvxpy>=1.2.1->pyscnn) (2.0.13)\n",
            "Requirement already satisfied: scs>=1.1.6 in /usr/local/lib/python3.10/dist-packages (from cvxpy>=1.2.1->pyscnn) (3.2.4.post1)\n",
            "Requirement already satisfied: setuptools>65.5.1 in /usr/local/lib/python3.10/dist-packages (from cvxpy>=1.2.1->pyscnn) (67.7.2)\n",
            "Requirement already satisfied: joblib>=1.1.1 in /usr/local/lib/python3.10/dist-packages (from scikit-learn>=1.0.0->pyscnn) (1.4.2)\n",
            "Requirement already satisfied: threadpoolctl>=2.0.0 in /usr/local/lib/python3.10/dist-packages (from scikit-learn>=1.0.0->pyscnn) (3.5.0)\n",
            "Requirement already satisfied: filelock in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (3.14.0)\n",
            "Requirement already satisfied: sympy in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (1.12)\n",
            "Requirement already satisfied: networkx in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (3.3)\n",
            "Requirement already satisfied: jinja2 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (3.1.4)\n",
            "Requirement already satisfied: fsspec in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (2023.6.0)\n",
            "Requirement already satisfied: nvidia-cuda-nvrtc-cu12==12.1.105 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (12.1.105)\n",
            "Requirement already satisfied: nvidia-cuda-runtime-cu12==12.1.105 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (12.1.105)\n",
            "Requirement already satisfied: nvidia-cuda-cupti-cu12==12.1.105 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (12.1.105)\n",
            "Requirement already satisfied: nvidia-cudnn-cu12==8.9.2.26 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (8.9.2.26)\n",
            "Requirement already satisfied: nvidia-cublas-cu12==12.1.3.1 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (12.1.3.1)\n",
            "Requirement already satisfied: nvidia-cufft-cu12==11.0.2.54 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (11.0.2.54)\n",
            "Requirement already satisfied: nvidia-curand-cu12==10.3.2.106 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (10.3.2.106)\n",
            "Requirement already satisfied: nvidia-cusolver-cu12==11.4.5.107 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (11.4.5.107)\n",
            "Requirement already satisfied: nvidia-cusparse-cu12==12.1.0.106 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (12.1.0.106)\n",
            "Requirement already satisfied: nvidia-nccl-cu12==2.20.5 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (2.20.5)\n",
            "Requirement already satisfied: nvidia-nvtx-cu12==12.1.105 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (12.1.105)\n",
            "Requirement already satisfied: triton==2.3.0 in /usr/local/lib/python3.10/dist-packages (from torch>=1.10.0->pyscnn) (2.3.0)\n",
            "Requirement already satisfied: nvidia-nvjitlink-cu12 in /usr/local/lib/python3.10/dist-packages (from nvidia-cusolver-cu12==11.4.5.107->torch>=1.10.0->pyscnn) (12.5.40)\n",
            "Requirement already satisfied: qdldl in /usr/local/lib/python3.10/dist-packages (from osqp>=0.4.1->cvxpy>=1.2.1->pyscnn) (0.1.7.post2)\n",
            "Requirement already satisfied: MarkupSafe>=2.0 in /usr/local/lib/python3.10/dist-packages (from jinja2->torch>=1.10.0->pyscnn) (2.1.5)\n",
            "Requirement already satisfied: mpmath>=0.19 in /usr/local/lib/python3.10/dist-packages (from sympy->torch>=1.10.0->pyscnn) (1.3.0)\n"
          ]
        }
      ],
      "source": [
        "!pip install pyscnn"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "5fG7YlZk38Hk"
      },
      "outputs": [],
      "source": [
        "from typing import Tuple, Optional, Union, Callable\n",
        "from typing_extensions import Literal\n",
        "import math\n",
        "\n",
        "import numpy as np\n",
        "from scipy.stats import ortho_group  # type: ignore\n",
        "\n",
        "Transform = Literal[\"cosine\", \"polynomial\"]\n",
        "\n",
        "Dataset = Tuple[np.ndarray, np.ndarray]"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "ifpo8q3B1tlM"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "from scnn.private.utils.data import gen_classification_data, gen_regression_data\n",
        "\n",
        "\n",
        "from scnn.models import ConvexGatedReLU, ConvexReLU\n",
        "from scnn.solvers import RFISTA, AL, LeastSquaresSolver, CVXPYSolver, ApproximateConeDecomposition\n",
        "from scnn.regularizers import NeuronGL1, L2, L1, FeatureGL1\n",
        "from scnn.metrics import Metrics\n",
        "from scnn.activations import sample_gate_vectors\n",
        "from scnn.optimize import optimize_model, optimize"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "HFmj3toLxZym"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "import torch\n",
        "import torchvision\n",
        "from torch import nn, optim\n",
        "import torch.nn.functional as F\n",
        "from torchvision import datasets, transforms\n",
        "from torch.autograd import Variable\n",
        "from torch.utils.data.sampler import Sampler\n",
        "import matplotlib.pyplot as plt\n",
        "import random\n",
        "import copy"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "A9cTGPKTFvV4",
        "outputId": "9d8706da-de65-4fe4-80f7-dd52dac9df8e"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Drive already mounted at /content/gdrive; to attempt to forcibly remount, call drive.mount(\"/content/gdrive\", force_remount=True).\n"
          ]
        }
      ],
      "source": [
        "from google.colab import drive\n",
        "drive.mount('/content/gdrive')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "DQ97qiMG26ra",
        "outputId": "ea02b10a-36f4-4cbd-92e6-b02935e1c2ec"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "cuda\n"
          ]
        }
      ],
      "source": [
        "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
        "print(device)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "miOHKtSS2Qrz"
      },
      "outputs": [],
      "source": [
        "import torch\n",
        "\n",
        "# Manually compute output from non-convex weights for sanity check:\n",
        "def gated_relu_prediction(X, gates, W1, W2):\n",
        "    \"\"\"\n",
        "    Compute the prediction of a Gated ReLU network.\n",
        "\n",
        "    Args:\n",
        "    - X: Input tensor of shape (N, D), where N is the number of samples and D is the number of features.\n",
        "    - gates: Gate vectors tensor of shape (D, P), where P is the number of neurons in the hidden layer.\n",
        "    - W1: First layer weights of shape (P, D).\n",
        "    - W2: Second layer weights of shape (C, P), where C is the output dimension.\n",
        "\n",
        "    Returns:\n",
        "    - Output predictions of shape (N, C).\n",
        "    \"\"\"\n",
        "    # Step 1: Compute the intermediate activations (X * W1^T)\n",
        "    intermediate_activations = torch.mm(X, W1.T)  # Shape: (N, P)\n",
        "\n",
        "    # Step 2: Compute the gates activation (should be on X against the gates, then determine the gating on neurons)\n",
        "    gate_activations = torch.mm(X, gates) > 0  # Shape: (N, P)\n",
        "\n",
        "    # Step 3: Apply gating to intermediate activations\n",
        "    gated_activations = intermediate_activations * gate_activations.float()  # Element-wise multiplication\n",
        "\n",
        "    # Step 4: Compute the final output using W2\n",
        "    output = torch.mm(gated_activations, W2.T)  # Shape: (N, C)\n",
        "\n",
        "    return gated_activations, output"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "RHsdQviIMvpp"
      },
      "outputs": [],
      "source": [
        "def report_test_accuracy(grelu_model):\n",
        "        W1, W2 = grelu_model.get_parameters()\n",
        "        gates = grelu_model.G\n",
        "        W1 = torch.Tensor(W1).to(device)\n",
        "        W2 = torch.Tensor(W2).to(device)\n",
        "        gates = torch.Tensor(gates).to(device)\n",
        "\n",
        "        test_class_correct = list(0. for i in range(num_classes))\n",
        "        test_class_total = list(0. for i in range(num_classes))\n",
        "        test_overall_correct = 0\n",
        "        test_overall_total = 0\n",
        "\n",
        "        orig_model.eval()\n",
        "\n",
        "        with torch.no_grad():\n",
        "\n",
        "            for data in test_loader:\n",
        "\n",
        "                images, labels = data\n",
        "                images, labels = images.to(device), labels.to(device)\n",
        "\n",
        "                _ = orig_model(images)\n",
        "\n",
        "                input = activation[\"layer3\"]\n",
        "                target_output = activation[\"layer4\"]\n",
        "\n",
        "                _, batch_predictions = gated_relu_prediction(torch.from_numpy(input.reshape((images.shape[0], -1)).cpu().numpy()).to(device), gates, W1, W2)\n",
        "\n",
        "                # _, batch_predictions = gated_relu_prediction(torch.from_numpy(scaler.transform(input.reshape((images.shape[0], -1)).cpu().numpy())).to(device), gates, W1, W2)\n",
        "\n",
        "                output = orig_model.fc(torch.reshape(torch.Tensor(batch_predictions), (target_output.shape[0], -1)).to(device))\n",
        "\n",
        "                predicted = output.argmax(dim=1, keepdim=True)\n",
        "                predicted = predicted.squeeze()\n",
        "\n",
        "                for i in range(len(predicted)):\n",
        "                    index = labels[i].item()\n",
        "                    test_class_correct[index] += int(labels[i] == predicted[i].item())\n",
        "                    test_class_total[index] += 1\n",
        "\n",
        "        overall_correct = 0\n",
        "        overall_total = 0\n",
        "\n",
        "        for i in range(num_classes):\n",
        "\n",
        "            test_overall_correct += test_class_correct[i]\n",
        "            test_overall_total += test_class_total[i]\n",
        "\n",
        "        return 100.0 * test_overall_correct / test_overall_total"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "GiR5fOg3xfAs"
      },
      "outputs": [],
      "source": [
        "def report_test_accuracy_W2(W2_temp):\n",
        "\n",
        "        test_loss = 0\n",
        "        num_classes = 10\n",
        "        test_class_correct = list(0. for i in range(num_classes))\n",
        "        test_class_total = list(0. for i in range(num_classes))\n",
        "        test_overall_correct = 0\n",
        "        test_overall_total = 0\n",
        "\n",
        "        orig_model.eval()\n",
        "\n",
        "        mse = 0\n",
        "\n",
        "        with torch.no_grad():\n",
        "\n",
        "            for data in test_loader:\n",
        "\n",
        "                images, labels = data\n",
        "                images, labels = images.to(device), labels.to(device)\n",
        "\n",
        "                _ = orig_model(images)\n",
        "\n",
        "                input = activation[\"layer3\"]\n",
        "                target_output = activation[\"layer4\"]\n",
        "\n",
        "                _, batch_predictions = gated_relu_prediction(torch.from_numpy(input.reshape((images.shape[0], -1)).cpu().numpy()).to(device), gates, W1, W2_temp.T)\n",
        "\n",
        "                # _, batch_predictions = gated_relu_prediction(torch.from_numpy(scaler.transform(input.reshape((images.shape[0], -1)).cpu().numpy())).to(device), gates, W1, W2_temp.T)\n",
        "\n",
        "                output = orig_model.fc(orig_model.avgpool(batch_predictions.reshape(target_output.shape)).reshape(input.shape[0], -1))\n",
        "\n",
        "                predicted = output.argmax(dim=1, keepdim=True)\n",
        "                predicted = predicted.squeeze()\n",
        "\n",
        "                for i in range(len(predicted)):\n",
        "                    index = labels[i].item()\n",
        "                    test_class_correct[index] += int(labels[i] == predicted[i].item())\n",
        "                    test_class_total[index] += 1\n",
        "\n",
        "                del images, labels, input, target_output, batch_predictions, output, predicted\n",
        "                torch.cuda.empty_cache()\n",
        "\n",
        "        test_loss /= len(test_loader.dataset)\n",
        "\n",
        "        overall_correct = 0\n",
        "        overall_total = 0\n",
        "\n",
        "        for i in range(num_classes):\n",
        "\n",
        "            test_overall_correct += test_class_correct[i]\n",
        "            test_overall_total += test_class_total[i]\n",
        "\n",
        "        _, test_predictions = gated_relu_prediction(X_test_SCNN, gates, W1, W2_temp.T)\n",
        "\n",
        "        test_mse = torch.mean((test_predictions - y_test_SCNN)**2).item()\n",
        "\n",
        "        _, train_predictions = gated_relu_prediction(X_train_SCNN_new, gates, W1, W2_temp.T)\n",
        "\n",
        "        train_mse = torch.mean((train_predictions - y_train_SCNN_new)**2).item()\n",
        "\n",
        "        return 100.0 * test_overall_correct / test_overall_total, train_mse, test_mse"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "5H7bC2BFF67r"
      },
      "outputs": [],
      "source": [
        "import torch\n",
        "\n",
        "def count_effective_entries(W1, sparse_W2, p, hidden_dim):\n",
        "    \"\"\"\n",
        "    Calculate the number of contributing elements in W1 given W1 and sparse_W2.\n",
        "\n",
        "    Args:\n",
        "    - W1 (torch.Tensor): First layer weights of shape (P, D).\n",
        "    - sparse_W2 (torch.Tensor): Sparse second layer weights of shape (C, P).\n",
        "    - p (int): Number of smaller matrices concatenated in W1 and sparse_W2.\n",
        "    - hidden_dim (int): Hidden dimension of the smaller matrices.\n",
        "\n",
        "    Returns:\n",
        "    - int: Number of contributing elements in W1.\n",
        "    \"\"\"\n",
        "\n",
        "    num_contributing_elements = 0\n",
        "\n",
        "    for i in range(p):\n",
        "        # Extract the i-th smaller matrices from W1 and sparse_W2\n",
        "        W1_i = W1[i * hidden_dim: (i + 1) * hidden_dim]\n",
        "        sparse_W2_i = sparse_W2[:, i * hidden_dim: (i + 1) * hidden_dim]\n",
        "\n",
        "        # Find the non-zero indices in the i-th smaller matrix of sparse_W2\n",
        "        non_zero_indices_W2 = sparse_W2_i.nonzero(as_tuple=False)\n",
        "\n",
        "        # Find the unique column indices (corresponding to elements in W1_i)\n",
        "        unique_indices_W2 = non_zero_indices_W2[:, 1].unique()\n",
        "\n",
        "        # Extract relevant columns in W1_i\n",
        "        contributing_W1_i = W1_i[unique_indices_W2]\n",
        "\n",
        "        # Count the non-zero elements in the extracted relevant columns\n",
        "        num_contributing_elements += contributing_W1_i.nonzero(as_tuple=False).size(0)\n",
        "\n",
        "    return num_contributing_elements"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "JpB0bHBg40Ja",
        "outputId": "13678f3f-6d38-4055-c290-5597779060b2"
      },
      "outputs": [
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "/usr/local/lib/python3.10/dist-packages/torchvision/models/_utils.py:208: UserWarning: The parameter 'pretrained' is deprecated since 0.13 and may be removed in the future, please use 'weights' instead.\n",
            "  warnings.warn(\n",
            "/usr/local/lib/python3.10/dist-packages/torchvision/models/_utils.py:223: UserWarning: Arguments other than a weight enum or `None` for 'weights' are deprecated since 0.13 and may be removed in the future. The current behavior is equivalent to passing `weights=ResNet18_Weights.IMAGENET1K_V1`. You can also use `weights=ResNet18_Weights.DEFAULT` to get the most up-to-date weights.\n",
            "  warnings.warn(msg)\n"
          ]
        },
        {
          "data": {
            "text/plain": [
              "ResNet(\n",
              "  (conv1): Conv2d(3, 64, kernel_size=(7, 7), stride=(2, 2), padding=(3, 3), bias=False)\n",
              "  (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "  (relu): ReLU(inplace=True)\n",
              "  (maxpool): MaxPool2d(kernel_size=3, stride=2, padding=1, dilation=1, ceil_mode=False)\n",
              "  (layer1): Sequential(\n",
              "    (0): BasicBlock(\n",
              "      (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "    (1): BasicBlock(\n",
              "      (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "  )\n",
              "  (layer2): Sequential(\n",
              "    (0): BasicBlock(\n",
              "      (conv1): Conv2d(64, 128, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (downsample): Sequential(\n",
              "        (0): Conv2d(64, 128, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "        (1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      )\n",
              "    )\n",
              "    (1): BasicBlock(\n",
              "      (conv1): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "  )\n",
              "  (layer3): Sequential(\n",
              "    (0): BasicBlock(\n",
              "      (conv1): Conv2d(128, 256, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (downsample): Sequential(\n",
              "        (0): Conv2d(128, 256, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "        (1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      )\n",
              "    )\n",
              "    (1): BasicBlock(\n",
              "      (conv1): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "  )\n",
              "  (layer4): Sequential(\n",
              "    (0): BasicBlock(\n",
              "      (conv1): Conv2d(256, 512, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (downsample): Sequential(\n",
              "        (0): Conv2d(256, 512, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "        (1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      )\n",
              "    )\n",
              "    (1): BasicBlock(\n",
              "      (conv1): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "  )\n",
              "  (avgpool): AdaptiveAvgPool2d(output_size=(1, 1))\n",
              "  (fc): Linear(in_features=512, out_features=1000, bias=True)\n",
              ")"
            ]
          },
          "execution_count": 11,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "import torchvision.models as models\n",
        "checkpoint = torch.load('/content/gdrive/My Drive/From Non-Convex to Convex/Fine-tune Pre-trained Resnet18/10 Class Experiments/TransformedImages/FFTSGDCosine/epoch189_testloss_0.0047605928719043735_testacc_87.61')\n",
        "orig_model = torchvision.models.resnet18(pretrained=True).to(device)\n",
        "orig_model.load_state_dict(checkpoint['model_state_dict'])\n",
        "orig_model.eval()"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "ih9OeeNW42KG",
        "outputId": "aa0cf9b2-430b-4552-849b-9a389d8b83c9"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "<torch.utils.hooks.RemovableHandle at 0x7d80e51982b0>"
            ]
          },
          "execution_count": 12,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "activation = {}\n",
        "def get_activation(name):\n",
        "    def hook(model, input, output):\n",
        "        activation[name] = output.to(device)\n",
        "    return hook\n",
        "\n",
        "orig_model.maxpool.register_forward_hook(get_activation('layer0'))\n",
        "orig_model.layer1.register_forward_hook(get_activation('layer1'))\n",
        "orig_model.layer2.register_forward_hook(get_activation('layer2'))\n",
        "orig_model.layer3.register_forward_hook(get_activation('layer3'))\n",
        "orig_model.layer4.register_forward_hook(get_activation('layer4'))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "ArRh32Q76DjL"
      },
      "outputs": [],
      "source": [
        "num_train_samples = 10\n",
        "num_train_gaussian_samples = 0"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "9xWBuVET_tCO"
      },
      "source": [
        "### Create Dataset of N samples/class"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "PUXYwua428If",
        "outputId": "7d85492a-60eb-4213-bb81-69a4c974c3ac"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Files already downloaded and verified\n",
            "Files already downloaded and verified\n"
          ]
        }
      ],
      "source": [
        "torch.manual_seed(7)\n",
        "\n",
        "transform = transforms.Compose([transforms.ToTensor(),\n",
        "                                transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))])\n",
        "\n",
        "transform_train = transforms.Compose([\n",
        "    transforms.RandomCrop(32, padding=4),\n",
        "    transforms.RandomHorizontalFlip(),\n",
        "    transforms.ToTensor(),\n",
        "    transforms.Normalize((0.4914, 0.4822, 0.4465), (0.2023, 0.1994, 0.2010)),\n",
        "])\n",
        "\n",
        "transform_test = transforms.Compose([\n",
        "    transforms.ToTensor(),\n",
        "    transforms.Normalize((0.4914, 0.4822, 0.4465), (0.2023, 0.1994, 0.2010)),\n",
        "])\n",
        "\n",
        "train_dataset = datasets.CIFAR10('./data', train=True, download=True, transform=transform_train)\n",
        "test_dataset = datasets.CIFAR10('./data', train=False, download=True, transform=transform_test)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "uOWCRoq5V1T8"
      },
      "outputs": [],
      "source": [
        "targets = list(range(10))\n",
        "\n",
        "random.seed(0)\n",
        "\n",
        "train_indices = [i for i, label in enumerate(train_dataset.targets) if label in targets]\n",
        "\n",
        "sampled_train_indices = []\n",
        "for target in targets:\n",
        "    sampled_train_indices += random.sample([i for i, label in enumerate(train_dataset.targets) if label == target], num_train_samples)\n",
        "\n",
        "train_subset = torch.utils.data.Subset(train_dataset, sampled_train_indices)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 1000
        },
        "id": "G1v7UqAL3LEg",
        "outputId": "f91224e0-2a2d-4dff-bef2-68491997e29e"
      },
      "outputs": [
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "data": {
            "image/png": "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",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "0 (tensor([[[-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         ...,\n",
            "         [-0.7038, -0.2967, -0.4518,  ..., -0.5681, -0.5681, -0.4712],\n",
            "         [-0.6650, -0.3936, -0.4712,  ..., -0.6069, -0.3549, -0.2773],\n",
            "         [-0.5875, -0.4518, -0.4324,  ..., -0.3161, -0.0641, -0.1029]],\n",
            "\n",
            "        [[-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         ...,\n",
            "         [-0.0386,  0.2958,  0.1974,  ...,  0.1384,  0.1778,  0.2761],\n",
            "         [ 0.0598,  0.3154,  0.2564,  ...,  0.1778,  0.1778,  0.1974],\n",
            "         [ 0.0204,  0.1778,  0.1974,  ...,  0.2761,  0.2958,  0.2171]],\n",
            "\n",
            "        [[-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         ...,\n",
            "         [-0.4655, -0.1143, -0.2313,  ..., -0.3484, -0.3484, -0.2313],\n",
            "         [-0.3874, -0.1533, -0.2704,  ..., -0.3874, -0.2313, -0.1728],\n",
            "         [-0.4655, -0.3679, -0.4069,  ..., -0.3289, -0.1533, -0.2118]]]), 0) plane\n"
          ]
        },
        {
          "data": {
            "image/png": "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vfpaU+/c3CxTfffOgktzLx1imO0mSS2G2zzeOAtu6BDn0oz0PFPrROND8/r6Oufa2IGDtuZNLYZi6zy9mHtpP8MPcntMajx9GWdAAAAvCCAAgBcEEADACwIIAOAFAQQA8IIAAgB4QQABALwggAAAXhBAAAAvCCAAgBc9dhTPofekxhNutUMMk0TOF5Mvs43XsahynzoiSdqw5S/OtRX795t6DxntPornysLppt6Z2dnOtYdraky9161+wlQvpTtXhgfbxuUMGul+G+7futPUWwPcRyuljBlvaj04PcO5NjvHNuIpe4h7b8n2SH7Ta5tMvQ9u3epe3NJq6q1k91+e6UPc7yftbVG9d/j0dZwBAQC8IIAAAF4QQAAALwggAIAXBBAAwAsCCADgBQEEAPCCAAIAeEEAAQC8IIAAAF4QQAAAL+KCIAh8L+KjGhoaFA6HFYlElJqa6vQ1bYb+CWe2LHzExu2NpvrHn1jjXBse5D7zTJKOHG1yrt252zbH7M677nSuverzA029U0zV0sEq99p3Kv9q6t0Scv+p+O1z6029n/79H51rL0hMNvU+WrnfvbghYuptZ3gsH59oa90eNRS32Hor5l56wQD32uCE1Lb5tL/HOQMCAHhhCqCSkhJNnTpVKSkpysjI0Pz581VRUdGpZsaMGYqLi+u03XLLLV26aABA72cKoLKyMhUVFWnz5s168cUXFYvFNGvWLDU3N3equ+mmm1RTU9OxPfDAA126aABA72d6P6B169Z1+nzlypXKyMhQeXm5pk//23utJCcnKysrq2tWCADok87qOaBI5IMn9tLSOr+p0WOPPab09HRNmDBBxcXFamk59RNj0WhUDQ0NnTYAQN93xu+I2t7erttvv11XXnmlJkyY0HH5ddddp+HDhysnJ0c7duzQnXfeqYqKCj355JMn7VNSUqJ77733TJcBAOilzjiAioqKtGvXLm3cuLHT5TfffHPHxxMnTlR2drZmzpypyspKjR49+hN9iouLtXTp0o7PGxoaNGzYsDNdFgCglzijAFqyZImeffZZbdiwQUOHDv3U2vz8fEnS3r17TxpAoVBIoVDoTJYBAOjFTAEUBIFuu+02rV27VqWlpRo5cuRpv2b79u2SpOzs7DNaIACgbzIFUFFRkVatWqWnn35aKSkpqq2tlSSFw2H1799flZWVWrVqlb7yla9o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",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "1 (tensor([[[-2.4291e+00, -2.4291e+00, -2.4291e+00,  ..., -2.4291e+00,\n",
            "          -2.4291e+00, -2.4291e+00],\n",
            "         [-2.4291e+00, -2.4291e+00, -2.4291e+00,  ..., -2.4291e+00,\n",
            "          -2.4291e+00, -2.4291e+00],\n",
            "         [ 1.8550e+00,  1.2735e+00,  9.4391e-01,  ..., -1.3241e+00,\n",
            "          -1.2854e+00, -9.5581e-01],\n",
            "         ...,\n",
            "         [ 3.2359e-01,  9.4391e-01,  1.1959e+00,  ...,  6.5314e-01,\n",
            "           5.3683e-01,  4.5929e-01],\n",
            "         [ 4.7867e-01,  1.3510e+00,  1.4479e+00,  ...,  1.6851e-01,\n",
            "          -1.0288e-01, -4.4721e-02],\n",
            "         [ 8.2760e-01,  1.4673e+00,  1.1378e+00,  ...,  9.8268e-01,\n",
            "           5.1744e-01,  4.0113e-01]],\n",
            "\n",
            "        [[-2.4183e+00, -2.4183e+00, -2.4183e+00,  ..., -2.4183e+00,\n",
            "          -2.4183e+00, -2.4183e+00],\n",
            "         [-2.4183e+00, -2.4183e+00, -2.4183e+00,  ..., -2.4183e+00,\n",
            "          -2.4183e+00, -2.4183e+00],\n",
            "         [ 1.9478e+00,  1.3381e+00,  1.0038e+00,  ..., -9.2357e-01,\n",
            "          -8.6457e-01, -6.2857e-01],\n",
            "         ...,\n",
            "         [ 4.3344e-01,  1.0628e+00,  1.3184e+00,  ...,  8.0711e-01,\n",
            "           6.8911e-01,  6.1044e-01],\n",
            "         [ 5.5144e-01,  1.4364e+00,  1.5544e+00,  ...,  2.7610e-01,\n",
            "           7.6703e-04,  5.9768e-02],\n",
            "         [ 8.8578e-01,  1.5348e+00,  1.2004e+00,  ...,  1.0824e+00,\n",
            "           5.9077e-01,  4.7277e-01]],\n",
            "\n",
            "        [[-2.2214e+00, -2.2214e+00, -2.2214e+00,  ..., -2.2214e+00,\n",
            "          -2.2214e+00, -2.2214e+00],\n",
            "         [-2.2214e+00, -2.2214e+00, -2.2214e+00,  ..., -2.2214e+00,\n",
            "          -2.2214e+00, -2.2214e+00],\n",
            "         [ 2.1879e+00,  1.6221e+00,  1.3490e+00,  ..., -2.7036e-01,\n",
            "          -2.3134e-01, -1.5330e-01],\n",
            "         ...,\n",
            "         [ 7.6368e-01,  1.3880e+00,  1.6416e+00,  ...,  1.1344e+00,\n",
            "           1.0173e+00,  9.3927e-01],\n",
            "         [ 8.2221e-01,  1.7002e+00,  1.7977e+00,  ...,  6.4662e-01,\n",
            "           3.7348e-01,  4.3201e-01],\n",
            "         [ 1.0758e+00,  1.7197e+00,  1.3880e+00,  ...,  1.3490e+00,\n",
            "           8.8074e-01,  7.4417e-01]]]), 0) plane\n"
          ]
        },
        {
          "data": {
            "image/png": "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iyETQCtXrtTFF1+szMxMlZWV6Te/+Y3vJSXUt771LaWkpAzYpk6d6ntZZ2TTpk264YYbVFJSopSUFD333HMDrg+CQA888ICKi4s1YsQIVVRUaO/evX4WewZOt5+33377x47twoUL/Sx2kKqrq3XllVcqJydHBQUFWrJkierr6wfU9PT0qLKyUqNHj1Z2draWLl2qtrY2TyseHJf9nDt37seO59133+1pxYOzatUqzZw5s//NpuXl5XrxxRf7rz9bx3JIBNDPf/5zrVixQg8++KB+97vfadasWVqwYIEOHz7se2kJddlll6mlpaV/e/31130v6Yx0dXVp1qxZWrly5Umvf/TRR/XDH/5QTzzxhLZu3aoLL7xQCxYsUE9Pz1le6Zk53X5K0sKFCwcc26effvosrvDM1dXVqbKyUlu2bNHLL7+s3t5ezZ8/X11dXf01999/v55//nmtW7dOdXV1OnTokG666SaPq7Zz2U9JuvPOOwccz0cffdTTigdn7NixeuSRR7R9+3Zt27ZN119/vRYvXqw333xT0lk8lsEQcNVVVwWVlZX9/+/r6wtKSkqC6upqj6tKrAcffDCYNWuW72UkjaRg/fr1/f+Px+NBUVFR8N3vfrf/smPHjgWhUCh4+umnPawwMT66n0EQBMuWLQsWL17sZT3Jcvjw4UBSUFdXFwTBiWOXnp4erFu3rr/m97//fSAp2Lx5s69lnrGP7mcQBMHnPve54O/+7u/8LSpJRo0aFfzkJz85q8fynD8DOn78uLZv366Kior+y1JTU1VRUaHNmzd7XFni7d27VyUlJZo4caK++MUvqrm52feSkqaxsVGtra0Djms4HFZZWdmwO66SVFtbq4KCAk2ZMkX33HOPjh496ntJZyQSiUiS8vLyJEnbt29Xb2/vgOM5depUlZaWDunj+dH9/KOf/exnys/P1/Tp01VVVaXu7m4fy0uIvr4+rV27Vl1dXSovLz+rx/KcG0b6UUeOHFFfX58KCwsHXF5YWKi3337b06oSr6ysTKtXr9aUKVPU0tKihx56SJ/97Ge1Z88e5eTk+F5ewrW2tkrSSY/rH68bLhYuXKibbrpJEyZMUENDg77+9a9r0aJF2rx5s9LS0nwvzywej+u+++7TNddco+nTp0s6cTwzMjI0cuTIAbVD+XiebD8l6bbbbtP48eNVUlKiXbt26atf/arq6+v17LPPelyt3e7du1VeXq6enh5lZ2dr/fr1mjZtmnbu3HnWjuU5H0Dni0WLFvX/e+bMmSorK9P48eP1i1/8QnfccYfHleFM3XLLLf3/njFjhmbOnKlJkyaptrZW8+bN87iywamsrNSePXuG/HOUp3Oq/bzrrrv6/z1jxgwVFxdr3rx5amho0KRJk872MgdtypQp2rlzpyKRiJ555hktW7ZMdXV1Z3UN5/yf4PLz85WWlvaxV2C0tbWpqKjI06qSb+TIkZo8ebL27dvneylJ8cdjd74dV0maOHGi8vPzh+SxXb58uV544QW99tprAz42paioSMePH9exY8cG1A/V43mq/TyZsrIySRpyxzMjI0OXXHKJZs+ererqas2aNUs/+MEPzuqxPOcDKCMjQ7Nnz1ZNTU3/ZfF4XDU1NSovL/e4suTq7OxUQ0ODiouLfS8lKSZMmKCioqIBxzUajWrr1q3D+rhK0oEDB3T06NEhdWyDINDy5cu1fv16vfrqq5owYcKA62fPnq309PQBx7O+vl7Nzc1D6niebj9PZufOnZI0pI7nycTjccVisbN7LBP6koYkWbt2bRAKhYLVq1cHb731VnDXXXcFI0eODFpbW30vLWH+/u//PqitrQ0aGxuDf//3fw8qKiqC/Pz84PDhw76XNmgdHR3Bjh07gh07dgSSgu9973vBjh07gj/84Q9BEATBI488EowcOTLYsGFDsGvXrmDx4sXBhAkTgvfff9/zym0+aT87OjqCL3/5y8HmzZuDxsbG4JVXXgkuv/zy4NJLLw16enp8L93ZPffcE4TD4aC2tjZoaWnp37q7u/tr7r777qC0tDR49dVXg23btgXl5eVBeXm5x1XbnW4/9+3bFzz88MPBtm3bgsbGxmDDhg3BxIkTgzlz5nheuc3Xvva1oK6uLmhsbAx27doVfO1rXwtSUlKCX/3qV0EQnL1jOSQCKAiC4Ec/+lFQWloaZGRkBFdddVWwZcsW30tKqJtvvjkoLi4OMjIygjFjxgQ333xzsG/fPt/LOiOvvfZaIOlj27Jly4IgOPFS7G9+85tBYWFhEAqFgnnz5gX19fV+Fz0In7Sf3d3dwfz584OLLrooSE9PD8aPHx/ceeedQ+6Xp5Ptn6TgySef7K95//33g7/5m78JRo0aFWRlZQU33nhj0NLS4m/Rg3C6/Wxubg7mzJkT5OXlBaFQKLjkkkuCf/iHfwgikYjfhRv99V//dTB+/PggIyMjuOiii4J58+b1h08QnL1jyccxAAC8OOefAwIADE8EEADACwIIAOAFAQQA8IIAAgB4QQABALwggAAAXhBAAAAvCCAAgBcEEADACwIIAOAFAQQA8OL/A34jxVp1buGzAAAAAElFTkSuQmCC",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "2 (tensor([[[-2.4291e+00, -4.3242e-01, -3.3549e-01,  ..., -5.2934e-01,\n",
            "          -6.4565e-01, -7.8135e-01],\n",
            "         [-2.4291e+00, -2.5796e-01, -1.4165e-01,  ..., -2.1919e-01,\n",
            "          -2.9672e-01, -4.5180e-01],\n",
            "         [-2.4291e+00, -5.4873e-01, -5.0996e-01,  ..., -2.5336e-02,\n",
            "          -1.0288e-01, -2.3857e-01],\n",
            "         ...,\n",
            "         [-2.4291e+00, -2.4291e+00, -2.4291e+00,  ..., -2.4291e+00,\n",
            "          -2.4291e+00, -2.4291e+00],\n",
            "         [-2.4291e+00, -2.4291e+00, -2.4291e+00,  ..., -2.4291e+00,\n",
            "          -2.4291e+00, -2.4291e+00],\n",
            "         [-2.4291e+00, -2.4291e+00, -2.4291e+00,  ..., -2.4291e+00,\n",
            "          -2.4291e+00, -2.4291e+00]],\n",
            "\n",
            "        [[-2.4183e+00, -9.7567e-02,  7.6703e-04,  ...,  7.6703e-04,\n",
            "          -1.1723e-01, -2.5490e-01],\n",
            "         [-2.4183e+00, -1.8900e-02,  1.3844e-01,  ...,  2.1710e-01,\n",
            "           1.1877e-01, -3.8567e-02],\n",
            "         [-2.4183e+00, -3.1390e-01, -2.1557e-01,  ...,  3.5477e-01,\n",
            "           2.7610e-01,  1.3844e-01],\n",
            "         ...,\n",
            "         [-2.4183e+00, -2.4183e+00, -2.4183e+00,  ..., -2.4183e+00,\n",
            "          -2.4183e+00, -2.4183e+00],\n",
            "         [-2.4183e+00, -2.4183e+00, -2.4183e+00,  ..., -2.4183e+00,\n",
            "          -2.4183e+00, -2.4183e+00],\n",
            "         [-2.4183e+00, -2.4183e+00, -2.4183e+00,  ..., -2.4183e+00,\n",
            "          -2.4183e+00, -2.4183e+00]],\n",
            "\n",
            "        [[-2.2214e+00,  5.4907e-01,  6.4662e-01,  ...,  7.2466e-01,\n",
            "           6.0760e-01,  4.7103e-01],\n",
            "         [-2.2214e+00,  6.0760e-01,  7.8319e-01,  ...,  7.8319e-01,\n",
            "           7.0515e-01,  5.4907e-01],\n",
            "         [-2.2214e+00,  3.5397e-01,  5.1005e-01,  ...,  8.8074e-01,\n",
            "           8.0270e-01,  6.6613e-01],\n",
            "         ...,\n",
            "         [-2.2214e+00, -2.2214e+00, -2.2214e+00,  ..., -2.2214e+00,\n",
            "          -2.2214e+00, -2.2214e+00],\n",
            "         [-2.2214e+00, -2.2214e+00, -2.2214e+00,  ..., -2.2214e+00,\n",
            "          -2.2214e+00, -2.2214e+00],\n",
            "         [-2.2214e+00, -2.2214e+00, -2.2214e+00,  ..., -2.2214e+00,\n",
            "          -2.2214e+00, -2.2214e+00]]]), 0) plane\n"
          ]
        },
        {
          "data": {
            "image/png": "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eumlKioq0oMPPqj8/HzNnDkzknMDAGKcdQBt375dN954Y+jjJUuWSJLmzp2rVatW6Uc/+pHa2tp0991369ixY7ruuuu0adMmJScnR27qr0hMjPNeG7UpLhytlvVpUZkCiE22/+zk/In6KIozxhjXQ3xZMBiU3+/Xv/72I6WkevvRNXZEvuf+l6b0djJ8gQACes/2V/L3WdReHuf9L+PnQyAQOOPz+v05XAEAfRgBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwwnoX3PmSkPj54Qkxek4ClvV7LOtHWdSytgcI1+N6gCjiRzcAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgRJ9dxdPe3i7FJ3iq7ejs9t442VvPvuYjy3qb9TrJlr0/taz/p1+/7bn29qsGWfW+OOdrltMAbtmu1umy+PEWa3gEBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnOizu+B64tvUHR/nqfZwoNVz31plWM1xsd97bbZVZ2m3RW2DZe80i9o9HXa9f/lvv7eq79iyzHNt468arXrP+uZMz7VTvrXEqrcs/6wAXtiuduuyXR4XQ3gEBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADgRZ4wxrof4smAwKL/fr6sXPaABvmRPnzPikhs897/9b66zmic70Xvtx1ad7XxiWf/Ou95r39hy0Kr3x01brOov3bvSc+3Fn26z6h3vS/Fcm5xicTElTblhlufa+X/zpFVvy1HQj9j+v7yn3XvtNRd5W192vgQCAaWnp5/2fh4BAQCcIIAAAE5YB9DmzZt18803Kz8/X3FxcVq/fn3Y/Xfeeafi4uLCjunTp0dqXgBAP2EdQG1tbRo/fryWL19+2prp06erqakpdLz00kvnNCQAoP+xfj+gsrIylZWVnbHG5/MpNze310MBAPq/qDwHVF1drezsbI0aNUoLFy7U0aNHT1vb2dmpYDAYdgAA+r+IB9D06dP14osvqqqqSj/96U9VU1OjsrIydXef+n0AKysr5ff7Q0dBQUGkRwIA9EERf0vu2267LfTfY8eO1bhx4zRixAhVV1drypQpJ9VXVFRoyZI/v1VyMBgkhADgAhD1l2EPHz5cWVlZ2rt37ynv9/l8Sk9PDzsAAP1f1APo4MGDOnr0qPLy8qL9pQAAMcT6n+COHz8e9mimsbFRO3bsUGZmpjIzM/Xoo49q9uzZys3NVUNDg370ox9p5MiRmjZtWkQHBwDENusA2r59u2688cbQx188fzN37lytWLFCO3fu1AsvvKBjx44pPz9fU6dO1d///d/L5/NZfZ2eIbeqJznNU+1/WVjkue/gBKsx9KlFbapda223WAr1i412vd/d8pr34tQeu+ZNFovmJLV/ethz7b6DAaveDXss6i1Ps3bXMs+1v3nXbhPg1Kl3WdV//7Yz/+oDYkenZf1nHVEZo0+wDqDJkyfrTPtLX3/99XMaCABwYWAXHADACQIIAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOBExN8PKFJuW3CJBnp8a4YDFju+Pu2ymyMv2Xttu11rvbPNe+27b//arnnjLu+1l2XZ9f6k2ap8f8N+78X77EaJpiNHvNfWfmRxMSV93HHcqn7zu1s8197+zb+26v3tGy63qsfJTv12m6dm+3Oi3fJnVizhERAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgRJ9dxdPdLv2nx+kOWqyqyEuzm6M94L32PYvtN5JU+4ed3os/abBrXjjIc+l1s+ZYta79n3+0qu/e94lVfdR42+z0ZxZ/rj5ttFg3JOnTg3b1R77W4bm2K8GqtTZvL/RcO6VkilXvGX8x0m6YGNVpUdtus7dHUqvt7p4YwiMgAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgRJ/dBdd1UEpI9VabkOK9b/3HdnPsOdjquXbbH2qtencdOeK9eNc2q94afYXn0vlldgvyJiQvtap/6t9fsqg+aNU7+VLvi88KRoy26r3nI4v9ey3ed7VJknrsypNTEj3XdsUft+q9q+kDz7UfrbfbSfhvVTmea8dfUWzVe8a3rvNcO9b7t69XbNa7HbdZHCcpwC44AAAiiwACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADjRZ1fxXBQ8ooHd3nZWxLcP8dz317/ZaTXH4X3/13txqs1CDkkKeC/d531diiTpyGHPpW/86r9Ztb7K+wYUSdJj7x3wXLtv+wtWvbsCv/dcm5KYadX7ksu8r0p6Y92vrXrrwH6r8oYa72ueOtot/lxJSkzN8lybnGH3Pdz6x92ea3/56itWvZ95odBz7ZzZt1v1/vbMb1nV5/m913ZZ/pgItrfZfUIM4REQAMAJqwCqrKzU1VdfrbS0NGVnZ2vmzJmqr68Pq+no6FB5ebkGDx6s1NRUzZ49Wy0tLREdGgAQ+6wCqKamRuXl5dq6daveeOMNdXV1aerUqWpr+/NDxMWLF2vjxo1au3atampqdOjQIc2aNSvigwMAYpvVc0CbNm0K+3jVqlXKzs5WXV2dJk2apEAgoH/5l3/R6tWrddNNN0mSVq5cqcsvv1xbt27VNddcE7nJAQAx7ZyeAwoEPn+yMzPz8ycm6+rq1NXVpdLS0lDN6NGjVVhYqNraUz+J2tnZqWAwGHYAAPq/XgdQT0+P7r33Xl177bUaM2aMJKm5uVlJSUnKyMgIq83JyVFzc/Mp+1RWVsrv94eOgoKC3o4EAIghvQ6g8vJy7d69W2vWrDmnASoqKhQIBELHgQPeX7ILAIhdvfo9oEWLFunVV1/V5s2bNXTo0NDtubm5OnHihI4dOxb2KKilpUW5ubmn7OXz+eTz+XozBgAghlk9AjLGaNGiRVq3bp3eeustFRUVhd0/YcIEJSYmqqqqKnRbfX299u/fr5KSkshMDADoF6weAZWXl2v16tXasGGD0tLSQs/r+P1+DRw4UH6/X/PmzdOSJUuUmZmp9PR03XPPPSopKeEVcACAMFYBtGLFCknS5MmTw25fuXKl7rzzTknSk08+qfj4eM2ePVudnZ2aNm2ann322YgMCwDoP+KMMcb1EF8WDAbl9/v1ry/uUkpKmqfP+WD0MM/9H9n4R6t5ultsdnZ1WfXWrre911b9s11vHfdcmfjXj1t1nnzrYqv64iu91/7VCKvWSrSo/dSutRosFnisesHiWkqqeuQWu2E+s9vvZmXU0LPX/H9ZRZdYtU5J8b4kbf/BJqveave2K1KSErK877uTpOy8PKv6S4aN8Vw75brpVr27fN737/30G8OtekdbIBBQenr6ae9nFxwAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgRJ9dxfMXg67VgHhvq+p8D6323L+6vcdqnq5P9nkvfn+LVW81vOu9tn7T2WvCtFrWW5j8Q6vyS66f6rm24oelZy/6kmJv25ok2a/iOWJRu8viUkrST77lfX2UJOljm5VQUWT7Bi5f2Zh/ZrZ/H7ZYfdXRbtc6JcVylASL3nZrfvwjxnuuDWxYYdU72ljFAwDokwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAnbzU7nzdZP31Gcx9oZHbs99+1qGGQ3yF6LJV9Hdtn1Pr7PojiKu91sVT9uVb7vo42ea/8p8SdWvYc+eKvn2v0dVq21r8F77eHGP9k17yu73Wz9p2X9nsaojCFJSk/2XttjsTdOktotd8fZ1H9m9z0J7P6t3SwxhEdAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBN9dhVPzrh0xSd4W8ZTkBnw3Pe6sWOs5tjySaL34oZ9Vr318e/t6vuKVMt6f5Pn0j/8n9VWrWu3fdtzbdalXpc7fe7Zp9d4rm19/j6r3oiAoOVuJRvHLVfxoFd4BAQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4QQABAJzos7vgUtPzlTAgwVPtW6+v99y34Hq7RWYpaUc817Yf+K1V7z7D9q8hWclW5f7L8jzXdnYctuq9+dcrPdeOvupmq96t/zzPoprdYYAtHgEBAJywCqDKykpdffXVSktLU3Z2tmbOnKn6+vqwmsmTJysuLi7sWLBgQUSHBgDEPqsAqqmpUXl5ubZu3ao33nhDXV1dmjp1qtra2sLq5s+fr6amptCxdOnSiA4NAIh9Vs8Bbdq0KezjVatWKTs7W3V1dZo0aVLo9pSUFOXm5kZmQgBAv3ROzwEFAp+/EVxmZmbY7T//+c+VlZWlMWPGqKKiQu3tp3+CtrOzU8FgMOwAAPR/vX4VXE9Pj+69915de+21GjPmz+8yescdd2jYsGHKz8/Xzp07dd9996m+vl6vvPLKKftUVlbq0Ucf7e0YAIAY1esAKi8v1+7du7Vly5aw2+++++7Qf48dO1Z5eXmaMmWKGhoaNGLEiJP6VFRUaMmSJaGPg8GgCgoKejsWACBG9CqAFi1apFdffVWbN2/W0KFDz1hbXFwsSdq7d+8pA8jn88nn8/VmDABADLMKIGOM7rnnHq1bt07V1dUqKio66+fs2LFDkpSX5/2XEQEA/Z9VAJWXl2v16tXasGGD0tLS1NzcLEny+/0aOHCgGhoatHr1av3lX/6lBg8erJ07d2rx4sWaNGmSxo0bF5UTAADEJqsAWrFihaTPf9n0y1auXKk777xTSUlJevPNN7Vs2TK1tbWpoKBAs2fP1gMPPBCxgQEA/UOcMca4HuLLgsGg/H6/fOPGKi7B2y64gYne97t9erzJbqCP9nmv7em2691XpHv7Pn8hsfhGyy/g/Xveddj77j1JSk4Z77m24xO781T9a95rcy17Z422q+/s8F7btM+u9/EY/XOLPi8QCCg9Pf2097MLDgDgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCi1+8HFG2dwS4pvsdTbUf7Pu+Nmw/2bqD+LG2QVXnXx5bfw1Rv11GSFJ9o1brjk33ei/dZzj3I+3qdQaNPfquRM/lMfqv6jo9avBd3sVoHsYFHQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwIk4Y4xxPcSXBYNB+f12e7IAAH1PIBBQenr6ae/nERAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATVgG0YsUKjRs3Tunp6UpPT1dJSYlee+210P0dHR0qLy/X4MGDlZqaqtmzZ6ulpSXiQwMAYp9VAA0dOlSPPfaY6urqtH37dt10002aMWOG3n//fUnS4sWLtXHjRq1du1Y1NTU6dOiQZs2aFZXBAQAxzpyjQYMGmeeff94cO3bMJCYmmrVr14bu+/DDD40kU1tb67lfIBAwkjg4ODg4YvwIBAJn/Hnf6+eAuru7tWbNGrW1tamkpER1dXXq6upSaWlpqGb06NEqLCxUbW3taft0dnYqGAyGHQCA/s86gHbt2qXU1FT5fD4tWLBA69at0xVXXKHm5mYlJSUpIyMjrD4nJ0fNzc2n7VdZWSm/3x86CgoKrE8CABB7rANo1KhR2rFjh7Zt26aFCxdq7ty5+uCDD3o9QEVFhQKBQOg4cOBAr3sBAGLHANtPSEpK0siRIyVJEyZM0O9//3s99dRTmjNnjk6cOKFjx46FPQpqaWlRbm7uafv5fD75fD77yQEAMe2cfw+op6dHnZ2dmjBhghITE1VVVRW6r76+Xvv371dJScm5fhkAQD9j9QiooqJCZWVlKiwsVGtrq1avXq3q6mq9/vrr8vv9mjdvnpYsWaLMzEylp6frnnvuUUlJia655ppozQ8AiFFWAXT48GF997vfVVNTk/x+v8aNG6fXX39d3/jGNyRJTz75pOLj4zV79mx1dnZq2rRpevbZZ6MyOAAgtsUZY4zrIb4sGAzK7/e7HgMAcI4CgYDS09NPez+74AAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4QQABAJwggAAATvS5AOpjixkAAL10tp/nfS6AWltbXY8AAIiAs/0873O74Hp6enTo0CGlpaUpLi4udHswGFRBQYEOHDhwxt1CsY7z7D8uhHOUOM/+JhLnaYxRa2ur8vPzFR9/+sc51m9IF23x8fEaOnToae9PT0/v1xf/C5xn/3EhnKPEefY353qeXpZK97l/ggMAXBgIIACAEzETQD6fTw8//LB8Pp/rUaKK8+w/LoRzlDjP/uZ8nmefexECAODCEDOPgAAA/QsBBABwggACADhBAAEAnIiZAFq+fLkuueQSJScnq7i4WL/73e9cjxRRjzzyiOLi4sKO0aNHux7rnGzevFk333yz8vPzFRcXp/Xr14fdb4zRQw89pLy8PA0cOFClpaXas2ePm2HPwdnO88477zzp2k6fPt3NsL1UWVmpq6++WmlpacrOztbMmTNVX18fVtPR0aHy8nINHjxYqampmj17tlpaWhxN3DteznPy5MknXc8FCxY4mrh3VqxYoXHjxoV+2bSkpESvvfZa6P7zdS1jIoBefvllLVmyRA8//LDeffddjR8/XtOmTdPhw4ddjxZRV155pZqamkLHli1bXI90Ttra2jR+/HgtX778lPcvXbpUTz/9tJ577jlt27ZNF110kaZNm6aOjo7zPOm5Odt5StL06dPDru1LL710Hic8dzU1NSovL9fWrVv1xhtvqKurS1OnTlVbW1uoZvHixdq4caPWrl2rmpoaHTp0SLNmzXI4tT0v5ylJ8+fPD7ueS5cudTRx7wwdOlSPPfaY6urqtH37dt10002aMWOG3n//fUnn8VqaGDBx4kRTXl4e+ri7u9vk5+ebyspKh1NF1sMPP2zGjx/veoyokWTWrVsX+rinp8fk5uaaxx9/PHTbsWPHjM/nMy+99JKDCSPjq+dpjDFz5841M2bMcDJPtBw+fNhIMjU1NcaYz69dYmKiWbt2bajmww8/NJJMbW2tqzHP2VfP0xhjbrjhBvP973/f3VBRMmjQIPP888+f12vZ5x8BnThxQnV1dSotLQ3dFh8fr9LSUtXW1jqcLPL27Nmj/Px8DR8+XN/5zne0f/9+1yNFTWNjo5qbm8Ouq9/vV3Fxcb+7rpJUXV2t7OxsjRo1SgsXLtTRo0ddj3ROAoGAJCkzM1OSVFdXp66urrDrOXr0aBUWFsb09fzqeX7h5z//ubKysjRmzBhVVFSovb3dxXgR0d3drTVr1qitrU0lJSXn9Vr2uWWkX3XkyBF1d3crJycn7PacnBx99NFHjqaKvOLiYq1atUqjRo1SU1OTHn30UV1//fXavXu30tLSXI8Xcc3NzZJ0yuv6xX39xfTp0zVr1iwVFRWpoaFBP/7xj1VWVqba2lolJCS4Hs9aT0+P7r33Xl177bUaM2aMpM+vZ1JSkjIyMsJqY/l6nuo8JemOO+7QsGHDlJ+fr507d+q+++5TfX29XnnlFYfT2tu1a5dKSkrU0dGh1NRUrVu3TldccYV27Nhx3q5lnw+gC0VZWVnov8eNG6fi4mINGzZMv/jFLzRv3jyHk+Fc3XbbbaH/Hjt2rMaNG6cRI0aourpaU6ZMcThZ75SXl2v37t0x/xzl2ZzuPO++++7Qf48dO1Z5eXmaMmWKGhoaNGLEiPM9Zq+NGjVKO3bsUCAQ0C9/+UvNnTtXNTU153WGPv9PcFlZWUpISDjpFRgtLS3Kzc11NFX0ZWRk6LLLLtPevXtdjxIVX1y7C+26StLw4cOVlZUVk9d20aJFevXVV/X222+HvW1Kbm6uTpw4oWPHjoXVx+r1PN15nkpxcbEkxdz1TEpK0siRIzVhwgRVVlZq/Pjxeuqpp87rtezzAZSUlKQJEyaoqqoqdFtPT4+qqqpUUlLicLLoOn78uBoaGpSXl+d6lKgoKipSbm5u2HUNBoPatm1bv76uknTw4EEdPXo0pq6tMUaLFi3SunXr9NZbb6moqCjs/gkTJigxMTHsetbX12v//v0xdT3Pdp6nsmPHDkmKqet5Kj09Pers7Dy/1zKiL2mIkjVr1hifz2dWrVplPvjgA3P33XebjIwM09zc7Hq0iPm7v/s7U11dbRobG80777xjSktLTVZWljl8+LDr0XqttbXVvPfee+a9994zkswTTzxh3nvvPfOnP/3JGGPMY489ZjIyMsyGDRvMzp07zYwZM0xRUZH57LPPHE9u50zn2draan7wgx+Y2tpa09jYaN58803z9a9/3Vx66aWmo6PD9eieLVy40Pj9flNdXW2amppCR3t7e6hmwYIFprCw0Lz11ltm+/btpqSkxJSUlDic2t7ZznPv3r3mJz/5idm+fbtpbGw0GzZsMMOHDzeTJk1yPLmd+++/39TU1JjGxkazc+dOc//995u4uDjzm9/8xhhz/q5lTASQMcY888wzprCw0CQlJZmJEyearVu3uh4poubMmWPy8vJMUlKSufjii82cOXPM3r17XY91Tt5++20j6aRj7ty5xpjPX4r94IMPmpycHOPz+cyUKVNMfX2926F74Uzn2d7ebqZOnWqGDBliEhMTzbBhw8z8+fNj7i9Ppzo/SWblypWhms8++8z87d/+rRk0aJBJSUkxt9xyi2lqanI3dC+c7Tz3799vJk2aZDIzM43P5zMjR440P/zhD00gEHA7uKXvfe97ZtiwYSYpKckMGTLETJkyJRQ+xpy/a8nbMQAAnOjzzwEBAPonAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADjx/wAcAw8Ln6VQPgAAAABJRU5ErkJggg==",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "3 (tensor([[[-0.8007, -0.6069, -0.6263,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-1.4211, -1.2078, -1.4986,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-1.2078, -1.3823, -1.2078,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         ...,\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291]],\n",
            "\n",
            "        [[-0.3729, -0.1172, -0.2746,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-1.2382, -0.9629, -1.3366,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-1.1399, -1.2776, -1.1596,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         ...,\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183]],\n",
            "\n",
            "        [[-0.3289, -0.0948, -0.2118,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-1.1288, -0.8947, -1.2459,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-0.9337, -1.1288, -1.0508,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         ...,\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214]]]), 0) plane\n"
          ]
        },
        {
          "data": {
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAaAAAAGdCAYAAABU0qcqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAm2ElEQVR4nO3dfXiU9Z3v8U8SJxNjkoEY8yQBAyoUeegpSsxqKUoWSLsuKO36dK5F6+LRBs8qutXs1sfunlg9W6kuwullF+o5RRRXoLpWq9GEYxuoRClaNYU0XUCSUKiZCYkZZjP3+cNjahTk/oYZfnl4v67rvi6Y+eaX7z33zHxyZ2a+SfE8zxMAACdYqusGAAAjEwEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwImTXDfwafF4XPv27VN2drZSUlJctwMAMPI8T52dnSouLlZq6tHPcwZdAO3bt08lJSWu2wAAHKc9e/ZozJgxR70+aQG0YsUKPfjgg2pra9P06dP1yCOPaObMmcf8uuzs7GS1JEna2hI21acH/NeOyrD1kvtaq//igwdti//FFN+lvca+02zl6jHUxo1rGw6P+ffNll6sa1tvQ8CPWBLX7un1X9sZiegLZ5Qc8/k8KQH05JNPatmyZVq1apXKysq0fPlyzZs3T01NTcrPz//cr032r92ycnJM9UHDM1yO8Yk855RD/os/jBoX97+fyQ6gdEMtAQQMXDIDKN0QQB871vN5Ut6E8P3vf19LlizRtddeq8mTJ2vVqlXKzMzUv/7rvybj2wEAhqCEB9Dhw4fV2NioioqKP32T1FRVVFSooaHhM/XRaFSRSKTfBgAY/hIeQAcOHFBvb68KCgr6XV5QUKC2trbP1NfU1CgUCvVtvAEBAEYG558Dqq6uVjgc7tv27NnjuiUAwAmQ8Dch5OXlKS0tTe3t7f0ub29vV2Fh4Wfqg8GggsFgotsAAAxyCT8DSk9P14wZM1RbW9t3WTweV21trcrLyxP97QAAQ1RS3oa9bNkyLV68WOeee65mzpyp5cuXq6urS9dee20yvh0AYAhKSgBdfvnl+sMf/qC77rpLbW1t+uIXv6gXXnjhM29MAACMXCme53mum/ikSCSiUCiUtPXf7bDtruUzmqONbYduu8d/8T8/aFv86dpj13xs0fm2tY26DbWWqQmSZLnJ+fAnhqIBfP7TN8uHrXsNjUQiERXkhhQOh5XzOR+Kd/4uOADAyEQAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcSMosuMEszRi5AcOsCvPfYz9rkqHYMtBG0j9/z3/t1zaYlu6xzCeSlJmkWhxZMke3MM7o+CXz+FiZng4NBz/NZy1nQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwIkRNwsu1TjMylIftC0tlc61foV/DRv91/57vWnp+KKvmOots6+YNXb8uA0HN+vxGYqPH79nNpwBAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE6MuFE8acbIDRhmW5jHYJTl+q+98i9taz/xU/+19682LZ250DaKp9Nww2SbVpZihtqAcW1gMBgs43Usffit5QwIAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4MeJmwaVaBysZItqc5iFD7V9dZ1v7zHP91/50q23tZxpN5ZnfmGFb34D5bsDQxRkQAMCJhAfQPffco5SUlH7bpEmTEv1tAABDXFJ+BXfOOefo5Zdf/tM3OWnE/aYPAHAMSUmGk046SYWFhclYGgAwTCTlNaCdO3equLhY48eP19VXX63du3cftTYajSoSifTbAADDX8IDqKysTGvWrNELL7yglStXqqWlRV/+8pfV2dl5xPqamhqFQqG+raSkJNEtAQAGoYQHUGVlpb7xjW9o2rRpmjdvnp5//nl1dHToqaeeOmJ9dXW1wuFw37Znz55EtwQAGISS/u6AUaNG6eyzz9auXbuOeH0wGFQwGEx2GwCAQSbpnwM6dOiQmpubVVRUlOxvBQAYQhIeQLfddpvq6+v1+9//Xr/85S916aWXKi0tTVdeeWWivxUAYAhL+K/g9u7dqyuvvFIHDx7UaaedpgsvvFBbtmzRaaedluhvNTDGyI1Zintta8syFui192xrX2g447zvOdvaD/zYVJ4WNoziybK1YrkNTcfSiJFAgF3CA2jdunWJXhIAMAwxCw4A4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwIul/jmEkSeassYxgpu0LLl3iv9b7G9va315sq3+rw3/t1FGmpS3j96zz2izHM5ljAIHBwHIf91vLGRAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgxIgbxROI2+qDhvkTacZZL5ZRLxlfPNe2uMU9dxnr77PVjzXcMN22pdOME4osLIczmWOYJEb3jGTWMU/JYnnq9FvLGRAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHBixM2CyzRGbsBQb01z0+i4ognG1Q3W/9hW/9+utdVfda//2qXLbGsvmua79I+2lZVtGPCWmuQf5XoNg7isMwmTqTeJg8zSRsiAPMsMtmTeDS3H0m8tZ0AAACcIIACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMCJFM/zPNdNfFIkElEoFEra+q//zra7ozP914aybb0EDTO7sq3zvZb+k//an66wrX322abyt2rrfdfeautEyx/9ke/ayTd+07g6MLz1GOa7WWbSRSIRFeWGFA6HlZOTc9Q6zoAAAE6YA2jz5s265JJLVFxcrJSUFG3cuLHf9Z7n6a677lJRUZFOPvlkVVRUaOfOnYnqFwAwTJgDqKurS9OnT9eKFUf+tc0DDzyghx9+WKtWrdLWrVt1yimnaN68eerp6TnuZgEAw4f57wFVVlaqsrLyiNd5nqfly5frO9/5jhYsWCBJevzxx1VQUKCNGzfqiiuuOL5uAQDDRkJfA2ppaVFbW5sqKir6LguFQiorK1NDQ8MRvyYajSoSifTbAADDX0IDqK2tTZJUUFDQ7/KCgoK+6z6tpqZGoVCobyspKUlkSwCAQcr5u+Cqq6sVDof7tj179rhuCQBwAiQ0gAoLCyVJ7e3t/S5vb2/vu+7TgsGgcnJy+m0AgOEvoQFUWlqqwsJC1dbW9l0WiUS0detWlZeXJ/JbAQCGOPO74A4dOqRdu3b1/b+lpUXbt29Xbm6uxo4dq5tvvln/+I//qLPOOkulpaW68847VVxcrIULFyaybwDAEGcexVNXV6eLLrroM5cvXrxYa9asked5uvvuu/XDH/5QHR0duvDCC/Xoo4/qbJ/jW5I9iufffvWfpvruD/7ouzb8x0OmtX/3fqvv2n+49c9Ma+fu7fJd+99LskxrrzJVSzFD7Wjj2v6PjqTfvW9bvLTYVg8MMZZPZ1qmgUUiEeWGjj2Kx3wGNHv2bH1eZqWkpOi+++7TfffdZ10aADCCOH8XHABgZCKAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOmEfxDHXTJ6WZ6mPR03zXRjv910pS9nuZvmv/7xvtxy76hGfvud137Y9MK9vNN9T+NGldSJr3NVv9b99MTh/AIJFhqD3Q6b+202ctZ0AAACcIIACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEyNuFE8saqtPM0R0XoFt7QVF/r/gQNi29lvnzfJf/OyPTWufbmtF/yvovzZgPD4mO7fb6r+11H/to/9iWxsj1u9bbfU9PYd9104qTbet3eu/dvduz3ftoUP+ajkDAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4QQABAJwggAAATqR4nud/wM8JEIlEFAqFkrb+jn223Y3G/NcGAsZmDvgv3fZ6o2npsaX+b8P3Xm8wrV338HJT/YH33/Bde4ZpZekmQ+25xrVNIh/Y6rNHJaUNJMbuvV2m+lUrH/Fdu39/j2ntpt/+znftXXfeblr7/AvP8V3b+Gv/t0lXV0R/MadY4XBYOTk5R63jDAgA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABw4iTXDZxoL238qam+O5btvzhw9JETR1KUmea7tvbl501r//qdrb5r3/n1v5vWPv2rd5jqs08v9V27+Vf/Zlr7cUPtl0wrS3cZahfc/ze2xf/paVv9CPD8y1tM9XdW+78fvrGt3rT2/DmVpvrJk87yXVt+7oWmtWeW+b/nbtv2qmntGWX+R/G07H3Pd+2H3Yd81XEGBABwggACADhhDqDNmzfrkksuUXFxsVJSUrRx48Z+119zzTVKSUnpt82fPz9R/QIAhglzAHV1dWn69OlasWLFUWvmz5+v1tbWvu2JJ544riYBAMOP+U0IlZWVqqz8/BfogsGgCgsLB9wUAGD4S8prQHV1dcrPz9fEiRN144036uDBg0etjUajikQi/TYAwPCX8ACaP3++Hn/8cdXW1up73/ue6uvrVVlZqd7e3iPW19TUKBQK9W0lJSWJbgkAMAgl/HNAV1xxRd+/p06dqmnTpmnChAmqq6vTnDlzPlNfXV2tZcuW9f0/EokQQgAwAiT9bdjjx49XXl6edu3adcTrg8GgcnJy+m0AgOEv6QG0d+9eHTx4UEVFRcn+VgCAIcT8K7hDhw71O5tpaWnR9u3blZubq9zcXN17771atGiRCgsL1dzcrG9/+9s688wzNW/evIQ2DgAY2swBtG3bNl100UV9///49ZvFixdr5cqV2rFjh3784x+ro6NDxcXFmjt3rr773e8qGAwmruvjsP+3tab67y1/OEmdSApO8F1abpgHJUnvtzZbu/G/9vP3275g9FTfpdP/i20GV3j3Xt+1bxx8y7T2QkPtnP9hm2H30ELbzK64/P9q+vQxY0xr5xUV+K5d8dha09pLl1xtqk+WZVd+zVT/z//b9tnF95t3+6694x++Z1p78hen+K6t+ad7TWt3+5zZJklvtfjfx9jhw77qzAE0e/ZseZ531OtffPFF65IAgBGIWXAAACcIIACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwn/e0CD3awvZZnqnzk95Lt25/thWzNR//Pa3nzd/8wmSer58ICtlyT67j98w3ftGaefYVr7F69t9V276eWoae3AB92+a2v3+59JJ0nTZl5sqh8sMkbnmeoDwWzftbFop21tQ+1f/eWFprVfW7/aVP987Wu+a//P0+tNa09+w//MyKDlRpEUPtDqu7b8vLN81/b09Oj5p45dxxkQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4MSIG8WzdeuvTfWTx/ofg/E7w1gLSeqN+q//q4VzTWv/avMv/ffRbRshdOGsSab6stP9j2/57U7b8cnq3u+79uvnTzat/fVv+B8h1NzcYlr7fz60ylT/zu9to36S5eorrjXV/9c5X/Jde9HXr7S249ueVtv9Kj8331T/9bn+x9R80H6Oae2de/2P1Zr/lT8zrT35LP/nIK1/fNN3bSx62FcdZ0AAACcIIACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMCJFM/zPNdNfFIkElEoFEra+pPGjzXVZ8Uzfdd2Z+aa1o72vue7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",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "4 (tensor([[[-2.4291, -2.4291,  0.6725,  ...,  0.9633,  0.9633,  0.9051],\n",
            "         [-2.4291, -2.4291,  0.6919,  ...,  0.9827,  0.9633,  0.9439],\n",
            "         [-2.4291, -2.4291,  0.7113,  ...,  0.9827,  0.9827,  0.9633],\n",
            "         ...,\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291]],\n",
            "\n",
            "        [[-2.4183, -2.4183,  0.8661,  ...,  1.0234,  1.0234,  0.9841],\n",
            "         [-2.4183, -2.4183,  0.8858,  ...,  1.0431,  1.0234,  0.9841],\n",
            "         [-2.4183, -2.4183,  0.9054,  ...,  1.0431,  1.0431,  1.0038],\n",
            "         ...,\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183]],\n",
            "\n",
            "        [[-2.2214, -2.2214,  1.3685,  ...,  1.4075,  1.4075,  1.4075],\n",
            "         [-2.2214, -2.2214,  1.3295,  ...,  1.4270,  1.4270,  1.4270],\n",
            "         [-2.2214, -2.2214,  1.3295,  ...,  1.4465,  1.4465,  1.4465],\n",
            "         ...,\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214]]]), 0) plane\n"
          ]
        },
        {
          "data": {
            "image/png": "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",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "5 (tensor([[[ 2.4559,  2.4365,  2.4365,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [ 1.8356,  1.9325,  1.8938,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [ 0.6531,  0.8276,  1.0990,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         ...,\n",
            "         [ 1.0602,  0.8664,  0.7888,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [ 2.1264,  2.0682,  2.0101,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [ 2.4559,  2.4559,  2.4559,  ..., -2.4291, -2.4291, -2.4291]],\n",
            "\n",
            "        [[ 2.5968,  2.5968,  2.5575,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [ 2.1838,  2.2231,  2.1445,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [ 1.2201,  1.2988,  1.5348,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         ...,\n",
            "         [ 1.4168,  1.2398,  1.1611,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [ 2.3411,  2.2821,  2.2231,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [ 2.5575,  2.5575,  2.5575,  ..., -2.4183, -2.4183, -2.4183]],\n",
            "\n",
            "        [[ 2.7147,  2.6952,  2.7537,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [ 2.6952,  2.6757,  2.5586,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [ 2.3830,  2.3635,  2.3440,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         ...,\n",
            "         [ 2.3440,  2.1294,  2.0709,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [ 2.7147,  2.6562,  2.5976,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [ 2.6562,  2.6562,  2.6562,  ..., -2.2214, -2.2214, -2.2214]]]), 0) plane\n"
          ]
        },
        {
          "data": {
            "image/png": "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Hj9+vKqqqjRy5Ej96le/0s033+yxM/TVddddl/j3xRdfrPHjx2v06NHatGmTpk6d6rGz3qmtrdX777+f8e9RnsyJtvPWW29N/Pviiy9WWVmZpk6dqt27d2v06NGnus1eGzNmjLZv365IJKJf//rXmj9/vhoaGk5pD2n/ElxRUZFycnK+cARGW1ubSktLPXUVvCFDhuj888/Xrl27fLcSiL/vu9Ntv0rSqFGjVFRUlJH7duHChXrppZf05ptv9vjalNLSUh07dkyHDx/uUZ+p+/NE23k8VVVVkpRx+zMvL0/nnnuuJk6cqLq6Ok2YMEGPP/74Kd2XaR9AeXl5mjhxourr6xPnxeNx1dfXq7q62mNnwTpy5Ih2796tsrIy360EorKyUqWlpT32azQa1datW/v1fpWkffv26dChQxm1b51zWrhwodatW6c33nhDlZWVPS6fOHGicnNze+zP5uZm7dmzJ6P258m283i2b98uSRm1P48nHo8rFoud2n2Z0kMaArJmzRoXCoXcqlWr3IcffuhuvfVWN2TIENfa2uq7tZT5wQ9+4DZt2uRaWlrcb37zG1dTU+OKiorcwYMHfbfWa+3t7e7dd9917777rpPkHnnkEffuu++6P/zhD8455x566CE3ZMgQt2HDBrdjxw43a9YsV1lZ6T799FPPndt82Xa2t7e7O+64wzU2NrqWlhb3+uuvu69+9avuvPPOc52dnb5bT9qCBQtcOBx2mzZtcgcOHEicjh49mqi57bbb3IgRI9wbb7zhtm3b5qqrq111dbXHru1Otp27du1yDz74oNu2bZtraWlxGzZscKNGjXKTJ0/23LnN3Xff7RoaGlxLS4vbsWOHu/vuu11WVpZ79dVXnXOnbl9mRAA559zPfvYzN2LECJeXl+cuvfRSt2XLFt8tpdS8efNcWVmZy8vLc2effbabN2+e27Vrl++2+uTNN990kr5wmj9/vnPus0Ox7733XldSUuJCoZCbOnWqa25u9tt0L3zZdh49etRNmzbNDRs2zOXm5rqRI0e6W265JeMePB1v+yS5p59+OlHz6aefuu9+97vurLPOcoMGDXLXXHONO3DggL+me+Fk27lnzx43efJkV1hY6EKhkDv33HPdD3/4QxeJRPw2bvSd73zHjRw50uXl5blhw4a5qVOnJsLHuVO3L/k6BgCAF2n/HhAAoH8igAAAXhBAAAAvCCAAgBcEEADACwIIAOAFAQQA8IIAAgB4QQABALwggAAAXhBAAAAvCCAAgBf/H729W8Dt2atXAAAAAElFTkSuQmCC",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "6 (tensor([[[-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-1.7894, -1.7312, -1.7118,  ..., -1.7506, -2.4291, -2.4291],\n",
            "         [-1.7700, -1.7312, -1.6924,  ..., -1.7312, -2.4291, -2.4291],\n",
            "         ...,\n",
            "         [-1.5761, -1.5180, -1.4792,  ..., -1.5567, -2.4291, -2.4291],\n",
            "         [-1.5955, -1.5374, -1.4986,  ..., -1.5567, -2.4291, -2.4291],\n",
            "         [-1.5955, -1.5567, -1.4986,  ..., -1.5955, -2.4291, -2.4291]],\n",
            "\n",
            "        [[-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-1.4546, -1.4349, -1.3562,  ..., -1.4349, -2.4183, -2.4183],\n",
            "         [-1.4546, -1.3956, -1.3169,  ..., -1.4152, -2.4183, -2.4183],\n",
            "         ...,\n",
            "         [-1.2382, -1.1792, -1.1399,  ..., -1.1989, -2.4183, -2.4183],\n",
            "         [-1.2579, -1.1989, -1.1596,  ..., -1.2186, -2.4183, -2.4183],\n",
            "         [-1.2579, -1.2186, -1.1596,  ..., -1.2579, -2.4183, -2.4183]],\n",
            "\n",
            "        [[-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-0.7386, -0.6606, -0.6215,  ..., -0.6801, -2.2214, -2.2214],\n",
            "         [-0.7191, -0.6411, -0.6020,  ..., -0.6606, -2.2214, -2.2214],\n",
            "         ...,\n",
            "         [-0.5825, -0.5435, -0.4655,  ..., -0.5240, -2.2214, -2.2214],\n",
            "         [-0.6020, -0.5630, -0.4850,  ..., -0.5435, -2.2214, -2.2214],\n",
            "         [-0.6020, -0.5435, -0.4850,  ..., -0.5630, -2.2214, -2.2214]]]), 0) plane\n"
          ]
        },
        {
          "data": {
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAaAAAAGdCAYAAABU0qcqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAhkUlEQVR4nO3dfXRU9b3v8c+AyQAmMxBCnkoC4UGi8tBKIaYiRUmBeJYXJHbhQ2+hpVpoYAnUqml9bs8KxXMUtYj3Hnug3iti6REongUWgwnLGmiJpPhQcoAbCkoSCm1mQjBDFtn3D5dzOgqyf8kMv8zwfq211yIz3/nOd7tlPuzMnt94HMdxBADARdbL9gAAgEsTAQQAsIIAAgBYQQABAKwggAAAVhBAAAArCCAAgBUEEADAistsD/BZnZ2dOnbsmFJTU+XxeGyPAwAw5DiOWltblZOTo169zn+e0+MC6NixY8rNzbU9BgCgm44eParBgwef9/6YBdCqVav0xBNPqKmpSePGjdOzzz6riRMnXvBxqampkj4Z3OfzxWo8AIABv99v/JhPX8/PJyYB9Morr2jZsmV6/vnnVVhYqJUrV2r69Omqr69XRkbGFz7201+7+Xw+AggA4tiF3kbxxGIx0sLCQk2YMEG/+MUvJH3yvk5ubq4WL16sBx544AsfGwwG5ff7FQgECCAA6CG68p78hV7Ho34V3JkzZ1RbW6vi4uL/fpJevVRcXKyamprP1YdCIQWDwYgNAJD4oh5AJ06c0NmzZ5WZmRlxe2Zmppqamj5XX1FRIb/fH964AAEALg3WPwdUXl6uQCAQ3o4ePWp7JADARRD1ixDS09PVu3dvNTc3R9ze3NysrKysz9V7vV55vd5ojwEA6OGifgaUnJys8ePHq7KyMnxbZ2enKisrVVRUFO2nAwDEqZhchr1s2TLNnTtXX/3qVzVx4kStXLlSbW1t+s53vhOLpwMAxKGYBNCcOXP017/+VQ8//LCampr05S9/Wdu2bfvchQkAgEtXTD4H1B18DggAep64+BwQAABuEEAAACsIIACAFQQQAMAKAggAYAUBBACwggACAFhBAAEArCCAAABWEEAAACsIIACAFQQQAMAKAggAYAUBBACwggACAFhBAAEArCCAAABWEEAAACsIIACAFQQQAMAKAggAYAUBBACw4jLbAwDApeRQe7tR/cv/d4Pr2sKiAqPe37h6glF9tHEGBACwggACAFhBAAEArCCAAABWEEAAACsIIACAFQQQAMAKAggAYAUBBACwggACAFjBUjwA0E1/M6j97p2LjXp/dOQj17U33bzaqLdtnAEBAKwggAAAVhBAAAArCCAAgBUEEADACgIIAGAFAQQAsIIAAgBYQQABAKwggAAAVhBAAAArWAsOALrpru//wHVtTfU7Rr3T09Nc147MHGLU2zbOgAAAVkQ9gB599FF5PJ6IraCgINpPAwCIczH5FdzVV1+tN95447+f5DJ+0wcAiBSTZLjsssuUlZUVi9YAgAQRk/eADhw4oJycHA0bNkx33nmnjhw5ct7aUCikYDAYsQEAEl/UA6iwsFBr167Vtm3btHr1ajU0NOj6669Xa2vrOesrKirk9/vDW25ubrRHAgD0QB7HcZxYPkFLS4uGDBmiJ598UvPnz//c/aFQSKFQKPxzMBhUbm6uAoGAfD5fLEcDgKgoNbgMe8t/7DbqbXIZdv3+7Ua9Uw1qPR6PUW9JF3wdj/nVAf3799cVV1yhgwcPnvN+r9crr9cb6zEAAD1MzD8HdOrUKR06dEjZ2dmxfioAQByJegDde++9qq6u1uHDh/X222/rlltuUe/evXX77bdH+6kAAHEs6r+C+/DDD3X77bfr5MmTGjRokCZNmqRdu3Zp0KBB0X4qAIiJtxrfN6rfvvV3rms7Th426t1v8CzXtSbv6fQEUQ+g9evXR7slACABsRYcAMAKAggAYAUBBACwggACAFhBAAEArCCAAABWEEAAACsIIACAFQQQAMAKAggAYEXMv44BAOLNip88Y1TvC1245lMdXrNvBvhSXpJRfTzhDAgAYAUBBACwggACAFhBAAEArCCAAABWEEAAACsIIACAFQQQAMAKAggAYAUBBACwgqV4ACS8Vf/xn0b1p/9m1n/GpK+7rq15t86o98irc82GiSOcAQEArCCAAABWEEAAACsIIACAFQQQAMAKAggAYAUBBACwggACAFhBAAEArCCAAABWEEAAACtYCw6IA0+/sMF17d/+ZraQ2b33fd91bapR59hqNaj9zbpNRr0HZw41qu99+iPXtafbO416FxRcY1QfTzgDAgBYQQABAKwggAAAVhBAAAArCCAAgBUEEADACgIIAGAFAQQAsIIAAgBYQQABAKwggAAAVrAWHBAH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KykpKeJ41tfX68iRI3F1PC+0n+dSV1cnSXF1PM+ls7NToVDo4h7LqF7SECPr1693vF6vs3btWueDDz5w7r77bqd///5OU1OT7dGi5oc//KFTVVXlNDQ0OL///e+d4uJiJz093Tl+/Ljt0bqstbXV2bt3r7N3715HkvPkk086e/fudf7yl784juM4y5cvd/r37+9s3rzZ2bdvnzNz5kwnPz/f+fjjjy1PbuaL9rO1tdW59957nZqaGqehocF54403nGuuucYZOXKk097ebnt01xYuXOj4/X6nqqrKaWxsDG+nT58O1yxYsMDJy8tzduzY4ezZs8cpKipyioqKLE5t7kL7efDgQefxxx939uzZ4zQ0NDibN292hg0b5kyePNny5GYeeOABp7q62mloaHD27dvnPPDAA47H43F+97vfOY5z8Y5lXASQ4zjOs88+6+Tl5TnJycnOxIkTnV27dtkeKarmzJnjZGdnO8nJyc6XvvQlZ86cOc7Bgwdtj9Utb775piPpc9vcuXMdx/nkUuyHHnrIyczMdLxerzN16lSnvr7e7tBd8EX7efr0aWfatGnOoEGDnKSkJGfIkCHOXXfdFXf/eDrX/kly1qxZE675+OOPnR/84AfOgAEDnH79+jm33HKL09jYaG/oLrjQfh45csSZPHmyk5aW5ni9XmfEiBHOj370IycQCNgd3NB3v/tdZ8iQIU5ycrIzaNAgZ+rUqeHwcZyLdyz5OgYAgBU9/j0gAEBiIoAAAFYQQAAAKwggAIAVBBAAwAoCCABgBQEEALCCAAIAWEEAAQCsIIAAAFYQQAAAKwggAIAV/x90L7P3lZObFgAAAABJRU5ErkJggg==",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "7 (tensor([[[ 2.4947,  2.4753,  2.4365,  ...,  1.4867,  1.5061, -2.4291],\n",
            "         [ 2.4947,  2.4753,  2.4365,  ...,  1.5255,  1.5448, -2.4291],\n",
            "         [ 2.4947,  2.4753,  2.4365,  ...,  1.4285,  1.6224, -2.4291],\n",
            "         ...,\n",
            "         [ 2.4753,  2.4753,  2.4947,  ...,  2.4365,  2.4365, -2.4291],\n",
            "         [ 2.4753,  2.4559,  2.4365,  ...,  2.3978,  2.3784, -2.4291],\n",
            "         [ 2.2427,  2.1264,  2.0295,  ...,  1.9519,  1.9132, -2.4291]],\n",
            "\n",
            "        [[ 2.5771,  2.5575,  2.5181,  ...,  1.8298,  1.8298, -2.4183],\n",
            "         [ 2.5771,  2.5575,  2.5181,  ...,  1.8495,  1.8691, -2.4183],\n",
            "         [ 2.5771,  2.5575,  2.5181,  ...,  1.7118,  1.9478, -2.4183],\n",
            "         ...,\n",
            "         [ 2.5575,  2.5575,  2.5771,  ...,  2.5575,  2.5378, -2.4183],\n",
            "         [ 2.5575,  2.5378,  2.5181,  ...,  2.5181,  2.5181, -2.4183],\n",
            "         [ 2.3215,  2.2231,  2.1248,  ...,  2.0855,  2.1051, -2.4183]],\n",
            "\n",
            "        [[ 2.7342,  2.7147,  2.6562,  ...,  1.6416,  1.6807, -2.2214],\n",
            "         [ 2.7342,  2.7147,  2.6562,  ...,  1.6807,  1.7002, -2.2214],\n",
            "         [ 2.7342,  2.7147,  2.6562,  ...,  1.5636,  1.7587, -2.2214],\n",
            "         ...,\n",
            "         [ 2.7147,  2.7147,  2.7147,  ...,  2.6757,  2.6367, -2.2214],\n",
            "         [ 2.7147,  2.6952,  2.6562,  ...,  2.5781,  2.5781, -2.2214],\n",
            "         [ 2.4611,  2.3635,  2.2270,  ...,  2.0904,  2.1099, -2.2214]]]), 0) plane\n"
          ]
        },
        {
          "data": {
            "image/png": "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",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:matplotlib.image:Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers).\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "8 (tensor([[[-2.4291, -2.4291,  0.8470,  ...,  0.6338,  0.8082,  0.9633],\n",
            "         [-2.4291, -2.4291,  0.8470,  ...,  0.9827,  1.0990,  1.1378],\n",
            "         [-2.4291, -2.4291,  1.0021,  ...,  1.1765,  1.2153,  1.2735],\n",
            "         ...,\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291]],\n",
            "\n",
            "        [[-2.4183, -2.4183,  1.2004,  ...,  1.3184,  1.4364,  1.5544],\n",
            "         [-2.4183, -2.4183,  1.2988,  ...,  1.5741,  1.6331,  1.6331],\n",
            "         [-2.4183, -2.4183,  1.5348,  ...,  1.6724,  1.6921,  1.7118],\n",
            "         ...,\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183]],\n",
            "\n",
            "        [[-2.2214, -2.2214,  1.9148,  ...,  2.3050,  2.3830,  2.4221],\n",
            "         [-2.2214, -2.2214,  2.1294,  ...,  2.4611,  2.4611,  2.4416],\n",
            "         [-2.2214, -2.2214,  2.4221,  ...,  2.4611,  2.4611,  2.5001],\n",
            "         ...,\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214]]]), 0) plane\n"
          ]
        },
        {
          "data": {
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAaAAAAGdCAYAAABU0qcqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAj20lEQVR4nO3df3CV9Zn38U+CyQEkORhCfkkCAS3UQtIpSsyqiJIC6YwLBTuo7RRaHx1osIVo1bQq/ug2FmdWbQdx+0vqroilj0C1Lf6IJoxtQIlQpGoW2FhAkrBSOScEE7LJ/fzh41mjIPeVnJNvTvJ+zdwz5JyLK9fNDflwJydXEjzP8wQAQB9LdD0AAGBwIoAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOHGW6wE+qaurS4cPH1ZKSooSEhJcjwMAMPI8Ty0tLcrJyVFi4unvc/pdAB0+fFi5ubmuxwAA9NLBgwc1ZsyY0z4fs0/BrV69WuPGjdPQoUNVVFSkV1991dfvS0lJidVIAIA+dKaP5zEJoKeeekrl5eVauXKlXn/9dRUWFmr27Nk6cuTIGX8vn3YDgIHhjB/PvRiYNm2aV1ZWFnm7s7PTy8nJ8SorK8/4e0OhkCeJg4ODgyPOj1Ao9Jkf76N+B3Ty5EnV1dWppKQk8lhiYqJKSkpUW1v7qfr29naFw+FuBwBg4It6AL333nvq7OxUZmZmt8czMzPV1NT0qfrKykoFg8HIwQsQAGBwcP59QBUVFQqFQpHj4MGDrkcCAPSBqL8MOz09XUOGDFFzc3O3x5ubm5WVlfWp+kAgoEAgEO0xAAD9XNTvgJKTkzV16lRVVVVFHuvq6lJVVZWKi4uj/e4AAHEqJt+IWl5erkWLFunCCy/UtGnT9NBDD6m1tVXf+ta3YvHuAABxKCYBtHDhQv33f/+37rrrLjU1NemLX/yitmzZ8qkXJgAABq8Ez/M810N8XDgcVjAYdD0GAKCXQqGQUlNTT/u881fBAQAGJwIIAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4QQABAJwggAAATpzleoA+V+fZ6i0RPdTW2qQzhr2T+lG9tXeKoXa4rbWlfISttfk0Owy11r+GltnTjL2HGWqDxt7BE/5rv2G89jNs5fiEcDisYPDMV5Q7IACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMCJwbeKxxq5lh0oVkNi2Nu668WiK8b1MZJkXGcUMFyfow223qOMO22GG1bJGLfOKMXwd8W6cujIP/zXvvu2rfc3/sl/bbOtNfoId0AAACcIIACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMCJwbcLzrqXzLKvzbrbzRL/1t1usdwzFzDWW5aTGc9zqOHP0PpHONSwO+79d2293z9iq8+/0H/tMOO1t5Sn21qr6j/+7rt26Alb74x/Guu7tp+sI8QncAcEAHAi6gF09913KyEhodsxadKkaL8bAECci8mn4L7whS/oxRdf/N93ctbg+0wfAOCzxSQZzjrrLGVlZcWiNQBggIjJ14D27t2rnJwcjR8/Xl//+td14MCB09a2t7crHA53OwAAA1/UA6ioqEhr167Vli1btGbNGjU0NOiyyy5TS0vLKesrKysVDAYjR25ubrRHAgD0Q1EPoNLSUn3ta19TQUGBZs+erT/+8Y86duyYfvvb356yvqKiQqFQKHIcPHgw2iMBAPqhmL86YOTIkfrc5z6nffv2nfL5QCCgQMD6jSUAgHgX8+8DOn78uPbv36/s7OxYvysAQByJegDdcsstqqmp0TvvvKO//OUv+upXv6ohQ4bo2muvjfa7AgDEsah/Cu7QoUO69tprdfToUY0ePVqXXnqptm3bptGjR0f7XfWMYb2KJNv+FmucW3fDWFjOc6ixt2W1jmQ7T+vOFEN9Yoext6E+27ijpnGXrT5g+F7utKCtt2lT0j9svTv+7ee+a2fc/H1T7wxDbbGpM/pK1ANo/fr10W4JABiA2AUHAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOBHzH8fQ7wyJYb21dyx3wVl6W3fBWc/TwvhfokTDzrsu4y64RMNPCck739a78T9t9aH3/NeOM+6Cs5RX/abZ1rzR/8/3+s23R5pas18//nEHBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADgx+FbxWCPXUh/LOLeuyzGskTGznqdh9iTDah3JtnFouHGFUJqhebqttc6fYKs/YVnFY+wdMmzXefO+20y9b7m73Hctq3UGH+6AAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAE4NvF5xleZgkGfeHxaMhxj+TTuufoWG/W5exteX6DDfOPcJQa/0jmTTGVv/KH/zXdhXZer9Ta7hA7z9v6l32f9bahsGgwh0QAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwYvDtgrMuG7PsgrMuBEsz1BrWdUkyzWLe7Wb9MzTMnmicJWD4L1QsT9N6ebKDtvrgcP+1b9fYev/x3x7wX3xWtqn3OMPcGHy4AwIAOGEOoK1bt+qqq65STk6OEhIStGnTpm7Pe56nu+66S9nZ2Ro2bJhKSkq0d+/eaM0LABggzAHU2tqqwsJCrV69+pTPr1q1Sj/96U/16KOPavv27Tr77LM1e/ZstbW19XpYAMDAYf4aUGlpqUpLS0/5nOd5euihh3THHXdo7ty5kqTHH39cmZmZ2rRpk6655preTQsAGDCi+jWghoYGNTU1qaSkJPJYMBhUUVGRamtrT/l72tvbFQ6Hux0AgIEvqgHU1NQkScrMzOz2eGZmZuS5T6qsrFQwGIwcubm50RwJANBPOX8VXEVFhUKhUOQ4ePCg65EAAH0gqgGUlZUlSWpubu72eHNzc+S5TwoEAkpNTe12AAAGvqgGUH5+vrKyslRVVRV5LBwOa/v27SouLo7muwIAxDnzq+COHz+uffv2Rd5uaGjQrl27lJaWpry8PC1fvlw/+tGPdP755ys/P1933nmncnJyNG/evGjODQCIc+YA2rFjh6644orI2+Xl5ZKkRYsWae3atbr11lvV2tqqG2+8UceOHdOll16qLVu2aOjQodGbujesO1MChtqQsfdfDbXpttZJ5/uv7bBeGuN98xDDTpukE7bellVJAeP6mwxD7Thba40x1h86x3/t7+5fY2u+pcJ36bfufMLWG/gM5gCaMWOGPM877fMJCQm69957de+99/ZqMADAwOb8VXAAgMGJAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOGFexRP3DLvDzNa8Zav/+XL/tSOmmFp3/PAe37UpV51t6j3MuMjsuGG/25A2W++hSf5rsy17/SR9JYbrC39x319M9a+vesh/8fH3bMMoxXflK7v/09R5R4f/2gsN1xJ9784a/4s021v91XIHBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADiR4Hme53qIjwuHwwoGg7F7B2/F8HTvtq1X0VOXxGYOSdLl/ksLv2Fr/bWrTeUpl4/0XTtxkm2UKef4r006YOs9zrCK5/E120y9377vZtswMuy00Thb67MMf4jpw2298zJ9l87/zkJT6xvm5/uunel/25AkqT9tBXrXUPvA720f3x7+lx/5L371D4bO/yOpTqFQSKmpqaet4g4IAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4Mfh2we00nu4QQ+0hW2stW+a/9r9WG5tbGBewaYKtfOIs/7W3ftfUeqFh5d1Ty35t6q0tT/mvDeTZeltWu0lS1xFDsfH/lWcZrueYDFvvzpD/2qBxYZtafFd++ZqrTJ1//8OLTfVvGmrv/eU+U+/Ndz/qv/jd5029bX8RLXsAOyX9lV1wAID+iQACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADgx+Fbx7Daebiwj2rJdpfR6W+92y9oZy74hSRphrE8y1H7J1nrc+f5r34nlOiPrGhnr33HLqh/jaqVzJ/uvTU+39W4xrOI5ccLWW+3+Sy+dYup8br5t5dC7v/6V/+KjNabe9r1NFkMNtZYVXB2SXmQVDwCgfyKAAABOmANo69atuuqqq5STk6OEhARt2rSp2/OLFy9WQkJCt2POnDnRmhcAMECYA6i1tVWFhYVavfr0n0+fM2eOGhsbI8eTTz7ZqyEBAAPPWdbfUFpaqtLS0s+sCQQCysrK6vFQAICBLyZfA6qurlZGRoYmTpyopUuX6ujRo6etbW9vVzgc7nYAAAa+qAfQnDlz9Pjjj6uqqko/+clPVFNTo9LSUnV2dp6yvrKyUsFgMHLk5uZGeyQAQD9k/hTcmVxzzTWRX0+ZMkUFBQWaMGGCqqurNXPmzE/VV1RUqLy8PPJ2OBwmhABgEIj5y7DHjx+v9PR07dt36p+DHggElJqa2u0AAAx8MQ+gQ4cO6ejRo8rOzo71uwIAxBHzp+COHz/e7W6moaFBu3btUlpamtLS0nTPPfdowYIFysrK0v79+3XrrbfqvPPO0+zZs6M6OAAgvpl3wVVXV+uKK6741OOLFi3SmjVrNG/ePO3cuVPHjh1TTk6OZs2apfvuu0+ZmZm++sd8F9zfjbvgLGvMjCubhhruP5Net/VuWbDYf/H//MbWHP3cLFv5jO/6r80eb+tt2e+2+VFbb73rv/QC4zfDv2n9N/GOoXa4sbflg1CLsbf/Wa5eu8d3bccHYW1eOuaMu+DMd0AzZszQZ2XWc889Z20JABiE2AUHAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOBH1nwfU71kj17KGyVIrqcOwtmnU5bbeZ7281nft+5c12prreWM9+pbx+lS3+6+9+n5b7y5Db71p661d/ks7Pv2zyD5burHeMrtxaWQM7xPO/e7PfNduWJTiuzYc9hRceuY67oAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJwbfKh6rTkNtl7G3of5Em631mAv81w59/GlT78ZvXmgbRm8b69G3XvNdmZT2P6bOHYeOG6oPmXpLhjU/ln/HkuyreCzrdQLG3v53fAXn/8DU+dDD/2ycJbq4AwIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4Mvl1wQ2LY27xvyr8hJ2z1LYb68y8/29Q78dEqU/27S6YYqv9h6o1o8P+P4tz0VFPnd0KWHWmWvXGS6R9c0Lh/bdQ4W/3RDEOx8R/z5DLfpcf+7wpbb8e4AwIAOEEAAQCcIIAAAE4QQAAAJwggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcGHyreKzrcmIY0YmG3u2WjSaS1OW/9N2QrfUFJTmm+o6Vz/iuPXLPJbZhEAUtvivf2b/X1jop01AcwzVMR/5q/A3jjPXD/ZemXmrq/NYbd9tGiSPcAQEAnDAFUGVlpS666CKlpKQoIyND8+bNU319fbeatrY2lZWVadSoURoxYoQWLFig5ubmqA4NAIh/pgCqqalRWVmZtm3bphdeeEEdHR2aNWuWWltbIzUrVqzQM888ow0bNqimpkaHDx/W/Pnzoz44ACC+mb4GtGXLlm5vr127VhkZGaqrq9P06dMVCoX0q1/9SuvWrdOVV14pSXrsscf0+c9/Xtu2bdPFF18cvckBAHGtV18DCoU+/Op1WlqaJKmurk4dHR0qKSmJ1EyaNEl5eXmqra09ZY/29naFw+FuBwBg4OtxAHV1dWn58uW65JJLNHnyZElSU1OTkpOTNXLkyG61mZmZampqOmWfyspKBYPByJGbm9vTkQAAcaTHAVRWVqY9e/Zo/fr1vRqgoqJCoVAochw8eLBX/QAA8aFH3we0bNkyPfvss9q6davGjBkTeTwrK0snT57UsWPHut0FNTc3Kysr65S9AoGAAgHjj8sFAMQ90x2Q53latmyZNm7cqJdeekn5+fndnp86daqSkpJUVVUVeay+vl4HDhxQcXFxdCYGAAwIpjugsrIyrVu3Tps3b1ZKSkrk6zrBYFDDhg1TMBjU9ddfr/LycqWlpSk1NVU33XSTiouLeQUcAKAbUwCtWbNGkjRjxoxujz/22GNavHixJOnBBx9UYmKiFixYoPb2ds2ePVuPPPJIVIYFAAwcCZ7nea6H+LhwOKxgMBi7d3DYeLqW3XHGfW1JJ/zXDmm39R7mf72X0obYeo8zrL2SpC/n+a9t/H2dqffD119oGwa9M/5mW/1l8/zX/uYyW28Tw19CSZJtX5tl+eJr3pOmzvH4N/yjj+OhUEipqamnrWMXHADACQIIAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOBEj34cQ1yzrNaRLBs2JMNqHUnqeN9/7dA2W+9Mw1qg3CG29UTXFyeY6hdaVv18e6qp96V5f/Vd+7UvF5p64xT+69Q/2fi0JhUZisecuaSbQ4Za4yqeEbZ9U//+2i9818bjap1Y4Q4IAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4Mfh2wYWM9ZZdcP8w9j7ivzSx7aSp9Tl6z3ftwityTL1Nu91i7OqSAt+1P37kSVPvH3znWus4g8ABW/nxFv+1I75i7P2u/9pLrjC1/vWam03135hkKsf/xx0QAMAJAggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4MTgW8Xjf0PNhyyrezo8U+uk0Pu+a+dfmGbq/b0v+l+vU2jqHL8qll5jqn9z1xu+a//j5z+2jhOnThjLDbusCi+39X77Fd+ld9zyTVPrb02xjYKe4Q4IAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4Mfh2wRlXWVkMPe5/t5skFSZ2+K799ReNw6DX/v3f/sV37Xvv2ZYMbnn659Zx+omgrfwfhn9wXYdMrS9cfJXv2vvmjTb1Rt/gDggA4IQpgCorK3XRRRcpJSVFGRkZmjdvnurr67vVzJgxQwkJCd2OJUuWRHVoAED8MwVQTU2NysrKtG3bNr3wwgvq6OjQrFmz1Nra2q3uhhtuUGNjY+RYtWpVVIcGAMQ/09eAtmzZ0u3ttWvXKiMjQ3V1dZo+fXrk8eHDhysrKys6EwIABqRefQ0oFPrwp7WlpXX/YWlPPPGE0tPTNXnyZFVUVOjEidN/IbK9vV3hcLjbAQAY+Hr8Kriuri4tX75cl1xyiSZPnhx5/LrrrtPYsWOVk5Oj3bt367bbblN9fb2efvrpU/aprKzUPffc09MxAABxqscBVFZWpj179uiVV7r/WNwbb7wx8uspU6YoOztbM2fO1P79+zVhwoRP9amoqFB5eXnk7XA4rNzc3J6OBQCIEz0KoGXLlunZZ5/V1q1bNWbMmM+sLSoqkiTt27fvlAEUCAQUCAR6MgYAII6ZAsjzPN10003auHGjqqurlZ+ff8bfs2vXLklSdnZ2jwYEAAxMpgAqKyvTunXrtHnzZqWkpKipqUmSFAwGNWzYMO3fv1/r1q3TV77yFY0aNUq7d+/WihUrNH36dBUUFMTkBAAA8ckUQGvWrJH04Tebftxjjz2mxYsXKzk5WS+++KIeeughtba2Kjc3VwsWLNAdd9wRtYEBAANDgud5nushPi4cDisYNO6bsvij7XTPSfJfm7T3sKn3H5bm+K690NQZ/d2seTeY6l/Y/MsYTWJ1ha18hP+/uTNvn29q/eIPL7bNgj7z0cfxUCik1NTU09axCw4A4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwosc/DyheDR1uq0/v8F+b1Nlu6s1+8MHr+U2/MNVfdPl7vmt3bN1knMYiZKoeWvzZP67l41itM/hwBwQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4QQABAJwYdLvgkgy73SRp6IlO37W5aeeYegdso2AQe61mo+/atMxppt7vH3nNf/GIdFPvP//hu6Z6DC7cAQEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADhBAAEAnCCAAABODLpVPBnvt5jqCw3bdX74zyNNvW1LTQB//tH8qqm+9Lof+q694Xu21TpfSjKVY5DhDggA4AQBBABwggACADhBAAEAnCCAAABOEEAAACcIIACAEwQQAMAJAggA4AQBBABwggACADiR4Hme53qIjwuHwwoGgzHrX/GW7XR/PClGgwDAAPXRx/FQKKTU1NTT1nEHBABwwhRAa9asUUFBgVJTU5Wamqri4mL96U9/ijzf1tamsrIyjRo1SiNGjNCCBQvU3Nwc9aEBAPHPFEBjxozR/fffr7q6Ou3YsUNXXnml5s6dq7/97W+SpBUrVuiZZ57Rhg0bVFNTo8OHD2v+/PkxGRwAEN96/TWgtLQ0PfDAA7r66qs1evRorVu3TldffbUk6e2339bnP/951dbW6uKLL/bVj68BAUB8i/nXgDo7O7V+/Xq1traquLhYdXV16ujoUElJSaRm0qRJysvLU21t7Wn7tLe3KxwOdzsAAAOfOYDeeOMNjRgxQoFAQEuWLNHGjRt1wQUXqKmpScnJyRo5cmS3+szMTDU1NZ22X2VlpYLBYOTIzc01nwQAIP6YA2jixInatWuXtm/frqVLl2rRokV68803ezxARUWFQqFQ5Dh48GCPewEA4sdZ1t+QnJys8847T5I0depUvfbaa3r44Ye1cOFCnTx5UseOHet2F9Tc3KysrKzT9gsEAgoEAvbJAQBxrdffB9TV1aX29nZNnTpVSUlJqqqqijxXX1+vAwcOqLi4uLfvBgAwwJjugCoqKlRaWqq8vDy1tLRo3bp1qq6u1nPPPadgMKjrr79e5eXlSktLU2pqqm666SYVFxf7fgUcAGDwMAXQkSNH9M1vflONjY0KBoMqKCjQc889py9/+cuSpAcffFCJiYlasGCB2tvbNXv2bD3yyCMxGbyneFk1APQPg24XXD87XQAYcNgFBwDo1wggAIATBBAAwAkCCADgBAEEAHCCAAIAOEEAAQCcIIAAAE4QQAAAJ8zbsGMt1psK+IF3ABBbH32cPdPH834XQC0tLTHtH8s1PwCA/9XS0vKZH3P73S64rq4uHT58WCkpKUpISIg8Hg6HlZubq4MHD37mbqF4x3kOHIPhHCXOc6CJxnl6nqeWlhbl5OQoMfH0X+npd3dAiYmJGjNmzGmfT01NHdAX/yOc58AxGM5R4jwHmt6ep5/PNvEiBACAEwQQAMCJuAmgQCCglStXKhAIuB4lpjjPgWMwnKPEeQ40fXme/e5FCACAwSFu7oAAAAMLAQQAcIIAAgA4QQABAJyImwBavXq1xo0bp6FDh6qoqEivvvqq65Gi6u6771ZCQkK3Y9KkSa7H6pWtW7fqqquuUk5OjhISErRp06Zuz3uep7vuukvZ2dkaNmyYSkpKtHfvXjfD9sKZznPx4sWfurZz5sxxM2wPVVZW6qKLLlJKSooyMjI0b9481dfXd6tpa2tTWVmZRo0apREjRmjBggVqbm52NHHP+DnPGTNmfOp6LlmyxNHEPbNmzRoVFBREvtm0uLhYf/rTnyLP99W1jIsAeuqpp1ReXq6VK1fq9ddfV2FhoWbPnq0jR464Hi2qvvCFL6ixsTFyvPLKK65H6pXW1lYVFhZq9erVp3x+1apV+ulPf6pHH31U27dv19lnn63Zs2erra2tjyftnTOdpyTNmTOn27V98skn+3DC3qupqVFZWZm2bdumF154QR0dHZo1a5ZaW1sjNStWrNAzzzyjDRs2qKamRocPH9b8+fMdTm3n5zwl6YYbbuh2PVetWuVo4p4ZM2aM7r//ftXV1WnHjh268sorNXfuXP3tb3+T1IfX0osD06ZN88rKyiJvd3Z2ejk5OV5lZaXDqaJr5cqVXmFhoesxYkaSt3HjxsjbXV1dXlZWlvfAAw9EHjt27JgXCAS8J5980sGE0fHJ8/Q8z1u0aJE3d+5cJ/PEypEjRzxJXk1Njed5H167pKQkb8OGDZGat956y5Pk1dbWuhqz1z55np7neZdffrn3ve99z91QMXLOOed4v/zlL/v0Wvb7O6CTJ0+qrq5OJSUlkccSExNVUlKi2tpah5NF3969e5WTk6Px48fr61//ug4cOOB6pJhpaGhQU1NTt+saDAZVVFQ04K6rJFVXVysjI0MTJ07U0qVLdfToUdcj9UooFJIkpaWlSZLq6urU0dHR7XpOmjRJeXl5cX09P3meH3niiSeUnp6uyZMnq6KiQidOnHAxXlR0dnZq/fr1am1tVXFxcZ9ey363jPST3nvvPXV2diozM7Pb45mZmXr77bcdTRV9RUVFWrt2rSZOnKjGxkbdc889uuyyy7Rnzx6lpKS4Hi/qmpqaJOmU1/Wj5waKOXPmaP78+crPz9f+/fv1gx/8QKWlpaqtrdWQIUNcj2fW1dWl5cuX65JLLtHkyZMlfXg9k5OTNXLkyG618Xw9T3WeknTddddp7NixysnJ0e7du3Xbbbepvr5eTz/9tMNp7d544w0VFxerra1NI0aM0MaNG3XBBRdo165dfXYt+30ADRalpaWRXxcUFKioqEhjx47Vb3/7W11//fUOJ0NvXXPNNZFfT5kyRQUFBZowYYKqq6s1c+ZMh5P1TFlZmfbs2RP3X6M8k9Od54033hj59ZQpU5Sdna2ZM2dq//79mjBhQl+P2WMTJ07Url27FAqF9Lvf/U6LFi1STU1Nn87Q7z8Fl56eriFDhnzqFRjNzc3KyspyNFXsjRw5Up/73Oe0b98+16PExEfXbrBdV0kaP3680tPT4/LaLlu2TM8++6xefvnlbj82JSsrSydPntSxY8e61cfr9TzdeZ5KUVGRJMXd9UxOTtZ5552nqVOnqrKyUoWFhXr44Yf79Fr2+wBKTk7W1KlTVVVVFXmsq6tLVVVVKi4udjhZbB0/flz79+9Xdna261FiIj8/X1lZWd2uazgc1vbt2wf0dZWkQ4cO6ejRo3F1bT3P07Jly7Rx40a99NJLys/P7/b81KlTlZSU1O161tfX68CBA3F1Pc90nqeya9cuSYqr63kqXV1dam9v79trGdWXNMTI+vXrvUAg4K1du9Z78803vRtvvNEbOXKk19TU5Hq0qLn55pu96upqr6Ghwfvzn//slZSUeOnp6d6RI0dcj9ZjLS0t3s6dO72dO3d6krx//dd/9Xbu3On9/e9/9zzP8+6//35v5MiR3ubNm73du3d7c+fO9fLz870PPvjA8eQ2n3WeLS0t3i233OLV1tZ6DQ0N3osvvuh96Utf8s4//3yvra3N9ei+LV261AsGg151dbXX2NgYOU6cOBGpWbJkiZeXl+e99NJL3o4dO7zi4mKvuLjY4dR2ZzrPffv2effee6+3Y8cOr6Ghwdu8ebM3fvx4b/r06Y4nt7n99tu9mpoar6Ghwdu9e7d3++23ewkJCd7zzz/veV7fXcu4CCDP87yf/exnXl5enpecnOxNmzbN27Ztm+uRomrhwoVedna2l5yc7J177rnewoULvX379rkeq1defvllT9KnjkWLFnme9+FLse+8804vMzPTCwQC3syZM736+nq3Q/fAZ53niRMnvFmzZnmjR4/2kpKSvLFjx3o33HBD3P3n6VTnJ8l77LHHIjUffPCB953vfMc755xzvOHDh3tf/epXvcbGRndD98CZzvPAgQPe9OnTvbS0NC8QCHjnnXee9/3vf98LhUJuBzf69re/7Y0dO9ZLTk72Ro8e7c2cOTMSPp7Xd9eSH8cAAHCi338NCAAwMBFAAAAnCCAAgBMEEADACQIIAOAEAQQAcIIAAgA4QQABAJwggAAAThBAAAAnCCAAgBMEEADAif8HZLjAFUbtgq0AAAAASUVORK5CYII=",
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "9 (tensor([[[-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         [-2.4291, -2.4291, -2.4291,  ..., -2.4291, -2.4291, -2.4291],\n",
            "         ...,\n",
            "         [-2.4291, -2.4291, -0.7232,  ...,  1.9132,  1.8356,  1.7775],\n",
            "         [-2.4291, -2.4291, -0.0447,  ...,  1.8356,  1.7387,  1.7193],\n",
            "         [-2.4291, -2.4291,  1.0021,  ...,  1.7193,  1.7193,  1.5836]],\n",
            "\n",
            "        [[-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         [-2.4183, -2.4183, -2.4183,  ..., -2.4183, -2.4183, -2.4183],\n",
            "         ...,\n",
            "         [-2.4183, -2.4183,  0.3351,  ...,  1.9281,  1.8101,  1.7314],\n",
            "         [-2.4183, -2.4183,  0.6891,  ...,  1.8691,  1.7708,  1.7314],\n",
            "         [-2.4183, -2.4183,  1.5151,  ...,  1.8298,  1.7904,  1.7118]],\n",
            "\n",
            "        [[-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         [-2.2214, -2.2214, -2.2214,  ..., -2.2214, -2.2214, -2.2214],\n",
            "         ...,\n",
            "         [-2.2214, -2.2214,  0.9003,  ...,  2.0904,  2.0319,  1.9148],\n",
            "         [-2.2214, -2.2214,  1.2319,  ...,  2.0319,  2.0123,  1.9343],\n",
            "         [-2.2214, -2.2214,  2.0319,  ...,  1.9538,  1.9928,  1.9343]]]), 0) plane\n"
          ]
        }
      ],
      "source": [
        "# functions to show an image\n",
        "def imshow(img):\n",
        "    img = img / 2 + 0.5     # unnormalize\n",
        "    npimg = img.numpy()\n",
        "    plt.imshow(np.transpose(npimg, (1, 2, 0)))\n",
        "    plt.show()\n",
        "\n",
        "class_names = ('plane', 'car', 'bird', 'cat', 'deer', 'dog', 'frog', 'horse', 'ship', 'truck')\n",
        "\n",
        "nb_classes = len(class_names)\n",
        "\n",
        "count = 0\n",
        "\n",
        "for index, iter in enumerate(train_subset):\n",
        "\n",
        "    image, label = iter[0], iter[1]\n",
        "    count += 1\n",
        "    imshow(torchvision.utils.make_grid(image))\n",
        "    print(index, iter, class_names[label])\n",
        "    if count == 10:\n",
        "        break"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "AggACHm-OLDO"
      },
      "outputs": [],
      "source": [
        "batch_size = 128"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "7Z3EGjB9egOb"
      },
      "outputs": [],
      "source": [
        "num_classes = 10"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "M7pxhMeWV7Km"
      },
      "outputs": [],
      "source": [
        "train_loader = torch.utils.data.DataLoader(train_subset, batch_size = batch_size, num_workers = 2, pin_memory = True)\n",
        "validation_loader = torch.utils.data.DataLoader(train_dataset, batch_size = batch_size, num_workers = 2, pin_memory = True, shuffle = False)\n",
        "test_loader = torch.utils.data.DataLoader(test_dataset, batch_size = batch_size, num_workers = 2, pin_memory = True, shuffle = False)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "r6PTVXZ6_C9s",
        "outputId": "46310315-0f8a-4a54-9e0b-8c61efae2355"
      },
      "outputs": [
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "/usr/local/lib/python3.10/dist-packages/torch/nn/modules/conv.py:456: UserWarning: Plan failed with a cudnnException: CUDNN_BACKEND_EXECUTION_PLAN_DESCRIPTOR: cudnnFinalize Descriptor Failed cudnn_status: CUDNN_STATUS_NOT_SUPPORTED (Triggered internally at ../aten/src/ATen/native/cudnn/Conv_v8.cpp:919.)\n",
            "  return F.conv2d(input, weight, bias, self.stride,\n"
          ]
        }
      ],
      "source": [
        "X_test = []\n",
        "y_test = []\n",
        "\n",
        "for batch_idx, (data, target) in enumerate(test_loader):\n",
        "    data, target = data.to(device), target.to(device)\n",
        "\n",
        "    _ = orig_model(data.to(device))\n",
        "\n",
        "    x = activation[\"layer3\"].detach()\n",
        "    y = activation['layer4'].detach()\n",
        "\n",
        "    if batch_idx == 0:\n",
        "        X_test = copy.deepcopy(x)\n",
        "        y_test = copy.deepcopy(y)\n",
        "\n",
        "    else:\n",
        "        X_test = torch.cat((X_test, x), 0)\n",
        "        y_test = torch.cat((y_test, y), 0)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "dg4sfMD86mxo"
      },
      "outputs": [],
      "source": [
        "X_test_SCNN = torch.reshape(X_test, (len(X_test), -1))\n",
        "y_test_SCNN = torch.reshape(y_test, (len(y_test), -1))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "RHd6g_bt6mxo",
        "outputId": "1a60e753-bd17-4aa7-fd89-78c86584e580"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "torch.Size([10000, 1024])\n",
            "torch.Size([10000, 512])\n"
          ]
        }
      ],
      "source": [
        "print(X_test_SCNN.shape)\n",
        "print(y_test_SCNN.shape)"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "eknuxrkv_xZu"
      },
      "source": [
        "### Create Dataset of Input-Output Activations using pre-trained model"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "xmtYWBFIIdm1",
        "outputId": "dea3fedc-eb5d-43e6-dd4d-ab23268c3631"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Input image shape torch.Size([128, 3, 32, 32])\n",
            "Layer1 Input torch.Size([128, 64, 8, 8])\n",
            "Layer2 Input torch.Size([128, 64, 8, 8])\n",
            "Layer3 Input torch.Size([128, 128, 4, 4])\n",
            "Layer3 Output torch.Size([128, 256, 2, 2])\n",
            "Layer4 Output torch.Size([128, 512, 1, 1])\n"
          ]
        }
      ],
      "source": [
        "def inspect_shapes(model, x):\n",
        "        print(\"Input image shape\", x.shape)\n",
        "        out = model.conv1(x)\n",
        "        out = model.bn1(out)\n",
        "        out = model.relu(out)\n",
        "        out = model.maxpool(out)\n",
        "        print(\"Layer1 Input\", out.shape)\n",
        "        out = model.layer1(out)\n",
        "        print(\"Layer2 Input\", out.shape)\n",
        "        out = model.layer2(out)\n",
        "        print(\"Layer3 Input\", out.shape)\n",
        "        out = model.layer3(out)\n",
        "        print(\"Layer3 Output\", out.shape)\n",
        "        out = model.layer4(out)\n",
        "        print(\"Layer4 Output\", out.shape)\n",
        "\n",
        "with torch.no_grad():\n",
        "\n",
        "    for data in validation_loader:\n",
        "\n",
        "        images, labels = data\n",
        "        images, labels = images.to(device), labels.to(device)\n",
        "\n",
        "        inspect_shapes(orig_model, images)\n",
        "\n",
        "        break"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "922JhWo5IdnB",
        "outputId": "def0d8d6-7c4b-4806-d47e-8e56edded12e"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "<torch.utils.hooks.RemovableHandle at 0x7857a251d090>"
            ]
          },
          "execution_count": 21,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "activation = {}\n",
        "def get_activation(name):\n",
        "    def hook(model, input, output):\n",
        "        activation[name] = output.to(device)\n",
        "    return hook\n",
        "\n",
        "orig_model.maxpool.register_forward_hook(get_activation('layer0'))\n",
        "orig_model.layer1.register_forward_hook(get_activation('layer1'))\n",
        "orig_model.layer2.register_forward_hook(get_activation('layer2'))\n",
        "orig_model.layer3.register_forward_hook(get_activation('layer3'))\n",
        "orig_model.layer4.register_forward_hook(get_activation('layer4'))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "Hai6rBHJ-j33"
      },
      "outputs": [],
      "source": [
        "import copy"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "2N1z9U2H79RB"
      },
      "outputs": [],
      "source": [
        "X_train = []\n",
        "y_train = []\n",
        "\n",
        "for batch_idx, (data, target) in enumerate(train_loader):\n",
        "    data, target = data.to(device), target.to(device)\n",
        "\n",
        "    _ = orig_model(data.to(device))\n",
        "\n",
        "    x = activation[\"layer3\"].detach()\n",
        "    y = activation['layer4'].detach()\n",
        "\n",
        "    # print(x.shape)\n",
        "    # print(y.shape)\n",
        "    # print(data.shape)\n",
        "\n",
        "    # break\n",
        "\n",
        "    if batch_idx == 0:\n",
        "        X_train = copy.deepcopy(x)\n",
        "        y_train = copy.deepcopy(y)\n",
        "\n",
        "    else:\n",
        "        X_train = torch.cat((X_train, x), 0)\n",
        "        y_train = torch.cat((y_train, y), 0)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "TfHibCsF80ik",
        "outputId": "7f5f1945-56b9-4e06-b99e-da11e4376da6"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "torch.Size([10, 256, 2, 2])\n",
            "torch.Size([10, 512, 1, 1])\n"
          ]
        }
      ],
      "source": [
        "print(X_train.shape)\n",
        "print(y_train.shape)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "rlDkR8kv_4jR"
      },
      "outputs": [],
      "source": [
        "X_train_SCNN = torch.reshape(X_train, (len(X_train), -1))\n",
        "y_train_SCNN = torch.reshape(y_train, (len(y_train), -1))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "W4GWdwkLAWZd",
        "outputId": "5ab9ad6f-753a-43bc-e915-a18731adfd17"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "torch.Size([10, 1024])\n",
            "torch.Size([10, 512])\n"
          ]
        }
      ],
      "source": [
        "print(X_train_SCNN.shape)\n",
        "print(y_train_SCNN.shape)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "pJI2lwbeCBwH"
      },
      "outputs": [],
      "source": [
        "gaussian_model = copy.deepcopy(orig_model)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "Wm6db9V6CP-n",
        "outputId": "14249ce0-f059-4ca7-fa07-60c295364d46"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "<torch.utils.hooks.RemovableHandle at 0x7fc15ee79ab0>"
            ]
          },
          "execution_count": 39,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "gaussian_model.maxpool.register_forward_hook(get_activation('layer0'))\n",
        "gaussian_model.layer1.register_forward_hook(get_activation('layer1'))\n",
        "gaussian_model.layer2.register_forward_hook(get_activation('layer2'))\n",
        "gaussian_model.layer3.register_forward_hook(get_activation('layer3'))\n",
        "gaussian_model.layer4.register_forward_hook(get_activation('layer4'))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "R8lFQH1D6utW",
        "outputId": "ef024adc-e8bb-4cdd-c1c2-653a92736a57"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "ResNet(\n",
              "  (conv1): Conv2d(3, 64, kernel_size=(7, 7), stride=(2, 2), padding=(3, 3), bias=False)\n",
              "  (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "  (relu): ReLU(inplace=True)\n",
              "  (maxpool): MaxPool2d(kernel_size=3, stride=2, padding=1, dilation=1, ceil_mode=False)\n",
              "  (layer1): Sequential(\n",
              "    (0): BasicBlock(\n",
              "      (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "    (1): BasicBlock(\n",
              "      (conv1): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(64, 64, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(64, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "  )\n",
              "  (layer2): Sequential(\n",
              "    (0): BasicBlock(\n",
              "      (conv1): Conv2d(64, 128, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (downsample): Sequential(\n",
              "        (0): Conv2d(64, 128, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "        (1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      )\n",
              "    )\n",
              "    (1): BasicBlock(\n",
              "      (conv1): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(128, 128, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(128, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "  )\n",
              "  (layer3): Sequential(\n",
              "    (0): BasicBlock(\n",
              "      (conv1): Conv2d(128, 256, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (downsample): Sequential(\n",
              "        (0): Conv2d(128, 256, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "        (1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      )\n",
              "    )\n",
              "    (1): BasicBlock(\n",
              "      (conv1): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(256, 256, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(256, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "  )\n",
              "  (layer4): Sequential(\n",
              "    (0): BasicBlock(\n",
              "      (conv1): Conv2d(256, 512, kernel_size=(3, 3), stride=(2, 2), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (downsample): Sequential(\n",
              "        (0): Conv2d(256, 512, kernel_size=(1, 1), stride=(2, 2), bias=False)\n",
              "        (1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      )\n",
              "    )\n",
              "    (1): BasicBlock(\n",
              "      (conv1): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn1): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "      (relu): ReLU(inplace=True)\n",
              "      (conv2): Conv2d(512, 512, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1), bias=False)\n",
              "      (bn2): BatchNorm2d(512, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)\n",
              "    )\n",
              "  )\n",
              "  (avgpool): AdaptiveAvgPool2d(output_size=(1, 1))\n",
              "  (fc): Linear(in_features=512, out_features=1000, bias=True)\n",
              ")"
            ]
          },
          "execution_count": 41,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "data_gaussian = torch.normal(0, 5, size=(num_train_gaussian_samples, 3, 32, 32)).to(device)\n",
        "\n",
        "\n",
        "gaussian_model.train()\n",
        "with torch.no_grad():\n",
        "    _ = gaussian_model(data_gaussian)\n",
        "\n",
        "    X_train_gaussian = activation[\"layer3\"].detach()\n",
        "    y_train_gaussian = activation[\"layer4\"].detach()\n",
        "\n",
        "gaussian_model.eval()"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 159
        },
        "id": "S8QPH0Jk5deB",
        "outputId": "6917ed41-c715-477a-aacb-11bde89792fd"
      },
      "outputs": [],
      "source": [
        "X_train_gaussian = torch.reshape(X_train_gaussian, (len(X_train_gaussian), -1))\n",
        "y_train_gaussian = torch.reshape(y_train_gaussian, (len(y_train_gaussian), -1))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "oSboIL608U3X",
        "outputId": "112bad19-3457-44af-9288-b6b1289a1dcf"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "torch.Size([1000, 1024])"
            ]
          },
          "execution_count": 42,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "X_train_gaussian.shape"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "qthCEwZh7yir"
      },
      "outputs": [],
      "source": [
        "# X_train_SCNN_new = torch.cat((X_train_SCNN, X_train_gaussian), 0)\n",
        "# y_train_SCNN_new = torch.cat((y_train_SCNN, y_train_gaussian), 0)\n",
        "# X_train_SCNN_new = X_train_gaussian\n",
        "# y_train_SCNN_new = y_train_gaussian\n",
        "X_train_SCNN_new = X_train_SCNN\n",
        "y_train_SCNN_new = y_train_SCNN"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "H-ga45U7_38H"
      },
      "source": [
        "### Train a SCNN GReLU Network with regularization"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "BWoqosej86OU"
      },
      "outputs": [],
      "source": [
        "num_train_samples = 10\n",
        "num_train_gaussian_samples = 0\n",
        "\n",
        "debugging_data = torch.load('/content/gdrive/My Drive/From Non-Convex to Convex/Fine-tune Pre-trained Resnet18/10 Class Experiments/debugging_scnn_numtrain{}_gaussian{}'.format(num_train_samples, num_train_gaussian_samples))\n",
        "\n",
        "X_train_SCNN_new = debugging_data['X_train_SCNN_new']\n",
        "y_train_SCNN_new = debugging_data['y_train_SCNN_new']\n",
        "W1 = debugging_data['W1']\n",
        "W2 = debugging_data['W2']\n",
        "gates = debugging_data['gates']\n",
        "gated_activations = debugging_data['gated_activations']\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "vvM27zyfF7HN"
      },
      "outputs": [],
      "source": [
        "import copy\n",
        "\n",
        "X_train_SCNN_new_orig = copy.deepcopy(X_train_SCNN_new)\n",
        "y_train_SCNN_new_orig = copy.deepcopy(y_train_SCNN_new)\n",
        "X_test_SCNN_orig = copy.deepcopy(X_test_SCNN)\n",
        "y_test_SCNN_orig = copy.deepcopy(y_test_SCNN)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "xtgkk2luF_ZP"
      },
      "outputs": [],
      "source": [
        "from sklearn import preprocessing\n",
        "scaler = preprocessing.StandardScaler().fit(X_train_SCNN_new.cpu().numpy())\n",
        "X_train_SCNN_new = torch.from_numpy(scaler.transform(X_train_SCNN_new.cpu().numpy())).to(device)\n",
        "X_test_SCNN =  torch.from_numpy(scaler.transform(X_test_SCNN.cpu().numpy())).to(device)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "UE11kdD_uHPt"
      },
      "outputs": [],
      "source": [
        "lam = 5e-8\n",
        "max_neurons = 100\n",
        "G = sample_gate_vectors(123, X_train_SCNN_new.shape[-1], max_neurons)\n",
        "metrics = Metrics(metric_freq=25, model_loss=True, train_accuracy=False, train_mse=True, test_mse=True, test_accuracy=False, neuron_sparsity=True)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "Rn8gDWsxuH_C"
      },
      "outputs": [],
      "source": [
        "# Instantiate convex model and other options.\n",
        "model = ConvexGatedReLU(G, c=y_train_SCNN_new.shape[-1])\n",
        "solver = RFISTA(model)\n",
        "regularizer = L1(lam)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 191,
          "referenced_widgets": [
            "85634e7f94dc430c910f6b56f96d4d41",
            "3e7b644c59da472784104cd16c5d83f6",
            "3ddd4bdfd4bc4aed9954646969ccc5e7",
            "c45ea1db6fb44eedaee1b94d6d3829bb",
            "c97d596d1703495bb7da15526d037b12",
            "8d43325a440a4e05bb36352b841fa856",
            "d6feeae7d8da4e98934474ad17c189a2",
            "a7b4591ae08f45e39739bcc3bfc1aeb5",
            "e944c6a17f6d4217a968ec019867b660",
            "096dbbb65dbe43e49dd49d4f038893d8",
            "4b9222767f6b429db911283903ead483"
          ]
        },
        "id": "PBsA61mqU2a4",
        "outputId": "32dc9134-e3fc-44a1-87a8-e7f4a1a9887c"
      },
      "outputs": [
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "INFO:scnn:Pre-Optimization Metrics: Train Set objective: 0.06752537935972214, Train Set grad_norm: 0.0021077762357890606, Train Set base_objective: 0.06752537935972214, Train Set squared_error: 69.14598846435547, Test Set nc_squared_error: 65.74641418457031, group_sparsity: 1.0, \n",
            "\u001b[38;20m2024-05-30 19:12:57,585 - scnn - INFO - Pre-Optimization Metrics: Train Set objective: 0.06752537935972214, Train Set grad_norm: 0.0021077762357890606, Train Set base_objective: 0.06752537935972214, Train Set squared_error: 69.14598846435547, Test Set nc_squared_error: 65.74641418457031, group_sparsity: 1.0,  (iterative.py:131)\u001b[0m\n"
          ]
        },
        {
          "data": {
            "application/vnd.jupyter.widget-view+json": {
              "model_id": "85634e7f94dc430c910f6b56f96d4d41",
              "version_major": 2,
              "version_minor": 0
            },
            "text/plain": [
              "fista:   0%|          | 0/10000 [00:00<?, ?it/s]"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Train Set objective: 0.06752537935972214, Train Set grad_norm: 0.0021077762357890606, Train Set base_objective: 0.06752537935972214, Train Set squared_error: 69.14598846435547, Test Set nc_squared_error: 65.74641418457031, group_sparsity: 1.0, \n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "INFO:scnn:Termination criterion satisfied at iteration 11/10000. Exiting optimization loop.\n",
            "\u001b[38;20m2024-05-30 19:13:43,394 - scnn - INFO - Termination criterion satisfied at iteration 11/10000. Exiting optimization loop. (iterative.py:250)\u001b[0m\n",
            "INFO:scnn:Post-Optimization Metrics: Train Set objective: 5.443900590762496e-05, Train Set grad_norm: 8.89074101451115e-07, Train Set base_objective: 5.443900590762496e-05, Train Set squared_error: 0.05574554204940796, Test Set nc_squared_error: 60477.71484375, group_sparsity: 0.0, \n",
            "\u001b[38;20m2024-05-30 19:13:47,602 - scnn - INFO - Post-Optimization Metrics: Train Set objective: 5.443900590762496e-05, Train Set grad_norm: 8.89074101451115e-07, Train Set base_objective: 5.443900590762496e-05, Train Set squared_error: 0.05574554204940796, Test Set nc_squared_error: 60477.71484375, group_sparsity: 0.0,  (iterative.py:184)\u001b[0m\n"
          ]
        }
      ],
      "source": [
        "grelu_model, grelu_metrics = optimize_model(\n",
        "    model,\n",
        "    solver,\n",
        "    metrics,\n",
        "    X_train_SCNN_new.cpu().numpy(),\n",
        "    y_train_SCNN_new.cpu().numpy(),\n",
        "    X_test_SCNN.cpu().numpy(),\n",
        "    y_test_SCNN.cpu().numpy(),\n",
        "    # regularizer=regularizer,\n",
        "    verbose=True,\n",
        "    device='cuda',\n",
        "    # return_convex=True\n",
        ")"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "VA0tIkwbrWn8",
        "outputId": "8f748a44-0f7a-477a-9384-ea91c6502f5d"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Parameter Shape:  (51200, 1024)\n",
            "#Non-zero entries:  52428800\n",
            "Total number of parameters:  52428800\n",
            "\n",
            "\n",
            "Parameter Shape:  (512, 51200)\n",
            "#Non-zero entries:  51200\n",
            "Total number of parameters:  26214400\n",
            "\n",
            "\n",
            "[0.00219308 0.0506067  0.11118023 0.17192176 0.2326482  0.29339906\n",
            " 0.3542356  0.41462727 0.47551103 0.53645461 0.59724173 0.65774029\n",
            " 0.66543882]\n"
          ]
        }
      ],
      "source": [
        "for p in grelu_model.get_parameters():\n",
        "    print(\"Parameter Shape: \", p.shape)\n",
        "    print(\"#Non-zero entries: \", np.count_nonzero(p))\n",
        "    print(\"Total number of parameters: \", np.size(p))\n",
        "    print(\"\\n\")\n",
        "\n",
        "print(grelu_metrics.time)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "cw4ZY-LLMy6i",
        "outputId": "8ae69147-2ec0-4e34-b170-10e152367e08"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "81.9"
            ]
          },
          "execution_count": 92,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "report_test_accuracy(grelu_model)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "rxcTBHMOz3MM",
        "outputId": "edd7fb17-e8e9-4f81-e01a-f6719854069f"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "[0.00219308 0.0506067  0.11118023 0.17192176 0.2326482  0.29339906\n",
            " 0.3542356  0.41462727 0.47551103 0.53645461 0.59724173 0.65774029\n",
            " 0.66543882]\n",
            "13\n",
            "0\n"
          ]
        }
      ],
      "source": [
        "print(grelu_metrics.time)\n",
        "print(len(grelu_metrics.time))\n",
        "print(np.argmin(grelu_metrics.test_mse))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "RX4aTTAhsgw4",
        "outputId": "519641d0-29c0-4496-a709-45e48a551e18"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "SCNN Iteration: 1\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:14:32,704 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  78.62\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.051334863000192854\n",
            "SCNN Iteration: 2\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:14:57,059 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  80.06\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.11321017499903974\n",
            "SCNN Iteration: 3\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:15:25,164 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  80.79\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.174565721998988\n",
            "SCNN Iteration: 4\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:15:57,566 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  81.11\n",
            "Total number of parameters:  52479999\n",
            "Total Time:  0.23611401699963608\n",
            "SCNN Iteration: 5\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:16:33,757 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  81.4\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.30079550000027666\n",
            "SCNN Iteration: 6\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:17:14,659 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  81.72\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.36226844300108496\n",
            "SCNN Iteration: 7\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:17:59,071 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  81.93\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.42475824399934936\n",
            "SCNN Iteration: 8\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:18:47,731 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  81.92\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.4908903799996551\n",
            "SCNN Iteration: 9\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:19:40,456 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  81.92\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.5501851740009442\n",
            "SCNN Iteration: 10\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:20:37,811 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  81.9\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.6176078750013403\n",
            "SCNN Iteration: 11\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:21:39,442 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  81.9\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.6749752919986349\n",
            "SCNN Iteration: 12\n"
          ]
        },
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "WARNING:scnn:Max iterations reached before termination criterion was satisfied. Exiting optimization loop.\n",
            "\u001b[33;20m2024-05-30 19:22:41,144 - scnn - WARNING - Max iterations reached before termination criterion was satisfied. Exiting optimization loop. (iterative.py:315)\u001b[0m\n"
          ]
        },
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Achieved Test Accuracy:  81.9\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.6721668799982581\n",
            "SCNN Iteration: 13\n",
            "Achieved Test Accuracy:  81.9\n",
            "Total number of parameters:  52480000\n",
            "Total Time:  0.6776038089992653\n"
          ]
        }
      ],
      "source": [
        "test_accuracies = []\n",
        "model_training_times = []\n",
        "\n",
        "best_acc = 0\n",
        "best_model = grelu_model\n",
        "best_metrics = grelu_metrics\n",
        "\n",
        "for iter in range(1, 13+1):\n",
        "\n",
        "    print(\"SCNN Iteration:\", iter)\n",
        "\n",
        "    # Instantiate convex model and other options.\n",
        "    model = ConvexGatedReLU(G, c=y_train_SCNN_new.shape[-1])\n",
        "    lam = 5e-8\n",
        "    max_neurons = 100\n",
        "\n",
        "    G = sample_gate_vectors(123, X_train_SCNN_new.shape[-1], max_neurons)\n",
        "    metrics = Metrics(metric_freq=25, model_loss=True, train_accuracy=False, train_mse=True, test_mse=True, test_accuracy=False, neuron_sparsity=True)\n",
        "    # solver = RFISTA(model, tol=5e-6, max_iters=iter)\n",
        "    solver = RFISTA(model, max_iters=iter)\n",
        "\n",
        "    regularizer = L1(lam)\n",
        "    grelu_model, grelu_metrics = optimize_model(\n",
        "        model,\n",
        "        solver,\n",
        "        metrics,\n",
        "        X_train_SCNN_new.cpu().numpy(),\n",
        "        y_train_SCNN_new.cpu().numpy(),\n",
        "        X_test_SCNN.cpu().numpy(),\n",
        "        y_test_SCNN.cpu().numpy(),\n",
        "        # regularizer=regularizer,\n",
        "        verbose=False,\n",
        "        device='cuda',\n",
        "        # return_convex=True\n",
        "    )\n",
        "\n",
        "    model_training_times.append(grelu_metrics.time[-1])\n",
        "\n",
        "    temp_test_acc = report_test_accuracy(grelu_model)\n",
        "\n",
        "    num_params = 0\n",
        "    for p in grelu_model.get_parameters():\n",
        "        num_params += np.count_nonzero(p)\n",
        "\n",
        "    print(\"Achieved Test Accuracy: \", temp_test_acc)\n",
        "    print(\"Total number of parameters: \", num_params)\n",
        "    print(\"Total Time: \", grelu_metrics.time[-1])\n",
        "\n",
        "\n",
        "    test_accuracies.append(temp_test_acc)\n",
        "\n",
        "    if best_acc < temp_test_acc:\n",
        "        best_acc = temp_test_acc\n",
        "        best_model = grelu_model\n",
        "        best_metrics = grelu_metrics"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "-2rx_IPoXiBv",
        "outputId": "b4094cd6-3ed0-49e6-ac6f-0fc45b4cd506"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "array([0.00241856, 0.05081834, 0.11200875, 0.17727745, 0.24012243,\n",
              "       0.30110677, 0.36349919, 0.42475529, 0.42475824])"
            ]
          },
          "execution_count": 99,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "best_metrics.time"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "z6UHdUzN8YHM",
        "outputId": "def7c698-b461-4f56-bb2e-9bce0d781dbc"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "81.93\n",
            "6\n",
            "0.42475824399934936\n"
          ]
        }
      ],
      "source": [
        "print(np.max(test_accuracies))\n",
        "print(np.argmax(test_accuracies))\n",
        "print(model_training_times[np.argmax(test_accuracies)])"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "X6-o78kwMajK",
        "outputId": "276e3b79-8f2b-4745-d424-aca89d8c77c9"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Parameter Shape:  (51200, 1024)\n",
            "#Non-zero entries:  52428800\n",
            "Total number of parameters:  52428800\n",
            "\n",
            "\n",
            "Parameter Shape:  (512, 51200)\n",
            "#Non-zero entries:  51200\n",
            "Total number of parameters:  26214400\n",
            "\n",
            "\n"
          ]
        }
      ],
      "source": [
        "for p in best_model.get_parameters():\n",
        "    print(\"Parameter Shape: \", p.shape)\n",
        "    print(\"#Non-zero entries: \", np.count_nonzero(p))\n",
        "    print(\"Total number of parameters: \", np.size(p))\n",
        "    print(\"\\n\")"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "V_JhyyOjBVDw",
        "outputId": "7bd1cee9-c72d-43e0-e270-d36e58d70d12"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "81.93"
            ]
          },
          "execution_count": 102,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "report_test_accuracy(best_model)"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "TMhzRdTPSrjb"
      },
      "source": [
        "## Experiment with Sparse Coding on Non-Convex Model"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "F3sSKI6K41UM"
      },
      "outputs": [],
      "source": [
        "# import copy\n",
        "# best_model = copy.deepcopy(grelu_model)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "1HRD0TlIS80M",
        "outputId": "714fd91b-73a2-4a13-adf8-7b0a2ed9c85e"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "torch.Size([51200, 1024])\n",
            "torch.Size([512, 51200])\n",
            "torch.Size([1024, 51200])\n"
          ]
        }
      ],
      "source": [
        "W1, W2 = best_model.get_parameters()\n",
        "gates = best_model.G\n",
        "\n",
        "W1 = torch.Tensor(W1).to(device)\n",
        "W2 = torch.Tensor(W2).to(device)\n",
        "gates = torch.Tensor(gates).to(device)\n",
        "\n",
        "print(W1.shape)\n",
        "print(W2.shape)\n",
        "print(gates.shape)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "zPwtIdE0wEtJ",
        "outputId": "5a9f9a9d-2e16-4a9d-f1f2-13cec17c4db0"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "tensor([[-0.0741, -0.0201,  0.0806,  ..., -0.0280,  0.3806,  0.5281],\n",
            "        [-0.0339, -0.0153, -0.0433,  ..., -0.0203,  0.6882,  0.4908],\n",
            "        [-0.0611, -0.0297, -0.0281,  ...,  0.0160,  0.4207,  0.3955],\n",
            "        ...,\n",
            "        [ 0.0086,  0.8051,  0.6742,  ..., -0.0144, -0.0500, -0.0740],\n",
            "        [ 0.2427,  0.4299,  0.2098,  ...,  0.1244, -0.0056, -0.0276],\n",
            "        [-0.0083,  0.8928,  0.5527,  ...,  0.0230, -0.0315, -0.0465]],\n",
            "       device='cuda:0')\n",
            "torch.Size([100, 512])\n",
            "torch.Size([100, 51200])\n"
          ]
        }
      ],
      "source": [
        "gated_activations, predictions = gated_relu_prediction(X_train_SCNN_new, gates, W1, W2)\n",
        "print(predictions)\n",
        "print(predictions.shape)\n",
        "print(gated_activations.shape)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "JIxxW-cadyab",
        "outputId": "9e1cb1a0-1ec6-4dc1-d87c-25aac7f98820"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "<torch.autograd.grad_mode.set_grad_enabled at 0x7d8082adfd30>"
            ]
          },
          "execution_count": 105,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "torch.set_grad_enabled(True)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "6BvFmLTZDrr-"
      },
      "outputs": [],
      "source": [
        "# debugging_data = {}\n",
        "\n",
        "# debugging_data['X_train_SCNN_new'] = X_train_SCNN_new_orig\n",
        "# debugging_data['y_train_SCNN_new'] = y_train_SCNN_new_orig\n",
        "# debugging_data['W1'] = W1\n",
        "# debugging_data['W2'] = W2\n",
        "# debugging_data['gates'] = gates\n",
        "# debugging_data['gated_activations'] = gated_activations\n",
        "\n",
        "# torch.save(debugging_data, '/content/gdrive/My Drive/From Non-Convex to Convex/Fine-tune Pre-trained Resnet18/10 Class Experiments/debugging_scnn_numtrain{}_gaussian{}'.format(num_train_samples, num_train_gaussian_samples))"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "yzz7sXWzThHT"
      },
      "source": [
        "### Compute \\beta using adelie"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "g5yl_X3VThHZ",
        "outputId": "fd540c5f-6313-4ea8-bbde-154f12e4c769"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Requirement already satisfied: adelie in /usr/local/lib/python3.10/dist-packages (1.1.34)\n",
            "Requirement already satisfied: numpy<2,>=1.24.2 in /usr/local/lib/python3.10/dist-packages (from adelie) (1.25.2)\n",
            "Requirement already satisfied: scipy<2,>=1.10.1 in /usr/local/lib/python3.10/dist-packages (from adelie) (1.11.4)\n",
            "Requirement already satisfied: matplotlib<4,>=3.7.0 in /usr/local/lib/python3.10/dist-packages (from adelie) (3.7.1)\n",
            "Requirement already satisfied: ipython<9,>=8.17.2 in /usr/local/lib/python3.10/dist-packages (from adelie) (8.24.0)\n",
            "Requirement already satisfied: decorator in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (4.4.2)\n",
            "Requirement already satisfied: jedi>=0.16 in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (0.19.1)\n",
            "Requirement already satisfied: matplotlib-inline in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (0.1.7)\n",
            "Requirement already satisfied: prompt-toolkit<3.1.0,>=3.0.41 in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (3.0.43)\n",
            "Requirement already satisfied: pygments>=2.4.0 in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (2.16.1)\n",
            "Requirement already satisfied: stack-data in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (0.6.3)\n",
            "Requirement already satisfied: traitlets>=5.13.0 in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (5.14.3)\n",
            "Requirement already satisfied: exceptiongroup in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (1.2.1)\n",
            "Requirement already satisfied: typing-extensions>=4.6 in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (4.11.0)\n",
            "Requirement already satisfied: pexpect>4.3 in /usr/local/lib/python3.10/dist-packages (from ipython<9,>=8.17.2->adelie) (4.9.0)\n",
            "Requirement already satisfied: contourpy>=1.0.1 in /usr/local/lib/python3.10/dist-packages (from matplotlib<4,>=3.7.0->adelie) (1.2.1)\n",
            "Requirement already satisfied: cycler>=0.10 in /usr/local/lib/python3.10/dist-packages (from matplotlib<4,>=3.7.0->adelie) (0.12.1)\n",
            "Requirement already satisfied: fonttools>=4.22.0 in /usr/local/lib/python3.10/dist-packages (from matplotlib<4,>=3.7.0->adelie) (4.51.0)\n",
            "Requirement already satisfied: kiwisolver>=1.0.1 in /usr/local/lib/python3.10/dist-packages (from matplotlib<4,>=3.7.0->adelie) (1.4.5)\n",
            "Requirement already satisfied: packaging>=20.0 in /usr/local/lib/python3.10/dist-packages (from matplotlib<4,>=3.7.0->adelie) (24.0)\n",
            "Requirement already satisfied: pillow>=6.2.0 in /usr/local/lib/python3.10/dist-packages (from matplotlib<4,>=3.7.0->adelie) (9.4.0)\n",
            "Requirement already satisfied: pyparsing>=2.3.1 in /usr/local/lib/python3.10/dist-packages (from matplotlib<4,>=3.7.0->adelie) (3.1.2)\n",
            "Requirement already satisfied: python-dateutil>=2.7 in /usr/local/lib/python3.10/dist-packages (from matplotlib<4,>=3.7.0->adelie) (2.8.2)\n",
            "Requirement already satisfied: parso<0.9.0,>=0.8.3 in /usr/local/lib/python3.10/dist-packages (from jedi>=0.16->ipython<9,>=8.17.2->adelie) (0.8.4)\n",
            "Requirement already satisfied: ptyprocess>=0.5 in /usr/local/lib/python3.10/dist-packages (from pexpect>4.3->ipython<9,>=8.17.2->adelie) (0.7.0)\n",
            "Requirement already satisfied: wcwidth in /usr/local/lib/python3.10/dist-packages (from prompt-toolkit<3.1.0,>=3.0.41->ipython<9,>=8.17.2->adelie) (0.2.13)\n",
            "Requirement already satisfied: six>=1.5 in /usr/local/lib/python3.10/dist-packages (from python-dateutil>=2.7->matplotlib<4,>=3.7.0->adelie) (1.16.0)\n",
            "Requirement already satisfied: executing>=1.2.0 in /usr/local/lib/python3.10/dist-packages (from stack-data->ipython<9,>=8.17.2->adelie) (2.0.1)\n",
            "Requirement already satisfied: asttokens>=2.1.0 in /usr/local/lib/python3.10/dist-packages (from stack-data->ipython<9,>=8.17.2->adelie) (2.4.1)\n",
            "Requirement already satisfied: pure-eval in /usr/local/lib/python3.10/dist-packages (from stack-data->ipython<9,>=8.17.2->adelie) (0.2.2)\n"
          ]
        }
      ],
      "source": [
        "!yes | pip install --no-cache-dir adelie"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "wxRrjxGUThHa"
      },
      "outputs": [],
      "source": [
        "import adelie as ad"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "Mm1ppW9agks7",
        "outputId": "59f5cc15-8bc8-4ded-fdd5-020b7d682294"
      },
      "outputs": [
        {
          "name": "stderr",
          "output_type": "stream",
          "text": [
            "/usr/local/lib/python3.10/dist-packages/adelie/matrix.py:427: UserWarning: Detected matrix to be C-contiguous. Performance may improve with F-contiguous matrix.\n",
            "  warnings.warn(\n",
            "100%|\u001b[1;32m█\u001b[0m\u001b[1;32m█\u001b[0m\u001b[1;32m█\u001b[0m\u001b[1;32m█\u001b[0m\u001b[1;32m█\u001b[0m\u001b[1;32m█\u001b[0m\u001b[1;32m█\u001b[0m\u001b[1;32m█\u001b[0m\u001b[1;32m█\u001b[0m\u001b[1;32m█\u001b[0m| 100/100 [00:01:46<00:00:00, 0.93it/s] [dev:100.0%]\n"
          ]
        }
      ],
      "source": [
        "state = ad.grpnet(\n",
        "    X=gated_activations.cpu().numpy().astype(np.float64),\n",
        "    glm=ad.glm.multigaussian(y=y_train_SCNN_new.cpu().numpy().astype(np.float64)),\n",
        "    intercept=False,\n",
        "    n_threads=2,\n",
        "    groups=\"grouped\",\n",
        "    # lmda_path_size=50,\n",
        "    min_ratio=1e-3,\n",
        "    # lmda_path=np.logspace(-5, -8, 100),\n",
        "    early_exit=False\n",
        ")"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "EZRdlCO0hcqZ",
        "outputId": "8b0f287b-e35e-4661-d588-714282f76344"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "[6.39390075e-05 5.96297324e-05 5.56108880e-05 5.18629003e-05\n",
            " 4.83675144e-05 4.51077058e-05 4.20675974e-05 3.92323821e-05\n",
            " 3.65882509e-05 3.41223252e-05 3.18225947e-05 2.96778583e-05\n",
            " 2.76776700e-05 2.58122877e-05 2.40726259e-05 2.24502115e-05\n",
            " 2.09371423e-05 1.95260490e-05 1.82100587e-05 1.69827617e-05\n",
            " 1.58381804e-05 1.47707400e-05 1.37752416e-05 1.28468364e-05\n",
            " 1.19810026e-05 1.11735231e-05 1.04204651e-05 9.71816061e-06\n",
            " 9.06318911e-06 8.45236050e-06 7.88269969e-06 7.35143211e-06\n",
            " 6.85597019e-06 6.39390075e-06 5.96297324e-06 5.56108880e-06\n",
            " 5.18629003e-06 4.83675144e-06 4.51077058e-06 4.20675974e-06\n",
            " 3.92323821e-06 3.65882509e-06 3.41223252e-06 3.18225947e-06\n",
            " 2.96778583e-06 2.76776700e-06 2.58122877e-06 2.40726259e-06\n",
            " 2.24502115e-06 2.09371423e-06 1.95260490e-06 1.82100587e-06\n",
            " 1.69827617e-06 1.58381804e-06 1.47707400e-06 1.37752416e-06\n",
            " 1.28468364e-06 1.19810026e-06 1.11735231e-06 1.04204651e-06\n",
            " 9.71816061e-07 9.06318911e-07 8.45236050e-07 7.88269969e-07\n",
            " 7.35143211e-07 6.85597019e-07 6.39390075e-07 5.96297324e-07\n",
            " 5.56108880e-07 5.18629003e-07 4.83675144e-07 4.51077058e-07\n",
            " 4.20675974e-07 3.92323821e-07 3.65882509e-07 3.41223252e-07\n",
            " 3.18225947e-07 2.96778583e-07 2.76776700e-07 2.58122877e-07\n",
            " 2.40726259e-07 2.24502115e-07 2.09371423e-07 1.95260490e-07\n",
            " 1.82100587e-07 1.69827617e-07 1.58381804e-07 1.47707400e-07\n",
            " 1.37752416e-07 1.28468364e-07 1.19810026e-07 1.11735231e-07\n",
            " 1.04204651e-07 9.71816061e-08 9.06318911e-08 8.45236050e-08\n",
            " 7.88269969e-08 7.35143211e-08 6.85597019e-08 6.39390075e-08]\n"
          ]
        }
      ],
      "source": [
        "lmda_max = state.lmdas\n",
        "print(state.lmdas)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "N9zfEciOGS-Z",
        "outputId": "5e9345b3-77d3-4b92-9f4d-e2929e4d4286"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "(81.93, 0.0019938938785344362, 0.8845712542533875)\n",
            "52428800\n"
          ]
        }
      ],
      "source": [
        "print(report_test_accuracy_W2(W2.T))\n",
        "print(count_effective_entries(W1, W2, 512, 100))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "uXidosPjgTB9",
        "outputId": "f71498b8-508c-43ca-a464-f8af1f6d3beb"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "100"
            ]
          },
          "execution_count": 112,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "len(state.lmdas)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "oqExOFLoGN25",
        "outputId": "0fc25a15-913a-40f7-d434-90374e44a084"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "99 (81.19, 9.161977686744649e-06, 1.156794786453247) 813056\n",
            "98 (81.17, 1.0514170753594954e-05, 1.1561691761016846) 808960\n",
            "97 (81.17, 1.2064673683198635e-05, 1.1555016040802002) 807936\n",
            "96 (81.16, 1.3841213331033941e-05, 1.1547878980636597) 800768\n",
            "95 (81.14, 1.5876785255386494e-05, 1.1540260314941406) 801792\n",
            "94 (81.14, 1.8208973415312357e-05, 1.1532135009765625) 798720\n",
            "93 (81.14, 2.0879942894680426e-05, 1.1523501873016357) 796672\n",
            "92 (81.16, 2.3937487640068866e-05, 1.1514558792114258) 794624\n",
            "91 (81.15, 2.7437279641162604e-05, 1.1505252122879028) 791552\n",
            "90 (81.13, 3.144276706734672e-05, 1.1495466232299805) 788480\n",
            "89 (81.13, 3.6022174754180014e-05, 1.1485143899917603) 785408\n",
            "88 (81.11, 4.1255931137129664e-05, 1.147402286529541) 780288\n",
            "87 (81.13, 4.723730307887308e-05, 1.1462008953094482) 777216\n",
            "86 (81.16, 5.407040225691162e-05, 1.144948959350586) 774144\n",
            "85 (81.12, 6.186862447066233e-05, 1.143614411354065) 769024\n",
            "84 (81.12, 7.07737126504071e-05, 1.1421715021133423) 764928\n",
            "83 (81.11, 8.09203993412666e-05, 1.1405797004699707) 758784\n",
            "82 (81.1, 9.247202251572162e-05, 1.1388602256774902) 751616\n",
            "81 (81.09, 0.00010562027455307543, 1.1370128393173218) 743424\n",
            "80 (81.08, 0.00012057193089276552, 1.1350536346435547) 733184\n",
            "79 (81.1, 0.00013757192937191576, 1.1329705715179443) 729088\n",
            "78 (81.07, 0.00015687603445257992, 1.130853295326233) 720896\n",
            "77 (81.07, 0.00017876990023069084, 1.1286823749542236) 717824\n",
            "76 (81.06, 0.0002035612560575828, 1.1264175176620483) 705536\n",
            "75 (81.05, 0.00023160118144005537, 1.12405526638031) 695296\n",
            "74 (81.02, 0.0002632146351970732, 1.1214302778244019) 678912\n",
            "73 (80.97, 0.00029889922006987035, 1.1186026334762573) 671744\n",
            "72 (81.02, 0.00033913159859366715, 1.1155595779418945) 664576\n",
            "71 (81.05, 0.0003843631420750171, 1.112614393234253) 648192\n",
            "70 (81.02, 0.00043515529250726104, 1.1096874475479126) 635904\n",
            "69 (81.0, 0.0004921681247651577, 1.1067525148391724) 626688\n",
            "68 (80.93, 0.0005560858407989144, 1.1037408113479614) 618496\n",
            "67 (80.89, 0.000627389526925981, 1.0998369455337524) 608256\n",
            "66 (80.87, 0.0007064169622026384, 1.0937241315841675) 593920\n",
            "65 (80.89, 0.0007941914955154061, 1.0872130393981934) 584704\n",
            "64 (80.86, 0.0008913354249671102, 1.0805178880691528) 566272\n",
            "63 (80.82, 0.0009980470640584826, 1.0734410285949707) 547840\n",
            "62 (80.68, 0.0011156783439218998, 1.0661605596542358) 533504\n",
            "61 (80.6, 0.001243454753421247, 1.0586352348327637) 506880\n",
            "60 (80.57, 0.0013825856149196625, 1.051561713218689) 499712\n",
            "59 (80.58, 0.0015345000429078937, 1.0452779531478882) 482304\n",
            "58 (80.48, 0.001699729822576046, 1.0388903617858887) 467968\n",
            "57 (80.44, 0.001877494971267879, 1.0323152542114258) 448512\n",
            "56 (80.31, 0.0020686236675828695, 1.0269230604171753) 431104\n"
          ]
        }
      ],
      "source": [
        "for j in range(100-1, 50, -1):\n",
        "    sparse_W2 = torch.Tensor(state.betas[j].toarray()[0].reshape(W2.T.shape)).to(device)\n",
        "    print(j, report_test_accuracy_W2(sparse_W2), count_effective_entries(W1, sparse_W2.T, 512, 100))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "qDMmV3gXS2Fg"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "import torch\n",
        "from scipy.optimize import lsq_linear\n",
        "import time\n",
        "\n",
        "# Initialize matrices and parameters\n",
        "sparse_W2 = torch.Tensor(state.betas[5].toarray()[0].reshape(W2.T.shape)).to(device)\n",
        "\n",
        "adelie_metric = report_test_accuracy_W2(sparse_W2)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "NVTJewrqUjic",
        "outputId": "7eabd667-2bdb-4b2b-c865-faf5d0489fa1"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Regularization Path Index:  91\n",
            "Non-Zero Entries in W2:  171520\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (74.25, 0.0003675807674881071, 0.12104099988937378)\n"
          ]
        }
      ],
      "source": [
        "print(\"Regularization Path Index: \", j)\n",
        "# print(\"Lambda at Index \", state.lmdas[j])\n",
        "print(\"Non-Zero Entries in W2: \", torch.count_nonzero(sparse_W2).item())\n",
        "print(\"Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE: \", adelie_metric)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "CSDkwuRNWxTA"
      },
      "outputs": [],
      "source": [
        "def regularized_polishing(sparse_W2, lmda_range):\n",
        "    polish_W2 = torch.zeros(sparse_W2.shape).to(device)\n",
        "\n",
        "    A = gated_activations\n",
        "    X = sparse_W2\n",
        "    B = y_train_SCNN_new\n",
        "\n",
        "    lsq_time = 0\n",
        "    best_acc = 0\n",
        "    best_lmda = 0\n",
        "\n",
        "    for k, lambda_reg in enumerate(lmda_range):\n",
        "\n",
        "        for col_idx in range(B.shape[1]):\n",
        "\n",
        "            non_zero_indices = torch.where(X[:, col_idx] != 0)[0]\n",
        "            if len(non_zero_indices) == 0:\n",
        "                continue\n",
        "\n",
        "            A_prime = A[:, non_zero_indices]\n",
        "            b = B[:, col_idx]\n",
        "\n",
        "            # Add regularization\n",
        "            reg_identity = np.sqrt(lambda_reg) * np.identity(len(non_zero_indices))\n",
        "            A_prime_reg = torch.vstack([A_prime, torch.Tensor(reg_identity).to(device)])\n",
        "            b_reg = torch.hstack([b, torch.zeros(len(non_zero_indices)).to(device)])\n",
        "\n",
        "            start_time = time.time()\n",
        "            x = torch.linalg.lstsq(A_prime_reg, b_reg).solution\n",
        "            lsq_time += time.time() - start_time\n",
        "\n",
        "            polish_W2[non_zero_indices, col_idx] = x\n",
        "\n",
        "        polish_metric = report_test_accuracy_W2(polish_W2.to(device))\n",
        "\n",
        "        if best_acc < polish_metric[0]:\n",
        "            best_acc = polish_metric[0]\n",
        "            best_metric = polish_metric\n",
        "            best_lmda = lambda_reg\n",
        "\n",
        "        print(\"k: \", k, \", At lmda=\", lambda_reg,\", Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE: \", polish_metric)\n",
        "\n",
        "    return best_lmda, best_metric"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "SHSIOLil3r3l",
        "outputId": "c9244864-93bc-4e36-f1ab-2c2f1d302e60"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "k:  0 , At lmda= 0.09999999999999999 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 5.187255737837404e-06, 0.11778374761343002)\n",
            "k:  1 , At lmda= 0.10985411419875583 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 6.2201374930737074e-06, 0.11778418719768524)\n",
            "k:  2 , At lmda= 0.12067926406393287 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 7.454358183167642e-06, 0.11778461933135986)\n",
            "k:  3 , At lmda= 0.1325711365590109 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.31, 8.92771640792489e-06, 0.11778515577316284)\n",
            "k:  4 , At lmda= 0.14563484775012436 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 1.0684927474358119e-05, 0.11778567731380463)\n",
            "k:  5 , At lmda= 0.15998587196060582 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 1.277865612792084e-05, 0.11778630316257477)\n",
            "k:  6 , At lmda= 0.1757510624854792 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 1.527028507553041e-05, 0.11778698116540909)\n",
            "k:  7 , At lmda= 0.193069772888325 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.29, 1.8232050933875144e-05, 0.11778776347637177)\n",
            "k:  8 , At lmda= 0.21209508879201902 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.29, 2.1747979189967737e-05, 0.11778867244720459)\n",
            "k:  9 , At lmda= 0.2329951810515372 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.29, 2.5915978767443448e-05, 0.1177896186709404)\n",
            "k:  10 , At lmda= 0.2559547922699536 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.31, 3.0849634640617296e-05, 0.11779072135686874)\n",
            "k:  11 , At lmda= 0.281176869797423 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.31, 3.6680296034319326e-05, 0.11779194325208664)\n",
            "k:  12 , At lmda= 0.3088843596477481 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.31, 4.355951750767417e-05, 0.11779330670833588)\n",
            "k:  13 , At lmda= 0.3393221771895328 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.29, 5.1661187171703205e-05, 0.11779484897851944)\n",
            "k:  14 , At lmda= 0.372759372031494 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 6.118462624726817e-05, 0.11779657006263733)\n",
            "k:  15 , At lmda= 0.4094915062380424 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.29, 7.23565899534151e-05, 0.11779852211475372)\n",
            "k:  16 , At lmda= 0.4498432668969445 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.31, 8.543531293980777e-05, 0.11780069023370743)\n",
            "k:  17 , At lmda= 0.49417133613238345 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 0.0001007121245493181, 0.11780314892530441)\n",
            "k:  18 , At lmda= 0.5428675439323859 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.31, 0.00011851558519992977, 0.11780592054128647)\n",
            "k:  19 , At lmda= 0.5963623316594643 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.34, 0.0001392135163769126, 0.1178090050816536)\n",
            "k:  20 , At lmda= 0.6551285568595507 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.31, 0.00016321729344781488, 0.11781249940395355)\n",
            "k:  21 , At lmda= 0.7196856730011519 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.33, 0.0001909834099933505, 0.11781642585992813)\n",
            "k:  22 , At lmda= 0.7906043210907697 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.35, 0.0002230174286523834, 0.11782083660364151)\n",
            "k:  23 , At lmda= 0.8685113737513525 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.36, 0.00025987683329731226, 0.11782577633857727)\n",
            "k:  24 , At lmda= 0.9540954763499939 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.37, 0.0003021726734004915, 0.11783135682344437)\n",
            "k:  25 , At lmda= 1.0481131341546857 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.34, 0.0003505741769913584, 0.11783759295940399)\n",
            "k:  26 , At lmda= 1.1513953993264467 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.34, 0.00040580949280411005, 0.11784462630748749)\n",
            "k:  27 , At lmda= 1.2648552168552958 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.33, 0.00046867161290720105, 0.11785247176885605)\n",
            "k:  28 , At lmda= 1.3894954943731375 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 0.0005400190129876137, 0.11786127835512161)\n",
            "k:  29 , At lmda= 1.5264179671752334 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 0.0006207818514667451, 0.11787114292383194)\n",
            "k:  30 , At lmda= 1.6768329368110073 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.29, 0.0007119634537957609, 0.11788217723369598)\n",
            "k:  31 , At lmda= 1.8420699693267153 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.29, 0.0008146515465341508, 0.1178944781422615)\n",
            "k:  32 , At lmda= 2.0235896477251565 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.31, 0.0009300195379182696, 0.11790824681520462)\n",
            "k:  33 , At lmda= 2.2229964825261943 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 0.0010593390325084329, 0.11792358011007309)\n",
            "k:  34 , At lmda= 2.4420530945486507 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.33, 0.0012039884459227324, 0.11794067174196243)\n",
            "k:  35 , At lmda= 2.6826957952797246 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.34, 0.0013654663925990462, 0.11795970052480698)\n",
            "k:  36 , At lmda= 2.94705170255181 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.35, 0.0015454096719622612, 0.11798088252544403)\n",
            "k:  37 , At lmda= 3.2374575428176433 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.31, 0.0017456042114645243, 0.11800440400838852)\n",
            "k:  38 , At lmda= 3.5564803062231283 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.32, 0.001968013821169734, 0.11803048849105835)\n",
            "k:  39 , At lmda= 3.906939937054615 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.34, 0.0022147984709590673, 0.11805945634841919)\n",
            "k:  40 , At lmda= 4.291934260128776 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.29, 0.0024883404839783907, 0.11809152364730835)\n",
            "k:  41 , At lmda= 4.714866363457392 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.3, 0.002791275968775153, 0.11812704801559448)\n",
            "k:  42 , At lmda= 5.17947467923121 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.22, 0.0031265143770724535, 0.11816630512475967)\n",
            "k:  43 , At lmda= 5.6898660290182965 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.26, 0.0034972825087606907, 0.11820966750383377)\n",
            "k:  44 , At lmda= 6.250551925273969 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.25, 0.003907138481736183, 0.11825749278068542)\n",
            "k:  45 , At lmda= 6.866488450042998 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.2, 0.004360005259513855, 0.11831017583608627)\n",
            "k:  46 , At lmda= 7.543120063354615 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.17, 0.00486019067466259, 0.11836816370487213)\n",
            "k:  47 , At lmda= 8.286427728546842 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.14, 0.005412401631474495, 0.11843185871839523)\n",
            "k:  48 , At lmda= 9.102981779915218 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.12, 0.006021758075803518, 0.11850173026323318)\n",
            "k:  49 , At lmda= 10.0 , Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (76.04, 0.006693764589726925, 0.11857825517654419)\n"
          ]
        }
      ],
      "source": [
        "lmda_range = np.logspace(-1, 1, 50)\n",
        "# lmda_range = np.linspace(1, 100, 100)\n",
        "best_lmda_reg, best_polish_metric = regularized_polishing(sparse_W2, lmda_range)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "FHbUpk85SavC",
        "outputId": "64b7b9fd-12ce-440c-b11c-f9f1df21b59e"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Regularization Path Index:  27\n",
            "Lambda at Index  3.4543500413045654e-05\n",
            "Non-Zero Entries in W2:  359424\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (83.79, 0.0036227128002792597, 0.03769033029675484)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (83.45, 0.00044954760232940316, 0.04145615175366402)\n",
            "\n",
            "\n"
          ]
        }
      ],
      "source": [
        "polish_W2 = torch.zeros(sparse_W2.shape)\n",
        "\n",
        "A = gated_activations.cpu().numpy()\n",
        "X = sparse_W2.cpu().numpy()\n",
        "B = y_train_SCNN_new.cpu().numpy()\n",
        "\n",
        "lsq_time = 0\n",
        "lambda_reg = 0.9 # Regularization parameter\n",
        "\n",
        "for col_idx in range(B.shape[1]):\n",
        "\n",
        "    non_zero_indices = np.where(X[:, col_idx] != 0)[0]\n",
        "    if len(non_zero_indices) == 0:\n",
        "        continue\n",
        "\n",
        "    A_prime = A[:, non_zero_indices]\n",
        "    b = B[:, col_idx]\n",
        "\n",
        "    # Add regularization\n",
        "    reg_identity = np.sqrt(lambda_reg) * np.identity(len(non_zero_indices))\n",
        "    A_prime_reg = np.vstack([A_prime, reg_identity])\n",
        "    b_reg = np.hstack([b, np.zeros(len(non_zero_indices))])\n",
        "\n",
        "    start_time = time.time()\n",
        "    x = lsq_linear(A_prime_reg, b_reg)['x']\n",
        "    lsq_time += time.time() - start_time\n",
        "\n",
        "    polish_W2[non_zero_indices, col_idx] = torch.Tensor(x)\n",
        "\n",
        "polish_metric = report_test_accuracy_W2(polish_W2.to(device))\n",
        "\n",
        "print(\"Regularization Path Index: \", j)\n",
        "print(\"Lambda at Index \", state.lmdas[j])\n",
        "print(\"Non-Zero Entries in W2: \", torch.count_nonzero(sparse_W2).item())\n",
        "print(\"Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE: \", adelie_metric)\n",
        "print(\"Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE: \", polish_metric)\n",
        "print(\"\\n\")\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "NLUjtzhUHdVA"
      },
      "outputs": [],
      "source": [
        "import torch.nn as nn\n",
        "import torch.optim as optim\n",
        "\n",
        "class SimpleLinearModel(nn.Module):\n",
        "    def __init__(self, input_dim, output_dim):\n",
        "        super(SimpleLinearModel, self).__init__()\n",
        "        self.linear = nn.Linear(input_dim, output_dim, bias=False)\n",
        "\n",
        "    def forward(self, x):\n",
        "        return self.linear(x)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "Ns1ah8mYMzW9",
        "outputId": "eafab69d-447e-4a11-e597-31483b3ba45b"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "0\n",
            "1\n",
            "2\n",
            "3\n",
            "4\n",
            "5\n",
            "6\n",
            "7\n",
            "8\n",
            "9\n",
            "10\n",
            "11\n",
            "12\n",
            "13\n",
            "14\n",
            "15\n",
            "16\n",
            "17\n",
            "18\n",
            "19\n",
            "20\n",
            "21\n",
            "22\n",
            "23\n",
            "24\n",
            "25\n",
            "26\n",
            "27\n",
            "28\n",
            "29\n",
            "30\n",
            "31\n",
            "32\n",
            "33\n",
            "34\n",
            "35\n",
            "36\n",
            "37\n",
            "38\n",
            "39\n",
            "40\n",
            "41\n",
            "42\n",
            "43\n",
            "44\n",
            "45\n",
            "46\n",
            "47\n",
            "48\n",
            "49\n",
            "50\n",
            "51\n",
            "52\n",
            "53\n",
            "54\n",
            "55\n",
            "56\n",
            "57\n",
            "58\n",
            "59\n",
            "60\n",
            "61\n",
            "62\n",
            "63\n",
            "64\n",
            "65\n",
            "66\n",
            "67\n",
            "68\n",
            "69\n",
            "70\n",
            "71\n",
            "72\n",
            "73\n",
            "74\n",
            "75\n",
            "76\n",
            "77\n",
            "78\n",
            "79\n",
            "80\n",
            "81\n",
            "82\n",
            "83\n",
            "84\n",
            "85\n",
            "86\n",
            "87\n",
            "88\n",
            "89\n",
            "90\n",
            "91\n",
            "92\n",
            "93\n",
            "94\n",
            "95\n",
            "96\n",
            "97\n",
            "98\n",
            "99\n",
            "100\n",
            "101\n",
            "102\n",
            "103\n",
            "104\n",
            "105\n",
            "106\n",
            "107\n",
            "108\n",
            "109\n",
            "110\n",
            "111\n",
            "112\n",
            "113\n",
            "114\n",
            "115\n",
            "116\n",
            "117\n",
            "118\n",
            "119\n",
            "120\n",
            "121\n",
            "122\n",
            "123\n",
            "124\n",
            "125\n",
            "126\n",
            "127\n",
            "128\n",
            "129\n",
            "130\n",
            "131\n",
            "132\n",
            "133\n",
            "134\n",
            "135\n",
            "136\n",
            "137\n",
            "138\n",
            "139\n",
            "140\n",
            "141\n",
            "142\n",
            "143\n",
            "144\n",
            "145\n",
            "146\n",
            "147\n",
            "148\n",
            "149\n",
            "150\n",
            "151\n",
            "152\n",
            "153\n",
            "154\n",
            "155\n",
            "156\n",
            "157\n",
            "158\n",
            "159\n",
            "160\n",
            "161\n",
            "162\n",
            "163\n",
            "164\n",
            "165\n",
            "166\n",
            "167\n",
            "168\n",
            "169\n",
            "170\n",
            "171\n",
            "172\n",
            "173\n",
            "174\n",
            "175\n",
            "176\n",
            "177\n",
            "178\n",
            "179\n",
            "180\n",
            "181\n",
            "182\n",
            "183\n",
            "184\n",
            "185\n",
            "186\n",
            "187\n",
            "188\n",
            "189\n",
            "190\n",
            "191\n",
            "192\n",
            "193\n",
            "194\n",
            "195\n",
            "196\n",
            "197\n",
            "198\n",
            "199\n",
            "200\n",
            "201\n",
            "202\n",
            "203\n",
            "204\n",
            "205\n",
            "206\n",
            "207\n",
            "208\n",
            "209\n",
            "210\n",
            "211\n",
            "212\n",
            "213\n",
            "214\n",
            "215\n",
            "216\n",
            "217\n",
            "218\n",
            "219\n",
            "220\n",
            "221\n",
            "222\n",
            "223\n",
            "224\n",
            "225\n",
            "226\n",
            "227\n",
            "228\n",
            "229\n",
            "230\n",
            "231\n",
            "232\n",
            "233\n",
            "234\n",
            "235\n",
            "236\n",
            "237\n",
            "238\n",
            "239\n",
            "240\n",
            "241\n",
            "242\n",
            "243\n",
            "244\n",
            "245\n",
            "246\n",
            "247\n",
            "248\n",
            "249\n",
            "250\n",
            "251\n",
            "252\n",
            "253\n",
            "254\n",
            "255\n",
            "256\n",
            "257\n",
            "258\n",
            "259\n",
            "260\n",
            "261\n",
            "262\n",
            "263\n",
            "264\n",
            "265\n",
            "266\n",
            "267\n",
            "268\n",
            "269\n",
            "270\n",
            "271\n",
            "272\n",
            "273\n",
            "274\n",
            "275\n",
            "276\n",
            "277\n",
            "278\n",
            "279\n",
            "280\n",
            "281\n",
            "282\n",
            "283\n",
            "284\n",
            "285\n",
            "286\n",
            "287\n",
            "288\n",
            "289\n",
            "290\n",
            "291\n",
            "292\n",
            "293\n",
            "294\n",
            "295\n",
            "296\n",
            "297\n",
            "298\n",
            "299\n",
            "300\n",
            "301\n",
            "302\n",
            "303\n",
            "304\n",
            "305\n",
            "306\n",
            "307\n",
            "308\n",
            "309\n",
            "310\n",
            "311\n",
            "312\n",
            "313\n",
            "314\n",
            "315\n",
            "316\n",
            "317\n",
            "318\n",
            "319\n",
            "320\n",
            "321\n",
            "322\n",
            "323\n",
            "324\n",
            "325\n",
            "326\n",
            "327\n",
            "328\n",
            "329\n",
            "330\n",
            "331\n",
            "332\n",
            "333\n",
            "334\n",
            "335\n",
            "336\n",
            "337\n",
            "338\n",
            "339\n",
            "340\n",
            "341\n",
            "342\n",
            "343\n",
            "344\n",
            "345\n",
            "346\n",
            "347\n",
            "348\n",
            "349\n",
            "350\n",
            "351\n",
            "352\n",
            "353\n",
            "354\n",
            "355\n",
            "356\n",
            "357\n",
            "358\n",
            "359\n",
            "360\n",
            "361\n",
            "362\n",
            "363\n",
            "364\n",
            "365\n",
            "366\n",
            "367\n",
            "368\n",
            "369\n",
            "370\n",
            "371\n",
            "372\n",
            "373\n",
            "374\n",
            "375\n",
            "376\n",
            "377\n",
            "378\n",
            "379\n",
            "380\n",
            "381\n",
            "382\n",
            "383\n",
            "384\n",
            "385\n",
            "386\n",
            "387\n",
            "388\n",
            "389\n",
            "390\n",
            "391\n",
            "392\n",
            "393\n",
            "394\n",
            "395\n",
            "396\n",
            "397\n",
            "398\n",
            "399\n",
            "400\n",
            "401\n",
            "402\n",
            "403\n",
            "404\n",
            "405\n",
            "406\n",
            "407\n",
            "408\n",
            "409\n",
            "410\n",
            "411\n",
            "412\n",
            "413\n",
            "414\n",
            "415\n",
            "416\n",
            "417\n",
            "418\n",
            "419\n",
            "420\n",
            "421\n",
            "422\n",
            "423\n",
            "424\n",
            "425\n",
            "426\n",
            "427\n",
            "428\n",
            "429\n",
            "430\n",
            "431\n",
            "432\n",
            "433\n",
            "434\n",
            "435\n",
            "436\n",
            "437\n",
            "438\n",
            "439\n",
            "440\n",
            "441\n",
            "442\n",
            "443\n",
            "444\n",
            "445\n",
            "446\n",
            "447\n",
            "448\n",
            "449\n",
            "450\n",
            "451\n",
            "452\n",
            "453\n",
            "454\n",
            "455\n",
            "456\n",
            "457\n",
            "458\n",
            "459\n",
            "460\n",
            "461\n",
            "462\n",
            "463\n",
            "464\n",
            "465\n",
            "466\n",
            "467\n",
            "468\n",
            "469\n",
            "470\n",
            "471\n",
            "472\n",
            "473\n",
            "474\n",
            "475\n",
            "476\n",
            "477\n",
            "478\n",
            "479\n",
            "480\n",
            "481\n",
            "482\n",
            "483\n",
            "484\n",
            "485\n",
            "486\n",
            "487\n",
            "488\n",
            "489\n",
            "490\n",
            "491\n",
            "492\n",
            "493\n",
            "494\n",
            "495\n",
            "496\n",
            "497\n",
            "498\n",
            "499\n",
            "500\n",
            "501\n",
            "502\n",
            "503\n",
            "504\n",
            "505\n",
            "506\n",
            "507\n",
            "508\n",
            "509\n",
            "510\n",
            "511\n",
            "Non-Zero Entries in W2:  119296\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (83.27, 0.013254854828119278, 0.03978775069117546)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (83.66, 0.007808864116668701, 0.04021880775690079)\n"
          ]
        }
      ],
      "source": [
        "polish_W2 = torch.zeros(sparse_W2.shape)\n",
        "\n",
        "A = gated_activations\n",
        "X = sparse_W2\n",
        "B = y_train_SCNN_new\n",
        "\n",
        "learning_rate = 1e-3\n",
        "num_epochs = 200\n",
        "\n",
        "for col_idx in range(B.shape[1]):\n",
        "\n",
        "    print(col_idx)\n",
        "\n",
        "    non_zero_indices = torch.nonzero(X[:, col_idx], as_tuple=False).squeeze()\n",
        "    if len(non_zero_indices) == 0:\n",
        "        continue\n",
        "\n",
        "    A_prime = A[:, non_zero_indices]\n",
        "    b = B[:, col_idx]\n",
        "\n",
        "    model = SimpleLinearModel(A_prime.shape[1], 1).to(device)\n",
        "    optimizer = optim.Adam(model.parameters(), lr=learning_rate)\n",
        "    criterion = nn.MSELoss()\n",
        "\n",
        "    A_prime = A_prime.to(device)\n",
        "    b = b.to(device).view(-1, 1)\n",
        "\n",
        "    for epoch in range(num_epochs):\n",
        "        model.train()\n",
        "        optimizer.zero_grad()\n",
        "        outputs = model(A_prime)\n",
        "        loss = criterion(outputs, b)\n",
        "        loss.backward()\n",
        "        optimizer.step()\n",
        "\n",
        "    with torch.no_grad():\n",
        "        polished_weights = model.linear.weight.squeeze().cpu()\n",
        "        polish_W2[non_zero_indices, col_idx] = polished_weights\n",
        "\n",
        "polish_metric = report_test_accuracy_W2(polish_W2.to(device))\n",
        "\n",
        "print(\"Non-Zero Entries in W2: \", torch.count_nonzero(sparse_W2).item())\n",
        "print(\"Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE: \", adelie_metric)\n",
        "print(\"Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE: \", polish_metric)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "CW0JhkD7OFeA",
        "outputId": "f39ee3ff-ebf7-457b-9a7b-27c2b65c9827"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "torch.Size([51200, 1024])\n",
            "torch.Size([512, 51200])\n",
            "torch.Size([51200, 512])\n"
          ]
        }
      ],
      "source": [
        "print(W1.shape)\n",
        "print(W2.shape)\n",
        "print(sparse_W2.shape)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 245
        },
        "id": "nZJP2jplN6_G",
        "outputId": "7ee100fe-3d9c-4971-a89d-b09f2f765321"
      },
      "outputs": [],
      "source": [
        "print(torch.count_nonzero(W1))\n",
        "print(torch.count_nonzero(sparse_W2))\n",
        "print(W1.shape)\n",
        "print(sparse_W2.shape)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "WYD1bp-eS4Bm",
        "outputId": "eb8299ad-d3bf-41ad-cbc2-ddd81b05388a"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Number of contributing elements in W1: 13107200\n"
          ]
        }
      ],
      "source": [
        "import torch\n",
        "\n",
        "def count_effective_entries(W1, sparse_W2, p, hidden_dim):\n",
        "    \"\"\"\n",
        "    Calculate the number of contributing elements in W1 given W1 and sparse_W2.\n",
        "\n",
        "    Args:\n",
        "    - W1 (torch.Tensor): First layer weights of shape (P, D).\n",
        "    - sparse_W2 (torch.Tensor): Sparse second layer weights of shape (C, P).\n",
        "    - p (int): Number of smaller matrices concatenated in W1 and sparse_W2.\n",
        "    - hidden_dim (int): Hidden dimension of the smaller matrices.\n",
        "\n",
        "    Returns:\n",
        "    - int: Number of contributing elements in W1.\n",
        "    \"\"\"\n",
        "\n",
        "    num_contributing_elements = 0\n",
        "\n",
        "    for i in range(p):\n",
        "        # Extract the i-th smaller matrices from W1 and sparse_W2\n",
        "        W1_i = W1[i * hidden_dim: (i + 1) * hidden_dim]\n",
        "        sparse_W2_i = sparse_W2[:, i * hidden_dim: (i + 1) * hidden_dim]\n",
        "\n",
        "        # Find the non-zero indices in the i-th smaller matrix of sparse_W2\n",
        "        non_zero_indices = sparse_W2_i.nonzero(as_tuple=False)\n",
        "\n",
        "        # Count the unique columns (corresponding to elements in W1_i)\n",
        "        unique_indices = non_zero_indices[:, 1].unique()\n",
        "\n",
        "        # Calculate the number of contributing elements in W1_i\n",
        "        num_contributing_elements += unique_indices.numel() * W1_i.size(1)\n",
        "\n",
        "    return num_contributing_elements\n",
        "\n",
        "\n",
        "num_contributing_elements = count_effective_entries(W1, W2, 512, 25)\n",
        "print(f\"Number of contributing elements in W1: {num_contributing_elements}\")\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "B6hQk7yVd5O9",
        "outputId": "3e4a6f2a-70dd-491a-f810-84cb423e9a48"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "tensor(9384774, device='cuda:0')"
            ]
          },
          "execution_count": 70,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "torch.count_nonzero(W1)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "-NHm4FGqT90C",
        "outputId": "69f99ffd-b0f1-4b2a-b336-05d0905d0823"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "tensor(52428800, device='cuda:0')"
            ]
          },
          "execution_count": 63,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "torch.count_nonzero(W1)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "wDes81sWEk3X",
        "outputId": "1273550e-c324-44f3-90a8-4a0bbf5ccc36"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Regularization Path Index:  99\n",
            "Lambda at Index  4.3935962268179724e-07\n",
            "Non-Zero Entries in W2:  193536\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.41, 0.02725278027355671, 0.02337065152823925)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.29, 0.019084082916378975, 0.02685443125665188)\n",
            "\n",
            "\n",
            "Regularization Path Index:  98\n",
            "Lambda at Index  4.6028006186117416e-07\n",
            "Non-Zero Entries in W2:  183808\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.36, 0.02741214632987976, 0.023429229855537415)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.31, 0.019461434334516525, 0.026548640802502632)\n",
            "\n",
            "\n",
            "Regularization Path Index:  97\n",
            "Lambda at Index  4.821966435007673e-07\n",
            "Non-Zero Entries in W2:  177152\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.3, 0.027569202706217766, 0.02348593808710575)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.38, 0.019740628078579903, 0.026364348828792572)\n",
            "\n",
            "\n",
            "Regularization Path Index:  96\n",
            "Lambda at Index  5.05156799673706e-07\n",
            "Non-Zero Entries in W2:  173056\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.25, 0.02772974595427513, 0.02354501746594906)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.32, 0.019869668409228325, 0.026256419718265533)\n",
            "\n",
            "\n",
            "Regularization Path Index:  95\n",
            "Lambda at Index  5.292102209669876e-07\n",
            "Non-Zero Entries in W2:  166912\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.19, 0.027892041951417923, 0.023605309426784515)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.4, 0.020091399550437927, 0.026119884103536606)\n",
            "\n",
            "\n",
            "Regularization Path Index:  94\n",
            "Lambda at Index  5.544089640223151e-07\n",
            "Non-Zero Entries in W2:  154112\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.17, 0.028053883463144302, 0.023665618151426315)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.31, 0.020590852946043015, 0.025742296129465103)\n",
            "\n",
            "\n",
            "Regularization Path Index:  93\n",
            "Lambda at Index  5.808075641975761e-07\n",
            "Non-Zero Entries in W2:  147968\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.15, 0.02821645885705948, 0.023728495463728905)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.36, 0.020808206871151924, 0.025525353848934174)\n",
            "\n",
            "\n",
            "Regularization Path Index:  92\n",
            "Lambda at Index  6.084631535927754e-07\n",
            "Non-Zero Entries in W2:  139264\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.19, 0.028380339965224266, 0.023793034255504608)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.24, 0.02123541384935379, 0.02536950632929802)\n",
            "\n",
            "\n",
            "Regularization Path Index:  91\n",
            "Lambda at Index  6.374355846958685e-07\n",
            "Non-Zero Entries in W2:  136192\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.1, 0.02854572981595993, 0.023859435692429543)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.25, 0.021331971511244774, 0.025367172434926033)\n",
            "\n",
            "\n",
            "Regularization Path Index:  90\n",
            "Lambda at Index  6.677875599160826e-07\n",
            "Non-Zero Entries in W2:  130048\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.13, 0.02871299535036087, 0.023929473012685776)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.19, 0.02157478779554367, 0.025243058800697327)\n",
            "\n",
            "\n",
            "Regularization Path Index:  89\n",
            "Lambda at Index  6.995847672850605e-07\n",
            "Non-Zero Entries in W2:  126464\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.13, 0.028883378952741623, 0.024003956466913223)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.18, 0.021693505346775055, 0.02516273967921734)\n",
            "\n",
            "\n",
            "Regularization Path Index:  88\n",
            "Lambda at Index  7.328960226195212e-07\n",
            "Non-Zero Entries in W2:  120832\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.13, 0.029057273641228676, 0.024081140756607056)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.17, 0.021913742646574974, 0.024958636611700058)\n",
            "\n",
            "\n",
            "Regularization Path Index:  87\n",
            "Lambda at Index  7.677934184531006e-07\n",
            "Non-Zero Entries in W2:  116224\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.15, 0.02923128567636013, 0.024157583713531494)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.13, 0.02216649428009987, 0.02486388571560383)\n",
            "\n",
            "\n",
            "Regularization Path Index:  86\n",
            "Lambda at Index  8.043524800596953e-07\n",
            "Non-Zero Entries in W2:  109568\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.11, 0.029402723535895348, 0.02423611655831337)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.1, 0.022456448525190353, 0.02471240796148777)\n",
            "\n",
            "\n",
            "Regularization Path Index:  85\n",
            "Lambda at Index  8.426523289059714e-07\n",
            "Non-Zero Entries in W2:  104960\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (85.04, 0.02956990711390972, 0.02431296557188034)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.03, 0.02265106327831745, 0.024541780352592468)\n",
            "\n",
            "\n",
            "Regularization Path Index:  84\n",
            "Lambda at Index  8.827758538867933e-07\n",
            "Non-Zero Entries in W2:  98304\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (84.89, 0.029783418402075768, 0.024442138150334358)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.03, 0.022930623963475227, 0.024505529552698135)\n",
            "\n",
            "\n",
            "Regularization Path Index:  83\n",
            "Lambda at Index  9.248098907141524e-07\n",
            "Non-Zero Entries in W2:  92672\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (84.79, 0.029970332980155945, 0.024543767794966698)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.03, 0.023154081776738167, 0.02446586824953556)\n",
            "\n",
            "\n",
            "Regularization Path Index:  82\n",
            "Lambda at Index  9.688454098478343e-07\n",
            "Non-Zero Entries in W2:  90112\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (84.83, 0.03016400896012783, 0.02465249039232731)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.27, 0.023259053006768227, 0.02441144920885563)\n",
            "\n",
            "\n",
            "Regularization Path Index:  81\n",
            "Lambda at Index  1.0149777133745505e-06\n",
            "Non-Zero Entries in W2:  87040\n",
            "Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE:  (84.79, 0.03036494366824627, 0.024766070768237114)\n",
            "Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE:  (85.19, 0.023380709812045097, 0.024396872147917747)\n",
            "\n",
            "\n"
          ]
        }
      ],
      "source": [
        "for j in range(100-1, 80, -1):\n",
        "    sparse_W2 = torch.Tensor(state.betas[j].toarray()[0].reshape(W2.T.shape)).to(device)\n",
        "\n",
        "    adelie_metric = report_test_accuracy_W2(sparse_W2)\n",
        "\n",
        "    polish_W2 = torch.zeros(sparse_W2.shape)\n",
        "\n",
        "    A = gated_activations.cpu().numpy()\n",
        "    X = sparse_W2.cpu().numpy()\n",
        "    B = y_train_SCNN_new.cpu().numpy()\n",
        "\n",
        "    lsq_time = 0\n",
        "\n",
        "    for col_idx in range(B.shape[1]):\n",
        "\n",
        "        non_zero_indices = np.where(X[:, col_idx] != 0)[0]\n",
        "        if len(non_zero_indices) == 0:\n",
        "            continue\n",
        "\n",
        "        A_prime = A[:, non_zero_indices]\n",
        "        b = B[:, col_idx]\n",
        "\n",
        "        start_time = time.time()\n",
        "        x = lsq_linear(A_prime, b)['x']\n",
        "        lsq_time += time.time() - start_time\n",
        "\n",
        "        polish_W2[non_zero_indices, col_idx] = torch.Tensor(x)\n",
        "\n",
        "    polish_metric = report_test_accuracy_W2(polish_W2.to(device))\n",
        "\n",
        "    print(\"Regularization Path Index: \", j)\n",
        "    print(\"Lambda at Index \", state.lmdas[j])\n",
        "    print(\"Non-Zero Entries in W2: \", torch.count_nonzero(sparse_W2).item())\n",
        "    print(\"Adelie Acc, Adelie Train Fitting MSE, Adelie Test Fitting MSE: \", adelie_metric)\n",
        "    print(\"Polish Acc, Polish Train Fitting MSE, Polish Test Fitting MSE: \", polish_metric)\n",
        "    print(\"\\n\")\n",
        "\n",
        "    # break"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "oW_f2HJqYl-l"
      },
      "outputs": [],
      "source": [
        "resnet_model = torch.vision"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "A5U3UHu8y7K3",
        "outputId": "f69b69af-5010-4b1f-9dae-52e87e0e3033"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "tensor(1461340, device='cuda:0')\n",
            "tensor(82944, device='cuda:0')\n",
            "torch.Size([29774, 1024])\n",
            "torch.Size([29774, 512])\n",
            "tensor(524288, device='cuda:0')\n"
          ]
        }
      ],
      "source": [
        "print(torch.count_nonzero(W1))\n",
        "print(torch.count_nonzero(sparse_W2))\n",
        "print(W1.shape)\n",
        "print(sparse_W2.shape)\n",
        "print(count_effective_entries(W1.T, sparse_W2))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "ghq7s_js3lhJ",
        "outputId": "cbadead7-06b9-4a3b-9258-818046e726f2"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "[ 0.20026419  2.07875891  2.7065717   3.38680148  3.11622945  4.37225341\n",
            "  5.67254587  7.51369981 10.79961778 14.19090436]\n"
          ]
        }
      ],
      "source": [
        "def _squash_times(ts):\n",
        "    idx = 0\n",
        "    new_ts = []\n",
        "    while idx < len(state.n_valid_solutions):\n",
        "        if state.n_valid_solutions[idx]:\n",
        "            new_ts.append(ts[idx])\n",
        "        else:\n",
        "            t = 0\n",
        "            while (idx < len(state.n_valid_solutions)) and (not state.n_valid_solutions[idx]):\n",
        "                t += ts[idx]\n",
        "                idx += 1\n",
        "            if idx < len(state.n_valid_solutions):\n",
        "                t += ts[idx]\n",
        "            new_ts.append(t)\n",
        "        idx += 1\n",
        "    return np.array(new_ts)\n",
        "\n",
        "times = [\n",
        "    _squash_times(state.benchmark_screen),\n",
        "    _squash_times(state.benchmark_fit_screen),\n",
        "    _squash_times(state.benchmark_fit_active),\n",
        "    _squash_times(state.benchmark_kkt),\n",
        "    _squash_times(state.benchmark_invariance),\n",
        "]\n",
        "n_iters = np.min([state.lmdas.shape[0]] + [len(t) for t in times])\n",
        "times = [t[:n_iters] for t in times]\n",
        "\n",
        "print(np.sum(np.array(times), axis=0))\n",
        "# lmdas = state.lmdas[:n_iters]\n",
        "# tlmdas = -np.log(lmdas)\n",
        "\n",
        "# total_times = np.array([np.sum(t) for t in times])\n",
        "# total_times_sum = np.sum(total_times)\n",
        "# total_times = np.concatenate([\n",
        "#     total_times,\n",
        "#     [state.total_time - total_times_sum] # unaccounted time\n",
        "# ])\n",
        "# if state.relative:\n",
        "#     total_times /= state.total_time"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "h87lSUD16Ohm",
        "outputId": "b666edd5-0b58-42f4-81a6-7aa7a8699f4b"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "[0.41672119 0.33862039 0.357245   0.27353804 0.55361135 0.53692663\n",
            " 0.51079233 0.62687451 0.55269774 0.57478042 0.63567376 0.73897639\n",
            " 0.82905542 0.71272457 0.83544406 0.84233217 0.98265396 1.00370689\n",
            " 0.95545849 1.05272947 1.09467282 0.92055892 1.05225221 1.07737592\n",
            " 1.41694707 1.24034325 1.25049976 1.24412095 1.34562271 1.36238078\n",
            " 1.54552273 1.52167405 1.59424301 1.60077025 1.68276707 1.74041869\n",
            " 1.96878832 2.04793219 1.89150406 2.01812957 2.3258553  2.39484772\n",
            " 2.5883428  2.6733306  2.90010137 2.93703562 3.01302274 3.12015757\n",
            " 3.23869335 3.60618738 3.60704117 3.95585748 3.87859708 4.04924621\n",
            " 4.30284006 4.62996607 4.83976279 4.70799463 5.50414197 5.51614732\n",
            " 5.62039845 5.92023813 6.00028475 6.10545274 6.3914993  6.45294663\n",
            " 5.58704854 5.72708899 4.55563233 5.05010731 3.89282604 4.13044593\n",
            " 3.79641526 4.01165832 4.55051303 4.35695354 4.55431971 4.7729302\n",
            " 4.83475242 4.92590377 5.16870114 5.27154202 5.21893521 5.27899541\n",
            " 5.51010492 5.55791224 5.88560497 5.64892717 5.77407884 5.92346898\n",
            " 6.18488137 6.02768564 6.15959706 6.59850243 6.49124969 6.69610861\n",
            " 7.38251801 7.03449435 7.39357457 7.59501281]\n"
          ]
        }
      ],
      "source": [
        "def _squash_times(ts):\n",
        "    idx = 0\n",
        "    new_ts = []\n",
        "    while idx < len(state.n_valid_solutions):\n",
        "        if state.n_valid_solutions[idx]:\n",
        "            new_ts.append(ts[idx])\n",
        "        else:\n",
        "            t = 0\n",
        "            while (idx < len(state.n_valid_solutions)) and (not state.n_valid_solutions[idx]):\n",
        "                t += ts[idx]\n",
        "                idx += 1\n",
        "            if idx < len(state.n_valid_solutions):\n",
        "                t += ts[idx]\n",
        "            new_ts.append(t)\n",
        "        idx += 1\n",
        "    return np.array(new_ts)\n",
        "\n",
        "times = [\n",
        "    _squash_times(state.benchmark_screen),\n",
        "    _squash_times(state.benchmark_fit_screen),\n",
        "    _squash_times(state.benchmark_fit_active),\n",
        "    _squash_times(state.benchmark_kkt),\n",
        "    _squash_times(state.benchmark_invariance),\n",
        "]\n",
        "n_iters = np.min([state.lmdas.shape[0]] + [len(t) for t in times])\n",
        "times = [t[:n_iters] for t in times]\n",
        "\n",
        "print(np.sum(np.array(times), axis=0))\n",
        "# lmdas = state.lmdas[:n_iters]\n",
        "# tlmdas = -np.log(lmdas)\n",
        "\n",
        "# total_times = np.array([np.sum(t) for t in times])\n",
        "# total_times_sum = np.sum(total_times)\n",
        "# total_times = np.concatenate([\n",
        "#     total_times,\n",
        "#     [state.total_time - total_times_sum] # unaccounted time\n",
        "# ])\n",
        "# if state.relative:\n",
        "#     total_times /= state.total_time"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "3Iez7nQpEC-f"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "from scipy.optimize import lsq_linear"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 211
        },
        "id": "X8eYyOKm5J6U",
        "outputId": "b9f05ad0-f332-494a-92b7-39f59f5f0eab"
      },
      "outputs": [],
      "source": [
        "polish_W2 = torch.zeros(sparse_W2.shape)\n",
        "\n",
        "A = gated_activations.cpu().numpy()\n",
        "X = sparse_W2.cpu().numpy()\n",
        "B = y_train_SCNN_new.cpu().numpy()"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 211
        },
        "id": "9X5XDFYRycqk",
        "outputId": "515be764-96ec-4ebf-a2c4-d9b8841017bb"
      },
      "outputs": [],
      "source": [
        "import time\n",
        "\n",
        "lsq_time = 0\n",
        "\n",
        "for col_idx in range(B.shape[1]):\n",
        "\n",
        "    print(col_idx)\n",
        "\n",
        "    non_zero_indices = np.where(X[:, col_idx] != 0)[0]\n",
        "    if len(non_zero_indices) == 0:\n",
        "        continue\n",
        "\n",
        "    A_prime = A[:, non_zero_indices]\n",
        "    b = B[:, col_idx]\n",
        "\n",
        "    start_time = time.time()\n",
        "    x = lsq_linear(A_prime, b)['x']\n",
        "    lsq_time += time.time() - start_time\n",
        "\n",
        "    polish_W2[non_zero_indices, col_idx] = torch.Tensor(x)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "eNp4a8M37Z2V"
      },
      "outputs": [],
      "source": [
        "polish_W2 = polish_W2.to(device)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "z8_PYPLxvP42",
        "outputId": "43809957-9880-4d2a-e4aa-80a1b6374950"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Test set: Average loss: 0.0, Accuracy: 8485.0/10000.0 (84.85%)\n",
            "\n"
          ]
        }
      ],
      "source": [
        "test_loss = 0\n",
        "num_classes = 10\n",
        "test_class_correct = list(0. for i in range(num_classes))\n",
        "test_class_total = list(0. for i in range(num_classes))\n",
        "test_overall_correct = 0\n",
        "test_overall_total = 0\n",
        "\n",
        "orig_model.eval()\n",
        "\n",
        "\n",
        "test_loss = 0\n",
        "test_class_correct = list(0. for i in range(num_classes))\n",
        "test_class_total = list(0. for i in range(num_classes))\n",
        "test_overall_correct = 0\n",
        "test_overall_total = 0\n",
        "\n",
        "with torch.no_grad():\n",
        "\n",
        "    for data in test_loader:\n",
        "\n",
        "        images, labels = data\n",
        "        images, labels = images.to(device), labels.to(device)\n",
        "\n",
        "        _ = orig_model(images)\n",
        "\n",
        "        input = activation[\"layer3\"]\n",
        "        target_output = activation[\"layer4\"]\n",
        "\n",
        "        # print(input.shape)\n",
        "        # print(target_output.shape)\n",
        "        # print(input.reshape((images.shape[0], -1)).cpu().numpy().shape)\n",
        "        # print(relu_model(input.reshape((images.shape[0], -1)).cpu().numpy()).shape)\n",
        "        # print(torch.reshape(torch.Tensor(relu_model(input.reshape((images.shape[0], -1)).cpu().numpy())), target_output.shape).shape)\n",
        "        # print(orig_model.avgpool(target_output).shape)\n",
        "        # print(orig_model.fc(orig_model.avgpool(target_output)).shape)\n",
        "        _, batch_predictions = gated_relu_prediction(input.reshape((images.shape[0], -1)), gates, W1, polish_W2.T)\n",
        "\n",
        "        output = orig_model.fc(orig_model.avgpool(batch_predictions.reshape(target_output.shape)).reshape(input.shape[0], -1)).to(device)\n",
        "\n",
        "        predicted = output.argmax(dim=1, keepdim=True)\n",
        "        predicted = predicted.squeeze()\n",
        "\n",
        "        for i in range(len(predicted)):\n",
        "            index = labels[i].item()\n",
        "            test_class_correct[index] += int(labels[i] == predicted[i].item())\n",
        "            test_class_total[index] += 1\n",
        "\n",
        "test_loss /= len(test_loader.dataset)\n",
        "\n",
        "overall_correct = 0\n",
        "overall_total = 0\n",
        "\n",
        "for i in range(num_classes):\n",
        "\n",
        "    test_overall_correct += test_class_correct[i]\n",
        "    test_overall_total += test_class_total[i]\n",
        "\n",
        "print('Test set: Average loss: {}, Accuracy: {}/{} ({}%)\\n'.format(\n",
        "    test_loss, test_overall_correct, test_overall_total, 100.0 * test_overall_correct / test_overall_total))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "FFBIrV20Muv5",
        "outputId": "56540bd9-5e51-4330-b00a-ca8d469a969f"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "28.847305297851562"
            ]
          },
          "execution_count": 77,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "lsq_time"
      ]
    }
  ],
  "metadata": {
    "accelerator": "GPU",
    "colab": {
      "collapsed_sections": [
        "9xWBuVET_tCO",
        "eknuxrkv_xZu"
      ],
      "gpuType": "T4",
      "machine_shape": "hm",
      "provenance": []
    },
    "kernelspec": {
      "display_name": "Python 3",
      "name": "python3"
    },
    "language_info": {
      "name": "python"
    },
    "widgets": {
      "application/vnd.jupyter.widget-state+json": {
        "096dbbb65dbe43e49dd49d4f038893d8": {
          "model_module": "@jupyter-widgets/base",
          "model_module_version": "1.2.0",
          "model_name": "LayoutModel",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "3ddd4bdfd4bc4aed9954646969ccc5e7": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "FloatProgressModel",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "FloatProgressModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "ProgressView",
            "bar_style": "danger",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_a7b4591ae08f45e39739bcc3bfc1aeb5",
            "max": 10000,
            "min": 0,
            "orientation": "horizontal",
            "style": "IPY_MODEL_e944c6a17f6d4217a968ec019867b660",
            "value": 11
          }
        },
        "3e7b644c59da472784104cd16c5d83f6": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "HTMLModel",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_8d43325a440a4e05bb36352b841fa856",
            "placeholder": "​",
            "style": "IPY_MODEL_d6feeae7d8da4e98934474ad17c189a2",
            "value": "fista:   0%"
          }
        },
        "4b9222767f6b429db911283903ead483": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "DescriptionStyleModel",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "85634e7f94dc430c910f6b56f96d4d41": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "HBoxModel",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HBoxModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HBoxView",
            "box_style": "",
            "children": [
              "IPY_MODEL_3e7b644c59da472784104cd16c5d83f6",
              "IPY_MODEL_3ddd4bdfd4bc4aed9954646969ccc5e7",
              "IPY_MODEL_c45ea1db6fb44eedaee1b94d6d3829bb"
            ],
            "layout": "IPY_MODEL_c97d596d1703495bb7da15526d037b12"
          }
        },
        "8d43325a440a4e05bb36352b841fa856": {
          "model_module": "@jupyter-widgets/base",
          "model_module_version": "1.2.0",
          "model_name": "LayoutModel",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "a7b4591ae08f45e39739bcc3bfc1aeb5": {
          "model_module": "@jupyter-widgets/base",
          "model_module_version": "1.2.0",
          "model_name": "LayoutModel",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "c45ea1db6fb44eedaee1b94d6d3829bb": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "HTMLModel",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_096dbbb65dbe43e49dd49d4f038893d8",
            "placeholder": "​",
            "style": "IPY_MODEL_4b9222767f6b429db911283903ead483",
            "value": " 11/10000 [00:45&lt;11:45:11,  4.24s/it]"
          }
        },
        "c97d596d1703495bb7da15526d037b12": {
          "model_module": "@jupyter-widgets/base",
          "model_module_version": "1.2.0",
          "model_name": "LayoutModel",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "d6feeae7d8da4e98934474ad17c189a2": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "DescriptionStyleModel",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "e944c6a17f6d4217a968ec019867b660": {
          "model_module": "@jupyter-widgets/controls",
          "model_module_version": "1.5.0",
          "model_name": "ProgressStyleModel",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "ProgressStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "bar_color": null,
            "description_width": ""
          }
        }
      }
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}
