{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "7ed13b1f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\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(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "print(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "ba437766",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 3500x1000 with 1 Axes>"
      ]
     },
     "metadata": {},
     "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_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 = 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": 3,
   "id": "44b35df4",
   "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": "code",
   "execution_count": null,
   "id": "3ebf58ed",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_epochs = 1000\n",
    "n_batches = 100\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_p1().to(device)\n",
    "    criterion = nn.CrossEntropyLoss()\n",
    "    # Define parameter groups\n",
    "    optimizer = torch.optim.SGD([\n",
    "        {'params': model.key_layer.parameters(), 'lr': 2e-1},\n",
    "        {'params': model.query_layer.parameters(), 'lr': 2e-1},\n",
    "        {'params': model.value_layer.parameters(), 'lr': 1e-2},\n",
    "        {'params': model.prediction_layer.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",
    "        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,
   "id": "6c9cd219",
   "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,
   "id": "11f84eae",
   "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,
   "id": "1e9de6ea",
   "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=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(\"params_increased_fafo_2.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d099580c",
   "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=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_increased_lr_qk_std.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bbfc4511",
   "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())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8f1705d4",
   "metadata": {},
   "outputs": [],
   "source": [
    "attn_values = np.array(attn_values)[None,:]\n",
    "prediction = np.array(prediction)[None,:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "29ac0946",
   "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 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(\"params_increased_fafo_test.pdf\", bbox_inches='tight')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "15d82af3",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d22467f0",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "26568dd3",
   "metadata": {},
   "source": [
    "# increasing lr\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "66cf6754",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_epochs = 2000\n",
    "n_batches = 100\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",
    "\n",
    "\n",
    "\n",
    "# total number of training steps\n",
    "total_steps = n_epochs * n_batches\n",
    "\n",
    "# def increasing_lr_lambda(step):\n",
    "#     progress = step / total_steps\n",
    "#     scale = 1.0 + 9.0 * (progress ** 0.1) \n",
    "#     return scale\n",
    "\n",
    "\n",
    "for seed in n_seeds:\n",
    "    torch.manual_seed(seed)\n",
    "    model = Net_p1().to(device)\n",
    "    criterion = nn.CrossEntropyLoss()\n",
    "    # Define parameter groups\n",
    "    optimizer = torch.optim.SGD([\n",
    "        {'params': model.key_layer.parameters(), 'lr': 1e-2},\n",
    "        {'params': model.query_layer.parameters(), 'lr': 1e-2},\n",
    "        {'params': model.value_layer.parameters(), 'lr': 1e-2},\n",
    "        {'params': model.prediction_layer.parameters(), 'lr': 1e-2}])\n",
    "    \n",
    "    \n",
    "    scheduler = torch.optim.lr_scheduler.LambdaLR(\n",
    "        optimizer,\n",
    "        lr_lambda=[lambda step: 1 + (40 - 1) * step / total_steps,  # key_layer\n",
    "                   lambda step: 1 + (40 - 1) * step / total_steps, # query_layer\n",
    "                   lambda step: 1.0,      # value_layer\n",
    "                   lambda step: 1.0       # prediction_layer\n",
    "                  ])\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",
    "        # Print current learning rates for debugging\n",
    "        #for i, param_group in enumerate(optimizer.param_groups):\n",
    "        #    print(f\"Group {i} LR: {param_group['lr']:.6f}\")\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",
    "            scheduler.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,
   "id": "7d40c649",
   "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}')\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1782bc13",
   "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,
   "id": "514c1fe1",
   "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=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(\"params_increasing_fafo.pdf\", bbox_inches='tight')\n",
    "plt.show()\n",
    "\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4e207933",
   "metadata": {},
   "source": [
    "# CACO Setting"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d6b7fd53",
   "metadata": {},
   "source": [
    "### Increased Learning Rate"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "b23aa0d2",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Net_CACO(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(Net_CACO, self).__init__()\n",
    "        self.key_layer = nn.Linear(50, 50, bias=False)\n",
    "        self.prediction_layer = nn.Linear(50, 50, bias=False)\n",
    "\n",
    "    def forward(self, x):\n",
    "        # x: (batch, seq_len, vocab_size)\n",
    "        key = self.key_layer(x)  # (B, T, D)\n",
    "        query = x[:, -1:, :]     # (B, 1, D)\n",
    "\n",
    "        # Attention scores: (B, 1, D) x (B, D, T) -> (B, 1, T)\n",
    "        scores = torch.matmul(query, key.transpose(-2, -1)) / math.sqrt(key.size(-1))\n",
    "        attn = F.softmax(scores, dim=-1)  # (B, 1, T)\n",
    "\n",
    "        # Context: (B, 1, T) x (B, T, D) -> (B, 1, D)\n",
    "        context = torch.matmul(attn, x)\n",
    "        output = self.prediction_layer(context)  # (B, 1, D)\n",
    "\n",
    "        return output.squeeze(1), attn.squeeze(1)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "bafe74d6",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  5%|██                                      | 100/2000 [00:35<09:57,  3.18it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.284\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:09<09:52,  3.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.212\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:43<09:44,  2.91it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.466\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:17<08:54,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:0.928\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:50<08:28,  2.95it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.493\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:24<07:46,  3.00it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.216\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [03:58<07:52,  2.75it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.101\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:33<06:42,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.056\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [05:07<06:12,  2.96it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.036\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:41<05:39,  2.94it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.026\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:14<04:54,  3.06it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.020\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [06:47<04:23,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.016\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:20<03:44,  3.12it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.013\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [07:54<03:35,  2.78it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.011\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:28<02:49,  2.95it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.010\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [09:03<02:10,  3.06it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [09:39<01:46,  2.81it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [10:14<01:08,  2.92it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.007\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [10:50<00:32,  3.06it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.006\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:26<00:00,  2.91it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.006\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "  5%|██                                      | 100/2000 [00:35<10:49,  2.92it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.284\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:11<10:22,  2.89it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.212\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:48<11:54,  2.38it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.469\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:23<09:37,  2.77it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:0.946\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:58<08:16,  3.02it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.542\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:34<08:04,  2.89it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.269\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [04:10<07:51,  2.76it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.133\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:45<06:51,  2.92it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.074\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [05:20<06:26,  2.85it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.047\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:56<06:35,  2.53it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.033\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:29<04:47,  3.14it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.024\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [07:01<04:23,  3.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.019\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:36<03:49,  3.05it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.016\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [08:09<03:20,  3.00it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.013\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:42<02:38,  3.15it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.011\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [09:15<02:05,  3.20it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.010\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [09:48<01:40,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [10:21<01:08,  2.93it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [10:56<00:36,  2.74it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.007\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:30<00:00,  2.90it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.006\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "  5%|██                                      | 100/2000 [00:35<10:38,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.286\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:10<10:25,  2.88it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.214\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:45<09:21,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.469\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:19<08:56,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:0.942\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:53<08:06,  3.08it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.531\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:28<08:47,  2.65it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.254\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [04:05<07:54,  2.74it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.122\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:40<07:10,  2.79it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.068\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [05:14<06:16,  2.92it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.043\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:48<05:51,  2.84it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.030\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:23<05:45,  2.60it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.023\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [07:00<04:40,  2.85it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.018\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:34<03:46,  3.09it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.015\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [08:10<03:54,  2.56it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.012\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:45<02:48,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.011\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [09:21<02:19,  2.87it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [09:56<01:40,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [10:31<01:07,  2.97it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.007\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [11:07<00:40,  2.49it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.007\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:44<00:00,  2.84it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.006\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "  5%|██                                      | 100/2000 [00:35<11:08,  2.84it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.277\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:12<11:20,  2.65it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.209\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:48<10:36,  2.67it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.469\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:24<08:47,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:0.952\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:59<08:57,  2.79it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.555\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:33<08:17,  2.81it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.285\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [04:07<07:33,  2.87it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.145\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:43<07:22,  2.71it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.082\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [05:19<07:40,  2.39it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.053\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:55<06:04,  2.75it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.037\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:33<05:39,  2.65it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.027\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [07:09<05:03,  2.63it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.021\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:45<04:11,  2.78it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.017\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [08:21<03:46,  2.65it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.014\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:58<03:13,  2.58it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.012\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [09:32<02:17,  2.91it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.011\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [10:06<01:46,  2.80it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [10:39<01:01,  3.23it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [11:14<00:34,  2.88it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:50<00:00,  2.82it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.007\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "  5%|██                                      | 100/2000 [00:35<11:38,  2.72it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.280\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:08<09:40,  3.10it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.205\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:42<09:44,  2.91it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.455\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:18<09:29,  2.81it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:0.924\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:55<08:27,  2.96it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.519\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:30<09:03,  2.57it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.257\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [04:05<07:59,  2.71it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.130\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:39<06:34,  3.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.074\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [05:11<05:54,  3.11it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.048\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:44<05:35,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.034\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:17<05:01,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.025\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [06:50<04:21,  3.06it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.020\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:22<03:46,  3.09it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.016\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [07:55<03:21,  2.97it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.014\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:28<02:43,  3.06it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.012\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [09:01<02:09,  3.09it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.010\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [09:35<01:37,  3.09it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [10:08<01:06,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [10:41<00:32,  3.08it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.007\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:15<00:00,  2.96it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.007\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "n_epochs = 2000\n",
    "n_batches = 100\n",
    "\n",
    "batch_size = 32\n",
    "seq_len = 128\n",
    "\n",
    "n_seeds = [1234,1235,1236,1237,1238]\n",
    "\n",
    "Attn_Values = []\n",
    "Prediction_Values = []\n",
    "\n",
    "for seed in n_seeds:\n",
    "    torch.manual_seed(seed)\n",
    "    model = Net_CACO().to(device)\n",
    "    \n",
    "    criterion  = nn.CrossEntropyLoss()\n",
    "    \n",
    "    lr_query_key = 0.05 # 0.02, 0.03,0.04,0.05,0.1\n",
    "    \n",
    "    lr_prediction = 0.01\n",
    "    \n",
    "    optimizer = optim.SGD([\n",
    "        {\"params\":model.key_layer.parameters(), \"lr\":lr_query_key},\n",
    "        {\"params\":model.prediction_layer.parameters(),\"lr\":lr_prediction}])\n",
    "    \n",
    "    \n",
    "    running_loss =0.0\n",
    "    \n",
    "    common_token_attn = []\n",
    "    distinct_token_attn = []\n",
    "    \n",
    "    \n",
    "    \n",
    "    epsilon = 1e-8\n",
    "    \n",
    "    for epoch in tqdm(range(n_epochs)):\n",
    "        model.train()\n",
    "        \n",
    "        for j in range(n_batches):\n",
    "            \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",
    "            \n",
    "            optimizer.zero_grad()\n",
    "            \n",
    "            outputs,_ = model(inputs)\n",
    "            loss = criterion(outputs,labels)\n",
    "            \n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            \n",
    "            running_loss +=loss.item()\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": 43,
   "id": "de260395",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss:0.006\n",
      "accuracy:1.000\n"
     ]
    }
   ],
   "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": 44,
   "id": "1f554333",
   "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": 45,
   "id": "e038607a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAioAAAInCAYAAAC2rnJtAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuNSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/xnp5ZAAAACXBIWXMAAA9hAAAPYQGoP6dpAAB4R0lEQVR4nO3dd3xT5f4H8M/JaDpJoYWW1TIEShmC7F1GWSKCBQRUwAEier0qjqteBdTfRb3e60YcXIoKokxBEUGm7CGjTAuUDloKLXTRNkmT5/dHbGjoIDlNTtP28369+no1OU+e8+Sb0/bb5zxDEkIIEBEREXkgVVU3gIiIiKg8TFSIiIjIYzFRISIiIo/FRIWIiIg8FhMVIiIi8lhMVIiIiMhjMVEhIiIij8VEhYiIiDwWExUiIiLyWExUiIiIyGMxUSEiIiKPxUSFiIiIPBYTFSIiIvJYTFSIiIjIY2mqugGOSElJwY4dOxAfH4/s7GwAgF6vR6tWrdC/f380bdq0iltIRERE7uDRicr58+fx1FNPYdOmTQAAIYTdcUmSAADDhg3DRx99hDvuuEPxNhIREZH7SOLWv/4eIiEhAd27d0dmZiaioqIwbNgwtGrVCnXq1AEA5OTkID4+Hhs3bsSOHTsQHByM/fv3o3nz5lXcciIiInIVj01UJk+ejNWrV2Pt2rUYPnx4hWV/+eUXjB07FjExMVi6dKlCLSQiIiJ389hEpUGDBhgxYgSWLFniUPkpU6Zg48aNuHLliptbRkRERErx2Fk/eXl5aNSokcPlGzVqhLy8PDe2iIiIiJTmsT0qHTt2BAD88ccf0GgqHvNrMplw1113QZIkHD9+XInmERERkQI8tkdl+vTpOHHiBIYOHYrdu3eXmvEDWGcB7dq1C0OHDsWpU6cwY8aMKmgpERERuYvH9qgIITBjxgwsWrQIkiTBz88PzZs3h16vBwBkZ2cjISEBN27cgBACjz32GL744osqbjURERG5kscmKsW2bduGL7/8Ejt27EBaWprdsYYNG2LAgAGYMWMGoqKibluXwWCAwWCwe06n00Gn07myyUREROQiHp+olJSfn2+3Mq2vr69Tr587dy7mzZtn91yXwS+ia/Q/XNZGIiIAeDXxsapuQq2z8b5VVd2EWmf6EPefo1olKpVVVo/K7I/zodawR4WIXIuJivKYqChPiUTFo5fQd7WybvOoNeYqag0RERHdTo1IVDIzM/Hpp59CkiS89tprVd0cIiIicpEakahkZGRg7ty5TFSIiIhqmBqRqAQHB+P111+37aZMRERENUONSFSCgoIwd+7cqm4GERERuZjHrkxLREREVGMSlUWLFuGRRx6p6mYQERGRC9WYRGXXrl1YsmRJVTeDiIiIXKjGJCpERERU83jsYNqvv/7aqfLnzp1zU0uIiIioqnhsojJt2jSnphsLITg9mYiIqIbx2ETFy8sLjRo1wuOPP+5Q+RUrVuDIkSNubhUREREpyWMTlQ4dOiApKQkvvfSSQ+XPnDnDRIWIiKiG8djBtF26dEFGRgaSk5OruilERERURTy2R6Vfv3749ddfER8fj6ZNm962fN++fRVoFRERESlJEkKIqm5EVZr5zvWqbgIR1UCvJj5W1U2odTbet6qqm1DrTB/i/nN47K0fIiIiIiYqRERE5LGYqBAREZHHYqJCREREHouJChEREXksJipERETksZioEBERkcdiokJEREQei4kKEREReSwmKkREROSxmKgQERGRx2KiQkRERB6LiQoRERF5LCYqRERE5LGYqBAREZHHYqJCREREHouJChEREXksJipERETksTRV3YDaTucFRHfzRuc2XgjSqyAEkH7NjEOnjdh22ACzRX7dAb4ShvbwRoeWWtSro4KpSCA1w4x9J4zYfdzoujdRzTDmyvHzltCxlRYR4RqEhWhQr44KKhWQVyCQmFaEfSeMOBpvklV30xA1Ot6hRViIGiH11PD3leDjJaHAKJCeacGJCybsOGJAfqFw8buqOk0XrHK4bOGfJ3D1gznlHlfVCYR//+HwaX8X1EENIGm9YMnNgelyCgzxJ5H72zrAYpbVTp/OPeHXcyC8mraAyj8AwmyGOSsThvhTyNu5EaaUi7LqrSoFeddxLm4rks7uRXryKeRcS4WwFMHHvx5Cw9qjXY+xaNUpusI6jIV5OLhlMeKPbkJ2RgoklQp1GzRDRJe7cVfUg1BrvCrVxrN/bMTJ/WuRnnwCBXlZUKk1CAgMQZM7uqFz/wfQoGnbStVflSQhRM35KZZh5jvXq+zc9eqo8NwkfwQHqgEABqOASgVoNRIAIOlyET5Ynod8g/MfUViIGk9P8Ie/r7XTrNAgoNUAarW17pMXTFiwKq9Sf5SrI8ZcWZ8+H2h7/wBgNAlYBODtdfO5E+dN+HxtHkxFztU9cYgPorp429VttgA+upt15+ZbsGBVHhJS5f3BrYxXEx9zeZ2N5n9VcQG1Bmr/AABAzua1yF7zTZnFfLr0Rr3JM6Hy8QMAWIwGwFxkewwAKbMfgijId66BGg2CH3sePh272Z6yFBZAUmsgabUAAGExI2v118jb+pNzdTtg432OJ3LO+O/f2sFiuXmBarQ6SJIaJuPN+DSP7I/R0z+C1sun1OuzMy/h+w8fQk7mJevrvXwgLGaYi6z/vDRoGokJT8fC21fvdNuKTEasX/Q0zsdtsz2n1fnCYjbBXGT9J0CSVBhw30voOmia0/XfzvQhLq+yFPaoVBGVBMyK8UNwoBpZuRbE/nwDZxKLIAG4K0KLB4f7ISxUg4fv8cOnK/OcqtvbC3hynPUPZlqmGYt/uoGky2aoVUDfO3UYP9gH7VpoMWGwD77bXOCeN+iBGHPlqdUSElKLsDfOgFMJRcjItmZpQXVUGNHbG33v1KF9Sy0eGOaL2J+d+6OYkGZG5rZ8nEspwuVMCwr+Si51WqBTay/EDPRBHT8VnrjPH69/kY3CGtChlfpyxclPwOB7EBgzDQBwY/eWMsv4dO6FoIefgaRSI+/3Tcjd9jOKLqcAACSdN7RNmsO3Uw/A7HxyV2dYjC1Jyd3xC3I3roY5+xogSdA2aY664x+G7o5IBN43FYb4UzAlX3D6HFXBYilCaHhHtO85Fs0i+yEwuCkAIDszBfs2foa4PSuRcGonNi97HSOn/dv+teYirFk4EzmZl+Cnr4+RU95FeERvCIsFZ49sxKZl/8SV5FP4OfYFxMz6wum27f91oS1J6dR/MnoMm4mAwBAIiwXpKaewbeW/cOn8YWxf/Taa3NEVoWHtKx8QhXGMShXp2cELTRpY88TP1+bhTKI1WxcADp8xYemvNwAAHVpq0SbcuXwyuoc39P4qGE0Cn6zIQ9Jl6y8cswXYccSA9bsKAQB9O+nQoG7tuQQYc+X997tcvPNNLnYeNdqSFADIzLHg24352HnEAADo2V6HugFSedWUaf9JIzYfMCAh1WxLUgDAYLIeW/yT9fOs46dCxzsq161eXfj1HgwAMJw7haIrqaWOq+oEou7kxyGp1Li+MhbXv/vclqQAgDAUwnj+NLJWxUIYDc6fv8cAANbbTlnff2VNUgBACJiSL+DqgvnWHhaVCr539ZLxDqvGhL8vwYMvrkCn/pNtSQoA6IOaYNgD/4c7+94PADh1cB1yrqfZvfbk/jXISP0TAHDvYx8jPKI3AEBSqRDRZSSiJ70BAEg4uQOJZ/Y63baTB9YCAJq06o4h989BQGCIrf7QsPa474nPodX5AkLgzyO/Ol2/J6g9vzE9TK/21l+cZxJNZXZLHzptwtUs6/M92zn3S7a4/MHTRmRml77PsP1wIQoNAmqVhO5O1l2dMebK+zOp4vs5u4/f/GMYHuraDt6E1JvnDnQyCaqOvFq0gbah9Y9oXjm9KQED74baLwDGpAvI27re5W1Q6+sCAIxJ58s8LgrzbQmUpPMus4wnCmvds8Lj7XuPs32fnnjC7tjJfWsBAE1b90CjFp1LvTaiy93QBzWxlv0r6XDGjeyrAFBuT4nOJwD1GjQHAJgMTt7K8xBMVKqAVgO0bGz9pXzyQvkDCU/9dSyyudbhukPqqRCkV1dYt8EEnEux/hKPbOZ43dUZY+6ZTEU3e0JULv5tdEeTm4lPRlbNHxhU3Jtiyb+Bgj/2lFnG968ejxsHdrqlDUUZ6QAAr7CWZR6XvH2hadAIAGBMLDuZqY40Gp3te0uJAcgmYwEuXfgDgHUMS1kkSUKzyH4AgMTTu50+t/6vHp70pBNlHjcU5OLalQQAQEg1vO0DMFGpEg2D1FCprP/hpWaUfx84NcP6y1Xvr4Kvt2P/ETYKVpd4fUV1W481LFG+JmPMPVPrsJtJ26WrlR/wqlFbx79E3aXDw6OsA0OvXDPj+Dl5M4uqC0nnDd+7rLcU8g/tgjCVHpCjDmoATWAQAMCUfB7aRmGo9/AzaDT/KzT5cDka/t8XCHr0OXi1aCO7HXm/W28teLduj8D7H4NaX892TNu0OerPehkqbx8YLpxBvpuSpaqQHH/A9n39xq1t32dePg8hrL9Tghu1Kvf1xcdu5FxFwY0sp87dqd8kWxt++34ecrOsyaIQAulJJ7H6s8dhMuSjUfPOiOw+2qm6PQUH01YBvf/N/DArt/zZJVm5N/8LDPSXHJpmaV93+f9FFh/z0UnQaa3/8ddkjLnn8dFJGN7T2v0fn2xC+jX5vR4fzw60zdwq6VxKERaty0OR8pN+FOXbpQ9U3tbZJnm7fyuzjPavngwA8GoRAf3ICZC0WliMBogiIzR1g6Dp0gc+nXsh5+fvkfPLSqfbkbdjI9SBQQgYMhoBA0YgYMAIu1k/5uzryPl1NXI2rABEzejlKszPwf5NnwMAmrTsinohLWzHbmRfsX1fPHakLAH6m8fysq/Axy/Q4fN37v8AcrMu49Bv/8PRnctwdOcyu1k/fnXqo/vQGeg14kmoVNXznyQmKlXAu8QQBaOp/D+ExhLd4jovx/67Lznt01jBH8Jb6zZU0I6agDH3LBKAh0f5IjDAOgB5eSVnQuXcsECjlqDzkmyfx5lEE1ZvL8D1ChLTmsKvj3WOqDE5odyZNJLvzanH+nsmwnwtA9eWLoThzzhACGhCm6DuxOnwbt0e+nsmwZSWjIKj+51riBDI/nEpTGkpqHv/Y1B5+9gSKACQtFqofHwheenK7PWpboTFgg1LXsSN7KvQaHUYfP9rdseNhTds32u0pact246VmNJsKvEaR0gqFfqPno2g0Duw5Yc3YDLk241FKTIZYCjIhcmYD61X9RkXVBITFSJS3IQhPraZOMs351f6ts+rC3Ns3wf4SujRzgsjennjH1MC8MueQtusq5pI07ApdM2ttxtu7Cl7EC1gHQtR4hEyvvw3TMkJtmeKLqcg47P5aDj3E6j1dVFn5ASnExWVXwCCpj8P79btUXj6KLJ//gGm1GRIXl7QNW8N/ZiH4N9/OLzbdsKV/752c1ZQNbV15f/hwgnr1ODBE15H/cYRirchP+8a1n/1dyTHH0B4RB/0Hvkkghu1RpGxEKkJR7Hzx/dw7PfvcPH0Lkx8dmmFPTueimNUqkDJ9Ry8tOX/1+5VoivbYHTsv8LCEuW8KhizKafu6owx9xwxA30w8K+F2n7Yko89ca79zzo3X+C3gwZ8vCIPEMDdfXzQoWXNHcDsXzyI1mjAjQM7yi1nMdxM1gxn4+ySlGLCUIi8Hb8AALyaNIMqwLkFyOpN/Zs1SfnzBK5+/CaMF85CFObDkpOFgmMHcOU/r8Kcmw1N/VDoxzzoVN2eZvvqd3Bkx7cAgIExL6NDiZk/xby8b/ZiFZnK7zUsMt48pi3xGkf88vU/kBx/AE1adce4pxahccsu0PkEwE9fH606RWPic8vg418X2RnJ+H3te07V7SmYqFSB7LwS4yAqmDYZGFBi7EOeY3/Y7Osu/+MtPlZgELVirARj7hnui/JBdHdrkrJyaz62HnJ+rQ5HXUwz22Za9b2zhk4JV2vg2906m6Tg6L4KV5I1Z2XavjeVWDvlViWPaerVd7gpmtDG8GnfBQCQu6Xsqc+WvBzk77cmUz6dejhct6fZseZdHNryPwDAgLEvoUs5K7766RvYvi8e5FqW3Oybx/xLvOZ2Mi+fR8JJazy7DX74ll6zv9oQEITI7mMAAH8e24zquBg9E5UqkJZphsVivVgaVTADpFGw9ePJzrM4vF9JyVknFddtPZZWwSyVmoQxr3r3RflgaA9rkrJqWz5+O+i+JKVYcbLZoG71HER4Oz4du0H9V69HeSvRFjOlpUA4tNpsyT92jv9R04beXAit6Orl8ttx1bogmkrn7XSPjSfYvvodHPxtEQCg/5gX0G3II+WWDQptCUmy/k7JSI0vt1zxMb869Z0aSJuZds72vT44rNxydRuEA7D23OTnZpZbzlMxUakCpiLg/CXrf3rtKlivo3gtj1MJjv/7nX7Ngsxs6y+jdi3KrttLe3ONiVMXa8e/9ox51YoZaJ+kbD7g/iQFAIID/9p3qYbeaiseRGu6kgZD/MmKCxeZYDh3CgCgDW1SbjFtQ+sxYbGgKPOq440pMYunop4YdYnkRBiq19ih7avfsfWk9B/zArpHV7ylgdbLB41b3AUAuHjq9zLLCCFw8fQuAEB42z5Otac4CQKAnGuXyi2Xn3MzOdHqfJ06hydgolJF9p6w3pdvHa5Bs4al/9vrEqFF/b/+C9x30rl7+MXlu0Z4IahO6Y84qrMO3joJZovAASfrrs4Y86oRM9D+do8rkpQyerhLaVPic77dCrnVkbpuMLwjOgAAbuytuDel2I291oGfujYdoG3avNRxSecN//7DAQDGi/Gw5OWUKlMeY4kxL/79h5VZRvLSwa9HlLV8ykVZy/RXlZJJyoCxL902SSnWrucYAEBS/H6kJRwrdfzsH78gOyPZWvavWzSOatA00vb9sd+/K7OM0ZBvW/G2fuM28GKiQo7aF2dEypUiqCQJj4/1t+0tIwG4q411gzzAurPs2UT7X7Kj+nhj4Ut1sfClumX+Udy8vxDZeRbovCQ8Od4fYSHWX9ZqFdC/kxfu6WedCrfrqAFXrteMtQwcwZgrr+SYlBVbnLvd06u9ly3mrZvaT1CsF6DCq9MC0O9OLwTr7T+PugEShvXQ4Yn7/KGSJOQVWLDlUPX6z90Rfr0HQVKpIcxFtgTkdvIP7oQh4U9IKhWCp78AXZsOtqxPE9oYwTP/AbW+LoTFjOx1y0q9vs7dE9B0wSo0XbAK6lt6TczXrqLg+EEA1ltS9aY+DXXwXzNMVGp4tWiDBs++AU39UABA7pZ1ct+64kqOSYmKebnC2z23atdjLIIbtQaEwI9f/c22n4+wWHD2j1+waZl1SnPzyP4Ijyi9/9Hunz/Ge0+2wXtPtkF2pv3YIn1QY7TsMBAAcD5uGzbEvoCsq0kQQsBsNuHShT/w/QcP2RKhroMdb7cn4fTkKmIRwGerbuDZSf4IDlTj2YkBMBgFJOnmrJSky0X433rn5tQD1hkun67Mw9MT/NEoWI1XptVBgUFAqwE0amvdJxNMWLG19uziCzDmSqsbINlu91gsAkN7eNsel+W3g4VO9bY0DdHggeHWX2GmIoFCo4BWI9mta3M1y4wv1txAzo0adutHkuDXcxAAoPDEH7DkZDn2OiGQ8fk7aPD0XGgbNUWDv8+1zgYym6H6a50VUWTC9e+/guHPspdkr8i1bz5F/af+Ca/wO+DXYwD8egyAxVAISaOBpL755yZn81rboFpPl3Mt1TYmRZJUOLDpSxzY9GW55bsNeQTdhjxqe6xSazB25mf4/sMpyMm8hBUfT7OumyIsKDJZr/cGTSNx98PyZuQMe/BfWPXpY0hPOolTB9fh1MF10Hj5wFJkgsVy8x+ubkMeRbseY2Sdo6oxUalCmTkWvLk4B9HdvdG5tReC9CpYLEBiWhEOnjZi22EDzDL/+U5KN2PeohwM6+mNDi21qBuggsEkkJBahH0njNhz3OjEMLmagzFXjqrE/RmVSoLev+L7NboKpo3fKivPgi/W5qF1Uw2aNdIg0F8FPx8JQgCZ2WakXDHj2DkTDp4ywlTz7vpAF9ERmiBrj0ZeBWunlMWSk4XLbz+PgAEj4NOlD7QNGkHSeqEoIx2Ff55A3pb1MKUly2qX5UYu0v/9Mvx6DoRP517WKc5+/hBmM8zXMmC4cBZ5uzbBeP6MrPqrgigx9kYIC/JzMyosbyxj4z99UBNMe2UdDv72P8Qf24zsjBRIag1CGt6BiC6jcFfUg1Br5M1M8/Wvhwee/wEn96/B2T824krKGRTmZ0OlViOgXkM0bt4ZHfvejyZ3dJVVvyeQRHWcq+RCM9+5XtVNIKIa6NVEx8YwkOtsvG9VVTeh1pk+xP3n4BgVIiIi8lhMVIiIiMhjMVEhIiIij8VEhYiIiDwWExUiIiLyWExUiIiIyGMxUSEiIiKPxUSFiIiIPBYTFSIiIvJYshIVi6X2bKpGREREVUdWohIWFoa5c+ciJSXl9oWJiIiIZJKVqKSmpuLNN99E8+bNMWbMGGzcuNHV7SIiIiKq3BgVs9mM9evX4+6770aLFi3w9ttv48qVK65qGxEREdVyshKVevXqoeSmy0IIXLx4Ea+++irCwsIwadIkbN++3VVtJCIiolpKVqJy+fJl/Pjjjxg/fjx0Op3dMaPRiB9++AGDBw9G27Zt8eGHHyIrK8sVbSUiIqJaRlaiotFocM899+D7779Heno6Fi1ahIEDB0KSJADWHhYhBM6ePYvnnnsOTZo0wezZs5Gbm+vSxhMREVHNVul1VAICAvDwww9jy5YtuHDhAnr37g0AkCQJkiRBCIH8/Hx88MEH6N69OzIyMirdaCIiIqodXLLgW3p6Ot5880306dMHe/futfWslCSEwJ9//om33nrLFackIiKiWqBSicqePXswefJkhIeHY+7cubh06RKAm7d+GjdujHfffRezZ8+GSqWCEALr1q1zScOJiIio5tPIedGiRYvw6aef4tixYwCsiUnJ8SkdOnTA888/j0mTJkGjsZ4iKysLixYt4iJxRERE5DBZicr06dNt409KJiiDBw/GCy+8gKFDh5Z6TevWrQFY114hIiIicoSsRKUktVqN8ePH44UXXkCnTp3KLRceHo4BAwZU9nRERERUi8hOVHx9ffHYY4/h2WefRVhY2G3LT5gwARMmTJB7OiIiIqqFZCUq//d//4cnnngCgYGBLm4OERER0U2yZv2YTCZ89NFHWLlyZbllLl26hFOnTuHUqVOyG0dERES1m6welblz50KSJMTExGDcuHFllnnmmWewevVqSJKEoqKiSjWSiIiIaqdKD6atSMmNC+VITk5GRkYGIiMjbXsKCSGwfPlyHDlyBDqdDgMHDsSgQYNc0VwiIiLyMG5LVDIzM2W/1mKxYPr06YiNjQUANGzYEGvWrEGHDh0wZMgQ7N2715YE/etf/8K4ceOwfPnyMlfEJSIiourL4UTljTfeKPXcqVOnynz+0qVL+P33360n0DifCy1btgyLFy9GWFgY+vbti927d2Py5MmYMmUK9u/fj7///e8YOHAgrly5gvfeew8rV67EwoUL8cQTTzh9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gf79+9ueK97vx2Aw4Pvvv8eQIUPQpk0bvPfee65pKREREdU6lZ71k5SUhG+++Qbffvstzp49W/oEkgSz2XMn1HPWDxERkTweO+unpLCwMLz66qs4ffo0Dhw4gL/97W/w9va29bAQERERyVXpRKVYcnIyNm/ejM2bN6OwsNBV1RIREVEtVqllY/Py8rBixQp888032Llzp22ALXtTiIiIyBVkJSobN27E119/jXXr1qGgoAAAbBsRFicrAQEBmDRpEmbMmOG61hIREVGtIitRGTlypF1SUtyDIoRA165dMWPGDEyaNAl+fn6uaykRERHVOi7ZMTAgIACTJ0/GjBkz0KlTJ1dUSURERCQ/URFCoHv37pgxYwYmTpwIX19fV7aLiIiISF6iMmvWLMyYMQMdO3Z0dXuIiIiIbGQlKp988omr20FERERUSqXHqKxZswbr169HQkIC8vPzsX//fhw8eBAFBQXQaDTo3bu3K9pJREREtZDsROXKlSu47777sHfvXgA3pycDwOeff47FixcDAOLi4hAZGemCphIREVFtI2tl2qKiIowePRp79uxBWVsFPfDAA7bn165dW6kGEhERUe0lK1H5+uuvceDAAbv1U0rq37+/bRbQrl27KtlEIiIiqq1kJSrfffed7fvHH38cffr0sTuuVqvRrl07CCFw5syZyrWQiIiIai1ZicrRo0cBAC1btsRnn32G0NDQUmVCQkIAAOnp6fJbR0RERLWarEQlOzsbkiShbdu25ZYxGAwArONZiIiIiOSQlagEBAQAAC5fvlzmcYvFghMnTgAA9Hq9zKYRERFRbScrUWndujWEEDh8+DB2795d6vi///1vpKWl3bbXhYiIiKgistZRGTp0KPbv3w8hBAYNGgR/f3/bsXbt2tkNoB06dGjlW0lERES1kqwelSeeeAJ16tQBAJhMJly/fh2AdZry6dOnbdOV69Spg+nTp7uoqURERFTbyEpUQkNDsXjxYmg0GkiSVOoLADQaDb766is0aNDApQ0mIiKi2kNWogIAY8eOxbZt22x7+QghbD0pvXr1wpYtWxATEyO7YYWFhfjPf/6D0aNH47777sPnn38Ok8lUZtkPP/wQLVq0kH0uIiIi8kyV2pSwd+/e+P3335GRkYGLFy8CAMLDw1G/fv1KNcpgMCAqKgoHDx60JT8//vgjPvnkE6xYsQIRERF25bOyspCYmFipcxIREZHnkd2jUlJwcDC6du2Krl27VjpJAYD//ve/OHDgAEaNGoU9e/bg4MGDePzxx3H69Gn069cPf/zxhwtaTURERJ6uUj0q7vL999+jdevWWL16NdRqNQCgS5cuuOeee3D//fcjOjoav/76K7p27VrFLSUiIiJ3cihRqcz4D0mScP78eadeEx8fj8cee8yWpBQbMWIEtmzZgqFDh2Lo0KHYuHEjunfvLrttRERE5NkcSlQuXrwISZJK7ZLsiOJZQM7QarXw9vYu81i3bt2wefNmREdHY9iwYdiwYYPT9RMREVH14NQYlbKmIlf0JVd4eDji4uLKPd61a1ds3rwZADB8+HAcOHBA9rmIiIjIczmcqBRPP3bmS67evXtj+/btyM7OLrdMcbKiVquxceNG2eciIiIiz+VQomKxWGR/mc1mpxs1evRoFBYWYsGCBRWWK05WuPEhERFRzeSRs35GjBiBgoKCUoNpy9KlSxecP38eOTk5ty1rMBhgMBjsnjMXGaDW6GS3lYiIiNzHJeuoANZbQ/n5+a6qDjqdDhqNY3lU3bp1ER4eftty8+fPh16vt/s6su39yjaViIiI3EQSlRhMcvnyZcyfPx/r169HcnIyhBAoKirC559/jvT0dGg0GrzyyiuubG+llNWjMvvjfPaoEBERybDwpbpuP4fsWz/79u3D6NGjkZmZaRs4WzzT59KlS3jrrbcgSRJ69+6NqKgolzS2LDk5OVi7di0AYMqUKRWW1el00OnskxK1xvkxNERERKQMWbd+rl27hvvuuw8ZGRkASq+VMm7cONv37l7nJC0tDdOmTcPDDz/s1vMQERGR8mT1qHzyySe4fPlyuYvAdezYEcHBwcjMzMS+ffsq3ciK6PV6TJkypVLrthAREZFnktWjsn79etv3y5Ytw5gxY0qViYyMhBAC586dk904R4SGhiI2NhaLFy9263mIiIhIebISlfj4eEiShK5du2LixIllTiMODAwEYL1NRERERCSHrFs/BQUFAICQkJByy2RlZQEAVCr5M6CTk5OxZMkS7NixA/Hx8baVavV6PVq1aoWoqCg89NBDCAsLk30OIiIi8lyysoh69epBCIGzZ8+WeTw3Nxd//PEHACAoKEhWw95//320bt0ar7/+OrZs2YJr167Bz88Pfn5+uHbtGrZs2YLXXnsN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LITATz/9VG59r776KtasWWN7/Prrr+Pjjz+2++yvXr2K3bt32x77+/vjww8/xLhx4zBu3DiMHz++1NTf9evXl3vOxo0b45///KddmwMCAmyPS34+jrp1XEXJsSklvy+rrLuUjCsA7N271xazcePG4YcffrA7XlHMAOD999+Hj48PACAkJMTWK1FMTtxux9XX27hx4zB48GDb44p+Nsmz8dYPKS4xMbHUcyEhIQ699p///Cd+//13GAwGpKam4qWXXrId8/LyQpcuXTB58mTMmDEDXl5eLmtzeW6dvqzX6+0elxxQe+HCBbtjJW/feKJbBzF36NChVJk777wTK1assD1OSEgot76S09PHjh1b5uyuixcvQghhe5yVlWV3m6QsFZ2zc+fOpWbe6PV65ObmAgCMRmOFdd/KaDRi2bJltscqlQr333+/7XGfPn0QFhaGpKQkADfXVGnfvr1T53HWrTH48ccfKyyfnJwMs9kMtVpd6pi/vz8iIiLsnqvounYVV19vzvxskmdjjwopruRsCQBo2rQpmjZt6tBrBwwYgOPHj+Pvf/872rdvbxvXAFj/iOzduxd/+9vfMHHiRJe2uTy39nyU9Yu/uiqZMACO93o5Ys2aNfjss89cUteNGzfKPVZWz1RlPqP169eXmjnTs2dPNGnSBE2aNEHTpk1LrVfjiVO+LRYLCgoKyjzm6pg5ytXXW03+2axtmKiQok6dOoX//e9/ds85O6CtdevW+OCDDxAXF4f8/HwkJSVh/fr1doNK16xZY/cfmiv/yMrVokULu8e3TuksT1W1/dZp0SUHChc7fvx4ha8p6cknn0SdOnXsHt86kDU8PNzu/UZEREAIUeFXycGm7nZr0mGxWHDp0iW7r5KDvQHrgOviQaDOcvSzLxl3SZKQmpp627j5+/vLalNl2lkRV19vVHMwUSFFCCGwZs0aDBw4EPn5+bbnQ0JC8MILLzhcT2xsLDZs2GDrttVoNGjatClGjRplN1YEgG1hOQC2++3FquL+9OjRo6FS3fyR2759O9544w27P2Lir4XxSs6guLXtqamp7m8sgFGjRtk9/uKLL+zWHvn999+xevVq22NJknD33XeXW1/Xrl3x008/2d6PEAIzZ87EokWLbGUaNGhgN77ozJkzePvtt+3G9gDW2UPbtm3Do48+ajebw53S09PLXBvldspaU6XkZ5qZmVnubQhHr9uS4y+EEHjyySeRk5NTqtzx48fx2muvYeHChQ633xGu+Ply9fVGNQfHqJDbzJo1Cz4+PsjJycGxY8fsxigA1nvG69atc2rg6Nq1a/Hjjz/C19cXbdu2RWhoKNRqNc6dO4dTp07Zymk0GrRq1cr2+NZ77m+++SZ27Nhh+w//22+/hbe3t5y36bBWrVrhySefxMcff2x7bs6cOVi4cKFtevKxY8dw+fJlJCQk2LYRaNCgAerVq4dr164BAOLj49GpUye0bNkSkiThsccew/Dhw13e3hEjRiAqKgrbt28HAFy7dg133XUXunXrBpPJhIMHD9olWdOmTUPbtm0rrLNfv35Ys2YNRo8eDaPRCCEEpk+fDpVKZVvc7O2338bgwYNtdb/88sv46KOP0L59e+h0OqSnp+PkyZO2hPehhx5y+XsvyzfffGP3fmNiYspdxfj999/Hc889Z3scGxtr94c4IiICR44cAQDk5eWhY8eOiIyMhFqtxujRozFlyhRbuZIWL16Mc+fO2X5m3n//fTRt2hSzZ8/G4sWLbcn5mjVrsHnzZtx1110IDAxEVlYWTp06ZUuA58yZU9lw2ImIiLCbQj927Fj06NEDOp0OLVu2xDvvvHPbOtxxvVENodT0Iqr5cJtFmkp+9e7dW5w/f77Meiqa6njvvfc6VP/8+fPt6iwoKBBhYWHlls/NzRVCOD89edu2bXbHb/f6oqIiMXPmzNu2/9bpnS+88EK5ZcubAns7jiz4lpmZKfr373/b9sbExIjCwsIK6y85FXX16tV2U01VKpVYsmSJ7fgPP/xw2wXfir9+//13h+MvhBDh4eF2ZRzVvn17u9etWLGi3LLJyclCkiRbWS8vL5GZmWk7/umnn5b7fmbPnm1XV/fu3cstGxcXZysXFxdnN32/oq8333zT7hwlj5U1Xfp21/3Ro0eFRqMp81wlp2/fbpkAd11vQrhvmj+5H2/9kFtpNBrUqVMHzZo1Q//+/fH0009jz5492L17d6kxG4745z//iTfffBMjR45Eq1atUK9ePajVavj6+qJ169Z48MEHsX37dvzjH/+we523tze2bt2KiRMn2nphqoJarcZnn32GAwcO4PHHH7dtA6DVahESEoKePXvi5ZdfttuUEQD+7//+D2+99RYiIyPd3vNTUr169bBt2zb88MMPGDNmDJo0aQKdTgdvb280a9YM999/PzZu3IiVK1dCp9M5XO/YsWMRGxtruxVmsVjw8MMP23YjHj9+PM6ePYs33ngDffv2RVBQEDQaDby9vREeHo5hw4bhzTffRFxcHPr27euW917SoUOH7KZFBwQEVHjboUmTJujTp4/t8a2zhWbNmoUFCxagc+fOdgujlWX9+vWYPn06mjZtWuHeQe3bt8exY8fw5ZdfYuTIkWjUqBF0Op3t2urTpw9mz56NLVu24JVXXnHkbTvszjvvxMaNGzF48GAEBgbKHrPiruuNqjdJiFuGWhMRERF5CPaoEBERkcdiokJEREQei4kKEREReSwmKkREROSxmKgQERGRx2KiQkRERB6LiQoRERF5LCYqRERE5LGYqBAREZHHYqJCREREHouJChEREXksJipERETksf4fZjSK2MRvNp0AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 600x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "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(\"params_increased_caco_1.pdf\", bbox_inches='tight')\n",
    "plt.show()\n",
    "\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e9a9b5b8",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5d39679f",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dd23cf05",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "id": "2b9fede2",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  5%|██                                      | 100/2000 [00:33<10:49,  2.93it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.285\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:06<09:44,  3.08it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.220\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:39<08:56,  3.17it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.500\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:12<08:44,  3.05it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:1.035\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:45<08:17,  3.02it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.711\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:18<07:43,  3.02it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.468\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [03:51<06:56,  3.12it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.293\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:24<06:49,  2.93it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.178\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [04:58<06:06,  3.00it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.110\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:32<05:34,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.071\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:06<05:03,  2.96it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.048\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [06:39<04:16,  3.12it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.034\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:11<03:50,  3.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.025\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [07:45<03:18,  3.02it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.020\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:17<02:45,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.016\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [08:51<02:11,  3.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.013\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [09:25<01:48,  2.77it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.011\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [09:58<01:07,  2.94it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [10:32<00:32,  3.05it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:06<00:00,  3.00it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.007\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "  5%|██                                      | 100/2000 [00:33<10:39,  2.97it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.285\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:07<10:03,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.219\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:40<09:33,  2.96it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.496\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:14<08:51,  3.01it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:1.032\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:47<08:00,  3.12it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.711\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:20<07:37,  3.06it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.475\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [03:54<07:08,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.304\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:27<06:40,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.192\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [05:00<05:55,  3.10it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.122\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:34<05:46,  2.89it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.080\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:08<05:00,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.055\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [06:42<04:36,  2.90it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.039\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:15<03:53,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.029\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [07:49<03:22,  2.96it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.023\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:23<02:54,  2.86it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.018\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [08:57<02:09,  3.08it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.015\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [09:30<01:40,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.012\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [10:03<01:06,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.010\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [10:37<00:33,  3.01it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:11<00:00,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.008\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "  5%|██                                      | 100/2000 [00:33<10:43,  2.95it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.287\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:07<10:18,  2.91it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.221\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:40<09:07,  3.10it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.499\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:13<08:52,  3.01it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:1.035\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:46<08:24,  2.97it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.713\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:19<07:56,  2.94it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.474\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [03:51<06:53,  3.14it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.302\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:25<06:43,  2.97it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.189\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [04:58<05:47,  3.16it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.119\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:31<05:29,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.077\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:05<05:02,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.053\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [06:39<04:26,  3.00it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.038\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:11<03:50,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.028\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [07:44<03:15,  3.07it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.022\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:17<02:47,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.017\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [08:50<02:12,  3.01it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.014\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [09:24<01:38,  3.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.012\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [09:57<01:06,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.010\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [10:30<00:34,  2.90it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:04<00:00,  3.01it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.007\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "  5%|██                                      | 100/2000 [00:33<10:28,  3.02it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.278\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:07<10:02,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.215\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:41<09:29,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.497\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:14<08:46,  3.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:1.035\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:47<08:28,  2.95it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.715\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:20<08:01,  2.91it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.478\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [03:53<07:12,  3.01it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.309\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:27<06:56,  2.88it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.196\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [04:59<06:21,  2.88it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.125\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:32<05:20,  3.12it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.083\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:06<04:57,  3.02it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.057\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [06:39<04:30,  2.96it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.041\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:13<03:46,  3.09it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.031\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [07:47<03:25,  2.92it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.024\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:20<02:43,  3.05it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.019\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [08:53<02:22,  2.80it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.015\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [09:27<01:39,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.013\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [10:01<01:07,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.011\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [10:34<00:32,  3.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:08<00:00,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.008\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "  5%|██                                      | 100/2000 [00:34<10:45,  2.94it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[100,  100] loss:3.281\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|████                                    | 200/2000 [01:08<10:05,  2.97it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[200,  100] loss:2.214\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|██████                                  | 300/2000 [01:42<09:45,  2.90it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[300,  100] loss:1.490\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████                                | 400/2000 [02:16<08:46,  3.04it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[400,  100] loss:1.024\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████                              | 500/2000 [02:50<08:08,  3.07it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[500,  100] loss:0.701\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████                            | 600/2000 [03:24<07:38,  3.05it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[600,  100] loss:0.463\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|██████████████                          | 700/2000 [03:57<07:10,  3.02it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[700,  100] loss:0.295\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████████████████                        | 800/2000 [04:30<06:27,  3.10it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[800,  100] loss:0.185\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|██████████████████                      | 900/2000 [05:03<06:45,  2.71it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[900,  100] loss:0.118\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|███████████████████▌                   | 1000/2000 [05:36<05:30,  3.03it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1000,  100] loss:0.078\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████▍                 | 1100/2000 [06:10<05:23,  2.78it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1100,  100] loss:0.054\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████▍               | 1200/2000 [06:43<04:17,  3.11it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1200,  100] loss:0.039\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|█████████████████████████▎             | 1300/2000 [07:16<03:48,  3.06it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1300,  100] loss:0.029\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|███████████████████████████▎           | 1400/2000 [07:50<03:29,  2.87it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1400,  100] loss:0.023\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|█████████████████████████████▎         | 1500/2000 [08:24<02:51,  2.91it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1500,  100] loss:0.018\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|███████████████████████████████▏       | 1600/2000 [08:58<02:10,  3.06it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1600,  100] loss:0.015\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|█████████████████████████████████▏     | 1700/2000 [09:32<01:36,  3.10it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1700,  100] loss:0.012\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████    | 1800/2000 [10:05<01:04,  3.09it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1800,  100] loss:0.010\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|█████████████████████████████████████  | 1900/2000 [10:38<00:33,  2.99it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1900,  100] loss:0.009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████| 2000/2000 [11:11<00:00,  2.98it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2000,  100] loss:0.008\n",
      "Finished Training\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "n_epochs = 2000\n",
    "n_batches = 100\n",
    "\n",
    "batch_size = 32\n",
    "seq_len = 128\n",
    "\n",
    "n_seeds = [1234,1235,1236,1237,1238]\n",
    "\n",
    "Attn_Values = []\n",
    "Prediction_Values = []\n",
    "\n",
    "# total number of training steps\n",
    "total_steps = n_epochs * n_batches\n",
    "\n",
    "\n",
    "\n",
    "for seed in n_seeds:\n",
    "    torch.manual_seed(seed)\n",
    "    model = Net_CACO().to(device)\n",
    "    \n",
    "    criterion  = nn.CrossEntropyLoss()\n",
    "    \n",
    "    lr_query_key = 0.01\n",
    "    \n",
    "    lr_prediction = 0.01\n",
    "    \n",
    "    optimizer = optim.SGD([\n",
    "        {\"params\":model.key_layer.parameters(), \"lr\":lr_query_key},\n",
    "        {\"params\":model.prediction_layer.parameters(),\"lr\":lr_prediction}])\n",
    "    \n",
    "    scheduler = torch.optim.lr_scheduler.LambdaLR(\n",
    "        optimizer,\n",
    "        lr_lambda=[lambda step: 1 + (10- 1) * step / total_steps,  # key_layer\n",
    "                   lambda step: 1\n",
    "                  ])\n",
    "\n",
    "    \n",
    "    \n",
    "    running_loss =0.0\n",
    "    \n",
    "    common_token_attn = []\n",
    "    distinct_token_attn = []\n",
    "    \n",
    "    \n",
    "    \n",
    "    for epoch in tqdm(range(n_epochs)):\n",
    "        model.train()\n",
    "        \n",
    "        for j in range(n_batches):\n",
    "            \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",
    "            \n",
    "            optimizer.zero_grad()\n",
    "            \n",
    "            outputs,_ = model(inputs)\n",
    "            loss = criterion(outputs,labels)\n",
    "            \n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            scheduler.step()\n",
    "            \n",
    "            running_loss +=loss.item()\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": 56,
   "id": "fe81edda",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss:0.007\n",
      "accuracy:1.000\n"
     ]
    }
   ],
   "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": 57,
   "id": "ed6e5265",
   "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": 58,
   "id": "84573886",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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Llm3fi5evAhJktC5c2enIqVIq1at0LlzZwghcOnSpQoHSURERDWTrEJFrVYDAIxGY5n7mEwmp32JiIiI3CWrUImKioIQAnv37sW2bdtKbN++fTt2794NSZIQFRVV4SCJiIioZpLVR6Vr1644ffo0zGYz+vXrh4SEBLRt2xaSJCEpKQmLFy+GxWKBJEno2rWrp2MmIiKiGkISMpZFXrduHeLj4x3DkCVJctpe1KQkSVizZg3uuOMOz0TrBY+/e7WyQyAiIqqSZr0c5vVjyLr1079/f4wcOdKpSCmaQwWA47nhw4f7dJFCREREvk32ooTfffcdHnnkEQBwmp226Ptx48bhu+++q2B4REREVJPJuvVT3LFjx7By5UqcOXMGQgg0btwYd955J1q2bOmpGL2Kt36IiIjkUeLWj6zOtJs3b3Z83717d7zwwgseC6g8JpMJixYtwp49e1BQUIBGjRohISEBt9xyiyLHJyIiImXJuqKiUqkgSRJiYmJw+vRpjwc1a9YsrFy5EsuXL4dKZb87dejQIQwdOhQpKSlOt5rUajX+9a9/4aWXXpJ1LF5RISIiksdnr6gYDAbk5OSgdevWno4HAPDtt98iKCjIUaSYzWbce++9SElJwYgRIzB06FCEhobi8OHD+Oijj/Dqq6+iRYsWGDp0qFfiISIiosohqzNt+/btIYRAenq6p+MBAPz1119OfVxWr16N06dP44033sCCBQswZswYDBkyBC+++CL27NmDyMhI/O9///NKLERERFR5ZBUqTz31FAD7Ssk7d+70aEAAcO3aNQQFBTkeHzt2DJIkYdKkSSX2jYqKwj333IM///zT43EQERFR5ZJ166dTp0546KGH8N133+HOO+/ECy+8gD59+qBevXqO2zXFRUdHu9V+w4YNcfToUcdjf39/AIBWqy11f61WiwoOXiIiIiIfJKtQadSoESRJgiRJuHr1Kv75z3+Wua8kSbBYLG61f+edd+Lzzz/H4cOH0bp1a8csuN9//z2effZZp32zs7OxYsWKUldxJiIioqpN9oRvRYqm0S/vy12vvfYaDAYD+vfvj7lz56JevXqYPn06XnnlFUyZMgUbN27E/v37MX/+fPTq1Qtnz57Fk08+WdG3QkRERD6mQsOTy3tp8XWArFar24Ht378f99xzD86ePQtJkhAZGYnLly/DZrM57SeEwPPP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CzG4qx/89BI6t9Zh5dZCme+iamHOlcecK485Vx5z7l2yhidTxWg1QNP69hrx8Glzmfsd+Xtbq8Zal9uuU0uFcIO63LaNZuDkWfsvSqtGrrddlTHnymPOlcecK485974KFyoWiwWbN2/G3Llz8fnnn3sipmqvbrgaKpW9E9T5y2X33zl/2V49G4JUCPBzbeHHehHXlzEov237troRNWPZA+Zcecy58phz5THn3lehQuWrr75CvXr10LdvXzzyyCN4+umnAQAPPPAAmjRpgubNmyM7O9sjgVYnhqDrac/KLbsvc1bu9ct8oUGundjObZd9r7Jom79egr56FuFOmHPlMefKY86Vx5x7n+xCZerUqXjiiSdw+fJlxwrKRQOI4uPjcebMGZw8eRLLli3zVKzVhp/u+vcmc9kntslyfZte59qJ7VdsP1PZVyFltV2VMefKY86Vx5wrjzn3PlmFyp49ezBjxgwA9uHHN074dvfddzuGMK9fv76CIRIREVFNJatQ+fTTTx1XTxo0aID69es7bQ8PD8ctt9wCIQT2799f4SCrm8JikwjqtGVXvzrN9W1Gk2vT3RQW209XziVAOW1XZcy58phz5THnymPOvU9WobJ582YAQFBQEPbs2YMuXbqU2KdRo0YAgNTUVPnRVVPZecXuVQaXfWKHBhe7P5nn2snn3HbZP96ibQVGAWM5lxSrC+Zcecy58phz5THn3ierUElPT4ckSejevTsiIyNL3Uev1wMArl27Jj+6aio90wqbzX6i1iunl3a9CPuPJzvPhvxC107s4j3Dy2/bvi29nJ7k1QlzrjzmXHnMufKYc++TVahoNPYx4zZb2b2Qi66k+Pv7yzlEtWa2AKfO2ce9ty5nTH3RePsjya6XyBlXbMjMtp+srZuU3rZOC9zSwP4zPHKmGpbfpWDOlcecK485Vx5z7n2yCpW6detCCIE//vgDeXkl1y3Yt28fDhw4AEmS0LBhwwoHWR3tOGS/sdksRoNGdUtWyre30CIyzP78zsPurYxZtH/HFjqEh5T8Ecd10MNPL8FqE9jlZttVGXOuPOZcecy58phz75JVqHTv3h0AkJubi/j4eJw+fdqx7T//+Q8GDx7seNytW7cKhlg97Uwy4exFC1SShEnDgtA8xl4RSwBua25fxAoADp0yl5hu+a4efpj1chhmvRxW6om79o9CZOfZ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",
      "text/plain": [
       "<Figure size 600x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "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(\"params_increasing_caco.pdf\", bbox_inches='tight')\n",
    "plt.show()\n",
    "\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c0a24f31",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 50,
   "id": "be6f461f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.2"
      ]
     },
     "execution_count": 50,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "2e-1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "be1ba54f",
   "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.20"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
