{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cpu\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from torch.nn import functional as F\n",
    "\n",
    "from matplotlib import pyplot as plt\n",
    "from tqdm import tqdm as tqdm\n",
    "import math\n",
    "from matplotlib.colors import ListedColormap\n",
    "import seaborn as sns\n",
    "\n",
    "%matplotlib inline\n",
    "#device = torch.device(\"mps\" if torch.backends.mps.is_built() else \"cpu\")\n",
    "device = \"cpu\"\n",
    "print(device)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## create dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAB7wAAAC7CAYAAAD2d+GQAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/MnkTPAAAACXBIWXMAAAsTAAALEwEAmpwYAAAPRElEQVR4nO3dX4ild33H8c93Z3ZrWlO2jttNdrPtuiClQdIImyDoRQiBrH+o3hQMtHhRyE0LCpZie1MsFO+kN96EGgxUFKnSBrFIahUrtLqrVet2TQ0hqUExf6zohqxhs99e7JSu6z4zJ+2cec5v5/WCYc555nD43Aw/dt57zqnuDgAAAAAAAACMZt/cAwAAAAAAAADg/0LwBgAAAAAAAGBIgjcAAAAAAAAAQxK8AQAAAAAAABiS4A0AAAAAAADAkARvAAAAAAAAAIa0vownffWr1vr4sf3LeGoABvWdRw/OPWHSpQNLOQ53xKUDNfeESZdW+Khf5W1ZvzT3gkm/sP/i3BMm/dL6i3NPmPTL+16Ye8KkG1f4v7f+xzd/ce4JAAAAAAzkJ/mvZ7v70NXXl/IX/uPH9ucrnz22jKcGYFBvfdM75p4w6cLxjbknTDp/9MDcEyY9f2R1Y/wLN61uVN53+MLcEyadOPzs3BMm3bHx5NwTJt1z49m5J0y664bV/V2498jtc08AAAAAYCD/0H9zzT8SrvBrPgAAAAAAAABgmuANAAAAAAAAwJAEbwAAAAAAAACGJHgDAAAAAAAAMCTBGwAAAAAAAIAhCd4AAAAAAAAADEnwBgAAAAAAAGBIgjcAAAAAAAAAQxK8AQAAAAAAABiS4A0AAAAAAADAkARvAAAAAAAAAIYkeAMAAAAAAAAwJMEbAAAAAAAAgCEJ3gAAAAAAAAAMaaHgXVWnqurRqnqsqt637FEAAAAAAAAAsJ1tg3dVrSX5UJI3J7k1yX1VdeuyhwEAAAAAAADAVhZ5hfedSR7r7se7+8UkH0/y9uXOAgAAAAAAAICtLRK8jyb57hX3n9q89jOq6v6qOlNVZ5557qWd2gcAAAAAAAAA17RI8K5rXOufu9D9QHef7O6ThzbW/v/LAAAAAAAAAGALiwTvp5Icu+L+LUm+t5w5AAAAAAAAALCYRYL36SSvrarXVNWBJO9M8vByZwEAAAAAAADA1ta3e0B3X6yqP0zy2SRrSR7s7rNLXwYAAAAAAAAAW9g2eCdJd38myWeWvAUAAAAAAAAAFrbIW5oDAAAAAAAAwMoRvAEAAAAAAAAYkuANAAAAAAAAwJAEbwAAAAAAAACGJHgDAAAAAAAAMCTBGwAAAAAAAIAhCd4AAAAAAAAADEnwBgAAAAAAAGBIgjcAAAAAAAAAQxK8AQAAAAAAABiS4A0AAAAAAADAkARvAAAAAAAAAIYkeAMAAAAAAAAwpPW5BwCwN1x8/Im5J0xaX+FtB+cesIWDcw9gTzmdtbknTDqd2+aeMOkDcw/YwvqJ43NPmHTh+MbcEyadP3pg7gmTnj9Sc0+Y9MJNl+aeMGnf4QtzT9jSicPPzj1h0h0bT849YdI9N56de8Kku25Y3d+He4/cPvcEAABgQF7hDQAAAAAAAMCQBG8AAAAAAAAAhiR4AwAAAAAAADAkwRsAAAAAAACAIQneAAAAAAAAAAxJ8AYAAAAAAABgSII3AAAAAAAAAEMSvAEAAAAAAAAYkuANAAAAAAAAwJAEbwAAAAAAAACGJHgDAAAAAAAAMCTBGwAAAAAAAIAhCd4AAAAAAAAADEnwBgAAAAAAAGBIgjcAAAAAAAAAQxK8AQAAAAAAABjStsG7qh6sqqer6lu7MQgAAAAAAAAAFrHIK7w/kuTUkncAAAAAAAAAwMuybfDu7i8m+eEubAEAAAAAAACAhe3YZ3hX1f1Vdaaqzjzz3Es79bQAAAAAAAAAcE07Fry7+4HuPtndJw9trO3U0wIAAAAAAADANe1Y8AYAAAAAAACA3SR4AwAAAAAAADCkbYN3VX0syT8n+Y2qeqqqfn/5swAAAAAAAABga+vbPaC779uNIQAAAAAAAADwcnhLcwAAAAAAAACGJHgDAAAAAAAAMCTBGwAAAAAAAIAhCd4AAAAAAAAADEnwBgAAAAAAAGBIgjcAAAAAAAAAQxK8AQAAAAAAABiS4A0AAAAAAADAkARvAAAAAAAAAIYkeAMAAAAAAAAwJMEbAAAAAAAAgCEJ3gAAAAAAAAAMSfAGAAAAAAAAYEjrcw8AAAD2nouPPzH3hEnrK7zt4NwDtnBw7gHsOaezNveESadz29wTJn1g7gFbWD9xfO4Jky4c35h7wqTzRw/MPWHS80dq7gmTXrjp0twTJu07fGHuCZNOHH527gmT7th4cu4Jk+658ezcEybddcPq/i7ce+T2uScAMAiv8AYAAAAAAABgSII3AAAAAAAAAEMSvAEAAAAAAAAYkuANAAAAAAAAwJAEbwAAAAAAAACGJHgDAAAAAAAAMCTBGwAAAAAAAIAhCd4AAAAAAAAADEnwBgAAAAAAAGBIgjcAAAAAAAAAQxK8AQAAAAAAABiS4A0AAAAAAADAkARvAAAAAAAAAIYkeAMAAAAAAAAwJMEbAAAAAAAAgCEJ3gAAAAAAAAAMadvgXVXHqurzVXWuqs5W1bt3YxgAAAAAAAAAbGV9gcdcTPLe7v5aVd2Y5KtV9Uh3//uStwEAAAAAAADApG1f4d3d3+/ur23e/kmSc0mOLnsYAAAAAAAAAGzlZX2Gd1UdT/L6JF++xs/ur6ozVXXmmede2qF5AAAAAAAAAHBtCwfvqnplkk8meU93//jqn3f3A919srtPHtpY28mNAAAAAAAAAPBzFgreVbU/l2P3R7v7U8udBAAAAAAAAADb2zZ4V1Ul+XCSc939weVPAgAAAAAAAIDtLfIK7zcm+b0kd1fV1ze/3rLkXQAAAAAAAACwpfXtHtDdX0pSu7AFAAAAAAAAABa20Gd4AwAAAAAAAMCqEbwBAAAAAAAAGJLgDQAAAAAAAMCQBG8AAAAAAAAAhiR4AwAAAAAAADAkwRsAAAAAAACAIQneAAAAAAAAAAxJ8AYAAAAAAABgSII3AAAAAAAAAEMSvAEAAAAAAAAYkuANAAAAAAAAwJAEbwAAAAAAAACGJHgDAAAAAAAAMCTBGwAAAAAAAIAhrc89AAAAAAAuPv7E3BMmra/wtoNzD9jCwbkHsKecztrcEyadzm1zT5j0gbkHbGH9xPG5J0y6cHxj7gmTzh89MPeELT1/pOaeMOmFmy7NPWHSvsMX5p4w6cThZ+eeMOmOjSfnnjDpnhvPzj1h0l03rO7vwtrN177uFd4AAAAAAAAADEnwBgAAAAAAAGBIgjcAAAAAAAAAQxK8AQAAAAAAABiS4A0AAAAAAADAkARvAAAAAAAAAIYkeAMAAAAAAAAwJMEbAAAAAAAAgCEJ3gAAAAAAAAAMSfAGAAAAAAAAYEiCNwAAAAAAAABDErwBAAAAAAAAGJLgDQAAAAAAAMCQBG8AAAAAAAAAhiR4AwAAAAAAADCkbYN3Vb2iqr5SVd+oqrNV9f7dGAYAAAAAAAAAW1lf4DE/TXJ3d5+vqv1JvlRVf9/d/7LkbQAAAAAAAAAwadvg3d2d5Pzm3f2bX73MUQAAAAAAAACwnYU+w7uq1qrq60meTvJId3/5Go+5v6rOVNWZZ557aYdnAgAAAAAAAMDPWih4d/dL3X17kluS3FlVr7vGYx7o7pPdffLQxtoOzwQAAAAAAACAn7VQ8P4f3f2jJF9IcmoZYwAAAAAAAABgUdsG76o6VFUHN2/fkOSeJN9e8i4AAAAAAAAA2NL6Ao+5OclDVbWWy4H8E9396eXOAgAAAAAAAICtbRu8u/ubSV6/C1sAAAAAAAAAYGEv6zO8AQAAAAAAAGBVCN4AAAAAAAAADEnwBgAAAAAAAGBIgjcAAAAAAAAAQxK8AQAAAAAAABiS4A0AAAAAAADAkARvAAAAAAAAAIYkeAMAAAAAAAAwJMEbAAAAAAAAgCEJ3gAAAAAAAAAMSfAGAAAAAAAAYEiCNwAAAAAAAABDErwBAAAAAAAAGJLgDQAAAAAAAMCQqrt3/kmrnkny5A493auTPLtDzwXA9cHZAMCVnAsAXM3ZAMDVnA0A4/v17j509cWlBO+dVFVnuvvk3DsAWB3OBgCu5FwA4GrOBgCu5mwAuH55S3MAAAAAAAAAhiR4AwAAAAAAADCkEYL3A3MPAGDlOBsAuJJzAYCrORsAuJqzAeA6tfKf4Q0AAAAAAAAA1zLCK7wBAAAAAAAA4OesdPCuqlNV9WhVPVZV75t7DwC7q6oerKqnq+pbV1x7VVU9UlXf2fz+K3NuBGB3VdWxqvp8VZ2rqrNV9e7N684HgD2oql5RVV+pqm9sngvv37zuXADY46pqrar+tao+vXnf2QBwnVrZ4F1Va0k+lOTNSW5Ncl9V3TrvKgB22UeSnLrq2vuSfK67X5vkc5v3Adg7LiZ5b3f/ZpI3JPmDzX8nOB8A9qafJrm7u38rye1JTlXVG+JcACB5d5JzV9x3NgBcp1Y2eCe5M8lj3f14d7+Y5ONJ3j7zJgB2UXd/MckPr7r89iQPbd5+KMk7dnMTAPPq7u9399c2b/8kl/+AdTTOB4A9qS87v3l3/+ZXx7kAsKdV1S1J3prkr6647GwAuE6tcvA+muS7V9x/avMaAHvb4e7+fnI5eiT51Zn3ADCTqjqe5PVJvhznA8CetfmWtV9P8nSSR7rbuQDAXyb54ySXrrjmbAC4Tq1y8K5rXOtdXwEAAKycqnplkk8meU93/3juPQDMp7tf6u7bk9yS5M6qet3MkwCYUVW9LcnT3f3VubcAsDtWOXg/leTYFfdvSfK9mbYAsDp+UFU3J8nm96dn3gPALquq/bkcuz/a3Z/avOx8ANjjuvtHSb6Q5FScCwB72RuT/HZVPZHLH5V6d1X9dZwNANetVQ7ep5O8tqpeU1UHkrwzycMzbwJgfg8nedfm7Xcl+bsZtwCwy6qqknw4ybnu/uAVP3I+AOxBVXWoqg5u3r4hyT1Jvh3nAsCe1d1/0t23dPfxXO4K/9jdvxtnA8B1q7pX913Cq+otufxZG2tJHuzuv5h3EQC7qao+luSuJK9O8oMkf5bkb5N8IsmvJfnPJL/T3T+caSIAu6yq3pTkn5L8W/738/j+NJc/x9v5ALDHVNVtSR7K5b8d7Uvyie7+86raiHMBYM+rqruS/FF3v83ZAHD9WungDQAAAAAAAABTVvktzQEAAAAAAABgkuANAAAAAAAAwJAEbwAAAAAAAACGJHgDAAAAAAAAMCTBGwAAAAAAAIAhCd4AAAAAAAAADEnwBgAAAAAAAGBIgjcAAAAAAAAAQ/pvRUtpln5OytkAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 2520x720 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "seq_class = []\n",
    "total_tokens = 50\n",
    "offset = 10 \n",
    "\n",
    "for i in range(4):\n",
    "    temp = np.zeros(total_tokens)\n",
    "    temp[:offset] = 1\n",
    "    temp[(i+1)*offset:((i+2)*offset)] = np.arange(0.1,1.1,0.1)\n",
    "    temp = temp/sum(temp)\n",
    "    seq_class.append(temp)\n",
    "    assert(sum(seq_class[i])==1)\n",
    "plt.figure(figsize=(35,10))\n",
    "plt.imshow(np.vstack(seq_class))\n",
    "    \n",
    "def sample_sequence(next_token,seq_len):\n",
    "    '''\n",
    "    function to sample a sequence based on p(l|m,n) for given m and n.\n",
    "    '''\n",
    "    query_tokens = [5,6,7,8]\n",
    "    next_tokens = [1,2,3,4]\n",
    "    if next_token==next_tokens[0]:\n",
    "        context_tokens = np.random.choice(np.arange(50),p=seq_class[0],size=seq_len-2)\n",
    "        seq = np.append(context_tokens,[query_tokens[0],next_token])\n",
    "    if next_token==next_tokens[1]:\n",
    "        context_tokens = np.random.choice(np.arange(50),p=seq_class[1],size=seq_len-2)\n",
    "        seq = np.append(context_tokens,[query_tokens[1],next_token])\n",
    "    if next_token==next_tokens[2]:\n",
    "        context_tokens = np.random.choice(np.arange(50),p=seq_class[2],size=seq_len-2)\n",
    "        seq = np.append(context_tokens,[query_tokens[2],next_token])\n",
    "    if next_token==next_tokens[3]:\n",
    "        context_tokens = np.random.choice(np.arange(50),p=seq_class[3],size=seq_len-2)\n",
    "        seq = np.append(context_tokens,[query_tokens[3],next_token])\n",
    "    return seq\n",
    "\n",
    "def generate_data(batch_size,seq_len):\n",
    "    batch_of_seq = []\n",
    "    next_tokens = [1,2,3,4]\n",
    "    for i in range(batch_size):\n",
    "        next_token = np.random.choice(next_tokens,p=[0.25]*4)\n",
    "        generated_seq = sample_sequence(next_token,seq_len)\n",
    "        batch_of_seq.append(torch.tensor(generated_seq))\n",
    "    return torch.vstack(batch_of_seq)\n",
    "\n",
    "\n",
    "def analyse_model(model,batch_size,seq_len):\n",
    "    test_inputs = generate_data(batch_size,seq_len)\n",
    "    tinputs,tlabels  = test_inputs[:,:-1].to(device),test_inputs[:,-1].to(device)\n",
    "    \n",
    "    model.eval()\n",
    "    with torch.no_grad():\n",
    "        model_output,model_attention = model(tinputs)\n",
    "    #model_output = nn.Softmax(dim=1)(model_output)\n",
    "\n",
    "    common_tokens_attn = []\n",
    "    distinct_tokens_attn = []\n",
    "    for i in range(batch_size):\n",
    "        indices_1 = torch.logical_and(tinputs[i]>=0,tinputs[i]<10)\n",
    "        indices_2 = torch.logical_and(tinputs[i]>=10,tinputs[i]<50)\n",
    "        common_tokens_attn.append(sum(model_attention[i][indices_1]).item())\n",
    "        distinct_tokens_attn.append(sum(model_attention[i][indices_2]).item())\n",
    "    return np.mean(common_tokens_attn),np.mean(distinct_tokens_attn)\n",
    "\n",
    "\n",
    "class Net_p1(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(Net_p1,self).__init__()\n",
    "        self.key_layer = nn.Linear(50,50,bias=False)\n",
    "        self.query_layer = nn.Linear(50,50,bias=False)\n",
    "        self.value_layer = nn.Linear(50,50,bias=False)\n",
    "        self.prediction_layer = nn.Linear(50,50,bias=False)\n",
    "        \n",
    "        #self.key_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "        #self.query_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "        self.value_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "        self.prediction_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "    def forward(self,x):\n",
    "        batch,seq_len,vocab_size = x.shape\n",
    "        \n",
    "        key = self.key_layer(x)\n",
    "        query = torch.unsqueeze(self.query_layer(x[:,-1]),dim=1)\n",
    "        \n",
    "        #print(key.shape,query.shape)\n",
    "        scores = (query@key.transpose(-2,-1))*(1/math.sqrt(key.size(-1)))\n",
    "        \n",
    "        \n",
    "        \n",
    "        \n",
    "        attn = nn.Softmax(dim=-1)(scores)\n",
    "        x1 = self.value_layer(x)\n",
    "        context =torch.matmul(attn,x1)\n",
    "    \n",
    "        output = self.prediction_layer(context)\n",
    "        #print(output.shape)\n",
    "        return output[:,0,:],attn[:,0,:]\n",
    "    \n",
    "class Net_p2(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(Net_p2,self).__init__()\n",
    "        self.key_layer = nn.Linear(50,50,bias=False)\n",
    "        self.query_layer = nn.Linear(50,50,bias=False)\n",
    "        #self.value_layer = nn.Linear(50,50,bias=False)\n",
    "        self.prediction_layer = nn.Linear(50,50,bias=False)\n",
    "        \n",
    "        #self.key_query_layer.weight = nn.Parameter(torch.zeros(50,50))\n",
    "        self.key_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "        self.query_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "        #self.value_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "        self.prediction_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "    def forward(self,x):\n",
    "        batch,seq_len,vocab_size = x.shape\n",
    "        \n",
    "        key = self.key_layer(x)\n",
    "        query = torch.unsqueeze(self.query_layer(x[:,-1]),dim=1)\n",
    "        \n",
    "        #print(key.shape,query.shape)\n",
    "        scores = (query@key.transpose(-2,-1))*(1/math.sqrt(key.size(-1)))\n",
    "        \n",
    "        \n",
    "        \n",
    "        \n",
    "        attn = nn.Softmax(dim=-1)(scores)\n",
    "        x1 = x #self.value_layer(x)\n",
    "        context =torch.matmul(attn,x1)\n",
    "    \n",
    "        output = self.prediction_layer(context)\n",
    "        #print(output.shape)\n",
    "        return output[:,0,:],attn[:,0,:]\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_heatmap(model,epoch_no):\n",
    "    model.eval()\n",
    "    attn_values = []\n",
    "    prediction  = []\n",
    "    with torch.no_grad():\n",
    "        running_loss = 0.0\n",
    "        for j in range(200):\n",
    "            data = generate_data(batch_size,seq_len)\n",
    "            inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "            indices = inputs>9\n",
    "            inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "            outputs,attn  = model(inputs)\n",
    "            #print(outputs.shape)\n",
    "            outputs = nn.Softmax(dim=1)(outputs)\n",
    "            #print(outputs[np.arange(128),labels])\n",
    "            #print(outputs.shape)\n",
    "\n",
    "            for i in range(outputs.shape[0]):\n",
    "                attn_values.append(sum(attn[i][indices[i]]).item())\n",
    "                prediction.append(outputs[i,labels[i]].item())\n",
    "    return attn_values,prediction"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## FAFO"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  4%|█▌                                      | 100/2500 [01:18<31:28,  1.27it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  250] loss:2.901\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  8%|███▏                                    | 200/2500 [02:36<29:57,  1.28it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  250] loss:1.310\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 12%|████▊                                   | 300/2500 [03:54<28:28,  1.29it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  250] loss:0.081\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 16%|██████▍                                 | 400/2500 [05:12<27:19,  1.28it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  250] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 500/2500 [06:33<26:41,  1.25it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  250] loss:0.005\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 24%|█████████▌                              | 600/2500 [07:51<24:23,  1.30it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  250] loss:0.003\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 28%|███████████▏                            | 700/2500 [09:08<23:17,  1.29it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  250] loss:0.002\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 32%|████████████▊                           | 800/2500 [10:26<21:58,  1.29it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  250] loss:0.002\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|██████████████▍                         | 900/2500 [11:44<20:43,  1.29it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  250] loss:0.001\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|███████████████▌                       | 1000/2500 [13:01<19:25,  1.29it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  250] loss:0.001\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 44%|█████████████████▏                     | 1100/2500 [14:18<17:57,  1.30it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  250] loss:0.001\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 44%|█████████████████▏                     | 1100/2500 [14:19<18:13,  1.28it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Finished Training\n"
     ]
    }
   ],
   "source": [
    "n_epochs = 2500\n",
    "n_batches = 250\n",
    "\n",
    "batch_size= 32 # actual batch size is times 4 here 128\n",
    "seq_len = 64\n",
    "\n",
    "n_seeds = [1234]#ß,1235,1236]\n",
    "\n",
    "Attn_Values  = []\n",
    "Prediction_Values = []\n",
    "#deltas_list = []\n",
    "for seed in n_seeds:\n",
    "    torch.manual_seed(seed)\n",
    "    model_1 = Net_p1().to(device)\n",
    "    criterion = nn.CrossEntropyLoss()\n",
    "    optimizer = optim.SGD(model_1.parameters(),lr=1e-2)\n",
    "\n",
    "    running_loss = 0.0\n",
    "\n",
    "    common_token_attn = []\n",
    "    distinct_token_attn = []\n",
    "    #deltas = []\n",
    "    epsilon = 1e-8\n",
    "    #generate_heatmap(model,0)\n",
    "    for epoch in tqdm(range(n_epochs)):\n",
    "        model_1.train()\n",
    "        # record conservation error\n",
    "#         with torch.no_grad():\n",
    "#             C1 = model.value_layer.weight.matmul(model.value_layer.weight.t())\n",
    "#             C2 = model.prediction_layer.weight.t().matmul(model.prediction_layer.weight)\n",
    "            \n",
    "#             denom = max(torch.norm(C1).item(), torch.norm(C2).item(), epsilon)\n",
    "#             delta = torch.norm(C2 - C1).item() / denom\n",
    "#             deltas.append(delta)\n",
    "        for j in range(n_batches):\n",
    "            data = generate_data(batch_size,seq_len)\n",
    "\n",
    "            inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "            inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "\n",
    "            optimizer.zero_grad()\n",
    "\n",
    "            outputs,_ = model_1(inputs)\n",
    "            loss = criterion(outputs,labels)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            running_loss +=loss.item()\n",
    "        if (running_loss/n_batches)<=0.001:\n",
    "            break\n",
    "        if (epoch+1)%100 ==0:\n",
    "            print(f'[{epoch+1},{j+1:5d}] loss:{running_loss/(100*n_batches):.3f}')\n",
    "            running_loss = 0.0\n",
    "            #generate_heatmap(model,epoch+1)\n",
    "    print(\"Finished Training\")\n",
    "    #deltas_list.append(deltas)\n",
    "    attn,preds = generate_heatmap(model_1,epoch+1)\n",
    "    Attn_Values.append(attn)\n",
    "    Prediction_Values.append(preds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss:0.001\n",
      "accuracy:1.000\n"
     ]
    }
   ],
   "source": [
    "correct = 0 \n",
    "total = 0\n",
    "model_1.eval()\n",
    "with torch.no_grad():\n",
    "    running_loss = 0.0 \n",
    "    for j in range(500):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model_1(inputs)\n",
    "        loss = criterion(outputs,labels)\n",
    "        running_loss +=loss.item()\n",
    "        correct += sum(outputs.max(1)[1]==labels).item()\n",
    "        total += len(labels)\n",
    "print(f'loss:{running_loss/(j+1):.3f}')\n",
    "print(f'accuracy:{correct/total:.3f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [],
   "source": [
    "attn_values_1 = np.mean(Attn_Values,axis=0)\n",
    "prediction_1 = np.mean(Prediction_Values,axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZAAAAGHCAYAAACebATvAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/MnkTPAAAACXBIWXMAAAsTAAALEwEAmpwYAAA+3UlEQVR4nO3deZxcVZn/8c/TazoLhCwkgUACyBIIiySsCZAFVHbQQSMIxBEj6ojgxoAjgssgoyKIQgYXlhEExgUEFWQNBNnDD4aQBQgJSQiB7J2k093V9fz+uLdyq6urq29XV3VVJd/361Wvqjp3O/X07Xrq3nvuOebuiIiIdFdVqSsgIiKVSQlERETyogQiIiJ5UQIREZG8KIGIiEhelEBERCQvSiAiIpIXJRAREcmLEoiIiOSlppQbN7NRwHDAgZXuvqSU9RERkfhKcgRiZpeY2VJgEfAM8CywyMyWmtnFpaiTiIh0T68fgZjZd4BvAtcADwErAQN2Bj4CXGVm/d39B71dNxERic96uzPF8MjjInf/cyfTPw7c4O679mrFRESkW0pxCmswMD/H9AXATr1UFxERyVMpEsjzwBVmVpc5ISz7djiPiIiUsVKcwhoLPAw0AE8RXANxgtZYxwCbgRPcfW6vVkxERLql1xMIgJkNAD4DHEmQOADeI2iRdae7b+j1SomISLeUJIH0hJnNAGYAHHPmteP2P2J6aStUJr695IJSV6FsrLjot6WuQtn47X3JUldBytDMS3eyQqyn4u5Ed/eb3X28u49X8hARKZ2ySyBm9oiZvVXqeoiISG4l7cqkEy8AS0tdCRERya3sEoi7X1bqOoiISNfK8RTWbmamq6AiImWu7BIIMAg4v9SVEBGR3ErRmeJ5Xcyye69UREREeqQU10BuJbjbvLMbUMrxqEhERDKU4sv6XeA8dx+Q7QFMKEGdRESkm0qRQF4CDs0x3QnGBxERkTJWilNYPwH655j+JjC5l+oiIiJ56vUE4u5PdTF9EzCrl6ojIiJ50gVrERHJixKIiIjkRQlERETyogQiIiJ5UQIREZG8KIGIiEhelEBERCQvSiAiIpIXJRAREcmLEoiIiORFCURERPKiBCIiInkpRW+8Zc2AKePrOeaQegbvWEXjZuel+S3cP7uJltaulx+3Xy0H7FnL7sOqGTG4mupq49s3rWf1hmTR695TVX37M+BjH6fhoMOp2WkwyS1NtK5Yyvr776LlrXlZl+l/7EfZadoMAJZ/czrJTY1db2eHgQyYdBK1u+1J3e57Uj1gRzY98zhr/ucXBf08+VixbAl/vvvXLFm0gLVrPqAtkWDw0OEcPO5oTj7zXAYOGrJ13j/9/mb+fNevs65n2vSLOPnMz3S5vR9++0Lmvzan0+kHHHw4//690sclXU//RwDG7lnDiUc3MHJoNYk2Z/6SBH96oonV68v//yTd9h4LJZAMZ01tYMr4Pry8oIVHXtjC8MHVTBlXz27Dqrn+ro2dDqOYctyH6xk9ooZlH7TxwbokwwdX90q9e6p60FB2vvgqrL4Pm/75GIn338Ua+lK36yiqBw7KukzVjjux4+nnkNzSRFWfhtjbqh22Kzt87BMk1nxAy5K3aBiba3iY3rVm9UrWrV3FuCMnMWjwzlRVV7NsyZs8/o97efaph/nBdb9jx4x4nPO5Sxiww8B2ZaP32i/W9k4767NMOuH0DuXPzn6Y//fCbD582DF5f5Zi6en/yCH71DLjjH4se7+NPz3RRJ96Y+r4er55zgCuvn0D6zd2tYbysb3HQgkkzYghVUwaV8+cBS3cfO+mreWr1iWZdkJfxo+p5YV5uX9W3PLAJtZvdJIO045vqJgEMnj6RVBdzXs//BrJDetiLbPTpz5PYtVKWt9dSr8jjou9rZZ33mL5tz5LcuMGqvoNYNcf35pfpYvggIMP54CDD+9Qvu8BH+YX/3U5Tz32AKd8/Lx208YdcRxDh+2S1/YOPOSIrOX33fNbamvrmDDpY3mtt1h6+j9SVQXTju/L2g1JfnpHI83hrHMXtXL5+QM4ZUIDdzy0udgfoyAUC10DaeewMXVUmfHYi83tyme/0kxzi3PEAfVdrmNtY5A8Kkn9h/an/kP70/jwvUHyqKrGautyLtNw8OE0HDSetXf+NyS7d6jtzVtIbtzQgxr3viFDRwCwaWP2U3RNmzfS1pYoyLYWzH2ZFcuXMO7ISfQfsGNB1lkoPf0f2We3GgYOqOLpV1u2fmECLHu/jYXvJBg/po6qCvlWUix0BNLOqBE1JJPO4hXtvwgSbcEfddSIyjia6K4+BwSnkBJrVjHki5fRZ/8PY9XVtK58lw1//182P/9ku/mtTwMDP3UBm2Y/TMuSN0tR5aJraWmmeUsTrS3NLF/6NnffFlyHOGTc0R3mvfyr57ClaRNVVdXsuff+nPGpz3FwlvnieuKRvwAw6YTT8l5HsfT0f2TUiOArZ9Hyjsn27XcT7De6lmGDqlixqvzP/ysWSiDtDOxfxcYmJ9HWcdq6jUn2GllDdRW0le/fMy814emXQed8kcT7K1hz+w1YTS0Dpp7K4Olfxaqq2fTs41vnH3jGuZhVse7eO0pV5aKb9fB93H7zT7a+H7LzCC685Hvse8CHt5b17TeAyR89k733O5B+/XZgxfIlPHT/Xfz0+5dwwVe+w7FTT+n2dps2b+T5px9l6LBd2P+gwwryWQqpp/8jA/vb1nk7Lu9bt1HOX5opioUSSDt1NWTdGQBaE8EftK7WaGqusHNUXUhdAPctTbx/3XchPBWz+f89xy7fv5EdTz+HTc89Ae7U7bkv/SaewJpbr8e3lPf52Z4Yd8RxjNh1NFu2bGbJooW8/PyTNG5Y226ej5326Q7LHXv8qVx20ae54zc/4/Cjp9CnoW+3tvvMk/+gpXkLxx5/KmbWo89QDD39H6mrDT5TIsvZvvTlK4FioWsg7bQkoKaTo87amuAP2dK6bSUPAG9tAWDTi7O3Jg8Ab9pE06svUL3jTsFRSnUNg87+Is0LXmXzi7NLVd1eMWjIMMYecjjjj5zEJ86ewYyLv8vdt/2Cv/zh1pzLDdhhIFM+9nE2b2rkjfmvdnu7sx75C1VV1Rw79dQ8a15cPf0fSU2ryfLTtdL+xxQLJZB21m1M0r/Bsu4UA/tX0bg5uc2dvgJoW7saIGvrq7b1QVlV3/70P+5j1AzfhcZH76dm6PCtDwuPYGqG7Ez14GG9Ve1etfvovRm15748+rc/dDnv0J2DC+6NMVuzpSxd/CaL3nidgw49kkGDd86nmkXX0/+R9FMzHZfv/JROOVIsdAqrnSUrEhywRy2jR9Tw5rLol3hNNYzcuZo3lhWmlU25aVn8Bhz7UaoHDu4wrXqn4J6HZON6agYNxaqqGfpv38m6nmGX/hfJLU0s/1rXN9BVopaWLWyM0XrsvXeXAnS4X6QrTzx8HwDHZbkvpFz09H9kSXjBec9da5i/pP28e+xSQ1Ozs3JNeX9ppigWOgJp58V5LSTdmTK+ffO7iQfXU19nPD+3ZWvZDv2MYYOqqN0GUvDmV54n2bSZvocfi9X32VpetcNAGg46nNaV75L44D02PfMYq3714w6PLQtfA2DN7b9gzW0/j1ZcVU3NsF2p3mlI5ibL1rq1q7KWv/7qiyx7ZxEf2ncsAG1tCTZv2thhvtUfrOSxB/9I/wE7svd+B20tTyQSvLtsMas+eC/r+ltbW/jnrAfZceAgPnzYxAJ8kuLo6f/IwqUJ1jUmmXBQHfW1UfmuQ6vZZ/caXprf0t1W4SWjWOgIpJ13VyWZNaeZyeP68IUz4LVFrVvvLF34TisvvB7tEGce18BRB9Zz7Z2NLFwa/Xr40Mga9t4tCOvuYTO9SePq2bwlOFz9+zNbeu8DxeRNm1j3p9sZdM6FDPvm1Wz852NYTQ39j/koVlPD2nuC7jpaly+hdfmSDss3jB0PQNP/vdiuK5PqgYMY8d2fs2Xha3xw3XfbLbPDxz4BgNUF/3y1u47aWtb85jya33y98B80hltvuoZ1a1ez/0HjGTJ0OK2tLbz95jyenf0wDQ19OfuzFwOwpamJr804g3FHHMcuu42mX/+gFdash+9jS1MTX/rG96lLS8ZrV7/PpV/+JPuNPZRv/3Bmh+2+9OwsNjau5+SPn0t1dfn+W/b0fySZhHse3cwFp/fj6+cM4OlXWuhTB1MP60PjZueB2U0l+mTdp1gogXRwz6NBHzQTD65n7F61bGpyHp/TzP1PNXXZLQHAfqNqOGVi+249Tjg8+iIpxwQCsOnph0lu2sCAE85gx1OngTstby9g9S0/o2XRgoJvb8fTzm73vm73oF8sgPV/vbtkCeSoYz/CU4/9jacf/1twDcOMIUOHM+WjZ3LSmecyZOjwoL719Rx21GTeemMuLz03i+Ytm+m/w0AOOOhwTv74uey1zwHd2u6s8N6P444vv3s/MvX0f2TOglZu+uMmTjy6D5+Y3EBrm7Mg7P9pXZl33ZFpe4+FuZd/JTtz4TVrK7fyBfbtJReUugplY8VFvy11FcrGb+8r83MgUhIzL92pIO2DdQ1ERETyogQiIiJ5UQIREZG8KIGIiEheYiUQM/uUmdV2PaeIiGwv4h6B/B5418x+amb7F7NCIiJSGbpzCmswcDHwf2b2TzObbmbd62pURES2GXETyFeBf4avDTgS+A2wwsxmmtn4YlRORETKV6wE4u43uPtEYBRwKTCHIJEMAD4PPGdmL5jZyUWrqYiIlJVutcJy92Xu/mPgS8CstEkGjAP+YmafL2D9RESkTMVOIGbW18w+b2YvAc8Ax4aTHPgzMJsgkfx7wWspIiJlJ1Znimb2S+AcglNWqT5UNgC/BX7u7ovNrBp4D9i9GBUVEZHyErc33i+mvX4LuAH4rbtvHRDB3dvMbDXQvVF0RESkInWnO/dZwHXAX7yTLnzdfb9CVEpERMpf3ATyYXd/pag1ERGRihL3IvrLZpZ1gF8ze8zMHi1gnUREpAJ05xRWZwOQTIJYg2+JiMg2pEe98ZrZQeFLJRARke1MpwnEzL5rZm1m1hYVBe/Tyl8Op73XnY2a2aCM90eb2UQza+hsGRERKS9dHYEY7U9dWZYHwH1xNmZme5vZQuADM3vGzIaY2eMENyE+CcwzM7XkEhGpALmugSwm6q7kOILTVE+mTXdgNfAs8MuY2/sx8DYwAzgf+DvQCOwGJIFbgGuA02OuT0RESqTTBOLutwG3AZhZMiyb3MPtTQCmuvurYZco64Dj3H15uJ3LCZKKiIiUubitsPYo0PbqgdTd65sIjjoa06ZvAHQdRESkAnSaQMzsWAB3f5KgG3fMbFRn84fzdeU14PPhkcbngFXANCB1k+LZwMJYNRcRkZLKdQTyBMERQk34OldTXe9iXSlXAfcC3wBagY8AvzGzjwBtwKEESURERMpcV1/6mS2wesTdHzKzMQRjh7zo7kvCI50vA32BS9398Z5uR0REii9XAvke0VHHVYXaoLsvJmjhlXq/Ergi7vJmNoOgFRfHnHkt+x8xvVBVq2g/HPXrUlehfNyXLHUNRLYL1knHuhXhwmvWVm7lRURKZOalO/X4jBL0sCuTQjOzR8zsrVLXQ0REuparFdZj3ViPu/vUAtTnBWBpAdYjIiJFlusayCTidZJoMefrkrtfVoj1iIhI8XWnFVZBmNlIgiFyjwaGEySflcDTwEx3X1bobYqISOHl6sqk4NdHzGwiQVclK4B/AI8RJKmdgbOAi8zsRHd/utDbFhGRwurOgFKFcB1wi7tflG2imV0fznNYL9ZJRETyEKsrk9TrXGJ2ZXIAcE6O6TcR3uMhIiLlrbe7MllB0CPvgk6mTwjnERGRMterXZkAPwFmmtnhwMMEF8+d4GL6CcB04OICbEdERIqsV7sycfcbzWw1cAlBb7zV4aQ24CXgPHe/pxDbEhGR4srVCuvKtNeF7AvrbuBuM6sFhoTFq9y9tVDbEBGR4utWKywzGwwcS/DFvwp40t1X57PhMGHoeoeISIWKnUDM7ErgUqAurbjFzH5UyCMUERGpDLFuFjSzbxJ0uV5PcDE99agHrjCzrxethiIiUpbi3m3+5fC5CbgT+FH43ESQSL5S+KqJiEg5i3sKaxhBi6zT3f2RVKGZnQA8RNAViYiIbEfiHoHMDZ+fzSh/Jnx+rTDVERGRShE3gfw7wb0aX8oo/xLQClxeyEqJiEj5686AUmuBq83s3wgGfRoZPj4ALgMeQUREthv5DCi1a/hIGRrOKyIi25FCDShVkBEJRUSkcvTqgFIiIrLtUJIQEZG8dLcvrIOAfYGGzGnufnuhKlVKBkwZX88xh9QzeMcqGjc7L81v4f7ZTbTE7O5x7J41nHh0AyOHVpNoc+YvSfCnJ5pYvT5Z1LoXmmIRUSwiikVke4+FuXd9+cLM+gP3ApM7mcXdvbeHx+XCa9YW/NrLJ6c2MGV8H15e0MLct1sZPriayYfW88ayBNfftbHLiz2H7FPLjDP6sez9Np5+pYU+9cbU8fUkk3D17RtYv7FyLhcpFhHFIqJYRCo1FjMv3akQ4zvFPgL5D2BKITZYzkYMqWLSuHrmLGjh5ns3bS1ftS7JtBP6Mn5MLS/M6/xnRVUVTDu+L2s3JPnpHY00h7POXdTK5ecP4JQJDdzx0OZif4yCUCwiikVEsYgoFvGvgZxO0NLqwfC9A9cSdOm+kAINOFVqh42po8qMx15sblc++5VmmlucIw6oz7n8PrvVMHBAFU+/2rJ1ZwBY9n4bC99JMH5MHVUVctVJsYgoFhHFIqJYxE8go8LnC1IF7v4NgsSyD/BegetVEqNG1JBMOotXJNqVJ9qCP+qoEdWdLBktD7BoeaLDtLffTdBQbwwbVOZ7REixiCgWEcUioljETyCp82XvAQkAM9sBeDks/2aB61USA/tXsbHJSbR1nLZuY5IBfauozhGxgf1t67wdl/et26gEikVEsYgoFhHFIn4CWRM+DyDougTgBuBn4etdClmpUqmrIevOANCaCP6gdbWdX3tKTUt0/EERa/lyolhEFIuIYhFRLOInkDfD592B5wiOSD4DzCC4HjKv8FXrfS0JqOnkqLO2JvhDtrR23ioiNa0mS9OEOMuXE8UiolhEFIuIYhE/gdwN/AMYAfwA2EI0KmEzwVC3FW/dxiT9GyzrTjGwfxWNm5O05WianeuwM9fhajlSLCKKRUSxiCgWMROIu9/o7ie6+z/cfQ5wAHAJwUiEY9390WJWsrcsWZGgqsoYPaL9T4Kaahi5czVL3uvkeDVteYA9d+34k2KPXWpoanZWrinvHSJFsYgoFhHFIqJY5NGViZntRHD/yO3u/kt3X1T4apXGi/NaSLozZXz75ncTD66nvs54fm7L1rId+gUtJGrT/vYLlyZY15hkwkF11NdG5bsOrWaf3Wt4aX4LyfLeH7ZSLCKKRUSxiCgWMe9EBzCzqcA1wIfTiucAl7p75tghvaIYd6J/6vgGJo8L7ix9bVFwZ+mUcfW8tTzBz34f3Vl6/kl9OerAeq69s5GFS6OrYIfuW8sFp6fdWVoHUw/rgztcfduGrYetlUCxiCgWEcUiUqmx6NU70c3sROA+oJr2XbyPAx40s9Pc/cGsC1eYex4N+qCZeHA9Y/eqZVOT8/icZu5/qilWn/VzFrRy0x83ceLRffjE5AZa25wFYd82lfSPAYpFOsUiolhEtvdYxO0L62Xg4PDtsvCRGpEQ4P+5+6FFqWEOxTgCERHZ1hXqCCTuNZD9CJrrfsvdd3f3o919d4Kx0gHGFKIyIiJSOeImkHfD55syym8Mn5cXpjoiIlIp4iaQG8LnCRnlR4XPPy9MdUREpFJ0ehHdzK7IKFoB/NnM/gS8Q3BX+pkERx8Di1VBEREpT7laYV0JWRsSnJ3xvgG4AvhegeokIiIVoKtmvOXdk5eIiJRMrgTS2fC1IiIinScQd5/VmxUREZHKEndMdADM7EjgJGBn4H3gr+7+XDEqJiIi5S12AjGzmcDnM4q/bWYz3f3Lha2WiIiUu1j3gZjZdILBoyzL40IzO69YFRQRkfIU90bCGeHzEuBigvs/vgosJkgiXyh0xUREpLzFPYU1luCekFPd/bVUoZk9DrwaThcRke1I3COQuvB5WUb5sozpIiKynYibQJaGzz8xs4EAZrYj8OOM6SIisp2Im0AeILjW8VlgtZmtB9YA/0pwauv+4lRPRETKVdwE8gOCDhRTLa8GpL1eAvywKLUTEZGyFSuBuPtq4AjgNwS98ibC518BR7n7mqLVUEREylKXrbDMrJZo3I9L3T3zZsIeMbPDCLqGX+LuLxZy3SIiUjxdJhB3bzWzx8K3u/dkY2b2n8CT7v6gmQ0F/kJwZJMAaszsGeAMd/+gJ9sREZHii3sNZBnB9Y4NPdze+QSnviBowWXAvu5eB+wNtAI/7eE2RESkF8RNIDeHzz3tsmQwsD58fRzwdXd/A8Dd3wIuAT7Sw22IiEgviHsnej1Bs90bzOw04GWgKX0Gd48zIuHbBHetLwaSWaY7wQiHIiJS5uImkO8QDW97QvjIFCeB/DfBzYgLgRvC159x97fMbA/gZ8CDMeskIiIl1J3xQHINb5tt7PSOM7lfZ2a7A68BbwGjgYVmlgjrMgeY1o06iYhIicRNIAUb3tbdv2Zm/w2cBuxJcB1mBfA08Ii7x0pGIiJSWrESSKGHt3X3BUT9aHWLmc0g7F7+mDOvZf8jphewZiIiElfOBBJ2mPgV4PCw6FngRndfV+R6dcrdbyZsFXbhNWt1tCIiUiKdJhAzG0yQMPZMKz4Z+FczO9LdVxW6Mmb2CLCHu+9V6HWLiEhh5boP5DvAXnQcwnYP4D+KVJ8XgCeLtG4RESmgXKewTiZoXfUWcBNBsvkC8CHgFIKhbQvK3S8r9DpFRKQ4ciWQ3cLn09x9PoCZ/Q2YC4zMd4NmNhL4InA0MJwgSa0kaIU1090zRz0UEZEylOsUVh1AKnmEr+eFL2vz2ZiZTQTmAWcRJKI7gbvC12cBr5vZhHzWLSIivStOd+67keUmwsxyd38nxvauA25x94s62db14TyHxViXiIiUUJz7QBZnvPcs5R5zXQcA5+SYfhPhPR4iIlLe4vTGm9kKq6vyXFYAuU5RTSDq7l1ERMpYrqOGd4jZx1U3/ASYaWaHAw8TXDx3govpJwDTKULrLhERKbxOE4i7jy70xtz9RjNbTTDux+eA6nBSG/AScJ6731Po7YqISOF1pzfegnD3u4G7w7HWh4TFq9y9tbfrIiIi+ev1BJISJgxd7xARqVBxh7QVERFpRwlERETyogQiIiJ5UQIREZG8KIGIiEheYrfCMrOjgXOBUUCfjMnu7lMLWTERESlvsRKImZ0H3NLZZAp/x7qIiJS5uEcg/078/q5ERGQ7EDeBjCY4yvgv4HfAJnTUISKyXYubQBYBY4D/dPfGItZHREQqRNxWWNcQnML6dBHrIiIiFSTuEcgUYB1wk5ldACwA0js/dHf/XIHrVhIGTBlfzzGH1DN4xyoaNzsvzW/h/tlNtMTs7nHsnjWceHQDI4dWk2hz5i9J8Kcnmli9PlnUuheaYhFRLCKKRWR7j4W5d30pw8ySdHHNw92rc00vhguvWVvw6zCfnNrAlPF9eHlBC3PfbmX44GomH1rPG8sSXH/Xxi4v/ByyTy0zzujHsvfbePqVFvrUG1PH15NMwtW3b2D9xsq5dKRYRBSLiGIRqdRYzLx0p4I0iupOb7y5Nlg5f/EcRgypYtK4euYsaOHmezdtLV+1Lsm0E/oyfkwtL8zr/GdFVRVMO74vazck+ekdjTSHs85d1Mrl5w/glAkN3PHQ5mJ/jIJQLCKKRUSxiCgW8a+B7NHFY8+i1K6XHTamjiozHnuxuV357FeaaW5xjjigPufy++xWw8ABVTz9asvWnQFg2fttLHwnwfgxdVRVyL3/ikVEsYgoFhHFImYCcfclXT2KXdHeMGpEDcmks3hFol15oi34o44akfss3agRwQHdouWJDtPefjdBQ70xbFCZ7xEhxSKiWEQUi4hi0c2+sMzsDDO70czuCt8fY2bHmln/4lSvdw3sX8XGJifR1nHauo1JBvStojpHxAb2t63zdlzet26jEigWEcUiolhEFIuYCcTMqs3sPuCPwIXAWeGkbwGPA58pTvV6V10NWXcGgNZE8Aetq+38UlBqWqLjD4pYy5cTxSKiWEQUi4hiEf8I5GLgVDpeSL8lLDu1gHUqmZYE1HRy1FlbE3z0ltbO2wukptVkaZoQZ/lyolhEFIuIYhFRLOInkPMJWlpdl1H+cvg8plAVKqV1G5P0b7CsO8XA/lU0bk7SlqNpdq7DzlyHq+VIsYgoFhHFIqJYxE8gHwqfr8gofzd8Hl6Y6pTWkhUJqqqM0SPa/ySoqYaRO1ez5L1OjlfTlgfYc9eOPyn22KWGpmZn5Zry3iFSFIuIYhFRLCKKRfwEkopEZq7dN3yOec9leXtxXgtJd6aMb9/8buLB9dTXGc/PbdlatkO/oIVEbdrffuHSBOsak0w4qI762qh816HV7LN7DS/NbyFZ3vvDVopFRLGIKBYRxSL+jYTzgHHAN1MFZnYkcEP4dm6B61US765KMmtOM5PH9eELZ8Bri4I7S6eMq2fhO6288Hq0Q5x5XANHHVjPtXc2snBp8EsimYR7Ht3MBaf34+vnDAjuLK2DqYf1oXGz88DsphJ9su5TLCKKRUSxiCgW8RPIbcB44DKiu86fDp8d+J8C16tk7nk06INm4sH1jN2rlk1NzuNzmrn/qaZYt9vPWdDKTX/cxIlH9+ETkxtobXMWhH3brKugLhpAsUinWEQUi8j2Hou4fWEZ8GfgtCyTHwBO9zgrKrBi9IUlIrKt69W+sNzdzexM4JMETXZ3Bt4nSB53A30JBpkSEZHtRNwx0S9x958RJIu7M6YNBP4OHFXw2omISNmK2wrrp2b2xcxCMxsKzAIOL2itRESk7HWno5UbzOyzqTdmthvwFHBgwWslIiJlL24rrMcIRiW82cy2AM8DjwCjwun/VYS6iYhIGYt7BHIy8BDBjYS3Ac8QJA8HvuHulxWneiIiUq7ijgfSDJxO0OqqBhhCcPf5ee5+bfGqJyIi5arTU1hmltnvFcCrBKeyGoBngb1S87n794pSQxERKUu5roFcSe6xzieGjxQlEBGR7UhXF9Hj3q2oO8JFRLYzuRLI5F6rhYiIVJxOE4i7z+rNioiISGWJex8IsLUL95OI+sL6q7s/V4yKiYhIeYudQMxsJvD5jOJvm9lMd/9yYaslIiLlLtZ9IGY2HZhBcFE983GhmZ1XrAqKiEh5insn+ozweQlwMXAm8FVgMUES+UKhKyYiIuUt7imssQRNdU9199dShWb2OMHNhWOLUDcRESljcY9A6sLnZRnlyzKmi4jIdiJuAlkaPv8kHEAKM9sR+HHG9LyY2QXh+kREpELETSAPEFzr+Cyw2szWA2uAfyU4tXV/D+txI7BLD9chIiK9KFdniucBuPvtwA8ILpzvHk4ekDbrYuCHcTZmZhty1OMFM0uG29whzvpERKR0cl1EvxVIAre7+2ozO4IgkZwEDAU+IDgyucLd18TcngFPAH/IKPs18J/A8u5UXkRESid2Z4ruvpKONxJ216HA74FjgYvcfROAmf0KuNfdX+/h+kVEpJd0Z0z0HnP3N4CjgA3AHDM7rDe3LyIihdPlfSCdDCzVQdwBpdy9FbjEzB4B7jWzG1F38CIiFSfOjYTfjbmubg0o5e5/NbPxwO9i1kNERMpInFNY2fq/ynzkxd1XuPtUd6+Ke/3DzGaY2Ytm9uLrz92a76ZFRKSH4vzyv6rotegGd78ZuBngwmvW6tSXiEiJdJlA3L3XEkh4XWQPd9+rt7YpIiL5KbdrDy/Qw25RRESkd5RVAnH3y0pdBxERiSdXAvlsMTZoZiOBLwJHA8MJmvCuBJ4GZrp7Zo+/IiJShjpNIO5+W6E3ZmYTgb8DK4B/AI8RtOLaGTgLuMjMTnT3pwu9bRERKazePoV1HXCLu1+UbaKZXR/OozvURUTKXK92ZQIcAPwyx/Sb0OiGIiIVobcTyApgQo7pE8J5RESkzPX2KayfADPN7HDgYYKL505wMf0EYDpwcS/XSURE8hArgYQdKrq7fz/LtGMJJj7Z1Xrc/UYzWw1cAnwOqA4ntQEvAee5+z0x6y4iIiUU9wjkSoIjhQ4JhGCAqGTcdbn73cDdZlYLDAmLV4W99IqISIXo0SmsMAlAHh0qhglD1ztERCpUrjHRjwOOyyjLHBtkTPi8ucD1EhGRMpfrCGQSkJ4wjOxjgzgwr4B1EhGRChB3THTPeJ9uFfCtgtVIREQqQq4EcivBBXIj6HLEgclp0x1YDbzp7s1Fqp+IiJSpXH1hLQGWAJjZ7UGRz+qtiomISHmL2/R2epHrISIiFSZ2M14zOxs4FxgF9MmY7BpFUERk+xL3TvRvAj/qbDLRRXYREdlOxD0C+QJ53CwoIiLbrri98e5CcJTxb8AAd6/KeFR3sbyIiGxj4iaQueHz79x9U7EqIyIilSNuAkndkf4NM9OpLBERiX0N5FtAI/Bt4AIzewtI7z3X3X1qoStXCgZMGV/PMYfUM3jHKho3Oy/Nb+H+2U20xOwveOyeNZx4dAMjh1aTaHPmL0nwpyeaWL0+WdS6F5piEVEsIopFZHuPhbl33YDKzJJ03tLKCBJIr18HufCatQVv/fXJqQ1MGd+Hlxe0MPftVoYPrmbyofW8sSzB9Xdt7LK52SH71DLjjH4se7+Np19poU+9MXV8PckkXH37BtZvrJwGa4pFRLGIKBaRSo3FzEt3KsiZpO50577Nn7oaMaSKSePqmbOghZvvjS71rFqXZNoJfRk/ppYX5nX+s6KqCqYd35e1G5L89I5GmsNZ5y5q5fLzB3DKhAbueKgyOi5WLCKKRUSxiCgWMa+BZGl1tU22wjpsTB1VZjz2YvuuvWa/0kxzi3PEAfU5l99ntxoGDqji6Vdbtu4MAMveb2PhOwnGj6mjqrdHoc+TYhFRLCKKRUSxiH8RfbswakQNyaSzeEWiXXmiLfijjhqRO0+OGhEc0C1anugw7e13EzTUG8MGVUbIFYuIYhFRLCKKRTcTiJkdZGZnmdl5mY9iVbA3DexfxcYmJ9HWcdq6jUkG9K2iOkfEBva3rfN2XN63bqMSKBYRxSKiWEQUi/hdmfQH7qV9d+7pHLi9QHUqmboasu4MAK2J4A9aV2s0NWe/sFVXG+wQiY4/KNotXwkUi4hiEVEsIopF/COQ/wCmEFxI7+xR8VoSUNPJUWdtTfARW1o7bxWRmlaTJS3HWb6cKBYRxSKiWEQUi/gJ5HSCo4wHw/cOXEswGuFC4KrCV633rduYpH+DZd0pBvavonFzkrYcTbNzHXbmOlwtR4pFRLGIKBYRxSJ+AhkVPl+QKnD3bxAkln2A9wpcr5JYsiJBVZUxekT7nwQ11TBy52qWvNfJ8Wra8gB77trxJ8Ueu9TQ1OysXFPeO0SKYhFRLCKKRUSxiJ9AUqeo3gMSAGa2A/ByWP7NAterJF6c10LSnSnj2ze/m3hwPfV1xvNzW7aW7dAvaCFRm/a3X7g0wbrGJBMOqqO+NirfdWg1++xew0vzW0iW9/6wlWIRUSwiikVEsYh/I+EaYDgwAPggfH0DkLrLZZfCV633vbsqyaw5zUwe14cvnAGvLQruLJ0yrp6F77TywuvRDnHmcQ0cdWA9197ZyMKlwS+JZBLueXQzF5zej6+fMyC4s7QOph7Wh8bNzgOzm0r0ybpPsYgoFhHFIqJYxE8gbxIkjd2B54AzgM+E0xyYV/Calcg9jwZ90Ew8uJ6xe9Wyqcl5fE4z9z/VFGvUrDkLWrnpj5s48eg+fGJyA61tzoKwb5t1FdRFAygW6RSLiGIR2d5jEbcvrC8DpwA/I7hwPptoWNstwKnu/mixKtmZYvSFJSKyrevVvrDc/ZfAL1PvzewA4DSC6yF/d/dFhaiMiIhUjrg3El4J/Nbd3wFw97eB64tYLxERKXPdGVBqkZk9YmafNrPcvYSJiMg2rzsdrVQRdGXyO2CFmf3CzMYVp1oiIlLu4iaQk4D/ATYS3BMyEPgi8LyZvWJmFxWneiIiUq7ijgfyoLufDwwDPkXQsWILQTI5kKB1loiIbEe6MyIh7r4F+F8zew6YD3yju+sQEZFtQ+wvfzMbDHwS+DRwNO174d3U2XIiIrJtituM92/A1LT5U4njn8BvgbsLXzURESlncY9APpb2egXB4FG3uPvCwldJREQqQdwEkgDuJzja+Lu7l3kfkSIiUmxxE8gu7r6qqDUREZGKErcvrFUAZnYG8BFgkLtPM7NjCK6HzHH3jUWrpYiIlJ1Y94GYWbWZ/QX4I3AhcFY46VvA40Rdu4uIyHYi7p3oFxN0557ZBfAtYdmpBayTiIhUgLgJ5HyCgaOuyyhPDWk7plAVEhGRyhA3gXwofL4io/zd8Hl4YaojIiKVIm4CaQufqzPK9w2fWwtTHRERqRRxE0hqzPNvpgrM7EjgN+HbuXE3aGbTzOx/zezmzO7gzWyImWl0QxGRChA3gdxGcLH8Mtg6VvzTwKHh+/+JsxIz+2w4bxLYDfinmc1Im6UaGBWzTiIiUkJxbyS8ETiBYBz0TA8AM2Ou52LgK+4+E8DMTgHuMLN6d78h5jpERKQMxL2R0M3sTIKxQE4BdgbeJ0ged7u751o+zYeAB9PW+4CZnQT8zcyqgd93p/IiIlI6sbtzD5PEXeEjX+uBEcDitPU+bWYnA38Lp4mISAXoNIGY2bHdWZG7PxljtueBE4FnMpadHZ7O+lt3tikiIqWT6wjkCaIL5l3xLtaV8jOCwag6rsD9yTCJnB9zmyIiUkJdfelndl3SI+4+C5iVY/oTBIlLRETKXK4EclWv1aIbwma/MwCOOfNa9j9iemkrJCKynbL4DaiKz8weAfZw973izH/hNWvLp/IiIhVi5qU7FeTsUuxWWABmVkNw8+Bgd/97ISqQ4QVgaRHWKyIiBRY7gZjZWcAvgCGEF83N7FFgD+BCd/9HTyvj7pf1dB0iItI74g4odQzBTX5DCC6spw5//gqMBv4l7gbNbKSZ/dDMHjezeWb2evj6B2Y2snvVFxGRUonbF9Zl4bwLMsofCZ+PirMSM5tI0DHjWQQdMN5JcGPi3LDsdTObELNOIiJSQnFPYR1JcNrqVOCNtPJUz7m7xlzPdcAt7n5Rtolmdn04z2Ex1yciIiUS9wikX/j8TkZ5Q8ZzVw4Afplj+k3A2JjrEhGREoqbQJaHz5mnqr4RPi+LuZ4VQK5TVBPCeUREpMzFPYX1EPAF4N5UgZnNB/YmOLX1UMz1/ASYaWaHAw8DK8PlhxN0Fz+doMt3EREpc3ETyA8IWloNJuofa2+C1lirgavjrMTdbzSz1cAlwOeIhshtA14CznP3e2LWSURESijWKSx3X05weukfBKMJWvj8D+CYcHos7n63ux8J9CW4+L4r0Nfdj1TyEBGpHN0ZD2Qh8DEz6wMMAta4+5Z8N+zureh6h4hIxYp7EX0rd9/i7u+mkoeZHWRmfy181UREpJzlPAIxs77A6cDuBEPY3u/uq8JpI4EfAdMocLfvIiJS/nKNSDgSmA3slla8MRx+tg74M9CfIHmoV1wRke1MriOQKwmOPNINAH5LcOPggLBsA0HzXBER2Y7kSiBTCY4s3gH+RHCkcSbwoXB6G8Fd5d9399XFrKSIiJSfXAlkRPj8MXdfAGBmNwOvEySWc939riLXT0REylSuVlh1AKnkEb6enzb9D8WqlIiIlL8u7wMxsys6mXS5WdT4yt2/V6hKiYhI+YtzI+F3M957J+VKICIi25GuEkjc+zvUjFdEZDuTK4Fc1Wu1EBGRitNpAnF3JRAREelU7M4UtxcGTBlfzzGH1DN4xyoaNzsvzW/h/tlNtLTGW8fYPWs48egGRg6tJtHmzF+S4E9PNLF6fbKodS80xSKiWEQUi8j2Hgtzr9zLFxdes7bglf/k1AamjO/DywtamPt2K8MHVzP50HreWJbg+rs2dnmx55B9aplxRj+Wvd/G06+00KfemDq+nmQSrr59A+s3Vk68FYuIYhFRLCKVGouZl+5UkP4LdQSSZsSQKiaNq2fOghZuvnfT1vJV65JMO6Ev48fU8sK8zn9WVFXBtOP7snZDkp/e0UhzOOvcRa1cfv4ATpnQwB0PbS72xygIxSKiWEQUi4hikUd37tuyw8bUUWXGYy82tyuf/UozzS3OEQfU51x+n91qGDigiqdfbdm6MwAse7+Nhe8kGD+mjqoKibhiEVEsIopFRLFQAmln1Igakkln8YpEu/JEW/BHHTWiupMlo+UBFi1PdJj29rsJGuqNYYMqI+SKRUSxiCgWEcVCCaSdgf2r2NjkJNo6Tlu3McmAvlVU54jYwP62dd6Oy/vWbVQCxSKiWEQUi4hioQTSTl0NWXcGgNZE8Aetq+382lNqWqLjD4pYy5cTxSKiWEQUi4hioQTSTksCajo56qytCf6QLa2dt4pITavJ0jQhzvLlRLGIKBYRxSKiWCiBtLNuY5L+DZZ1pxjYv4rGzUnacjTNznXYmetwtRwpFhHFIqJYRBQLJZB2lqxIUFVljB7R/idBTTWM3LmaJe91cryatjzAnrt2/Emxxy41NDU7K9eU9w6RolhEFIuIYhFRLJRA2nlxXgtJd6aMb9/8buLB9dTXGc/PbdlatkO/oIVEbdrffuHSBOsak0w4qI762qh816HV7LN7DS/NbyFZ3vvDVopFRLGIKBYRxUJ3onfwqeMbmDwuuLP0tUXBnaVTxtXz1vIEP/t9dGfp+Sf15agD67n2zkYWLo2ugh26by0XnJ52Z2kdTD2sD+5w9W0bth62VgLFIqJYRBSLSKXGQneiF8k9jwZ90Ew8uJ6xe9Wyqcl5fE4z9z/VFKvP+jkLWrnpj5s48eg+fGJyA61tzoKwb5tK+scAxSKdYhFRLCLbeyx0BCIisp0p1BGIroGIiEhelEBERCQvSiAiIpIXJRAREcmLEoiIiORFCURERPKiBCIiInlRAhERkbwogYiISF6UQEREJC9KICIikhclEBERyYsSiIiI5EUJRERE8qIEIiIieVECERGRvCiBiIhIXpRAREQkL0ogIiKSFyUQERHJixKIiIjkRQlERETyogQiIiJ5UQIREZG8KIGIiEhelEBERCQv5u6lroMUgJnNcPebS12PcqBYRBSLiGIRKVQsdASy7ZhR6gqUEcUiolhEFItIQWKhBCIiInlRAhERkbwogWw7dG43olhEFIuIYhEpSCx0EV1ERPKiIxAREcmLEoiIiORFCaRCmNmXzOxtM9tiZi+Z2TE55p1kZveZ2Qoz22xmr5rZv/ZmfYupO7HIWG5vM2s0s43FrmNv6W4sLHCxmc03s+ZwH/lRb9W3mPKIxUfN7Jlwn1gV/s/s01v1LQYzO9bM/mJmy83MzWx6jGUONLNZZtYULneFmVmc7SmBVAAz+xRwPfCfwIeBfwJ/N7PdO1nkaOD/gH8BxgI3ATeb2dm9UN2iyiMWqeXqgLuAJ4teyV6SZyx+CnwJuBQYA5zENhCT7sbCzPYA7gOeCuc/HmgA/tYrFS6e/sBrwFeBpq5mNrMdgIeBlcBhwEXAN4Gvxdqau+tR5g/gOeBXGWVvAFd3Yx33AH8s9WcpVSyAnwG3ANOBjaX+HKWIBbAv0AqMKXXdyyAW/wK0AdVpZZMBB4aU+vMUKCYbgeldzPNFYAPQkFb2H8BywkZWuR46Ailz4S/nccA/Mib9g+BII64dgLWFqlcp5BsLMzsZOIXg19U2Ic9YnA4sAj5mZovMbLGZ3WZmOxexqkWXZyxeJEimF5hZtZkNAM4HXnD3VUWrbPk5CnjK3dOPVh4CdgFGd7WwEkj5GwJUExxiplsJDI+zAjM7BZhK5beD73YszGwE8CvgXHdvLG71elU++8WewChgGsGR2LnAfsD9ZlbJ3wXdjoW7LwZOAK4CmoH1wIEEPzS2J8PJHrfUtJwqeafZ3mTesGNZyjowswnAncBF7v58MSpWAt2Jxe+Am9z92eJWqWS6E4sqoJ4gmT7p7k8RJJHDCc5/V7rYsTCz4cBvgNsJPvskoBG4p8KTaT6yxS1beQfbW6Aq0SqCc7WZvwZ2puMvh3bMbCLwd+AKd7+pONXrVfnEYgrwXTNLmFmC4EujX/i+kjvXyycWK4CEuy9MK3sDSAA5GyGUuXxi8WVgk7t/y91fdvcngc8Ax9G9U8OV7j2yxw26+H4BJZCy5+4twEsEh9vpTiBoaZKVmR1LkDyucvfrilbBXpRnLA4EDkl7XEHQOuUQ4H8LX8vekWcsngZqzGyvtLI9gRpgScEr2UvyjEVfgqSTLvV+e/pefAY4xsz6pJWdALwLLO5y6VK3FNAjVmuKTwEtwAUETS+vJ2hhMSqcfjXwaNr8k4BNwI8Jfl2kHkNL/Vl6OxZZlp/OttMKq7v7RRXBF+0sgqarHw5fPwtUlfrz9HIspgBJ4LvA3sChwIPAO0C/Un+eHsShP9GPpc0EP5gOAXbvJA47EhyF3EXQ5P/jBK2yvh5re6X+wHrE3jG+RPCLoDn8Ejg2bdqtwOKM957lsbi3613qWGRZdptJIPnEAhhBcOTVCLwP3AEMK/XnKFEspgFzwkTzAXA/sH+pP0cPYzCpk//9W3PE4UCCe4G2EJzm/C4xmvC6uzpTFBGR/GxP5/pERKSAlEBERCQvSiAiIpIXJRAREcmLEoiIiORFCURERPKiBFIhzOzKcICY1KPVzNaY2VwzuzXs8ypzmUlp81+Z5zavzDYoTUZ9JuXzmbpRj+mpusScf3FGrDp7LO5GHZ5ILZfv5yg2M/t9xuc7sZP5Lg7jeXGWaQPT/u5nFLvOGdsui/1N4qspdQUkbzXATuFjf+B8M7sB+KoX7uae74bPswhuQCqV6QR9FAFcWbpqlC8z6w+cllF8LkF3NpkuJuiVdwlwXca0gUR/99uAewtUxTjKZX+TmHQEUpmuIvjbjQAuJOiKGuArwHdSM7n7E+5u4ePKQlbA3a9MW/cThVx3T7n76LS6WcY0S3uMLlEVi+ETBP07pTs9TCwVr5z3t+2ZEkiF8sB77v7fBL/QUy41s0HQ+SksMzvYzP4Ujn/cbGarzWyOmf23mdWGp4zSj2KOS1vPE+E6sp5SSJ/PzE4ysxcsGGv5LTP7lln7sZbN7ENm9qvwtFOzma21YJzqk8xsdFiP47KsvyBHWWZ2mpk9Em63xczeMbPfmNnoGMueHNbZLRhjfNewfIyZ/U8Y3xYze9/M/mBmB2Usf2va5znazH4X1mN1OH+s8V5C56a9viV87kvQt1Fqe5PCuI0Ki0aln84L95G309Zzftr0W9PWc6SZ/dnMVlpwKvXd8LOMzvh8T6Qtv58FY3U3mtl7ZvZrC4ZTTZ2izHd/qzGzS8L9d5MF46G/bmbfM7N+GfXp1r4pMZS67xY9YvdxcyVRvzZXZpm+IG36Wd6xX5wrw7K+BP3+ZOsvxwk6Y5ueY/oTWeozKa0eqbL1BJ3VZS7/mbR5jyTokynbdq4kGBGts3p4N2KXdRngshzrX0Nav0jAE+nrAD5C0HeQE4w/Pywsn0jQiV22dTYBx6St89a0aWuzzP9IzM+3C0FPsk7Qt9N+aet4OG2+SZ3Uywn6kLoyx/Rbw3V8kqD792zzrAb2zRazTj7fr8P5pufYbqf7G8EgUn/LsexLpHWMmFbe5b6pR7yHjkC2HfPTXo/OMd8YghHcAL4F9AGGEnzxXU0wXsSt3v7UzyyPTh9MilmfHcL17QT8W1p5+i/l3xAkLICZwEiCc/AnAi+7++KwHrNSC3gnp6a6y8xGAd8L364jOMrZEfhBWLYTHa8PpJadQnBtoB54meALLTV2wq+ABoLrC+PCeT5MkLT7AL/spEpvA3sB+xB0cggw1YIRFbtyNtHZhD+4+3zg9fD9FDPbBaJTmkRdty9Ji+doD05z7pG23tvSpk83s77ATQRf3KlEVU8wlngLMIigB+hsnic45XoEQWeHAJ8xM+vB/jaNYF+B4O+wF0Gv0w+FZYcCX82yXJx9U2JQAtl2xP1bLiX4BQnBP+ClBF1br3T3y919S4Hqs5JgIKt1BBdjU0ZBcOqK4OI/wFvAl919ubuvd/cH3f2+AtWjMx8lakRyuwcj9G0g+KWbGhN7srUfJyHlfoIk8Rwwxd1XA5jZ3gRfqhB8zpcIvixfJkjSAAd2cmrqCndf5O5vAE+llY+K8VnSv/j+mPFcBXw6xjrimECQJCD4cp5P8PkeB+rC8swxOVK+7sEp1+eB18KyemBYD+pzctrr74fxW0mwT6eclGW5nPumxKcEsu3YL+31253N5O7vE1x4X0XwJXAVcDfwhpk9lTovXQBvuXtqgJ5NaeWpL+T0L44F7p4s0HbjGpr2+p3Ui7DOy8O3NURfmOlSF6ufD7+EUnbOMm82g7OULUh7nS1eWZnZWCB1bWUJUBuWzUubrVC/rON8vj6Z1x5CeX2+LmT9G9J+cKxsde5q35SYlEC2AWb2L8CHwrebgUdzze/uvyE41D+QoPXOz8NJEwmG+iyE1rTteZbp6cNl7mu5x6HOtnxPfZD2eutwrmZWDewavk0QXAvJ9Hj4/BVrf2/K+2mvH/b2Lb5Sp92q3H1ulnW2pr3uzudNTw6jCK7H/B9wZ1r5wWFSibP+XNPSP9+vcny+TZkLunu+ny+XrH/DjNfpdU7pat+UmJRAKpQFhpnZhQTXElJ+5O5rcyw3xMx+AhxF8M91P5B+uij9n291+DzKzHYqUNUBcPc3ic7T7wX83Mx2MbMBZjbVzE7PUg/M7JACVeFBolN555rZxPDo6wqia0SPdXJK7wyC01IQjLf+NYDw9FNqvPETLLhhb2D4GG9mVxCM/FYQYdI9O+bsn0l7nYrnkFTLsSzTAPbOOJr4J8HFcAhaaJ0d/r2GmtkEM/sxnVw3iqm7+9sDaa+/bWZ7mNkw4Edp5X/tQX2kK6W+iq9HvAe5W8ikHteTNpIY2VthjexiHWekLf9AlulXZqnPpLRl2rWeyVK+OK3sKILR4LLV48q0+b6RZfoT3Yjd1uUyynvSCmsY8Eba/J8Py48jaG2Vs1VROO+taeWjOymflONzTUmb79ks0/dJm76UcNha4BdZ6nVr2nKvZZk+PZz2aaIWX9ket3YWs2zlGZ+7W/sb+bfC6nLf1CPeQ0cglauNoPXQ6wQXAie4+1dT3245rAV+BrxAcB2kjaAp7T+Bs9393rR5v0LwD7qWInD3ZwjGa/4NwXnrVoImls8RtNpJ+SVBK60VFPB0lrtfDZwOPBZuNwEsA34LHOrur+dYdiVBU973wqKZZjbN3WcRtL66PVxXK0EyehW4Abi8UPWn/VHF7VnquJBgvHMIfjgcF76+kuBI6IPMZULnEgxxuiHLOn9PcKrzjwSnIRPhel4ErgF+2s3PkK5b+5sH1zFOA75OcES4meCi/jzg+wRD2nY4nSaFoyFtRUQkLzoCERGRvCiBiIhIXpRAREQkL0ogIiKSFyUQERHJixKIiIjkRQlERETyogQiIiJ5UQIREZG8/H/tt2C7abpZ4wAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "bin_edges = [np.array([0, 0.2, 0.4, 0.6, 0.8, 1]), np.array([0, 0.2, 0.4, 0.6, 0.8, 1])]\n",
    "\n",
    "num = 64\n",
    "histograms = []\n",
    "for attn_values, prediction in zip(Attn_Values, Prediction_Values):\n",
    "    hist, _, _ = np.histogram2d(attn_values, prediction, bins=bin_edges)\n",
    "    histograms.append(hist)\n",
    "    \n",
    "# Stack and compute mean and variance\n",
    "histograms = np.stack(histograms)\n",
    "mean_hist = np.mean(histograms, axis=0)\n",
    "var_hist = np.var(histograms, axis=0)\n",
    "\n",
    "\n",
    "mean_hist_norm = np.round(mean_hist/num,1)\n",
    "\n",
    "\n",
    "# Assuming `mean_hist_norm` and `var_hist_norm` are your 2D arrays of the same shape\n",
    "# If you prefer to show std instead of variance in the annotation, convert variance to std:\n",
    "std_hist_norm = np.round(np.sqrt(var_hist/num), 1)  # If you want std, otherwise use var_hist for variance.\n",
    "\n",
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(mean_hist_norm, dtype=object)\n",
    "for i in range(mean_hist_norm.shape[0]):\n",
    "    for j in range(mean_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"{mean_hist_norm[i, j]:.1f}\"#\\n(±{std_hist_norm[i, j]:.1f})\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(mean_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=14)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=14)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params1.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(std_hist_norm, dtype=object)\n",
    "for i in range(std_hist_norm.shape[0]):\n",
    "    for j in range(std_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"±{std_hist_norm[i, j]:.1f}\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(std_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=14)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=14)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params1_std.pdf\", bbox_inches='tight')\n",
    "plt.show()\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": 38,
   "metadata": {},
   "outputs": [],
   "source": [
    "separability = []\n",
    "with torch.no_grad():\n",
    "    for j in range(200):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        indices = inputs>9\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model_1(inputs)\n",
    "        #print(outputs.shape)\n",
    "        #outputs = nn.Softmax(dim=1)(outputs)\n",
    "        #print(outputs[np.arange(128),labels])\n",
    "        #print(outputs.shape)\n",
    "        \n",
    "        #print(outputs.shape,attn.shape)\n",
    "        \n",
    "        for i in range(outputs.shape[0]):\n",
    "            \n",
    "            #print(attn[i],labels[i],indices[i]\n",
    "                  \n",
    "            max_value = torch.max(attn[i][indices[i]]).item()\n",
    "            min_value = torch.min(attn[i][~indices[i]]).item()\n",
    "            #prediction.append(outputs[i,labels[i]].item())\n",
    "            \n",
    "            separability.append(max_value-min_value)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.0035905217997787985"
      ]
     },
     "execution_count": 41,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.array(separability).mean()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "metadata": {},
   "outputs": [],
   "source": [
    "# torch.manual_seed(1234)\n",
    "# model = Net_p1().to(device)\n",
    "model_1.eval()\n",
    "attn_values = []\n",
    "prediction  = []\n",
    "with torch.no_grad():\n",
    "    running_loss = 0.0\n",
    "    for j in range(500):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        indices = inputs>9\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model_1(inputs)\n",
    "        #print(outputs.shape)\n",
    "        outputs = nn.Softmax(dim=1)(outputs)\n",
    "        #print(outputs[np.arange(128),labels])\n",
    "        \n",
    "        for i in range(attn.shape[0]):\n",
    "            attn_values.append(sum(attn[i][indices[i]]).item())\n",
    "            prediction.append(outputs[i,labels[i]].item())\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {},
   "outputs": [],
   "source": [
    "attn_values = np.array(attn_values)[None,:]\n",
    "prediction = np.array(prediction)[None,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "num = 160\n",
    "histograms = []\n",
    "for attn, pred in zip(attn_values, prediction):\n",
    "    hist, _, _ = np.histogram2d(attn, pred, bins=bin_edges)\n",
    "    histograms.append(hist)\n",
    "    \n",
    "# Stack and compute mean and variance\n",
    "histograms = np.stack(histograms)\n",
    "mean_hist = np.mean(histograms, axis=0)\n",
    "var_hist = np.var(histograms, axis=0)\n",
    "\n",
    "\n",
    "mean_hist_norm = np.round(mean_hist/num,1)\n",
    "\n",
    "\n",
    "# Assuming `mean_hist_norm` and `var_hist_norm` are your 2D arrays of the same shape\n",
    "# If you prefer to show std instead of variance in the annotation, convert variance to std:\n",
    "std_hist_norm = np.round(np.sqrt(var_hist/num), 1)  # If you want std, otherwise use var_hist for variance.\n",
    "\n",
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(mean_hist_norm, dtype=object)\n",
    "for i in range(mean_hist_norm.shape[0]):\n",
    "    for j in range(mean_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"{mean_hist_norm[i, j]:.1f}\" #\\n(±{std_hist_norm[i, j]:.1f})\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(mean_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params1_test.pdf\", bbox_inches='tight')\n",
    "plt.show()\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": []
  },
  {
   "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": "markdown",
   "metadata": {},
   "source": [
    "## FACO"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  4%|█▌                                      | 100/2500 [01:11<28:22,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  250] loss:2.522\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  8%|███▏                                    | 200/2500 [02:22<27:17,  1.40it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  250] loss:1.030\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 12%|████▊                                   | 300/2500 [03:33<25:57,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  250] loss:0.564\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 16%|██████▍                                 | 400/2500 [04:44<24:54,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  250] loss:0.376\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 500/2500 [05:55<23:34,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  250] loss:0.280\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 24%|█████████▌                              | 600/2500 [07:06<22:49,  1.39it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  250] loss:0.222\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 28%|███████████▏                            | 700/2500 [08:17<21:13,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  250] loss:0.183\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 32%|████████████▊                           | 800/2500 [09:28<20:02,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  250] loss:0.156\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|██████████████▍                         | 900/2500 [10:39<18:51,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  250] loss:0.132\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|███████████████▌                       | 1000/2500 [11:50<17:42,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  250] loss:0.087\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 44%|█████████████████▏                     | 1100/2500 [13:01<16:36,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  250] loss:0.033\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 48%|██████████████████▋                    | 1200/2500 [14:12<15:42,  1.38it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  250] loss:0.015\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 52%|████████████████████▎                  | 1300/2500 [15:23<14:13,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  250] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 56%|█████████████████████▊                 | 1400/2500 [16:34<12:59,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  250] loss:0.006\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1500/2500 [17:45<11:47,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  250] loss:0.005\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 64%|████████████████████████▉              | 1600/2500 [18:56<10:51,  1.38it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  250] loss:0.004\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 68%|██████████████████████████▌            | 1700/2500 [20:07<09:25,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  250] loss:0.003\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 72%|████████████████████████████           | 1800/2500 [21:18<08:14,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  250] loss:0.003\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 76%|█████████████████████████████▋         | 1900/2500 [22:29<07:05,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  250] loss:0.002\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 2000/2500 [23:41<05:54,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  250] loss:0.002\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 84%|████████████████████████████████▊      | 2100/2500 [24:54<04:43,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2100,  250] loss:0.002\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 88%|██████████████████████████████████▎    | 2200/2500 [26:06<03:33,  1.41it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2200,  250] loss:0.002\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 92%|███████████████████████████████████▉   | 2300/2500 [27:17<02:20,  1.42it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2300,  250] loss:0.002\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 96%|█████████████████████████████████████▍ | 2400/2500 [28:27<01:10,  1.42it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2400,  250] loss:0.001\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2500/2500 [29:42<00:00,  1.40it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2500,  250] loss:0.001\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "n_epochs = 2500\n",
    "n_batches = 250\n",
    "\n",
    "batch_size= 32 # actual batch size is times 4 here 128\n",
    "seq_len = 64\n",
    "\n",
    "n_seeds = [1234]#,1235,1236]\n",
    "\n",
    "Attn_Values  = []\n",
    "Prediction_Values = []\n",
    "#deltas_list = []\n",
    "for seed in n_seeds:\n",
    "    torch.manual_seed(seed)\n",
    "    model_2 = Net_p2().to(device)\n",
    "    criterion = nn.CrossEntropyLoss()\n",
    "    optimizer = optim.SGD(model_2.parameters(),lr=1e-2)\n",
    "\n",
    "    running_loss = 0.0\n",
    "\n",
    "    common_token_attn = []\n",
    "    distinct_token_attn = []\n",
    "    #deltas = []\n",
    "    epsilon = 1e-8\n",
    "    #generate_heatmap(model,0)\n",
    "    for epoch in tqdm(range(n_epochs)):\n",
    "        model_2.train()\n",
    "        # record conservation error\n",
    "#         with torch.no_grad():\n",
    "#             C1 = model.value_layer.weight.matmul(model.value_layer.weight.t())\n",
    "#             C2 = model.prediction_layer.weight.t().matmul(model.prediction_layer.weight)\n",
    "            \n",
    "#             denom = max(torch.norm(C1).item(), torch.norm(C2).item(), epsilon)\n",
    "#             delta = torch.norm(C2 - C1).item() / denom\n",
    "#             deltas.append(delta)\n",
    "        for j in range(n_batches):\n",
    "            data = generate_data(batch_size,seq_len)\n",
    "\n",
    "            inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "            inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "\n",
    "            optimizer.zero_grad()\n",
    "\n",
    "            outputs,_ = model_2(inputs)\n",
    "            loss = criterion(outputs,labels)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            running_loss +=loss.item()\n",
    "        if (running_loss/n_batches)<=0.001:\n",
    "            break\n",
    "        if (epoch+1)%100 ==0:\n",
    "            print(f'[{epoch+1},{j+1:5d}] loss:{running_loss/(100*n_batches):.3f}')\n",
    "            running_loss = 0.0\n",
    "            #generate_heatmap(model,epoch+1)\n",
    "    print(\"Finished Training\")\n",
    "    #deltas_list.append(deltas)\n",
    "    attn,preds = generate_heatmap(model_2,epoch+1)\n",
    "    Attn_Values.append(attn)\n",
    "    Prediction_Values.append(preds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "metadata": {},
   "outputs": [],
   "source": [
    "# plt.plot(deltas_list[2])\n",
    "# plt.ylabel(r\"$\\Delta$\")\n",
    "# plt.xlabel(\"epochs\")\n",
    "# plt.savefig(\"conserve_quantity.pdf\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss:0.001\n",
      "accuracy:1.000\n"
     ]
    }
   ],
   "source": [
    "correct = 0 \n",
    "total = 0\n",
    "model_2.eval()\n",
    "with torch.no_grad():\n",
    "    running_loss = 0.0 \n",
    "    for j in range(500):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model_2(inputs)\n",
    "        loss = criterion(outputs,labels)\n",
    "        running_loss +=loss.item()\n",
    "        correct += sum(outputs.max(1)[1]==labels).item()\n",
    "        total += len(labels)\n",
    "print(f'loss:{running_loss/(j+1):.3f}')\n",
    "print(f'accuracy:{correct/total:.3f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "metadata": {},
   "outputs": [],
   "source": [
    "#  sum(((np.array(Attn_Values[0]) + np.array(Attn_Values[1]) +  \n",
    "#        np.array(Attn_Values[2]) +  np.array(Attn_Values[3]) +  \n",
    "#        np.array(Attn_Values[4]) ) /5)>=0.4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "metadata": {},
   "outputs": [],
   "source": [
    "#generate_heatmap(model,epoch+1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "metadata": {},
   "outputs": [],
   "source": [
    "attn_values_1 = np.mean(Attn_Values,axis=0)\n",
    "prediction_1 = np.mean(Prediction_Values,axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "bin_edges = [np.array([0, 0.2, 0.4, 0.6, 0.8, 1]), np.array([0, 0.2, 0.4, 0.6, 0.8, 1])]\n",
    "\n",
    "num = 64\n",
    "histograms = []\n",
    "for attn_values, prediction in zip(Attn_Values, Prediction_Values):\n",
    "    hist, _, _ = np.histogram2d(attn_values, prediction, bins=bin_edges)\n",
    "    histograms.append(hist)\n",
    "    \n",
    "# Stack and compute mean and variance\n",
    "histograms = np.stack(histograms)\n",
    "mean_hist = np.mean(histograms, axis=0)\n",
    "var_hist = np.var(histograms, axis=0)\n",
    "\n",
    "\n",
    "mean_hist_norm = np.round(mean_hist/num,1)\n",
    "\n",
    "\n",
    "# Assuming `mean_hist_norm` and `var_hist_norm` are your 2D arrays of the same shape\n",
    "# If you prefer to show std instead of variance in the annotation, convert variance to std:\n",
    "std_hist_norm = np.round(np.sqrt(var_hist/num), 1)  # If you want std, otherwise use var_hist for variance.\n",
    "\n",
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(mean_hist_norm, dtype=object)\n",
    "for i in range(mean_hist_norm.shape[0]):\n",
    "    for j in range(mean_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"{mean_hist_norm[i, j]:.1f}\"#\\n(±{std_hist_norm[i, j]:.1f})\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(mean_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params2.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZIAAAGKCAYAAAAmB8cMAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/MnkTPAAAACXBIWXMAAAsTAAALEwEAmpwYAAA9X0lEQVR4nO3deZxcVZ3//9en13QWSMIaSEgAiYb9S+KwC0nQEZDFH4rIqKCDCMPMiI7ICIpRZBCVTUZhcFBAUXQQGXBGUQhhU9YgDpEkSMgGgUjInra7q+vz++Pcyq2urqq+3bWn38/H4z6q6txbt059qro/de659xxzd0RERIaqqdYVEBGRxqZEIiIiJVEiERGRkiiRiIhISZRIRESkJEokIiJSEiUSEREpiRKJiIiURIlERERKokQiIiIlaanVC5tZK/A2YHxU9BbwZ3fvqVWdRERk8KreIjGzA83sHmAD8ALwSLS8AGwws3vM7KBq10tERIbGqjloo5kdDdwPLAfuBBYQWiIGjAP2A04HpgB/6+6PVq1yIiIyJNVOJL8DVgGnu3tvgW2agZ8Cu7v74VWrnIiIDEm1E8kW4ER3f2iA7WYBv3T3kdWpmYiIDFW1+0jWAXsm2G7PaFsREalz1T5r6w7gW2aWAn7m7n/NXmlmI4APAt8AflDluomIyBBU+9BWO/B94MNAN/AKsBZwwmnAewJthD6Ss929q2qVExGRIalqItn6ouH03pOBfQkJxAhnby0A7nX356teKRERGZKaJJJSmdm5wLkAR7//mun7Hnp2bSskItJgbrp4nJVrXw05RIq73+zuM9x9hpKIiEht1WUiMbMvmtmXal0PEREZWM3G2hrAHEK/yeU1roeIiAygXhPJXoREIiIida4uE4m7L691HUREJJl67SN5l5nNrXU9RERkYHWZSICdgGNqXQkRERlYVQ9tmdkeCTfdqaIVERGRsql2H8lSwnAoA7GE24mISI1VO5F0EmZDvGuA7WYQXbkuIiL1rdqJ5Hmg191vKbaRma1DiUREpCFUu7P9WWB6wm11HYmISAOodovk6wx8WAt3/zn1e0aZiIhkqWoicfdXgVer+ZoiIlJZ+tUvIiIlUSIREZGSKJGIiEhJlEhERKQkSiQiIlISJRIRESmJEomIiJREiUREREqiRCIiIiVRIhERkZLU5Zzt9c6AWTPaOfrgdnbYvomNW5xnF3Zz32OddPck38/+e7Vw/BEdTNypmVSvs3BZirvndbJmfbpidS83xSKmWMQUi9hwiIW5N/b8Ueddtbbqb+D02R3MmjGC5xZ1s+CVHnbdoZmZh7Tz0soU19+5KdGMXAdPbeXcU0excnUvjz/fzYh2Y/aMdtJpuPL2Dazf1Bifi2IRUyxiikWsXmNx08XjyjbCulokWaZOauGzZ47hmh9vZPGKVN5tJuzYxLHT25m/qJub79m8tfzNdWnOePdIZkxr5ekXi//MaGqCM44bydoNaa6+YyNd0eYLlvRwyVljeN+RHdxx/5ayva+hUCxiikVMsYgpFll1rOmrN6B3TmujyYy5z3T1KX/s+S66up1D92sfcB9TJ7UwdkwTj/+xe+uXAmDl6l4WL08xY1obTQ3wySgWMcUipljEhkssGuCjqAwDRnVYn6WjPbT0Otqt37pMG3DyhBbSaWfpqr6/QFK94YOdPKF5wNeePCE0BJe82v9XzCuvpehoN3YZX72PRrGIKRYxxSKmWBQ3bA9tjd+uiSvO3z7vuvNPG92v7NIb17NmQ5qxo5vY1Omkevs/b92mNHtPbKG5CXqL9H+NHW1bt++/D4+2aWLVm9XpRFMsYopFTLGIKRbFDdtEsn5zmuvu3NinbOLOzXxg1kjumruFlat7+20P0NZC3i8FQE8qfKhtrUZnV+HOr7bW8MVI5Tmsmr2PalEsYopFTLGIKRbFDdtEkuqFhcv6fjLpKKEvf723YOdZdwrGtOXfZ2tL+DC7e4qfQZFZ35In+kn3UU6KRUyxiCkWMcWiuGHbRzJU6zalGd1htOQ5tDl2dBMbt6SLNlPDPuLmaP99FG7G1hvFIqZYxBSL2HCJhRLJIC1blaKpyZgyoe/Pg5bm0NRd9nqBdmzOPgD22r3/T4w9d2uhs8t54636/yNRLGKKRUyxiA2XWCiRZFm8IsV5V60t2EwFeObFbtLuzJrR97S9ow5qp73NeGpBd5/y7UaFMypas74Di1ekWLcxzZEHttHeGpfvvlMzU/do4dmF3VubzbWiWMQUi5hiEVMsYsP2yva2Vjh4nwIHL/P4w0vdW4cz+NBxHcycHq5UfWFJuFJ11vR2Xn41xbU/6Xul6lknjOTwA9r7XbR0yNtbOeeUrCtV22D2O0fgDlfetmFrc7YaFIuYYhFTLGLbYix0ZXsZjOlo4hMnjUq8/aU3pljTE9L+zx4M49scdVA7++/dyuZO56H5Xdz3aGei4Q4A5i/q4cafb+b4I0Zw2swOenqdRdHYOdX8AwHFIptiEVMsYopFccO2RSIiMpyVs0WiPhIRESmJEomIiJREiUREREqSKJGY2YfMrHXgLUVEZLhJ2iL5CfCamV1tZvtWskIiItJYBnNoawfgQuD/zOx3Zna2mY2sTLVERKRRJE0knwZ+F9034DDgFmCVmd1kZjMqUTkREal/iRKJu9/g7kcBk4GLgfmEhDIG+CTwpJk9bWYnVqymIiJSlwZ11pa7r3T3bwL/ADyctcqA6cC9ZvbJMtZPRETqXOJEYmYjzeyTZvYs8HvgXdEqB34BPEZIKP9a9lqKiEjdSjTWlpl9B/g7wqGszGX1G4DvA99296Vm1gy8DuxRiYqKiEh9Sjpo4/lZ918GbgC+7+6bMoXu3mtma4DxZayfiIjUucGM/vswcB1wrxcY6dHd31GOSomISONImkj+n7s/X9GaiIhIQ0ra2f6cmeWdBszM5prZg2Wsk4iINJDBHNoqNHb9sZB4fhYREdnGlDRDopkdGN1NnEjMbARhQq3OrLIDgGnAq+7+eCl1EhGR6ip4aMvMvmxmvWbWGxeFx1nlz0XrXh/ohaLrUH5COG14o5ndEJV/F/gDcCfwiJk9aWbbl/CeRESkigZqkeQezip0eOu/E7zWF4BTgKuB9cA/m1kbcAbwKeApwhhe3wQuIQzFIiIida5YIllKPAzKMYTDV49krXdgDfAE8J0Er3U6cJm7fwvAzJ4Cfgtc5O7/GW3zx6g18vcokYiINISCicTdbwNuAzCzdFQ2s4TXmgg8nfX4SUIL56mc7Z4G5pTwOiIiUkVJO9v3LMNrvQlMynqcGUplUs52e0TbiohIAyiYSMzsXQDu/ghh+HjMbHKh7aPtipkHfMXMVgEbgasIAz1eZma/j8br2ovQP/L7wbwJERGpnWItknlAOtpmHsVP8fUB9gXwReBB4DfR45eAo4GfAS+b2VpgHCHJzBlgXyIiUicGc9ZWoTO2EnH3FWZ2MHAk0AY84O5dZvZe4Bxgf8JpxLe6+7JSXktERKqnWCL5KnEr5CvleDF330I4Uyu7rItkZ32JiEgdKnbW1pys+2VJJOViZucC5wIc/f5r2PfQs2tbIRGRYWxQU+1Wg5l90cy+VGwbd7/Z3We4+wwlERGR2ip21tbcQezH3X12GeoDoaPdgMvLtD8REamgYn0kx5JsMEZLuF1Se1Fix76IiFTPYMfaqjh3X17t1xQRkaEr1tled/0nIiJSf6qaLMysw8wuNLOHzOwNM+uOljeisgvNbGQ16yQiIqVJNERK5n4xAw2RYmaTgLnAFOBx4C7gLcLhs3HAvsA3gAvMbLYOcYmINIZqDpFyHdAJ7OPuS/NtYGZTgHuAa4HTBtifiIjUgaoNkQIcB3ykUBIBiAZuvAz4YYmvJSIiVVLNIVIGc4pwOU8nFhGRCqrmECkPAFeY2Qvu/kq+DaJDW5eTMx6XiIjUr6QTWwFgZjsA7wJ2JEw+9Yi7r0n49AuBh4DFZvYE8AKwltD6GA/sR5izfSnwmcHUS0REaidxIjGzOYR51NuyirvN7OtJWizuvtLMDiQMtngScCohgUBIKAuAi4DvRaMEi4hIA0iUSMzsIuCyPKvaCTMcbnL3qwfaj7t3AtdHi4iIbAOSXpB4QXTbCfwY+Hp020k4m+ufyl81ERFpBEkPbe1C6Ms4xd0fyBSa2buB+4GdK1A3ERFpAElbJAui2ydyyn8f3b5QnuqIiEijSZpI/hXoBf4hp/wfgB7gknJWSkREGsdgJrZaC1xpZv8IrAAmRstfgC8QrhMREZFhZigTW+0eLRk7RduKiMgwVK6JrTSkiYjIMKWJrUREpCRKFiIiUpLBjrV1IPB2oCN3nbvfXq5KiYhI40g6RMpowoRTMwts4sCwSSQGzJrRztEHt7PD9k1s3OI8u7Cb+x7rpLsn+X7236uF44/oYOJOzaR6nYXLUtw9r5M169MVq3u5KRYxxSKmWMSGQyySHtr6IjCLEJNCy7DxwdkdfHD2SFa92ctPH9jC/EXdzJrezj+cNjpxIA6e2so/fGA0rS1w97xOfvNUF/tMauGivxvD9qMbJ5yKRUyxiCkWseEQi6SJ5BRCq+PX0WMHriEMJb+Y8kx8VXNTJ7Vw08XjmDqpcENtwo5NHDu9nfmLuvmPezbz2PPd3DW3k/+a28k7JrcyY1rrgK/T1ARnHDeStRvSXH3HRh5+rov7n/gr3/7ZJrYbZbzvyH5HDqtOsYgpFjHFIqZYZNUx4XaTo9tzMgXu/jlCgpkKvF7metWtd05ro8mMuc909Sl/7PkuurqdQ/drH3AfUye1MHZME4//sZuurKbtytW9LF6eYsa0Npoa4DQIxSKmWMQUi9hwiUXSl8+0nV4HUgBmth3wXFR+UZnrVXEGjOqwPktHe3ibHe3Wb10mAJMntJBOO0tXpfrsL9UbPtjJE5oHfO3JE8IvmCWvpvqte+W1FB3txi7jq/fNUCxiikVMsYgpFsUlPWvrLWBXYAxhSJRdgRuAzARUu5W/apU1frsmrjh/+7zrzj9tdL+yS29cz5oNacaObmJTp5Pq7f+8dZvS7D2xheYm6C3S/zU2Oqa5blP/jdZt8mibJla9WZ1ONMUipljEFIuYYlFc0kTyZ0Ly2AN4kjC74UeidQ68WPaaVdj6zWmuu3Njn7KJOzfzgVkjuWvuFlau7u23PUBbC3m/FAA9qfChtrUanV2FL/Zvaw1fjFT/Hxh99lEtikVMsYgpFjHForikieSnhNbHBOBrwHuBEdG6vxKm4G0oqV5YuKzvJ5OOEvry13tZvCLPpwZ0p2BMW95VtLaED7O7p/iIMZn1LXmin3Qf5aRYxBSLmGIRUyyKS3Rgzd2/6+7Hu/tv3H0+sB/wGcLMiPu7+4OVrGQ9WbcpzegOoyXPoc2xo5vYuCVdtJka9hE3R/vvo3Aztt4oFjHFIqZYxIZLLAbdQ2Nm4wgtmdvd/TvuvqT81apfy1alaGoypkzo+/OgpTk0dZe9XqAdm7MPgL127/8TY8/dWujsct54q/7/SBSLmGIRUyxiwyUWiROJmc02s2cI144sBN40s6fNbFbFaldli1ekOO+qtQWbqQDPvNhN2p1ZM/qetnfUQe20txlPLejuU77dqHBGRWvWd2DxihTrNqY58sA22rNOI999p2am7tHCswu7tzaba0WxiCkWMcUipljEzH3gY2tmdjzw30Az/a9iTwEnu/uv+z2xCs67au2QDg62tcLB+xQ4eJnHH17q3jqcwYeO62Dm9BE8t6ibF5b0sOsOzcya3s7Lr6a49ieb+oypf9YJIzn8gHau+fHGPl+4Q97eyjmnjGLl6l4ef76bEW0w+50jcIcrb9uwtTlbDYpFTLGIKRaxbTEWN108rmw99Ek72/8ta9uV0ZKZIbElWl+TRDJUYzqa+MRJoxJvf+mNKdb0hLT/swfD+DZHHdTO/nu3srnTeWh+F/c92pl4Ypb5i3q48eebOf6IEZw2s4OeXmdRNHZONf9AQLHIpljEFIuYYlFc0hZJJ9AGXOzu38oq/zzwdaDL3Wtynf5QWyQiIsNZOVskSftIXotub8wp/250+2p5qiMiIo0maSK5Ibo9Mqf88Oj22+WpjoiINJqCfSRmdllO0SrgF2Z2N7CccJX7+wmtkbGVqqCIiNS3Yp3tcyBvX9CZOY87gMuAr5apTiIi0kAGOmur9jOmiIhIXSuWSApNqysiIrJVwUTi7g9XsyIiItKYkl6QCICZHQacAOwMrAb+x92frETFRESkMSROJGZ2E/DJnOJLzewmd7+gvNUSEZFGkeg6EjM7GziX0Pmeu5xnZh+rVAVFRKS+Jb0g8dzodhlwIeH6kU8DSwnJ5FPlrpiIiDSGpIe29idcU3KSu7+QKTSzh4A/RutFRGQYStoiyYyfvDKnfGXOehERGWaSJpIV0e23zGwsgJltD3wzZ72IiAwzSRPJLwl9IR8H1pjZeuAt4BOEQ173VaZ6IiJS75Imkq8RBmrMnKk1Juv+MuCKitRORETqXqJE4u5rgEOBWwijAKei2+8Bh7v7W0OtgJm9w8zOMLPTzWyvoe5HRERqY8CztsyslXjekYvdPfeixETM7J+AZne/Lno8ArgDOJV4cEg3s1uBc929dyivIyIi1TVgi8Tde4C50TKihNe6ANiU9fgq4L3AF4BDgOnAF4EPR7ciItIAkl5HshKYBGwo4bX2AF7KenwG8CV3vyar7Dkzc8IFjl8p4bVERKRKkna23xzdljIUymZg+6zH44Bn8mz3NDChhNcREZEqStoiaSec7nuDmZ0MPAd0Zm/g7gPNkPgQ4XThe6PHzxLmPHkkZ7tZhDPERESkASRNJF8innb33dGSa6BE8mXgCTO7C7gm2ued0QWOD0T7fy9wHvD5hPUSEZEaG8x8JMWm3c03t3vfDdxfNLPZwK3Ao1n7/OdoAegGvpY5s0tEROpf0kRSlml33f0ZMzsg2t+RwG6Efpo1wALgV6VckyIiItWXKJGUc9pdd3fi04lFRKTBFT1ry8y2N7Mvmtm90XJJZtDGWjKzc83sGTN75k9P3lrr6oiIDGsWGgh5VpjtADwB5A5b8gpwmLu/WZEKmX0xqtflSbY/76q1A/bPiIhIXzddPK5Yv/egFGuRfAnYm/5T6+5JZa88nxMtIiLSAIolkhMJZ2P9GfgX4KLovgHvq2Cd9qJ/K0hEROpUsc72SdHtye6+EMDM/pdwdtXESlXI3XUxoohIAynWImkDyCSR6P6L0d3WSlZKREQaR5Jh5CeR52LE3PIkLQkz6yAMyHgKsC9hvC2AtcCfgP8Gbnb3LUkqLyIitZfkOpKlOY89T7kPtK8o8cwFpgCPA3cRxu8yQkLZF/gGcIGZzdYhLhGRxpAkkeS2RrxA+UCuIwz0uI+7L837QmZTgHuAa4HTBrl/ERGpgWKJZDkJxtAahOOAjxRKIgDuvtTMLgN+WMbXFRGRCiqYSNx9SplfazBJSRcZiog0iKQTW5XDA8AVZrZnoQ2iQ1uXA7+tVqVERKQ0gxlGvlQXEia3WmxmTwAvEM7WcmA8sB9wGKET/zNVrJeIiJSgaonE3Vea2YHAucBJwKmEBAIhoSwgXD3/PZ3+KyLSOKrZIsHdO4Hro0VERLYB1ewjERGRbZASiYiIlESJRERESpK4j8TMjgA+CkwGRuSsdnefXc6KiYhIY0iUSMzsY8APCq1GFxCKiAxbSVsk/8rgx9YSEZFhIGkimUJodXwD+BGwGbVCRESE5IlkCTAN+Dd331jB+oiISINJetbWVYRDWx+uYF1ERKQBJW2RzALWATea2TnAIqAna727+9+XuW4iItIAkiaSs4j7RKZHS65hk0gMmDWjnaMPbmeH7ZvYuMV5dmE39z3WSXfPgE/fav+9Wjj+iA4m7tRMqtdZuCzF3fM6WbM+XbG6l5tiEVMsYopFbDjEwtwH7jM3s4Fq6u7eXJ4qDc55V62teqf/6bM7mDVjBM8t6mbBKz3sukMzMw9p56WVKa6/c1OisxAOntrKuaeOYuXqXh5/vpsR7cbsGe2k03Dl7RtYv6kxzmVQLGKKRUyxiNVrLG66eFzZzsRN2iIpOIfItmTqpBY+e+YYrvnxRhavSOXdZsKOTRw7vZ35i7q5+Z7NW8vfXJfmjHePZMa0Vp5+sfjPjKYmOOO4kazdkObqOzbSFW2+YEkPl5w1hvcd2cEd99d2AGTFIqZYxBSLmGKRVcckG7n7soGWSle0XrxzWhtNZsx9pqtP+WPPd9HV7Ry6X/uA+5g6qYWxY5p4/I/dW78UACtX97J4eYoZ09poaoDBaxSLmGIRUyxiwyUWgxpG3sxOBd4DjHf3M8zsaMIhwPnuvqkC9asYA0Z29G3ZdbTb1ttROeu2dDoOTJ7QQjrtLF3V9xdIqjd8sJMnDHyEb/KEEPYlr/b/FfPKayneMaWVXcY3serN6hz7VCxiikVMsYgpFsUlHSKlGbgbeB/xkChnAJ8HTgAuAG6qUB0rYvx2TVxx/vZ5151/2uh+ZZfeuJ41G9KMHd3Epk4n1dv/ees2pdl7YgvNTdBb5DMdO9q2bt9/Hx5tU70vhmIRUyxiikVMsSguaYvkQsKshrl+AJwYrWuoRLJ+c5rr7ux7beXEnZv5wKyR3DV3CytX9/bbHqCthbxfCoCeVPhQ21qNzq7CnV9treGLkcpzWDV7H9WiWMQUi5hiEVMsihvs6b/XE5JKxnPR7bQy1qkqUr2wcFnfTyYdJfTlr/cW7DzrTsGYtvz7bG0JH2Z3T/EzKDLrW/JEP+k+ykmxiCkWMcUiplgUl7SL5m3R7WU55a9Ft7uWpzr1b92mNKM7jJY8hzbHjm5i45Z00WZq2EfcHO2/j8LN2HqjWMQUi5hiERsusUiaSDKNs9xwvD26HcRlNY1t2aoUTU3GlAl9fx60NIem7rLXC7Rjc/YBsNfu/X9i7LlbC51dzhtv1f8fiWIRUyxiikVsuMQiaSJ5Mbq9KFNgZocBt0QPF5SzUrWyeEWK865aW7CZCvDMi92k3Zk1o+9pe0cd1E57m/HUgu4+5duNMnYZ30Rr1ndg8YoU6zamOfLANtpb4/Ldd2pm6h4tPLuwe2uzuVYUi5hiEVMsYopFLOmV7RcAN1B46Ph/dPcby1mxpIZ6ZXtbKxy8T4GDl3n84aXurcMZfOi4DmZOD1eqvrAkXKk6a3o7L7+a4tqf9L1S9awTRnL4Ae39Llo65O2tnHNK1pWqbTD7nSNwhytv27C1OVsNikVMsYgpFrFtMRa1uLL9u8C7gZPzrPslDXbGFsCYjiY+cdKoxNtfemOKNT0h7f/swTC+zVEHtbP/3q1s7nQemt/FfY92Jp6kZf6iHm78+WaOP2IEp83soKfXWRSNnVPNPxBQLLIpFjHFIqZYFJeoRQJgZgacTjjVd2dgNSGJ/BQY6e6bizy9Ymox1paISKOreovEzD7j7tcSksZPc9aNBX4FHF6uSomISONI2tl+tZmdn1toZjsBDwN/U9ZaiYhIwxjMUF83mNnHMw/MbBLwKHBA2WslIiINI2ln+1zCLIk3m9lfgaeAB4DJ0fpvVKBuIiLSAJK2SE4E7idckHgb8HtCEnHgc+7+hcpUT0RE6l3S+Ui6gFMIZ2m1ADsSrmb/mLtfU7nqiYhIvSt4aMvMcsfVAvgj4RBXB/AEsHdmO3f/akVqKCIida1YH8kcCl/JDnBUtGQokYiIDEMDdbYnvWBFFwWKiAxTxRLJzKrVQkREGlbBROLuD1ezIiIi0piSXkcCbB06/gTisbb+x92frETFRESkMSROJGZ2E/DJnOJLzewmd7+gvNUSEZFGkeg6EjM7GziX0Pmeu5xnZh+rVAVFRKS+Jb2y/dzodhlwIfB+4NPAUkIy+VS5KyYiIo0h6aGt/Qmn+J7k7i9kCs3sIcJFivtXoG4iItIAkrZIMnNMrswpX5mzfkjMrM3M/tnMditlPyIiUn1JE8mK6PZb0URWmNn2wDdz1g9VB3AtsHeJ+xERkSpLemjrl4Q+kY8DHzezTcDoaJ0D9w20AzN7pMjqZkJfy7+b2XrA3f2YhHUTEZEaKtgiMbOPZZ2N9TVgOfGZWmOy7i8DrkjwWkcB+wC9eZZ0tE0657GIiNS5Yi2SWwn/0G939zVmdighoZwA7AT8hdBSuczd30rwWpcClwALgS+4+7rMiuhw2VvAp929WMtFRETqzEB9JFsHbXT3N9z9k+6+u7u3Rbefcvc3kryQu18JHATsCSzKufZEgz6KiDSowczZXjJ3X+Lu7wU+A3zDzB42s2nVrIOIiJTXgJ3tBSa46mcwE1u5+4/N7H+BbwHPAbegVomISENKctbWlxPua1ATW0V9JOeY2W3ATSSf+0REROpIkkSS5B/8kFsT7v4osN9Qny8iIrWVJJF8peK1GCQzO5do/K+j338N+x56dm0rJCIyjJl7/saEmaUJFwY2V7VCZl+M6nV5ku3Pu2qt+lZERAbppovHla07YVATW1XJHMLhtESJREREaqseE8leqONdRKRhFEskH69aLbK4+/JavK6IiAxNwUTi7rdVsyIiItKYqnplu5l1mNmFZvaQmb1hZt3R8kZUdqGZjaxmnUREpDRV6yMxs0nAXGAK8DhwF2GgRgPGAfsC3wAuMLPZOsQlItIYqtnZfh3QCezj7kvzbWBmU4B7CJNcnValeomISAmqmUiOAz5SKIkAuPvSaGyvH1atViIiUpJEfSRmdpmZfanAuneZ2bsS7GYwFw7qIkMRkQaRtEUyh/DPPd9FgvMIE2ANtK8HgCvM7AV3fyXfBtGhrcuB3yasl4iI1FhJh7bMrDVzN8HmFwIPAYvN7AngBWAtIUGNJwzceBiwlDBfiYiINICCicTMjgGOySnLnZskMynVloFeyN1XmtmBhMEWTwJOJSQQCAllAXAR8D13H3B/IiJSH4q1SI4FshOHkX9uEgdeTPJi7t4JXB8tIiKyDRjo0FbmkJXnPM72JvD5stVIREQaSrFEciuhI90IFxI6MDNrvQNrgD+7e1eF6iciInWu2Fhby4BlAGZ2eyjyh6tVMRERaQyJztpy97MrXA8REWlQiU//NbMzgY8Ck4EROavd3fcuZ8VERKQxJEokZnYR8PVCq9GV6CIiw1bSFsmn0KyFIiKSR9L5SHYjtDr+ERjj7k05S3PlqigiIvUsaSJZEN3+yN03V6oyIiLSeJImkswV7p8zMx3iEhGRrZL2kXwe2AhcCpxjZi8DPVnr3d1nl7ty9cqAWTPaOfrgdnbYvomNW5xnF3Zz32OddPcM+PSt9t+rheOP6GDiTs2kep2Fy1LcPa+TNevTFat7uSkWMcUipljEhkMszH3gE67MLE3hM7OMkEhq0k9y3lVrq37G2OmzO5g1YwTPLepmwSs97LpDMzMPaeellSmuv3NTolPYDp7ayrmnjmLl6l4ef76bEe3G7BntpNNw5e0bWL+pMU6EUyxiikVMsYjVayxuunhc2Y4uDWYY+W3+kNbUSS189swxXPPjjSxekcq7zYQdmzh2ejvzF3Vz8z1xd9Gb69Kc8e6RzJjWytMvFv+Z0dQEZxw3krUb0lx9x0a6os0XLOnhkrPG8L4jO7jj/toOgKxYxBSLmGIRUyyy6phkozxnaQ3bs7beOa2NJjPmPtN3eLHHnu+iq9s5dL/2AfcxdVILY8c08fgfu7d+KQBWru5l8fIUM6a10ZS096qGFIuYYhFTLGLDJRYN8FFUhgGjOqzP0tEeGl0d7dZvXaY5NnlCC+m0s3RV318gqd7wwU6eMHBOnTwhNASXvNr/V8wrr6XoaDd2GV+9j0axiCkWMcUiplgUN6gZEqOJqd4OdOSuc/fby1Wpahi/XRNXnL993nXnnza6X9mlN65nzYY0Y0c3sanTSfX2f966TWn2nthCcxP0Fun/Gjvatm7ffx8ebdPEqjer04mmWMQUi5hiEVMsiks6RMpo4B76DiOfzYGGSiTrN6e57s6Nfcom7tzMB2aN5K65W1i5urff9gBtLeT9UgD0pMKH2tZqdHYV7vxqaw1fjFSew6rZ+6gWxSKmWMQUi5hiUVzSFskXgVmVrEi1pXph4bK+n0w6SujLX+8t2HnWnYIxbfn32doSPszunuJnUGTWt+SJftJ9lJNiEVMsYopFTLEoLumBtVMIrY5fR48duIYwO+Ji4Cvlr1p9WrcpzegOoyXPoc2xo5vYuCVdtJka9hE3R/vvo3Aztt4oFjHFIqZYxIZLLJImksnR7TmZAnf/HCHBTAVeL3O96tayVSmamowpE/r+PGhpDk3dZa8XaMfm7ANgr937/8TYc7cWOrucN96q/z8SxSKmWMQUi9hwiUXSRJI5APc6kAIws+2A56Lyi8pcr5pYvCLFeVetLdhMBXjmxW7S7sya0fe0vaMOaqe9zXhqQXef8u1GhTMqWrO+A4tXpFi3Mc2RB7bR3hqX775TM1P3aOHZhd1bm821oljEFIuYYhFTLGJJ+0jeAnYFxgB/ie7fAGSugtmt/FWrrLZWOHifAgcv8/jDS91098Brb6Z5eH4XM6eP4FOnwgtLwpWqs6a3s3h5D0//qe8X4/3HdHD4Ae19LlpKp+FnD27hnFNG8S9/NyZcqdoGs985go1bnF8+1lnOtzogxSKmWMQUi5hiUVzSRPJnQvLYA3gSOBX4SLTOgRfLXrMKG9PRxCdOGpV4+0tvTLGmJ6T9nz0Yxrc56qB29t+7lc2dzkPzu7jv0c7EM3zNX9TDjT/fzPFHjOC0mR309DqLorFz1lV56AfFIqZYxBSLmGJRXNKxti4A3gdcS+hgf4x4ut2/Aie5+4OVqmQxtRhrS0Sk0VV9rC13/w7wncxjM9sPOJnQX/Ird19SrgqJiEhjSXpB4hzg++6+HMDdXwGur2C9RESkQQxmYqslZvaAmX3YzAYeaUxERIaFwYz01UQYIuVHwCoz+3czm16ZaomISKNImkhOAH4IbCJcUzIWOB94ysyeN7N/rkz1RESk3iWdj+TX7n4WsAvwIcIAjt2EpHIA4WwuEREZhgY1jLy7/xX4LzN7ElgIfG6w+xARkW1L4iRgZjsApwMfBo4gtEYy5yFvLvQ8ERHZtiU9/fd/gdlZ22cSyO+A7wM/LX/VRESkESRtkbw36/4qwiRWP3D3xeWvkoiINJKkiSQF3EdoffzK3et//GYREamKpIlkN3d/s6I1ERGRhpR0rK03AczsVOA9wHh3P8PMjib0l8x3900Vq6WIiNStRNeRmFmzmd0L/Bw4D/hgtOrzwEPEQ8qLiMgwk/TK9gsJw8jnDjv8g6jspDLWSUREGkjSRHIWYQKr63LKM1PtTitXhUREpLEkTSRvi24vyyl/LbrdtTzVERGRRpM0kfRGt8055W+PbnvKUx0REWk0SRNJZk72izIFZnYYcEv0cEGSnZjZ7mY2x8y+Z2afMbPt82wzzczmJqyXiIjUWNJEchuhU/0LsHW++seBQ6LHPxxoB2Y2BXge+BKhc/5qYJGZzc7ZdDvgmIT1EhGRGkuaSL4L3Es8UGP28j/ATQn28TVgNbCnu+8K7AcsAv7XzM4cZL1FRKROJL0g0c3s/YS5SN4H7ExICr8EfuruXuz5kaOBi7PmfX/RzGYB3wFuN7Ox7v7dobwJERGpncTDyEfJ4s5oGYodgVdz9tkLnGdma4EbzGwMMG+I+xcRkRoomEjM7F2D2ZG7PzLAJssJh7MezfPcL5jZJuBK4FeDeV0REamtYi2SecQd6wPxAfYF8AjwdxToT3H3K8xsI5q2V0SkoQz0zz93SJRS3AycYWY7uPuafBu4+7fNbDXwt2V8XRERqaBiieQr5Xwhd38WeDbBdqX0w4iISJUVTCTuXtZEUk5mdi5wLsDR77+GfQ89u7YVEhEZxpJeRwKAmbWY2d+Y2fGVqpCZfdHMvlRsG3e/2d1nuPsMJRERkdpKnEjM7IOE03d/T5h2FzN70MyWmNl7ylinOdEiIiINIOnEVkcDPyFcC5K5oh3CVe1TgA+UsU57RYuIiDSApC2SL0TbLsopfyC6PbxcFXL35e6+rFz7ExGRykqaSA4jXCuSOxPikuh297LVSEREGkrSRDIqul2eU96Rc1uUmXWY2YVm9pCZvWFm3dHyRlR2oZmNTFgnERGpA0nH2noVmEz/Q1ifi25XDrQDM5sEzCX0qTwO3AW8RehvGQfsC3wDuMDMZmcGdxQRkfqWNJHcD3wKuCdTYGYLgX0Ih7zuT7CP64BOYB93X5pvg2jOknsIw6SclrBuIiJSQ0kPbX0NWAOMJR5/ax9Ca+ItwmCLAzkOuLRQEgGI1l0WbSsiIg0gUSJx91eBI4HfAGlCAklHj4+O1g+4m0HUazDbiohIDQ1mPpLFwHvNbAQwHnjL3f86iNd6ALjCzF5w91fybRAd2roc+O0g9isiIjWUOJFkRMnjtcxjMzsQuNLdTxzgqRcCDwGLzewJ4AVgLaH1MZ4wV8lhwFLgM4Otl4iI1EbRRBKdinsKsAdhat373P3NaN1E4OvAGSQYbt7dV0ZJ51zC9SinEhIIhISyALgI+J67bxnKmxERkeorNkPiROAxYFJW8SYzOxFoA34BjCYkkUR9Gu7eCVwfLSIisg0o1iKZQ2iJZBsDfJ9wAeKYqGwD8K2y10xERBpCsUQym9DSWA7cTWh5vB94W7S+F/gOcHmhGQ9FRGTbVyyRTIhu3+vuiwDM7GbgT4QE89FoNkMRERnGil1H0gaQSSLR/YVZ6++qVKVERKRxDHj6r5ldVmDVJWbxyVru/tVyVUpERBpHkutIvpzz2AuUK5GIiAxDAyWSAa8PiWhIExGRYapYIvlK1WohIiINq2AicXclEhERGdCgx9qScLxv1ox2jj64nR22b2LjFufZhd3c91gn3T3J97P/Xi0cf0QHE3dqJtXrLFyW4u55naxZn65Y3ctNsYgpFjHFIjYcYmHujd29cd5Va6v+Bk6f3cGsGSN4blE3C17pYdcdmpl5SDsvrUxx/Z2bEnUYHTy1lXNPHcXK1b08/nw3I9qN2TPaSafhyts3sH5TY3wuikVMsYgpFrF6jcVNF49L2gc+ILVIskyd1MJnzxzDNT/eyOIVqbzbTNixiWOntzN/UTc337N5a/mb69Kc8e6RzJjWytMvFv+Z0dQEZxw3krUb0lx9x0a6os0XLOnhkrPG8L4jO7jj/tqOW6lYxBSLmGIRUyyy6ljTV29A75zWRpMZc5/p6lP+2PNddHU7h+7XPuA+pk5qYeyYJh7/Y/fWLwXAytW9LF6eYsa0Npoa4JNRLGKKRUyxiA2XWDTAR1EZBozqsD5LR3to6XW0W791mTbg5AktpNPO0lV9f4GkesMHO3lC84CvPXlCaAguebX/r5hXXkvR0W7sMr56H41iEVMsYopFTLEobtge2hq/XRNXnL993nXnnza6X9mlN65nzYY0Y0c3sanTSfX2f966TWn2nthCcxP0Fun/Gjvatm7ffx8ebdPEqjer04mmWMQUi5hiEVMsihu2iWT95jTX3bmxT9nEnZv5wKyR3DV3CytX9/bbHqCthbxfCoCeVPhQ21qNzq7CnV9treGLkcpzWDV7H9WiWMQUi5hiEVMsihu2iSTVCwuX9f1k0lFCX/56b8HOs+4UjGnLv8/WlvBhdvcUP4Mis74lT/ST7qOcFIuYYhFTLGKKRXHDto9kqNZtSjO6w2jJc2hz7OgmNm5JF22mhn3EzdH++yjcjK03ikVMsYgpFrHhEgslkkFatipFU5MxZULfnwctzaGpu+z1Au3YnH0A7LV7/58Ye+7WQmeX88Zb9f9HoljEFIuYYhEbLrFQIsmyeEWK865aW7CZCvDMi92k3Zk1o+9pe0cd1E57m/HUgu4+5duNCmdUtGZ9BxavSLFuY5ojD2yjvTUu332nZqbu0cKzC7u3NptrRbGIKRYxxSKmWMSG7ZXtba1w8D4FDl7m8YeXurcOZ/Ch4zqYOT1cqfrCknCl6qzp7bz8aoprf9L3StWzThjJ4Qe097to6ZC3t3LOKVlXqrbB7HeOwB2uvG3D1uZsNSgWMcUipljEtsVY6Mr2MhjT0cQnThqVePtLb0yxpiek/Z89GMa3Oeqgdvbfu5XNnc5D87u479HOxOPpz1/Uw40/38zxR4zgtJkd9PQ6i6Kxc6r5BwKKRTbFIqZYxBSL4oZti0REZDgrZ4tEfSQiIlISJRIRESmJEomIiJREiUREREqiRCIiIiVRIhERkZIokYiISEmUSEREpCRKJCIiUhIlEhERKYkSiYiIlESJRERESqJEIiIiJVEiERGRkiiRiIhISZRIRESkJEokIiJSEiUSEREpiRKJiIiURIlERERKokQiIiIlUSIREZGSKJGIiEhJlEhERKQkSiQiIlISJRIRESmJEomIiJTE3L3WdZAyMbNz3f3mWtejHigWMcUipljEyhkLtUi2LefWugJ1RLGIKRYxxSJWtlgokYiISEmUSEREpCRKJNsWHfuNKRYxxSKmWMTKFgt1touISEnUIhERkZIokdQ5M5tkZneZ2Xoz22Bmd5vZHgmeN8PMbjazhWa2xcyWm9kdZrZnNepdCUONRZ79fMHM3Mweq0Q9q6HUWJjZNDP7LzN708w6zWyRmX26knWulFJiYWZ7mNlt0d/HFjNbbGZfM7NRla53JZjZRDO7wcx+H70fN7MpCZ87wsy+aWarou/E783sXUmeq0RSx8xsJDAXeAdwFvBRYB/goQRf9DOA/YBvA8cD/wocAjxjZpMqVukKKTEW2fvZC7gUWF2JelZDqbEwsxnAk0A7cA5wAnA10FypOldKKbGI1j8AvAv4EnAi8J/AvwDfr2C1K+ltwOnAWuDRQT73FuCTwGXA+4BVwP1mdvCAz3R3LXW6AJ8GeoG3ZZXtCaSAzw7w3J3ylE0G0sBXa/3eqhmLnP3cD/wHMA94rNbvqwbfiyZgAfCLWr+POojFewAH3pNT/vXo+SNr/f6GEI+mrPvnRO9vSoLnHRRt+/GsshZgEXDvQM9Xi6S+nQw84e5/zhS4+yvA48ApxZ7o7n/JU7YM+Auwe5nrWQ1DjkWGmZ1JaJV9oSI1rJ5SYnEssC9wTcVqV12lxKItut2QU76OkHCtTHWsGndPD/GpJwM9wE+z9pUC7gT+1szaiz1ZiaS+7Qe8kKd8AeGfwaCY2TRgZ+DFEutVCyXFwszGAdcCn3f3t8pct2orJRZHRbcjzOwJM+sxs9Vm9m0z6yhrLaujlFg8ALwEXGVm+5rZaDObRWjl3OTum8tb1bq2H/CKu2/JKV9ASLhvK/ZkJZL6Np5wrDPXW8C4wezIzFqAmwgtkltKr1rVlRqLbwKLgVvLWKdaKSUWu0W3PwV+A7wb+AbhMMiPy1XBKhpyLNz9r4TEmjnctxF4EPgl8I/lrWbdKxbHzPqCWspeHSm3fBf6DKXJ/e/AEcCJ7p7vC9MIhhQLMzsa+BhwiEcHf7cBQ/1eZH48/sjdL4vuzzOzZuDrZravu/+pLDWsnqF+L0YQEurOhE765cDfEDqbU8D5ZaxjvTNK+F+jRFLf1pL/l8A48v96yMvMriQM0HaWu/+mTHWrtlJi8R+EVthKMxsblbUAzdHjTnfvKlM9q6GUWKyJbn+bU/4bQifzwUAjJZJSYvH3hD6jt7n7y1HZI2a2HrjZzG5y9+fLVtP69haQ75TpcVnrC9Khrfq2gHDsMte+JPxjN7NLCaf+ftrdf1jGulVbKbGYBpxH+MeSWY4EDovuN9ovz1JisSC6zf31mfnlOdTO2lopJRYHAGuzkkjGU9HttBLr1kgWAHtGp1Nn2xfoBv7c/ykxJZL6di9wWHTtAwDRxUVHRuuKMrN/Br4GXOruN1SqklVSSixm5lmeJ3TSzgTuqkB9K6mUWPwK6ALem1P+t9HtM2WqY7WUEovXgXFmltuRfGh0+2q5KtkA7gVagQ9mCqJ+1Q8BvxmwxV7r8561FD23exThl8D/EU5lPJnwD3AJMDpru8mEY7qXZZWdQfh1+SvCL+/sZd9av7dqxqLA/ubRuNeRlBQL4MtR+b8BxxFarJ3ArbV+b9WMBTCFcOrvYsLFjDOBi6KyZ8i6JqORFuAD0XIjoeV5fvT4mAG+F3cSWujnALMJP7D+SuhbLP6atX7TWgb8UuwB/Dz6cm8E7iHnAqPoD8KBOVllt0Zl+ZZ5tX5f1YxFgX01bCIpNRaEw1ifjf4BdwPLgK8CrbV+XzWIxb7Az4AVUTJdDHwLGFfr91VCPIr+3ReJRQfh+qLXowTyJHBsktfU6L8iIlIS9ZGIiEhJlEhERKQkSiQiIlISJRIRESmJEomIiJREiUREREqiRNJAzGxONHVmZukxs7fMbIGZ3WpmR+Z5zrFZ288Z4mvOMbOzB6jPsUN5T4Oox9mZuiTcfmlOrAotSwdRh3mZ5w31fVSamf0k5/0dX2C7C6N4Xphn3disz/3UStc557Xr4vsmg6NBGxtbC2FQtXGEC6vOMrMbCONqleuf3Zej24ep7RDsZwPHRPfn1K4a9cvMRhOu7M72UcLoBrkuJFzhvAy4LmfdWOLP/TbCBX7VUi/fNxkEtUga11cIn98EwoCE66PyfyLMPw2Au89zd4uWOeWsgLvPydr3vHLuu1TuPiWrbpazzrKWKTWqYiWcBuQOundKlGAaXj1/34Y7JZIG5sHr7v4fhF/sGReb2XgofGjLzA4ys7vN7FUz6zKzNWY238z+w8xao0NJ2a2aY7L2My/aR95DDdnbmdkJZva0mXWa2ctm9nkz6/OP3czeZmbfiw5HdZnZWjP7ffTcKVE9jsmz/7K0uszsZDN7IHrdbjNbbma3RIP/DfTcE6M6u5ktNLPdo/JpZvbDKL7dFmYhvMvMDsx5/q1Z7+cIM/tRVI810fa7DuKtfDTr/g+i25HA/5f1esdGcZscFU3OPswXfUdeydrPWVnrb83az2Fm9gsze8PCIdbXovcyJef9zct6/jvM7F4z22hmr5vZf5rZdtF2pXzfWszsM9H3d7OZ/dXM/mRmXzWzUTn1GdR3UxKq9bgwWgY1hs4c4nFz5uRZvyhr/QejsmNzn0P45/IXCo/JM5qQmAYasye7Psdm1SNTtp4wcGTu8z+Ste1hhPGR8r3OHOJxgfIug4hd3ucQ5m8vtP+3yBrgkjA+19Z9AO8hjEnkhEEDd4nKjwK2FNhnJ3B01j5vzVq3Ns/2DyR8f7sBvdFz5gPvyNrHb7O2O7ZAvRxYmvOZ5i63Rvs4nTDoX75t1gBvzxezAu/vP6Ptzi7yugW/b0Az8L9FnvssMGoo300tyRe1SLYtC7PuTymy3TRgx+j+54ERwE6Ef4BXAil3v9X7HhJ62OPDCscmrM920f7G0Xfq0uxfzrcQEheEqYAnEo7RHw885+5Lo3o8nHmCFzhkNVhmNpkwWCHAOkKrZ3vC0PtE9b6uwHNnEfoO2oHnCP/Y3ohWf48wAN4yYHq0zf8jJO8RwHcKVOkVYG9gKrA6KpttZhMSvJ0ziY8w3OXuC4nn45hlZrtBfKgzqhvAsqx4TvFw+HPPrP3elrX+bAvzVdxI+AeeSVjthJFzuwmTTH2zQB2fIhyKPZQwlD3AR8zMSvi+nUH4rkD4HPYGdgXuj8oOIczBnivJd1MSUiLZtiT9PFcQflFC+EO8GJgFvOHul3iYy7oc3iAMVb2O0GmbMRnCIS3CSQIALwMXuPur7r7e3X/t7v9dpnoU8rfEJ5zc7u6PuPsGwi/fN6PymRamZM11HyFZPAnMcvc1AGa2D+GfK4T3+Szhn+ZzhGQNcECBQ1aXufsSd38JeDSrfHKC95L9D/DnObdNwIcT7COJI4lnJDyE8OOlC3gIaIvK313guf/i4VDsU4S5YCAkoV1KqM+JWfcvj+L3BuE7nXFCnucV/W7K4CiRbFvekXX/lUIbuftqQgf9m4R/Bl8hzF39kpk9mjluXQYvu3tvdH9zVnnmH3P2P5BF7l7t2fl2yrq/PHMnqnNmUqMW8k/lmunUfir6Z5Sxc8LX3iFP2aKs+/nilZeZ7Q9k+l6WAa1R2YtZm5Xrl3aS9zcit28iMqT3N4C8nyFxiwvy13mg76YMghLJNsLMPgBkZnrbAjxYbHt3v4VwCOAAwtk+345WHQVcUKZq9WS9nudZ/0bW/bebWbHvY77nl+ovWfe3zldtZs3A7tHDFPnnq34ouv0n63tty+qs+7/1vmeIZQ7HNbn7Avrrybo/mPebnSQmE/pr/g/4cVb5QVFySbL/Yuuy39/3iry/zblPdPehvr9i8n6GOfez65wx0HdTBkGJpIFZsIuZnUfoa8j4uruvLfK8Hc3sW8DhhD+y+4Dsw0jZf4RrotvJZjauTFUHwN3/THwcf2/g22a2m5mNMbPZZnZKnnpgZgeXqQq/Jj7E91EzOypqjV1G3Ic0t8ChvlMJh6sAvmxmnwWIDkstjsrfbeHCv7HRMsPMLiPMRFcWUfI9M+HmH8m6n4nnjpkzzfKsA9gnp3XxO0KnOYQzus6MPq+dzOxIM/smBfqVEhrs9+2XWfcvNbM9zWwX4OtZ5f9TQn0kiVr39mtJvlD8jJrMcj2ECcui5xybtW5OVDZxgH2cmvX8X+ZZPydPfY7Nek6fs23ylC/NKjsc2FSgHnOytvtcnvXzBhG7rc/LKS/lrK1dgJeytv9kVH4M4eysomchRdvemlU+pUD5sUXe16ys7Z7Is35q1voVRNPHAv+ep163Zj3vhTzrz47WfZj4DLF8y62FYpavPOd9D+r7xtDP2hrwu6kl+aIWSWPrJZxt9CdCh+GR7v7pzH+5ItYC1wJPE/pJegmn4P4OONPd78na9p8If6hrqQB3/z1wMKFFtYxwyGE9oRP7qaxNv0M4q2sVZTzM5e5XEub6nhu9bgpYCXyfMFf1n4o89w3CKcCvR0U3mdkZ7v4w4Wyt26N99RCS0h+BG4BLylV/+rYybs9Tx8XAE9HDifQdHeBO+h4ayvZR4BHC9LW5+/wJ4RDozwmHJ1PRfp4BrgKuHuR7yDao75uHfo6TgX8htBC3EDr/XwQuB97leQ6zSXlpql0RESmJWiQiIlISJRIRESmJEomIiJREiUREREqiRCIiIiVRIhERkZIokYiISEmUSEREpCRKJCIiUpL/H5PLLMfIiRZWAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(std_hist_norm, dtype=object)\n",
    "for i in range(std_hist_norm.shape[0]):\n",
    "    for j in range(std_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"±{std_hist_norm[i, j]:.1f}\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(std_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params2_std.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 58,
   "metadata": {},
   "outputs": [],
   "source": [
    "# torch.manual_seed(1234)\n",
    "# model = Net_p1().to(device)\n",
    "model_2.eval()\n",
    "attn_values = []\n",
    "prediction  = []\n",
    "with torch.no_grad():\n",
    "    running_loss = 0.0\n",
    "    for j in range(500):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        indices = inputs>9\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model_2(inputs)\n",
    "        #print(outputs.shape)\n",
    "        outputs = nn.Softmax(dim=1)(outputs)\n",
    "        #print(outputs[np.arange(128),labels])\n",
    "        \n",
    "        for i in range(attn.shape[0]):\n",
    "            attn_values.append(sum(attn[i][indices[i]]).item())\n",
    "            prediction.append(outputs[i,labels[i]].item())\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(64000, 16000, 16000)"
      ]
     },
     "execution_count": 59,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "128*500, len(attn_values), len(prediction),"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "metadata": {},
   "outputs": [],
   "source": [
    "attn_values = np.array(attn_values)[None,:]\n",
    "prediction = np.array(prediction)[None,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "num = 160\n",
    "histograms = []\n",
    "for attn, pred in zip(attn_values, prediction):\n",
    "    hist, _, _ = np.histogram2d(attn, pred, bins=bin_edges)\n",
    "    histograms.append(hist)\n",
    "    \n",
    "# Stack and compute mean and variance\n",
    "histograms = np.stack(histograms)\n",
    "mean_hist = np.mean(histograms, axis=0)\n",
    "var_hist = np.var(histograms, axis=0)\n",
    "\n",
    "\n",
    "mean_hist_norm = np.round(mean_hist/num,1)\n",
    "\n",
    "\n",
    "# Assuming `mean_hist_norm` and `var_hist_norm` are your 2D arrays of the same shape\n",
    "# If you prefer to show std instead of variance in the annotation, convert variance to std:\n",
    "std_hist_norm = np.round(np.sqrt(var_hist/num), 1)  # If you want std, otherwise use var_hist for variance.\n",
    "\n",
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(mean_hist_norm, dtype=object)\n",
    "for i in range(mean_hist_norm.shape[0]):\n",
    "    for j in range(mean_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"{mean_hist_norm[i, j]:.1f}\" #\\n(±{std_hist_norm[i, j]:.1f})\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(mean_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params2_test.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "# num =160\n",
    "\n",
    "\n",
    "# plt.figure(figsize=(6,6))\n",
    "# im = plt.hist2d(np.array(attn_values),np.array(prediction),[[0,0.2,0.4,0.6,0.8,1],[0,0.2,0.4,0.6,0.8,1]])\n",
    "\n",
    "\n",
    "# ax = sns.heatmap(np.round(im[0].transpose()/num,1),vmin=5,vmax=70,annot=True,fmt=\"g\",cmap=sns.color_palette(\"coolwarm\"),\n",
    "#                  yticklabels=[0.2,0.4,0.6,0.8,1.],\n",
    "#                  xticklabels=[0.2,0.4,0.6,0.8,1],annot_kws={\"size\":18},cbar=False)\n",
    "# ax.invert_yaxis()\n",
    "\n",
    "# plt.xlabel(r\"distinct token attention\",fontweight=\"bold\",fontsize=20)\n",
    "# plt.ylabel(r\"true token probability\",fontweight=\"bold\",fontsize=20) # change xlabel based on algo\n",
    "# # plt.xticks([1,2,3,4,5],weight=\"bold\",fontsize=18)\n",
    "# # plt.yticks([1,2,3,4,5],weight=\"bold\", va=\"top\",fontsize=18)\n",
    "# plt.savefig(\"heatmap_Setting_1_len_\"+str(seq_len)+\".pdf\",bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# import numpy as np    \n",
    "# hist, bin_edges = np.histogram(attn_values, bins = [0,0.2,0.4,0.6,0.8,1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# hist,bin_edges"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# 7697+8278+12+13"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# temp_1 = model.value_layer.weight.cpu().detach().numpy()\n",
    "# temp_2 = model.prediction_layer.weight.cpu().detach().numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# U1,S1,V1 = np.linalg.svd(temp_1)\n",
    "# U2,S2,V2 = np.linalg.svd(temp_2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# V1.T[:,4]@U2[:,4]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# S2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# S1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "metadata": {},
   "outputs": [],
   "source": [
    "separability_1 = []\n",
    "separability_2 = []\n",
    "with torch.no_grad():\n",
    "    for j in range(200):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        indices = inputs>9\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        \n",
    "        outputs_1, attn_1 = model_1(inputs)\n",
    "        outputs_2, attn_2 = model_2(inputs)\n",
    "\n",
    "        \n",
    "        for i in range(outputs_1.shape[0]):\n",
    "            \n",
    "            #print(attn[i],labels[i],indices[i]\n",
    "                  \n",
    "            max_value = torch.max(attn_1[i][indices[i]]).item()\n",
    "            min_value = torch.min(attn_1[i][~indices[i]]).item()\n",
    "            #prediction.append(outputs[i,labels[i]].item())\n",
    "            \n",
    "            separability_1.append(max_value-min_value)\n",
    "            \n",
    "        for i in range(outputs_2.shape[0]):\n",
    "            \n",
    "            #print(attn[i],labels[i],indices[i]\n",
    "                  \n",
    "            max_value = torch.max(attn_2[i][indices[i]]).item()\n",
    "            min_value = torch.min(attn_2[i][~indices[i]]).item()\n",
    "            #prediction.append(outputs[i,labels[i]].item())\n",
    "            \n",
    "            separability_2.append(max_value-min_value)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(0.0036011230207805057, 0.0006917548140273317)"
      ]
     },
     "execution_count": 67,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.array(separability_1).mean(), np.std(np.array(separability_1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(0.10556412973190163, 0.022467949260446995)"
      ]
     },
     "execution_count": 68,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.array(separability_2).mean(), np.std(np.array(separability_2))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## CAFO"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Net_p3(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(Net_p3,self).__init__()\n",
    "        self.key_query_layer = nn.Linear(50,50,bias=False)\n",
    "        #self.query_layer = nn.Linear(50,50,bias=False)\n",
    "        self.value_layer = nn.Linear(50,50,bias=False)\n",
    "        self.prediction_layer = nn.Linear(50,50,bias=False)\n",
    "        \n",
    "        #self.key_query_layer.weight = nn.Parameter(torch.zeros(50,50))\n",
    "        self.key_query_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "        #self.query_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "        self.value_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "        self.prediction_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "    def forward(self,x):\n",
    "        batch,seq_len,vocab_size = x.shape\n",
    "        \n",
    "        key = self.key_query_layer(x)\n",
    "        query = torch.unsqueeze(x[:,-1],dim=1)\n",
    "        \n",
    "        #print(key.shape,query.shape)\n",
    "        scores = (query@key.transpose(-2,-1))*(1/math.sqrt(key.size(-1)))\n",
    "        \n",
    "        \n",
    "        \n",
    "        \n",
    "        attn = nn.Softmax(dim=-1)(scores)\n",
    "        x1 = self.value_layer(x)\n",
    "        context =torch.matmul(attn,x1)\n",
    "    \n",
    "        output = self.prediction_layer(context)\n",
    "        #print(output.shape)\n",
    "        return output[:,0,:],attn[:,0,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_epochs = 5000\n",
    "n_batches = 200\n",
    "\n",
    "batch_size= 32 # actual batch size is times 4 here 128\n",
    "seq_len = 64\n",
    "\n",
    "n_seeds = [1234,1235,1236,1237,1238]\n",
    "\n",
    "Attn_Values  = []\n",
    "Prediction_Values = []\n",
    "deltas_list = []\n",
    "for seed in n_seeds:\n",
    "    torch.manual_seed(seed)\n",
    "    model = Net_p3().to(device)\n",
    "    criterion = nn.CrossEntropyLoss()\n",
    "    optimizer = optim.SGD(model.parameters(),lr=1e-2)\n",
    "\n",
    "    running_loss = 0.0\n",
    "\n",
    "    common_token_attn = []\n",
    "    distinct_token_attn = []\n",
    "    deltas = []\n",
    "    epsilon = 1e-8\n",
    "    #generate_heatmap(model,0)\n",
    "    for epoch in tqdm(range(n_epochs)):\n",
    "        model.train()\n",
    "        # record conservation error\n",
    "        with torch.no_grad():\n",
    "            C1 = model.value_layer.weight.matmul(model.value_layer.weight.t())\n",
    "            C2 = model.prediction_layer.weight.t().matmul(model.prediction_layer.weight)\n",
    "            \n",
    "            denom = max(torch.norm(C1).item(), torch.norm(C2).item(), epsilon)\n",
    "            delta = torch.norm(C2 - C1).item() / denom\n",
    "            deltas.append(delta)\n",
    "        for j in range(n_batches):\n",
    "            data = generate_data(batch_size,seq_len)\n",
    "\n",
    "            inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "            inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "\n",
    "            optimizer.zero_grad()\n",
    "\n",
    "            outputs,_ = model(inputs)\n",
    "            loss = criterion(outputs,labels)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            running_loss +=loss.item()\n",
    "        if (running_loss/n_batches)<=0.001:\n",
    "            break\n",
    "        if (epoch+1)%100 ==0:\n",
    "            print(f'[{epoch+1},{j+1:5d}] loss:{running_loss/(100*n_batches):.3f}')\n",
    "            running_loss = 0.0\n",
    "            #generate_heatmap(model,epoch+1)\n",
    "    print(\"Finished Training\")\n",
    "    deltas_list.append(deltas)\n",
    "    attn,preds = generate_heatmap(model,epoch+1)\n",
    "    Attn_Values.append(attn)\n",
    "    Prediction_Values.append(preds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.plot(deltas_list[2])\n",
    "plt.ylabel(r\"$\\Delta$\", fontweight=\"bold\", fontsize=16)\n",
    "plt.xlabel(\"epochs\", fontweight=\"bold\", fontsize=16)\n",
    "plt.title(r\"Conservation of W_O^{\\top}W_O-W_VW_V^{\\top}\")\n",
    "plt.savefig(\"conserve_quantity.pdf\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "correct = 0 \n",
    "total = 0\n",
    "model.eval()\n",
    "with torch.no_grad():\n",
    "    running_loss = 0.0 \n",
    "    for j in range(500):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model(inputs)\n",
    "        loss = criterion(outputs,labels)\n",
    "        running_loss +=loss.item()\n",
    "        correct += sum(outputs.max(1)[1]==labels).item()\n",
    "        total += len(labels)\n",
    "print(f'loss:{running_loss/(j+1):.3f}')\n",
    "print(f'accuracy:{correct/total:.3f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "attn_values_1 = np.mean(Attn_Values,axis=0)\n",
    "prediction_1 = np.mean(Prediction_Values,axis=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "bin_edges = [np.array([0, 0.2, 0.4, 0.6, 0.8, 1]), np.array([0, 0.2, 0.4, 0.6, 0.8, 1])]\n",
    "\n",
    "num = 64\n",
    "histograms = []\n",
    "for attn_values, prediction in zip(Attn_Values, Prediction_Values):\n",
    "    hist, _, _ = np.histogram2d(attn_values, prediction, bins=bin_edges)\n",
    "    histograms.append(hist)\n",
    "    \n",
    "# Stack and compute mean and variance\n",
    "histograms = np.stack(histograms)\n",
    "mean_hist = np.mean(histograms, axis=0)\n",
    "var_hist = np.var(histograms, axis=0)\n",
    "\n",
    "\n",
    "mean_hist_norm = np.round(mean_hist/num,1)\n",
    "\n",
    "\n",
    "# Assuming `mean_hist_norm` and `var_hist_norm` are your 2D arrays of the same shape\n",
    "# If you prefer to show std instead of variance in the annotation, convert variance to std:\n",
    "std_hist_norm = np.round(np.sqrt(var_hist/num), 1)  # If you want std, otherwise use var_hist for variance.\n",
    "\n",
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(mean_hist_norm, dtype=object)\n",
    "for i in range(mean_hist_norm.shape[0]):\n",
    "    for j in range(mean_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"{mean_hist_norm[i, j]:.1f}\"#\\n(±{std_hist_norm[i, j]:.1f})\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(mean_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params3.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(std_hist_norm, dtype=object)\n",
    "for i in range(std_hist_norm.shape[0]):\n",
    "    for j in range(std_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"±{std_hist_norm[i, j]:.1f}\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(std_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params3_std.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# torch.manual_seed(1234)\n",
    "# model = Net_p1().to(device)\n",
    "model.eval()\n",
    "attn_values = []\n",
    "prediction  = []\n",
    "with torch.no_grad():\n",
    "    running_loss = 0.0\n",
    "    for j in range(500):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        indices = inputs>9\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model(inputs)\n",
    "        #print(outputs.shape)\n",
    "        outputs = nn.Softmax(dim=1)(outputs)\n",
    "        #print(outputs[np.arange(128),labels])\n",
    "        \n",
    "        for i in range(attn.shape[0]):\n",
    "            attn_values.append(sum(attn[i][indices[i]]).item())\n",
    "            prediction.append(outputs[i,labels[i]].item())\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "attn_values = np.array(attn_values)[None,:]\n",
    "prediction = np.array(prediction)[None,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "num = 160\n",
    "histograms = []\n",
    "for attn, pred in zip(attn_values, prediction):\n",
    "    hist, _, _ = np.histogram2d(attn, pred, bins=bin_edges)\n",
    "    histograms.append(hist)\n",
    "    \n",
    "# Stack and compute mean and variance\n",
    "histograms = np.stack(histograms)\n",
    "mean_hist = np.mean(histograms, axis=0)\n",
    "var_hist = np.var(histograms, axis=0)\n",
    "\n",
    "\n",
    "mean_hist_norm = np.round(mean_hist/num,1)\n",
    "\n",
    "\n",
    "# Assuming `mean_hist_norm` and `var_hist_norm` are your 2D arrays of the same shape\n",
    "# If you prefer to show std instead of variance in the annotation, convert variance to std:\n",
    "std_hist_norm = np.round(np.sqrt(var_hist/num), 1)  # If you want std, otherwise use var_hist for variance.\n",
    "\n",
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(mean_hist_norm, dtype=object)\n",
    "for i in range(mean_hist_norm.shape[0]):\n",
    "    for j in range(mean_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"{mean_hist_norm[i, j]:.1f}\" #\\n(±{std_hist_norm[i, j]:.1f})\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])# Create annotation labels with both mean and std (formatted as strings)\n",
    "\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(mean_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params3_test.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    for j in range(200):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        indices = inputs>9\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model(inputs)\n",
    "        #print(outputs.shape)\n",
    "        #outputs = nn.Softmax(dim=1)(outputs)\n",
    "        #print(outputs[np.arange(128),labels])\n",
    "        #print(outputs.shape)\n",
    "        \n",
    "        print(outputs.shape,attn.shape)\n",
    "        \n",
    "        for i in range(outputs.shape[0]):\n",
    "            \n",
    "            print(attn[i],labels[i],indices[i])\n",
    "            attn_values.append(sum(attn[i][indices[i]]).item())\n",
    "            print(attn_values[-1])\n",
    "            break\n",
    "            #prediction.append(outputs[i,labels[i]].item())\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## CACO"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Net_p4(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(Net_p4,self).__init__()\n",
    "        self.key_query_layer = nn.Linear(50,50,bias=False)\n",
    "\n",
    "        self.prediction_layer = nn.Linear(50,50,bias=False)\n",
    "        \n",
    "\n",
    "#         self.key_query_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "#         self.prediction_layer.weight = nn.Parameter(torch.eye(50,50)*0.000001)\n",
    "    def forward(self,x):\n",
    "        batch,seq_len,vocab_size = x.shape\n",
    "        \n",
    "        key = self.key_query_layer(x)\n",
    "        query = torch.unsqueeze(x[:,-1],dim=1)\n",
    "        \n",
    "        #print(key.shape,query.shape)\n",
    "        scores = (query@key.transpose(-2,-1))*(1/math.sqrt(key.size(-1)))\n",
    "        \n",
    "        \n",
    "        \n",
    "        \n",
    "        attn = nn.Softmax(dim=-1)(scores)\n",
    "        x1 = x #self.value_layer(x)\n",
    "        context =torch.matmul(attn,x1)\n",
    "    \n",
    "        output = self.prediction_layer(context)\n",
    "        #print(output.shape)\n",
    "        return output[:,0,:],attn[:,0,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_epochs = 2500\n",
    "n_batches = 200\n",
    "\n",
    "batch_size= 32 # actual batch size is times 4 here 128\n",
    "seq_len = 64\n",
    "\n",
    "n_seeds = [1234,1235,1236]\n",
    "\n",
    "Attn_Values  = []\n",
    "Prediction_Values = []\n",
    "#deltas_list = []\n",
    "for seed in n_seeds:\n",
    "    torch.manual_seed(seed)\n",
    "    model = Net_p4().to(device)\n",
    "    criterion = nn.CrossEntropyLoss()\n",
    "    optimizer = optim.SGD(model.parameters(),lr=1e-2)\n",
    "\n",
    "    running_loss = 0.0\n",
    "\n",
    "    common_token_attn = []\n",
    "    distinct_token_attn = []\n",
    "    #deltas = []\n",
    "    epsilon = 1e-8\n",
    "    #generate_heatmap(model,0)\n",
    "    for epoch in tqdm(range(n_epochs)):\n",
    "        model.train()\n",
    "        # record conservation error\n",
    "#         with torch.no_grad():\n",
    "#             C1 = model.value_layer.weight.matmul(model.value_layer.weight.t())\n",
    "#             C2 = model.prediction_layer.weight.t().matmul(model.prediction_layer.weight)\n",
    "            \n",
    "#             denom = max(torch.norm(C1).item(), torch.norm(C2).item(), epsilon)\n",
    "#             delta = torch.norm(C2 - C1).item() / denom\n",
    "#             deltas.append(delta)\n",
    "        for j in range(n_batches):\n",
    "            data = generate_data(batch_size,seq_len)\n",
    "\n",
    "            inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "            inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "\n",
    "            optimizer.zero_grad()\n",
    "\n",
    "            outputs,_ = model(inputs)\n",
    "            loss = criterion(outputs,labels)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            running_loss +=loss.item()\n",
    "        if (running_loss/n_batches)<=0.001:\n",
    "            break\n",
    "        if (epoch+1)%100 ==0:\n",
    "            print(f'[{epoch+1},{j+1:5d}] loss:{running_loss/(100*n_batches):.3f}')\n",
    "            running_loss = 0.0\n",
    "            #generate_heatmap(model,epoch+1)\n",
    "    print(\"Finished Training\")\n",
    "    #deltas_list.append(deltas)\n",
    "    attn,preds = generate_heatmap(model,epoch+1)\n",
    "    Attn_Values.append(attn)\n",
    "    Prediction_Values.append(preds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "correct = 0 \n",
    "total = 0\n",
    "model.eval()\n",
    "with torch.no_grad():\n",
    "    running_loss = 0.0 \n",
    "    for j in range(500):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model(inputs)\n",
    "        loss = criterion(outputs,labels)\n",
    "        running_loss +=loss.item()\n",
    "        correct += sum(outputs.max(1)[1]==labels).item()\n",
    "        total += len(labels)\n",
    "print(f'loss:{running_loss/(j+1):.3f}')\n",
    "print(f'accuracy:{correct/total:.3f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "bin_edges = [np.array([0, 0.2, 0.4, 0.6, 0.8, 1]), np.array([0, 0.2, 0.4, 0.6, 0.8, 1])]\n",
    "\n",
    "num = 64\n",
    "histograms = []\n",
    "for attn_values, prediction in zip(Attn_Values, Prediction_Values):\n",
    "    hist, _, _ = np.histogram2d(attn_values, prediction, bins=bin_edges)\n",
    "    histograms.append(hist)\n",
    "    \n",
    "# Stack and compute mean and variance\n",
    "histograms = np.stack(histograms)\n",
    "mean_hist = np.mean(histograms, axis=0)\n",
    "var_hist = np.var(histograms, axis=0)\n",
    "\n",
    "\n",
    "mean_hist_norm = np.round(mean_hist/num,1)\n",
    "\n",
    "\n",
    "# Assuming `mean_hist_norm` and `var_hist_norm` are your 2D arrays of the same shape\n",
    "# If you prefer to show std instead of variance in the annotation, convert variance to std:\n",
    "std_hist_norm = np.round(np.sqrt(var_hist/num), 1)  # If you want std, otherwise use var_hist for variance.\n",
    "\n",
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(mean_hist_norm, dtype=object)\n",
    "for i in range(mean_hist_norm.shape[0]):\n",
    "    for j in range(mean_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"{mean_hist_norm[i, j]:.1f}\"#\\n(±{std_hist_norm[i, j]:.1f})\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(mean_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params4.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(std_hist_norm, dtype=object)\n",
    "for i in range(std_hist_norm.shape[0]):\n",
    "    for j in range(std_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"±{std_hist_norm[i, j]:.1f}\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(std_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labels\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params4_std.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# torch.manual_seed(1234)\n",
    "# model = Net_p1().to(device)\n",
    "model.eval()\n",
    "attn_values = []\n",
    "prediction  = []\n",
    "with torch.no_grad():\n",
    "    running_loss = 0.0\n",
    "    for j in range(500):\n",
    "        data = generate_data(batch_size,seq_len)\n",
    "        inputs,labels = data[:,:-1].to(device),data[:,-1].to(device)\n",
    "        indices = inputs>9\n",
    "        inputs = F.one_hot(inputs,num_classes=50).float()\n",
    "        outputs,attn  = model(inputs)\n",
    "        #print(outputs.shape)\n",
    "        outputs = nn.Softmax(dim=1)(outputs)\n",
    "        #print(outputs[np.arange(128),labels])\n",
    "        \n",
    "        for i in range(attn.shape[0]):\n",
    "            attn_values.append(sum(attn[i][indices[i]]).item())\n",
    "            prediction.append(outputs[i,labels[i]].item())\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "attn_values = np.array(attn_values)[None,:]\n",
    "prediction = np.array(prediction)[None,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "num = 160\n",
    "histograms = []\n",
    "for attn, pred in zip(attn_values, prediction):\n",
    "    hist, _, _ = np.histogram2d(attn, pred, bins=bin_edges)\n",
    "    histograms.append(hist)\n",
    "    \n",
    "# Stack and compute mean and variance\n",
    "histograms = np.stack(histograms)\n",
    "mean_hist = np.mean(histograms, axis=0)\n",
    "var_hist = np.var(histograms, axis=0)\n",
    "\n",
    "\n",
    "mean_hist_norm = np.round(mean_hist/num,1)\n",
    "\n",
    "\n",
    "# Assuming `mean_hist_norm` and `var_hist_norm` are your 2D arrays of the same shape\n",
    "# If you prefer to show std instead of variance in the annotation, convert variance to std:\n",
    "std_hist_norm = np.round(np.sqrt(var_hist/num), 1)  # If you want std, otherwise use var_hist for variance.\n",
    "\n",
    "# Create annotation labels with both mean and std (formatted as strings)\n",
    "annot_array = np.empty_like(mean_hist_norm, dtype=object)\n",
    "for i in range(mean_hist_norm.shape[0]):\n",
    "    for j in range(mean_hist_norm.shape[1]):\n",
    "        annot_array[i, j] = f\"{mean_hist_norm[i, j]:.1f}\" #\\n(±{std_hist_norm[i, j]:.1f})\"  # mean ± std\n",
    "\n",
    "# Define tick positions for edges of bins (for correct labeling)\n",
    "x_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "y_edges = np.array([0, 0.2, 0.4, 0.6, 0.8, 1])\n",
    "xtick_positions = np.arange(len(x_edges) - 1) + 1.0  # Right edges\n",
    "ytick_positions = np.arange(len(y_edges) - 1) + 1.0  # Top edges\n",
    "\n",
    "# Plot heatmap for mean values (color intensity)\n",
    "plt.figure(figsize=(6, 6))\n",
    "ax = sns.heatmap(mean_hist_norm.T, annot=annot_array.T, cmap=sns.color_palette(\"coolwarm\"), \n",
    "                 annot_kws={\"size\":18}, cbar=False, vmin=5, vmax=70,fmt=\"\")\n",
    "\n",
    "# Adjust tick positions for x and y axes (move them to the edges)\n",
    "ax.set_xticks(xtick_positions)\n",
    "ax.set_yticks(ytick_positions)\n",
    "\n",
    "# Set the labels for ticks (x and y edges)\n",
    "ax.set_xticklabels(x_edges[1:])\n",
    "ax.set_yticklabels(y_edges[1:])\n",
    "\n",
    "# Make the tick labels bold\n",
    "ax.tick_params(axis='x', labelsize=16)  # Bold x-axis labelsimport numpy as np\n",
    "ax.tick_params(axis='y', labelsize=16)  # Bold y-axis labels\n",
    "\n",
    "\n",
    "# Invert y-axis to align with typical heatmap style\n",
    "ax.invert_yaxis()\n",
    "\n",
    "# Labels and title\n",
    "plt.xlabel(r\"Distinct Token Attention\", fontweight=\"bold\", fontsize=16)\n",
    "plt.ylabel(r\"Relevant Token Probability\", fontweight=\"bold\", fontsize=16)\n",
    "#plt.title(\"Mean (±Std Dev) Heatmap\")\n",
    "\n",
    "# Save the figure\n",
    "plt.savefig(\"Params4_test.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
