{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "4ce6f328",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "import sys\n",
    "sys.path.append(\"../../src\")\n",
    "import model.sdes as sdes\n",
    "import model.generate as generate\n",
    "import model.image_unet as image_unet\n",
    "import model.util as model_util\n",
    "from plot.plot import plot_mnist_digits\n",
    "import torch\n",
    "import torchvision\n",
    "import numpy as np\n",
    "import scipy.ndimage\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.font_manager as font_manager\n",
    "import os\n",
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5f452f18",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plotting defaults\n",
    "font_list = font_manager.findSystemFonts(fontpaths=[\"/home/anon/modules/fonts\"])\n",
    "for font in font_list:\n",
    "    font_manager.fontManager.addfont(font)\n",
    "plot_params = {\n",
    "    \"figure.titlesize\": 22,\n",
    "    \"axes.titlesize\": 22,\n",
    "    \"axes.labelsize\": 20,\n",
    "    \"legend.fontsize\": 18,\n",
    "    \"font.size\": 13,\n",
    "    \"xtick.labelsize\": 16,\n",
    "    \"ytick.labelsize\": 16,\n",
    "    \"font.family\": \"Roboto\",\n",
    "    \"font.weight\": \"bold\",\n",
    "    \"svg.fonttype\": \"none\"\n",
    "}\n",
    "plt.rcParams.update(plot_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "df22d86f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define device\n",
    "if torch.cuda.is_available():\n",
    "    DEVICE = \"cuda\"\n",
    "else:\n",
    "    DEVICE = \"cpu\""
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6dfb808b",
   "metadata": {},
   "source": [
    "### Define constants and paths"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "2cb7c2d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "models_base_path = \"/data/anon/branched_diffusion/models/trained_models/\"\n",
    "\n",
    "branched_model_path = os.path.join(models_base_path, \"mnist_continuous_alldigits/2/epoch_90_ckpt.pth\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "7ee483d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define the branches\n",
    "classes = list(range(10))\n",
    "branch_defs = [((0, 1, 2, 3, 4, 5, 6, 7, 8, 9), 0.4854854854854855, 1), ((1, 2, 3, 4, 5, 6, 7, 8, 9), 0.44744744744744747, 0.4854854854854855), ((2, 3, 4, 5, 6, 7, 8, 9), 0.43343343343343343, 0.44744744744744747), ((2, 3, 4, 5, 7, 8, 9), 0.4164164164164164, 0.43343343343343343), ((3, 4, 5, 7, 8, 9), 0.3743743743743744, 0.4164164164164164), ((3, 4, 5, 8, 9), 0.3683683683683684, 0.3743743743743744), ((3, 4, 5, 9), 0.35235235235235235, 0.3683683683683684), ((3, 4, 5), 0.3483483483483483, 0.35235235235235235), ((3, 5), 0.27127127127127126, 0.3483483483483483), ((0,), 0, 0.4854854854854855), ((1,), 0, 0.44744744744744747), ((6,), 0, 0.43343343343343343), ((2,), 0, 0.4164164164164164), ((7,), 0, 0.3743743743743744), ((8,), 0, 0.3683683683683684), ((9,), 0, 0.35235235235235235), ((4,), 0, 0.3483483483483483), ((5,), 0, 0.27127127127127126), ((3,), 0, 0.27127127127127126)]\n",
    "\n",
    "classes_049 = [0, 4, 9]\n",
    "branch_defs_049 = [\n",
    "    ((0, 4, 9), 0.5, 1),\n",
    "    ((0,), 0, 0.5),\n",
    "    ((4, 9), 0.35, 0.5),\n",
    "    ((4,), 0, 0.35),\n",
    "    ((9,), 0, 0.35)\n",
    "]\n",
    "\n",
    "classes_0497 = [0, 4, 9, 7]\n",
    "branch_defs_0497 = [\n",
    "    ((0, 4, 7, 9), 0.5, 1),\n",
    "    ((0,), 0, 0.5),\n",
    "    ((4, 7, 9), 0.38, 0.5),\n",
    "    ((7,), 0, 0.38),\n",
    "    ((4, 9), 0.35, 0.38),\n",
    "    ((4,), 0, 0.35),\n",
    "    ((9,), 0, 0.35)\n",
    "]\n",
    "\n",
    "classes_7 = [7]\n",
    "branch_defs_7 = [\n",
    "    ((7,), 0, 0.38)\n",
    "]\n",
    "\n",
    "input_shape = (1, 28, 28)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "5be5f5ec",
   "metadata": {},
   "outputs": [],
   "source": [
    "sde = sdes.VariancePreservingSDE(0.1, 20, input_shape)\n",
    "t_limit = 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "0364c0a2",
   "metadata": {},
   "outputs": [],
   "source": [
    "out_path = \"/home/anon/branched_diffusion/figures/mnist_efficiency\"\n",
    "\n",
    "os.makedirs(out_path, exist_ok=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c77b6452",
   "metadata": {},
   "source": [
    "### Create data loader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "98285ad2",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset = torchvision.datasets.MNIST(\n",
    "    \"/data/anon/datasets\", train=True, transform=(lambda img: (np.asarray(img)[None] / 256 * 2) - 1)\n",
    ")\n",
    "\n",
    "data_loader = torch.utils.data.DataLoader(dataset, batch_size=128, shuffle=True, num_workers=2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "51fbc829",
   "metadata": {},
   "outputs": [],
   "source": [
    "# TODO: this is currently rather inefficient; a decision-tree-style structure\n",
    "# would be better\n",
    "\n",
    "def class_time_to_branch(c, t, branch_defs):\n",
    "    \"\"\"\n",
    "    Given a class and a time (both scalars), return the\n",
    "    corresponding branch index.\n",
    "    \"\"\"\n",
    "    for i, branch_def in enumerate(branch_defs):\n",
    "        if c in branch_def[0] and t >= branch_def[1] and t <= branch_def[2]:\n",
    "            return i\n",
    "    raise ValueError(\"Undefined class and time\")\n",
    "        \n",
    "def class_time_to_branch_tensor(c, t, branch_defs):\n",
    "    \"\"\"\n",
    "    Given tensors of classes and a times, return the\n",
    "    corresponding branch indices as a tensor.\n",
    "    \"\"\"\n",
    "    return torch.tensor([\n",
    "        class_time_to_branch(c_i, t_i, branch_defs) for c_i, t_i in zip(c, t)\n",
    "    ], device=DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c35fc1e8",
   "metadata": {},
   "source": [
    "### Import models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "40e98368",
   "metadata": {},
   "outputs": [],
   "source": [
    "branched_model = model_util.load_model(\n",
    "    image_unet.MultitaskMNISTUNetTimeConcat, branched_model_path\n",
    ").to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5f55c8b",
   "metadata": {},
   "source": [
    "### Generating multiple digit classes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "c1e50cd6",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 79s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 79s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 78s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 78s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 78s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 78s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 78s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 78s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 78s\n",
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 2\n",
      "Sampling class: 3\n",
      "Sampling class: 4\n",
      "Sampling class: 5\n",
      "Sampling class: 6\n",
      "Sampling class: 7\n",
      "Sampling class: 8\n",
      "Sampling class: 9\n",
      "Total time taken: 78s\n",
      "Average time taken: 78.73 +/- 0.11\n"
     ]
    }
   ],
   "source": [
    "# Sample each class individually without taking advantage of branches\n",
    "num_linear_trials = 10\n",
    "linear_times = []\n",
    "for _ in range(num_linear_trials):\n",
    "    time_a = time.time()\n",
    "    for class_to_sample in classes:\n",
    "        print(\"Sampling class: %s\" % class_to_sample)\n",
    "        generate.generate_continuous_branched_samples(\n",
    "            branched_model, sde, class_to_sample,\n",
    "            lambda c, t: class_time_to_branch_tensor(c, t, branch_defs),\n",
    "            sampler=\"pc\", t_limit=t_limit, num_steps=1000\n",
    "        )\n",
    "    time_b = time.time()\n",
    "    time_taken = time_b - time_a\n",
    "    linear_times.append(time_taken)\n",
    "    print(\"Total time taken: %ds\" % time_taken)\n",
    "print(\"Average time taken: %.2f +/- %.2f\" % (np.mean(linear_times), np.std(linear_times) / np.sqrt(len(linear_times))))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "0de8f9d2",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Sampling branch 1/19\n",
      "Sampling branch 2/19\n",
      "Sampling branch 3/19\n",
      "Sampling branch 4/19\n",
      "Sampling branch 5/19\n",
      "Sampling branch 6/19\n",
      "Sampling branch 7/19\n",
      "Sampling branch 8/19\n",
      "Sampling branch 9/19\n",
      "Sampling branch 10/19\n",
      "Sampling branch 11/19\n",
      "Sampling branch 12/19\n",
      "Sampling branch 13/19\n",
      "Sampling branch 14/19\n",
      "Sampling branch 15/19\n",
      "Sampling branch 16/19\n",
      "Sampling branch 17/19\n",
      "Sampling branch 18/19\n",
      "Sampling branch 19/19\n",
      "Total time taken: 37s\n",
      "Average time taken: 37.30 +/- 0.03\n"
     ]
    }
   ],
   "source": [
    "# Sample each digit by taking advantage of branches\n",
    "num_branched_trials = 10\n",
    "branched_times = []\n",
    "\n",
    "# Sort the branches by starting time point (in reverse order), and generate along those\n",
    "# branches, caching results; this guarantees that we will always find a cached batch\n",
    "# (other than the first one)\n",
    "cache = {}\n",
    "sorted_branch_defs = sorted(branch_defs, key=(lambda t: -t[1]))\n",
    "\n",
    "for _ in range(num_branched_trials):\n",
    "    time_a = time.time()\n",
    "    # First branch\n",
    "    print(\"Sampling branch 1/%d\" % len(sorted_branch_defs))\n",
    "    branch_def = sorted_branch_defs[0]\n",
    "    samples = generate.generate_continuous_branched_samples(\n",
    "        # Specify arbitrary class\n",
    "        branched_model, sde, branch_def[0][0],\n",
    "        lambda c, t: class_time_to_branch_tensor(c, t, branch_defs),\n",
    "        sampler=\"pc\", t_limit=branch_def[2], t_start=branch_def[1],\n",
    "        num_steps=int(1000 * (branch_def[2] - branch_def[1]))\n",
    "    )\n",
    "    for class_i in branch_def[0]:\n",
    "        cache[class_i] = (branch_def[1], samples)\n",
    "\n",
    "    for i, branch_def in enumerate(sorted_branch_defs[1:]):\n",
    "        print(\"Sampling branch %d/%d\" % (i + 2, len(sorted_branch_defs)))\n",
    "        cached_time, cached_samples = cache[branch_def[0][0]]\n",
    "        assert cached_time == branch_def[2]\n",
    "        samples = generate.generate_continuous_branched_samples(\n",
    "            branched_model, sde, branch_def[0][0],\n",
    "            lambda c, t: class_time_to_branch_tensor(c, t, branch_defs),\n",
    "            sampler=\"pc\", t_limit=branch_def[2], t_start=branch_def[1],\n",
    "            num_steps=int(1000 * (branch_def[2] - branch_def[1])),\n",
    "            initial_samples=cached_samples\n",
    "        )\n",
    "        for class_i in branch_def[0]:\n",
    "            cache[class_i] = (branch_def[1], samples)\n",
    "\n",
    "    time_b = time.time()\n",
    "    time_taken = time_b - time_a\n",
    "    branched_times.append(time_taken)\n",
    "    print(\"Total time taken: %ds\" % time_taken)\n",
    "print(\"Average time taken: %.2f +/- %.2f\" % (np.mean(branched_times), np.std(branched_times) / np.sqrt(len(branched_times))))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "38c99f3d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(8, 8))\n",
    "labels = [\"Linear\", \"Branched\"]\n",
    "times = [np.mean(linear_times), np.mean(branched_times)]\n",
    "errors = [np.std(linear_times) / np.sqrt(num_linear_trials), np.std(branched_times) / np.sqrt(num_branched_trials)]\n",
    "ax.bar(labels, times, color=\"mediumorchid\")\n",
    "ax.errorbar(labels, times, yerr=errors, fmt=\"none\", color=\"darkmagenta\")\n",
    "ax.set_xlabel(\"Time to generate one batch of each class\")\n",
    "ax.set_ylabel(\"Seconds\")\n",
    "plt.show()\n",
    "fig.savefig(\n",
    "    os.path.join(out_path, \"mnist_efficiency.svg\"),\n",
    "    format=\"svg\"\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
