{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "import matplotlib\n",
    "from matplotlib import pyplot as plt\n",
    "from scipy.stats import truncnorm, bernoulli\n",
    "from torch import nn"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<torch._C.Generator at 0x7f11cc3df570>"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "num_samples = 33000\n",
    "\n",
    "seed = 2009\n",
    "rng = np.random.default_rng(seed)\n",
    "\n",
    "torch.manual_seed(seed)  # global seed\n",
    "# torch.use_deterministic_algorithms(True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "p1_dist = truncnorm(a=-1, b=1, loc=-1, scale=0.1)\n",
    "p2_dist = truncnorm(a=-1, b=1, loc=1, scale=0.1)\n",
    "\n",
    "def logp(x):\n",
    "    # Note: p1 & p2 will return -np.inf if out of domain\n",
    "    p1_logprob = p1_dist.logpdf(x)\n",
    "    p2_logprob = p2_dist.logpdf(x)\n",
    "    return 0.5 * np.maximum(p1_logprob, p2_logprob)\n",
    "\n",
    "q1_dist = truncnorm(a=-1, b=1, loc=-1, scale=0.2)\n",
    "q2_dist = truncnorm(a=-1, b=1, loc=1, scale=0.2)\n",
    "\n",
    "def logq(x):\n",
    "    # Note: q1 & q2 will return -np.inf if out of domain\n",
    "    q1_logprob = q1_dist.logpdf(x)\n",
    "    q2_logprob = q2_dist.logpdf(x)\n",
    "    return 0.5 * np.maximum(q1_logprob, q2_logprob)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "# NOTE: truncnorm parameters a, b are defined over the domain of a standard normal,\n",
    "# so low = scale*a + loc, high = scale*b + loc\n",
    "\n",
    "p1 = p1_dist.rvs(num_samples, random_state=rng)\n",
    "p2 = p2_dist.rvs(num_samples, random_state=rng)\n",
    "u = bernoulli(0.5).rvs(num_samples, random_state=rng)\n",
    "p = (p1 * u) + (p2 * (1 - u))\n",
    "\n",
    "q1 = q1_dist.rvs(num_samples, random_state=rng)\n",
    "q2 = q2_dist.rvs(num_samples, random_state=rng)\n",
    "u = bernoulli(0.5).rvs(num_samples, random_state=rng)\n",
    "q = (q1 * u) + (q2 * (1 - u))\n",
    "\n",
    "m = truncnorm(a=-1.2, b=1.2, loc=0, scale=1.0).rvs(num_samples, random_state=rng)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "_ = plt.hist(p, bins=100, density=True, label=\"p\", alpha=0.4)\n",
    "_ = plt.hist(q, bins=100, density=True, label=\"q\", alpha=0.4)\n",
    "_ = plt.hist(m, bins=100, density=True, label=\"m\", alpha=0.4)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [],
   "source": [
    "class MLP(nn.Module):\n",
    "    \n",
    "    def __init__(self, n_layers=1, n_hiddens=128, insize=1, outsize=2):\n",
    "        super().__init__()\n",
    "        self.fc1 = nn.Linear(insize, n_hiddens)\n",
    "        self.fc2 = nn.Linear(n_hiddens, outsize)\n",
    "        self.relu = nn.ReLU()\n",
    "    \n",
    "    def forward(self, x):\n",
    "        if isinstance(x, np.ndarray):\n",
    "            x = torch.from_numpy(x.astype(np.float32))\n",
    "        x = self.fc1(x)\n",
    "        x = self.relu(x)\n",
    "        x = self.fc2(x)\n",
    "        return x\n",
    "    \n",
    "def copy_model(m1, m2):\n",
    "    params1 = m1.named_parameters()\n",
    "    params2 = m2.named_parameters()\n",
    "\n",
    "    dict_params2 = dict(params2)\n",
    "\n",
    "    for name1, param1 in params1:\n",
    "        if name1 in dict_params2:\n",
    "            dict_params2[name1].data.copy_(param1.data)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Losses"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_tre_loss(model, p_batch, q_batch, m_batch):\n",
    "    \n",
    "    batch_size = len(p_batch)\n",
    "    \n",
    "    # Model has 2-dimensional output. \n",
    "    # First dim classifies between p & m, whilst second dim classifies between m & q.\n",
    "    p_logits = model(p_batch)  # (batch_size, 2)\n",
    "    q_logits = model(q_batch)  # (batch_size, 2)\n",
    "    m_logits = model(m_batch)  # (batch_size, 2)\n",
    "    \n",
    "    loss1_logits = torch.cat([p_logits[:, 0], m_logits[:, 0]], dim=0)\n",
    "    loss1_targets = torch.cat([torch.zeros(batch_size), torch.ones(batch_size)], dim=0).to('cuda')\n",
    "    loss1 = nn.BCEWithLogitsLoss()(loss1_logits, loss1_targets)\n",
    "    \n",
    "    loss2_logits = torch.cat([m_logits[:, 1], q_logits[:, 1]], dim=0)\n",
    "    loss2_targets = torch.cat([torch.zeros(batch_size), torch.ones(batch_size)], dim=0).to('cuda')\n",
    "    loss2 = nn.BCEWithLogitsLoss()(loss2_logits, loss2_targets)\n",
    "    \n",
    "    return 0.5 * (loss1 + loss2)\n",
    "    \n",
    "\n",
    "def compute_binary_sdre_loss(model, p_batch, q_batch, m_batch):\n",
    "    \n",
    "    batch_size = len(p_batch)\n",
    "    \n",
    "    # Model has 2-dimensional output. \n",
    "    # First dim classifies between p & m, whilst second dim classifies between q & m.\n",
    "    p_logits = model(p_batch)  # (batch_size, 2)\n",
    "    q_logits = model(q_batch)  # (batch_size, 2)\n",
    "    m_logits = model(m_batch)  # (batch_size, 2)\n",
    "    \n",
    "    loss1_logits = torch.cat([p_logits[:, 0], m_logits[:, 0]], dim=0)\n",
    "    loss1_targets = torch.cat([torch.ones(batch_size), torch.zeros(batch_size)], dim=0)\n",
    "    loss1 = nn.BCEWithLogitsLoss()(loss1_logits, loss1_targets)\n",
    "    \n",
    "    # Note: the 0-1 targets for q & m are switched (relative to TRE)\n",
    "    loss2_logits = torch.cat([q_logits[:, 1], m_logits[:, 1]], dim=0)\n",
    "    loss2_targets = torch.cat([torch.ones(batch_size), torch.zeros(batch_size)], dim=0)\n",
    "    loss2 = nn.BCEWithLogitsLoss()(loss2_logits, loss2_targets)\n",
    "    \n",
    "    return 0.5 * (loss1 + loss2)\n",
    "\n",
    "\n",
    "def compute_multiclass_sdre_loss(model, p_batch, q_batch, m_batch):\n",
    "    \n",
    "    batch_size = len(p_batch)\n",
    "    \n",
    "    # Model has 2-dimensional output. \n",
    "    # First dim classifies between p & m, whilst second dim classifies between q & m.\n",
    "    all_inputs = torch.cat([p_batch, q_batch, m_batch], dim=0)\n",
    "    all_logits = model(all_inputs)  # (3* batch_size, 3)\n",
    "    \n",
    "    # logits for reference class (m) are fixed to 1\n",
    "#     ones = torch.ones((3*batch_size, 1), dtype=torch.float32)\n",
    "#     all_logits = torch.cat([all_logits], dim=1)  # (batch_size, 3)\n",
    "\n",
    "    \n",
    "    p_targets = torch.ones(batch_size) * 0\n",
    "    q_targets = torch.ones(batch_size) * 1\n",
    "    m_targets = torch.ones(batch_size) * 2\n",
    "    all_targets = torch.cat([p_targets, q_targets, m_targets]).long().to('cuda')\n",
    "    \n",
    "    loss = nn.CrossEntropyLoss()(all_logits, all_targets)\n",
    "    \n",
    "    return loss\n",
    "    \n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Training code"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {},
   "outputs": [],
   "source": [
    "from tqdm import trange\n",
    "\n",
    "def transform(x):\n",
    "    return torch.from_numpy(x.astype(np.float32)).view(-1, 1)\n",
    "\n",
    "def train_model(model, loss_type, num_iters=100000, batch_size=512, lr=1e-3):\n",
    "    \n",
    "    rng = np.random.default_rng(5672) # ensure all models use same minibatches\n",
    "    \n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=lr)\n",
    "    model = model.to('cuda')\n",
    "    for i in trange(num_iters):\n",
    "        \n",
    "        # sample minibatch\n",
    "        p_batch = transform(rng.choice(p, size=batch_size)).to('cuda')\n",
    "        q_batch = transform(rng.choice(q, size=batch_size)).to('cuda')\n",
    "        m_batch = transform(rng.choice(m, size=batch_size)).to('cuda')\n",
    "        \n",
    "        if loss_type == \"tre\":\n",
    "            loss = compute_tre_loss(model, p_batch, q_batch, m_batch)\n",
    "        elif loss_type == \"binary_sdre\":\n",
    "            loss = compute_binary_sdre_loss(model, p_batch, q_batch, m_batch)\n",
    "        elif loss_type == \"multiclass_sdre\":\n",
    "            loss = compute_multiclass_sdre_loss(model, p_batch, q_batch, m_batch)\n",
    "        \n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        optimizer.zero_grad()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [],
   "source": [
    "tre_model = MLP()\n",
    "binary_sdre_model = MLP()\n",
    "multiclass_sdre_model = MLP(outsize=3)\n",
    "\n",
    "copy_model(binary_sdre_model, tre_model)  # ensure initial weights are identical to tre model\n",
    "# copy_model(multiclass_sdre_model, tre_model)  # ensure initial weights are identical to tre model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "MLP(\n",
       "  (fc1): Linear(in_features=1, out_features=128, bias=True)\n",
       "  (fc2): Linear(in_features=128, out_features=3, bias=True)\n",
       "  (relu): ReLU()\n",
       ")"
      ]
     },
     "execution_count": 39,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "multiclass_sdre_model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100000/100000 [05:42<00:00, 291.76it/s]\n"
     ]
    }
   ],
   "source": [
    "train_model(tre_model, \"tre\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  1%|▏         | 1272/100000 [00:03<04:29, 366.32it/s]\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-14-ce09d9fb1400>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0mtrain_model\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mbinary_sdre_model\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"binary_sdre\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
      "\u001b[0;32m<ipython-input-11-bb891a4882f6>\u001b[0m in \u001b[0;36mtrain_model\u001b[0;34m(model, loss_type, num_iters, batch_size, lr)\u001b[0m\n\u001b[1;32m     23\u001b[0m             \u001b[0mloss\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mcompute_multiclass_sdre_loss\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mmodel\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mp_batch\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mq_batch\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mm_batch\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     24\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 25\u001b[0;31m         \u001b[0mloss\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbackward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     26\u001b[0m         \u001b[0moptimizer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mstep\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     27\u001b[0m         \u001b[0moptimizer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mzero_grad\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/ananconda3/envs/torch1.8/lib/python3.8/site-packages/torch/tensor.py\u001b[0m in \u001b[0;36mbackward\u001b[0;34m(self, gradient, retain_graph, create_graph, inputs)\u001b[0m\n\u001b[1;32m    243\u001b[0m                 \u001b[0mcreate_graph\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mcreate_graph\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    244\u001b[0m                 inputs=inputs)\n\u001b[0;32m--> 245\u001b[0;31m         \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mautograd\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mbackward\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mgradient\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mretain_graph\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcreate_graph\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0minputs\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0minputs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    246\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    247\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0mregister_hook\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mhook\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m/disk_c/han/ananconda3/envs/torch1.8/lib/python3.8/site-packages/torch/autograd/__init__.py\u001b[0m in \u001b[0;36mbackward\u001b[0;34m(tensors, grad_tensors, retain_graph, create_graph, grad_variables, inputs)\u001b[0m\n\u001b[1;32m    143\u001b[0m         \u001b[0mretain_graph\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mcreate_graph\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    144\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 145\u001b[0;31m     Variable._execution_engine.run_backward(\n\u001b[0m\u001b[1;32m    146\u001b[0m         \u001b[0mtensors\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mgrad_tensors_\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mretain_graph\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcreate_graph\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0minputs\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    147\u001b[0m         allow_unreachable=True, accumulate_grad=True)  # allow_unreachable flag\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "train_model(binary_sdre_model, \"binary_sdre\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|███▉      | 39595/100000 [01:18<02:03, 489.28it/s]"
     ]
    }
   ],
   "source": [
    "train_model(multiclass_sdre_model, \"multiclass_sdre\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Estimate KL between p & q"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "True KL: 0.29198621762480786 \n",
      " TRE KL: -0.1928389072418213\n"
     ]
    }
   ],
   "source": [
    "p = p.reshape(-1, 1)  # samples from p\n",
    "true_kl = (logp(p) - logq(p)).mean()\n",
    "\n",
    "# tre_logits = tre_model(torch.from_numpy(p).float().to('cuda')).detach().cpu().numpy()\n",
    "# tre_kl = (tre_logits[:, 0] + tre_logits[:, 1]).mean()\n",
    "\n",
    "# binary_sdre_logits = binary_sdre_model(p).detach().numpy()\n",
    "# binary_sdre_kl = (binary_sdre_logits[:, 0] - binary_sdre_logits[:, 1]).mean()\n",
    "\n",
    "# print(f\"True KL: {true_kl} \\n TRE KL: {tre_kl} \\n binary sDRE KL: {binary_sdre_kl} \"\n",
    "#       f\"\\n multiclass sDRE KL: {multiclass_sdre_kl}\")\n",
    "print(f\"True KL: {true_kl} \\n TRE KL: {tre_kl}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "p = p.reshape(-1, 1)  # samples from p\n",
    "true_kl = (logp(p) - logq(p)).mean()\n",
    "\n",
    "# tre_logits = tre_model(torch.from_numpy(p).float().to('cuda')).detach().cpu().numpy()\n",
    "# tre_kl = (tre_logits[:, 0] + tre_logits[:, 1]).mean()\n",
    "\n",
    "# binary_sdre_logits = binary_sdre_model(p).detach().numpy()\n",
    "# binary_sdre_kl = (binary_sdre_logits[:, 0] - binary_sdre_logits[:, 1]).mean()\n",
    "\n",
    "multiclass_sdre_logits = multiclass_sdre_model(torch.from_numpy(p).float().to('cuda')).detach().cpu().numpy()\n",
    "multiclass_sdre_kl = (multiclass_sdre_logits[:, 0] - multiclass_sdre_logits[:, 1]).mean()\n",
    "\n",
    "# print(f\"True KL: {true_kl} \\n TRE KL: {tre_kl} \\n binary sDRE KL: {binary_sdre_kl} \"\n",
    "#       f\"\\n multiclass sDRE KL: {multiclass_sdre_kl}\")\n",
    "# print(f\"True KL: {true_kl} \\n TRE KL: {tre_kl}\")\n",
    "print(f\"\\n multiclass sDRE KL: {multiclass_sdre_kl}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Plot estimated log p / q from all models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "x_axis = torch.linspace(-1.5, 1.5, 512).view(-1, 1)\n",
    "\n",
    "tre_logits = tre_model(x_axis).detach().numpy()\n",
    "tre_p_over_q = tre_logits[:, 0] + tre_logits[:, 1]\n",
    "\n",
    "binary_sdre_logits = binary_sdre_model(x_axis).detach().numpy()\n",
    "binary_sdre_p_over_q = binary_sdre_logits[:, 0] - binary_sdre_logits[:, 1]\n",
    "\n",
    "multiclass_sdre_logits = multiclass_sdre_model(x_axis).detach().numpy()\n",
    "multiclass_sdre_p_over_q = multiclass_sdre_logits[:, 0] - multiclass_sdre_logits[:, 1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/w0/wqfh_8bn4cs94nhj93py3hk00000gp/T/ipykernel_10289/1667444262.py:2: RuntimeWarning: invalid value encountered in subtract\n",
      "  true_ratio = logp(x_ax) - logq(x_ax)\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAegAAAI/CAYAAAC8g0oGAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/MnkTPAAAACXBIWXMAAAsTAAALEwEAmpwYAACcSElEQVR4nOzdd3hUVfrA8e+Z9A4hjTTSaKFD6F0RUBAQwXXVVWxgXXV/rmV1Xd1dXcuubW2gImJdQaUoiIBSpIgJvZOEAAkJSQiklynn90diBKUEksydmbyf55knM3du7nnnZjLvnHNPUVprhBBCCOFYTEYHIIQQQojfkgQthBBCOCBJ0EIIIYQDkgQthBBCOCBJ0EIIIYQDkgQthBBCOCB3owM4VUhIiI6LizM6DCGEEMJu0tLSCrXWob/e7lAJOi4ujtTUVKPDEEIIIexGKXXoTNuliVsIIYRwQJKghRBCCAckCVoIIYRwQM2eoJVSY5VS+5RS6UqpR5q7PCGEEMIVNGsnMaWUG/A6cBmQDfyklFqktd7dnOUKIYSwn2qLlaLyGo6X1VBSZaZ3bGu8PdyMDsvpNXcv7n5AutY6E0Ap9SkwEZAE3QjFJ47z7y/WsutwPr7KjI+ppvanMuOjavBRZjyVBTcFbkrjpsB0yn0fDzc8vbzx8vbGx9sbb28f/Hy88fXxwd/Xj6CgQLx9A8DTHzx8wS8U3D2NftlCiMaoKYeKIrBUgbkCzHU/LVVgrqy9WavRNhs1FgvlVWYqqutuNWYqqi2Um6HErCipURTXwMlqTXE1lJoVlXhSqb2owJuRXeN4+JoR4OFj9Kt2as2doKOAI6c8zgb6N3OZLu+n+c/zj5y3QNVtsDVzgTd9BfFDm7kQIUSz+uldWP7X8+6mAK+6W3BDjquAX39/PwAHt3xEfL/xFxqlOIXh46CVUtOB6QCxsbEGR+P4aiw23jranpzwR7lpWKfab6ju3r/96e4FygQoUHW3n+8DWC1oazUVlVWUlJVTUlZBSUUF5eXlbM44yu5DeSS1gmkpYUSEdDDwFQshmkTiJVR5BLJw53FWppdSVVfjxcObAP8AWgUF0ioggCB/L1r5eNHKz4vWfl609vOmtZ83rfw88HPTKJsFbGaw1oC17r6lur4WXllewj8WpKL3evOvfka/aOfW3Ak6B4g55XF03bZ6WutZwCyAlJQU3czxOL2Ve46RWtmWuy+ZAB3DGnUsBfgFgl84tD1l+4hRsGxXHo8v2MlHq63MjHVjUECjihJCGGx9WQR//i6Wo8Wh3DYknsm9o4kM8iHQxx318xf3JuADBOUlM3N1BnccL6ddG78mO3ZL09y9uH8C2iul4pVSnsC1wKJmLtOlfZZ6hPBAL4a1/82scE1qTJcIFt49mLatvLnpvU0s3Jpz/l8SQjic8moLTyzcyXXv/Iinu4l5Mwby2LhkOrcNJMjXo0mT889uHhSHu8nE22szm/zYLUmzJmittQW4B1gG7AE+01rvas4yXdmxkipW7y9gcu9o3ExN/0/1a5GtfJh3xyB6x7bmvk+38uaqDLSWRg4hnMWPmce5/JW1fLDxELcMjmfJH4eSEtegK8uNEhbozeTeUcxLzaawrLrZy3NVzT4OWmu9RGvdQWudqLV+urnLc2VfbM7BpmFqn2i7lRnk48HcW/txZY9InvtmL08s3IXVJklaCEdWWWPlqcW7uPbtjQB8evsAnrgyGR9P+w19un1YAjVWG++vz7Jbma7G8E5iomG01sxLO0JKu9YkhPrbtWwvdzde+V1PIoO8mbkmk7ySKl69tpdd/9mFEA2TmlXEn+dv52BhOTcNbMfDl3fC19P+H/WJof6MTg5n7oZD3DE8ET8vSTcXSqb6dBKbD58ks6CcqSn2qz2fymRSPHpFZ568MpkVe45x3TsbKSqvMSQWIcRvVZmtPP31bqbO3IDZauPj2/vz1MSuhiTnn80YnkhxpZn//XTk/DuL35AE7STmpx3Bx8ONcd0jDY1j2uB43ry+D7uPlnD1m+s5dLzc0HiEELDl8AmueHUtb689yHX9Yvnm/mEMSgwxOix6x7amX3ww7/5wELO1uSdscD2SoJ1AZY2VxdtyubxbBP4O0Ew0tmsEH9/enxMVNUx+Yz3bjpw0OiQhWqQqs5Vnl+7l6jfXU1Vj5YNb+/H0Vd0c4nPiZ3cMTyDnZCVfbT9qdChORxK0E/hmVy5l1Ram9ok5/8520qddMJ/fOQhfLzeunbWR7/YeMzokIVqU7dknufK/P/DW6gyuSYnhmweGMbSZh19ejBEdwugQ7s/M1ZkyCuQCSYJ2Ap/9lE1MsA/945t/eMSFSAz15/M7B5EU5s9t76fy8Y+HjQ5JCJdXY7Hxn2/3cdUb6ymtsvDezX159uruBHp7GB3aGZlMiunDEtmbV8rq/QVGh+NUJEE7uCNFFWzIPM6U3jGY7DD2+UKFBXjz6fQBDOsQyl++3MHTX+/GIteahGgWO3OKmfDaD/z3u3Su6hXFsgeGMbKRMwraw4QekbQN8uat1RlGh+JUJEE7uPlp2SgFV/eJMjqUs/LzcuedG1O4cWA73l57kGnv/cQJ6eEtRJMxW228vGI/k15fx/HyGt69KYV/T+1BkI9j1pp/zdPdxK1D4tmYWcRW6bPSYJKgHZjNppmfls3gxBCiW/saHc45ubuZ+PvErjw/pTubDhZx5Ws/sPtoidFhCeH09uSWMOn1dby84gBX9ohk+QPDuLRzuNFhXbBr+8US6O3OrDVSi24oSdAObGPmcXJOVho29vliXJMSw2d3DMRi1Ux+cx2Lt0nPTSEuhsVq47XvDjDhtR84VlLFzD/04aXf9aSVr3Ouze7v5c4fBrZj6c48DhbK8MyGkATtwOalZRPg7c6YLhFGh3JBesa0YtG9g+kWFcS9n2zhX0v2yPSgQlyA/cdKmfzmev797X7Gdm3Ltw8Md7rPgTO5aVAcHm6yiEZDSYJ2UCVVZpbuzOXKHpF4ezjflJphAd58dNsAbhgQy8w1mUx7bxMnK+S6tBDnYrHaeHNVBuNf/YHsE5W8cX1v/vv7XgT7OWet+dfCAry5unc089OyKSiVRTTORxK0g/p6ey5VZptdF8Zoap7uJv45qRvPXd2NHzOLmPDaOvbkynVpIc4kPb+MKW9t4Llv9nJp5zC+fWAYV3Rre/5fdDLThyVgttqYs/6g0aE4PEnQDmpe6hGSwvzpGdPK6FAa7Xd9Y/l0xgCqLVYmv7FeZhQS4hRWm+btNZlc8epaso6X8+rve/HG9b0J8fcyOrRmER/ix9guEXyw4RBl1Rajw3FokqAdUHp+KZsPn2Rqn+hmWUzdCL1jW7P4niEkRwZyz8dbeHbpXrkuLVq8g4XlXDNzA08v2cPwDqF8+8AwJvSIdJn/+7OZMTyRkioLn26SyY3ORRK0A5qXlo2bSXFVb8cd+3wxwgK9+eT2AVzfP5a3Vmdw85yf5Lq0aJFsNs3sHw5y+StrOHCslJd+14NZf+hDWIC30aHZRc+YVgxIqF1Eo8YiExudjSRoB2Ox2vhicw4jO4a65D+rp7uJp6/qxr8md2NDRiETXlvH3jy5Li1ajkPHy7n27Y38/avdDEoMYfmfhnNVL9dpLWuoGcMTyS2ukqGY5yAJ2sGsOVBAQWk1UxxoYYzm8Pt+sXw6fSBV5trr0kt25BodkhDNymbTzN2QxdiX17LnaAkvTOnOuzelEB7oel/EG2JEh1A6hgcwc02GLKJxFpKgHcy81GyC/Ty5pJPjz6/bWH3atWbxvUPoFBHAXR9t5vlv5Lq0cE1Hiiq4/p0feWLhLvrGB/Ptn4YxNSWmxdWaT6WUYsbwBPYfK+P7fflGh+OQJEE7kKLyGlbsOcaknlF4ureMP014oDefTB/A7/vF8MaqDG59/yeKK8xGhyVEk9Ba89GPhxj78hp25BTz7ORuvH9zX9oG+RgdmkO4skckkUHevLVaJi45k5aRBZzEwq05mK3aqab2bApe7m78a3J3nr6qK+vSC5n4+g/sP1ZqdFhCNErOyUpunL2Jx77cSc/YVnxz/1Cu7RfbomvNv+bhZuLWoQlsOljE5sMnjA7H4UiCdiDzUrPpGhVI57aBRodiiOv7t+OT2wdQXmPlqtfX8c1OuS4tnI/Wmv/9dJgxL60h7dAJ/jmpKx/e2t/hF7wxyrV9Ywjy8WCW1KJ/QxK0g9iZU8zu3BKmunjnsPNJiQtm8T1DaB8ewB0fbubfy/Zhk+vSwknkFlcy7b2fePjzHXSNCmTZ/cO4YUA7qTWfg5+XOzcObMey3XlkFJQZHY5DkQTtIOanZePpZmJiz0ijQzFcRJA3/5sxgN+lxPDa9+ncNjeV4kq5Li0cl9a1S8OOfmkNmw4W8dSELnx82wBigqXW3BA/L6LxjiyicRpJ0A6g2mJlwdYcLusS7rRLyTU1L3c3nr26G/+Y1JU1+wuY9Po6Dsh1aeGA8kuquO39VB6ct41OEQEsvW8oNw2Kw2SSWnNDhfh7MbVPNJ+n5ZBfWmV0OA5DErQDWLknn5MVZqdeGKM5KKX4w4B2fDJ9AKVVFia9vo5lu/KMDksIoLbWvGBLDpe9tIYf0gv56/hk/jd9IHEhfkaH5pRuH5qAxWbjvXVZRofiMCRBO4B5qUeICPRmaPtQo0NxSH3jgll872CSwgOY8UEaLy7fL9elhaEKSquZ8UEa9/9vK4mhfiy9byi3DomXWnMjxIX4cXnXtny48RClVXJJCyRBG+5YSRWr9xcwuXcUbvLPfVZtg3z43/QBTO0TzasrDzD9g1RK5J9YGGDxtqOMfmk1q/YX8JcrOjHvjkEkhPobHZZLmDE8gdIqC59uOmJ0KA5BErTBvticg03DFGnePi9vDzeen9Kdv0/swqp9tdel0/Ol16ewj+Nl1dz90Wbu/WQLscG+LPnjEKYPS5Qv1k2oe3QrBiW2kUU06jQqQSulXlBK7VVKbVdKfamUanXKc48qpdKVUvuUUmMaHakL0lozL+0IKe1ayzfwBlJKcePAOD66rT8llWYmvb6O5buPGR2WcHFLd+Qy+qU1LN99jIfGduTzOweRFBZgdFguacbwRPJKqli4NcfoUAzX2Br0cqCr1ro7sB94FEAplQxcC3QBxgJvKKXcGlmWy9l8+CSZBeUtbuawptA/oQ2L7hlCQqgft89N5eUVcl1aNL0T5TX88ZMt3PnRZtq28mbxvUO4a0QS7m7S+NhchrUPoVNEALPWZLb4/+lGvcu01t9qrS11DzcCP2eaicCnWutqrfVBIB3o15iyXNG81CP4eLgxrruMfb4Yka18+GzGQK7uHc3LKw4w/YM06Vwimsy3u/K47KU1LN2Zy/9d1oEv7xpMxwipNTc3pRR3DE/kQH4Z3+1t2YtoNOXXwFuApXX3o4BTr/Jn120TdSpqLHy1PZcrurXF38vd6HCclreHG/+e2p0nr0zm+335THp9ncxGJBqluMLMn/63lekfpBEa4MXCu4dw76Xt8ZBas92M696WqFY+zFyTYXQohjrvO04ptUIptfMMt4mn7PMYYAE+utAAlFLTlVKpSqnUgoKCC/11p/XNzjzKqi3SvN0ElFJMGxzPR7f152SFmUmvrWPlHrkuLS7cd3uPcdlLq1m47Sh/vLQ9C+8eTHJky5wb30gebiZuGxrPT1knSDtUZHQ4hjlvgtZaj9Jadz3DbSGAUmoaMB64Xv+y6nYOcOqk0tF12850/Fla6xStdUpoaMsZBzwvNZvYYF/6xwcbHYrLGJDQhkX3DiEuxI9b30/l1ZUHWvw1LNEwxZVm/jxvG7fMSaW1rycL7x7Mny7r0GKWfXVEv+sbQytfjxa9FGVje3GPBR4CJmitK055ahFwrVLKSykVD7QHNjWmLFdypKiCDZnHmdInWibRb2JRrXyYd8dAJveK4sXl+7njwzTKqi3n/0XRYq3eX8DYl9fw+eZs7h6ZyKJ7B9M1KsjosFo8X093bhwYx/Ldx1rscMrGfj18DQgAliultiql3gLQWu8CPgN2A98Ad2utrY0sy2XMT8tGKbhaxj43C28PN/5zTQ+eGJ/Myr2116Uz5bq0+JXSKjOPfrGdm2Zvws/LnS/vGsyfx3TCy10GnDiKmwa2w8vdxNtrWmYturG9uJO01jFa6551tztOee5prXWi1rqj1nrpuY7TkthstaveDE4MIaqVj9HhuCylFLcMieeDW/tRVF7DxNfX8d1euS4tav1woJCxL6/lfz8dYcbwBL66dwg9YloZHZb4lTb+XlyTEsOXW3I4VtLyFtGQCyx2tjHzODknK6VzmJ0MSgxh0T2DiQ325db3U3ntuwP80lVCtDTl1RYeX7CDG979ES93E/PuGMSjl3fG20NqzY7q50U0Zq87aHQodicJ2s4+Sz1CgLc7Y7pEGB1KixHd2pfP7xzEpJ5R/Pvb/dz54Wa5Lt0Cbcg4zpiX1/DRj4e5bUg8S+4bSp92rY0OS5xHbBtfrujWlo83Hm5x8+9LgrajkiozS3fmMaFHpHxjtzNvDzdevKYHj4/rzPI9x7jq9XVkFZYbHZawg4oaC08u2sXv396Iu0nx2YyBPD4+Wf4HncgdwxMprbbwyY+HjQ7FriRB29FX23KpttiYmhJz/p1Fk1NKcdvQBD64pR+FZdVMeO0Hvt/XsmcqcnWbDhZx+StrmbM+i2mD4lhy31D6xsnQRmfTNSqIIUkhvPvDQaotLae/sSRoO5qXdoT2Yf70iJYhHEYalBTConuGEN3al1vm/MTr36fLdWkXU1lj5R9f7eZ3szZg05pPpw/gyQld8PWUWfuc1YzhCeSXVrNwy1GjQ7EbSdB2kp5fypbDJ5maImOfHUFMcO116Su7R/LCsn3c/fFmyuW6tEtIO3SCca+u5d0fDnJD/3Z8c98wBiS0MTos0UhDkkJIbhvIzDUZLWYCIknQdjIvLRs3k2JSL5mS3FH4eLrxyrU9eeyKznyzM4/Jb6zn0HG5Lu2sqsxW/rVkD1PfWk+1xcZHt/XnH5O64idz3bsEpRQzhieQUVDOihYyla8kaDuwWG18sTmHkR1DCQvwNjoccQqlFLcPS2DuLf05VlrFlf/9gdX7W86c8K5i65GTjHt1LTPXZPK7vrF8c/9QBieFGB2WaGLjurUlurUPM1vIxCWSoO1gzYECCkqrmdJHOoc5qiHtQ1h8zxAiW/lw83ubeHNVhlyXdgLVFivPf7OXyW+so6LGytxb+vGvyd0I8PYwOjTRDNzdTNw+NIG0QydIzXL9RTQkQdvBvNRsgv08uaRTmNGhiHOICfbli7sGcUW3tjz3zV7u+WQLFTVyXdpR7cgu5sr//sAbqzKY0ieaZQ8MY1iHlrPgTks1NSWa1r4evLXa9ZeilATdzIrKa1ix5xhX9YqSlXGcgK+nO//9fS8evbwTS3fkMvG1dezJLTE6LHGKKrOVF5btZdIb6yiuNPPetL48P6UHgVJrbhF8Pd25aVAcK/bkc+BYqdHhNCvJGM1swZYczFYtU3s6kdrOKInMvaU/JyvNTHxtHe/+cLDF9Bx1ZD9lFXHFq2t5/fsMJvWM4tv7hzNSWqZanBsHxuHtYWKWi1+LlgTdzOalZdMtKohOEbLou7MZ0j6EZffXNpv+46vd3PTeJvJb4IT9jqC0ysxfF+xk6lsbqLHYmHtLP/5zTQ+CfKXW3BIF+3nyu5QYFmzNIbe40uhwmo0k6Ga0M6eYPbklUnt2YsF+nrx9Yx+evqorP2UVMerF1Xy48ZDUpu3ou73HGP3SGj788RC3DI6v/9IkWrbbhiZg0/DeuiyjQ2k2kqCb0fy0bDzdTEzoEWl0KKIRlFJc378dS/44lK5RQTy+YCdXvbmeTQddvxepkY4UVXD3x5u5ZU4qAd7ufH7nIJ64MlnGNQugtlPnuG5t+fjHwxRXuuYiGpKgm0m1xcqCrTlc1iWcVr6eRocjmkBCqD8f3dafl3/Xk7ziSq6ZuYGb39vE7qPSiawpFVeYefrr3Vz6n9Ws3HOMB0Z14Kt7h9I7VlaeEqebPiyBsmoLH/14yOhQmoV8FW0mK/fkc7LCzNQ+0rztSpSqnQ1ubNcI3l+fxRurMrji1bVc2SOS24fG0z26ldEhOq2KGgsfbjzE699nUFJV+7/zp8s6EhEkk/uIM+saFcTQ9iG8ty6LWwbHu9wKZZKgm8m81CNEBHoztL1cK3NF3h5uzBieyLX9Ypm1JoM567JYvO0ovWJbMW1QHJd3bSvD6hqorNrC3A1ZvLP2IEXlNQzvEMojl3eic1vpWCnO747hiVz/zo8s2JLDtf1ijQ6nSSlHmi0pJSVFp6amGh1Go+UVVzHo2ZXcOSKRP4/pZHQ4wg5Kq8x8npbN3A2HyCwsJ8Tfi+v6x3Jt3xgiW/kYHZ7DMVttpGadYPH2oyzeepTSagsjOoZy7yXt6dNOmrJFw2mtufK1H6iotrLiT8MxmZxvMSKlVJrWOuXX26UG3Qy+2JKNTSNTe7YgAd4eTBscz40D41ibXsj767P473cHeHXlAXrHtuLyrm0Z2SmUxFD/Frua2cmKGlbtK2Dl3nxW78unpMqCt4eJK7q15aaBcfSIaWV0iMIJKaWYMSyRez/Zwre7jzG2a4TRITUZqUE3Ma01l/5nNW38PZl3xyCjwxEGOny8gsXbj/L19lx2181GFhlUe9mjT7vWdI8Jon1YAG5O+I2/IQpKq0nNKmJTVhE/ZRWx+2gJNg0h/p6M7BjGpZ3DGdo+RHpli0azWG1c8p/VBPt58uVdg5zuS7DUoO1k8+ETZBaWc8fwRKNDEQaLbePL3SOTuHtkEtknKlizv5A1+wtYujOX/6UeAcDHw40ukYHEh/jRro0vsW38iG7tQ1QrH0L8vRw+eVttmuNl1WQWlnMgv4z0Y6XsP1bGgfwyCsuqAfD2MNEzphX3XNKekR1D6RHdyimbIYXjql1EI56/LtzFT1kn6BcfbHRITUISdBObl5qNj4cbV3Rva3QowoFEt/bluv6xXNc/FptNk3W8nO3ZxWw9cpLdR0tYvb+A/NLq037H3aSICPImslVtwm57yv2wQC9a+3rSytcDHw+3Jq0xaK2pqLFSUmWmtMrC8bIajpVUkVdSRV5x3a2kimMlVeSXVmM9ZdIWfy93ksL8GdkxlA7hAfRu15puUUHSYU40uyl9YnhpxQHeWp0hCVr8VkWNha+253JFt7b4S7OdOAuTSZEQ6k9CqD+TekXVb6+osXC4qIKjJyvJOVlF7slKjp6s5OjJKjYdLCKvpOq0ZPgzT3cTrXw8Thticmq+/vmuTYNNa2w2jVXr2sd19602jda1NeIaq+2M5QAEeLkTHuRNRKA3SUkhRAR6Ex7kTbtgX9qH+xMR6O10zYvCNfh4ujFtUBwvLt/PvrxSOkYEGB1So0kWaULf7MyjrNoiU3uKi+Lr6U6niMCzzttutWnyS6s4erKK/JIqTlaaOVlh5mRlDSfLzdRYbQCnrWP98z2twaRqvxyYlMJNqbr74PbztrrHnu4mAr09CPTxINDbg9a+HvVJWa4XC0f2hwHteHNVBrPWZPKfa3oYHU6jyX9bE5qXmk1ssC/9XaR5RTgWN5OibZAPbYNk2JYQZ9Laz5Pf9Y3hw42H+L/RHZx+iKNcGGoiR4oq2JB5nKl9oqWJTwghDHLb0Hg0MPuHg0aH0miSoJvIvLRslIKrZWpPIYQwTHRrX67s3pZPNh2muMK5F9GQBN0EbDbN52nZDEkKcfomFSGEcHbThyVSXmPlQydfRKNJErRS6v+UUlopFVL3WCmlXlVKpSultiulejdFOY5qQ+Zxck5WMkVqz0IIYbjkyECGdwjlvXUHqTJbjQ7nojU6QSulYoDRwOFTNl8OtK+7TQfebGw5jmxe6hECvN0Z08V1ppgTQghnNmN4AoVlNXyxOcfoUC5aU9SgXwIe4pcRHQATgbm61kaglVLKJWfuKKkys3RnHhN6RLrcUmdCCOGsBia0oXt0ELPWZJx1XL+ja1SCVkpNBHK01tt+9VQUcOSUx9l121zOV9tyqbbYmJoiC2MIIYSj+HkRjazjFXy7K8/ocC7KecdBK6VWAGdqu30M+Au1zdsXTSk1ndpmcGJjnW8tz3lpR2gf5k+P6CCjQxFCCHGKsV0jaNfGl7dWZzC2a4TTDYE9bw1aaz1Ka9311zcgE4gHtimlsoBoYLNSKgLIAU6tUkbXbTvT8WdprVO01imhoaGNfT12lZ5fypbDJ7kmJcbp/vBCCOHq3EyK24cmsC27mB8PFhkdzgW76CZurfUOrXWY1jpOax1HbTN2b611HrAIuLGuN/cAoFhrnds0ITuOeWnZuJnUafMpCyGEcBxT+kQT4u/JW6szjA7lgjXXOOgl1Naw04G3gbuaqRzDWKw2vticw8iOYYQGeBkdjhBCiDPw9qhdRGPVvgL21K3L7iyaLEHX1aQL6+5rrfXdWutErXU3rXVqU5XjKFbvL6CgtFoWxhBCCAd3w4B2+Hq68faaTKNDuSAyk9hFmpeaTRs/Ty7pFGZ0KEIIIc6hla8n1/aNZdG2o+ScrDQ6nAaTBH0RisprWLn3GJN6ReHhJqdQCCEc3a1D4wF4d63zLKIh2eUiLNiSg9mqpXlbCCGcRFQrHyb0iOTTnw5zsqLG6HAaRBL0RZiXlk23qCA6RQQaHYoQQogGmj48gYoaKx9scI5FNCRBX6CdOcXsyS2R2rMQQjiZThGBjOwYypz1WU6xiIYk6As0Py0bT3cTE3pEGh2KEEKICzRjeCLHy2uYn5ZtdCjnJQn6AlRbrCzYmsPo5HBa+XoaHY4QQogL1D8+mB4xrXh7babDL6IhCfoCrNidz8kKsyyMIYQQTkopxR3DEjh0vIJvdjr2IhqSoC/AvLQjtA3yZkhSiNGhCCGEuEiju0QQH+LHW6sz0Npxa9GSoBsor7iKNfsLmNw7CjeTLIwhhBDO6udFNHbkFLMh47jR4ZyVJOgG+mJLNjYNU/pI87YQQji7yb2jCPH34i0Hnv5TEnQDaK2Zn5pN37jWxIf4GR2OEEKIRvL2cOPmwXGs2V/A7qOOuYiGJOgG2Hz4BJmF5UyV2rMQQriMG/q3w8/TjZlrHHMpSknQDTAvNRtfTzeu6N7W6FCEEEI0kSBfD37fL5avtudypKjC6HB+QxL0eVTUWPhqey5XdGuLv5e70eEIIYRoQrcOjUcB7/7geItoSII+j6U78iirtjC1j0ztKYQQrqZtkA8Te0bxv5+OcKLcsRbRkAR9HvPSjtCujS/94oONDkUIIUQzmDE8gUqzlbkOtoiGtNmew+HjFWzMLOL/LuuAUjL2WfyipKSE/Px8zGaz0aGIM/Dw8CAsLIzAQFlxTpxfh/AALu0Uxvsbspg+LAEfTzejQwIkQZ/T/M3ZKAVXS/O2OEVJSQnHjh0jKioKHx8f+fLmYLTWVFZWkpOTAyBJWjTIjOGJXDNzA/PTjvCHgXFGhwNIE/dZ2Wyaz9OyGZIUQmQrH6PDEQ4kPz+fqKgofH19JTk7IKUUvr6+REVFkZ+fb3Q4wkn0jWtNr9hWzFqbicVqMzocQBL0WW3IPE7OyUqmSO1Z/IrZbMbHR760OTofHx+5BCEaTCnFjGGJHCmqZKmDLKIhCfos5qUeIdDbnTFdIowORTggqTk7PvkbiQs1OjmchBA/Zq5xjEU0JEGfQUmVmaU785jQMxJvD8foLCCEEKJ5mUyK6cMS2JlTwrp04xfRkAR9Bou3HaXaYpOpPYVLGTFiBPfcc4/RYQBwxx13cP/99xsdhhC/MalXFKEBXg4x/ack6DOYl5pNh3B/ukcHGR2KEHbX3NdttdYsWrSISZMmNWs5QlwMbw83bhkcz9oDhezMKTY0FknQv5KeX8rWIyeZ2idGrmEJlzFt2jRWr17N66+/jlIKpRRZWVmsWrUKpRRLliyhX79+eHp6smzZMrTWPP/88yQmJuLj40O3bt348MMPz1vG+PHj+ec//0l4eDj+/v7cfPPNVFZWnrbfTz/9RHV1NUOGDAEgPT2dESNG4O3tTceOHfnqq6/w9/dnzpw5zXU6hDin6/rH4u/lzkyDl6KUcdC/Mi81GzeTYlKvKKNDEaLJvPLKK+zfv59OnTrxzDPPABAaGkpWVhYADz/8MP/5z39ISkoiICCAxx9/nPnz5/P666/TsWNHNmzYwO23307r1q0ZN27cWctZvXo1Pj4+rFy5kpycHG655RYefvhhXn311fp9FixYwLhx43B3d8dms3HVVVfRunVrNmzYQEVFBffddx/V1dXNej6EOJcgHw+u6x/LO2szeWhMR2KCfQ2JQ2rQp7BYbXyxJYeRHcMIDfAyOhwhmkxQUBCenp74+voSERFBREQEbm6/dIB88sknGT16NAkJCfj6+vLiiy/yzjvvMHbsWOLj47nuuuu4/fbbef31189ZjpubG++99x5du3ZlzJgxPPfcc8ycOZPy8vL6fRYuXFjfvL1ixQp2797Nhx9+SK9evRg8eDAvv/wyFoulWc6DEA11y+B43EyKd9YaV4uWGvQpVu8voKC0mqkpMvZZXJinFu+y+6LvyZGB/O3KLk1yrJSUlPr7u3fvpqqqirFjx552mcdsNhMXF3fO43Tv3h1/f//6xwMHDqSmpoaMjAy6d+9Oeno6mZmZjBkzBoA9e/YQFRVFbGxs/e/0798fk0nqDsJYEUHeTOoZxf9Sj/DHS9vTxt/+lTZJ0KeYl5pNiL8nl3QKMzoUIezKz8+v/r7NVjuL0uLFi09LnFA7x3VjLFiwgEsvvfS08oRwVDOGJzAvLZu5Gw7xwGUd7F5+oxO0Uupe4G7ACnyttX6obvujwK112/+otV7W2LKaU1F5DSv3HuOmgXF4uMm3d3Fhmqom25w8PT2xWq3n3S85ORkvLy8OHTrEJZdcckFl7Nixg/Ly8voEvHHjRjw9PUlMTARqm7dvuumm+v07d+5MTk4OR44cISamdljjpk2b6r8kCGGkpLAARnUOZ+6GLGYMT8DX07512kaVppQaCUwEemitq5VSYXXbk4FrgS5AJLBCKdVBa33+TweDLNiSg9mqmZoiY5+Fa4qLi2PTpk1kZWXh7+9PcPCZl1ANCAjgwQcf5MEHH0RrzbBhwygrK2Pjxo2YTCamT59+1jIsFgu33HILTzzxBEePHuWRRx7h9ttvx8/Pj4KCAjZu3Mj8+fPr9x81ahSdOnXixhtv5KWXXqKyspIHHngAd3dp3BOO4Y7hCUx56xif/XSEaYPj7Vp2Y6uKdwLPaq2rAbTWP89MPxH4VGtdrbU+CKQD/RpZVrPRWvNZ6hG6RwfRMSLA6HCEaBYPPvggnp6eJCcnExoayuHDh8+67z/+8Q+efPJJ/v3vf9OlSxcuu+wyPv/8c+Ljz/0BNXz4cLp06cLIkSO56qqruOSSS3j++eeB2ibzvn37Eh4eXr+/yWTiyy+/xGaz0b9/f2688UYef/xxvLykk6ZwDClxwfRp15q31x60+yIajf2a2gEYqpR6GqgCHtRa/wREARtP2S+7bptD2nW0hL15pfxjouM3UwpxsTp06MCGDRtO2xYXF3fGOYeVUtx7773ce++9F1zOE088wRNPPPGb7af23v51XKtXr77gcoSwlxnDEpj+QRpf78hlYk/7pbLzJmil1ArgTCtGPFb3+8HAAKAv8JlSKuFCAlBKTQemA7/pkGIv81KP4OluYkIPh/0OIYTTGzx4ML///e+NDkOICzaqcziJoX7MXJ3JhB6RdpvE6rxN3FrrUVrrrme4LaS2ZvyFrrUJsAEhQA5w6sXc6LptZzr+LK11itY6JTQ0tPGv6AJVW6ws3HaU0cnhBPk2roeqEOLsHnroofqOYEI4E5OpdinK3bklrD1QaLdyG9vEvQAYCXyvlOoAeAKFwCLgY6XUi9R2EmsPbGpkWc1ixe58TlaYpXOYEI3UlFNzlpWVNdmxhGgKE3tF8p/l+5i5JoNhHexTmWxsJ7HZQIJSaifwKXBTXW16F/AZsBv4BrjbUXtwz0s7Qtsgb4YkhRgdihBCCAfl5V67iMaPmUVkn6iwS5mNqkFrrWuAG87y3NPA0405fnPLK65izf4C7hqRhJtJFsYQQghxdtcPaMe47m2Jbm2fublb9GDDL7ZkY9MwpY9M7SmEEOLc/L3c8feyX9pssVNmaa2Zl5pNv7hg4kJk2kEhhBCOpcUm6LRDJzhYWM4UWRhDCCGEA2qxCXpeaja+nm6M69bW6FCEEEKI32iRCbqixsJX249yRbe2+NnxeoIQQgjRUC0yQS/dkUd5jZWp0jlMtCAjRozgnnvuMTqMBvv000/p2bOn0WEIYZgWmaDnpR0hro0v/eLPvJqPEC2Z2Ww2OgTg7HN3C9FStLgEffh4BRszi5jSJ9pu86kKYbRp06axevVqXn/9dZRSKKXIyspi1apVKKVYsmQJ/fr1w9PTk2XLlvHkk0/StWvX044xZ84c/P39T9u2ePFi+vTpg7e3N/Hx8Tz22GPU1NScNY6fj7F48WI6dOiAt7c3I0eOJDMz87T9zGYzS5cuZeLEiQBUVFQwbdo0/P39CQ8P55lnnmH8+PFMmzataU6QEA6oxSXo+ZuzUQom95bmbdFyvPLKKwwcOJCbb76Z3NxccnNzT5sX++GHH+af//wne/fupX///g065rJly7j++uu555572LVrF7Nnz2b+/Pn85S9/OefvVVdX89RTT/Hee++xYcMGrFYrkydPPm1Vre+//56goCB69eoF1C6VuXz5cj7//HNWrlzJli1bWLNmzUWcCSGcR4vqIWWzaT5Py2ZIUgiRrXyMDkcIuwkKCsLT0xNfX18iIn67ON2TTz7J6NGjL+iYTz/9NH/+85+5+eabAUhMTOS5557jhhtu4IUXXjhrC5XFYuGVV15h8ODBAHzwwQckJCSwcuVKRo0aBdQ2b/9cey4rK+Pdd99l9uzZjBkzBoD33nuP6Gj5ki1cW4tK0OszjpNzspKHL+9kdCjC1Sx9BPJ22LfMiG5w+bNNcqiUlJQL/p20tDQ2bdrEc889V7/NZrNRWVlJXl4ebdueeQijyWSiX79+9Y/btWtHZGQku3fvrk/QixYtql98IyMjg5qaGgYOHFj/O/7+/nTr1u2CYxbCmbSoBD0v7QiB3u6MTg43OhQhHIqf3+mz6ZlMptOanOG3ncdsNht/+9vfmDp16m+Od76lY8/V/yM1NZWysjKGDx9+vrCFcGktJkEXV5r5ZmceU1Oi8fZwMzoc4WqaqCbbnDw9PbFaG7aoXGhoKMeOHUNrXZ9Mt27deto+vXv3Zu/evSQlJV1QHDabjU2bNjFo0CAADh8+zNGjR+ncuTNQ27w9btw43N1rP54SExPx8PBg48aNJCQkAFBeXs7OnTtJTEy8oLKFcCYtJkF/tf0o1RYbU/vIus+iZYqLi2PTpk1kZWXh7+9PcPDZhxmOGDGCoqIinnnmGa699lpWrVrF/PnzT9vniSeeYPz48bRr145rrrkGd3d3du7cyaZNm3j++efPemx3d3fuv/9+XnnlFXx8fHjggQfo0qXLadef//rXv9bv7+/vz6233srDDz9MaGgokZGR/P3vf2/wlw0hnFWL6cU9LzWbjuEBdI8OMjoUIQzx4IMP4unpSXJyMqGhoRw+fPis+3bu3Jk333yTWbNm0b17d5YvX/6b3tljxozh66+/5vvvv6dfv37069ePZ599ltjY2HPG4eXlxWOPPcaNN95I//79sdlsfPHFFyilyMzMZP/+/YwdO/a03/n3v//NyJEjueqqqxg5ciRdu3Zl2LBhF38yhHAC6tfXmYyUkpKiU1NTm/y46fmljHpxDY+P68xtQxOa/PiiZdmzZ099c6y4MHPmzOGee+6hrKzsjM+/9NJLLF++nCVLlpz3WOPHjyckJKS+M9mZyN9KOAOlVJrW+jc9NVtEDXpeajbuJsWkXlFGhyKEOIeoqCgeffRRo8MQwiG4/DVos9XG55tzGNkpjBB/L6PDEUKcwzXXXGN0CEI4DJdP0Kv3FVBYVi0LYwjhAKZNm9Zk03N+9dVXTXIcIRyVyzdxz0s7Qoi/JyM7hRkdihBCCNFgLp2gj5dVs3JPPpN6RuHh5tIvVQghhItx6ay1YOtRLDbN1BQZ+yyaliONfhBnJn8j4excNkFrrZmXeoTu0UF0jAgwOhzhQjw8PKisrDQ6DHEelZWVeHh4GB2GEBfNZRP0rqMl7M0rldqzaHJhYWHk5ORQUVEhtTQHpLWmoqKCnJwcwsKk74lwXi7bi9vH041rUqKZ0D3S6FCEiwkMDATg6NGjv1lAQjgGDw8PwsPD6/9WQjgjl03QiaH+PD+lh9FhCBcVGBgoH/5CiGblsk3cQgghhDOTBC2EEEI4IEnQQgghhAOSBC2EEEI4IEnQQgghhANyqPWglVIFwKEmPGQIUNiEx3N2cj5OJ+fjF3IuTifn43RyPn7RHOeindY69NcbHSpBNzWlVOqZFsFuqeR8nE7Oxy/kXJxOzsfp5Hz8wp7nQpq4hRBCCAckCVoIIYRwQK6eoGcZHYCDkfNxOjkfv5BzcTo5H6eT8/ELu50Ll74GLYQQQjgrV69BCyGEEE7JpRK0UmqqUmqXUsqmlDprLzulVJZSaodSaqtSKtWeMdrTBZyPsUqpfUqpdKXUI/aM0Z6UUsFKqeVKqQN1P1ufZT9r3Xtjq1Jqkb3jbE7n+1srpbyUUv+re/5HpVScAWHaTQPOxzSlVMEp74fbjIjTHpRSs5VS+UqpnWd5XimlXq07V9uVUr3tHaO9NOBcjFBKFZ/yvniiOeJwqQQN7AQmA2sasO9IrXVPFx86cN7zoZRyA14HLgeSgd8rpZLtE57dPQKs1Fq3B1bWPT6Tyrr3Rk+t9QT7hde8Gvi3vhU4obVOAl4CnrNvlPZzAe/9/53yfnjHrkHa1xxg7DmevxxoX3ebDrxph5iMModznwuAtae8L/7eHEG4VILWWu/RWu8zOg5H0cDz0Q9I11pnaq1rgE+Bic0fnSEmAu/X3X8fmGRcKIZoyN/61HM0H7hUKaXsGKM9taT3/nlprdcARefYZSIwV9faCLRSSrW1T3T21YBzYRculaAvgAa+VUqlKaWmGx2MwaKAI6c8zq7b5orCtda5dffzgPCz7OetlEpVSm1USk2yT2h20ZC/df0+WmsLUAy0sUt09tfQ9/7VdU2685VSMfYJzSG1pM+KhhiolNqmlFqqlOrSHAW4N8dBm5NSagUQcYanHtNaL2zgYYZorXOUUmHAcqXU3rpvTE6nic6HyzjX+Tj1gdZaK6XONoShXd37IwH4Tim1Q2ud0dSxCqewGPhEa12tlJpBbevCJQbHJIy3mdrPiTKl1BXAAmqb/puU0yVorfWoJjhGTt3PfKXUl9Q2dTllgm6C85EDnForiK7b5pTOdT6UUseUUm211rl1TXP5ZznGz++PTKXUKqAX4AoJuiF/65/3yVZKuQNBwHH7hGd35z0fWutTX/s7wPN2iMtRudRnRWNorUtOub9EKfWGUipEa92kc3S3uCZupZSfUirg5/vAaGo7U7VUPwHtlVLxSilP4FrApXoun2IRcFPd/ZuA37QwKKVaK6W86u6HAIOB3XaLsHk15G996jmaAnynXXeyhPOej19dY50A7LFjfI5mEXBjXW/uAUDxKZeMWhSlVMTPfTOUUv2ozaVN/0VWa+0yN+Aqaq+LVAPHgGV12yOBJXX3E4Btdbdd1DYFGx67Ueej7vEVwH5qa4mufD7aUNt7+wCwAgiu254CvFN3fxCwo+79sQO41ei4m/gc/OZvDfwdmFB33xuYB6QDm4AEo2M2+Hz8q+5zYhvwPdDJ6Jib8Vx8AuQC5rrPjVuBO4A76p5X1PZ6z6j730gxOmYDz8U9p7wvNgKDmiMOmUlMCCGEcEAtrolbCCGEcAaSoIUQQggHJAlaCCGEcECSoIUQQggHJAlaCCGEcECSoIUQQggHJAlaCCGEcECSoIUQQggHJAlaCCGEcECSoIUQQggHJAlaCCGEcECSoIUQQggHJAlaCCGEcECSoIUQQggHJAlaCCGEcECSoIUQQggH5G50AKcKCQnRcXFxRochhBBC2E1aWlqh1jr019sdKkHHxcWRmppqdBhCCCGE3SilDp1puzRxCyGEEA5IErQQQgjhgCRBCyGEEA6o2RO0UmqsUmqfUipdKfVIc5cnhBBCuIJm7SSmlHIDXgcuA7KBn5RSi7TWu5uzXHF+1RYrbkrh7iaNKEK0VDabJvtEJSVVZqw2jcVmw2LVdfd/+Wmx2k57bLNpfDzd8Pdyx9/bHT9PdwK83fHzcsffyx1Pd/lcaQrN3Yu7H5Cutc4EUEp9CkwEJEE3RnYqZK4CcwWYK6GmvPZ+TQWYy8FcBdoKNitWm5XqGjM1ZjNmixWzxUKF1USFxYRVuePt7YOfrw9B/n4E+Pli8vQDTz/w8gfPgNqfnSdAUJTRr1oI0RhHt8KhdVBTTtHJExw8mk9uwXGUuRxfqvFRNbhhxRMbJmy41d1+vm/BDQtumHHHjBsWXfu4GHeO4UWZ9qYCb8rxJiEynMuuuQvPkDijX7VTa+4EHQUcOeVxNtC/mct0fYfWwXf/AGUCDz+0py82dx+qlDcV2otSqztlNVBeo6mwmLDhjQ1ftDLh6+lOoI8i0AO0tZryikoKjxdy8vgxvJWF1h5m/FU1XtZyTLaa2vLa9pAELYSzO7QOlv0FAD/tTgLexLn74h7oD15+2NwCwOQOJjeUyQQmNzB5oEwmlHLDpC0obcFks6BsZrTFjM1ag7ZUg+UkbjXluFkrcLdUYMq38V1qfy4ZG2fsa3Zyho+DVkpNB6YDxMbGGhyNY7PZNEeLK8loPZnMEaPZf7yGA/nlpBeUcbLCXL+fr6cbSWH+JIX6kxjmX3s/zJ92wb5nbNIuKK3mx4PHWZ9xnPXphWQdrwAg3FcxIt6H3rnhDPAvJzbYF6WU3V6vEKIJ9ZnGQjWS/1twgOGdInl+Snfa+Hs1eTHaZuOKF5fjn+nLJU1+9JaluRN0DhBzyuPoum31tNazgFkAKSkpupnjcQoWq41DRRWk55eddssoKKOixlq/X2tfD5LC/Lm8a9v6JJwU5k/bQG9MpoYn0tAAL8Z3j2R890gAck5Wsj69kPUZx/k+vZD/7doL7CWqlQ+DEtswOCmEQYltCAv0buqXLoRoJt9llvGnRVkMSArnjRt64+Xu1izlKJOJCX2TeHbpXjIKykgM9W+WcloCpXXz5USllDuwH7iU2sT8E3Cd1nrXmfZPSUnRLWkmsSqzlYyCuuSbX8aBukScdbwcs/WXv0vbIG+SwvxJDK1NwO3rEnFzfPv9Na01GQXlrM8oZH36cTZkHqe4sra2nhTmz+DENgxKCmFAfBuCfD2aPR4hxIXbfPgE1729kaQwfz65fQAB3s37v5pfUsXAZ79j+rAEHh7bqVnLcgVKqTStdcpvtjdngq4r+ArgZcANmK21fvps+7pqgi6uNNcn4fSCX2rER05U8PPpNymIDfYlKSzgtNpwYqhfs/8zXQirTbP7aAnrMwpZl3Gcnw4WUWm2YlLQNSqIQYm1teu+ccH4eDbPN3QhRMOl55cy5a0NBPl4MP+OQYQGNP8Xe4Bb5vzErqPFrH/kUtwuoEWvJTIsQV8IZ07QWmsKyqp/UxtOzy8jv7S6fj9PdxMJIX4knlITTgrzJ66NH94ezpfQaiw2th45ybr0QjZkHGfLkROYrRpPNxO9YlsxKDGEwUlt6BHTCg8Z0iWEXeUVVzH5jXXUWG18fucg2rXxs1vZS3fkcudHm3nv5r6M7Bhmt3KdkSToJmKzaXJOVp5+fbigjAPHSimpstTv5+/lXttB61fN0jHBvi79bbKixsKmg0VsyDjOuoxCdh0tQevajmv94oMZnBjCoKQ2dI4IvKDr5EKIC1NcYeaamRvIOVnJp9MH0DUqyK7l11hs9H9mBYMSQ3j9+t52LdvZnC1BG96L21GZrTYOHS/nwLFfkvDPHbWqzLb6/dr4eZIU5s+VPSLra8PtwwIID/RqkT2efT3dGdExjBF135hPlNfw48HjrEuvTdir9u0Baju4DUxsU98kHh/i1yLPlxDNocps5fa5qWQWlvH+zf3snpyhtrVwUq8oPtp4mBPlNbT287R7DM6uxSfoihoLmQXl9bXhA/mlpOeXceh4BRbbL60LUa18SAzzZ0BCm1+uEYf6y5vuPFr7eTK2a1vGdm0L1Da5rc8oZF36cdZnFLJkRx5Q2xHu52Q9OCmEiCDpIS7ExbBYbfzxky38dKiI//6+F4OSQgyLZWqfGN5bl8XCrTlMGxxvWBzOqsU0cZ+sqDlDs3QZOScr6/dxMynatfH9pVk63J+k0AASQv3w82rx32WanNaarOMV9dev12cUcqJuPHdCqF9tsk4MYWBiG1r5yhchIc5Ha81fvtzJJ5sO8+SVyQ6RFMe9uhaAr/841OBIHFeLa+JOzy9lzvqsuoRcTmHZLx21vNxNJIb606dda37XN+a0jloyh6z9KKWID/EjPsSPGwa0w2bT7Mkrqb1+nV7Il5tz+HDjYZSC5LaB9eOv+8YFyxcmIc7gpRUH+GTTYe4emegQyRlgap9only8m91HS0iODDQ6HKfisjXozYdPcNPsTfVN0e3Df26WDiCqtY9Ld9RyFWarje3ZJ2uvX6cXsuXwSWqsNtxNqr6H+KDENvSKbS1frESL98HGQ/x1wU6uSYnmuau7O0yfjhPlNfR/ZiXXD4jlb1d2MToch9TienH//Loc5U0qGq+yxkrqoSLWpR9nQ0YhO3KKsWnw8XCjb3xwfZN4cmSgfAETLcrSHbnc9fFmLu0Uxls39HG4Veru+iiNDRnH+fEvo+TL9Bm0uCZuScyux8fTjaHtQxnaPhSonQBmY+bx+ibxZ5fuBSDIx4MBCcF1TeIhJIZKD3HhujZkHOe+T7fSO7Y1//19b4dLzgBTU2JYsiOPlXuOcXm3tkaH4zRcNkEL1xfk48GYLhGM6RIB1E4vuCGzNlmvSz/Osl3HAAgP9KpvDh+UFEJUKx8jwxaiSWitWbTtKI98voN2bXx596YUh529b1j7UMIDvZiXli0J+gJIghYuIyzQm4k9o5jYMwqtNUeKKlmXUci69ELW7C/gyy2167QkhfkzOjmcy5LD6RHdSiZMEU5n/7FS/rZwFxsyj9MjphXv3Jji0CMd3EyKyb2jmbk6g/ySKllop4Fc9hq0EKfSWrPvWCnr0o/z3d5jbMwswmrThAV4MSo5nNHJ4QxMbNNsK/wI0RRKqsy8vPwA72/Iwt/LnQfHdOS6frFO0ecis6CMS/6zmkcu78QdwxONDsehtLhOYkKcS3GFme/35fPt7jxW7SugosaKv5c7wzuGMjo5nBEdwwjycZxFSkTLZrNpPt+czXPf7ON4eTXX9o3lz2M6EuxkEyVd/eZ6TlbUsOJPw6VfyClaXCcxIc4lyNeDSb2imNQriiqzlQ0Zx/l2dx7Ld+fz9fZc3E2KgYltGJ0czqjkcNoGyXVrYYwth0/w5OLdbDtykl6xrXhvWl+6Rdt/6s6mMLVPNI98sYMtR07SO7a10eE4PKlBC3EKm02z5cjJ2mS96xiZheUAdI8O4rLO4YzuEkGHcH/59i+aXX5pFc9/s4/5admEBnjx6OWdmNQzyqn7TJRWmen39Eom9YrkX5O7Gx2Ow5AmbiEuQnp+WV3N+hhbDp8EoF0b3/pk3adda6e4/iecR43Fxpz1B3l1ZTrVFiu3DkngnkuS8HeR2fP+9NlWvt11jJ8eG+Wwvc7tTRK0EI2UX1LFij21163Xpx+nxmoj2M+TSzuFMbpLBEOSQuQDRzTK9/vy+cfi3WQWlnNJpzD+Oj6Z+BD7reFsDxsyjvP7tzfy0u96cFWvaKPDcQiSoIVoQmXVFlbvK+Db3Xl8tzef0ioL3h4mhrUP5bLkcC7tHO50HXiEcbIKy/nHV7tZuTef+BA/nhifzMhOYUaH1SxsNs3wf39PTGtfPr59gNHhOATpJCZEE/L3cmdc97aM696WGouNTQeL6pvCv919DJOCvnHBXJYczujkCGLb+BodsnBAZdUWXvsundk/HMTDTfHo5Z24eXC8S0+HaTIppvSO4aUV+zlSVEFMsPxvnI3UoIVoQlprduaU1CfrvXmlAHSKCGB0cu116y6RgdLJrIXTWrNgaw7/WrKX/NJqru4dzcNjO7aYCTyyT1Qw9Pnvue/S9tw/qoPR4RhOmriFMMDh4xV8uzuPb3cfIzWrCJuGyCDv2pp1lwj6xQfj4YBzJ4vmsz37JE8u2sXmwyfpER3EkxO60KsFDjm64Z0fOVhYztqHRjp1z/SmIAlaCIMVldewck9tE/jaAwVUmW0EertzSacwLkuOYHjHUJfpqSt+q7Csmhe+2cdnaUdo4+fJQ2M7MaV3dItNTgu35nDfp1v5+Lb+DEoKMTocQ8k1aCEMFuznydSUGKamxFBZY2XtgQK+3X2MlXuOsWDrUTzdTAxOasNlyRGMSg4jLKBlNHe6OrPVxtwNh3h5xX4qa6zcNiSeey9tT6B3y56pbkyXCAK83ZmXlt3iE/TZSA1aCINZrDbSDp3g293HWL77GIeLKlAKesa04oqutR3RImUFLqe09kABTy3eTXp+GcM6hPLE+GSSwvyNDsth/OXLHXyxOZtNj41q0V9YpIlbCCfw86Iey3cdY9nuPHbmlACQ0q4147u35YrubaVm7QQOH6/gn1/v5tvdx4gN9uWJ8clc2jlMOgf+ypbDJ7jqjfX8a3I3ft8v1uhwDCMJWggnlFVYztc7clm87Sh780oxKegf34Yre0QytmuEjLV2MBU1Ft74PoNZazNxNynuHpnErUPi8faQCWzORGvN6JfWEODtzhd3DTY6HMNIghbCyR04Vsri7bl8tf0omQXluJkUQ5JCGN+9LaO7RMjqWwbSWrN4ey7/WrKH3OIqJvWM5JHLOxMRJK0d5zNrTQbPLNnLij8NIykswOhwDCEJWggXobVmd24JX22vrVlnn6jE083EsA6hXNmjLaM6h+MnvcHtZvfREp5ctItNWUV0jQrkySu7kBIXbHRYTiO/tIqB//qO24bG8+jlnY0OxxCSoIVwQVprtmUXs3jbUb7enkteSRXeHiYu7RTOpF5RDO8Q6tKzUhmpuMLMf5bv48ONh2jl68mfx3TkmpQYWTzlItz2/k9syy5mwyOX4N4C5wWQYVZCuCClFD1jWtEzphWPXdGZ1EMn+Gp7bbL+ekcurX09GN89kkm9ougd20o6KTUBm03zWeoRnl+2j5MVNfxhQDv+dFlHgnzlEsPFmtInhhV78llzoIBLOoUbHY7DkBq0EC7IbLXxw4FCvtySw7e786gy24gN9mVSrygm9YwkIVSG+lyMbUdO8sTCnWzLLqZvXGuemtCV5MhAo8NyejUWGwP+tZL+8cG8eUMfo8OxO6lBC9GCeLiZGNkpjJGdwiirtvDNzjwWbMnhv98d4NWVB+gR04qrekZyZY9I2vh7GR2uwysqr+GFZXv59KcjhPh78dLvejCpZ5S0SDQRT3cTV/WKYu6GLIrKa2R0Qh2pQQvRguQVV7F421G+3JLD7twS3EyK4R1CmdQriss6h8t61r9itWk+/vEQ//52P+XVFm4eHMcfL21PQAueVKO57M0rYezLa3lifDK3DIk3Ohy7kk5iQojT7MsrZcHWHBZuyeFocRV+nm6M7dqWq3pFMTCxTYvv7JR2qIi/LtjF7twSBiW24akJXWgf3jKHAdnLlf/9AYtNs/S+oUaHYleSoIUQZ2SzaX48WMSCLTks2ZFLabWFsAAvJvaM5Kpe0XRuG9CimnLzS6t4dulevticQ9sgbx4fl8wV3SJa1DkwytwNWTyxcBdf3TuErlFBRodjN5KghRDnVWW28t3efL7cksOqffmYrZqO4QFM6hXFxJ6RLj0nuNlq4/31Wby84gDVFiu3D03gnkuS8PWUrjr2crKihn5Pr+S6/rE8OaGL0eHYjSRoIcQFOVFew9c7cvlySw5ph06gFPSLC+aKbm0Z0yXCpWbJ2pBxnL8t2sn+Y2UM7xDK365Mlp7uBrn7482sSy/kx79cipd7y+gT0SwJWin1AnAlUANkADdrrU/WPfcocCtgBf6otV52vuNJghbCMR0+XsGCrTks3naUA/llAPSObcXYrhFc3rUtMcG+Bkd4cY4UVfD8sn0s3naU6NY+/O3KLoySRS0MtXp/ATfN3sQb1/fmim5tjQ7HLporQY8GvtNaW5RSzwForR9WSiUDnwD9gEhgBdBBa2091/EkQQvh+NLzy/hmZy7f7Pplta0ukYGM7RLB5d0inGI+5ZyTlby9JpOPfjyESSnuHJHIHcMTZVELB2C1aYY89x0dIwKYc3M/o8Oxi2YZB621/vaUhxuBKXX3JwKfaq2rgYNKqXRqk/WGxpQnhDBeUpg/91zSnnsuac+Rogq+2ZnH0p25/Gf5fv6zfD9JYf6M7RLB2K4RdIkMdJja6KHj5SzdmcfSnXlsO3ISN5PimpRo7ru0g0s11zs7N5Nicu8o3lyVQV5xVYv+2zRl74dbgP/V3Y+iNmH/LLtumxDChcQE+3L7sARuH5bAsZIqlu3KY+mOPN5Ylc5r36cTE+zD2C4RDGkfSveoIFrbcQIKs9XG9uxifjhQyDe78tiTW1vb7xYVxJ/HdOTK7pHEtnHOpnlXN6VPDK9/n8EXW7K5a0SS0eEY5rwJWim1Aog4w1OPaa0X1u3zGGABPrrQAJRS04HpALGxLXfBbiGcXXigNzcOjOPGgXEcL6tmxZ5jLN2Zx5z1Wby99iAAscG+dI8Ookd0K5LC/IkL8SO6tQ8eF7lAgtaa8horJ8prKCqvIbe4iszCMjYdLOKng0WU19ReVevTrjWPj+vMmC4RTnu9vCWJD/Gjb1xr5qdmc+fwRIdphbG38yZorfWocz2vlJoGjAcu1b9c0M4BYk7ZLbpu25mOPwuYBbXXoM8fshDC0bXx9+J3fWP5Xd9YSqvM7MguZlt2MduzT7L50Am+2p5bv6+7SRET7EtssC9hAV6EBHjh4+GGm0nhblKYrTbKa6yUVpk5UWGuT8YnKmo4UW6mxmr7TfmJoX5M7h3NwMQ29I8PlulMndDUPjE89Pl2Nh8+QZ92LXP5zsZ2EhsLvAgM11oXnLK9C/Axv3QSWwm0l05iQgiA42XVHCws52BhOVnHa38eKaqkoLSawrJqLLbTP5c83Uz4ebnR2s+TYF/P03629vWofxwe6E1sG1+CfGQqTmdXVm2h39MrmNAjkmev7m50OM2quRbLeA3wApbXNUFs1FrfobXepZT6DNhNbdP33edLzkKIlqONvxdt/L1IifttzUhrjU2DxWbDYtV4uJlkTesWyN/LnSu6tWXxtqM8cWVyi5wwplHveq11ktY6Rmvds+52xynPPa21TtRad9RaL218qEKIlkAphZtJ4eXuhp+XuyTnFmxqn2jKa6ws3ZFndCiGkHe+EEIIh9QvPph2bXyZl3bE6FAMIQlaCCGEQ1JKMaV3NBszizh8vMLocOxOErQQQgiHdXWfaJSC+ZuzjQ7F7iRBCyGEcFiRrXwYkhTC52nZ2GwtaySuJGghhBAObWpKDDknK1mfcdzoUOxKErQQQgiHNjo5nEBv9xbXWUwStBBCCIfm7eHGhJ6RfLMzj+JKs9Hh2I0kaCGEEA5vap8Yqi02vtp+1OhQ7EYStBBCCIfXPTqIjuEBzEttOb25JUELIYRweEoppqZEs/XISQ4cKzU6HLuQBC2EEMIpTOoVhbtJMS+tZdSiJUELIYRwCiH+XozsFMYXm3Mwn2GZUVcjCVoIIYTTmNonmsKyalbvKzj/zk5OErQQQginMbJTGCH+ni1iTLQkaCGEEE7Dw83EVb2iWLknn+Nl1UaH06wkQQshhHAqU1NisNg0C7a69phoSdBCCCGcSofwAHpEBzEv9Qhau+4CGpKghRBCOJ0pKTHszStlZ06J0aE0G0nQQgghnM6E7pF4uptcurOYJGghhBBOJ8jXgzFdIli49ShVZqvR4TQLSdBCCCGc0jUp0RRXmlmx55jRoTQLSdBCCCGc0qDEECKDvPnMRRfQkAQthBDCKbmZFFf3iWbtgQJyiyuNDqfJSYIWQgjhtKb0iUZr+GJzjtGhNDlJ0EIIIZxWuzZ+9IsPdskx0ZKghRBCOLVrUmLIOl5B6qETRofSpCRBCyGEcGpXdIvAz9ONeamuNSZaErQQQgin5uvpzrjubflqey7l1Rajw2kykqCFEEI4vakpMVTUWFmyI9foUJqMJGghhBBOL6Vda+JD/JiX5jpjoiVBCyGEcHpKKab0iWbTwSIOHS83OpwmIQlaCCGES5jcOwqTgvkuUouWBC2EEMIltA3yYWj7UD5Py8Zqc/4x0ZKghRBCuIypKdEcLa5iXXqh0aE0miRoIYQQLmNU53CCfDxcorOYJGghhBAuw9vDjYk9I1m2K4/iCrPR4TRKkyRopdT/KaW0Uiqk7rFSSr2qlEpXSm1XSvVuinKEEEKI85naJ4Yai41F248aHUqjNDpBK6VigNHA4VM2Xw60r7tNB95sbDlCCCFEQ3SNCqRTRADznXzqz6aoQb8EPASc2mVuIjBX19oItFJKtW2CsoQQQohzUkoxNSWGbdnF7MsrNTqci9aoBK2UmgjkaK23/eqpKODUry7ZdduEEEKIZjepZyTuJuXUC2icN0ErpVYopXae4TYR+AvwRGMCUEpNV0qlKqVSCwoKGnMoIYQQAoA2/l5c2jmMBVtzMFttRodzUc6boLXWo7TWXX99AzKBeGCbUioLiAY2K6UigBwg5pTDRNdtO9PxZ2mtU7TWKaGhoY19PUIIIQRQ21mssKyG7/fmGx3KRbnoJm6t9Q6tdZjWOk5rHUdtM3ZvrXUesAi4sa439wCgWGvtOkuMCCGEcHgjOoYS4u/ltGOim2sc9BJqa9jpwNvAXc1UjhBCCHFG7m4mru4dxfd78yksqzY6nAvWZAm6riZdWHdfa63v1lonaq27aa1Tm6ocIYQQoqGmpkRjsWkWbDnjVVaHJjOJCSGEcFlJYQH0jGnFZ6lH0Nq5FtCQBC2EEMKlTU2JZv+xMrZnFxsdygWRBC2EEMKlXdkjEi93E/PSnGtMtLvRATSUzWajsLCQkydPYrVajQ5HtBDe3t5ER0fj4eFhdChCiIsU6O3B2K4RLNp6lMfHJePt4WZ0SA3iNAk6OzsbpRRxcXF4eHiglDI6JOHitNYcP36c7Oxs4uPjjQ5HCNEI16TEsHDrUb7dfYwJPSKNDqdBnKaJu7y8nKioKDw9PSU5C7tQStGmTRuqqqqMDkUI0UgDE9oQ1crHqab+dJoEDWAyOVW4wgXIl0EhXIPJpLi6TzQ/pBeSc7LS6HAaRDKeEEKIFmFqn2i0hi+cZGYxSdDNbMSIEdxzzz2N3sdZ+fv7M2fOHKPDEEIIYoJ9GZAQzPzN2U4xJloStAP44osv+Ne//mV0GC7r2WefZdKkSUaHIYRwANekxHDoeAWbDhYZHcp5SYJ2AMHBwQQEBDRrGTabzSGHp5nN5mYvY+HChZKghRAAXN61Lf5e7k6xgIYkaDuwWCzcd999tG7dmtatW/PnP/8Zm+2X9Ul/3cQdFxfHP//5T2bMmEFgYCDR0dG88MILpx3zxRdfpHv37vj5+REVFcVtt93GyZMn65+fM2cO/v7+LFmyhK5du+Lp6cm6devw8PAgLy/vtGM99thjdO/e/byvo7i4mD/84Q+EhYXh7e1NQkICL7/8cv3z6enpjBgxAm9vbzp27MhXX3112u9nZWWhlOKTTz7hkksuwcfHh5kzZwLw3nvvkZycjLe3Nx06dOCll1467Rz92pNPPknXrl155513iI2NxcfHh0mTJlFYWHjafnl5eaSmpjJ+/HgA8vPzmThxIj4+PrRr147Zs2fTtWtXnnzyyfO+fiGE8/PxdGN897Ys2ZFLWbXF6HDOSRK0HXz00UfYbDY2bNjAzJkzmTVr1mmJ7UxeeuklunXrxubNm3n44Yd56KGH2LBhQ/3zJpOJl19+mV27dvHxxx+zadMm7r333tOOUVVVxT/+8Q9mzpzJ7t276dWrF4mJicydO7d+H5vNxty5c7n11lvP+zoef/xxduzYwVdffcW+ffuYPXs2UVFR9ce56qqr6l/n7NmzefLJJ6mu/u0KMo8++ih33XUXu3fvZtKkSbz99tv85S9/4e9//zt79uzhP//5D8899xxvvPHGOePJysriww8/ZOHChaxYsYIDBw5wyy23nLbPokWLGDhwICEhIQBMmzaN9PR0VqxYwYIFC5g7dy5ZWVnnfe1CCNcxNSWaihorS7Y79irITjNRyZk8tXgXu4+W2LXM5MhA/nZllwv6nbZt2/Lqq6+ilKJTp07s37+fF198kT/96U9n/Z3Ro0fX16rvvfdeXn31VVauXMnAgQMBuP/+++v3jYuL4/nnn2fixIm8//779cPRrFYrr732Gn369Knf97bbbuPdd9/loYceAmDZsmXk5+dzww03nPd1HDp0iN69e9OvXz8A2rVrV//cihUr2L17NwcPHiQ2NhaAl19+maFDh/7mOPfeey9Tpkypf/yPf/yD559/vn5bfHw8jzzyCG+88cY5O89VVlYyd+7c+vJmzpzJ0KFDOXDgAO3btwdOb97ev38/S5cu5YcffmDw4MEAvP/++yQkJJz3tQshXEfv2NYkhPoxL+0I1/SNMTqcs5IatB0MGDDgtPG0AwcOJCcnh5KSs3+5+HWTc2RkJPn5+fWPv/vuOy677DKio6MJCAhg8uTJ1NTUnNZ87e7uTs+ePU87zk033URmZibr168HYPbs2UyaNIk2bdqc93Xceeed/O9//6NHjx48+OCDrF69uv65PXv2EBUVVZ8sAfr373/GsespKSn19wsKCjhy5AgzZszA39+//vbII4+QkZFxznjOVt6ePXsAKCsrY+XKlUycOLE+RpPJVP8FA2q/ZERGOsesQkKIpqGUYkqfaH7KOsHBwnKjwzkrp65BX2hN1pn8eu5npVT9NdlDhw4xbtw4br/9dv7+97/Tpk0bNm/ezO9//3tqamrqf8fLyws3t9PnnA0NDWXChAnMnj2bjh07smjRIhYvXtygmC6//HIOHTrE0qVLWblyJePGjWPq1Km89957F/Ta/Pz86u///JreeustBg0adEHHOZ9vvvmGpKQkEhMTT9suk48IIa7uHc2/l+1jftoR/jymk9HhnJHUoO3gxx9/PG3M3caNG4mMjCQwMPCijpeamkpNTQ0vvfQSAwcOpEOHDhw9erTBv3/77bfz2WefMXPmTCIiIhg1alSDfzckJIQ//OEPzJkzh3fffZf333+f6upqOnfuTE5ODkeO/DKN3qZNm87Z0QsgPDycyMhIMjIySEpK+s3tXM5WXufOnYHf9t7u1KkTNpuNTZs21W87fPjwBZ07IYRrCA/0ZniHUD5Py8Fqc8wx0ZKg7eDo0aPcf//97Nu3j/nz5/PCCy/wwAMPXPTx2rdvj81m4+WXX+bgwYN88skn5+10dqrLLruMNm3a8NRTTzFt2rQGT6H6xBNPsGDBAg4cOMCePXv44osvSEhIwMvLi1GjRtGpUyduvPFGtm7dyoYNG3jggQdwdz9/I81TTz3F888/z0svvcS+ffvYuXMnc+fOPe/YcB8fH2666ab68u644w7GjRtH+/btsVgsfP311/XN2wAdO3Zk7NixzJgxgw0bNrB161amTZuGj49Pg16/EMK1TE2JIa+kih/SC8+/swEkQdvB9ddfj9VqpX///tx+++3ceuutjUrQ3bt355VXXuHFF18kOTmZd955h3//+98N/n2lFDfffDNms5mbb765wb/n5eXFY489Ro8ePRg8eDClpaX1zeMmk4kvv/wSm81G//79ufHGG3n88cfx8vI673Fvu+02Zs+ezQcffECPHj0YOnQos2bNOu8KUnFx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\n",
      "text/plain": [
       "<Figure size 576x720 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "x_ax = x_axis.numpy().squeeze()\n",
    "true_ratio = logp(x_ax) - logq(x_ax)\n",
    "\n",
    "fig, axs = plt.subplots(3, 1, figsize=(8, 10))\n",
    "axs = axs.ravel()\n",
    "\n",
    "axs[0].plot(x_ax, tre_p_over_q, label=\"tre p/q\")\n",
    "axs[0].plot(x_ax, true_ratio, label=\"true p/q\")\n",
    "axs[0].legend(fontsize=14)\n",
    "\n",
    "axs[1].plot(x_ax, binary_sdre_p_over_q, label=\"binary_sdre p/q\")\n",
    "axs[1].plot(x_ax, true_ratio, label=\"true p/q\")\n",
    "axs[1].legend(fontsize=14)\n",
    "\n",
    "axs[2].plot(x_ax, multiclass_sdre_p_over_q, label=\"multiclass_sdre p/q\")\n",
    "axs[2].plot(x_ax, true_ratio, label=\"true p/q\")\n",
    "_ = axs[2].legend(fontsize=12)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "torch1.8",
   "language": "python",
   "name": "torch1.8"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
