{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 89,
   "id": "8ef6e587",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def attention(q, k, v):\n",
    "    \"\"\"\n",
    "    q: Queries tensor of shape (batch_size, d_k, seq_len_q)\n",
    "    k: Keys tensor of shape (batch_size, d_k, seq_len_k)\n",
    "    v: Values tensor of shape (batch_size, d_v, seq_len_k)\n",
    "    \"\"\"\n",
    "\n",
    "    # Compute attention scores\n",
    "    scores = torch.einsum('bdk,bdq->bkq', k, q) \n",
    "    #numerical stability\n",
    "    abs_scores = scores.abs() \n",
    "    idx = abs_scores.argmax(dim=1)\n",
    "    idx_expanded = idx.unsqueeze(1)\n",
    "    maxabs_vals = scores.gather(dim=1, index=idx_expanded)\n",
    "\n",
    "    result = maxabs_vals.expand_as(scores)\n",
    "\n",
    "    scores = scores-result\n",
    "\n",
    "    # Apply softmax to get attention weights\n",
    "    attn = F.softmax(scores/torch.sqrt(torch.tensor([k.size(-2)])), dim=-2)\n",
    "    \n",
    "    output = torch.einsum('bvk,bkq->bvq', v, attn) \n",
    "\n",
    "    return output\n",
    "\n",
    "def expression_1(Z_star, delta_m):\n",
    "    \"\"\"\n",
    "    Computes the expression from the image:\n",
    "    (2 / delta_m) * [ReLU(Z + delta_m) - ReLU(Z + delta_m / 2)] +\n",
    "    (2 / delta_m) * [ReLU(-Z + delta_m) - ReLU(-Z + delta_m / 2)] - 1\n",
    "    \n",
    "    Parameters:\n",
    "    - Z_star: torch.Tensor, the input tensor Z^*\n",
    "    - delta_m: float or torch scalar, the scalar delta_m\n",
    "\n",
    "    Returns:\n",
    "    - torch.Tensor with the same shape as Z_star\n",
    "    \"\"\"\n",
    "    term1 = F.relu(Z_star + delta_m) - F.relu(Z_star + delta_m / 2)\n",
    "    term2 = F.relu(-Z_star + delta_m) - F.relu(-Z_star + delta_m / 2)\n",
    "    \n",
    "    result = (2 / delta_m) * (term1 + term2)\n",
    "    return result\n",
    "\n",
    "#FFN_1\n",
    "def FFN_1(x):\n",
    "    token_len = x.size(-2)-x.size(-1)\n",
    "    seq_len = x.size(-1)\n",
    "    identity_n = torch.eye(seq_len)\n",
    "    mat_1_1 = torch.cat([torch.zeros((seq_len,token_len)),identity_n],dim=-1)\n",
    "    # Construct the components\n",
    "    top_row = torch.stack([torch.zeros((1, seq_len))]*x.size(0), dim=0)  # 0_{1 x n} (because 0 and 0 concatenated)\n",
    "\n",
    "    identity = torch.eye(seq_len - 1)    # I_{n-1}\n",
    "    minus_ones = -torch.ones((seq_len - 1, 1))  # -1_{n-1}\n",
    "\n",
    "    # Concatenate horizontally: [I_{n-1} | -1_{n-1}]\n",
    "    bottom_block = torch.cat([identity, minus_ones], dim=1)\n",
    "\n",
    "    # Concatenate vertically with the top row\n",
    "    # result_matrix = torch.cat([top_row, bottom_block], dim=0)\n",
    "    result_matrix = bottom_block\n",
    "    mat1 = torch.matmul(result_matrix,mat_1_1)\n",
    "\n",
    "    z_1 = torch.einsum(\"it,bts->bis\",mat1,x)\n",
    "\n",
    "    #numerical stable\n",
    "    z_1 = 1e3*z_1\n",
    "    idx = (z_1.abs()).argmin(dim=1)\n",
    "    idx_expanded = idx.unsqueeze(1)\n",
    "    minabs_vals = z_1.gather(dim=1, index=idx_expanded)\n",
    "    z_1 = z_1-minabs_vals.expand_as(z_1)\n",
    "    z_1 = 1e4*z_1\n",
    "    idx = (z_1.abs()).argmin(dim=1)\n",
    "    idx_expanded = idx.unsqueeze(1)\n",
    "    minabs_vals = z_1.gather(dim=1, index=idx_expanded)\n",
    "    z_1 = z_1-minabs_vals.expand_as(z_1)\n",
    "    z_1 = 1e4*z_1\n",
    "    idx = (z_1.abs()).argmin(dim=1)\n",
    "    idx_expanded = idx.unsqueeze(1)\n",
    "    minabs_vals = z_1.gather(dim=1, index=idx_expanded)\n",
    "    z_1 = z_1-minabs_vals.expand_as(z_1)\n",
    "\n",
    "\n",
    "    o_1 = expression_1(z_1,delta_m=1e-4)\n",
    "    \n",
    "    o_1 = torch.cat([top_row, o_1], dim=-2)\n",
    "    \n",
    "    o_1 = torch.cat([torch.zeros(x.size(0),token_len,seq_len), o_1], dim=-2)\n",
    "    return o_1\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "id": "07192f8b",
   "metadata": {},
   "outputs": [],
   "source": [
    "M=10000 #which\n",
    "\n",
    "class construction():\n",
    "    def __init__(self,seq_len,token_len,m):\n",
    "        zeros = torch.zeros(size=(seq_len,token_len))\n",
    "        identity = torch.eye(seq_len)\n",
    "        self.W_K = torch.concat([zeros,(1/m) * identity],dim=-1)  # (seq_len,token_len+seq_len)\n",
    "        self.W_Q = torch.concat([zeros,(m**0.5) * identity],dim=-1) # (seq_len,token_len+seq_len)\n",
    "        self.W_V = torch.concat((torch.eye(token_len),torch.zeros(size=(token_len,seq_len))),dim=-1) # (token_len,token_len+seq_len)\n",
    "\n",
    "construction1 = construction(seq_len=8,token_len=10,m=M)\n",
    "\n",
    "def SA_2(x):\n",
    "    k = torch.einsum(\"kx,bxs->bks\",construction1.W_K,x)\n",
    "    q = torch.einsum(\"qx,bxs->bqs\",construction1.W_Q,x)\n",
    "    v = torch.einsum(\"vx,bxs->bvs\",construction1.W_V,x)\n",
    "    output = attention(q,k,v)\n",
    "    return output\n",
    "\n",
    "class SA_1(nn.Module):\n",
    "    def __init__(self,d_hidden,d_input,seq_len):\n",
    "        super().__init__()\n",
    "        self.W_K = nn.Linear(d_input-seq_len,d_hidden,bias=False)\n",
    "        self.W_Q = nn.Linear(d_input,d_hidden,bias=False)\n",
    "        #random initialization\n",
    "        nn.init.xavier_normal_(self.W_K.weight,gain=1.0)\n",
    "        nn.init.xavier_normal_(self.W_Q.weight,gain=1.0)\n",
    "        token_len = d_input - seq_len\n",
    "        self.pre = torch.concat((torch.eye(token_len),torch.zeros(size=(token_len,seq_len))),dim=-1)\n",
    "\n",
    "    def forward(self,x_p):\n",
    "        k = self.W_K(torch.einsum(\"ti,bis->bts\",self.pre,x_p).permute(0,2,1)).permute(0,2,1)\n",
    "  \n",
    "        q = self.W_Q(x_p.permute(0,2,1)).permute(0,2,1)\n",
    "        output = attention(q,k,x_p)\n",
    "\n",
    "        return output\n",
    "\n",
    "class induction_head(nn.Module):\n",
    "    def __init__(self,d_hidden,d_input,seq_len):\n",
    "        super().__init__()\n",
    "        self.sa_1 = SA_1(d_hidden,d_input,seq_len)\n",
    "    \n",
    "    def forward(self,x):\n",
    "        x_temp = x\n",
    "        x = self.sa_1(x)\n",
    "        x = FFN_1(x)\n",
    "        x = M*x+x_temp\n",
    "        x = SA_2(x)\n",
    "        return x\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 91,
   "id": "c0b76887",
   "metadata": {},
   "outputs": [],
   "source": [
    "Total = 50\n",
    "rounds  = 1000\n",
    "avg = []\n",
    "\n",
    "for _ in range(Total):\n",
    "    dif = []\n",
    "    for round in range(rounds):   \n",
    "        batch_size = 2\n",
    "        tokens = torch.cat((torch.tensor([[5,7],[6,4],[7,-5],[3,-1],[1,2],[5,6],[7,8],[-9,0],[-1,8],[0,0]]),torch.randint(-10,10,(10,5)),torch.tensor([[5],[6],[7],[3],[1],[5],[7],[-9],[-1],[0]])),dim=-1)\n",
    "        tensor = torch.cat((tokens,torch.eye(8)),dim=-2)\n",
    "        tensor = torch.stack(batch_size*[tensor])\n",
    "        induction_head_0 = induction_head(d_hidden=100,d_input=18,seq_len=8)\n",
    "        output = induction_head_0(tensor)\n",
    "        induction = output[0,:,1]\n",
    "        _ = torch.sum(torch.abs(induction - torch.tensor([7,4,-5,-1,2,6,8,0,8,0])))\n",
    "        _ = _.detach().numpy()\n",
    "        dif.append(_)\n",
    "\n",
    "    _ = sum(dif)/rounds\n",
    "    avg.append(_)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 92,
   "id": "22fb5647",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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M/Xr1zc9e/a0HzM9Pd2sW7fOSDJ33HGHvd0YY9LS0kzjxo1NWFiYefXVV823335rnnnmGePv72+GDRtmP86+ffvs561Pnz7m3//+t1m2bJnZt2+fWb58ufHz8zNXX321SUhIMF9//bUZNmyYkWTefPNN+z7efPNNI8k0a9bMPPzww+abb74xCxcuNDVr1jR9+vRx+KzOfJ/feecdY7PZTGxsrPnkk0/M559/bm655RZTuXJl8+23317w52iMMRkZGSY0NNTUrVvXLFiwwHz99ddm5MiRRpKJi4uzrzdmzBgTGBhoMjMzHbafN2+ekWS2bt1qjPHM979Lly6mRo0aJi8vzxjj/PeysIbJkycX2WdYWJgZOnSo/b2VczZ06FBjs9nMY489Zj9foaGhJigoyGGfDz30kKlSpYqZM2eOWbFihfniiy/MjBkzzMsvv3zBnwEqHoIdvNLRo0fNVVddZSQZScbX19f06NHDxMfHm5MnT9rX27t3r6lcubK59957nd53QUGBOXfunDlw4ICRZP773//alz3//PNGktm3b1+R7dq2bWsPEueLj483lSpVMj/88IND+7///W8jySxdutTeJskEBweb48ePO1Vr4T9s586dM+fOnTOHDx82EyZMMJLMnXfeaV/vz8GuMAjNnDnTYX8JCQlGknnttdcu+bmKY2W/lwpr57vUujt37iwSPM4PdoXCwsLMzTffXGR7SWbEiBEObQ899JCpVq2aOXDggEP7rFmzjCSzfft2Y8z/BbuIiIgi/4PRqlUr06lTJ3Pu3DmH9ltuucU0aNDA5OfnG2P+LyT84x//cFhv5syZRpJJS0szxjj3fT516pSpVauW6d+/v0N7fn6+6dChg+natesFtzXG2L8/GzZscGiPi4szNpvN7Nq1yxhjzNatW4ucU2OM6dq1q4mKirK/L83vf1pamnn66aeNJLNgwQL7ela+l1aD3aXOWeF3c8yYMQ7rvffee0aSwz7btWtnYmNjnfrsqLi4FAuvVLt2ba1evVo//PCDZsyYoVtvvVU///yzJk6cqMjISPsl1MTEROXn52vEiBEX3V9GRoaGDx+uxo0by8fHR76+vgoLC5Mk7dy587Jq/eKLL9SuXTt17NhReXl59tf1119vv5R7vr59+6pmzZpO73/79u3y9fWVr6+vGjZsqNmzZ+vee+/V66+/fsFt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rV692tMlVUirVq1S//791bBhQ9lsNn366acOy40xmjJliho2bKjAwED17t1b27dv90yxFUx8fLyuvPJKVa9eXfXq1VNsbKx27drlsA7nx3Pmz5+v9u3b2+fbio6O1ldffWVfzrkpO+Lj42Wz2TR69Gh7G+fHc6ZMmSKbzebwql+/vn15aZwbgp2LJSQkaPTo0XryySeVnJysq6++WjfeeKNSU1M9XVqFc+rUKXXo0EGvvPJKsctnzpypOXPm6JVXXtEPP/yg+vXrq1+/fvbHzsF9kpKSNGLECK1fv16JiYnKy8tTTEyMTp06ZV+H8+M5jRo10owZM7Rx40Zt3LhRffv21a233mr/B4hzUzb88MMPeu2119S+fXuHds6PZ7Vt21ZpaWn2148//mhfVirnxsClunbtaoYPH+7Q1qpVKzNhwgQPVQRjjJFk/vOf/9jfFxQUmPr165sZM2bY286ePWuCg4PNggULPFBhxZaRkWEkmaSkJGMM56csqlmzplm4cCHnpow4efKkadGihUlMTDS9evUyjzzyiDGG3x1Pmzx5sunQoUOxy0rr3NBj50K5ubnatGmTYmJiHNpjYmK0du1aD1WF4uzbt0/p6ekO58rf31+9evXiXHlAZmamJKlWrVqSOD9lSX5+vpYsWaJTp04pOjqac1NGjBgxQjfffLOuu+46h3bOj+ft3r1bDRs2VHh4uO6++27t3btXUumdmwr3rFh3Onr0qPLz8xUSEuLQHhISovT0dA9VheIUno/iztWBAwc8UVKFZYzR2LFjddVVV6ldu3aSOD9lwY8//qjo6GidPXtW1apV03/+8x+1adPG/g8Q58ZzlixZos2bN+uHH34osozfHc/q1q2bFi9erCuuuEK//fabpk+frh49emj79u2ldm4Idm5gs9kc3htjirShbOBced7IkSO1detWfffdd0WWcX48p2XLlkpJSdGJEyf08ccfa+jQoUpKSrIv59x4xsGDB/XII49o2bJlCggIuOB6nB/PuPHGG+3/HRkZqejoaEVEROjtt99W9+7dJbn/3HAp1oXq1KmjypUrF+mdy8jIKJLQ4VmFo5Q4V5718MMP67PPPtOKFSvUqFEjezvnx/P8/PzUvHlzdenSRfHx8erQoYNefPFFzo2Hbdq0SRkZGYqKipKPj498fHyUlJSkl156ST4+PvZzwPkpG6pWrarIyEjt3r271H53CHYu5Ofnp6ioKCUmJjq0JyYmqkePHh6qCsUJDw9X/fr1Hc5Vbm6ukpKSOFelwBijkSNH6pNPPtHy5csVHh7usJzzU/YYY5STk8O58bBrr71WP/74o1JSUuyvLl266N5771VKSoqaNWvG+SlDcnJytHPnTjVo0KD0fndcNgwDxhhjlixZYnx9fc0bb7xhduzYYUaPHm2qVq1q9u/f7+nSKpyTJ0+a5ORkk5ycbCSZOXPmmOTkZHPgwAFjjDEzZswwwcHB5pNPPjE//vijGTRokGnQoIHJysrycOXeLy4uzgQHB5uVK1eatLQ0++v06dP2dTg/njNx4kSzatUqs2/fPrN161bzxBNPmEqVKplly5YZYzg3Zc35o2KN4fx40qOPPmpWrlxp9u7da9avX29uueUWU716dXsGKI1zQ7Bzg3/9618mLCzM+Pn5mc6dO9uncEDpWrFihZFU5DV06FBjzB9DzydPnmzq169v/P39zTXXXGN+/PFHzxZdQRR3XiSZN998074O58dz7r//fvvfsLp165prr73WHuqM4dyUNX8Odpwfzxk4cKBp0KCB8fX1NQ0bNjS333672b59u315aZwbmzHGuK7/DwAAAJ7CPXYAAABegmAHAADgJQh2AAAAXoJgBwAA4CUIdgAAAF6CYAcAAOAlCHYAAABegmAHAADgJQh2AFAGDRs2TLGxsZ4uA0A5Q7ADUGENGzZMNptNNptNPj4+atKkieLi4vT77797ujQAKBGCHYAK7YYbblBaWpr279+vhQsX6vPPP9c//vEPT5cFACVCsANQofn7+6t+/fpq1KiRYmJiNHDgQC1btkySVFBQoGnTpqlRo0by9/dXx44d9fXXX9u3XblypWw2m06cOGFvS0lJkc1m0/79+yVJb731lmrUqKFvvvlGrVu3VrVq1exhslB+fr7Gjh2rGjVqqHbt2nr88cfFY7wBlATBDgD+v7179+rrr7+Wr6+vJOnFF1/U7NmzNWvWLG3dulXXX3+9BgwYoN27d1va7+nTpzVr1iy98847WrVqlVJTUzVu3Dj78tmzZ2vRokV644039N133+n48eP6z3/+49LPBqBiINgBqNC++OILVatWTYGBgYqIiNCOHTs0fvx4SdKsWbM0fvx43X333WrZsqWee+45dezYUXPnzrV0jHPnzmnBggXq0qWLOnfurJEjR+p///ufffncuXM1ceJE/eUvf1Hr1q21YMECBQcHu/JjAqggfDxdAAB4Up8+fTR//nydPn1aCxcu1M8//6yHH35YWVlZOnz4sHr27Omwfs+ePbVlyxZLx6hSpYoiIiLs7xs0aKCMjAxJUmZmptLS0hQdHW1f7uPjoy5dunA5FoBl9NgBqNCqVq2q5s2bq3379nrppZeUk5OjqVOn2pfbbDaH9Y0x9rZKlSrZ2wqdO3euyDEKL+2ev09CGwB3INgBwHkmT56sWbNmKTs7Ww0bNtR3333nsHzt2rVq3bq1JKlu3bqS5DAQIiUlxdLxgoOD1aBBA61fv97elpeXp02bNpXwEwCoyLgUCwDn6d27t9q2bat//vOfeuyxxzR58mRFRESoY8eOevPNN5WSkqL33ntPktS8eXM1btxYU6ZM0fTp07V7927Nnj3b8jEfeeQRzZgxQy1atFDr1q01Z84ch5G2AOAsgh0A/MnYsWP117/+VT///LOysrL06KOPKiMjQ23atNFnn32mFi1aSPrjEusHH3yguLg4dejQQVdeeaWmT5+uO++809LxHn30UaWlpWnYsGGqVKmS7r//ft12223KzMx0x8cD4MVshhs9AAAAvAL32AEAAHgJgh0AAICXINgBAAB4CYIdAACAlyDYAQAAeAmCHQAAgJcg2AEAAHgJgh0AAICXINgBAAB4CYIdAACAlyDYAQAAeAmCHQAAgJf4f4TDG2UoYnY6AAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "x = list(range(len(avg)))\n",
    "y = avg\n",
    "plt.scatter(x, y,s=1)\n",
    "plt.xlabel('Round')\n",
    "plt.ylabel('Average Sum of Absolute Difference')\n",
    "plt.title('Scatter Plot of Difference over Rounds')\n",
    "plt.tight_layout()\n",
    "plt.savefig(\"scatter.svg\",format = \"svg\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 93,
   "id": "c7540bb2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "np.float32(0.14)"
      ]
     },
     "execution_count": 93,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import numpy as np\n",
    "np.mean(avg)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "aipy12dit",
   "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.12.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
