{
  "cells": [
    {
      "cell_type": "code",
      "execution_count": 5,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "3bc26880",
        "outputId": "b5a50305-e34f-4dce-a8d2-8310679538fe"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "<module 'hopfield' from '/content/gdrive/MyDrive/Hopfield/hopfield.py'>"
            ]
          },
          "metadata": {},
          "execution_count": 5
        }
      ],
      "source": [
        "from importlib import reload\n",
        "from math import sqrt\n",
        "import numpy as np\n",
        "import matplotlib.pyplot as plt\n",
        "from sklearn.utils import check_random_state\n",
        "import pandas as pd\n",
        "import seaborn as sns\n",
        "\n",
        "import hopfield\n",
        "reload(hopfield)"
      ],
      "id": "3bc26880"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "bEE219a79rh3"
      },
      "outputs": [],
      "source": [
        "import torch\n",
        "import torchvision\n",
        "from torchvision import transforms\n",
        "\n",
        "\n",
        "def load_mnist(batch_size=1000, test_batch_size=1000,\n",
        "               norm_factor=1, shuffle=True, redownload=False):\n",
        "    transform = transforms.Compose([transforms.ToTensor()])\n",
        "    trainset = torchvision.datasets.MNIST(root='./mnist_data', train=True,\n",
        "                                         download=redownload, transform=transform)\n",
        "    print(\"trainset: \", trainset)\n",
        "    trainloader = torch.utils.data.DataLoader(trainset, batch_size=batch_size,\n",
        "                                              shuffle=shuffle)\n",
        "    print(\"trainloader: \", trainloader)\n",
        "    trainset = list(iter(trainloader))\n",
        "\n",
        "    testset = torchvision.datasets.MNIST(root='./mnist_data', train=False,\n",
        "                                         download=redownload,\n",
        "                                         transform=transform)\n",
        "    testloader = torch.utils.data.DataLoader(testset,\n",
        "                                             batch_size=test_batch_size,\n",
        "                                             shuffle=shuffle)\n",
        "    testset = list(iter(testloader))\n",
        "    for i,(img, label) in enumerate(trainset):\n",
        "        trainset[i] = (img.reshape(len(img), 784) / norm_factor ,label)\n",
        "    for i,(img, label) in enumerate(testset):\n",
        "        testset[i] = (img.reshape(len(img), 784) / norm_factor ,label)\n",
        "    return trainset, testset"
      ],
      "id": "bEE219a79rh3"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "6BpZG_tf4VXL"
      },
      "outputs": [],
      "source": [
        "from math import floor\n",
        "\n",
        "def mask_bottom(img, theta=0.5, copy=True):\n",
        "    if copy:\n",
        "        img = img.copy()\n",
        "    nrows = len(img)\n",
        "    thetac = 1 - theta\n",
        "    img[:, int(floor(thetac * nrows)):] = -1\n",
        "    return img"
      ],
      "id": "6BpZG_tf4VXL"
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "1d119fe6"
      },
      "source": [
        "# Memory Capacity"
      ],
      "id": "1d119fe6"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "127867a3",
        "outputId": "ea4416f6-f787-4fa2-e989-bcfa128e4e12"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "trainset:  Dataset MNIST\n",
            "    Number of datapoints: 60000\n",
            "    Root location: ./mnist_data\n",
            "    Split: Train\n",
            "    StandardTransform\n",
            "Transform: Compose(\n",
            "               ToTensor()\n",
            "           )\n",
            "trainloader:  <torch.utils.data.dataloader.DataLoader object at 0x7ea2a55059f0>\n"
          ]
        }
      ],
      "source": [
        "# prepare MNIST data\n",
        "trainset, testset = load_mnist(batch_size=60000, shuffle=True,\n",
        "                               redownload=False)\n",
        "imgs, labels = trainset[0]\n",
        "imgs = imgs.numpy().reshape((-1, 28, 28))\n",
        "labels = labels.numpy()\n",
        "imgs = 2 * (imgs >= 0.4) - 1."
      ],
      "id": "127867a3"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "045beaef",
        "outputId": "9b3ac9f0-a62d-4f24-8b0f-71ed9367f44a"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Run 1 / 10\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=10000, metrics={'succ': 0.0, 'duration': 2.56916880607605}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=5000, metrics={'succ': 0.0, 'duration': 1.0297729969024658}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=2000, metrics={'succ': 0.0, 'duration': 0.8097014427185059}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=1000, metrics={'succ': 0.0, 'duration': 0.7773270606994629}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=900, metrics={'succ': 0.0, 'duration': 0.7715716361999512}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=800, metrics={'succ': 0.0, 'duration': 0.7738566398620605}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=700, metrics={'succ': 0.0, 'duration': 0.3551034927368164}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=600, metrics={'succ': 0.0, 'duration': 0.3497505187988281}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=500, metrics={'succ': 0.0, 'duration': 0.3429996967315674}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=400, metrics={'succ': 0.0, 'duration': 0.3337574005126953}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=300, metrics={'succ': 0.0, 'duration': 0.3310396671295166}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=200, metrics={'succ': 0.0, 'duration': 0.3268873691558838}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=100, metrics={'succ': 0.0, 'duration': 0.3260672092437744}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=90, metrics={'succ': 0.0, 'duration': 0.33039045333862305}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=80, metrics={'succ': 0.0, 'duration': 0.32724952697753906}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=70, metrics={'succ': 0.0, 'duration': 0.32227206230163574}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=60, metrics={'succ': 0.0, 'duration': 0.32619810104370117}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=50, metrics={'succ': 0.0, 'duration': 0.32759952545166016}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=40, metrics={'succ': 0.0, 'duration': 0.3194549083709717}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=30, metrics={'succ': 0.0, 'duration': 0.3196530342102051}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=20, metrics={'succ': 0.0, 'duration': 0.3226616382598877}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=10, metrics={'succ': 0.0, 'duration': 0.3097059726715088}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=9, metrics={'succ': 0.0, 'duration': 0.3221251964569092}\n",
            "float32 float32\n",
            "conf={'mode': 'Classical', 'scale': 784, 'method': 'Classical'}, DB size=8, metrics={'succ': 0.0, 'duration': 0.32763195037841797}\n",
            "float32 float32\n"
          ]
        }
      ],
      "source": [
        "reload(hopfield)\n",
        "\n",
        "backend = \"cuda\"  # change to \"cpu\" if you don't have GPU\n",
        "simulated = False\n",
        "shape = (28, 28)\n",
        "group_sizes = [7, 28, 49, 112]\n",
        "n_runs = 10\n",
        "if simulated:\n",
        "    n_runs = 10\n",
        "    group_sizes = [7, 28, 49, 112]\n",
        "N = np.prod(shape)  # input dimension (i.e number of input neurons)\n",
        "mem_sizes = np.concatenate((\n",
        "        np.arange(1, 11),\n",
        "        np.arange(10, 110, 10),\n",
        "        np.arange(100, 1100, 100),\n",
        "        [1000, 2000, 5000, 10000]  # comment this out if lacking memory\n",
        "        ))\n",
        "mem_sizes = np.unique(mem_sizes)[::-1]\n",
        "modes = [\"Classical\", \"Demircigil-expo\", \"Demircigil-poly\", \"Ours\"]\n",
        "res = []\n",
        "rng = check_random_state(0)\n",
        "for runid in range(n_runs):\n",
        "    print(\"Run %d / %d\" % (runid + 1, n_runs))\n",
        "    if simulated:\n",
        "        data = rng.choice([-1, 1.], size=(10000, *shape))\n",
        "    else:\n",
        "        data = imgs.copy()\n",
        "        rng.shuffle(data)\n",
        "    mem_sizes = [M for M in mem_sizes if M <= len(data)]\n",
        "    for theta in [0, .25, .35][::-1]:\n",
        "        data = data.reshape((len(data), *shape))\n",
        "        if True:\n",
        "            corrupt_data = [mask_bottom(x, theta=theta, copy=True)\n",
        "                            for x in data]\n",
        "            corrupt_data = np.asanyarray(corrupt_data)\n",
        "        else:\n",
        "            corrupt_data = data.copy()\n",
        "            mask = np.less_equal(rng.rand(*corrupt_data.shape), theta)\n",
        "            corrupt_data[mask] = -1\n",
        "        data = data.reshape((len(data), N))\n",
        "        corrupt_data = corrupt_data.reshape((len(corrupt_data), N))\n",
        "        confs = []\n",
        "        for mode in modes:\n",
        "            if mode == \"Ours\":\n",
        "                confs += [dict(mode=mode, group_size=gs,\n",
        "                              method=\"%s (group size=%s)\" % (mode, gs))\n",
        "                         for gs in group_sizes]\n",
        "            elif mode == \"Demircigil-poly\":\n",
        "                confs += [dict(mode=mode, deg=deg, scale=N,\n",
        "                               method=\"Poly HN (deg=%d)\" % deg)\n",
        "                for deg in N // np.array(group_sizes)]\n",
        "            elif mode == \"Demircigil-expo\":\n",
        "                scale = 10.\n",
        "                confs += [dict(mode=mode, scale=scale,\n",
        "                               method=\"Exponential HN\")]\n",
        "            elif mode == \"Classical\":\n",
        "                confs += [dict(mode=mode, scale=N,\n",
        "                               method=\"Classical\")]\n",
        "            else:\n",
        "                confs += [dict(mode=mode, method=mode)]\n",
        "\n",
        "        for conf in confs:\n",
        "            # XXX rm hack !\n",
        "            for M in mem_sizes:\n",
        "                this_res = hopfield.run_experiment(conf, data[:M],\n",
        "                                                   corrupt_data[:M],\n",
        "                                                   mega_bytes=400,\n",
        "                                                   true_input_data=data[:M],\n",
        "                                                   backend=backend)\n",
        "                for stuff in this_res:\n",
        "                    stuff[\"runid\"] = runid\n",
        "                    stuff[\"M\"] = M\n",
        "                    stuff[\"theta\"] = theta\n",
        "                res += this_res"
      ],
      "id": "045beaef"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "9fdc821c"
      },
      "outputs": [],
      "source": [
        "import pandas as pd\n",
        "import seaborn as sns\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "df = pd.DataFrame(res)\n",
        "df = df[df.deg != 2]\n",
        "df.succ = df.succ.apply(lambda s: s.item() if hasattr(s, \"item\") else s)\n",
        "df[\"M_retrieved\"] = df.succ * df.M\n",
        "df[\"succ\"] = df.succ * 100\n",
        "# df = df[df.group_size != 14]\n",
        "df.method = df.method.apply(lambda m: m.replace(\"group size\", \"$N/k$\"))\n",
        "df.method = df.method.apply(lambda m: m.replace(\", ug=False\",\n",
        "                                                \"\"))\n",
        "df.method = df.method.apply(lambda m: m.replace(\"Dermicigil (expo)\",\n",
        "                                                \"Exponential HN\"))\n",
        "df.method = df.method.apply(lambda m: m.replace(\"Demircigil (poly, d=\",\n",
        "                                                \"Poly HN (deg=\"))\n",
        "df.method = df.method.apply(lambda m: m.replace(\"Hopfield (ug=False)\",\n",
        "                                                \"Classical HN\"))\n",
        "df.method = df.apply(lambda row: row.method if row[\"mode\"] != \"Ours\"\n",
        "                        else \"PSHN ($k=%d$)\" % (N // row.group_size), axis=1)\n",
        "# df.method = df.group_size.apply(lambda gs: \"$k=%d$\" % (784 // gs))\n",
        "thetas = sorted(df.theta.unique())\n",
        "_, axes = plt.subplots(1, len(thetas), figsize=(len(thetas) * 3, 1.5),\n",
        "                       sharey=True, gridspec_kw = {'wspace':0.05, 'hspace':0})\n",
        "axes = np.ravel([axes])\n",
        "y = \"succ\"\n",
        "for ax, theta in zip(axes, thetas):\n",
        "    if y == \"succ\":\n",
        "        ax.axhline(100, linestyle=\"--\", c=\"k\")\n",
        "        ax.axhline(0, linestyle=\"--\", c=\"k\")\n",
        "    subdf = df[df.theta == theta]\n",
        "    sns.lineplot(data=subdf, x=\"M\", y=y, hue=\"method\",\n",
        "                 err_style=\"bars\", ax=ax, linewidth=2)\n",
        "    ax.set_xscale(\"log\")\n",
        "    ax.set_xlabel(\"Number of patterns in database\", fontsize=8)\n",
        "    if y == \"succ\":\n",
        "        ax.set_ylabel(\"Retrieval accuracy (%)\", fontsize=8)\n",
        "    elif y == \"M_retrieved\":\n",
        "        ax.set_ylabel(\"# patterns correctly retrieved\")\n",
        "        ax.set_yscale(\"log\")\n",
        "    elif y == \"duration\":\n",
        "        ax.set_ylabel(\"Duration (seconds)\")\n",
        "    else:\n",
        "        raise NotImplementedError(y)\n",
        "    if len(thetas) != 1:\n",
        "        ax.set_title(\"Noise-level $\\\\theta=%s$\" % (\n",
        "            int(theta) if theta == np.floor(theta) else (\"%0.2f\" % theta)),\n",
        "                    fontsize=8)\n",
        "    ax.legend_.remove()\n",
        "loc = (-2., 1.2)\n",
        "frameon = True\n",
        "if len(thetas) == 1:\n",
        "    loc = (0, 1)\n",
        "    frameon = False\n",
        "plt.legend(loc=loc, ncol=5, fontsize=8, frameon=frameon,\n",
        "           handlelength=1)\n",
        "if simulated:\n",
        "    tag = \"simulated\"\n",
        "else:\n",
        "    tag = \"mnist\"\n",
        "filename = \"benchmark-%s-all-M=%s.png\" % (tag, np.max(mem_sizes))\n",
        "plt.savefig(filename, dpi=200, bbox_inches=\"tight\")\n",
        "print(filename)"
      ],
      "id": "9fdc821c"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "MI-xbEsX7VJX"
      },
      "outputs": [],
      "source": [
        "df_old = df\n",
        "df.to_pickle(filename.replace(\".png\", \".pkl\"))"
      ],
      "id": "MI-xbEsX7VJX"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "doVOTOTs6vLt"
      },
      "outputs": [],
      "source": [
        "df = pd.read_pickle(\"benchmark-simulated-all-M=10000.pkl\")"
      ],
      "id": "doVOTOTs6vLt"
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "552e8a0a"
      },
      "source": [
        "# Visual Demo of Retrieval"
      ],
      "id": "552e8a0a"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "142471a3",
        "outputId": "9c452011-9c6b-403f-e0c2-bcf4f3a54d9b"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Running Demircigil-expo ...\n",
            "Running Demircigil-poly ...\n",
            "Running Demircigil-poly ...\n",
            "Running Demircigil-poly ...\n",
            "Running Ours ...\n",
            "Pytoch: Using device cuda:0\n",
            "Running Ours ...\n",
            "Pytoch: Using device cuda:0\n",
            "Running Ours ...\n",
            "Pytoch: Using device cuda:0\n"
          ]
        }
      ],
      "source": [
        "reload(hopfield)\n",
        "from sklearn.utils import check_random_state\n",
        "\n",
        "rng = check_random_state(0)\n",
        "# rng.shuffle(imgs)\n",
        "\n",
        "M = 100\n",
        "group_size = 7\n",
        "permute_neurons = True\n",
        "true_imgs = imgs[:M]  # queries\n",
        "backend = \"cuda\"\n",
        "if True:\n",
        "    theta = 0.35\n",
        "    corrupt_imgs = [mask_bottom(img.T, theta=theta).T for img in true_imgs]\n",
        "else:\n",
        "    noise_std = 0.001\n",
        "    rng = check_random_state(0)\n",
        "    corrupt_imgs = true_imgs + rng.randn(*true_imgs.shape) * noise_std\n",
        "n_queries = len(corrupt_imgs)\n",
        "X = imgs[:M].reshape((M, -1))\n",
        "Q = np.reshape(corrupt_imgs, (n_queries, -1))\n",
        "\n",
        "modes = [\"Demircigil-expo\", \"Demircigil-poly\", \"Ours\"]\n",
        "group_sizes = np.array([112, 49, 7])\n",
        "ks = N // group_sizes\n",
        "confs = []\n",
        "for mode in modes:\n",
        "    if mode == \"Ours\":\n",
        "        confs += [dict(mode=mode, group_size=gs,\n",
        "                       method=\"PSHN ($k=%d$)\" % k)\n",
        "        for gs, k in zip(group_sizes, ks)]\n",
        "    elif mode == \"Demircigil-poly\":\n",
        "        deg = k\n",
        "        confs += [dict(mode=mode, deg=deg, scale=N,\n",
        "                        method=\"Poly HN (deg=%d)\" % deg) for deg in ks]\n",
        "    elif mode == \"Demircigil-expo\":\n",
        "        scale = 10.\n",
        "        confs += [dict(mode=mode, scale=scale,\n",
        "                        method=\"Expo HN\")]\n",
        "    elif mode == \"Classical\":\n",
        "        confs += [dict(mode=mode, scale=N,\n",
        "                        method=\"Classical\")]\n",
        "    else:\n",
        "        confs += [dict(mode=mode, method=mode)]\n",
        "\n",
        "recon_imgs = {}\n",
        "for conf in confs:\n",
        "    method = conf.pop(\"method\")\n",
        "    recon_imgs[method] = hopfield.update(X, Q,\n",
        "                                 backend=backend,\n",
        "                                 **conf)"
      ],
      "id": "142471a3"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 305
        },
        "id": "72e94cd7",
        "outputId": "962cb1da-bc18-438a-f6fc-6547623449ff"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "Generating plots...\n"
          ]
        },
        {
          "data": {
            "image/png": "iVBORw0KGgoAAAANSUhEUgAABCoAAAEPCAYAAACEDCXoAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAqyklEQVR4nO3de3yU1Z3H8W8CiYRcJtyUhEsoaLhKYtHqigJCF42IolUqN6OAlo2uuKtFFiJaVoTaVyurfXlthKoEW8RwURS0BLlssd7QCoKrJRBIcAUSE0gChJz9w+WRSWaSSTJJzjz5vF+veWnmeZ6ZM893npPJj3POhBljjAAAAAAAACwQ3tINAAAAAAAAOINCBQAAAAAAsAaFCgAAAAAAYA0KFQAAAAAAwBoUKgAAAAAAgDUoVAAAAAAAAGtQqAAAAAAAANagUAEAAAAAAKxBoQIAAAAAAFiDQgUAAAAAALAGhQoAAAAAAGAN6woVAwcO1BtvvBHQvsuWLdPll18elOfdtGmT4uPj/W5PS0vT008/HZTnQvCFhYVpx44dDTq2uLhYYWFhysvLa9Dxq1atUq9evRp0LNBc6urjWtK+ffuUnJysEydO+Ny+Y8cOhYWFNXOr6rZt2zZdccUVzfZ8ZNj8FixYoLlz5wbt8cjQPnfeeadeeOGFlm5Gs3v11Vc1fvx4r/suvvhi5eTkNGs7Ro8erXfffbdZn9MtbMmwLmTsX6hkWJcmy9g0kXfeecdcccUVJjo62sTFxZlrrrnGfPTRR031dI2Wm5trPB5PSzfD9ZrqfSHJfPLJJw06tqioyEgye/fubdDxOTk5JikpqUHHusXw4cNNZGSkiY6Odm6dOnVqkbb4ei9Uv77T09ONJLN27Vqv/Twej8nNzW36RjbC2ee6Q4cOZvjw4ebDDz+s87jG9HH+jn344YfNDTfc4PyclJRkoqOjzaFDh5z7PvnkE1PXr5rbbrvNPPnkk363B/IYTWHBggVe7+n27dsbSWblypXOPiNGjDCrVq2q1+OSYfMpKCgwY8eONQkJCX5/T+Tn55ubb77ZeDwe4/F4zOjRo51tpaWl5txzzzWFhYVex5Bh86krw7///e9m9OjRplOnTkaSKSoq8tr+m9/8xlx44YUmNjbWdOvWzdx///3mxIkTzva9e/eahIQEU1FREfS21/Y+2b17t7nuuutMp06dTGxsrOnbt69ZtGiR17FPPPFEjcc8+xwMHz7chIeHm08//dTZHshnmtOnT5sf/ehH5rPPPnPuO3XqlGnXrp356quvGveiqzm7D42OjjZt27Y1F154obN906ZNJjU1NajPGUxkaMxTTz1lhgwZYiIjI736mrOtXr3apKSkmPbt25uEhATzzDPPONtaOmMyrDvD2rZXVFSY6dOnm169epmYmBjTt29fk5WV5bVPU2XcJCMq1qxZo3Hjxum2225TYWGh8vLyNGzYMA0bNkwffvihz2MqKytljGmK5sASDXlfIHT8+te/1rFjx5zb4cOHW7pJterUqZPmzJmjqqqqlm5KvZ051wUFBbrooot0ww03tHSTHO3atdP8+fMD3v/IkSN6/fXXNWnSpCZsVcPMmTPH6z390ksvyePxKC0tzdknPT1dv//97+v92GTYPMLDw3XNNddo1apVPrcfP35cV111lVJSUpSfn6/Dhw/r0UcfdbbHxMQoLS1NWVlZNY4lw+ZRV4YREREaP368li5d6nP76dOnlZWVpSNHjmj79u3atGmTHnnkEWd7r169lJycrNdeey34jZf/98mYMWOUkpKi/fv3q6ioSCtXrlTv3r3r/fgdOnTQf/zHf9TrmHXr1qljx4668MILnfv27Nmjtm3bNqgNtTm7Dz127Jj69++vW2+91dk+bNgwFRcXa9u2bUF93mBq7RkmJiYqMzNTd955p8/tb7/9tjIyMrR48WKVlJRo586dGjFihLPdhozJsPYMa9teWVmphIQEvfvuuyopKdHSpUt1//33a8OGDc4+TZVx0AsVxhjNnDlTs2fP1p133qnY2FgnvJ///Od64IEHnH3DwsL0+9//XoMGDVJ0dLSOHTumXr16ef0yeuqpp9SjRw916tRJmZmZSk1NdX4ZLV26VKmpqc6+vXr10uOPP67LLrtMsbGxGj58uPLz853ts2bNUlJSkmJjYzVgwACtWLEi4Nc1YsQILV68WNIPQzeffvppdevWTR06dNDixYu1e/duXXrppYqLi9O4ceN0/Phx5/jJkycrMTFRcXFxGjJkiHJzc70ev7bXKUnvvvuufvKTnyg+Pl4DBw7UmjVrAm67DQJ9Xxw6dEiTJ09WQkKC4uPjNWzYMJWXl9e57Wy333677rvvPufn6lM7Tpw4oX/5l39Rx44d9aMf/ajGhxNjjJ588kn169dP8fHxGjFihL744gtn+4EDBzR69Ggny127dgX5bLnLxx9/LI/Ho88//1ySVFRUpJ49e+qPf/yjpO/zmjp1qsaNG6eYmBgNHjxYW7dudY4vLS3VXXfdpYSEBCUkJGjGjBle11ZD/fznP1dZWZleeeWVRj9WS2nXrp2mTZumgwcP6siRI/rmm280fvx4denSRT179tTcuXNVWVlZ47jVq1erd+/eXsXh7du3q2PHjqqoqGhUm2bNmqUlS5bo66+/Dmj/9evXq3///urYsaNzX3FxscaPH6/4+Hj169dPmzdv9jrm1KlTmjdvnvr06aNOnTrp+uuvV0FBgbN9586dzu+Bq666SrNmzfL60NRQWVlZmjBhgqKiopz7Ro0apU2bNqm0tLRBj0mGTZvheeedp4yMDP3kJz/xuX3p0qXq3LmzMjMzFRsbq7Zt2+qSSy7x2mfUqFG1/s4lw5bNsG/fvpo2bZoGDRrkc/uDDz6oSy65RBEREerevbtuu+02r98xUt0ZB8PZ75NvvvlGX3/9tX7xi1+offv2atOmjQYOHKhbbrml3o+bkZGhbdu21cinNmvWrNHIkSO97tuxY4cGDRqksLAwGWO0cOFCDRw4MKifcf72t79p165duv322537wsLCNHLkyJD4XNtaM7zppps0btw4de7c2ef2hx56SPPmzdOIESPUpk0bdejQQf369XO225QxGfrOsLbt0dHRmj9/vvr06aOwsDBddtlluuqqq7z60abKOOiFii+//FJ5eXmaOHFijW0TJ07U1q1bvf64zM7O1oYNG1RSUqLo6Giv/f/yl79o3rx5WrlypQoLCxUeHq6dO3fW+vyvvPKKli9frm+//VbR0dF66KGHnG0pKSn64IMPVFxcrHnz5mnKlCnau3dvg15naWmp8vLytHfvXq1YsUIPPPCAHnjgAb322mvKz8/XV199peeee87Zf9SoUfriiy905MgR3Xrrrbr55pudD7Z1vc7PPvtMt9xyixYtWqSjR4/queee05QpU7Rnz54Gtb0lBPK+OH78uMaOHau2bdtq165dOnz4sB577DGFh4erqqrK77b6WrBggf7617/q888/1yeffKLXX3/da/szzzyjrKwsrV27VocPH9ZNN92ksWPH6uTJk057ExISdOjQIS1btqxVzm2tjx//+Md6+OGHdeutt6q8vFzTpk3TlVdeqfT0dGef7OxsTZs2TcXFxcrIyND111+v4uJiSdLMmTP11Vdf6fPPP9ff//537d69W//2b//W6HZFREToP//zPzVv3jy/c7JtV1ZWpj/84Q9KSkpSp06dNHHiREVERGjv3r3asmWLVq1apccff7zGcWPGjFFZWZnee+89574lS5ZowoQJateuXaPalJycrClTpigzMzOg/Xfs2OH1gUaS7r33XhUXFysvL08bN27USy+95LV97ty52rZtm7Zu3arCwkIlJyc7/0J36tQpXX/99UpLS9ORI0e0aNEivfjii17HX3fddYqPj/d7279/f412HjhwQOvXr9f06dO97u/Ro4fatWvnFOLqiwybL0Nf3nvvPXXv3l1paWnq2LGjhgwZonXr1nntM2DAgFrXQCLDls2wvt577z0NHjzY6766Mg6Gs98n5557rvr27as77rhDf/7zn7Vv374GP27Hjh314IMPavbs2QEf4yvvTz/9VIMHD9bRo0c1duxYffTRR9q+fbsGDBjg7JORkVFrZtULQNVlZWUpLS1NiYmJXvc3x/kPBjKs6fjx4/roo4908OBBJScnq2vXrrrllltUWFjotZ8tGZNh41VUVOhvf/tb8/SjwZ5LsnXrViPJlJeX19i2a9cuI8kcOHDAmO//CcHk5OR47ZOUlOTcN3XqVHP33Xc7206ePGk8Ho9ZsmSJMcaYJUuWmJSUFK9jz54T9corr5hBgwb5bWtKSop55ZVXjDF1zxs9e45Sbm6uCQ8PN2VlZc72Ll26mGeffdb5+Ze//KWZNGmS38eLj483W7duDeh1ZmRkmPvuu8/r+IkTJ5r58+f7fXzbBPK+2LRpk4mOjvY6r2ds377d7zZjvOeKpaenm5kzZzrbqs8T6927t/nTn/7k9dhnbx8wYECNOeeJiYlm8+bNZv/+/UaS+eabb5xtixYtYo2K4cNNu3btnDneHo/H/PSnP3W2V1VVmWuuucYMHjzY9O7d23z33XfOtvT0dJOWlub1eP369TMvv/yyOX36tImMjDTbt293tm3bts2cc8455vTp0z7bIsnExMR4tSU6OrrGGhUzZ840VVVV5qKLLnKu7VBZo+LMuT7vvPPM1VdfbT799FNz4MABI8lrXvqyZcvMBRdcYIyp2cfNmjXLpKenG2OMKS8vNx6Px3zwwQc+nzM3N9dI8jqn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\n",
            "text/plain": [
              "<Figure size 1350x300 with 18 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "print(\"Generating plots...\")\n",
        "nrows = min(M, 2)\n",
        "ncols = 2 + len(list(recon_imgs.keys()))\n",
        "_, all_axes = plt.subplots(nrows, ncols, figsize=(ncols * 1.5, nrows * 1.5))\n",
        "all_axes = np.reshape(all_axes, [nrows, ncols])\n",
        "idx = -1\n",
        "for which, axes in zip([\"Original image\", \"Corrupt\"] + list(recon_imgs.keys()),\n",
        "                       all_axes.T):\n",
        "    if which == \"Original image\":\n",
        "        slate = true_imgs\n",
        "    elif which == \"Corrupt\":\n",
        "        slate = corrupt_imgs\n",
        "    else:\n",
        "        slate = recon_imgs[which]\n",
        "    if which == \"Corrupt\":\n",
        "        title = \"Occluded\"\n",
        "    else:\n",
        "        title = which\n",
        "    for i, (ax, img) in enumerate(zip(axes, slate)):\n",
        "        ax.matshow(img.reshape((28, 28)));\n",
        "        ax.margins(x=0)\n",
        "        # ax.axis(\"off\")\n",
        "        ax.set_xticks([])\n",
        "        ax.set_yticks([])\n",
        "        if i == 0:\n",
        "            ax.set_title(title, y=1.01, fontsize=9)\n",
        "# plt.tight_layout()\n",
        "plt.subplots_adjust(wspace=.05, hspace=.05)\n",
        "plt.savefig(\"feature-to-proto-type-nplots=%d\" % nrows, dpi=200, bbox_inches=\"tight\");"
      ],
      "id": "72e94cd7"
    },
    {
      "cell_type": "code",
      "execution_count": 2,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "7026ae0e",
        "outputId": "8eabab36-d787-421e-d088-845c44bce1f7"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Mounted at /content/gdrive\n"
          ]
        }
      ],
      "source": [
        "from google.colab import drive\n",
        "drive.mount(\"/content/gdrive\")"
      ],
      "id": "7026ae0e"
    },
    {
      "cell_type": "code",
      "execution_count": 4,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "cExQhSwdo2h3",
        "outputId": "41dbda49-ca12-425f-f839-a26ad714663b"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "/content/gdrive/MyDrive/Hopfield\n"
          ]
        }
      ],
      "source": [
        "%cd /content/gdrive/MyDrive/Hopfield/"
      ],
      "id": "cExQhSwdo2h3"
    },
    {
      "cell_type": "markdown",
      "source": [
        "Some Benchmarks\n",
        "---"
      ],
      "metadata": {
        "id": "eItSmmHflGID"
      },
      "id": "eItSmmHflGID"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "d775bd20"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "import torch\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "\n",
        "backend = \"cpu\"\n",
        "m = M = k = g = 200\n",
        "X = np.random.choice([-1, 1.], size=(M, k, g))\n",
        "X8 = X.astype(\"int8\")\n",
        "Q = np.random.choice([-1, 1.], size=(m, k, g))\n",
        "Q8 = Q.astype(\"int8\")\n",
        "Xtorch = torch.Tensor(X).to(backend.replace(\"gpu\", \"cuda\"))\n",
        "Xtorch8 = Xtorch.type(torch.int8)\n",
        "Qtorch = torch.Tensor(Q).to(Xtorch.device)\n",
        "Qtorch8 = Qtorch.type(torch.int8)\n",
        "device = jax.devices(backend=backend)[0]\n",
        "Xjax = jax.device_put(X, device)\n",
        "Xjax8 = Xjax.astype(\"int8\")\n",
        "Qjax = jax.device_put(Q, device)\n",
        "Qjax8 = Qjax.astype(\"int8\")"
      ],
      "id": "d775bd20"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "98f69e73",
        "outputId": "df1679d2-aa0b-40be-e361-4ac083be992c"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "1.13 s ± 46.6 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
          ]
        }
      ],
      "source": [
        "%%timeit\n",
        "np.einsum(\"mkg,Mkg->mMk\", Q, X);  # Numpy: floating point prec"
      ],
      "id": "98f69e73"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "Acylt5vQZkWr",
        "outputId": "c22f3903-cc04-4de0-dcab-b79891314134"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "433 ms ± 4.8 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
          ]
        }
      ],
      "source": [
        "%%timeit\n",
        "np.einsum(\"mkg,Mkg->mMk\", Q8, X8);  # Numpy: int8 prec"
      ],
      "id": "Acylt5vQZkWr"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "1e11b460",
        "outputId": "afb55ddb-3b0f-4f98-e29a-9d51179a57a2"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "53.5 ms ± 1.76 ms per loop (mean ± std. dev. of 7 runs, 10 loops each)\n"
          ]
        }
      ],
      "source": [
        "%%timeit\n",
        "jnp.einsum(\"mkg,Mkg->mMk\", Qjax, Xjax);  # JAX: floating point prec"
      ],
      "id": "1e11b460"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "K_qVp6wFaU-W",
        "outputId": "3fb702f2-8a6c-4a19-f911-d16d018255d5"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "671 ms ± 13.8 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
          ]
        }
      ],
      "source": [
        "%%timeit\n",
        "jnp.einsum(\"mkg,Mkg->mMk\", Qjax8, Xjax8);  # JAX: int8 prec"
      ],
      "id": "K_qVp6wFaU-W"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "5f9e9206",
        "outputId": "a6dbb94e-0e0d-449e-8046-c33d62be5d9f"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "11.1 ms ± 1.54 ms per loop (mean ± std. dev. of 100 runs, 10 loops each)\n"
          ]
        }
      ],
      "source": [
        "%%timeit -r 100 -n 10\n",
        "torch.einsum(\"mkg,Mkg->mMk\", Qtorch, Xtorch);  # PyTorch: floating point prec"
      ],
      "id": "5f9e9206"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "d2377a2d",
        "outputId": "bbe4f7e1-d075-4cd6-bf81-0bbcdefa8167"
      },
      "outputs": [
        {
          "name": "stdout",
          "output_type": "stream",
          "text": [
            "680 ms ± 4.62 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
          ]
        }
      ],
      "source": [
        "%%timeit\n",
        "torch.einsum(\"mkg,Mkg->mMk\", Qtorch8, Xtorch8);  # PyTorch: int8 prec"
      ],
      "id": "d2377a2d"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "ZILseSnBZs78"
      },
      "outputs": [],
      "source": [
        "from sklearn.utils import check_random_state\n",
        "\n",
        "\n"
      ],
      "id": "ZILseSnBZs78"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "0Q1-gI0cKOuF",
        "outputId": "e829aac5-9730-430c-b286-bcc020075faf"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "<module 'hopfield' from '/content/gdrive/MyDrive/Hopfield/hopfield.py'>"
            ]
          },
          "execution_count": 18,
          "metadata": {},
          "output_type": "execute_result"
        }
      ],
      "source": [
        "from importlib import reload\n",
        "import hopfield\n",
        "reload(hopfield)"
      ],
      "id": "0Q1-gI0cKOuF"
    },
    {
      "cell_type": "code",
      "execution_count": 20,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "a8rWcNMnKqQm",
        "outputId": "3d50e1c5-81dd-429a-ebf8-e66b03a44e10"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=1/5, gs=20, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=1/5, gs=25, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=1/5, gs=40, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=1/5, gs=50, M=20000, acc=96.11%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=1/5, gs=100, M=20000, acc=0.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=2/5, gs=20, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=2/5, gs=25, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=2/5, gs=40, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=2/5, gs=50, M=20000, acc=95.92%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=2/5, gs=100, M=20000, acc=0.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=3/5, gs=20, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=3/5, gs=25, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=3/5, gs=40, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=3/5, gs=50, M=20000, acc=96.08%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=3/5, gs=100, M=20000, acc=0.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=4/5, gs=20, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=4/5, gs=25, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=4/5, gs=40, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=4/5, gs=50, M=20000, acc=96.24%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=4/5, gs=100, M=20000, acc=0.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=5/5, gs=20, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=5/5, gs=25, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=5/5, gs=40, M=20000, acc=100.00%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=5/5, gs=50, M=20000, acc=96.02%\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "Pytoch: Using device cuda:0\n",
            "runid=5/5, gs=100, M=20000, acc=0.00%\n"
          ]
        }
      ],
      "source": [
        "rng = check_random_state(0)\n",
        "N = 200\n",
        "\n",
        "n_runs = 5\n",
        "res = []\n",
        "for runid in range(n_runs):\n",
        "    data = rng.choice([-1, 1.], size=(10000, N))\n",
        "    Ms = np.concatenate((np.arange(1, 20),\n",
        "                        np.arange(20, 100, 10),\n",
        "                        np.arange(100, 1100, 100),\n",
        "                        np.arange(1000, 21000, 1000)))\n",
        "    for group_size in range(20, N):\n",
        "        if N % group_size != 0:\n",
        "            continue\n",
        "        model = hopfield.SPSHopfieldNet(n_features=N, group_size=group_size)\n",
        "        for M in Ms:\n",
        "            X = data[:M]\n",
        "            Q = X\n",
        "            TQ = model.retrieve(X=X, Q=Q, backend=\"cuda\")\n",
        "            acc = np.all(TQ == X, axis=1).mean()\n",
        "            res.append(dict(runid=runid, group_size=group_size, M=M, acc=acc))\n",
        "        print(\"runid=%d/%d, gs=%d, M=%d, acc=%.2f%%\" % (runid + 1, n_runs,\n",
        "                                                        group_size, M,\n",
        "                                                        100 * acc))"
      ],
      "id": "a8rWcNMnKqQm"
    },
    {
      "cell_type": "code",
      "execution_count": 25,
      "metadata": {
        "id": "p3AH7xMOLJG5"
      },
      "outputs": [],
      "source": [
        "import pandas as pd\n",
        "import seaborn as sns\n",
        "\n",
        "df = pd.DataFrame(res)"
      ],
      "id": "p3AH7xMOLJG5"
    },
    {
      "cell_type": "code",
      "execution_count": 37,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "cNyizr2-LdYm",
        "outputId": "2fbfcb62-1c2c-4d56-f0b6-0a4175ca3c01"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "    runid   gs   k      M\n",
            "0       0   20  10  20000\n",
            "1       0   25   8  20000\n",
            "2       0   40   5  20000\n",
            "3       0   50   4   8000\n",
            "4       0  100   2     13\n",
            "5       1   20  10  20000\n",
            "6       1   25   8  20000\n",
            "7       1   40   5  20000\n",
            "8       1   50   4   9000\n",
            "9       1  100   2     11\n",
            "10      2   20  10  20000\n",
            "11      2   25   8  20000\n",
            "12      2   40   5  20000\n",
            "13      2   50   4   9000\n",
            "14      2  100   2     11\n",
            "15      3   20  10  20000\n",
            "16      3   25   8  20000\n",
            "17      3   40   5  20000\n",
            "18      3   50   4   8000\n",
            "19      3  100   2     10\n",
            "20      4   20  10  20000\n",
            "21      4   25   8  20000\n",
            "22      4   40   5  20000\n",
            "23      4   50   4   8000\n",
            "24      4  100   2     12\n"
          ]
        }
      ],
      "source": [
        "gamma = 0.02\n",
        "mask = \"acc >= %f\" % (1 - gamma)\n",
        "fixed = []\n",
        "for (runid, gs), subdf in df.query(mask).groupby([\"runid\", \"group_size\"]):\n",
        "    M = subdf[\"M\"].max()\n",
        "    k = N // gs\n",
        "    line = dict(runid=runid, gs=gs, k=k, M=M)\n",
        "    fixed.append(line)\n",
        "fixed = pd.DataFrame(fixed)\n",
        "print(fixed)"
      ],
      "id": "cNyizr2-LdYm"
    },
    {
      "cell_type": "code",
      "execution_count": 67,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 216
        },
        "id": "Qzjd0vj1OOen",
        "outputId": "5149bf62-4144-40e7-f86d-6d879c95c0c6"
      },
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 300x210 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "import seaborn as sns\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "fixed[\"bound\"] = fixed.gs ** (fixed.k - 1) / (2.1 * np.log(N))\n",
        "mask = fixed.k <= 4\n",
        "plt.figure(figsize=(3, 2.1))\n",
        "sns.lineplot(data=fixed[mask], x=\"k\", y=\"M\", err_style=\"bars\",\n",
        "             label=\"Experiment\", linewidth=2)\n",
        "sns.lineplot(data=fixed[mask], x=\"k\", y=\"bound\", linestyle=\"--\", c=\"k\",\n",
        "             label=\"Theory $\\\\frac{N_1^{k-1}}{2\\\\log N}$\", linewidth=2)\n",
        "plt.ylabel(\"Storage Capacity $M_N$\", fontsize=10)\n",
        "plt.xlabel(\"Number of groups $k$\", fontsize=10)\n",
        "plt.yscale(\"log\")\n",
        "plt.legend(loc=\"best\", frameon=False, fontsize=10, handlelength=.5)\n",
        "plt.tight_layout()\n",
        "plt.savefig(\"theory-storage-capacity-M=%d.pdf\" % Ms.max(), dpi=200,\n",
        "            bbox_inches=\"tight\")"
      ],
      "id": "Qzjd0vj1OOen"
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "Vj9rxewbPlg-"
      },
      "outputs": [],
      "source": [],
      "id": "Vj9rxewbPlg-"
    }
  ],
  "metadata": {
    "accelerator": "GPU",
    "colab": {
      "machine_shape": "hm",
      "provenance": []
    },
    "kernelspec": {
      "display_name": "Python 3 (ipykernel)",
      "language": "python",
      "name": "python3"
    },
    "language_info": {
      "codemirror_mode": {
        "name": "ipython",
        "version": 3
      },
      "file_extension": ".py",
      "mimetype": "text/x-python",
      "name": "python",
      "nbconvert_exporter": "python",
      "pygments_lexer": "ipython3",
      "version": "3.9.13"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 5
}