{
  "cells": [
    {
      "cell_type": "markdown",
      "id": "Br0PvlFNPwab",
      "metadata": {
        "id": "Br0PvlFNPwab"
      },
      "source": [
        "Preliminaries"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "id": "HZYXQ1mTxG1q",
      "metadata": {
        "id": "HZYXQ1mTxG1q"
      },
      "outputs": [],
      "source": [
        "# install dependencies\n",
        "import numpy as np\n",
        "import torch\n",
        "import torch.nn as nn\n",
        "from einops import rearrange"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "id": "3tj6x-Jleuej",
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "3tj6x-Jleuej",
        "outputId": "ea538941-11e6-4fd0-8570-894e38ef23ef"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "--2024-12-01 09:19:53--  https://nlp.stanford.edu/data/glove.twitter.27B.zip\n",
            "Resolving nlp.stanford.edu (nlp.stanford.edu)... 171.64.67.140\n",
            "Connecting to nlp.stanford.edu (nlp.stanford.edu)|171.64.67.140|:443... connected.\n",
            "HTTP request sent, awaiting response... 301 Moved Permanently\n",
            "Location: https://downloads.cs.stanford.edu/nlp/data/glove.twitter.27B.zip [following]\n",
            "--2024-12-01 09:19:53--  https://downloads.cs.stanford.edu/nlp/data/glove.twitter.27B.zip\n",
            "Resolving downloads.cs.stanford.edu (downloads.cs.stanford.edu)... 171.64.64.22\n",
            "Connecting to downloads.cs.stanford.edu (downloads.cs.stanford.edu)|171.64.64.22|:443... connected.\n",
            "HTTP request sent, awaiting response... 200 OK\n",
            "Length: 1520408563 (1.4G) [application/zip]\n",
            "Saving to: ‘glove.twitter.27B.zip’\n",
            "\n",
            "glove.twitter.27B.z 100%[===================>]   1.42G  4.99MB/s    in 4m 44s  \n",
            "\n",
            "2024-12-01 09:24:39 (5.10 MB/s) - ‘glove.twitter.27B.zip’ saved [1520408563/1520408563]\n",
            "\n",
            "Archive:  glove.twitter.27B.zip\n",
            "  inflating: glove.twitter.27B.25d.txt  \n",
            "  inflating: glove.twitter.27B.50d.txt  \n",
            "  inflating: glove.twitter.27B.100d.txt  \n",
            "  inflating: glove.twitter.27B.200d.txt  \n"
          ]
        }
      ],
      "source": [
        "!wget https://nlp.stanford.edu/data/glove.twitter.27B.zip\n",
        "!unzip glove.twitter.27B.zip"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# this cell contains the code for locality sensitive hash function used in kdeformer, with no changes compared to the kdeformer implementation\n",
        "# the code is taken from https://github.com/majid-daliri/kdeformer\n",
        "class CosineHammingLSH(torch.nn.Module):\n",
        "\n",
        "    def __init__(self, Bucket_size=64):\n",
        "        super().__init__()\n",
        "        self.Bucket_size = Bucket_size\n",
        "\n",
        "    def select(self, matrix, indices):\n",
        "        Offset = torch.zeros_like(indices)\n",
        "        n = matrix.shape[2]\n",
        "        Offset += n * torch.arange(Offset.shape[1], device=matrix.device).unsqueeze(0).unsqueeze(-1)\n",
        "        Offset += n * Offset.shape[1] * torch.arange(Offset.shape[0], device=matrix.device).unsqueeze(-1).unsqueeze(-1)\n",
        "        indices_flat = (indices + Offset).view(-1)\n",
        "        return torch.index_select(matrix.view(-1, matrix.shape[3]), 0, indices_flat).view(matrix.shape)\n",
        "    def forward(self, query, key, weight, K_sort_idx, Q_sort_idx, input_attn_mask=None):\n",
        "        num_blocks = key.shape[2] // self.Bucket_size\n",
        "        query_Bucket_size = query.shape[2] // num_blocks\n",
        "\n",
        "        query_sorted = self.select(query, Q_sort_idx)\n",
        "        key_sorted = self.select(key, K_sort_idx)\n",
        "        weight_sorted = self.select(weight, K_sort_idx)\n",
        "        key_split_per_block = key_sorted.view(-1, self.Bucket_size, key.shape[3])\n",
        "        query_split_per_block = query_sorted.view(-1, query_Bucket_size, query.shape[3])\n",
        "\n",
        "        weight_split_per_block = weight_sorted.view(-1, self.Bucket_size, weight.shape[3])\n",
        "        A_sparse = torch.exp(torch.einsum('bnd,bmd->bnm', query_split_per_block, key_split_per_block))\n",
        "\n",
        "        if input_attn_mask is not None:\n",
        "            mask_split_per_block = input_attn_mask.view(-1, self.Bucket_size, weight.shape[3]).unsqueeze(0).unsqueeze(0)\n",
        "            A_sparse *= mask_split_per_block\n",
        "\n",
        "        result = torch.bmm(A_sparse, weight_split_per_block)\n",
        "        result = result.view(query.shape[0], query.shape[1], query.shape[2], weight.shape[3])\n",
        "\n",
        "        Q_sort_idx_new = torch.argsort(Q_sort_idx, dim=2)\n",
        "        result = self.select(result, Q_sort_idx_new)\n",
        "\n",
        "        return result\n",
        "\n",
        "\n",
        "def unit_hamming_distance_array(size_n):\n",
        "    if size_n == 1:\n",
        "        return torch.tensor([0, 1], dtype=torch.long)\n",
        "    a = unit_hamming_distance_array(size_n - 1)\n",
        "    return torch.concat([a, torch.flip(a, dims=[0]) + 2 ** (size_n - 1)], 0)\n",
        "\n",
        "\n",
        "def power_method(A, itr_num=20, rng=None):\n",
        "    x = torch.randn(A.shape[0], A.shape[1], A.shape[3], device=A.device, dtype=A.dtype, generator=rng)\n",
        "    x = x / torch.linalg.norm(x, dim=2).unsqueeze(-1)\n",
        "    for _ in range(itr_num):\n",
        "        y = torch.einsum('bhnm,bhm->bhn', A, x)\n",
        "        x = y / torch.linalg.norm(y, dim=2).unsqueeze(-1)\n",
        "    return torch.linalg.norm(y, dim=2)\n",
        "\n",
        "\n",
        "class Angular_LSH(torch.nn.Module):\n",
        "\n",
        "    def __init__(self, num_projs, dim, device, dtype, rng=None):\n",
        "        super().__init__()\n",
        "        self.num_projs = num_projs\n",
        "        self.proj_dir = torch.randn(dim + (num_projs,), device=device, dtype=dtype, generator=rng)\n",
        "        self.perm = unit_hamming_distance_array(self.num_projs).to(device)\n",
        "\n",
        "    def hash(self, mat):\n",
        "        proj_mat = (torch.einsum('...nd,...dr -> ...nr', mat, self.proj_dir) > 0.).type(mat.dtype)\n",
        "        enc_vec = 2.** torch.arange(self.num_projs, dtype=mat.dtype, device=mat.device).reshape(1,1,-1)\n",
        "        bin_ids = torch.einsum(\"...nr,...r->...n\", proj_mat, enc_vec).long()\n",
        "        return self.perm[bin_ids]\n"
      ],
      "metadata": {
        "id": "KNSW-DBixV4E"
      },
      "id": "KNSW-DBixV4E",
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# this class contains our modification of the kdeformer implementation\n",
        "# this code is based on https://github.com/majid-daliri/kdeformer\n",
        "class Ourmodel(torch.nn.Module):\n",
        "\n",
        "    def __init__(self, sample_size=400, num_projs=7, Bucket_size=-1,scaling_factor = 1.5, **kwargs):\n",
        "        super().__init__()\n",
        "        self.sample_size = sample_size\n",
        "        self.num_projs = num_projs\n",
        "        self.Bucket_size = Bucket_size\n",
        "        self.scaling_factor = scaling_factor\n",
        "\n",
        "        if 'mask_size' in kwargs.keys():\n",
        "            self.mask_matrix = torch.ones(kwargs['mask_size'])\n",
        "        else:\n",
        "            self.mask_matrix = None\n",
        "\n",
        "    def calc_A_var(self, key, query, Q_sort_idx, value, batch_size, head_size, nr_samples):\n",
        "        Gram_V = torch.einsum('bhnt,bhnd->bhtd', value, value)\n",
        "        V_norm = power_method(Gram_V).unsqueeze(2)\n",
        "\n",
        "        P = torch.linalg.norm(value, dim=3) / V_norm\n",
        "        P += torch.ones_like(P) / key.shape[2]\n",
        "\n",
        "        P = torch.nn.functional.normalize(P, p=1, dim=2)\n",
        "        curr_samples = int(nr_samples*self.sample_size)\n",
        "        Pflat = P.view(-1, P.shape[2])\n",
        "        index = Pflat.multinomial(num_samples=curr_samples, replacement=True)\n",
        "\n",
        "        num_blocks = key.shape[2] // self.Bucket_size\n",
        "        bucket_size_query = query.shape[2] // num_blocks\n",
        "\n",
        "        sampled_set = index.view(batch_size, head_size, -1)\n",
        "\n",
        "        Offset = torch.zeros_like(sampled_set)\n",
        "        n = key.shape[2]\n",
        "        Offset += n * torch.arange(Offset.shape[1], device=query.device).unsqueeze(0).unsqueeze(-1)\n",
        "        Offset += n * Offset.shape[1] * torch.arange(Offset.shape[0], device=query.device).unsqueeze(-1).unsqueeze(-1)\n",
        "        sampled_set = (sampled_set + Offset).view(-1)\n",
        "\n",
        "        block_id = torch.div(index, self.Bucket_size, rounding_mode='floor')  # bh * s\n",
        "        bucket_member = Q_sort_idx.view(-1, bucket_size_query)  # b h num_block * q_block\n",
        "\n",
        "        Offset = torch.zeros_like(block_id)\n",
        "        Offset += num_blocks * torch.arange(Offset.shape[0], device=query.device).unsqueeze(-1)\n",
        "        block_sample = (block_id + Offset).view(-1)\n",
        "        query_sample_collision = bucket_member[block_sample, :]\n",
        "\n",
        "        Offset = torch.zeros_like(query_sample_collision)\n",
        "\n",
        "        Offset += query.shape[2] * torch.arange(Offset.shape[0], device=query.device).unsqueeze(-1)\n",
        "        query_sample_collision_flat = (query_sample_collision + Offset).view(-1)\n",
        "        if self.mask_matrix is None:\n",
        "            mask_matrix = torch.ones(batch_size, head_size, curr_samples, query.shape[2]).view(-1).to(query.device)\n",
        "        else:\n",
        "            mask_matrix = self.mask_matrix.to(query.device)\n",
        "\n",
        "        mask_matrix[query_sample_collision_flat] = 0\n",
        "        mask_matrix = mask_matrix.view(batch_size, head_size, curr_samples, query.shape[2])\n",
        "        mask_matrix = torch.transpose(mask_matrix, 2, 3)\n",
        "\n",
        "        Vpi = value.view(-1, value.shape[3])\n",
        "        Vpi = Vpi[sampled_set, :].view(batch_size, head_size, curr_samples, value.shape[3])\n",
        "\n",
        "        Kpi = key.view(-1, key.shape[3])\n",
        "        Kpi = Kpi[sampled_set, :].view(batch_size, head_size, curr_samples, key.shape[3])\n",
        "\n",
        "        Ppi = P.view(-1)\n",
        "        Ppi = Ppi[sampled_set].view(batch_size, head_size, curr_samples)\n",
        "        sig = 1.0 / (Ppi * curr_samples)\n",
        "\n",
        "        Api_res = torch.exp(torch.einsum('bhnd,bhsd->bhns', query, Kpi)) * mask_matrix\n",
        "        att_res = torch.einsum('bhns,bhsp->bhnp', Api_res, sig.unsqueeze(-1) * Vpi)\n",
        "\n",
        "\n",
        "        Api_var = torch.pow(Api_res,2)\n",
        "        Vpi_var = torch.pow(Vpi, 2)\n",
        "        att_var = torch.einsum('bhns,bhsp->bhnp', Api_var, sig.unsqueeze(-1) * Vpi_var)\n",
        "\n",
        "        return att_res,att_var\n",
        "\n",
        "    def calc_A_res(self, key, query, Q_sort_idx, value, batch_size, head_size, nr_samples):\n",
        "        Gram_V = torch.einsum('bhnt,bhnd->bhtd', value, value)\n",
        "        V_norm = power_method(Gram_V).unsqueeze(2)\n",
        "\n",
        "        P = torch.linalg.norm(value, dim=3) / V_norm\n",
        "        P += torch.ones_like(P) / key.shape[2]\n",
        "\n",
        "        P = torch.nn.functional.normalize(P, p=1, dim=2)\n",
        "        curr_samples = int(nr_samples*self.sample_size)\n",
        "        Pflat = P.view(-1, P.shape[2])\n",
        "        index = Pflat.multinomial(num_samples=curr_samples, replacement=True)\n",
        "\n",
        "        num_blocks = key.shape[2] // self.Bucket_size\n",
        "        bucket_size_query = query.shape[2] // num_blocks\n",
        "\n",
        "        sampled_set = index.view(batch_size, head_size, -1)\n",
        "\n",
        "        Offset = torch.zeros_like(sampled_set)\n",
        "        n = key.shape[2]\n",
        "        Offset += n * torch.arange(Offset.shape[1], device=query.device).unsqueeze(0).unsqueeze(-1)\n",
        "        Offset += n * Offset.shape[1] * torch.arange(Offset.shape[0], device=query.device).unsqueeze(-1).unsqueeze(-1)\n",
        "        sampled_set = (sampled_set + Offset).view(-1)\n",
        "\n",
        "        block_id = torch.div(index, self.Bucket_size, rounding_mode='floor')\n",
        "        bucket_member = Q_sort_idx.view(-1, bucket_size_query)\n",
        "\n",
        "        Offset = torch.zeros_like(block_id)\n",
        "        Offset += num_blocks * torch.arange(Offset.shape[0], device=query.device).unsqueeze(-1)\n",
        "        block_sample = (block_id + Offset).view(-1)\n",
        "        query_sample_collision = bucket_member[block_sample, :]\n",
        "\n",
        "        Offset = torch.zeros_like(query_sample_collision)\n",
        "\n",
        "        Offset += query.shape[2] * torch.arange(Offset.shape[0], device=query.device).unsqueeze(-1)\n",
        "        query_sample_collision_flat = (query_sample_collision + Offset).view(-1)\n",
        "        if self.mask_matrix is None:\n",
        "            mask_matrix = torch.ones(batch_size, head_size, curr_samples, query.shape[2]).view(-1).to(query.device)\n",
        "        else:\n",
        "            mask_matrix = self.mask_matrix.to(query.device)\n",
        "\n",
        "        mask_matrix[query_sample_collision_flat] = 0\n",
        "        mask_matrix = mask_matrix.view(batch_size, head_size, curr_samples, query.shape[2])\n",
        "        mask_matrix = torch.transpose(mask_matrix, 2, 3)\n",
        "\n",
        "        Vpi = value.view(-1, value.shape[3])\n",
        "        Vpi = Vpi[sampled_set, :].view(batch_size, head_size, curr_samples, value.shape[3])\n",
        "\n",
        "        Kpi = key.view(-1, key.shape[3])\n",
        "        Kpi = Kpi[sampled_set, :].view(batch_size, head_size, curr_samples, key.shape[3])\n",
        "\n",
        "        Ppi = P.view(-1)\n",
        "        Ppi = Ppi[sampled_set].view(batch_size, head_size, curr_samples)\n",
        "        sig = 1.0 / (Ppi * curr_samples)\n",
        "\n",
        "\n",
        "        Api = torch.exp(torch.einsum('bhnd,bhsd->bhns', query, Kpi)) * mask_matrix\n",
        "        att_res = torch.einsum('bhns,bhsp->bhnp', Api, sig.unsqueeze(-1) * Vpi)\n",
        "\n",
        "        return att_res\n",
        "\n",
        "    def select(self, matrix, indices):\n",
        "        Offset = torch.zeros_like(indices)\n",
        "        n = matrix.shape[2]\n",
        "        Offset += n * torch.arange(Offset.shape[1], device=matrix.device).unsqueeze(0).unsqueeze(-1)\n",
        "        Offset += n * Offset.shape[1] * torch.arange(Offset.shape[0], device=matrix.device).unsqueeze(-1).unsqueeze(-1)\n",
        "        indices_flat = (indices + Offset).view(-1)\n",
        "\n",
        "        return torch.index_select(matrix.view(-1, matrix.shape[3]), 0, indices_flat).view(matrix.shape)\n",
        "\n",
        "    def forward(self, query, key, value):\n",
        "        query = rearrange(query, 'b t h e -> b h t e').contiguous()\n",
        "        key = rearrange(key, 'b s h e -> b h s e').contiguous()\n",
        "        value = rearrange(value, 'b s h d -> b h s d').contiguous()\n",
        "\n",
        "        proj_shape = (query.shape[0], query.shape[1], query.shape[3])\n",
        "\n",
        "        lsh = Angular_LSH(self.num_projs, proj_shape, device=query.device, dtype=query.dtype)\n",
        "        _, K_sort_idx = torch.sort(lsh.hash(key), dim=2)\n",
        "        _, Q_sort_idx = torch.sort(lsh.hash(query), dim=2)\n",
        "\n",
        "        value_aug = torch.cat((value, torch.ones(value.shape[0], value.shape[1], value.shape[2], 1).to(value.device)), dim=3)\n",
        "        att_sparse = CosineHammingLSH(Bucket_size=self.Bucket_size)(query=query, key=key, weight=value_aug, K_sort_idx=K_sort_idx, Q_sort_idx=Q_sort_idx)\n",
        "\n",
        "        batch_size, head_size = query.shape[0], query.shape[1]\n",
        "        value_sorted = self.select(value_aug, K_sort_idx)\n",
        "        key_sorted = self.select(key, K_sort_idx)\n",
        "\n",
        "\n",
        "        # compute empirical variance and then take more samples for the queries with high empirical variance, this is the modification on top of kdeformer\n",
        "        att_res,att_var = self.calc_A_var(key=key_sorted, query=query, Q_sort_idx=Q_sort_idx, value=value_sorted, batch_size=batch_size, head_size=head_size,nr_samples=self.scaling_factor/1.5)\n",
        "        att_var = att_var[:,:,:,0]\n",
        "        att_var_mean = att_var.mean(dim=2, keepdim=True)\n",
        "        indicator = (att_var > att_var_mean).int()\n",
        "        att_res_large =self.calc_A_res(key=key_sorted, query=query, Q_sort_idx=Q_sort_idx, value=value_sorted, batch_size=batch_size, head_size=head_size, nr_samples = self.scaling_factor)\n",
        "        att_res_small =att_res\n",
        "        indicator_for_attn = indicator.unsqueeze(-1).expand(-1, -1, -1, att_res_large.shape[3])\n",
        "\n",
        "        att_final = att_sparse + att_res_large * indicator_for_attn + att_res_small * (1 - indicator_for_attn)\n",
        "        D_tilde = att_final[:, :, :, value_aug.shape[3] - 1]\n",
        "        est = att_final[:, :, :, :value_aug.shape[3] - 1] / D_tilde.unsqueeze(-1)\n",
        "        return est\n"
      ],
      "metadata": {
        "id": "H9tvZTd1izrX"
      },
      "id": "H9tvZTd1izrX",
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "Obtain runtimes of the above implementation for attention approximation for various sequence lengths"
      ],
      "metadata": {
        "id": "MwrscFJDyKSf"
      },
      "id": "MwrscFJDyKSf"
    },
    {
      "cell_type": "code",
      "source": [
        "import time\n",
        "import torch\n",
        "import sys\n",
        "import os\n",
        "from functools import partial\n",
        "\n",
        "\n",
        "torch.manual_seed(0)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "hUL5LcFRp1VB",
        "outputId": "8fc91132-2b2e-43de-de81-cc03cad9338b"
      },
      "id": "hUL5LcFRp1VB",
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "<torch._C.Generator at 0x7a491ea39810>"
            ]
          },
          "metadata": {},
          "execution_count": 252
        }
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "id": "HK-osNESbwe9",
      "metadata": {
        "id": "HK-osNESbwe9"
      },
      "outputs": [],
      "source": [
        "# this cell contains the functions for benchmarking the runtime of our (based on kdeformer) and exact implementation. the code is taken from the kdeformer repository.\n",
        "def etime():\n",
        "    \"\"\"Measures user and system time this process has used.\n",
        "\n",
        "    Returns the sum of user and system time.\"\"\"\n",
        "    user, sys, chuser, chsys, real = os.times()\n",
        "    return user+sys\n",
        "def get_glove(seq_len=8192, seed=1):\n",
        "\n",
        "    data = open(\"glove.twitter.27B.100d.txt\", \"r\").read().split(\"\\n\")\n",
        "    batch_size, head_size = 8, 1\n",
        "    max_seq_len = 16384\n",
        "    data_all = []\n",
        "    for i in range(batch_size * head_size * max_seq_len * 3):\n",
        "        data_i = [float(_x) for _x in data[i].split()[1:]]\n",
        "        if len(data_i) != 100:\n",
        "            data_i = [float(_x) for _x in data[i].split()]\n",
        "        data_all.append(data_i)\n",
        "    data_all = torch.tensor(data_all).double()\n",
        "\n",
        "    query = data_all[:batch_size*head_size*max_seq_len]\n",
        "    query = query.reshape(batch_size, max_seq_len, head_size, -1)\n",
        "    value = data_all[2*batch_size*head_size*max_seq_len:]\n",
        "    value = value.reshape(batch_size, max_seq_len, head_size, -1)\n",
        "    del data, data_all\n",
        "\n",
        "    query = query[:,:seq_len,:,:]\n",
        "    value = value[:,:seq_len,:,:1]\n",
        "\n",
        "    normalizer = 10**0.25\n",
        "    query /= normalizer\n",
        "    key = query.clone()\n",
        "\n",
        "    return query, key, value\n",
        "\n",
        "\n",
        "class ExactAttention(torch.nn.Module):\n",
        "\n",
        "    def __init__(self):\n",
        "        super().__init__()\n",
        "\n",
        "    def forward(self, query, key, value):\n",
        "        print(key.shape)\n",
        "        qk = torch.einsum(\"bhnd,bhmd->bhnm\", query.transpose(1,2), key.transpose(1,2))\n",
        "        attn = torch.softmax(qk, dim=-1)\n",
        "        return torch.einsum('bhnm,bhmd->bhnd', attn, value.transpose(1,2))\n",
        "\n",
        "\n",
        "@torch.no_grad()\n",
        "def run(seq_len=8192, device='cpu', sample_size=256,scaling_factor = 1.5):\n",
        "\n",
        "    query, key, value = get_glove(seq_len)\n",
        "    query, key, value = map(lambda _x: _x.to(device).double(), (query, key, value))\n",
        "\n",
        "\n",
        "    tic0 = etime()\n",
        "\n",
        "\n",
        "    batch_size = query.shape[0]\n",
        "    head_size = query.shape[2]\n",
        "\n",
        "    exact_attn = ExactAttention()\n",
        "    inputs = (query, key, value)\n",
        "\n",
        "    # 1. Exact attention\n",
        "    method = 'Exact'\n",
        "    tic = etime()\n",
        "    out_exact = exact_attn(query, key, value)\n",
        "    tim_exact = etime() - tic\n",
        "\n",
        "    print(f\"[TIM] computing exact attention           | {tim_exact:<12.8f} sec  | \")\n",
        "\n",
        "\n",
        "    out_exact = out_exact.to('cpu')\n",
        "    err_normalizer =  torch.linalg.norm(value, ord='fro', dim=(1, 3))\n",
        "\n",
        "    def compute_rel_err(A):\n",
        "        return torch.mean(torch.linalg.norm(A - out_exact, ord='fro', dim=(2,3)) / err_normalizer)\n",
        "\n",
        "    # 2. Ourmodel\n",
        "    method = 'Ourmodel'\n",
        "    our_attn = Ourmodel(num_projs=7, Bucket_size=sample_size//2 , sample_size=sample_size,scaling_factor=scaling_factor).to(device)\n",
        "    our_call = partial(our_attn, query=query, key=key, value=value)\n",
        "\n",
        "    tic = etime()\n",
        "    out_our = our_call()\n",
        "    tim_our = etime() - tic\n",
        "\n",
        "    out_our = out_our.to('cpu')\n",
        "    err_our = compute_rel_err(out_our).item()\n",
        "    del out_our\n",
        "    print(f\"{method:<16}      | samples {int(sample_size)} | rel error {err_our:.10f} | {tim_our:<12.8f} sec  | \")\n",
        "    print()\n",
        "    return tim_exact,tim_our,err_our\n",
        "\n",
        "\n",
        "\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "id": "snGp17a1emD2",
      "metadata": {
        "id": "snGp17a1emD2",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "9f5b2c2a-2619-4857-a3b4-88303e6a61ee"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[TIM] computing exact attention           | 0.55000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.0990508761 | 0.24000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 0.58000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.1013593102 | 0.22000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 0.60000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.0970531369 | 0.20000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 0.66000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.0955961739 | 0.24000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 0.71000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.0993005969 | 0.31000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 0.57000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.0985954552 | 0.20000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 0.58000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.1038777868 | 0.24000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 0.64000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.1074963057 | 0.24000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 0.72000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.0953673990 | 0.30000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 0.59000000   sec  | \n",
            "Ourmodel              | samples 200 | rel error 0.1032565463 | 0.22000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 1.25000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.1034144077 | 0.42000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 1.31000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.1104094248 | 0.59000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 1.45000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.1013650408 | 0.57000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 1.21000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.1007482866 | 0.42000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 1.18000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.0958857732 | 0.39000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 1.38000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.1048835532 | 0.57000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 1.99000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.1049981141 | 0.65000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.71000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.1031810086 | 0.53000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 1.44000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.1004732546 | 0.63000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 1.34000000   sec  | \n",
            "Ourmodel              | samples 300 | rel error 0.1009075245 | 0.64000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.12000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.1037055295 | 0.92000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.09000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.1055861228 | 1.00000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.31000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.1037605857 | 1.24000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.37000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.1018880174 | 1.24000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.16000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.0995303802 | 0.94000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.09000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.1048428542 | 0.98000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.39000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.1007084871 | 1.36000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.52000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.0979721287 | 1.40000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.11000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.0974000985 | 0.93000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 2.33000000   sec  | \n",
            "Ourmodel              | samples 400 | rel error 0.1004098447 | 1.19000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.55000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.0981358272 | 2.15000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.21000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.1004299144 | 1.66000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.27000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.0997118035 | 1.57000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.69000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.1008808555 | 2.07000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.36000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.1003338035 | 1.61000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.22000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.0962955950 | 1.53000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.43000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.0979430109 | 1.91000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.85000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.0973444493 | 1.95000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.24000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.1017114720 | 1.68000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 3.21000000   sec  | \n",
            "Ourmodel              | samples 500 | rel error 0.1111974964 | 1.66000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 8.33000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.1034001086 | 4.91000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 9.76000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.1015280952 | 2.38000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 4.69000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.1000588363 | 2.74000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 4.95000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.0983123919 | 2.98000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 4.62000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.1035613586 | 2.76000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 4.98000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.1078706641 | 2.75000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 5.27000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.1031066741 | 2.72000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 4.63000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.1100542104 | 2.42000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 5.39000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.1030127966 | 2.63000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 5.34000000   sec  | \n",
            "Ourmodel              | samples 600 | rel error 0.0969215581 | 2.92000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 6.44000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.1105948148 | 3.23000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 6.87000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.0961420376 | 3.77000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 6.77000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.1044175802 | 3.12000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 6.24000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.1013963309 | 3.13000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 6.92000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.0974536521 | 3.85000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 6.36000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.1032108719 | 3.11000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 6.37000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.1010375192 | 3.91000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 7.02000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.1034687302 | 3.85000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 6.27000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.1009420681 | 3.01000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 7.14000000   sec  | \n",
            "Ourmodel              | samples 700 | rel error 0.1007188957 | 3.93000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 9.06000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.0995020039 | 3.89000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 8.25000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.1037996243 | 3.93000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 9.61000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.1046866888 | 4.90000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 8.34000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.1023171194 | 3.89000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 9.12000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.1058644118 | 4.70000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 8.26000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.1127150200 | 3.95000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 8.29000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.1039098979 | 4.14000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 9.21000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.1038372849 | 4.57000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 8.36000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.1034559096 | 3.90000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 9.57000000   sec  | \n",
            "Ourmodel              | samples 800 | rel error 0.1073531347 | 4.90000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 10.79000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.1035630238 | 4.87000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 11.33000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.1044330756 | 5.70000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 10.79000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.0978799638 | 4.91000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 11.38000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.1073712694 | 5.47000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 11.55000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.0981304797 | 4.86000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 10.68000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.1019704606 | 5.86000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 12.00000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.1023400519 | 4.83000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 10.72000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.0947959675 | 5.53000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 12.14000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.1008603160 | 5.40000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 10.73000000  sec  | \n",
            "Ourmodel              | samples 900 | rel error 0.1019134164 | 4.87000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 13.77000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.1045379119 | 5.85000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 24.91000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.0958396009 | 12.31000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 21.05000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.1054687312 | 5.80000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 15.49000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.1000492673 | 7.21000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 13.52000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.0956974607 | 5.86000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 14.70000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.1022988820 | 6.63000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 13.25000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.0995770967 | 5.75000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 15.15000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.1028376102 | 5.89000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 13.32000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.0992201416 | 6.73000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 14.42000000  sec  | \n",
            "Ourmodel              | samples 1000 | rel error 0.1031700111 | 5.74000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 18.72000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.1017699287 | 8.23000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 16.41000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.1025845980 | 6.78000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 17.88000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.0971825183 | 8.18000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 16.36000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.1093003795 | 6.78000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 18.65000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.0972493919 | 6.98000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 16.40000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.0969394159 | 8.28000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 16.54000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.0967929468 | 6.67000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 17.85000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.1005029701 | 8.07000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 16.30000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.1035306494 | 6.78000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 18.24000000  sec  | \n",
            "Ourmodel              | samples 1100 | rel error 0.1033618722 | 7.87000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 22.09000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.1006116362 | 9.12000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 36.94000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.1022975129 | 14.59000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 32.88000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.1040958703 | 8.56000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 21.23000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.1028701010 | 7.30000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 22.16000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.0968593491 | 8.13000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 19.35000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.1026647822 | 7.85000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 20.07000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.1005183629 | 7.32000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 21.11000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.1066850892 | 8.93000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 19.47000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.1060483463 | 7.37000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 20.68000000  sec  | \n",
            "Ourmodel              | samples 1200 | rel error 0.1026154585 | 7.33000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 26.17000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1078544605 | 10.24000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 22.69000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1035621505 | 8.89000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 24.10000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1056018776 | 8.66000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 24.26000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1010145099 | 9.96000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 22.77000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1057884906 | 8.88000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 24.71000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1025029483 | 8.72000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 23.98000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1042842342 | 10.54000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 22.68000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1002704086 | 10.29000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 22.93000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1056907284 | 8.64000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 25.79000000  sec  | \n",
            "Ourmodel              | samples 1300 | rel error 0.1049093124 | 8.74000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 53.88000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1041161042 | 10.21000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 28.68000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1055593743 | 8.82000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 28.93000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1017223282 | 8.91000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 28.12000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1045919089 | 10.83000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 26.78000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1080715876 | 10.87000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 27.00000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1023025077 | 9.33000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 29.13000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1024633001 | 9.03000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 28.30000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1027869330 | 9.41000000   sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 27.48000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1038067712 | 10.79000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 51.36000000  sec  | \n",
            "Ourmodel              | samples 1400 | rel error 0.1001705441 | 22.64000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 34.94000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1057850678 | 10.25000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 32.69000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1059973210 | 10.84000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 31.83000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1042295468 | 12.19000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 30.59000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1052339304 | 11.88000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 30.72000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1056394693 | 10.61000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 31.97000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1014310790 | 10.56000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 61.32000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1054398725 | 15.32000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 37.54000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1021981337 | 10.22000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 32.78000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1028677085 | 12.15000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 31.18000000  sec  | \n",
            "Ourmodel              | samples 1500 | rel error 0.1044308842 | 11.76000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 35.87000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1040521221 | 12.99000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 35.37000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1020531689 | 12.13000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 35.62000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1049481356 | 11.20000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 38.99000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1066694144 | 10.99000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 40.52000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1053897786 | 11.16000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 39.18000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1038122394 | 11.10000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 36.93000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1110303901 | 12.46000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 36.97000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1061545316 | 12.61000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 36.15000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1068083783 | 12.99000000  sec  | \n",
            "\n",
            "[TIM] computing exact attention           | 35.63000000  sec  | \n",
            "Ourmodel              | samples 1600 | rel error 0.1021637693 | 13.21000000  sec  | \n",
            "\n"
          ]
        }
      ],
      "source": [
        "tim_our_arr =[]\n",
        "tim_exact_arr = []\n",
        "err_our_arr = []\n",
        "reps = 10\n",
        "for i in range(2,17):\n",
        "    scaling_factor = 1.5-(i-2)*0.05\n",
        "    tim_exact = 0\n",
        "    tim_our = 0\n",
        "    err_our = 0\n",
        "    for j in range(reps):\n",
        "        tim_exact_local,tim_our_local,err_our_local = run(seq_len=1000*i,sample_size=100*i,scaling_factor=scaling_factor)\n",
        "        tim_exact+=tim_exact_local\n",
        "        tim_our+=tim_our_local\n",
        "        err_our+=err_our_local\n",
        "\n",
        "    tim_our_arr.append(tim_our/reps)\n",
        "    tim_exact_arr.append(tim_exact/reps)\n",
        "    err_our_arr.append(err_our/reps)\n"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "samples_arr = [100*i for i in range(2,17)]\n",
        "print(tim_exact_arr)\n",
        "print(tim_our_arr)\n",
        "print(samples_arr)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "JIWKc-MHiekh",
        "outputId": "efa3988d-ebd4-4fa4-c1be-f4683d62468c"
      },
      "id": "JIWKc-MHiekh",
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[0.619999999999709, 1.5259999999994762, 2.2489999999987047, 3.4029999999998837, 5.796000000000641, 6.639999999999418, 8.807000000000334, 11.210999999999695, 15.957999999999448, 17.334999999999855, 23.597999999999228, 24.00799999999981, 32.96600000000035, 35.55600000000013, 37.12299999999923]\n",
            "[0.24100000000071303, 0.5410000000003492, 1.119999999999709, 1.7790000000000874, 2.920999999999185, 3.490999999998894, 4.277000000000044, 5.229999999999928, 6.776999999998952, 7.462000000000626, 8.650000000000365, 9.355999999999767, 11.083999999999651, 11.577999999999156, 12.083999999999651]\n",
            "[200, 300, 400, 500, 600, 700, 800, 900, 1000, 1100, 1200, 1300, 1400, 1500, 1600]\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "Plot of runtime of exact algorithm and our algorithm as a function of sequence length"
      ],
      "metadata": {
        "id": "AYsG8RBOxr7z"
      },
      "id": "AYsG8RBOxr7z"
    },
    {
      "cell_type": "code",
      "source": [
        "import matplotlib.pyplot as plt\n",
        "\n",
        "x= range(2000,17000,1000)\n",
        "y1 = tim_exact_arr\n",
        "y2 = tim_our_arr\n",
        "\n",
        "# Plot both curves\n",
        "plt.plot(x, y1, label=\"exact\", color=\"blue\")\n",
        "plt.plot(x, y2, label=\"our\", color=\"red\")\n",
        "\n",
        "# Add labels, legend, and grid\n",
        "plt.xlabel(\"seq length\")\n",
        "plt.ylabel(\"time (s)\")\n",
        "plt.legend()\n",
        "plt.grid()\n",
        "\n",
        "# Show the plot\n",
        "plt.show()"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 449
        },
        "id": "tBULuXPgV1N2",
        "outputId": "199956fe-b44f-49f9-b22c-d70156e3013c"
      },
      "id": "tBULuXPgV1N2",
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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jGrVqhkNGzY0HnnkEdd2b2//8ePHjapVqxp33nmnsWbNGmPv3r3G/Pnzjd27d7uOGT9+vBEREWHMmTPH2LRpk3HdddcZcXFxxqlTp1zHXHPNNUajRo2M1atXGz/99JNRs2ZNo2/fvq79SUlJRsWKFY3+/fsbW7duNT777DMjNDTUePfdd4u1ved68cUXjcjISOO7774z/vjjD2PGjBlGyZIljTfeeMN1jLe1f968ecZTTz1lzJo1ywCM2bNnZ9tfXO1duXKl4e/vb7zyyivGb7/9Zjz99NNGYGCgsWXLFsvaf+LECaNTp07GF198Yfz+++/GqlWrjJYtWxrNmjXLdg5Pbv/F+uBss2bNMho1amRUqlTJeP3117Pt8+Q+uFj7d+/ebZQtW9YYMWKEsX79emP37t3G119/bRw9etR1jDv9bvDpIHOuxMREAzCWLVtmGIb5lzowMNCYMWOG65jt27cbgLFq1SrDMMwfCD8/P+PIkSOuY6ZMmWKEh4cb6enphmEYxuOPP27Ur18/2/e65ZZbjK5duxZ1k/Lk5MmTRq1atYwFCxYYV199tSvI+EL7R44cabRp0+a8+x0OhxEVFWW8+uqrrm0nTpwwgoODjc8++8wwDMP47bffDMD49ddfXcd8//33hs1mM/766y/DMAzj7bffNsqUKePqE+f3rl27dmE3KV969Ohh3HXXXdm29e7d2+jfv79hGN7f/nP/ES/O9t58881Gjx49stXTqlUrY/DgwYXaxgu50C9xp19++cUAjP379xuG4V3tN4zz98Gff/5pVK5c2di6datRtWrVbEHGm/ogt/bfcsstxm233Xber3G33w0+fWvpXElJSQCULVsWgHXr1mG32+nUqZPrmDp16hAbG8uqVasAWLVqFQ0aNKBixYquY7p27UpycjLbtm1zHXP2OZzHOM9htSFDhtCjR48cNfpC+7/55huaN2/OTTfdRIUKFWjSpAnvv/++a/8ff/zBkSNHstUfERFBq1atsvVB6dKlad68ueuYTp064efnx5o1a1zHtG3blqCgINcxXbt2ZceOHfz7779F3czzuvLKK1m0aBE7d+4EYNOmTaxYsYJu3boB3t/+cxVne93578XZkpKSsNlslC5dGvCN9jscDm6//XZGjBhB/fr1c+z35j5wOBzMnTuXyy67jK5du1KhQgVatWqV7faTu/1uUJD5j8PhYOjQobRu3ZrLL78cgCNHjhAUFOT6C+xUsWJFjhw54jrm7D8o537nvgsdk5yczKlTp4qiOXn2+eefs379esaNG5djny+0f+/evUyZMoVatWoxf/587r//fh5++GE++ugjIKsNudV/dvsqVKiQbX9AQABly5bNVz9Z4YknnuDWW2+lTp06BAYG0qRJE4YOHUr//v2z1eat7T9Xcbb3fMe4U3+cPn2akSNH0rdvX9eCgL7Q/pdffpmAgAAefvjhXPd7cx8kJiaSkpLC+PHjueaaa/jxxx+54YYb6N27N8uWLQPc73eD169+nVdDhgxh69atrFixwupSis3Bgwd55JFHWLBgASEhIVaXYwmHw0Hz5s156aWXAGjSpAlbt27lnXfeYcCAARZXV/S+/PJLPv30UxISEqhfvz4bN25k6NChVKpUySfaL+dnt9u5+eabMQyDKVOmWF1OsVm3bh1vvPEG69evx2azWV1OsXM4HABcf/31DBs2DIDGjRvz888/884773D11VdbWV6udEUGePDBB/nuu+9YsmQJVapUcW2PiooiIyODEydOZDv+6NGjREVFuY45d6S28/3FjgkPDyc0NLSwm5Nn69atIzExkaZNmxIQEEBAQADLli3jzTffJCAggIoVK3p1+wGio6OpV69etm1169Z1jc53tiG3+s9uX2JiYrb9Z86c4fjx4/nqJyuMGDHCdVWmQYMG3H777QwbNsx1hc7b23+u4mzv+Y5xh/5whpj9+/ezYMEC19UY8P72//TTTyQmJhIbG+v6d3H//v08+uijVKtWDfDuPihXrhwBAQEX/XfRnX43+HSQMQyDBx98kNmzZ7N48WLi4uKy7W/WrBmBgYEsWrTItW3Hjh0cOHCA+Ph4AOLj49myZUu2H2rnX3znD0J8fHy2cziPcZ7DKh07dmTLli1s3LjR9dG8eXP69+/veu3N7Qdo3bp1jkfud+7cSdWqVQGIi4sjKioqW/3JycmsWbMmWx+cOHGCdevWuY5ZvHgxDoeDVq1auY5Zvnw5drvddcyCBQuoXbs2ZcqUKbL2XUxaWhp+ftn/GfD393f9r8zb23+u4myvu/69cIaYXbt2sXDhQiIjI7Pt9/b233777WzevDnbv4uVKlVixIgRzJ8/H/DuPggKCqJFixYX/HfR7X435mtosJe5//77jYiICGPp0qXG4cOHXR9paWmuY+677z4jNjbWWLx4sbF27VojPj7eiI+Pd+13PmLWpUsXY+PGjcYPP/xglC9fPtdHzEaMGGFs377dmDx5sts8fnyus59aMgzvb/8vv/xiBAQEGC+++KKxa9cu49NPPzXCwsKMTz75xHXM+PHjjdKlSxtff/21sXnzZuP666/P9XHcJk2aGGvWrDFWrFhh1KpVK9ujmCdOnDAqVqxo3H777cbWrVuNzz//3AgLC7P88esBAwYYlStXdj1+PWvWLKNcuXLG448/7jrG29p/8uRJY8OGDcaGDRsMwJg4caKxYcMG11M5xdXelStXGgEBAcaECROM7du3G88++2yxPHp7ofZnZGQY1113nVGlShVj48aN2f5dPPvpG09u/8X6IDfnPrVkGJ7dBxdr/6xZs4zAwEDjvffeM3bt2uV6LPqnn35yncOdfjf4dJABcv2YNm2a65hTp04ZDzzwgFGmTBkjLCzMuOGGG4zDhw9nO8++ffuMbt26GaGhoUa5cuWMRx991LDb7dmOWbJkidG4cWMjKCjIqF69erbv4U7ODTK+0P5vv/3WuPzyy43g4GCjTp06xnvvvZdtv8PhMEaPHm1UrFjRCA4ONjp27Gjs2LEj2zH//POP0bdvX6NkyZJGeHi4MXDgQOPkyZPZjtm0aZPRpk0bIzg42KhcubIxfvz4Im/bxSQnJxuPPPKIERsba4SEhBjVq1c3nnrqqWy/tLyt/UuWLMn17/2AAQMMwyje9n755ZfGZZddZgQFBRn169c35s6dW2TtdrpQ+//444/z/ru4ZMkS1zk8uf2GcfGfgXPlFmQ8uQ/y0v6pU6caNWvWNEJCQoxGjRoZc+bMyXYOd/rdYDOMs6bwFBEREfEgPj1GRkRERDybgoyIiIh4LAUZERER8VgKMiIiIuKxFGRERETEYynIiIiIiMdSkBERERGPpSAjIiIiHktBRkTcXrVq1Zg0aZLVZQAwffp0SpcubXUZIvIfBRkRkfNwpwAlIrlTkBERERGPpSAjIvkyc+ZMGjRoQGhoKJGRkXTq1InU1FTX/g8++IC6desSEhJCnTp1ePvtt7N9/S+//EKTJk0ICQmhefPmzJ49G5vNxsaNG/Ncw4kTJ7j77rspX7484eHhdOjQgU2bNrn2jxkzhsaNG/Pxxx9TrVo1IiIiuPXWWzl58qTrmJMnT9K/f39KlChBdHQ0r7/+Ou3atWPo0KEAtGvXjv379zNs2DBsNhs2my1bDfPnz6du3bqULFmSa665hsOHD+ejF0WksCjIiEieHT58mL59+3LXXXexfft2li5dSu/evXGuPfvpp5/yzDPP8OKLL7J9+3ZeeuklRo8ezUcffQRASkoK1157LfXq1WPdunWMGTOGxx57LN913HTTTSQmJvL999+zbt06mjZtSseOHTl+/LjrmD179jBnzhy+++47vvvuO5YtW8b48eNd+4cPH87KlSv55ptvWLBgAT/99BPr16937Z81axZVqlThueee4/Dhw9mCSlpaGhMmTODjjz9m+fLlHDhwoEDtEJFCkO/1skXEZ61bt84AjH379uW6v0aNGkZCQkK2bc8//7wRHx9vGIZhvPvuu0ZkZKRx6tQp1/4pU6YYgLFhw4bzft+qVasar7/+umEYhvHTTz8Z4eHhxunTp3N873fffdcwDMN49tlnjbCwMCM5Odm1f8SIEUarVq0MwzCM5ORkIzAw0JgxY4Zr/4kTJ4ywsDDjkUceyfX7Ok2bNs0AjN27d7u2TZ482ahYseJ56xeRohNgcY4SEQ/SqFEjOnbsSIMGDejatStdunThxhtvpEyZMqSmprJnzx4GDRrEPffc4/qaM2fOEBERAcD27dtp2LAhISEhrv3x8fH5qmHTpk2kpKQQGRmZbfupU6fYs2eP6321atUoVaqU6310dDSJiYkA7N27F7vdTsuWLV37IyIiqF27dp5qCAsLo0aNGrmeW0SKl4KMiOSZv78/CxYs4Oeff+bHH3/krbfe4qmnnmLNmjWEhYUB8P7779OqVascX1dYUlJSiI6OZunSpTn2nf1YdGBgYLZ9NpsNh8NRKDXkdm7jv9trIlK8NEZGRPLFZrPRunVrxo4dy4YNGwgKCmL27NlUrFiRSpUqsXfvXmrWrJntIy4uDoC6deuyefNmTp8+7Trf6tWr8/X9mzZtypEjRwgICMjxfcqVK5enc1SvXp3AwEB+/fVX17akpCR27tyZ7bigoCAyMzPzVZ+IFC8FGRHJszVr1vDSSy+xdu1aDhw4wKxZs/j777+pW7cuAGPHjmXcuHG8+eab7Ny5ky1btjBt2jQmTpwIQL9+/bDZbNxzzz389ttvzJs3jwkTJuSrhk6dOhEfH0+vXr348ccf2bdvHz///DNPPfUUa9euzdM5SpUqxYABAxgxYgRLlixh27ZtDBo0CD8/v2xPJ1WrVo3ly5fz119/cezYsXzVKSLFQ0FGRPIsPDyc5cuX0717dy677DKefvppXnvtNbp16wbA3XffzQcffMC0adNo0KABV199NdOnT3ddkSlZsiTffvstW7ZsoUmTJjz11FO8/PLL+arBZrMxb9482rZty8CBA7nsssu49dZb2b9/PxUrVszzeSZOnEh8fDzXXnstnTp1onXr1q7Hxp2ee+459u3bR40aNShfvny+6hSR4mEzdGNXRCy0b98+4uLi2LBhA40bN7asjtTUVCpXrsxrr73GoEGDLKtDRPJHg31FxCdt2LCB33//nZYtW5KUlMRzzz0HwPXXX29xZSKSHwoyIuKzJkyYwI4dOwgKCqJZs2b89NNPeR4wLCLuQbeWRERExGNpsK+IiIh4LAUZERER8VgKMiIiIuKxFGRERETEYynIiIiIiMdSkBERERGPpSAjIiIiHktBRkRERDzW/wN2EODf9T/l8wAAAABJRU5ErkJggg==\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "Plot of ratio of runtime of exact and our algorithm as a function of sequence length"
      ],
      "metadata": {
        "id": "GqEUmKZIxyl2"
      },
      "id": "GqEUmKZIxyl2"
    },
    {
      "cell_type": "code",
      "source": [
        "import matplotlib.pyplot as plt\n",
        "\n",
        "x= range(4000,17000,1000)\n",
        "y1 = np.array(tim_exact_arr[2:])\n",
        "y2 = np.array(tim_our_arr[2:])\n",
        "\n",
        "# Plot both curves\n",
        "plt.plot(x, (y1/y2), label=\"exact/our\", color=\"blue\")\n",
        "\n",
        "# Add labels, legend, and grid\n",
        "plt.xlabel(\"seq length\")\n",
        "plt.ylabel(\"ratio of runtime\")\n",
        "plt.legend()\n",
        "plt.grid()\n",
        "\n",
        "# Show the plot\n",
        "plt.show()"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 449
        },
        "id": "yvb5TU2h5u_L",
        "outputId": "1919115c-f57d-4c97-9a63-cd26b7d9d9c5"
      },
      "id": "yvb5TU2h5u_L",
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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35iF2t69ZkvLgw/CiRPuXy9jZtSqVVCwIMyfD0WKeDxsERHxIBU3PsCdNXFy4wwoT7PZzBzs2mUOvLbZ4LPPzAHH//2ve7PM3njDfE3GnlGaGSUikvupuPER11sTJy/MgPK0ggXNhf9++slcCPCvv8xuvFatzBad65k719z+AczVhzUzSkQkb1Bx40OutiZOXFzemQHlDTExZtfb2LEQFmaOwald2xyTc/581q+5dGZU//6aGSUikpeouPFBl6+JM3Fi3psB5WnBweZ2DTt2mC0waWnmbKr69c0Wr0sdP35xZtSdd5o7feeVgdUiIqLixmdlrInz1lvmppN5cQaUN1SubLZsffklFC9udsvdfjs89pg5XunymVHaM0pEJO9RcePDAgLMdWtWr87bM6A8zWYzu5x274Z+/czHPvzQbNnq1Mkco1OwoLlnlGZGiYjkPSpuxG8VKQIff2x201Wvbi7Ul5Bwcc+o6tWtjlBERG6Eihvxe82bm1PpR46E8uVh8mRz3JKIiORN6qgQAUJDYdgw8yYiInmbWm5ERETEp6i4EREREZ+i4kZERER8ioobERER8SkqbkRERMSnqLgRERERn6LiRkRERHyKihsRERHxKSpuRERExKeouBERERGfouJGREREfIqKGxEREfEpKm5ERETEp6i4EREREZ8SZHUAOc0wDABSUlI8fm273U5qaiopKSkEBwd7/Pq+RLlynXLlOuXKdcqVe5Qv13krVxnf2xnf49fid8XNqVOnAIiOjrY4EhEREXHXqVOnKFiw4DXPsRmulEA+xOFw8McffxAREYHNZvPotVNSUoiOjua3334jMjLSo9f2NcqV65Qr1ylXrlOu3KN8uc5buTIMg1OnTlG6dGkCAq49qsbvWm4CAgIoW7asV98jMjJSv/wuUq5cp1y5TrlynXLlHuXLdd7I1fVabDJoQLGIiIj4FBU3IiIi4lNU3HhQaGgow4cPJzQ01OpQcj3lynXKleuUK9cpV+5RvlyXG3LldwOKRURExLep5UZERER8ioobERER8SkqbkRERMSnqLgRERERn6Li5hrGjh2LzWbjqaeecj527tw54uLiKFq0KAUKFOBf//oXx44dy/S6xMREOnbsSHh4OFFRUQwZMoQLFy5kOmfp0qU0aNCA0NBQbrrpJqZOnZoDn8izDh8+zAMPPEDRokXJly8ftWvXZsOGDc7nDcNg2LBhlCpVinz58tGqVSv27t2b6RonT56kV69eREZGUqhQIfr168fp06cznbNt2zZuv/12wsLCiI6O5vXXX8+Rz+dJ6enpDB06lIoVK5IvXz4qV67Myy+/nGmPFH/N1/Lly+nUqROlS5fGZrMxe/bsTM/nZF5mzJhB9erVCQsLo3bt2ixYsMDjnzc7rpUru93O888/T+3atcmfPz+lS5emT58+/PHHH5muoVxdqX///thsNt5+++1MjytXF+3atYvOnTtTsGBB8ufPT+PGjUlMTHQ+n+u+Gw3J0rp164wKFSoYderUMQYNGuR8vH///kZ0dLSxZMkSY8OGDcatt95qNG3a1Pn8hQsXjFq1ahmtWrUyNm/ebCxYsMAoVqyY8cILLzjP2b9/vxEeHm48/fTTxs6dO413333XCAwMNBYuXJiTHzFbTp48aZQvX9548MEHjbVr1xr79+83Fi1aZOzbt895ztixY42CBQsas2fPNrZu3Wp07tzZqFixonH27FnnOe3atTPq1q1rrFmzxlixYoVx0003GT169HA+n5ycbJQoUcLo1auXsWPHDuOrr74y8uXLZ0yePDlHP292jR492ihatKgxf/5848CBA8aMGTOMAgUKGBMmTHCe46/5WrBggfHiiy8aM2fONABj1qxZmZ7Pqbz89NNPRmBgoPH6668bO3fuNF566SUjODjY2L59u9dz4Kpr5SopKclo1aqV8fXXXxu7d+82Vq9ebdxyyy1Gw4YNM11Ducps5syZRt26dY3SpUsbb731VqbnlCvTvn37jCJFihhDhgwxNm3aZOzbt8+YM2eOcezYMec5ue27UcVNFk6dOmVUqVLFSEhIMFq0aOEsbpKSkozg4GBjxowZznN37dplAMbq1asNwzB/SQICAoyjR486z5k0aZIRGRlppKWlGYZhGM8995xRs2bNTO95//33G23btvXyJ/Oc559/3rjtttuu+rzD4TBKlixpvPHGG87HkpKSjNDQUOOrr74yDMMwdu7caQDG+vXrned89913hs1mMw4fPmwYhmFMnDjRKFy4sDN3Ge9drVo1T38kr+rYsaPx73//O9Nj99xzj9GrVy/DMJSvDJf/w5qTebnvvvuMjh07ZoqnSZMmxmOPPebRz+gp1/rCzrBu3ToDMA4dOmQYhnJ1ud9//90oU6aMsWPH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337b6jAAmDp1KoUKFbI6DBH5h4obERE35KaiSkSypuJGREREfIqKGxHJtm+++YbatWuTL18+ihYtSqtWrThz5ozz+Y8//pgaNWoQFhZG9erVmThxYqbXr1u3jvr16xMWFkajRo2YNWsWNpuNLVu2uBxDUlISDz/8MMWLFycyMpI777yTrVu3Op8fMWIE9erV4/PPP6dChQoULFiQ7t27c+rUKec5p06dolevXuTPn59SpUrx1ltv0bJlS5566ikAWrZsyaFDhxg8eDA2mw2bzZYphkWLFlGjRg0KFChAu3btOHLkiBtZFBFPUXEjItly5MgRevTowb///W927drF0qVLueeee8jYk/fLL79k2LBhjB49ml27dvHqq68ydOhQPvvsMwBOnz7N3Xffzc0338zGjRsZMWIEzz77rNtxdOvWjePHj/Pdd9+xceNGGjRowF133cXJkyed5/z666/Mnj2b+fPnM3/+fJYtW8bYsWOdzz/99NP89NNPzJ07l4SEBFasWMGmTZucz8+cOZOyZcsyatQojhw5kql4SU1N5c033+Tzzz9n+fLlJCYm3tDnEBEPuKG9xEVE/rFx40YDMA4ePJjl85UrVzbi4+MzPfbyyy8bMTExhmEYxuTJk42iRYsaZ8+edT4/adIkAzA2b9581fctX7688dZbbxmGYRgrVqwwIiMjjXPnzl3x3pMnTzYMwzCGDx9uhIeHGykpKc7nhwwZYjRp0sQwDMNISUkxgoODjRkzZjifT0pKMsLDw41BgwZl+b4ZpkyZYgDGvn37nI+9//77RokSJa4av4h4T5DFtZWI5HF169blrrvuonbt2rRt25Y2bdpw7733UrhwYc6cOcOvv/5Kv379eOSRR5yvuXDhAgULFgRg165d1KlTh7CwMOfzMTExbsWwdetWTp8+TdGiRTM9fvbsWX799Vfn/QoVKhAREeG8X6pUKY4fPw7A/v37sdvt3HLLLc7nCxYsSLVq1VyKITw8nMqVK2d5bRHJWSpuRCRbAgMDSUhIYNWqVXz//fe8++67vPjii6xdu5bw8HAAPvroI5o0aXLF6zzl9OnTlCpViqVLl17x3KVTtIODgzM9Z7PZcDgcHokhq2sb/3TNiUjO0pgbEck2m81Gs2bNGDlyJJs3byYkJIRZs2ZRokQJSpcuzf79+7npppsy3SpWrAhAjRo12LZtG+fOnXNeb82aNW69f4MGDTh69ChBQUFXvE+xYsVcukalSpUIDg5m/fr1zseSk5P55ZdfMp0XEhJCenq6W/GJSM5ScSMi2bJ27VpeffVVNmzYQGJiIjNnzuTPP/+kRo0aAIwcOZIxY8bwzjvv8Msvv7B9+3amTJnC+PHjAejZsyc2m41HHnmEnTt3smDBAt588023YmjVqhUxMTHExsby/fffc/DgQVatWsWLL77Ihg0bXLpGREQEffv2ZciQIfz444/8/PPP9OvXj4CAgEyzoipUqMDy5cs5fPgwJ06ccCtOEckZKm5EJFsiIyNZvnw5HTp0oGrVqrz00kuMGzeO9u3bA/Dwww/z8ccfM2XKFGrXrk2LFi2YOnWqs+WmQIECzJs3j+3bt1O/fn1efPFFXnvtNbdisNlsLFiwgObNm/PQQw9RtWpVunfvzqFDhyhRooTL1xk/fjwxMTHcfffdtGrVimbNmjmnsGcYNWoUBw8epHLlyhQvXtytOEUkZ9gMdQqLSC5z8OBBKlasyObNm6lXr55lcZw5c4YyZcowbtw4+vXrZ1kcIuIeDSgWEfnH5s2b2b17N7fccgvJycmMGjUKgC5dulgcmYi4Q8WNiMgl3nzzTfbs2UNISAgNGzZkxYoVLg9KFpHcQd1SIiIi4lM0oFhERER8ioobERER8SkqbkRERMSnqLgRERERn6LiRkRERHyKihsRERHxKSpuRERExKeouBERERGf8v8GzkdL3WmBrgAAAABJRU5ErkJggg==\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "print(err_our_arr[2:])\n",
        "print(y1[2:])\n",
        "print(y2[2:])\n",
        "print((y1/y2)[2:])"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "QPUhybbMbb1j",
        "outputId": "1d856867-3bd3-4a03-92d2-6ed8f685c21f"
      },
      "id": "QPUhybbMbb1j",
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[0.10158040488867445, 0.10039842276432506, 0.1027826693858755, 0.10193825008211857, 0.1047441095421798, 0.1013258024763805, 0.10086967135817002, 0.10092146706911062, 0.1025266508614215, 0.10414791210149503, 0.10355913593755313, 0.10432530131326294, 0.10530819282738405]\n",
            "[ 5.796  6.64   8.807 11.211 15.958 17.335 23.598 24.008 32.966 35.556\n",
            " 37.123]\n",
            "[ 2.921  3.491  4.277  5.23   6.777  7.462  8.65   9.356 11.084 11.578\n",
            " 12.084]\n",
            "[1.98425197 1.9020338  2.05915361 2.14359465 2.35472923 2.32310373\n",
            " 2.72809249 2.56605387 2.97419704 3.07099672 3.07207878]\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "print(min(err_our_arr[2:]))\n",
        "print(max(err_our_arr[2:]))"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "XTYlIMFFun07",
        "outputId": "e86046c5-9ac6-4dea-b613-88221627b926"
      },
      "id": "XTYlIMFFun07",
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "0.10039842276432506\n",
            "0.10530819282738405\n"
          ]
        }
      ]
    }
  ],
  "metadata": {
    "colab": {
      "provenance": [],
      "machine_shape": "hm"
    },
    "kernelspec": {
      "display_name": "Python 3",
      "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.9.6"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 5
}