{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 135,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/usr/lib/python3.6/_collections_abc.py:841: MatplotlibDeprecationWarning: Support for setting the 'text.latex.preamble' or 'pgf.preamble' rcParam to a list of strings is deprecated since 3.3 and will be removed two minor releases later; set it to a single string instead.\n",
      "  self[key] = other[key]\n"
     ]
    }
   ],
   "source": [
    "%matplotlib inline\n",
    "import copy\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "# import torch.linalg\n",
    "from utils import FakeDL\n",
    "import algos.pyhessian as pyhessian \n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "from utils import kp_2d\n",
    "import matplotlib as mpl\n",
    "from tqdm import tqdm\n",
    "\n",
    "# mpl.use('Agg')\n",
    "plt.style.use('ggplot')\n",
    "plt.rc('font', **{'family': 'serif', 'serif': ['Computer Modern']})\n",
    "plt.rc('text', usetex=True)\n",
    "params= {'text.latex.preamble' : [r'\\usepackage{amsmath}',r'\\usepackage{amssymb}', r'\\usepackage{bm}']}\n",
    "plt.rcParams.update(params)\n",
    "\n",
    "\n",
    "dev = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "print(dev)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 136,
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_sharpness_traj(ss, eos, fig_size, exp_name):\n",
    "\n",
    "    plt.figure(figsize=fig_size)\n",
    "    plt.axhline(eos, label=r\"EoS Threshold ($2/\\eta$)\", linestyle='--', color='black')\n",
    "    plt.scatter(np.arange(len(ss)), ss, label=\"Numerical Sharpness\", s=7, zorder=100)\n",
    "    plt.legend(loc=1)\n",
    "    plt.xlabel('Iterations')\n",
    "    plt.ylabel('Sharpness')\n",
    "    plt.tight_layout()\n",
    "    plt.savefig(f\"./figs/{exp_name}_sharpness.pdf\")\n",
    "    plt.show()\n",
    "\n",
    "def plot_loss_traj(losses, fig_size, exp_name, logy=False):\n",
    "\n",
    "    plt.figure(figsize=fig_size)\n",
    "    plt.plot(np.arange(len(losses)), losses, label=\"Loss\", zorder=100)\n",
    "    plt.legend(loc=1)\n",
    "    plt.xlabel('Iterations')\n",
    "    plt.ylabel('Loss')\n",
    "    if logy:\n",
    "        plt.yscale('log')\n",
    "    plt.tight_layout()\n",
    "    plt.savefig(f\"./figs/{exp_name}_loss.pdf\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 141,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_dataset_single_dirc(n, d):\n",
    "    # assert n >= d\n",
    "    X = np.sqrt(n) * torch.qr(torch.normal(torch.zeros(n, d)))[0]\n",
    "    inc_rate = torch.diag(torch.Tensor([1] + [0] * (d-1)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_multi_dirc(n, d):\n",
    "    X = np.sqrt(n) * torch.qr(torch.normal(torch.zeros(n, d)))[0]\n",
    "    inc_rate = torch.diag(torch.Tensor(np.linspace(1, 1/d, d)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_factorization(n, d):\n",
    "    # assert n >= d\n",
    "    X = np.sqrt(n) * torch.eye(d)\n",
    "    inc_rate = torch.diag(torch.Tensor(np.linspace(1, 1/d, d)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_factorization_single(n, d):\n",
    "    # assert n >= d\n",
    "    X = np.sqrt(n) * torch.eye(d)\n",
    "    inc_rate = torch.diag(torch.Tensor([1] + [1e-5] * (d-1)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_scalar_factorization(n, d):\n",
    "    # assert n >= d\n",
    "    X = torch.eye(1)\n",
    "    Y = torch.eye(1)\n",
    "    return X.to(dev), Y.to(dev)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "class LinearNet(nn.Module):\n",
    "    def __init__(self, L, d):\n",
    "        super(LinearNet, self).__init__()\n",
    "        self.L = L\n",
    "        self.layers = []\n",
    "        for i in range(L):\n",
    "            self.layers.append(nn.Linear(d, d, bias=False))\n",
    "            self.add_module(\"W{}\".format(i+1), self.layers[-1])\n",
    "\n",
    "    def forward(self, x):\n",
    "        for i in range(self.L):\n",
    "            x = self.layers[i](x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "class LinearNetSF(nn.Module):\n",
    "    def __init__(self, L, d):\n",
    "        super(LinearNetSF, self).__init__()\n",
    "        self.L = L\n",
    "        self.layers = []\n",
    "        self.layers.append(nn.Linear(1, d, bias=False))\n",
    "        self.add_module(\"W{}\".format(1), self.layers[-1])\n",
    "        self.layers.append(nn.Linear(d, 1, bias=False))\n",
    "        self.add_module(\"W{}\".format(2), self.layers[-1])\n",
    "\n",
    "    def forward(self, x):\n",
    "        for i in range(self.L):\n",
    "            x = self.layers[i](x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "class LinearReLUNet(nn.Module):\n",
    "    def __init__(self, L, d):\n",
    "        super(LinearReLUNet, self).__init__()\n",
    "        self.L = L\n",
    "        self.layers = []\n",
    "        for i in range(L):\n",
    "            self.layers.append(nn.Linear(d, d, bias=False))\n",
    "            self.add_module(\"W{}\".format(i+1), self.layers[-1])\n",
    "\n",
    "    def forward(self, x):\n",
    "        for i in range(self.L):\n",
    "            x = self.layers[i](x)\n",
    "            if i != self.L - 1:\n",
    "                x = nn.functional.relu(x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 142,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_eigeninfo(net, dl, topn, criterion):\n",
    "    # Computing the top n eigenvectors using Power Iteration Method.\n",
    "    net.eval()\n",
    "    hessian_comp = pyhessian.hessian(net, criterion, dataloader=dl, cuda=False if dev==\"cpu\" else True)\n",
    "    eigenvals, eigenvecs = hessian_comp.eigenvalues(top_n=topn, tol=1e-4, maxIter=10000)\n",
    "    # print(eigenvals)\n",
    "    net.train()\n",
    "    return eigenvals, eigenvecs\n",
    "\n",
    "def compute_hvp(net, dl, topn, criterion, v):\n",
    "    # Computing the top n eigenvectors using Power Iteration Method.\n",
    "    net.eval()\n",
    "    hessian_comp = pyhessian.hessian(net, criterion, dataloader=dl, cuda=False if dev==\"cpu\" else True)\n",
    "    # print(\"conducting HVP with v{}\".format(v.shape))\n",
    "    val, hvp = hessian_comp.dataloader_hv_product(v)\n",
    "    return val, hvp\n",
    "\n",
    "def eigenthings_tensor_utils(t, device=None, out_device='cpu', symmetric=False, topn=-1):\n",
    "    t = t.to(device)\n",
    "    if topn >= 0:\n",
    "        _, eigenvals, eigenvecs = torch.svd_lowrank(t, q=min(topn, t.size()[0], t.size()[1]))\n",
    "        eigenvecs.transpose_(0, 1)\n",
    "    else:\n",
    "        if symmetric:\n",
    "            eigenvals, eigenvecs = torch.symeig(t, eigenvectors=True) # pylint: disable=no-member\n",
    "            eigenvals = eigenvals.flip(0)\n",
    "            eigenvecs = eigenvecs.transpose(0, 1).flip(0)\n",
    "        else:\n",
    "            _, eigenvals, eigenvecs = torch.svd(t, compute_uv=True) # pylint: disable=no-member\n",
    "            eigenvecs = eigenvecs.transpose(0, 1)\n",
    "    return eigenvals, eigenvecs\n",
    "\n",
    "\n",
    "def quadratic_loss(Y_hat, Y):\n",
    "    \"\"\"Batched quadratic loss\"\"\"\n",
    "    diff = torch.square(Y - Y_hat)\n",
    "    loss = diff.sum() * 0.5 / len(Y)\n",
    "    if len(Y_hat.shape) != 1:\n",
    "        loss\n",
    "    return loss"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_Wprod_singular_space(Ws):\n",
    "    W_prod = Ws[0].clone()\n",
    "    for W in Ws[1:]:\n",
    "        W_prod = W.matmul(W_prod)\n",
    "\n",
    "    u_prod, s_prod,v_prod = np.linalg.svd(W_prod)\n",
    "    return W_prod, u_prod, s_prod, v_prod"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 9999  Loss: 3.58984e-08 Sharpness: 199.188: 100%|██████████| 10000/10000 [02:23<00:00, 69.75it/s]\n",
      "Epoch 9999  Loss: 5.59372e-07 Sharpness: 298.558: 100%|██████████| 10000/10000 [02:19<00:00, 71.51it/s]\n",
      "Epoch 9999  Loss: 0.000382192 Sharpness: 403.419: 100%|██████████| 10000/10000 [02:43<00:00, 60.99it/s]\n"
     ]
    }
   ],
   "source": [
    "seed, eval_start, eval_end = 0, 0, 10000\n",
    "freq = 31\n",
    "d = 50 # layer width\n",
    "n = 50 # # of samples\n",
    "L = 5 # # of layers\n",
    "exp_name = 'ReLU_multilr'\n",
    "sharps = [200, 300, 400]\n",
    "\n",
    "res = {}\n",
    "\n",
    "for sharp in sharps:\n",
    "    eta = 2/sharp\n",
    "\n",
    "    torch.manual_seed(seed)\n",
    "    hessian_topn = 1\n",
    "\n",
    "    X, Y = get_dataset_multi_dirc(n, d)\n",
    "    X, Y = X.to(dev), Y.to(dev)\n",
    "\n",
    "    epochs = eval_end\n",
    "    hessian_eval_epochs = list(range(eval_start, eval_end, freq))\n",
    "\n",
    "    dataloader = FakeDL(X, Y, dev)\n",
    "    loss_record = []\n",
    "    eigenvals_record = []\n",
    "    eigenvecs_record = []\n",
    "\n",
    "    weight_record = []\n",
    "    weight_prod_record = []\n",
    "    Y_record = []\n",
    "\n",
    "    net = LinearReLUNet(L, d).to(dev)\n",
    "    for layer in net.layers:\n",
    "        nn.init.xavier_normal_(layer.weight, gain=1)\n",
    "\n",
    "    # criterion = nn.MSELoss(reduction='mean')\n",
    "    optimizer = torch.optim.SGD(net.parameters(), lr=eta)\n",
    "\n",
    "    trange = tqdm(range(epochs))\n",
    "    for epoch in trange:  # loop over the dataset multiple times\n",
    "        optimizer.zero_grad()\n",
    "        \n",
    "        Y_hat = net(X)\n",
    "        loss = quadratic_loss(Y_hat, Y)\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        \n",
    "        loss_record.append(loss.item())\n",
    "\n",
    "        if epoch in hessian_eval_epochs:\n",
    "            eigenvals, eigenvecs = compute_eigeninfo(net, dataloader, hessian_topn, quadratic_loss)\n",
    "            eigenvals_record.append(eigenvals)\n",
    "            eigenvecs_record.append(eigenvecs)\n",
    "\n",
    "            Ws = [W.weight.data.clone() for W in net.layers]\n",
    "            W_prod = Ws[0].clone()\n",
    "            for W in Ws[1:]:\n",
    "                W_prod = W.matmul(W_prod)\n",
    "            \n",
    "            weight_record.append(Ws)\n",
    "            weight_prod_record.append(W_prod)\n",
    "            Y_record.append(Y_hat)\n",
    "\n",
    "        if epoch % 3 == 0:\n",
    "            trange.set_description_str(\"Epoch {}  Loss: {:.6g} Sharpness: {:.6g}\".format(epoch, loss.item(), eigenvals_record[-1][0] if len(eigenvals_record) > 0 else -1))\n",
    "        \n",
    "        sharpness_array = np.array([v[0] for v in eigenvals_record])\n",
    "\n",
    "    res[sharp] = [loss_record, sharpness_array, hessian_eval_epochs]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [],
   "source": [
    "res_relu_deep = copy.deepcopy(res)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "323\n",
      "323\n",
      "323\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 288x360 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "f, (ax1, ax2) = plt.subplots(2, 1, sharex=True, figsize=(4, 5), gridspec_kw={'height_ratios': [0.6, 1]})\n",
    "for sharp in res:\n",
    "    loss_record, sharpness_array, hessian_eval_epochs = res[sharp]\n",
    "    print(len(sharpness_array))\n",
    "    ax1.plot(np.arange(len(loss_record)), np.array(loss_record), label=r'$\\eta=2/{}$'.format(sharp))\n",
    "    ax2.scatter(hessian_eval_epochs, sharpness_array, s=5, label=r'$\\eta=2/{}$'.format(sharp))\n",
    "    ax2.axhline(sharp, linestyle='--', color='black')\n",
    "    ax2.text(epochs * 0.95, sharp + 15, r'$\\lambda={}$'.format(sharp), ha='right')\n",
    "\n",
    "ax1.set_ylabel('Loss')\n",
    "ax1.set_yscale('log')\n",
    "ax1.legend(loc=1)\n",
    "\n",
    "\n",
    "ax2.legend(loc=4)\n",
    "ax2.set_ylabel('Sharpness ($\\lambda$)')\n",
    "ax2.set_xlabel('Iterations')\n",
    "ax2.set_ylim(0, 600)\n",
    "plt.tight_layout()\n",
    "plt.savefig(f\"./figs/Intro_{exp_name}.pdf\", bbox_inches='tight', pad_inches=0)\n",
    "plt.show()\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 143,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 99  Loss: 2.33042e-07 Sharpness: 23.3556: 100%|██████████| 100/100 [00:19<00:00,  5.06it/s]\n",
      "Epoch 99  Loss: 2.08536e-06 Sharpness: 32.3144: 100%|██████████| 100/100 [00:19<00:00,  5.19it/s]\n",
      "Epoch 99  Loss: 1.2804e-05 Sharpness: 37.463: 100%|██████████| 100/100 [00:21<00:00,  4.65it/s] \n"
     ]
    }
   ],
   "source": [
    "seed, eval_start, eval_end = 0, 0, 100\n",
    "freq = 1\n",
    "d = 10 # layer width\n",
    "n = 10 # # of samples\n",
    "L = 2 # # of layers\n",
    "exp_name = 'Linear_multilr_2layer'\n",
    "sharps = [30, 35, 40]\n",
    "\n",
    "preset_scales = [4, 0.1]\n",
    "\n",
    "res = {}\n",
    "X, Y = get_dataset_multi_dirc(n, d)\n",
    "X, Y = X.to(dev), Y.to(dev)\n",
    "\n",
    "net = LinearNet(L, d).to(dev)\n",
    "for i, layer in enumerate(net.layers):\n",
    "    nn.init.xavier_normal_(layer.weight, gain=1)\n",
    "    layer.weight.data *= preset_scales[i]\n",
    "sd = net.state_dict()\n",
    "\n",
    "for sharp in sharps:\n",
    "    eta = 2/sharp\n",
    "\n",
    "    torch.manual_seed(seed)\n",
    "    hessian_topn = 1\n",
    "    epochs = eval_end\n",
    "    hessian_eval_epochs = list(range(eval_start, eval_end, freq))\n",
    "\n",
    "    dataloader = FakeDL(X, Y, dev)\n",
    "    loss_record = []\n",
    "    eigenvals_record = []\n",
    "    eigenvecs_record = []\n",
    "\n",
    "    weight_record = []\n",
    "    weight_prod_record = []\n",
    "    Y_record = []\n",
    "\n",
    "    net = LinearNet(L, d).to(dev)\n",
    "    net.load_state_dict(sd)\n",
    "\n",
    "    # criterion = nn.MSELoss(reduction='mean')\n",
    "    optimizer = torch.optim.SGD(net.parameters(), lr=eta)\n",
    "\n",
    "    trange = tqdm(range(epochs))\n",
    "    for epoch in trange:  # loop over the dataset multiple times\n",
    "        optimizer.zero_grad()\n",
    "        \n",
    "        Y_hat = net(X)\n",
    "        loss = quadratic_loss(Y_hat, Y)\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        \n",
    "        loss_record.append(loss.item())\n",
    "\n",
    "        if epoch in hessian_eval_epochs:\n",
    "            eigenvals, eigenvecs = compute_eigeninfo(net, dataloader, hessian_topn, quadratic_loss)\n",
    "            eigenvals_record.append(eigenvals)\n",
    "            eigenvecs_record.append(eigenvecs)\n",
    "\n",
    "            Ws = [W.weight.data.clone() for W in net.layers]\n",
    "            W_prod = Ws[0].clone()\n",
    "            for W in Ws[1:]:\n",
    "                W_prod = W.matmul(W_prod)\n",
    "            \n",
    "            weight_record.append(Ws)\n",
    "            weight_prod_record.append(W_prod)\n",
    "            Y_record.append(Y_hat)\n",
    "\n",
    "        if epoch % 3 == 0:\n",
    "            trange.set_description_str(\"Epoch {}  Loss: {:.6g} Sharpness: {:.6g}\".format(epoch, loss.item(), eigenvals_record[-1][0] if len(eigenvals_record) > 0 else -1))\n",
    "        \n",
    "        sharpness_array = np.array([v[0] for v in eigenvals_record])\n",
    "\n",
    "    res[sharp] = [loss_record, sharpness_array, hessian_eval_epochs]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 144,
   "metadata": {},
   "outputs": [],
   "source": [
    "res_shallow = copy.deepcopy(res)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 145,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAARgAAAFgCAYAAABg/2YgAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8QVMy6AAAACXBIWXMAAAsTAAALEwEAmpwYAABYfUlEQVR4nO29eXhb1bnv/9my5EGy5UmRMtqxE0IIsQKEgGOImQxlCEmhDJfGObfcc0i53BN64d5T6MNpH05bStL7nPBrueRAuOe0T3FoKaelmZjiQLCL4zSlJFKYYzl2iBMpg+NJnrV/fyh7R6Ml2ZIs2+vzPHkU7bX32q/28PW73rXWuyRZlmUEAoEgAWjG2wCBQDB5EQIjEAgShhAYgUCQMITACASChCEERiAQJAwhMAKBIGFox9uAeNHW1hZxH5PJxOnTp5NgzeiZCDbCxLBT2BgforFx5syZIbcLD0YgECQMITACgSBhTJom0miRBwfh048hLQ3yCpFmzx1vkwSCScOUFhjZM4znpY1w6C/qNs1T/4o096JxtEowHsiyTF9fHx6PB0mSknZep9NJf39/0s43GhQbZVlGo9GQmZkZ9TWasgIjyzLy716GQ39Buvu/IpVchGfTj5APHRACMwXp6+tDp9Oh1Sb3ldBqtaSlpSX1nLHia+PQ0BB9fX1kZWVFdeyUjcHI9e8iv/8m0i3fRHPbt5AWWqF0AbL9r+NtmmAc8Hg8SReXiYhWq8Xj8US9/5QUGPnsKeTX/wMuWYL0re+o26XFS6HlCHJn+/gZJxgXktksmujEcq1SWmAcDgc2my1u9Xk8HmRZxlPzb+DxoFn7P5A0Fy6BVHYlAPLhj+N2ToEgFWlpaUnKecbNJ3Q4HJSWlqrfGxsb0ev1NDc3s3r1agAaGhqYP39+XM63e7sdV1c2K5v/jeyWz5Hu/wcwWWjrHOCwy03T2T7ae9NZNWMxiw5/BBU3xuW8AsF4UFNTA3iF5KmnnvIrs9vt6v/r6uoAqK+vV/fbuXMnRqORw4cP88gjj4zJjnHxYGw2Gy+99JL63eFwAGC1WjEYDOr3ioqKuJ2zbwiM2ny2lKzjzzev42XjMtZtc/Dfdzh4Yf9J6o928vnpPv55wVq2dFvo6x+M27kFgmRSV1fHihUrqK6upqWlRRURhUOHDlFWVobdbqe+vp7Kykrsdrv6D6CyshKj0egnRqNhXATGarWSnZ2tfm9oaECv1wNgNpvH/KNCsfKuxRTMzuBibSGvDl5CraOT4rwMvrvMwr/dWcrWey/ipVXzuCO/j7fNV/KTt7/EPTgcdzsEgkTT2tpKfX09AMXFxbS2tobcr6ysTPVaWltbKSsrY/v27RiNRgCKiorUekZLSoTN3W63n+B0dXUBXs/m5MmTlJWVYTAYxnQOSZK4/bZZ/OG3zazqK+TiskxK52eg0VwIWGXpJP7h5kXM/9df8ss5d/CjPcd4+sY5ZKendjeiIL54fvcy8rHmuNYpzSlB818eCluueBOLFy8GGFPzpLq62q/eVatWqd9bWlpYsmSJ3/6bN29mw4YNAHR0dJCfn6+WtbePrcMjJQQmHFVVVXGtT5euoXxFNgcPuPns4z5avxpgwaWZzCrSIZ0XGik9g+tuuJLMP/2Gn5f9V371Nxfry2fE1Q6BIBCj0Uh7ezv5+fmUlZWpMRSFlpaWsN6Er6D4YrfbqayspKysTN1WX18ftP8jjzzCunXrgoQnHqSEwOj1erq7uwGvN5OTkxPxmNraWmprawHYsGEDJpMp4jFarZa5pRaKS2SOHXXzt/1n+Hi/m6YvdFxSlsv8hUbS0zXIt3+LivffZPXpA7zBVXzzsiIun507th8ZJVqtNqrfMt5MBDtjsdHpdF4YB1P93xNoVTBarZZ58+Zx7NgxLr/8cgA0Go3fuJx58+Yxb968mOr98MMPefTRR/22+dar9NBarVZKSkr47W9/S35+Pp2dnWi1Wnp6eigsLFRtVMjIyIj6uqaEwFRUVKiBXafT6ae44aiqqvLzcKKZ8u477VyfA9fclMWJr7Uc+ayf/fWn+eu+05TMz6D04gzS71rLfc/9Cw03XM6zu7/g/7t9LulpiQ9ZTYTp+zAx7IzFxv7+/nEZUavVahkaGgK8wyiGhoaw2+1YrVZ1O3g9mF27doWsI1RTqqamhocffpihoSHq6uqorKykpaWF2bNnq/Xu3buXxYsXMzQ0RHt7O2VlZVxzzTUcOnSIa665hubmZlasWAHgZ0t/f3/QdQ2XrmFcBKaxsRGHw0FjYyPl5eWUlpaqY14MBoNf93UikSSJmXPSmTknnXNnhmj6sp8jn/fT/FU/c0rmU3TRlXy3aTs/nvstXth/ku8tn4FGDMgSJICWlhYqKysBbzNGebEViouLo47J1NXV8bOf/YzNmzdz7tw5XnzxRbVe3+bRmjVr2LFjBzt37gRg5cqVgLeXqa6uDqPRGNUf+5GQJsu6SPFKONXVOUzTZ/0cbx3A45GZ7jxA8/xZbG1P486F+fz9FeaEjvqcCJ4BTAw7Y7HR7XarPZnJxNeDUVi3bh1btmyJ+7lqamrCxmtGItDGUNdKJJyKkhxjGpddreemlUbmX5LBaVMZWV2zWJtj5tAXbv7fX10MeyaFJgtSlLy8vLjX2dHRkZB6IyEEJgyZWRouseq50fQxC5r+E+Ogh1vS8tE3pfFvu5w4z4mBeILE8POf/zzudR46dEhtAiUTITARSK+8gfk9B7jhg/VcUXKWvBwtxe5MGt/u5j93nOGzL3sZHBQejSC1UeI7yUYITASkzCw0//tnaIx5TP/Nk9xm2Mei8jQ6jUMM9cCRj/t584/neOudc3z+WS9dHcNMkrCWQDBmUqKbOtWR8gvR/O9n8Pzbs8hb/4252VspuaIC5/wr2DtUzImTMK1dx9A5+MrWD1qZPFMaMy3p5BdqMeanodWK3ifB1EMITJRIeQVonvw5fPkJnvd2Iv/lA8x1b3MfMDyjiC/ml3Mg91JODBagH0jHckLHuZNKYh6ZzGwNhYVacvPTyM1Lw5iXRnqGcCAF40NLSwvFxcUJP48QmBiQJAkuXkzaxYuRh4bA8QXyV5+QduQzFn20i0Xu3wNwMn8OttJyPstfyGkKyfSkY+rU0d6tJavlwmCu9ExJFRtjbho5uWnk54vmlWDsjCVdwzPPPMNTTz016m5tX4TAjBJJq4UFlyItuBQA2eMB53Hkps+Z4fiC6U37ueWj15EBp97E5yXL+NK8kOZ0Cz0DmeSjJd+tw9Kvw3gyDQlvE6pe00V2jgZj3gVPJzdfeDuC6FHSNRQXF7Nu3Tp1JK/CoUOHqK6uVidYPvXUU2zevBm73U5ZWRlbt25l165d6gTIsSAEJk5IGg3MmIM0Yw5cezMAsrsbHF8w46vPmH7kU66vr4WhQXrTMmiat4yv5izhi6xZfNmXDgMaCiQtFtKZ1ZuOsSeN4y0X4jaZeom8fG8TK6/A+0+IjiAUra2ttLa2UlxcHDFdgzJSV0nXAPDiiy/GrddJCEwCkfTZsHipN9cv59dgav4S/Rd2Fn9uY/Hel2B4CFmXjnPhVXxZehUO00UcONNF67l+0pEwa9JZaMhkpjaD4bMyJ49fGH9jyNaQV5hGfqGWApO3ieWbfkIwOv7fX500t/fFtc6S/Ez+4UpL2PJUSdegnHusNigIgUkikk53oVl1539B7u+DLz9B/uRvTD/8N6bb/0wlwKxiepZU8PncZXwi67E7e9nT3oFHhhythqV52czPykTnSeO0c4jjLV7R0Woh36SlwKSl0KwlryCNtDQhOBOBVEnXkJubq4pKfX09dXV13Hjj6NPHCoEZR6SMTChbilR23sNxtaE/8indH76H4a3XWCr/lqXTpiMtW0FPxQo+SSvk4IkePj7Rw97TnQDMzE7n6mIDl+j1ZA6k0X56iC8Oe//6atKgwKTFZNEyzeJtXons+ZEZydNIFMXFxbS0tISdXKg0d2Khvr5+RA9ECfaWlZVRXFzM1q1bKSoqArwTH/Pz88M2r6JFCEwKIZlnYlhkpbeiCrmrA/ngfuS//hn5rT+gf/N1ls0p4arlNyLdcB0nJT0ftXXz1+M97Gg+xxuedowZaVw1O5vyhdnMSMvg3KkhTruG+NzWx+dAeobENIuWaTN0mGdoyRAxnJTEbrdz2WWX+W0bTboGZbtvugZFQAC/JllHRwdLliyhuLhY3efo0aOsXbt2TL9FCEyKIuXkIq24BVbcgtzZjnzgz8j73kf+/b8j/+HXmJdczR0rbuGO6y+jd1jm4xM9NB7rpqG1i9qmDrLTNZTPyaHyMiNXGQ2cdQ1z6uQgrpNDHG/1NqnyC9OwzNQxfZaObKNGeDfjSCqla6ipqSEvL4+5c+eKdA0K8UrXMN5EslE+3or84W7kfe9DdydMm450/W1I19yMZMhmcNjDwRNu/tzayf5j3fQOecjP0nL9XCM3luYyJzedjvZhnG1DONsG6Wj3JjY35GiYMVvHjNm6qJpSk+Fa+iLSNYRnLOkahAczwZBmFSHd9/fId/0d8t8akD94C/n1XyFvexWp4ka0VatZNnsmy2Zn0z/k4cDxbj442sn2z8/yxmdnuagwk5tKc7nuYiMXL86k1+3BeXyQE8cHafq8nyOf9ZNl0DBzjo5ZRTqMeSJuk2wmU7oG4cGkGKOxUW51IL+3A3n/BzA8DJddjea2e5BKFqj7dPQNUXe0k91NHbSc6ycjTWLFXCO3XZTP/MJMAAb6PZw8PkjbsUFOO4eQZcjO0TCrOJ1ZxToM2RdGIU+2a5lKHkwiCBxsFwtj8WCEwKQYY7FR7mhHfm8X8t5d4O6BhVY0K++HBYtVL0SWZY6c7eOdr85Rd7ST/mGZiwozuWNBPtcW56A7n3d4oN9D27FBjrcOcPaUtxmVb0pjzlxvitEZM6dNqms52QVmLAiBQQiML3KfG/mDd5DffQM6z8FFi9Cs+jbSQqvffj0Dw+xt7uTNL9v5unOA3Iw0bl2Qx+0X5ZOXdaH17O7xcLxlgK+PDtDd5UGTBiXzcjDP9FBo1qZsE0oITHwQAoMQmFDIA/3I9buR3/5POHcWLi5D8801SPMX+e8nyxw66WbnF2c5cLwHrUbiurlGvnlJAUV5GX77nTs7zLHmAU4cG2JgwIM+W0NRSTpzStLJzEqtbm8hMPFBCAxCYEZCHhxArnsH+c3XvR5N2ZVo7l6LNLskaN/jnQPs+PwsexwdDAzLXDnTwN2LCllkzvLzVPJyCzh86CStjn7OnBpGkmD6LB3F89IxWVLDqxECEx+EwCAEJhrk/j7kPTuQ3/kj9LqRrr4e6ZvVSIXTgvbt7B/mrS/b2flFO539w1xsyuKeSwu4clY2Gknys7O7c5gWxwDHmgcYHJAxZGsonu/1atLTx8+rEQITnljywQiBQQhMLMg9Xchv/SfyHu8AK+mmO5FuvxdJH7z+d/+Qh9qmDv702VlcPYMU52Vwz6WFrL6ihPazZ/z2HR6WOXFskKNH+mk/M4wmDWYXpzN3fga5+clf1GwqC0w0+WB8B9Ft3rxZHci3c+dOjEajOtlRLFsiiAnJkIPmngfR/PRFpCuvRX73DTxPfRfP+7u8ibR8yNBquOPifP5tVSn/c/kMhj0y//phG2te+Yg9TecY8lnCJS1NYvbcdK6tyqHylmxmF6fzdcsAde928eGeLtqOedeaEiQWJR9MdXU1LS0talIphUOHDvmJS11dHQcPHgQuiE9lZSVGo9EvOdVoEAIzhZEKp6H5+8fQPLUJZhUjv/oSnh9/D/nwR0H7ajUSN5Tm8vzKEp5YMZMMbRq/bDzJIzscvHvkHIPD/sKRm69lyTI9N68ysmhJJn29Mh81uNmzs5OvPu2jv98TdA5BfGhtbVVnXo+UDyYU27dvx2g0AlBUVBR2Bne0jHokr8vlwmw2j+nkgtRAKp6H5n/9FA7tx/P6r/D84l9g8VI09/090ozZfvtqJImKIiN3Xl7CW4eO8pr9DC/sP8nrh09z72ITN5TkovNJEZGermHewkxKF2TgPDFE81f9fG7v48tP+5hdnE7pggxycpPffBqJw39z03luOK51GvPSWHxF+CbYeOWDUVI6KE2qjo4O8vPz1fL29vZR2aAQk8C8+uqrLF++nNraWgwGAxaLhZtuumlMBoTD6XTS09OD3W6nvLwciyX5U+inEpIkwWXlaBYvRX5vJ/LO1/D8y3qkG+5AuvO/eJNnBex/1ewcls3K5qO2Hn5nP31eaM5w7+JCbizNReuT/ErSSEyf5Z1Y2XlumOav+vm6ZYBWxwAmi5bSBRmYZ6RG79N4MF75YMYqIJGISWCqqqowm800NTWxYcOGqNpnDofDbzH7xsZG9Ho9zc3NrF69Ouxxzc3NlJeXq8eMtK8gfkhaHdItdyGX34D8pxpvr9P+D7y9TddWIWn8vQ1JkrhyVjZLZxr4W1sPv/URmvsWF3JDgNCA96/5kmV6FlozaW0a4OiRfv5S34MhR0Ppggxmz00f12VeRvI0EsV45YMJnD6Qm5urik5nZ6efNzMaYhIYh8NBc3MzFRUVgDeaPBI2m42tW7eyceNG9XgAq9WKy+UKEh9fFHFRPBhBcpGMeUh/94/I192G53cvI7/yAvIHb6F5YF3QQD3wCs3SWdlcMdOgejT/d/9JXv/kvNCU5JIWIDQZGRouWpTJvIUZtB0bxPFFP/aPevnc3kdxaTpzL8ogSz/1woTJygfT0tJCS0sL4I3bKM2pQ4cOqdsC00bESkwCY7FYaGhoYM2aNdTW1kYUGKvVyrZt29TvDQ0NWK3e4epmsxm73U5paSmNjY1+x5WVlWEwGLDZbJSVlYnm0TgiFc9D8/1nkQ/UI//nr/FsfBLpqusYXvc4EOxl+Ho0fz3ew2/tp3i+0evR3F9m4rq5xiCh0WgkZhenM6tIR/vpYRxf9nPki36avuhn5hwdpQsyyCuc3BP/xyMfjG/+l85Ob4bEsrIyDh06RF1dHUajccz5YGK6ayUlJZSUeEd/Wq3WmIO8breb7OwLbfmuri6AkB6KzWZj27ZtWCwWrFar8GLGEUmSkK6qRF5ylXf8zDtvcPp/7Ee67R6kW76JpEsPecyy2dlcOcvAX45381vbaX6x78R5oSlkRXGw0EiSRME0LQXTtLi7h2n+aoBWRz/HWwfJN6VRuiCDGbN0SJMwsblvE+jgwYNjSrZdWVnJp59+GvX+1dXVfsIz1rWQfIk5yFtRUcHu3bsTHuS1Wq2qtxOK2tpaamtrAdiwYQMmkylinVqtNqr9xpOUt/Ef/idDK++l5zcv0PenGjT73iP7wfVkXFUZNkB7x7Rp3LZkLvVNZ/iP/a0813CCP3x2jv92dRE3XmQKEhoATFA0FwYGPHz1aSef2s7xUYOb7Bwti6x5XHRJDukZI/c+xXItnU4nWu34eEmB5y0oKIi7LR0dHRQWFo66Xt/jMjIyor6uMY3kVbqmn3zySTXIG8mF+slPfsIPf/hDwOuKKcLR2NiI0+mMW/BWjORNLiaTiVN1tXh+9zKcOAaXLEFz/0NIs4pGPM4jy+w71sVrtjO0dPQz25jO/WUmrinKCS0055E9MifbvHGas6eH0WphTkk6JRdlYMgJLTRTeSRvIOOVDyahQd5AKioq1ECv0+kcc/tOML5Iiy5D86NfeLPqbX8Vz48fRbruNqTV30Yy5IQ8RiNJXFNkZPmcHPa1dvE7+2n+9cM2XrOnc9/iQq4N0XQCbzf3jNnpzJidzrmzQzi+7Odo0wDNXw1gmamlZEEGphROHTHexGshtViJKURvsVhoampi1apV1NbW4nQ6R9y/sbERh8OhBnGVHiObzYbBYAjbgySYOEhaLZqb7kTz05eQKr+BvPctPE89jOe9nUHTDnzRSBLXFBv5xR0lfP/amWgk2NRwgvW7mtnb3MHwCFMK8gq0XFFuoGqlkYsWZdB+ZpjGvT188HYXLU39DA3FPh1hkkzJSwqxXKuYJzs2NzfjcDiYN28ec+fOjdW2hCGaSMklnJ3y10fx/P7f4bNDMH02mvv+m3d1ywiehdp0sp+h5Vw/M3J03HtpIdeVBI+jCWR4WOZ4ywDNX/XTec6DLl2iqDSdy5dNp3+gI6rf09vbi06nS3ocZqKlaxgaGmJwcJCsrCy/feIym1rxSMxmMw6HgyVLlnD11VePwfT4IQQmuYxkpyzLcOgveF7/FbjaYNHlaO59EGn23Ij1emSZ/V9383v7aRzt/ZgNOr51aQE3leaq6TzDIcsyZ095RwkrS+yaZ2opmZ8RMUeNLMv09fXh8XiS2szKyMigv78/aecbDYqNsiyj0WjIzMwMukZxEZjAoG5jY2PKdB8LgUku0dgpDw0i730TecfvoLfXOxJ49Rqk3MijQ2VZ5q/He3jt8Gm+OtNHYZaWby4q4Jb5eWRqI7fse90eXG1pfH74HAP9Mtk5GubOz2B2STo6XerEaSbC/Y7GxrgEeXNy/AN3vmNaBIJAJK0OqWo18vIbkXe+hvz+m8h/qUO65S7v+JnMrPDH+oyjOXTSzeuHT/PvH7l4/fAZ7lyYz+0L8slOD99NnaXXsLS8kNklHtpaB2n+qp/DH/fymb1XzVFjzEutSZaTkZg8mNraWiRJwmKxqJMRfWdqjifCg0kuo1pexdWG54+/gY8aIDcfadUD3gXj0qJ70T9zuXn9kzN81NZDllbDbQvyuHNhAQVZof9OBtp47swQzUf6aWsdxOOBAlMaxfMzmDFbR1ra+Hg1E+F+j8WDiTnI29jYiM1mY968eQkbZDcahMAklzEtr3LkMzx/+DUc+cwbCL6rGi5fHnXsw3G2jz98eoaG1i40ksSNpUbuuqSQmUb/EcXhbBzo93CseYCjTQO4uz2kZ0jMKUmneF6639pPyWAi3O+kCgx4YzHNzc10d3fz7W9/O9bDE4IQmOQyVjtlWYaD+/G88Yp3oF7JAjR3/13Q0iojcaJrgDc+Pct7jg6GPDLlc7K5a1EhF5uyorJRlmVOOYdoOTKAs20QWQaTRUvxvHSmz9ShSYJXMxHud9IFRuGnP/0p//zP/zzaw+OKEJjkEi875eFh5H3vIW//LbSfhkWXoblrLdLci6Ku41zvEDu+aOetr9rpGfCwaFoW31xUwG1L5nL2zJnIFQB9vR5aHd65T71uWfVqikrTyQ4zUjgeTIT7PW4CU1tbS1VV1WgPjytCYJJLvO2UBwe8QeC3XofuLri8HM3qNUizos+B4h4cprapgx2fn8XVM8ScvCzuuMjIjaW5ZETR8wTeKQmuk0O0Oi54NYXT0igqPR+riXOemolwvxMqMPv37w871mWksmQjBCa5JMpOudeNXLsdefefoK8X6cprvcHg6bMjHqsw7JFpaO1i15FOPnN2k5ORxjfm53H7gjwK9bqo6+nr9XDsqDfrnrvbg04nMatYx5ySdHLz0+IyXmYi3O+ECsyTTz7JvHnzgrbLskxzczPPPvtsDKYmDiEwySXRdso9XcjvvIG8ZwcMDiKVX4e08n4kc+gHORSFhYXUfXqMbZ+f5S9fd6OR4NpiI3cuzOeiwvBd5EG2yDJnTnm9mhNfD+IZBmOehjklGcwq1pGRMfqkWBPhfid0HIzZbA47KbG7uzsK8wSC2JEMOUh3/x1y1Srkd/7obT7t/8C7WNzK+6ISGkmSuNSi51KLnhNdA+z6op3dTR18cLSThaYs7lyYT/mcnIhTESRJwmTWYTLrGBjw0NYySGvzAJ983Munh3qxzNRRVJLOtOlaNJMwV81YiOjBjLR6QCqtLCA8mOSSbDvljnbkt/+I/MFbMDSEdPV1SHfcO2LTKZSN7sFh9jR1sPOLdk52D1KYpeW2BXncMj+P3MzY5iF1nvOu0/11ywAD/TIZmRKzirwrWkY7iG8i3O9xC/KmEkJgkst42Sl3tCO/+wby3je9Tacrr0W64/6QeWhGsnHYI/O3th52fHGWQyfdaDUSK4pzuH1BPgtM0TefADweGdeJIY41XwgMG/PSmDNXx6zidDIywzehJsL9FgKDEJhkM952yp3nkHdvQ37/Tejv9fY63XEfUvH8mG081tHPm1+2856jk74hDxcVZnL7gnyuKcqJuvdJob/f24Q6dnSAjvZhJAmmTdcye653bE1gL9R4X8doEAKDEJhkkyp2yt2dyHt2Ir+3A9w93pnbd9wLF13KtGnTYrLRPTjM+45O3vyyna87B8hJ13DTvDxuvSiPGTnBeYcj0dUxzNctA3x9dIC+XhmtFmbMSWd2sY7C88mxUuU6joQQGITAJJtUs1PudSPvfcvbvd3VAfMWknf/39M5d0HM3cmyLGN3unnzy3Ps/7oLjwyXzzBw20V5XDkre8TUniHr83h7ob4+OsiJrwcYGoLMLG+85tIlZjx0pnQmPiEwCIFJNqlqpzzQj/xhLfI7b8AZF8wqRvrG3UjLViCNIpnUGfcgu5s6ePerc5zpHaIwS8st8/Oomp+LKYYxNQrDQ97cwsdbBnCdGEKWIduoUZdt0Sd5LlQ0CIFBCEyySXU75aEhsj8/SOfrv4a2VigwId28GunaW0ZMExGOYY/MgePdvPPVOT4+0YMkwZWzsvnG/Dwun2GI2asBb7ymqz2DLz45y9nT3rWw8wvTmFWczsw5uhGDw8lECAxCYJLNRLDTZDJxyuWCwx/hefsP8NWnoM9Guv52pJvuQDKOblnUk10D7G7qoLbpHOf6hjHptdw8P4+qebF7Ncp1dPd4aGv1dnl3dXiQJO/Ey1lFOqbPSkeXPn5NKCEwCIFJNhPBzkAb5abP8bzzRzi4H9K0SMtvQLr5m0gzop+G4MvgsMxfjnfxzlfnOHTSjUaCK2YYuGV+9LGaUNex89wwx1sHaGsdxN3jQaMB8wwds4p0mGfqkr5utxAYhMAkm4lgZ9jE5CePe7u4970HgwOw5Co0N38TFlw66mCr4tXscXTQ3jtEfpaWm0pzuXleLtNH6IGKlNv43NlhjrcM0HZskP4+mTQtWGbqmFXkHTmcjERZQmAQApNsJoKdEfPBdJ7zTkHY+yZ0d0LxfG+cZuk1owoIw4VYze4j5/jbiR48Mlgteqrm5bK8KIf0gMTl0V5H2SNz5vQQx1sGOfH1IIMDMlodTJ+lY+acdKZZtAnLXyMEBiEwyWYi2Bn1yzvQj9z4PvLubXDyOOSbkG68A2nFN5AMo887fdo9yHuODmqbOnB2D2JI13DdXCM3z8ujtCAzJht98XhkTjuHaGsd5MTxAYYGQZcuMWOWjhlFOkzm+M6JEgKDEJhkMxHsjNVG2ePxBoTf/RN8YYeMTKSKG5FuWoVkiX4WdyCe8+Nqdh85R+OxbgY9MqX5GVTNy+OupXMZ6I5u7aZQDA/LnDo5RNuxAU4eH2R4CNIzJK9nU6SjcNrYxUYIDEJgks1EsHNMeYNbHd68NAfqYHgYrMvQ3HQnLLSOaVBcV/8wdUc7qW06h6O9n/Q0iatnZ3PTvDysFv2oursVhodkXCcHOXFskJNtF8RmxmwdM+aMXmwmpcA4nU62bt2K1WqNKmueEJjkMhHsjIeNcke7d22nD972jhCePRfppju9s7l1sU8f8MVxto8P2/p5+zMn3QMeCvVabizJ5aZ5uaOamuDL0JDMqZODtLUO4mwbZHh49GKT0gLjcDj81qBubGxEr9fT3NzM6tWrwx7ndDrJzs7GYDBEdR4hMMllItgZTxvlwQHk/R8g126H4y2QbUSqvBXp+tuQ8gvHZGOb08Vfvu7mPUcHH58PDC+alsVN83K5pshIlm5sA+6GhmRcJ7yejfNEQDNqjnde1Ehik7ICY7PZ2Lp1Kxs3bgS8YuNyuSgvL6e2tpbS0lI/8fGlp6cHp9OJy+WirKwsotAIgUkuE8HORNgoyzJ8bsOzZwfYDoBGg3RFBdJNd0LpxTE3nwJtPOMe5P3mTvY0ddDWNUCmVqKiKIcbS3O51KxHM8Y5S6pnc+y8ZzPkEyCecz5AnBb7jO+4rOwYK1arlW3btqnfGxoasFq9y1KYzWbsdjulpaU0Njb6HVdWVkZTU5O67759+1ImubhgaiNJElyyhLRLliCfOon83i7kD3cjH6iHuRch3bQSaem1SLrY5ykBFOp13HNpId9aVMDnp3vZ09TBn1u6eM/Ridmg48ZSIzeUjDy2ZiS0WokZs9OZMTv9Qszm60Hajg3Q2jyATidhmaVlxuz4jLNJqMAE4na7/Zab7erqAgi5vrXFYsHhcOBwOMKm7BQIxhNp2nSk+/8eefW3kfe9j/zeDuR/fw759V95m0/X3YqUVzC6uiWJS6bpuWSanoeutLDvWBfvOTp4zX6G39nPcKk5ixtLc6koykGvG90EyTRfsTnfG3Xi6wGcx70zv7VauOamHEymUVUPJFlgYsFisQCEbUIJBKmClJmFdMPtyNfdCp8dwrNnB/Ku15Dfet07aO/GlaNqPilkaDVcX5LL9SW5nOoZZG9zB+85Onm+8SQvHXCyfI63CVU2hl6otDRvTGb6LB2eYZnTriGcbYNkG8cW/0mqwOj1ejVRuNvtJicnZ9R11dbWUltbC8CGDRswRSGzWq02qv3Gk4lgI0wMO8fFRvPNcN3NDJ34mt63/kDvnp14/lKHtvRi9HfcQ+a1VUjpGaO20WSCS4pn8PB1Mp+c7OLNT12899UpPjjaiTk7nVsWmrntEjNzC/Rj+xkWWFQ2Oht9SXgv0k9+8hN++MMfAqhNnqqqKrZt20ZZWVncPBQR5E0uE8HOVLBR7utFbtyL/P4ub9qI7Bxvyojrb0MqNMfFxoFhD3/5upv3HR3q9ISLCjO5oSSXFcU5GGNMZh5IygZ5GxsbcTgcNDY2Ul5eTmlpKQ6HA5vNhsFgEM0fwaRHysxCuv42b/PpCzue93d513t65w1Ysoz+1Q8gzyoZ0+C99DQN1xYbubbYyLneIT442sn7zR1s+auT//ibk6Uzs7mhJJcrZxnQpSU3x0zKDrSLFeHBJJeJYGeq2iifOYVc9zZy/bvewXvTZyFdfzvS8huR9NGN+4qGo+19vN/cyQfNHbT3DZOd7hWiG0pyudiUGbWopew4mGQiBCa5TAQ7U91GeXCQ7C8O0bn9d9D8pXfu09XXI91wO9LsuXE7z7BH5tDJHt5v7qTxWBcDwzIzcnTewPFcY8Qu75RtIgkEgvBIOh1Z199Kz+IrkY9+5Z2SsO895Lq34aJFSDfcgXR5OZJ2dGNqFNI0ElfMzOaKmdm4B4fZ19rF+82d/M52mt/aTnPJtCyuLzFybZGR7Iz45gQWHkyKMRFshIlh50S0Ue7uRP5wj3cFy1MnwZiHtOIW77iagvj2iJ3qGeSDZm+85uvOAbQaiWWzDFxfksvSmRfiNaKJhBCYZDMR7JzINsoeD3zyMZ73d8HhjwAJLrsKzfW3e2d0a+IXrJVlmaaz/ext7qCupZMOn3jNvYsLWVg0QzSRBILJhKTRQNlS0sqWeqck1L2D/OfdeD5uBPNM7yjha25CMox+LJl6LklifmEm8wszefAKMwdP9LD3aCd1Rzt5wDo2r0kIjECQ4kjTpiN9678ir3oA+a8fIn/wFvLr/4H8pxqkq1YgXXc7UslFcTlXmkZi6axsls7KZnDYM+ZubSEwAsEEQdKlIy2/AZbf4E2I9cHbyPv3In+4x5tP+Lpbka66DikjI3JlURCPMTOpsbKTQCCICamoFM3aR9D8n18jffu7MDiA/Jv/i+efvoPndy8jnzg23iYCwoMRCCY0UpYe6YY7kK+/Hb76xLs+9963kPfsgIvLvF5NHLq6R4sQGIFgEiBJEixYjLRgsXc5lg9rvYHhLf8HOScX6doq7yoJ06Yn1S4hMALBJEMy5iHddg/yN+72dnV/8Bby228gv/1HuPQKNNfdCmVXIqXFd1BdKITACASTFL+u7rOnkOt3I9e/i+eFZ7xrP624Benam8eUUzgSQmAEgimAVDANafW3ke+4D2wH8HzwNvL2V5F3/g6sV6G57huw6PK4DuADITACwZRC0mrhiuWkXbEc2XXCG6dp2IPnYCMUmpEqv4F0TRVSbn5czicERiCYokjmGUj3fAd59Rrkg43ecTVvvIK8/VW47Go0lbfCQuuYziEERiCY4kg6HdKyFbBsBfLJr897Ne/h+agBzQ+fA7N51HULgREIBCrS9NlI9/098l1r4dODSEXzxlSfEBiBQBCEpEuHJVeNuR4xVUAgECQMITACgSBhCIERCAQJY9JktBMIBKnHlPJgnnzyyfE2ISITwUaYGHYKG+PDWGycUgIjEAiSixAYgUCQMKaUwFRVVY23CRGZCDbCxLBT2BgfxmKjCPIKBIKEMaU8GIFAkFyEwAgEgoQhBEYgECQMITACgSBhCIERCAQJQwiMQCBIGEJgBAJBwhACIxAIEoYQGIFAkDAmTcrMtra2iPuYTCZOnz6dBGtGz0SwESaGncLG+BCNjTNnzgy5XXgwAoEgYQiBEQgECUMIjEAgSBhCYAQCQcIQAiMQCBLGpOlFihXP239EPlCPtGwFmlvvHm9zBIJJyZT1YOQD9dDa5P0UCAQJYcp6MNKyFcjnPwUCQWKYMgLjefuPnPl4H57Ll6O59W5vs0g0jQSChJLUJtK2bdvU/9fW1lJbW0tNTU3IfZXttbW1cTm3fKCeIccXokkkECSRpAmMzWajqalJ/X9ZWRlVVVW4XC5sNlvQ/nv27GH9+vWYzea4nF9atgJt6cWiSSQQJJFxaSK5XC5cLhcWiwWz2YzL5Qra57HHHsNqtcbtnJpb76awel3Kz/sQCCYTSfFgHA6Hn1hUVVWpa600NzdTWloadExzczM2m82vWSUQCCYWSRGY7u7ukNsV4QklMKtXr8ZqtdLV1RWyCSUQCFKfhDeRAr0XX+x2O6tXrw7a3tjYCEB5eTk5OTkhm1BKkBhgw4YNmEymiLZotdqo9htPJoKNMDHsFDbGh7HYmHCBUeItAE6nE4fDQWlpKbW1taq42Gw2rFYrPT09GAwGzGYzFotFPSbU0pW+zSwgqtjKZMm9kQpMBDuFjfEhpfPBlJeXU15eTnd3N263G/AKytatW1m/fj0PPviguu+Pf/xjAEpLS9m3bx+NjY1YLJaQTahYeeOTMzz46se88cmZMdclEAiiY9KsTR0po93jbzXTdLafeQUZbLqtJElWxc5E+IsGE8NOYWN8GIsHM2VG8q4oMpKW5qZiln68TREIpgxTRmDuurSQh667OOX/WggEk4kpO5taIBAkninjwRz5vI+GtmOYZ0rMX5g53uYIBFOCKePBtLUOcOZUP22tA+NtikAQlu9///txrW/z5s3q/3fu3EldXZ3ftkQzZQRmZlE6hdMymFmUPt6mCARhsVqt7Ny5My511dXVcfDgQcA7qBWgsrISo9Gofk80U0Zg5i/MZNV9c0TzSJDS3HnnnWzfvj3u9W7fvh2j0QhAUVER9fXJSVsyZQRGIJgI5Obm0tnZOeZ67HY7lZWV6veOjg7y8/PV7+3t7WM+RzRMmSBvIG98cob61k5WFBm569LC8TZHIPCjrq7OTyAAWlpawnoe1dXVft+TJSCRmLICU9/aSdPZfqBTCIwgZdi8eTMbN27khRdeCBKY4uJiiouLI9YR6L2A1zNSRKezs9PPm0kkU1ZgVhQZgc7znwLB+PPMM89QXV1NcXExra2tQeUtLS3s2rUr5LGPPPKI334tLS0AtLa2YrfbWbVqFYcOHVK3rViRnMyOU1Zg7rq0UHgugpShrq6OJUuWqB7KypUrg5pJxcXFfkISjpUrVwLevNZKPKesrIxDhw5RV1eH0WikrKwsAb8imKROdty2bZuaoqGxsRG9Xk9zc3PYnDAjlQcSabIjTJ6JZanARLBT2BgfUjpdg4Jv0m+HwwF4+/wNBoP6XSFSuUAgmBiMSzd1Q0MDer13VrPZbA4a9BOpPB4c+byPunc7OfJ5X9zrFggEXpISg1HSZiopLt1uN9nZ2Wp5V1eX3/6RykNxzz33+H1fuXIl3/nOd+jt7WXt2rUA6HQ6Bpxt0NnBkuu/y+XLHqCzy8mT//y/gupbu3Ytq1ev5vjx43zve98LKl+3bh233HILR44c4cknnwwqf/TRR6msrOTw4cM8/fTTQeVPPPEEy5Yt48CBA2zcuFHdrtPpGBwc5Omnn2bx4sXU1dXxy1/+Muj4DRs2MH/+fN599122bNkSVP6LX/yCWbNmsW3bNl555ZWg8i1btlBQUMBrr73G66+/HlT+yiuvkJWVxa9//euQI0v37t0LwIsvvhi0dlVmZqa6rtVzzz3Hhx9+6Feen5/Pyy+/DMCzzz7LRx995Fc+Y8YMnn/+eQB+9KMf8emnn/qVl5aW8vOf/xzwDq0P9HAXLVqkJi9bv349J06c8CtfunQpP/jBDwB46KGHgrp0r7nmGh577DHA2/3b1+f/R6iqqoqHH34YCH7uIPSz58u9997L/fffz9mzZ3nggQcYHBz0Kx+vZ08h8NlTnkmFUM9eQ0NDUD2QJIEJl/R7LATm5NXpdH7l2dnZmEwm3G63WiZJElJXJ3J/L/ldDgqnZZBvyg06FsBoNGIymejt7R2x/MyZMyHLc3NzMZlM5OfnhyzPy8vDZDKRl5fnVy5JEjqdjvz8fEwmE7m5oe1Tyo1GY8jygoKCqMpzcnJClhcWFqLX68nOzg5ZruRp1ev1QeU6nU7N4WowGILK09PT1fKsrKyg8oyMjKjLMzIygsqzsrIwmUxotdoRyxVbAssNBoNartPpGB4e9ivX6/V+5YGEevZ8ycnJUY9X7rcv4/XsKQQ+e4E2Rnr2fEl4kFfJwQuwadMmHn/8cWpqarBarVitVhobG3E6nX6B3EjloYg2yOuq2YJ8oB5p2Qrv8rEpxkQI+sHEsFPYGB9SOqNdqKTfFRUVqlvrdDrVLjMl6Xe48ngg1qQWCJLHuCT9Vjwam82GwWBQv/sm/Q5VLhAIJhZTJuk3TB53NBWYCHYKG+PDhBgHIxAIph5CYAQCQcKYcnOR/rT3LD2nPBimafjm9QXjbY5AMKmZch5MzykPRo+WnlOe8TZFIJj0TDmBMUzT0KkZwjBtyv10gSDpTLkmkmgWCQTJQ/wZFwgECWPKeTDhOPJ5H22tA8wsShcrDwgEcUJ4MOdpax2go90jFmYTCOKI8GDO412QbUAszCYQxBEhMOeZvzBTNI0Egjgz5QVGxF4EgsSRlBiMzWbDZrOpWc4cDgf33Xcf69evZ/369SEzsin7BmZLizcjxV5EWk2BYGwk3INxOBzYbDaqq6vZtm0bDoeD7u5ufv/736vlBoMh6Lg9e/awf/9+HnrooYTaN1LsRREfGBDejUAwChIuMKWlpWo+F5fLFZTbxeVyUV5eHnTcY489htVqTbR5IWMvSrMpI1MiN1+jio9oTglSjVR/JpMWg9m2bVuQN2Kz2cJmq2tublY/o1kXKZ4onktuvobKW4xB2xWPZjxurq/49ffJ6qdiQ7jycJ8j2R74+3y/m64NtiWSDYHniub6xXoO3/1M18b2m8Ld03A2RLpHkc6RkSkxPNRLmnZo1PdIeSb7eqO7TrE+J+GuY7TELDButxuHw6GmtywtLVWXGBmJ1atXs2nTJubNm6c2iWw2W1gvRREVJX4TuF9g0m8lifJIKImqI3HRwjSaj3RTMj8bkykf+9/aaT7STXZOJlrtsLq9oe0YHe0etFqZ8muD61WOy8xKo693OOJnyfxspk8PbaNSl7tniF63h7Q0GB5G/RzsH8TVJoctD/epHOdrQ9kV3nWLA3+f73etVourTaaj3RO1DUq5co7A+pXfGMoGpY5AW8L9LsXGwGsZ7jcF2u5ra7jfGe4+BtahfA9t+3DEe6TVyhj0aSGfJeWZdPcM0dE+HPE6xfqchLuO0RJ1Rrvm5mYaGhrIycnBbDZjMBjo6enB5XLR3d1NRUUFc+fODTpOya1bWlpKTU0NOTk5qnj85Cc/4Yc//GHQMY2NjYA33ea2bdswGAxUVVWNaF8iM9rVvdvp59FEq/79fTJ9vTKaNPAME/EzN1/D3d8uDWmjYkNGJmRmacKeK1x5tDYqxyvNwnCeSPm1s2n889d+5ZFsiLV8JBui+Rwe0kb0DsKdW/muNJFD/c5w9zGwjpHuSTQ2KufvaPeEfGZG80xG+5wo9zqhSb/tdjs5OTmsWbMm7D7Nzc0cPnyYxYsXBx1bUlICeL2f+fPnA95k3oHBXcUrMpvNWCwWdb9I4pJoAgPBgU0o5eUP9aDl5kf/smdkSmz//THMMyX1PL5lysMeyl0ebXMt8MHs75PVZmDlLUbmL8wMEliFwPhVJBuUcuUcgddPuV6hbFAIZ0sg3v36w76Q4f5ohGvWhPqdkZo0gXWEsj22P3qhxT7UvQh3nZLdrI9KYObNmxexGVRSUqIm9falqqqKffv2+XklCr6Lq4E36ffGjRspLS2ltraW7OxsLBbLuCf9Drx5gYKjfI+l7RwK5aUYGvKOHvAVrUgv1GgHCo4kEgrRjnKOZINSHngO3/pDlfsSrS0zi9LRauUg7yDUOUPZPtJvifZaR3puYiHW+xvuXMkeUDrmpN8ulwuz2Rwve0bNZEj6feTzPlxtckgPJtV6CVL9WoKwMV4kfV0kt9tNQ0MDtbW1WCwWdZlNwdiYvzCT8msv3MxUEhRfjnzeR0ObtymXqjYKUoOYBObw4cO8++67HD16lKqqKn70ox9F1YMkmFwoMaihIY0QGMGIRCUwtbW1bNu2jeXLl1NdXU1PTw89PT1CXKYoSnxDacoJBOGI2oN5/vnn/b7b7Xb27NnDTTfdFHejBKlNYFNOIAhHVJMdQ3UTl5WVYTabefXVV+NulEAwVfn+978fl3rq6uqoq6vjmWeeUbcp/1cmEieDMc2mLisrG/cxKgLBZMJqtbJz584x1WG326mvr6eyshK73Y7dbgdg69atVFRUUFRUFA9To2LMc5FSoYtaIJgs3HnnnfzTP/0TK1euHHUdZWVl6hy/1tZW9f8vvvgilZWVcbEzWqLyYA4fPhxVZdHuJxAIQpObm0tnZ2dc6tq8eTMbNmxQvx8+fJi6ujo2b94cl/qjISoPZvHixWzfvh2LxcLVV18dVN7Y2EhPT48I+AoEcaKuri7I22hpaaG+vj7k/tXV1UHbHnnkEdatW8eSJUvIzc3lkUceAaC+vj5k/Ykg6ibSqlWrcLlcbN26Ve2mzs7OxmAwhJ3oOJF445Mz1Ld2sqLIyF2XFo63OYIpyubNm9m4cSMvvPBCkAAUFxdTXFwcsQ4l5lJWVkZxcTFbt25V4y4rV64kPz+f1tbW+BsfgphiMGazecQJjxOZ+tZOms72A51CYATjwjPPPEN1dTXFxcUhBaClpYVdu3aFPFbxTsDroSiTjjs6OliyZAnFxcWqyBw9epS1a9cm4BcEM+WTfiueS35mGvMKMlhRFH5CoUCQKOrq6lQhAK+nEdiMKS4u9hOScKxZs4YdO3aovVFKwLimpoa8vDzmzp0bNtFbvBnzZMdosNls6qfSVqypqaG6upra2tqQXd2NjY3o9fqoM9qNdrLj428103S2n3kFGWy6rSSan5NQJsLkN5gYdgob48NYJjsmfFUBJem31WqlublZTUC1Z88e1q9fH7KbW9nHarViMBjU74lgRZFReC4CQYIYUxMpmlQN4ZJ+j5TUu6GhQS0zm83Y7faE5YS569JCEXMRCBJEzALz6quvsnz5cmprazEYDFgslqi6pwOTfo+U1Nvtdvslo+rq6opY/z333OP3feXKlXznO9+ht7dXDWjpdDoGBwcBuPfee7n//vs5e/Ys69atC6pv7dq1rF69muPHj/O9730vqHzdunXccsstHDlyhCeffDKo/NFHH6WyspLDhw/z9NNPB5U/8cQTLFu2jAMHDrBx40Z1u2Lj008/zeLFi6mrq+OXv/xl0PEbNmxg/vz5vPvuuyHXlfrFL37BrFmz2LZtG6+88kpQ+ZYtWygoKOC1117j9ddfDyp/5ZVXyMrK4te//nXIkaV79+4FvIO3AteuyszMVIejP/fcc3z44Yd+5fn5+bz88ssAPPvss3z00Ud+5TNmzFDnvv3oRz/i008/9SsvLS3l5z//OeAdWh/o4S5atIgf//jHAKxfv54TJ074lS9dupQf/OAHADz00EO0t7f7lV9zzTVqCpLq6mr6+vzXxaqqquLhhx8Ggp87CP3s+eL77D3wwAPqM6kwXs+eQuCz5/veQOhnr6GhIageGIXAVFVVYTabaWpqYsOGDWqXWCQCk35HSuodicCk3zqdzq88Ozsbk8mE2+1WyyRJUv+fk5OjJjIOPBbAaDRiMpno7e0dsfzMmTMhy3NzczGZTOTn54csz8vLw2QykZeX51eu2Jifn4/JZCI3Nzfk8Uq50WgMWV5QUBBVeU5OTsjywsJC9Ho92dnZIcuVRNB6vT6oXKfTqdfWYDAElaenp6vlWVlZQeUZGRlRl2dkZASVZ2VlYTKZ0Gq1I5YrtgSWGwwGv2djeHjYr1yv14/47IR69nzxffZ8n0mF8Xr2FAKfvUAbIz17vsQc5G1sbESSJJxOJ6tWrWL//v0hB98phEr6reTbDZfUu6amBqvVitVqpbGxEafTGTHQOxky2sHEsBEmhp3CxviQ1CCvxWLhyJEjrFq1itraWpxO54j72+12uru7AW/Tx2KxYDab1W4yp9Opxld6enoAqKiowOVyqeXJ6lITCATxJWaBKSkpUQfbWa1WVq1aNeL+VVVVuFwuv6TfpaWlaiJw36TeSrtZ+W6z2dS1lwQCwcQj5ibSq6++SkVFBbt3744pyJtoRBMpuUwEO4WN8SGpTaSqqirmzp1LU1MT3/72t0W6BoFAEJaYe5EcDgfNzc1UVFQAhFwLSSAQCGCUQd6mpqaog7wCgWDqErMHYzAYOHnyJHv27KGiokIIjEAgCEvMAmO323n88cdxuVzo9Xq1a1kgEAgCiVlglEFyCpNZYEQSKoFgbIwqyOtwODAYDImwJ6UQSagEgrERc5B31apV6PV6mpqayM7OZvny5YmwKyUQqRwEgrERswejDPefPn06siyzdetWv1nSkwmRykEgGBsxC0xNTQ3Tp09Xv0/GXiQRexEI4kPMArN8+XK/yYeTcWXHcLEXITwCQWzEHIORJAmXy6WO4N23b1/EY5ScL75r4ir5XMKtk6tsD0xmlAzCxV4U4alvjc/CWALBZCdmD+all15S4y/gTYM50mRHJSdvdXU127Ztw+Fw0N3dTVlZGRaLhU2bNoVMOLVnzx7279+f8PiO5+0/Ih+oR1q2As2tdwPhYy9ewelUhUd4NALByMQsMI8//jglJRey7yupL8MRKidvbW0tLpdLzQ2j5H7xZaScvfFEPlAPrU3IAOcFJhBfIfFdeSCwKTXZBCdRv8d3qZj2vmH1UzlPqPOGs2Wk7Q3Hj2HUyX51x+O3j/a6xON6RqojsDzS90QTs8D4ikuo7+HwzcnrG7fxnTjpy0g5e+OJtGwF8vnPcISLyQR6NJEEJ9x33xetc/AYFbP0ER/qSC9p4HHhXuJwx0X6PQ9dZwprQ7hP5RxKvelpMDCM+tnee5b61k7O9Q5xpncY6FTt8N3ma6dSl3Ksci5lf6XuwN8QydZQ51OOVcoCzxnp3oQ6rnPwggiOdN0Cr4NSR+A5A+0OvIeB1yuW52E0xJwPxuVysWnTJpxOJ9OnT+exxx6LOmXDpk2b+O53v6sO0nM4HNjt9hEFxDd9pi+BOXkHBgYinl+r1TI0NBSVrb5s/evX1H55iqoF01hz5eyg74H7FerTOeMe4Kx7kNM9A5j0OgoM6er3BdMM/Orbl/Pgqx/z5akeMtIk+odl9VMp90XZN7CuSMcGniPc8cpxvr8NCPl7Fkwz8MrfLWPtbw6EtD/cZ+A5lHrDXa+qBdOo/fKU3+9W9g28F4G/x6TXUZidQUGWLuQ5Itnqa4NyjyPdg0j3JlL5SNcNCHnucN8Dr2246xXpeYj2vUlPTw+5PWaB2bp1KzfffDNms5menh727NkzYla7UDl5FUHZtm1bSHHxzX4XKmdvKJKZcCrSYm1KeUFWGvlZWvWvivJ9ZA9GomKWHmDEv1BKXZGaGeH+ioY7Xin3/W2Bv0exM9Jf3kieTCDRNI0CbRnpL/BD112s3u9QvyFWGyN5kaO5N9FcR18PJtLzEHidwj2jsTxPvtcxHOESTsXcRKqoqFA9FoPBEDFfrt1uV5tRbreb+fPnA14PxHdlAavVSk9PDwaDAbPZrM55cjqdKdcVHtg0Gql8pHZvqGCyIoLKA+Lrym66rSSqNvjjbzWHFAqlPNTxvi9gYA9a4O8Z7WqYynHhpl6Euh6B2xRbzvUO+dUVaVBk4G8YDYHnCPwebXzD97hY/uiFOne4c0bqkIh0vO+9Gks3S8wezOHDh9WFtQF1VYFwqwv09PSwb98+srOzsdlsrFu3DpvNxnPPPUd2djbd3d1qQPeJJ55Q12mpra0lOzs7qhUFYPKlzAz8CxPtyxzuL3wkYgn+eQOo7pCxonidIx51xet+JzIwmqxnMtY/Cv7xttF7MDELzIMPPsj06dPVhdGUwx0OB//xH/8RS1VxZTwFJp4PYKCNsdadrF6CiSTWYyWR65cn6zqO5bkYS07emJtIjzzyCMuWLQvaHqm7ejKTyFnXsc6HEvOn4k+kJvFEYLyei5gF5s9//nNIgYm2u3oyMhkeQEF4hGiPnlGtKnD06FH1+549e+Jpz4TkrksL2XRbiXgIBYIAYvZgtmzZos6mlmUZh8OREusiCQSC1CNmgVm3bp1f1/RUjr0IBPHm+9//Pj//+c/HXE9dXR0A9fX1PPXUUwDs3LkTo9HI4cOHeeSRR8Z8jmiIuYnkKy4ul2tS5+RNFMpYlTc+OTPepghSDKvVys6dO8dUh91up76+nsrKSux2u/oPoLKyEqPRqH5PNDF7MC6Xi927d+N2u5FlmZ6eHr9xMYLIiFy/gnDceeed/NM//RMrV64cdR1lZWWqI9Da2kpZWRnPPPMMK1Z459sVFRVRX18fcZBsPIhZYHbv3k1FRQVNTU3q6FtBbIheJ0E4cnNz6eyMT76hzZs3s2HDBgA6OjrIz89Xy9rb2+NyjkiMaqpASUkJ3d3dmM1mvx4lQXSIbk9BJOrq6qisrPTb1tLSQn19fcj9q6urg7Y98sgjrFu3jiVLliTExmiIWWC6u7t57rnnWLNmDVu2bEGSpEmb9FsgSDabN29m48aNvPDCC0ECU1xcTHFxccQ6lPhKWVkZxcXFbN26ldzcXNVr6ezs9PNmEknMAuPbvlu+fLk6ZUAgEIyNZ555hurqaoqLi2ltbQ0qb2lpYdeuXSGP9e0Vqq+vV+OiHR0dLFmyhBUrVnDo0CHAG5dR4jGJJmaB8aWsrEzNzSsQCEZPXV0dS5YsUT2UlStXBjWTiouLo+peXrNmDTt27FB7o5SA8aFDh6irq8NoNCYlwAujmOzodrvVHC/gTfodqYlks9nUT6Wt2NjYiF6vD5uxLlJ5IJNtNnWqMxHsFDbGh7FMdox5HExNTY26fKzD4Yi4LpKS9NtqtdLc3KweB94+f4PB4CdYyjEjlQsEgolBwtdFCpX0W0mDCWA2m7Hb7eo+AA0NDSOWCwSCiUHMAqOsi5SdnY1er2ffvn1RzUXyTfrtdrv9gsNdXV1++0YqD8U999zj933lypV85zvfobe3l7Vr1wKg0+kYHBwE4N577+X+++/n7NmzrFu3DvnsKejsAGMuUsE01q5dy+rVqzl+/Djf+973gs63bt06brnlFo4cOcKTTz4ZVP7oo49SWVnJ4cOHefrpp4PKn3jiCZYtW8aBAwfUJFu+Nj799NMsXryYuro6fvnLXwYdv2HDBubPn8+7777Lli1bgsp/8YtfMGvWLLZt28Yrr7wSVL5lyxYKCgp47bXXeP3114PKX3nlFbKysvj1r38dcmTp3r17AXjxxReD1q7KzMxU17V67rnn+PDDD/3K8/PzefnllwF49tln+eijj/zKZ8yYwfPPPw/Aj370Iz799FO/8tLSUnU4/fe///0gD3fRokX8+Mc/BmD9+vWcOHHCr3zp0qX84Ac/AOChhx4KGhNyzTXX8NhjjwHe7t++vj6/8qqqKh5++GEg+LmD0M+eL77P3gMPPKA+kwrj9ewpBD57vu8NhH72GhoaguqBKAVm/fr1fhMcwSs0sixHXBdJYfXq1WzatIl58+ZFc8qIBCb91ul0fuXZ2dmYTCbcbrdaJkmS+v+cnBxMJhNw/qXu6kTu70XqktBZZmI0GjGZTPT29gbVDajlZ86cCVmem5uLyWQiPz8/ZHleXh4mk4m8vDy/csXG/Px8TCYTubm5IY9Xyo1GY8jygoKCqMpzcnJClhcWFqLX68nOzg5ZrtVqMZlM6PX6oHKdTqdeW4PBEFSenp6ulmdlZQWVZ2RkRF2ekZERVJ6VlYXJZEKr1Y5YrtgSWG4wGPyejeHhYb9yvV7vVx5IqGfPF99nz/eZVBivZ08h8NkLtDHSs+dLVEHe5ubmsPleRiqD0Em/u7q61JUCGhsbg9Ji+q4kEKo8FGMN8oZagG08mAhBP5gYdgob40PCM9qNJCCREk2FSvpdVlamCo/T6VRjOkrS74qKipDliURz691hF14TCASjI6pepNraWl599VV1WsCePXv46U9/yssvvxxxHExVVRUul8tvKRIlYGuz2TAYDOp3pd0crlwgEEwsomoi2e12dZlXu93Oq6++yrPPPgvA9u3bR1wXKVmIcTDJZSLYKWyMDwkfB+N2u9W1kBobG/2CutGu6igQCKYeUQlMd3e3+n+73e63jKskSfG3SiAQTAqiCvLKssz27ds5cuQIc+fOVb2WZGXFEggEE5OoBKaqqkrtjlZ6dJqbm+np6REejEAgCEvUI3kDu6NLSkqm9FpIAoEgMmNK1yAQTBZkWaavrw+Px5NUr9zpdNLf35+0840GxUZZltFoNGRmZkZ9jYTACARAX18fOp0OrTa5r4RWqyUtLS2p54wVXxuHhobo6+sjKysrqmNjTtcgEExGPB5P0sVlIqLVavF4PFHvLwRGIGDqDbdoaWkZ9bGxXCsh2QLBJERJl9HS0qKu7KjgO7wk0StACg9GIJhk1NXVsWLFCqqrq2lpaVFFROHQoUOUlZUlZQVIITACwSSjtbVVXT8p3AoF4E3ar3gtygqQ27dvx2j0LgiorAA5FpLSRFISQ508eZLq6mocDgdPPvkkFosF8P7QdevW+R1TU1NDdXU1tbW1EdNyCgQTHcWbUJYbGUvzxHcRNrvd7jcZuaWlJWghtkSuAJlwgbHZbJSVlWGxWNi0aZO6wsDvf/97wJuQymAwBB23Z88e9u/fLxZ1E6Qs8UxSZjQaaW9vJz8/n7KyMjWGohDrqo7gFZfKykq/fEr19fVB+ydyBciEC4zL5cLlcqnpHlwul59H4nK5KC8vDzruscce85tUKRCkGvKBemhtQoYxJysrLi6mpaUlbHK1aFd19KW+vn5ELygZK0AmXGB8xaS5uZmKigr1u+LdhKK5uVn9DJUuMzAnr5LjdCSUPLKpzESwESaGnbHY6HQ6Yx8Hc/V1DEsSaVdVjmkMjXKsRqNBq9Vis9m44oor/Oo8evRoyOTrAP/4j/8YtO03v/kNjz76KAAffPAB1113HUePHqWkpESt98MPP6SsrAytVktXVxdXXHEF119/PQcPHkSr1fL1119z/fXX+9kI/jmRIxHzwmujxeFwYLfbg3LvhnPvfPdR8vOOhEg4lVwmgp2x2Oh2u9Hr9Qm2KBitVsvQ0JDaBKqurmbz5s2sWLFi1Kli6+rqePjhh8nLy+PcuXO8+OKLVFZWBr1vHR0d7Nixg7y8POrq6tSVGmpqaigqKqK1tZXq6mrVRoVQ12pMOXnjQaC4wAUvJRDf9Jo5OTm4XK6E2ycQjCe+TaCDBw+OafxJZWVl0FIvocjNzVUFR1leFsLHdEZDUrqpa2trVXFRgrxOpzMouNvT0wN4s+Qp6u10OkVOXsGUIi8vL+51dnR0JKTeSCRcYGw2G1u3bmX9+vU8+OCDfmW+i6uBf9Lvffv20djYiMViEQIjmFIoTZV4cujQIT8vJVkkLQaTaEQMJrlMBDsnUgwmlRlLDEaM5BUIBAlDCIxAIEgYQmAEAkHCEAIjEExBxpIPJhaEwAgEk5Camhpqamp45plngsrsdjudnZ2Ad1BeXV2d337K/wPnQ40GITACwSRjLPlgALZu3UpFRQVFRUVjtkVktBMIJhmtra20traqo4NHygejDGhV8sEA6tSCeCAERiBIAVIlH4xy7rHaoCAERiAYJW98cob61k5WFBm569LCMdWVKvlgcnNzVVGpr6+nrq6OG2+8cdS/SwiMQDBK6ls7aTrbD3SOWWBSJR+MEndZuXIl+fn5YZtX0SIERiAYJSuKjEDn+c/4Ybfbueyyy/y2tbS0sGvXrpD7hxKRmpoadXtdXR2VlZW0tLT4BW59m2QdHR0sWbKE4uJidZ+jR4+ydu3aMf2WccnJC5Fz7jY2NqLX68MmnBIIxpu7Li0cs+ei0NLSogZW6+vrWbFihV95cXFx1PGQuro6fvazn7F582Y1H4xSr2/zaM2aNezYsUNNZKVMhqypqSEvL4+5c+eOOieNwrjk5LVarSPm3HU4HABYrVZcLhcOh0PMqBZMaiZrPphxyckLI+fcbWhoUMvMZjN2u10IjGDKMJnywYxbTt6Rcu663W6/XDFdXV2JNlMgSBkmUz6YpAV5HQ4HVqtV9UR8M9wpzaZYEEm/x5eJYGfCk37HiWScdyxdzTD6pN/jkpM3Us5dvV5Pd3c34PVmcnJyguqrqqry846iSSw02ZIkjScTwc5YbOzv7yctLS3BFgUzERNO9ff3B13XcU04FZiTN1zOXSUnb0VFhSo6TqdzzJFsgUAwPoxLTt5wOXd9c/IqxxoMBhHgFQjiTLLSNYicvCnGRLARJoadUzknrzLVoKWlRV3gXsF3BK/C5s2b1a7xnTt3YjQa1blIIievQCBQiTZdg+/+Bw8eBC6IT2VlJUajUf0+WoTACASTjNbWVnVi5EjpGkKxfft2jEbv1IeioqKwEyyjRcxFEghSgPFK16DMuFaaVB0dHX4L3re3t4/KBgUhMALBKDnyeR9trQPMLEpn/sLMMdU1XukaxiogkRACIxCMkrbWATraPcDAmAVmvNI1BGauy83NVUWns7PTz5sZDUJgBIJRMrMoHRg4/xk/kpWuoaWlRe2ubm1tVZtThw4dUrcFzuqOFSEwKYDn7T8iH6hHWrYCqteFLMOYD53t6qe0bAWaW+8OWx7uM9xxgdulZd4HK9y5fe2M1YZYbA1nQ6TfpdgYrW2e2+9HnjEL0tJgePjCp+H8nLiebr9tUm4+82b0Ms/o3S63DQeVK8gd7d7jA7b7lrd88QUrLrMitx2j7s1drFi2FNnZptZXlGfkv3/zTrUO3zr9zpGWRn3DPn7205+y+Ze/5FxnJy8++wyys426HdupvvdbyG3HIC2NO65YAmlp1Lz+BzrbzyKfOknZpZdy8Fw7dW+9idFoTP10DZOZuL1Ynefg3BnkjnbOfLwPz+XLgfMv1vkydOkwOKB+yh3tDI9QHu5TBrj1bm/drU1h61MHR/ns47uvYqcmRF3x+hzJhkjXQ7FRPns6uuvT2Q6FhSBpQPZc+FQY6LuwbXgIuacbhoe8/0Id4ysCyn7KcT4iNuQZhqEhis2FFN9xGwz0cchu55H7vwV9vcE2BJ77/Lno6VZtXHG5lU/e2eVvU18vDA9eqNPH5uqVt1J95+3qftUrb4U0LaRpvb+hcNqo3xEhMCPg+9cwpLcQ48sd9jO3AIrmQec5hhxfgDKoqbXpQlkoURqpfITP4Z885v3ue1xAfar3ACHPPeT4As6c8r7UgXXF24MJZUOk66HYGMv1Sc8M78HAhW3DQ96XOU0b+pi0NK+XoIiAsp9ynM/LLcueoHry8gNs8bUhxLkVjyScLQwP0+F2k1dQCJlZIcuDPpXzwJgERozkDYEqJIqA5BZ4/0oECoqyPQ4vkiJgaR/vY9jHg1HKwtkYrjwcwz95zPsiFs0j7YfPjao+xc4hxTsIqGs8CPXHQLmW0VyfWEbyRmzytB27IAJp2uBmjc/LLHmGkfWh64nm3Oq50jORZs4Je5wSh4kW3/PoCqeNeiSvEJgQqC9hoLAECEqsL3c0JHoI/miFKRCTyYSrZktc6koU4zVVIJIA+TLWqQKxnGu0jGWqgGgihUBatgL5/Gdg4DAVX6RY0Nx6N8TpN8SzrvEmnn9npdx87x+iJJDMcynEcq3GLel3qG2+REoKnkgCX5zJ9CIJQqPRaBgaGhq3pFMThaGhITSa6GcYjUvSbyBkInBfRkoKLhDEm8zMTPr6+ujv70eSpKSdNyMjg/7+/qSdbzQoNsqyjEajITMz+kGF45b0O9Q2X0ZKCi4QxBtJksjKykr6eSdb2otAxiXpt28CKd9E4L6MlBQcRE7e8WYi2ClsjA9jsTFpvUgOh8MvL2+4bYHU1NRgtVojejMi4VRymQh2ChvjQzQ2jnvCqVBCEk5cGhsb1cTgoZKCCwSCiUFSPBjfniAloBtqW09PDwaDAYfDgcViwWAwsGXLFqqqqkReXoFgAjIuSb9DbQP/pN+hkoKPlSeffDIu9SSSiWAjTAw7hY3xYSw2JjzIa7Va+dWvfhW0PdS2jRs3qv9P9tgXgUAQf0ROXoFAkDCmlMBMBK9oItgIE8NOYWN8GIuNk2ayo0AgSD2mlAcjGDvbtm1T/9/Y2IjNZvPbJpjYOBwOv++h7nEs933KCEyqvgzKiGTfLPKpaqvNZqOpqQm48CBarVZ1aMF443A4/MZQQepdS8UeZRS677bxttFms/HSSy+p30Pd41jv+5QQmFR8GeDCRNCqqipcLhc2my1lbQ2koaFBzQliNpvHvAJgPKitraW8vByn0zmqlyHROBwOzGYzVqsVs9mccjZarVaysy9kzwt1j2O971NCYFLxZQDvhE/FFmXSZ6ra6nA4/KZruN1uv4exq6trPMxSUcZMAaxevZrS0tKUvJZbt24FvPc+VW1UCHWPY73vU0JgUu1lUKiqqlIj9M3NzZSWlqasrd3d3eNtwogcOXKErq4uHA6H2tRItWtZWlqKxWJh/fr1ql2pZmO8mRICk+oo3kGqTocI9F4A9Hq9Kjput5ucnJzxMM2PnJwc9Rr6xmFShZ6eHvR6PVVVVbz00ks4nc7xNmlEQt3jWO/7lEjflYovgy++kz5T0VYlpw+gxjcqKirUeIHT6Rzz+jljZfr06aonYLFYOHLkSMpdy9raWu666y4MBgMlJSU0NjamnI2+hLvHsdz3KeHBVFRU+L0g4/0y+FJbW6uKi81mS0lby8vLKS8vp7u7G7fbDaB6CjabDYPBMO7eV1lZmeoROJ1O5s+fn5LXUkEJ6qaSjY2NjWpPHIS+x7He9ykz0K62tlYNpKbK6EmbzcZzzz1HdnY23d3daha/VLR1IlBbW0t2djZOp1MV7VS7ltu2bcNisdDd3a3ak2o2xpMpIzACgSD5TIkmkkAgGB+EwAgEgoQhBEYgECQMITACgSBhCIERRMTpdLJp0yaeeOIJdeG8ZLF+/fqknk8QX4TACCJisVioqKhg3rx56ohe39nA8SJUnc8//3zczyNIHkJgBDHT09PD7t27E16n0+lMySH/gugRAiOIGafTidvtVkd+Kmzbtk3NdeJ0OrHZbKxfvx6bzcamTZvo6ekBUNNS1NTU+I2+DawzOztbnX0MXg/H4XBgs9lU4XE4HOo5bDYbW7ZsoaenRx2R6nQ6/XLtCJKLEBhBzJSWlqLX6ykvL1eHiivNG6vVSlVVFVu3blXznmRnZ/P4449jMBjUfUtLS6moqFC9llB1GgwGzGYzgN/wdavVypEjR3A4HJSWlmI2m7FYLFitViwWC01NTTQ0NAAXmneC8UEIjCAuOBwODAYDTqcTp9PpN/s6cL7KmjVrVC9GmdsUCZvNpooNeCc3+npPvikPAKqrq3E6nTzxxBMJiRcJokMIjGBUKC+00qukCIrFYsFisbB8+fKQxympIa1WqzqxT2kmBdbpS2lpqd8Swor3oqB4RwrKJNKNGzei1+tTPjXCZEUIjCAiTqeThoYGmpqa1Jd/+fLlfp5BeXk5cCG/rG/aSt/9FBFxOBz09PTQ3d2tvvyBdTocDlwul7rMcE9Pjxp/UfLn+O7jW3727FnVlunTp6vZ7gTJRUx2FAgECUN4MAKBIGEIgREIBAlDCIxAIEgYQmAEAkHCEAIjEAgShhAYgUCQMITACASChCEERiAQJAwhMAKBIGH8/74jNsn+m1XRAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 288x360 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "f, (ax1, ax2) = plt.subplots(2, 1, sharex=True, figsize=(4, 5), gridspec_kw={'height_ratios': [0.6, 1]})\n",
    "for sharp in res:\n",
    "    loss_record, sharpness_array, hessian_eval_epochs = res[sharp]\n",
    "    ax1.plot(np.arange(len(loss_record)), np.array(loss_record), label=r'$\\eta=2/{}$'.format(sharp))\n",
    "    ax2.scatter(hessian_eval_epochs, sharpness_array, s=5, label=r'$\\eta=2/{}$'.format(sharp))\n",
    "    ax2.axhline(sharp, linestyle='--', color='black')\n",
    "    ax2.text(epochs * 0.95, sharp + 0.5, r'$\\lambda={}$'.format(sharp), ha='right')\n",
    "\n",
    "ax1.set_ylabel('Loss')\n",
    "ax1.set_yscale('log')\n",
    "ax1.legend(loc=1)\n",
    "\n",
    "\n",
    "ax2.legend(loc=4)\n",
    "ax2.set_ylabel('Sharpness ($\\lambda$)')\n",
    "ax2.set_xlabel('Iterations')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f\"./figs/IntroShallow_{exp_name}.pdf\", bbox_inches='tight', pad_inches=0)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.6.9"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "31f2aee4e71d21fbe5cf8b01ff0e069b9275f58929596ceb00d14d90e3e16cd6"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
