{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import math\n",
    "import os\n",
    "import warnings\n",
    "from dataclasses import dataclass\n",
    "from typing import Optional, Tuple\n",
    "import time\n",
    "import torch\n",
    "import torch.utils.checkpoint\n",
    "from packaging import version\n",
    "from torch import nn\n",
    "from torch.nn import BCEWithLogitsLoss, CrossEntropyLoss, MSELoss\n",
    "from torch.nn import functional as F\n",
    "\n",
    "from transformers.activations import ACT2FN\n",
    "from transformers.file_utils import (\n",
    "    ModelOutput,\n",
    "    add_code_sample_docstrings,\n",
    "    add_start_docstrings,\n",
    "    add_start_docstrings_to_model_forward,\n",
    "    replace_return_docstrings,\n",
    ")\n",
    "from transformers.modeling_outputs import (\n",
    "    BaseModelOutputWithPastAndCrossAttentions,\n",
    "    BaseModelOutputWithPoolingAndCrossAttentions,\n",
    ")\n",
    "from transformers.modeling_utils import (\n",
    "    PreTrainedModel,\n",
    "    apply_chunking_to_forward,\n",
    "    find_pruneable_heads_and_indices,\n",
    "    prune_linear_layer,\n",
    ")\n",
    "from transformers.utils import logging\n",
    "from transformers.models.bert.configuration_bert import BertConfig\n",
    "\n",
    "logger = logging.get_logger(__name__)\n",
    "\n",
    "class BertEmbeddings(nn.Module):\n",
    "    \"\"\"Construct the embeddings from word, position and token_type embeddings.\"\"\"\n",
    "\n",
    "    def __init__(self, config):\n",
    "        super().__init__()\n",
    "        self.word_embeddings = nn.Embedding(config.vocab_size, config.hidden_size, padding_idx=config.pad_token_id)\n",
    "        self.position_embeddings = nn.Embedding(config.max_position_embeddings, config.hidden_size)\n",
    "        self.token_type_embeddings = nn.Embedding(config.type_vocab_size, config.hidden_size)\n",
    "\n",
    "        # self.LayerNorm is not snake-cased to stick with TensorFlow model variable name and be able to load\n",
    "        # any TensorFlow checkpoint file\n",
    "        self.LayerNorm = nn.LayerNorm(config.hidden_size, eps=config.layer_norm_eps)\n",
    "        self.dropout = nn.Dropout(config.hidden_dropout_prob)\n",
    "        # position_ids (1, len position emb) is contiguous in memory and exported when serialized\n",
    "        self.position_embedding_type = getattr(config, \"position_embedding_type\", \"absolute\")\n",
    "        self.register_buffer(\"position_ids\", torch.arange(config.max_position_embeddings).expand((1, -1)))\n",
    "        if version.parse(torch.__version__) > version.parse(\"1.6.0\"):\n",
    "            self.register_buffer(\n",
    "                \"token_type_ids\",\n",
    "                torch.zeros(self.position_ids.size(), dtype=torch.long),\n",
    "                persistent=False,\n",
    "            )\n",
    "\n",
    "    def forward(\n",
    "        self, input_ids=None, token_type_ids=None, position_ids=None, inputs_embeds=None, past_key_values_length=0\n",
    "    ):\n",
    "        if input_ids is not None:\n",
    "            input_shape = input_ids.size()\n",
    "        else:\n",
    "            input_shape = inputs_embeds.size()[:-1]\n",
    "\n",
    "        seq_length = input_shape[1]\n",
    "\n",
    "        if position_ids is None:\n",
    "            position_ids = self.position_ids[:, past_key_values_length : seq_length + past_key_values_length]\n",
    "\n",
    "        # Setting the token_type_ids to the registered buffer in constructor where it is all zeros, which usually occurs\n",
    "        # when its auto-generated, registered buffer helps users when tracing the model without passing token_type_ids, solves\n",
    "        # issue #5664\n",
    "        if token_type_ids is None:\n",
    "            if hasattr(self, \"token_type_ids\"):\n",
    "                buffered_token_type_ids = self.token_type_ids[:, :seq_length]\n",
    "                buffered_token_type_ids_expanded = buffered_token_type_ids.expand(input_shape[0], seq_length)\n",
    "                token_type_ids = buffered_token_type_ids_expanded\n",
    "            else:\n",
    "                token_type_ids = torch.zeros(input_shape, dtype=torch.long, device=self.position_ids.device)\n",
    "\n",
    "        if inputs_embeds is None:\n",
    "            inputs_embeds = self.word_embeddings(input_ids)\n",
    "        token_type_embeddings = self.token_type_embeddings(token_type_ids)\n",
    "\n",
    "        embeddings = inputs_embeds + token_type_embeddings\n",
    "        if self.position_embedding_type == \"absolute\":\n",
    "            position_embeddings = self.position_embeddings(position_ids)\n",
    "            embeddings += position_embeddings\n",
    "        embeddings = self.LayerNorm(embeddings)\n",
    "        embeddings = self.dropout(embeddings)\n",
    "        return embeddings\n",
    "\n",
    "def lowdim_apply(linear : nn.Linear, input, mode:bool, factor:int, interpolate='avg'):\n",
    "    #print('low_dim_apply', mode, factor, interpolate)\n",
    "    if mode:\n",
    "        if interpolate == 'avg':\n",
    "            s = linear.weight.shape\n",
    "            w = F.adaptive_avg_pool2d(linear.weight.view(1, 1, s[0], s[1]), output_size = (s[0] // factor, s[1] // factor)).view(s[0]//factor, s[1]//factor)\n",
    "            s = linear.bias.shape\n",
    "            b = F.adaptive_avg_pool1d(linear.bias.view(1,1,s[0]), (s[0] // factor,)).view(s[0]//factor)\n",
    "        elif interpolate == 'max':\n",
    "            s = linear.weight.shape\n",
    "            w = F.adaptive_max_pool2d(linear.weight.view(1, 1, s[0], s[1]), output_size = (s[0] // factor, s[1] // factor)).view(s[0]//factor, s[1]//factor)\n",
    "            s = linear.bias.shape\n",
    "            b = F.adaptive_max_pool1d(linear.bias.view(1,1,s[0]), (s[0] // factor,)).view(s[0]//factor)\n",
    "        elif interpolate == 'output_only':\n",
    "            s = linear.weight.shape\n",
    "            w = F.adaptive_max_pool2d(linear.weight.view(1, 1, s[0], s[1]), output_size = (s[0], s[1] // factor)).view(s[0], s[1]//factor)\n",
    "            s = linear.bias.shape\n",
    "            b = F.adaptive_max_pool1d(linear.bias.view(1,1,s[0]), (s[0] // factor,)).view(s[0]//factor)\n",
    "        else:\n",
    "            s = linear.weight.shape\n",
    "            w = F.interpolate(\n",
    "                linear.weight.view(1, 1, s[0], s[1]), \n",
    "                size = (s[0] // factor, s[1] // factor),\n",
    "                mode = interpolate\n",
    "            ).view(s[0]//factor, s[1]//factor)\n",
    "            s = linear.bias.shape\n",
    "            b = F.interpolate(\n",
    "                linear.bias.view(1,1,s[0]), \n",
    "                size = (s[0] // factor,),\n",
    "                mode = interpolate\n",
    "            ).view(s[0]//factor)\n",
    "        return F.linear(input, w, b)\n",
    "    else:\n",
    "        return linear(input)\n",
    "\n",
    "class BertSelfAttention(nn.Module):\n",
    "    def __init__(self, config, position_embedding_type=None):\n",
    "        super().__init__()\n",
    "        if config.hidden_size % config.num_attention_heads != 0 and not hasattr(config, \"embedding_size\"):\n",
    "            raise ValueError(\n",
    "                f\"The hidden size ({config.hidden_size}) is not a multiple of the number of attention \"\n",
    "                f\"heads ({config.num_attention_heads})\"\n",
    "            )\n",
    "        \n",
    "        self.hidden_size = config.hidden_size\n",
    "        self.lowdim = False\n",
    "        self.lowdim_linear = False\n",
    "        self.lowdim_qk_linear = True\n",
    "        self.lowdim_qk = False\n",
    "        self.lowdim_factor = 4\n",
    "        self.lowdim_mode = 'avg'\n",
    "        self.lowdim_qk_linear_mode = 'avg'\n",
    "\n",
    "        self.num_attention_heads = config.num_attention_heads\n",
    "        self.attention_head_size = int(config.hidden_size / config.num_attention_heads)\n",
    "        self.all_head_size = self.num_attention_heads * self.attention_head_size\n",
    "\n",
    "        self.query = nn.Linear(config.hidden_size, self.all_head_size)\n",
    "        self.key = nn.Linear(config.hidden_size, self.all_head_size)\n",
    "        self.value = nn.Linear(config.hidden_size, self.all_head_size)\n",
    "\n",
    "        self.dropout = nn.Dropout(config.attention_probs_dropout_prob)\n",
    "        self.position_embedding_type = position_embedding_type or getattr(\n",
    "            config, \"position_embedding_type\", \"absolute\"\n",
    "        )\n",
    "        if self.position_embedding_type == \"relative_key\" or self.position_embedding_type == \"relative_key_query\":\n",
    "            self.max_position_embeddings = config.max_position_embeddings\n",
    "            self.distance_embedding = nn.Embedding(2 * config.max_position_embeddings - 1, self.attention_head_size)\n",
    "\n",
    "        self.is_decoder = config.is_decoder\n",
    "        self.print = True\n",
    "    \n",
    "    def update_lowdim(self):\n",
    "        def downsize_linear(linear):\n",
    "            factor = self.lowdim_factor\n",
    "            if self.lowdim_qk_linear_mode == 'avg':\n",
    "                s = linear.weight.shape\n",
    "                w = F.adaptive_avg_pool2d(linear.weight.view(1, 1, s[0], s[1]), output_size = (s[0] // factor, s[1])).view(s[0] // factor, s[1])\n",
    "                s = linear.bias.shape\n",
    "                b = F.adaptive_avg_pool1d(linear.bias.view(1,1,s[0]), (s[0] // factor,)).view(s[0]//factor)\n",
    "            elif self.lowdim_qk_linear_mode == 'absmax':\n",
    "                import numba\n",
    "                @numba.njit\n",
    "                def resize(x, factor):\n",
    "                    # x (SIZE, N) -> (SIZE // factor, N)\n",
    "                    SIZE, N = x.shape\n",
    "                    ret = np.empty((SIZE // factor, N), dtype=x.dtype)\n",
    "                    for i in range(N):\n",
    "                        for j in range(SIZE // factor):\n",
    "                            value = 0\n",
    "                            for k in range(factor):\n",
    "                                if abs(x[j*factor+k, i]) >= abs(value):\n",
    "                                    value = x[j*factor+k, i]\n",
    "                            ret[j, i] = value\n",
    "                    return ret\n",
    "                w = torch.tensor(resize(linear.weight.data.cpu().numpy(), factor))\n",
    "                b = torch.tensor(resize(linear.bias.data.cpu().numpy().reshape(-1, 1), factor)).view(-1)\n",
    "            else:\n",
    "                raise Exception('no')\n",
    "            l = nn.Linear(w.shape[1], w.shape[0])\n",
    "            l = l.to(linear.weight.device)\n",
    "            l.weight.data[:,:] = w\n",
    "            l.bias.data[:] = b\n",
    "            #print(linear.weight.shape, 'to', l.weight.shape)\n",
    "            return l\n",
    "        self.low_query = downsize_linear(self.query)\n",
    "        self.low_key = downsize_linear(self.key)\n",
    "\n",
    "    def resize_vector(self, x, use, factor=4):\n",
    "        #print('resize_vec', use, factor)\n",
    "        if not use: return x\n",
    "        #return F.interpolate(x, (model.config.hidden_size // factor,))\n",
    "        s = x.shape\n",
    "        return F.adaptive_avg_pool1d(x.reshape(s[0]*s[1], s[2], s[3]), output_size = (s[-1] // factor, )).view(s[0],s[1],s[2],s[3]//factor)\n",
    "\n",
    "    def transpose_for_scores(self, x):\n",
    "        new_x_shape = x.size()[:-1] + (self.num_attention_heads, x.shape[-1] // self.num_attention_heads)\n",
    "        x = x.view(*new_x_shape)\n",
    "        return x.permute(0, 2, 1, 3)\n",
    "\n",
    "    def forward(\n",
    "        self,\n",
    "        hidden_states,\n",
    "        attention_mask=None,\n",
    "        head_mask=None, #removed\n",
    "        encoder_hidden_states=None, #removed\n",
    "        encoder_attention_mask=None, #removed\n",
    "        past_key_value=None, #removed\n",
    "        output_attentions=False,\n",
    "    ):\n",
    "        assert encoder_hidden_states is None and encoder_attention_mask is None and head_mask is None and past_key_value is None\n",
    "\n",
    "        if self.is_decoder: raise Exception('removed')\n",
    "\n",
    "        if self.print: \n",
    "            print('SelfAttention.forward: hidden_states, attention_mask backuped')\n",
    "            self.last_hidden_states = hidden_states.clone().detach()\n",
    "            self.last_attention_mask = attention_mask.clone().detach()\n",
    "        if self.lowdim_qk_linear and self.lowdim:\n",
    "            mixed_query_layer = self.low_query(hidden_states)\n",
    "        else:\n",
    "            mixed_query_layer = lowdim_apply(self.query, hidden_states, self.lowdim and self.lowdim_linear, self.lowdim_factor, self.lowdim_mode)\n",
    "\n",
    "        is_cross_attention = encoder_hidden_states is not None\n",
    "        if is_cross_attention and past_key_value is not None:\n",
    "            raise Exception('oops')\n",
    "        elif is_cross_attention:\n",
    "            raise Exception('oops')\n",
    "        elif past_key_value is not None:\n",
    "            raise Exception('oops')\n",
    "        else:\n",
    "            if self.print: print('SelfAttention.forward: expected')\n",
    "            if self.lowdim_qk_linear and self.lowdim:\n",
    "                key_layer = self.transpose_for_scores(self.low_key(hidden_states))\n",
    "            else:\n",
    "                key_layer = self.resize_vector(\n",
    "                    self.transpose_for_scores(\n",
    "                        lowdim_apply(self.key, hidden_states, self.lowdim and self.lowdim_linear, self.lowdim_factor, self.lowdim_mode)\n",
    "                    ),\n",
    "                    self.lowdim and self.lowdim_qk,\n",
    "                    self.lowdim_factor)\n",
    "            value_layer = self.transpose_for_scores(\n",
    "                lowdim_apply(self.value, hidden_states, self.lowdim and self.lowdim_linear, self.lowdim_factor, self.lowdim_mode))\n",
    "\n",
    "        query_layer = self.resize_vector(\n",
    "            self.transpose_for_scores(mixed_query_layer),\n",
    "            self.lowdim and self.lowdim_qk, \n",
    "            self.lowdim_factor)\n",
    "\n",
    "        # Take the dot product between \"query\" and \"key\" to get the raw attention scores.\n",
    "        #print(query_layer.shape, key_layer.transpose(-1, -2).shape)\n",
    "        attention_scores = torch.matmul(query_layer, key_layer.transpose(-1, -2))\n",
    "\n",
    "        if self.position_embedding_type == \"relative_key\" or self.position_embedding_type == \"relative_key_query\":\n",
    "            raise Exception('lowdim not applied')\n",
    "\n",
    "        if self.lowdim: lowdim_factor = self.lowdim_factor\n",
    "        else: lowdim_factor = 1\n",
    "        \n",
    "        attention_scores = attention_scores / math.sqrt(self.attention_head_size // lowdim_factor)\n",
    "        if attention_mask is not None:\n",
    "            # Apply the attention mask is (precomputed for all layers in BertModel forward() function)\n",
    "            attention_scores = attention_scores + attention_mask\n",
    "\n",
    "        # Normalize the attention scores to probabilities.\n",
    "        attention_probs = nn.functional.softmax(attention_scores, dim=-1)\n",
    "\n",
    "        # This is actually dropping out entire tokens to attend to, which might\n",
    "        # seem a bit unusual, but is taken from the original Transformer paper.\n",
    "        attention_probs = self.dropout(attention_probs)\n",
    "\n",
    "        # Mask heads if we want to\n",
    "        if head_mask is not None: raise Exception()\n",
    "        if self.print: \n",
    "            print('SelfAttention.forward: attention_probs.shape', attention_probs.shape)\n",
    "            self.last_attention_probs = attention_probs.clone().detach()\n",
    "            print('SelfAttention.forward: last_attention_probs is cloned from attention_probs')\n",
    "\n",
    "        context_layer = torch.matmul(attention_probs, value_layer)\n",
    "\n",
    "        context_layer = context_layer.permute(0, 2, 1, 3).contiguous()\n",
    "        if self.lowdim_linear: \n",
    "            new_context_layer_shape = context_layer.size()[:-2] + (self.all_head_size // lowdim_factor,)\n",
    "        else: \n",
    "            new_context_layer_shape = context_layer.size()[:-2] + (self.all_head_size,)\n",
    "        context_layer = context_layer.view(*new_context_layer_shape)\n",
    "\n",
    "        outputs = (context_layer, attention_probs) if output_attentions else (context_layer,)\n",
    "        \n",
    "        return outputs\n",
    "\n",
    "\n",
    "class BertSelfOutput(nn.Module):\n",
    "    def __init__(self, config):\n",
    "        super().__init__()\n",
    "        self.dense = nn.Linear(config.hidden_size, config.hidden_size)\n",
    "        self.LayerNorm = nn.LayerNorm(config.hidden_size, eps=config.layer_norm_eps)\n",
    "        self.dropout = nn.Dropout(config.hidden_dropout_prob)\n",
    "\n",
    "    def forward(self, hidden_states, input_tensor):\n",
    "        hidden_states = self.dense(hidden_states)\n",
    "        hidden_states = self.dropout(hidden_states)\n",
    "        hidden_states = self.LayerNorm(hidden_states + input_tensor)\n",
    "        return hidden_states\n",
    "\n",
    "\n",
    "class BertAttention(nn.Module):\n",
    "    def __init__(self, config, position_embedding_type=None):\n",
    "        super().__init__()\n",
    "        self.self = BertSelfAttention(config, position_embedding_type=position_embedding_type)\n",
    "        self.output = BertSelfOutput(config)\n",
    "        self.pruned_heads = set()\n",
    "\n",
    "    def prune_heads(self, heads):\n",
    "        print('prune_heads called')\n",
    "        raise Exception('unknown function')\n",
    "        if len(heads) == 0:\n",
    "            return\n",
    "        heads, index = find_pruneable_heads_and_indices(\n",
    "            heads, self.self.num_attention_heads, self.self.attention_head_size, self.pruned_heads\n",
    "        )\n",
    "\n",
    "        # Prune linear layers\n",
    "        self.self.query = prune_linear_layer(self.self.query, index)\n",
    "        self.self.key = prune_linear_layer(self.self.key, index)\n",
    "        self.self.value = prune_linear_layer(self.self.value, index)\n",
    "        self.output.dense = prune_linear_layer(self.output.dense, index, dim=1)\n",
    "\n",
    "        # Update hyper params and store pruned heads\n",
    "        self.self.num_attention_heads = self.self.num_attention_heads - len(heads)\n",
    "        self.self.all_head_size = self.self.attention_head_size * self.self.num_attention_heads\n",
    "        self.pruned_heads = self.pruned_heads.union(heads)\n",
    "\n",
    "    def forward(\n",
    "        self,\n",
    "        hidden_states,\n",
    "        attention_mask=None,\n",
    "        head_mask=None,\n",
    "        encoder_hidden_states=None,\n",
    "        encoder_attention_mask=None,\n",
    "        past_key_value=None,\n",
    "        output_attentions=False,\n",
    "    ):\n",
    "        self_outputs = self.self(\n",
    "            hidden_states,\n",
    "            attention_mask,\n",
    "            head_mask,\n",
    "            encoder_hidden_states,\n",
    "            encoder_attention_mask,\n",
    "            past_key_value,\n",
    "            output_attentions,\n",
    "        )\n",
    "        attention_output = self.output(self_outputs[0], hidden_states)\n",
    "        outputs = (attention_output,) + self_outputs[1:]  # add attentions if we output them\n",
    "        return outputs\n",
    "\n",
    "\n",
    "class BertIntermediate(nn.Module):\n",
    "    def __init__(self, config):\n",
    "        super().__init__()\n",
    "        self.dense = nn.Linear(config.hidden_size, config.intermediate_size)\n",
    "        if isinstance(config.hidden_act, str):\n",
    "            self.intermediate_act_fn = ACT2FN[config.hidden_act]\n",
    "        else:\n",
    "            self.intermediate_act_fn = config.hidden_act\n",
    "\n",
    "    def forward(self, hidden_states):\n",
    "        hidden_states = self.dense(hidden_states)\n",
    "        hidden_states = self.intermediate_act_fn(hidden_states)\n",
    "        return hidden_states\n",
    "\n",
    "\n",
    "class BertOutput(nn.Module):\n",
    "    def __init__(self, config):\n",
    "        super().__init__()\n",
    "        self.dense = nn.Linear(config.intermediate_size, config.hidden_size)\n",
    "        self.LayerNorm = nn.LayerNorm(config.hidden_size, eps=config.layer_norm_eps)\n",
    "        self.dropout = nn.Dropout(config.hidden_dropout_prob)\n",
    "\n",
    "    def forward(self, hidden_states, input_tensor):\n",
    "        hidden_states = self.dense(hidden_states)\n",
    "        hidden_states = self.dropout(hidden_states)\n",
    "        hidden_states = self.LayerNorm(hidden_states + input_tensor)\n",
    "        return hidden_states\n",
    "\n",
    "\n",
    "class BertLayer(nn.Module):\n",
    "    def __init__(self, config):\n",
    "        super().__init__()\n",
    "        self.chunk_size_feed_forward = config.chunk_size_feed_forward\n",
    "        self.seq_len_dim = 1\n",
    "        self.attention = BertAttention(config)\n",
    "        self.is_decoder = config.is_decoder\n",
    "        self.add_cross_attention = config.add_cross_attention\n",
    "        if self.add_cross_attention:\n",
    "            if not self.is_decoder:\n",
    "                raise ValueError(f\"{self} should be used as a decoder model if cross attention is added\")\n",
    "            self.crossattention = BertAttention(config, position_embedding_type=\"absolute\")\n",
    "        self.intermediate = BertIntermediate(config)\n",
    "        self.output = BertOutput(config)\n",
    "\n",
    "    def forward(\n",
    "        self,\n",
    "        hidden_states,\n",
    "        attention_mask=None,\n",
    "        head_mask=None,\n",
    "        encoder_hidden_states=None,\n",
    "        encoder_attention_mask=None,\n",
    "        past_key_value=None,\n",
    "        output_attentions=False,\n",
    "    ):\n",
    "        if self.is_decoder: raise Exception('removed')\n",
    "        \n",
    "        # decoder uni-directional self-attention cached key/values tuple is at positions 1,2\n",
    "        self_attn_past_key_value = past_key_value[:2] if past_key_value is not None else None\n",
    "        self_attention_outputs = self.attention(\n",
    "            hidden_states,\n",
    "            attention_mask,\n",
    "            head_mask,\n",
    "            output_attentions=output_attentions,\n",
    "            past_key_value=self_attn_past_key_value,\n",
    "        )\n",
    "        attention_output = self_attention_outputs[0]\n",
    "\n",
    "        # if decoder, the last output is tuple of self-attn cache\n",
    "        if self.is_decoder:\n",
    "            pass\n",
    "        else:\n",
    "            outputs = self_attention_outputs[1:]  # add self attentions if we output attention weights\n",
    "\n",
    "        layer_output = apply_chunking_to_forward(\n",
    "            self.feed_forward_chunk, self.chunk_size_feed_forward, self.seq_len_dim, attention_output\n",
    "        )\n",
    "        outputs = (layer_output,) + outputs\n",
    "\n",
    "        return outputs\n",
    "\n",
    "    def feed_forward_chunk(self, attention_output):\n",
    "        intermediate_output = self.intermediate(attention_output)\n",
    "        layer_output = self.output(intermediate_output, attention_output)\n",
    "        return layer_output\n",
    "\n",
    "\n",
    "class BertEncoder(nn.Module):\n",
    "    def __init__(self, config):\n",
    "        super().__init__()\n",
    "        self.config = config\n",
    "        self.layer = nn.ModuleList([BertLayer(config) for _ in range(config.num_hidden_layers)])\n",
    "        self.gradient_checkpointing = False\n",
    "\n",
    "    def forward(\n",
    "        self,\n",
    "        hidden_states,\n",
    "        attention_mask=None,\n",
    "        head_mask=None,\n",
    "        encoder_hidden_states=None,\n",
    "        encoder_attention_mask=None,\n",
    "        past_key_values=None,\n",
    "        use_cache=None,\n",
    "        output_attentions=False,\n",
    "        output_hidden_states=False,\n",
    "        return_dict=True,\n",
    "    ):\n",
    "        all_hidden_states = () if output_hidden_states else None\n",
    "        all_self_attentions = () if output_attentions else None\n",
    "        all_cross_attentions = () if output_attentions and self.config.add_cross_attention else None\n",
    "\n",
    "        next_decoder_cache = () if use_cache else None\n",
    "        for i, layer_module in enumerate(self.layer):\n",
    "            if output_hidden_states:\n",
    "                all_hidden_states = all_hidden_states + (hidden_states,)\n",
    "\n",
    "            layer_head_mask = head_mask[i] if head_mask is not None else None\n",
    "            past_key_value = past_key_values[i] if past_key_values is not None else None\n",
    "\n",
    "            if self.gradient_checkpointing and self.training:\n",
    "                raise Exception('removed')\n",
    "            else:\n",
    "                layer_outputs = layer_module(\n",
    "                    hidden_states,\n",
    "                    attention_mask,\n",
    "                    layer_head_mask,\n",
    "                    encoder_hidden_states,\n",
    "                    encoder_attention_mask,\n",
    "                    past_key_value,\n",
    "                    output_attentions,\n",
    "                )\n",
    "\n",
    "            hidden_states = layer_outputs[0]\n",
    "            if use_cache:\n",
    "                next_decoder_cache += (layer_outputs[-1],)\n",
    "            if output_attentions:\n",
    "                all_self_attentions = all_self_attentions + (layer_outputs[1],)\n",
    "                if self.config.add_cross_attention:\n",
    "                    all_cross_attentions = all_cross_attentions + (layer_outputs[2],)\n",
    "\n",
    "        if output_hidden_states:\n",
    "            all_hidden_states = all_hidden_states + (hidden_states,)\n",
    "\n",
    "        if not return_dict:\n",
    "            return tuple(\n",
    "                v\n",
    "                for v in [\n",
    "                    hidden_states,\n",
    "                    next_decoder_cache,\n",
    "                    all_hidden_states,\n",
    "                    all_self_attentions,\n",
    "                    all_cross_attentions,\n",
    "                ]\n",
    "                if v is not None\n",
    "            )\n",
    "        return BaseModelOutputWithPastAndCrossAttentions(\n",
    "            last_hidden_state=hidden_states,\n",
    "            past_key_values=next_decoder_cache,\n",
    "            hidden_states=all_hidden_states,\n",
    "            attentions=all_self_attentions,\n",
    "            cross_attentions=all_cross_attentions,\n",
    "        )\n",
    "\n",
    "\n",
    "class BertPooler(nn.Module):\n",
    "    def __init__(self, config):\n",
    "        super().__init__()\n",
    "        self.dense = nn.Linear(config.hidden_size, config.hidden_size)\n",
    "        self.activation = nn.Tanh()\n",
    "\n",
    "    def forward(self, hidden_states):\n",
    "        # We \"pool\" the model by simply taking the hidden state corresponding\n",
    "        # to the first token.\n",
    "        first_token_tensor = hidden_states[:, 0]\n",
    "        pooled_output = self.dense(first_token_tensor)\n",
    "        pooled_output = self.activation(pooled_output)\n",
    "        return pooled_output\n",
    "\n",
    "\n",
    "class BertPreTrainedModel(PreTrainedModel):\n",
    "    \"\"\"\n",
    "    An abstract class to handle weights initialization and a simple interface for downloading and loading pretrained\n",
    "    models.\n",
    "    \"\"\"\n",
    "\n",
    "    config_class = BertConfig\n",
    "    #load_tf_weights = load_tf_weights_in_bert\n",
    "    base_model_prefix = \"bert\"\n",
    "    supports_gradient_checkpointing = True\n",
    "    _keys_to_ignore_on_load_missing = [r\"position_ids\"]\n",
    "\n",
    "    def _init_weights(self, module):\n",
    "        \"\"\"Initialize the weights\"\"\"\n",
    "        if isinstance(module, nn.Linear):\n",
    "            # Slightly different from the TF version which uses truncated_normal for initialization\n",
    "            # cf https://github.com/pytorch/pytorch/pull/5617\n",
    "            module.weight.data.normal_(mean=0.0, std=self.config.initializer_range)\n",
    "            if module.bias is not None:\n",
    "                module.bias.data.zero_()\n",
    "        elif isinstance(module, nn.Embedding):\n",
    "            module.weight.data.normal_(mean=0.0, std=self.config.initializer_range)\n",
    "            if module.padding_idx is not None:\n",
    "                module.weight.data[module.padding_idx].zero_()\n",
    "        elif isinstance(module, nn.LayerNorm):\n",
    "            module.bias.data.zero_()\n",
    "            module.weight.data.fill_(1.0)\n",
    "\n",
    "    def _set_gradient_checkpointing(self, module, value=False):\n",
    "        if isinstance(module, BertEncoder):\n",
    "            module.gradient_checkpointing = value\n",
    "\n",
    "\n",
    "class BertModel(BertPreTrainedModel):\n",
    "    \"\"\"\n",
    "\n",
    "    The model can behave as an encoder (with only self-attention) as well as a decoder, in which case a layer of\n",
    "    cross-attention is added between the self-attention layers, following the architecture described in [Attention is\n",
    "    all you need](https://arxiv.org/abs/1706.03762) by Ashish Vaswani, Noam Shazeer, Niki Parmar, Jakob Uszkoreit,\n",
    "    Llion Jones, Aidan N. Gomez, Lukasz Kaiser and Illia Polosukhin.\n",
    "\n",
    "    To behave as an decoder the model needs to be initialized with the `is_decoder` argument of the configuration set\n",
    "    to `True`. To be used in a Seq2Seq model, the model needs to initialized with both `is_decoder` argument and\n",
    "    `add_cross_attention` set to `True`; an `encoder_hidden_states` is then expected as an input to the forward pass.\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self, config, add_pooling_layer=True):\n",
    "        super().__init__(config)\n",
    "        self.config = config\n",
    "\n",
    "        self.embeddings = BertEmbeddings(config)\n",
    "        self.encoder = BertEncoder(config)\n",
    "\n",
    "        self.pooler = BertPooler(config) if add_pooling_layer else None\n",
    "\n",
    "        # Initialize weights and apply final processing\n",
    "        self.post_init()\n",
    "\n",
    "    def get_input_embeddings(self):\n",
    "        return self.embeddings.word_embeddings\n",
    "\n",
    "    def set_input_embeddings(self, value):\n",
    "        self.embeddings.word_embeddings = value\n",
    "\n",
    "    def _prune_heads(self, heads_to_prune):\n",
    "        \"\"\"\n",
    "        Prunes heads of the model. heads_to_prune: dict of {layer_num: list of heads to prune in this layer} See base\n",
    "        class PreTrainedModel\n",
    "        \"\"\"\n",
    "        for layer, heads in heads_to_prune.items():\n",
    "            self.encoder.layer[layer].attention.prune_heads(heads)\n",
    "\n",
    "\n",
    "    def forward(\n",
    "        self,\n",
    "        input_ids=None,\n",
    "        attention_mask=None,\n",
    "        token_type_ids=None,\n",
    "        position_ids=None,\n",
    "        head_mask=None,\n",
    "        inputs_embeds=None,\n",
    "        encoder_hidden_states=None,\n",
    "        encoder_attention_mask=None,\n",
    "        past_key_values=None,\n",
    "        use_cache=None,\n",
    "        output_attentions=None,\n",
    "        output_hidden_states=None,\n",
    "        return_dict=None,\n",
    "    ):\n",
    "        output_attentions = output_attentions if output_attentions is not None else self.config.output_attentions\n",
    "        output_hidden_states = (\n",
    "            output_hidden_states if output_hidden_states is not None else self.config.output_hidden_states\n",
    "        )\n",
    "        return_dict = return_dict if return_dict is not None else self.config.use_return_dict\n",
    "\n",
    "        if self.config.is_decoder:\n",
    "            use_cache = use_cache if use_cache is not None else self.config.use_cache\n",
    "        else:\n",
    "            use_cache = False\n",
    "\n",
    "        if input_ids is not None and inputs_embeds is not None:\n",
    "            raise ValueError(\"You cannot specify both input_ids and inputs_embeds at the same time\")\n",
    "        elif input_ids is not None:\n",
    "            input_shape = input_ids.size()\n",
    "        elif inputs_embeds is not None:\n",
    "            input_shape = inputs_embeds.size()[:-1]\n",
    "        else:\n",
    "            raise ValueError(\"You have to specify either input_ids or inputs_embeds\")\n",
    "\n",
    "        batch_size, seq_length = input_shape\n",
    "        device = input_ids.device if input_ids is not None else inputs_embeds.device\n",
    "\n",
    "        # past_key_values_length\n",
    "        past_key_values_length = past_key_values[0][0].shape[2] if past_key_values is not None else 0\n",
    "\n",
    "        if attention_mask is None:\n",
    "            attention_mask = torch.ones(((batch_size, seq_length + past_key_values_length)), device=device)\n",
    "\n",
    "        if token_type_ids is None:\n",
    "            if hasattr(self.embeddings, \"token_type_ids\"):\n",
    "                buffered_token_type_ids = self.embeddings.token_type_ids[:, :seq_length]\n",
    "                buffered_token_type_ids_expanded = buffered_token_type_ids.expand(batch_size, seq_length)\n",
    "                token_type_ids = buffered_token_type_ids_expanded\n",
    "            else:\n",
    "                token_type_ids = torch.zeros(input_shape, dtype=torch.long, device=device)\n",
    "\n",
    "        # We can provide a self-attention mask of dimensions [batch_size, from_seq_length, to_seq_length]\n",
    "        # ourselves in which case we just need to make it broadcastable to all heads.\n",
    "        extended_attention_mask: torch.Tensor = self.get_extended_attention_mask(attention_mask, input_shape, device)\n",
    "\n",
    "        # If a 2D or 3D attention mask is provided for the cross-attention\n",
    "        # we need to make broadcastable to [batch_size, num_heads, seq_length, seq_length]\n",
    "        if self.config.is_decoder and encoder_hidden_states is not None:\n",
    "            encoder_batch_size, encoder_sequence_length, _ = encoder_hidden_states.size()\n",
    "            encoder_hidden_shape = (encoder_batch_size, encoder_sequence_length)\n",
    "            if encoder_attention_mask is None:\n",
    "                encoder_attention_mask = torch.ones(encoder_hidden_shape, device=device)\n",
    "            encoder_extended_attention_mask = self.invert_attention_mask(encoder_attention_mask)\n",
    "        else:\n",
    "            encoder_extended_attention_mask = None\n",
    "\n",
    "        # Prepare head mask if needed\n",
    "        # 1.0 in head_mask indicate we keep the head\n",
    "        # attention_probs has shape bsz x n_heads x N x N\n",
    "        # input head_mask has shape [num_heads] or [num_hidden_layers x num_heads]\n",
    "        # and head_mask is converted to shape [num_hidden_layers x batch x num_heads x seq_length x seq_length]\n",
    "        head_mask = self.get_head_mask(head_mask, self.config.num_hidden_layers)\n",
    "\n",
    "        embedding_output = self.embeddings(\n",
    "            input_ids=input_ids,\n",
    "            position_ids=position_ids,\n",
    "            token_type_ids=token_type_ids,\n",
    "            inputs_embeds=inputs_embeds,\n",
    "            past_key_values_length=past_key_values_length,\n",
    "        )\n",
    "        encoder_outputs = self.encoder(\n",
    "            embedding_output,\n",
    "            attention_mask=extended_attention_mask,\n",
    "            head_mask=head_mask,\n",
    "            encoder_hidden_states=encoder_hidden_states,\n",
    "            encoder_attention_mask=encoder_extended_attention_mask,\n",
    "            past_key_values=past_key_values,\n",
    "            use_cache=use_cache,\n",
    "            output_attentions=output_attentions,\n",
    "            output_hidden_states=output_hidden_states,\n",
    "            return_dict=return_dict,\n",
    "        )\n",
    "        sequence_output = encoder_outputs[0]\n",
    "        pooled_output = self.pooler(sequence_output) if self.pooler is not None else None\n",
    "\n",
    "        if not return_dict:\n",
    "            return (sequence_output, pooled_output) + encoder_outputs[1:]\n",
    "\n",
    "        return BaseModelOutputWithPoolingAndCrossAttentions(\n",
    "            last_hidden_state=sequence_output,\n",
    "            pooler_output=pooled_output,\n",
    "            past_key_values=encoder_outputs.past_key_values,\n",
    "            hidden_states=encoder_outputs.hidden_states,\n",
    "            attentions=encoder_outputs.attentions,\n",
    "            cross_attentions=encoder_outputs.cross_attentions,\n",
    "        )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Some weights of the model checkpoint at bert-base-uncased were not used when initializing BertModel: ['cls.predictions.transform.dense.weight', 'cls.predictions.decoder.weight', 'cls.seq_relationship.bias', 'cls.predictions.transform.dense.bias', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.transform.LayerNorm.weight', 'cls.predictions.transform.LayerNorm.bias']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "Some weights of the model checkpoint at bert-base-uncased were not used when initializing BertModel: ['cls.predictions.transform.dense.weight', 'cls.predictions.decoder.weight', 'cls.seq_relationship.bias', 'cls.predictions.transform.dense.bias', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.transform.LayerNorm.weight', 'cls.predictions.transform.LayerNorm.bias']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n"
     ]
    }
   ],
   "source": [
    "from transformers import BertTokenizer\n",
    "from transformers import BertModel as RealBertModel\n",
    "hugging = 'google/bert_uncased_L-2_H-128_A-2'\n",
    "hugging = 'bert-base-uncased'\n",
    "model = BertModel.from_pretrained(hugging)\n",
    "model.cuda()\n",
    "model.eval()\n",
    "real_model = RealBertModel.from_pretrained(hugging)\n",
    "real_model.cuda()\n",
    "real_model.eval()\n",
    "tokenizer = BertTokenizer.from_pretrained(hugging)\n",
    "text = ['[CLS]Hello, World! I love to play with transformers.', '[CLS]This is the batch.', '[CLS]I love cats']\n",
    "tokenized_result = tokenizer(text, padding=True, return_tensors='pt')\n",
    "tokenized_result.input_ids = tokenized_result.input_ids.cuda()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "torch.Size([3, 14, 768])"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.encoder.layer[0].attention.self.lowdim = False\n",
    "model(tokenized_result.input_ids, output_attentions=True)\n",
    "att = model.encoder.layer[0].attention.self\n",
    "hidden_states = att.last_hidden_states.clone().detach()\n",
    "attention_mask = att.last_attention_mask.clone().detach()\n",
    "hidden_states.shape\n",
    "#att.query.weight.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "0.003365039825439453\n"
     ]
    }
   ],
   "source": [
    "t = time.time()\n",
    "normal_result = att.forward(\n",
    "    hidden_states = hidden_states,\n",
    "    attention_mask = attention_mask,\n",
    "    output_attentions=True,\n",
    ")\n",
    "print(time.time() - t)\n",
    "real_result = real_model.encoder.layer[0].attention.self.forward(\n",
    "    hidden_states = hidden_states,\n",
    "    attention_mask = attention_mask,\n",
    "    output_attentions=True,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(209.2968, device='cuda:0') tensor(52.4660, device='cuda:0')\n",
      "tensor(190.6799, device='cuda:0') tensor(48.0566, device='cuda:0')\n",
      "tensor(-8.6462, device='cuda:0') tensor(-2.0208, device='cuda:0')\n",
      "tensor(52.1677, device='cuda:0') tensor(12.6805, device='cuda:0')\n",
      "tensor(6.4235, device='cuda:0') tensor(0.6227, device='cuda:0')\n",
      "tensor(6.1717, device='cuda:0') tensor(0.0492, device='cuda:0')\n",
      "tensor(23.4705, device='cuda:0') tensor(5.1969, device='cuda:0')\n",
      "tensor(3.9314, device='cuda:0') tensor(-2.7471, device='cuda:0')\n",
      "tensor(38.7964, device='cuda:0') tensor(8.9564, device='cuda:0')\n",
      "tensor(0.8757, device='cuda:0') tensor(-0.6088, device='cuda:0')\n",
      "tensor(28.8300, device='cuda:0') tensor(5.6413, device='cuda:0')\n",
      "tensor(6.4083, device='cuda:0') tensor(5.7907, device='cuda:0')\n",
      "tensor(49.2122, device='cuda:0') tensor(10.4318, device='cuda:0')\n"
     ]
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "i = 12\n",
    "a = hidden_states[0,i]\n",
    "b_ = F.adaptive_avg_pool1d(hidden_states, output_size = (model.config.hidden_size // 2, ))\n",
    "b = b_[0, i]\n",
    "plt.bar(range(len(a)), a.cpu().numpy())\n",
    "plt.show()\n",
    "plt.bar(range(len(b)), b.cpu().numpy())\n",
    "plt.show()\n",
    "for k in range(13):\n",
    "    print(torch.sum(hidden_states[0,0] * hidden_states[0,k]), torch.sum(b_[0,0] * b_[0,k]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "64\n",
      "torch.Size([768, 768]) to torch.Size([384, 768])\n",
      "torch.Size([768, 768]) to torch.Size([384, 768])\n",
      "real\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAATQAAAD4CAYAAABi3BrkAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8rg+JYAAAACXBIWXMAAAsTAAALEwEAmpwYAAAadUlEQVR4nO3de5Ad5Xnn8e9PM5JGN4RAshASBrEIYkIMOGNim12cmItlm5L8B96FLC68oUrllHGcOCkXrLegilS22DjlTaqWtVHZsqmEhTjgbFRZYawAjuMkEImLASErDAKjEQLduAh0m8uzf5we58wczZzW20c95/T8PlVdnO7TTz890ujhffvtflsRgZlZFUyb7BMwM2sVFzQzqwwXNDOrDBc0M6sMFzQzq4zuMpPN6JoVs6bPTz+AlB47OJgeC6ACtT+Gi+WmwM89reD/s4aHCoUPLOhJjp1+oNjf2cC89F/v6W8NFMo9NDs9dyxK/zM/+vpbDLx1sMAvDHz8N+bEvv35zuGJZ448FBEri+RrpVIL2qzp8/nIss8mx0fPjPTkr+9NjwU0a1ZybBw+XCh3kWKqubMLpY4D7xSKf+2ac5NjT/v7Yn9nu35jUXLs6Q/uLJT7wPsXJ8cOfH5fcuxzX7w7OXbE3v1DPP7Qslz7Tl/y4sLCCVuo1IJmZp0gGCrcq5gcLmhmNkoAw3TmDfcuaGbWYJjObKEVumIsaaWkbZL6JN3cqpMys8kTBAMxnGtpN8ktNEldwJ3AlUA/sEnS+oh4vlUnZ2blC2CoQ7ucRVpolwB9EbE9Io4C9wGrW3NaZjaZholcS7spcg1tKbCjbr0f+LWxO0laA6wB6OmeVyCdmZUhgKEOnYXnhD8pEBFrI6I3InpndBW7J8rMyjGcc2k3RVpoO4Ez6taXZdvMrIMF0bHX0IoUtE3ACknLqRWya4HfbMlZmdmkiYCBzqxn6QUtIgYl3QQ8BHQB6yJiS8vOzMwmiRgq8vzwJCp0Y21EbAA2tOhczKwNBDA81VpoZlZdU7KFZmbVU7ux1gWtueFhOFRgKp2uAneZFJlLDWpXSlMNFRvg1pz0OcUm2+y96T+7Dh0plHveqwXmUztabD60nr1Hk2N/fOEDybGXzHojOXZEAAPRmXO/uoVmZqMEYqhDJ7N2QTOzBsPhLqeZVUAnX0PrzHalmZ1AYiim5VpyHa3JNGOSPi/pWUlPS/qJpPPrvrsli9sm6ePNcrmFZmaj1GasbU1bJ+c0Y/8nIr6Z7b8K+DqwMits1wK/DJwO/J2kcyNi3De4uIVmZqNEiKPRlWvJoek0YxHxdt3qHPjFg6Srgfsi4khEvAT0Zccbl1toZtZgOP81tIWSNtetr42ItXXreacZ+wLwZWAG8LG62MfGxC6d6GRc0MxslNqgQO7O296I6C2cM+JO4E5Jvwn8N+CGlOO4oJnZGMp9wT+H451m7D7gG4mxvoZmZqONDArkWXL4xTRjkmZQu8i/vn4HSSvqVj8FvJB9Xg9cK2lmNk3ZCuBfJkrmFpqZNRhq0Y21400zJul2YHNErAduknQFMAC8QdbdzPb7HvA8MAh8YaIRTnBBM7MxAjEQrSsNx5pmLCJurfv8pQli/wj4o7y5XNDMbJTjHBRoKy5oZjZKoJZ1OctW8vRBQRw6lB4/b05yqObNTc8LDO1If//L4asuLpT7wLL0v6aeN4tNPXpwUbH/U5/2j28mx+5ctaxQ7iWP7E+OffeiCW93amrm/9uUHPvJyz+THPvC9u8kx9Zr1ZMCZXMLzcxGiaCVt22UygXNzEapDQrkeqyp7bigmVkDDwqYWSUE6tgJHpPLsKQzJD0q6XlJWySNey+JmXWWIablWtpNkRbaIPD7EfGkpHnAE5I2jpnnyMw6TO29nO1XrPIo8ub0XcCu7PMBSVupTe3hgmbW0abom9NHSDoLuBh4/BjfrQHWAPRMS7+PzMzKUXuN3RQd5ZQ0F3gA+N0xM08CkE32thZgfveiDn3BvNnUEaGp1+UEkDSdWjG7JyK+35pTMrPJNuVurJUk4NvA1oj4eutOycwmU20+tKl3De1S4LPAs5Kezrb912yqEDPrWC2dsbZURUY5fwIdWsbNbFy12zY685+2nxQws1H8LKeZVYqnD8pjejcsXpQcfvDsBcmxs159JzkWYNry9ybHzvlp+lxqAMPT0+cFm/f4zwvlnn3OkkLx+y6cnxy7+F+K/Z29ff7JybGzdx0plPvIJz+YHDvrlQPpiVW8q1ibPshdTjOrCF9DM7NKqM224S6nmVVA7dEnFzQzqwS30MysQqbikwJmVkGdPMrZme1KMzuhhmNariUPSSslbZPUJ+nmY3z/5Wzm62ckPSzpzLrvhiQ9nS3rm+VyC83MRmnlOwUkdQF3AlcC/cAmSevHzGz9FNAbEQcl/Tbwx8B/yr47FBEX5c3nFpqZjRLAYEzLteRwCdAXEdsj4ihwH7B6VL6IRyPiYLb6GJB8J7kLmpk1aGGXcymwo269P9s2nhuBB+vWeyRtlvSYpE83S+Yup5mNFsfV5VwoaXPd+tpslurjJul6oBf4aN3mMyNip6SzgUckPRsRL453DBc0MxvlOCd43BsRvRN8vxM4o259WbZtFElXAF8FPhoRv3iQNiJ2Zv/dLulH1N5dMm5Bc5fTzBoMZ620ZksOm4AVkpZLmgFcC4warZR0MXAXsCoidtdtXyBpZvZ5IbVJZSd8q5xbaGY2SisneIyIQUk3AQ8BXcC6iNgi6XZgc0SsB74GzAX+qjazP69ExCrgfcBdkoapNb7uaPbe33IL2tAQejt9SpiePT3JsYeWzU2OBZjz5BvpwT0zC+U+Mj99sr2ui9KnPQIoOs/fws37k2OHe4r9es5/dl9y7I5PpU9zBfDev3wlOfbo2em546UWTB+EGBxuXectm5Z/w5htt9Z9vmKcuH8CfuV4crmFZmYN/OiTmVVDeD40M6sIvyTFzCplyha07FmtzcDOiLi6+CmZ2WQKxFALBwXK1IoW2peArcBJLTiWmbWBTh0UKFSGJS0DPgV8qzWnY2aTLaKlN9aWqmgL7U+BrwDzxttB0hpgDUBPV7F7wcysHNGGxSqP5BaapKuB3RHxxET7RcTaiOiNiN4Z02alpjOz0uRrnVWthXYpsErSJ4Ee4CRJfxER17fm1Mxssky5FlpE3BIRyyLiLGoPnD7iYmbW+SJgaFi5lnbj+9DMrEGnjnK2pKBFxI+AH7XiWGY2uYLO7XK6hWZmY7TnBf88XNDMrEHEZJ9BmnILWgRx9GhyuA4PJsfOeSJ9fiqAfVcsT4499SevFsp96uN70oP3v1UotwrO5da35ozmO43jnP/9UqHc+y4/Kzl28ROHC+UuMqfZvvelz/s3uKU1jyy5y2lmlVAb5Zy6z3KaWcW4y2lmleEup5lVQiAXNDOrjg7tcbqgmdkYAdGGjzXl4YJmZg3c5TSzyvAop5lVgp/lNLPqCKBDC1pn3g5sZidURL4lD0krJW2T1Cfp5mN8/2VJz0t6RtLDks6s++4GSS9kyw3NcrmgmdkYIobzLU2PVHvN5Z3AJ4DzgesknT9mt6eA3oh4P3A/8MdZ7CnAbcCvAZcAt0laMFE+FzQzaxQ5l+YuAfoiYntEHAXuA1aPShXxaEQczFYfA5Zlnz8ObIyI/RHxBrARWDlRMl9DM7PR4rgGBRZK2ly3vjYi1tatLwV21K33U2txjedG4MEJYpdOdDKlFrShuTN559+fkxz/1vL0013009nJsQCnPLk/PbjgGPj+3oXJsT1vTthCb2qwp1gj/py1O5rvNI5YUOzd1af87dbk2IMfPrdQ7jnPv5Ycu6RvKDn2pTfTp+caJf+v7N6I6G1FSknXA73AR1OP4S6nmR2Dci5N7QTqJ8Vblm0bnU26AvgqsCoijhxPbD0XNDNrNJxzaW4TsELSckkzqL0hbn39DpIuBu6iVsx21331EHCVpAXZYMBV2bZx+RqamY3WwvvQImJQ0k3UClEXsC4itki6HdgcEeuBrwFzgb+SBPBKRKyKiP2S/pBaUQS4PSImvPbjgmZmDVr56FNEbAA2jNl2a93nKyaIXQesy5urUJdT0smS7pf0M0lbJX24yPHMrE207raNUhVtof0Z8IOIuCbrHxcbSjSz9tChjz4lFzRJ84HLgM8BZDfNtWjM2Mwmk9qw9ZVHkS7ncmAP8B1JT0n6lqQ5Y3eStEbSZkmbB4+8WyCdmZUiBMM5lzZTpKB1Ax8AvhERFwPvAg0PnkbE2ojojYje7pkN9c7M2lGHXkMrUtD6gf6IeDxbv59agTOzTjfVClpEvAbskHRetuly4PmWnJWZTa4OLWhFRzm/CNyTjXBuB/5L8VMys0nVwRM8FipoEfE0tYdJzaxCOnWU008KmFkjFzQzqwq30HKILnH45K7k+APL8z3efyxL/uFI850mMpSee2jR/EKpZ+8ZTI6d9cKeQrk5XPDPbcb09NjB9HnBAKLA31nRf9CHz3lPcmz3gfT70+ONFv2TnorX0Mysgtp0BDMPFzQza+SCZmZVofTe+qRyQTOzRm6hmVkVKDzKaWZV4lFOM6sMt9DMrCrc5TSzagiPcppZlbiFZmaV4YJmZlXRqdfQCr2X08ysnbigmVmjFk7BLWmlpG2S+iQ1vEhJ0mWSnpQ0KOmaMd8NSXo6W9Y3y+Uup5mN1sJRTkldwJ3AldRerLRJ0vqIqH//yCvU3u/7B8c4xKGIuChvvlIL2rSjwbwd6XM9DXfPSM99OH1OMQC60huz037+eqHUQ6ctT46NIvORAcOnzisU/9a5c5NjT37mzUK5D3/kvOY7jaPntWLvkD28eHZybNcb6blVcA65X2jdNbRLgL6I2A4g6T5gNXUvVIqIl7PvCpdRdznNbBTxb89zNluAhSMvEs+WNWMOtxTYUbfen23Lqyc77mOSPt1sZ3c5zaxR/hba3og4kS9KOjMidko6G3hE0rMR8eJ4O7uFZmaj5Wyd5by1YydwRt36smxbvlOJ2Jn9dzvwI+DiifYvVNAk/Z6kLZKek3SvpJ4ixzOzNjGcc2luE7BC0vLs/b3XAk1HKwEkLZA0M/u8ELiUJi8zTy5okpYCvwP0RsQFQFd2smbW4VrVQouIQeAm4CFgK/C9iNgi6XZJqwAkfVBSP/AZ4C5JW7Lw9wGbJf0UeBS4Y8zoaIOi19C6gVmSBoDZwKsFj2dm7aCFTwpExAZgw5htt9Z93kStKzo27p+AXzmeXMkttKxv+yfU7iHZBbwVET8cu5+kNSMjIAMDxYbCzawEeW+qbcPHo4p0ORdQu59kOXA6MEfS9WP3i4i1EdEbEb3Tp89JP1MzK00LBwVKVWRQ4ArgpYjYExEDwPeBj7TmtMxsUnVoC63INbRXgA9Jmg0cAi4HNrfkrMxsUnXqBI9FrqE9DtwPPAk8mx1rbYvOy8wmSwdfQys0yhkRtwG3tehczKwNKFs6kR99MrNGbdj6ysMFzcwatOMIZh6lFjQNB93vpE8fpEifPogj6XkB9lx2WnLsSS8Xm4Jnz4Xpf00DcxYWyr1g02uF4uc/8EJyrOYWu81nxnM/S44d+PUPFMod3QU6ba/vTY8dLDhN1ggXNDOrBL/GzswqxS00M6sKX0Mzs+pwQTOzqnALzcyqIcg7eWPbcUEzs1FGXpLSiVzQzKyRC5qZVYWiMyuaC5qZjdamM2nk4YJmZg18Dc3MKsOPPplZdbiFZmaV0KYvQMnDBc3MGrmgNRfTxMBJ6XOaDc5Kn2NqaGGxOclOejl9PrWZT20vlPuUU85Njj3p6d2Fckd3V6H4N/5j+rxip/z0jUK5+XdLk0NnvrSnUOqhX1qcHKsF89MTHyr29wWdfWNtkdfYmVlFaThyLbmOJa2UtE1Sn6Sbj/H9ZZKelDQo6Zox390g6YVsuaFZLnc5zWy0Ft6HJqkLuBO4EugHNklaHxHP1+32CvA54A/GxJ5C7SVMvdkZPZHFjtt0dwvNzBpoON+SwyVAX0Rsj4ijwH3A6vodIuLliHiGxkfiPw5sjIj9WRHbCKycKFnTgiZpnaTdkp6r23aKpI1ZM3CjpAW5fjQz6wz538u5UNLmumXNmCMtBXbUrfdn2/I47tg8LbTv0lgVbwYejogVwMPZuplVhCLfAuyNiN66ZVJfNt60oEXEj4H9YzavBu7OPt8NfLq1p2VmkyaAiHxLczuBM+rWl2XbTkhs6jW0xRGxK/v8GjDuGLWkNSPN0YGBdxPTmVmZWngNbROwQtJySTOAa4H1OU/jIeAqSQuyy1pXZdvGVXhQICImHBOJiLUjzdHp04u9Z9HMTryR+9BydjknFBGDwE3UCtFW4HsRsUXS7ZJWAUj6oKR+4DPAXZK2ZLH7gT+kVhQ3Abdn28aVetvG65KWRMQuSUuAYndvmln7yN+dzHm42ABsGLPt1rrPm6h1J48Vuw5YlzdXagttPTByk9sNwN8kHsfM2lCrWmhly3Pbxr3APwPnSeqXdCNwB3ClpBeAK7J1M6uK/LdttJWmXc6IuG6cry5v8bmYWZtox9ZXHn70ycxGC2CoMyuaC5qZNXALLYdp7xxi5k+2JMef3vee5NijZ56aHAvQ89Le5Nh4T7Hcu1anT1307uIlhXIv2HakUPzCv3spOfbw+9Kn/wGY+WRfcuzRC88ulHv2tvSB/4Fl6b8vsbdF/6T91iczqwq30MysGtp0BDMPFzQzG0WAPChgZlXhN6ebWTW4y2lm1dHaZznL5IJmZg08ymlm1eEWmplVQniU08yqpDPrmQuamTXybRtmVh0uaGZWCUHjK387hAuamY0iwl1OM6uQ4c5sopVb0GbOQGce8+UuuRxcviA5tqf/QHIswNGl6bmn7y6W+8w/70qO7dm+q/lOExhYcnKh+H0fOys59qTthwrlPvShc5NjZ/38zUK537ngtPTcuwq8v3a4BS0rdznNrErc5TSz6nBBM7Nq6NyH0/O8l3OdpN2Snqvb9jVJP5P0jKS/lnTyCT1LMyvPyFuf8iw5SFopaZukPkk3H+P7mZL+Mvv+cUlnZdvPknRI0tPZ8s1mufK8Of27wMox2zYCF0TE+4F/BW7JcRwz6xCKyLU0PY7UBdwJfAI4H7hO0vljdrsReCMizgH+J/A/6r57MSIuypbPN8vXtKBFxI+B/WO2/TAiBrPVx4D0oUszaz8R+ZbmLgH6ImJ7RBwF7gNWj9lnNXB39vl+4HJJSjntPC20Zn4LeLAFxzGzdhDUbv/IszS3FNhRt96fbTvmPllD6S1g5F1+yyU9JenvJf2HZskKDQpI+iowCNwzwT5rgDUAPd0nFUlnZqU4rkGBhZI2162vjYi1LTqRXcB7I2KfpF8F/q+kX46It8cLSC5okj4HXA1cHjH+T5/9cGsB5s9a0plDJ2ZTTf6Ctjcieif4fidwRt36smzbsfbpl9QNzAf2ZXXlSO104glJLwLnApsZR1KXU9JK4CvAqog4mHIMM2tTAQwN51ua2wSskLRc0gzgWmD9mH3WAzdkn68BHomIkLQoG1RA0tnACmD7RMmattAk3Qv8OrWmZT9wG7VRzZnAxuza3WN5RiDMrBMERGuefYqIQUk3AQ8BXcC6iNgi6XZgc0SsB74N/LmkPmoDkNdm4ZcBt0saoPYw1ucjYn9jln/TtKBFxHXH2Pzt3D+RmXWeFt5YGxEbgA1jtt1a9/kw8JljxD0APHA8ufykgJmNNjLK2YFc0MysUYc++lRuQZOI6ekph3qS7rWrpT58JDkWYHDO3OTYGQODzXeawPD09J97cFGxW2Wm7y825jNz/vTk2O43i+WOGem3WeqdYlMXDc4+tflO4ziyaHZybHS34tZSXNDMrCIiYGhoss8iiQuamTVyC83MKsMFzcyqIfdzmm3HBc3MRguIFt1YWzYXNDNrlO+xprbjgmZmo0X4NXZmViEeFDCzqgi30MysGjr3rU8uaGY2mh9ON7OqCCD86JOZVUK0boLHsrmgmVmDcJfTzCqjQ1tomuCFTa1PJu0Bfj7BLguBvSWdjnM7dxVznxkRi4ocQNIPqJ1jHnsjYmWRfK1UakFrRtLmJq/Ecm7ndm4bV4umtzQzm3wuaGZWGe1W0Fr1Cnnndm7nnoLa6hqamVkR7dZCMzNL5oJmZpXRFgVN0kpJ2yT1Sbq5xLxnSHpU0vOStkj6Ulm5686hS9JTkv625LwnS7pf0s8kbZX04RJz/1725/2cpHsl9ZzgfOsk7Zb0XN22UyRtlPRC9t8FJeb+Wvbn/oykv5Z08onIPRVNekGT1AXcCXwCOB+4TtL5JaUfBH4/Is4HPgR8ocTcI74EbC05J8CfAT+IiF8CLizrHCQtBX4H6I2IC4Au4NoTnPa7wNibP28GHo6IFcDD2XpZuTcCF0TE+4F/BW45QbmnnEkvaMAlQF9EbI+Io8B9wOoyEkfEroh4Mvt8gNo/6qVl5AaQtAz4FPCtsnJmeecDlwHfBoiIoxHxZomn0A3MktQNzAZePZHJIuLHwP4xm1cDd2ef7wY+XVbuiPhhRAxmq48By05E7qmoHQraUmBH3Xo/JRaVEZLOAi4GHi8x7Z8CXwHKfnBuObAH+E7W3f2WpDllJI6IncCfAK8Au4C3IuKHZeQeY3FE7Mo+vwYsnoRzAPgt4MFJyl057VDQJp2kucADwO9GxNsl5bwa2B0RT5SRb4xu4APANyLiYuBdTlyXa5TsWtVqakX1dGCOpOvLyD2eqN27VPr9S5K+Su2yxz1l566qdihoO4Ez6taXZdtKIWk6tWJ2T0R8v6y8wKXAKkkvU+tmf0zSX5SUux/oj4iR1uj91ApcGa4AXoqIPRExAHwf+EhJueu9LmkJQPbf3WUml/Q54GrgP4dvBm2Zdihom4AVkpZLmkHtAvH6MhJLErXrSFsj4utl5BwREbdExLKIOIvaz/xIRJTSUomI14Adks7LNl0OPF9GbmpdzQ9Jmp39+V/O5AyKrAduyD7fAPxNWYklraR2qWFVRBwsK+9UMOkFLbs4ehPwELVf7O9FxJaS0l8KfJZa6+jpbPlkSbkn2xeBeyQ9A1wE/PcykmatwvuBJ4Fnqf0OntBHgSTdC/wzcJ6kfkk3AncAV0p6gVqr8Y4Sc/8vYB6wMfud++aJyD0V+dEnM6uMSW+hmZm1iguamVWGC5qZVYYLmplVhguamVWGC5qZVYYLmplVxv8HyNEaWJo6pVsAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "original\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAATQAAAD4CAYAAABi3BrkAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8rg+JYAAAACXBIWXMAAAsTAAALEwEAmpwYAAAadUlEQVR4nO3de5Ad5Xnn8e9PM5JGN4RAshASBrEIYkIMOGNim12cmItlm5L8B96FLC68oUrllHGcOCkXrLegilS22DjlTaqWtVHZsqmEhTjgbFRZYawAjuMkEImLASErDAKjEQLduAh0m8uzf5we58wczZzW20c95/T8PlVdnO7TTz890ujhffvtflsRgZlZFUyb7BMwM2sVFzQzqwwXNDOrDBc0M6sMFzQzq4zuMpPN6JoVs6bPTz+AlB47OJgeC6ACtT+Gi+WmwM89reD/s4aHCoUPLOhJjp1+oNjf2cC89F/v6W8NFMo9NDs9dyxK/zM/+vpbDLx1sMAvDHz8N+bEvv35zuGJZ448FBEri+RrpVIL2qzp8/nIss8mx0fPjPTkr+9NjwU0a1ZybBw+XCh3kWKqubMLpY4D7xSKf+2ac5NjT/v7Yn9nu35jUXLs6Q/uLJT7wPsXJ8cOfH5fcuxzX7w7OXbE3v1DPP7Qslz7Tl/y4sLCCVuo1IJmZp0gGCrcq5gcLmhmNkoAw3TmDfcuaGbWYJjObKEVumIsaaWkbZL6JN3cqpMys8kTBAMxnGtpN8ktNEldwJ3AlUA/sEnS+oh4vlUnZ2blC2CoQ7ucRVpolwB9EbE9Io4C9wGrW3NaZjaZholcS7spcg1tKbCjbr0f+LWxO0laA6wB6OmeVyCdmZUhgKEOnYXnhD8pEBFrI6I3InpndBW7J8rMyjGcc2k3RVpoO4Ez6taXZdvMrIMF0bHX0IoUtE3ACknLqRWya4HfbMlZmdmkiYCBzqxn6QUtIgYl3QQ8BHQB6yJiS8vOzMwmiRgq8vzwJCp0Y21EbAA2tOhczKwNBDA81VpoZlZdU7KFZmbVU7ux1gWtueFhOFRgKp2uAneZFJlLDWpXSlMNFRvg1pz0OcUm2+y96T+7Dh0plHveqwXmUztabD60nr1Hk2N/fOEDybGXzHojOXZEAAPRmXO/uoVmZqMEYqhDJ7N2QTOzBsPhLqeZVUAnX0PrzHalmZ1AYiim5VpyHa3JNGOSPi/pWUlPS/qJpPPrvrsli9sm6ePNcrmFZmaj1GasbU1bJ+c0Y/8nIr6Z7b8K+DqwMits1wK/DJwO/J2kcyNi3De4uIVmZqNEiKPRlWvJoek0YxHxdt3qHPjFg6Srgfsi4khEvAT0Zccbl1toZtZgOP81tIWSNtetr42ItXXreacZ+wLwZWAG8LG62MfGxC6d6GRc0MxslNqgQO7O296I6C2cM+JO4E5Jvwn8N+CGlOO4oJnZGMp9wT+H451m7D7gG4mxvoZmZqONDArkWXL4xTRjkmZQu8i/vn4HSSvqVj8FvJB9Xg9cK2lmNk3ZCuBfJkrmFpqZNRhq0Y21400zJul2YHNErAduknQFMAC8QdbdzPb7HvA8MAh8YaIRTnBBM7MxAjEQrSsNx5pmLCJurfv8pQli/wj4o7y5XNDMbJTjHBRoKy5oZjZKoJZ1OctW8vRBQRw6lB4/b05yqObNTc8LDO1If//L4asuLpT7wLL0v6aeN4tNPXpwUbH/U5/2j28mx+5ctaxQ7iWP7E+OffeiCW93amrm/9uUHPvJyz+THPvC9u8kx9Zr1ZMCZXMLzcxGiaCVt22UygXNzEapDQrkeqyp7bigmVkDDwqYWSUE6tgJHpPLsKQzJD0q6XlJWySNey+JmXWWIablWtpNkRbaIPD7EfGkpHnAE5I2jpnnyMw6TO29nO1XrPIo8ub0XcCu7PMBSVupTe3hgmbW0abom9NHSDoLuBh4/BjfrQHWAPRMS7+PzMzKUXuN3RQd5ZQ0F3gA+N0xM08CkE32thZgfveiDn3BvNnUEaGp1+UEkDSdWjG7JyK+35pTMrPJNuVurJUk4NvA1oj4eutOycwmU20+tKl3De1S4LPAs5Kezrb912yqEDPrWC2dsbZURUY5fwIdWsbNbFy12zY685+2nxQws1H8LKeZVYqnD8pjejcsXpQcfvDsBcmxs159JzkWYNry9ybHzvlp+lxqAMPT0+cFm/f4zwvlnn3OkkLx+y6cnxy7+F+K/Z29ff7JybGzdx0plPvIJz+YHDvrlQPpiVW8q1ibPshdTjOrCF9DM7NKqM224S6nmVVA7dEnFzQzqwS30MysQqbikwJmVkGdPMrZme1KMzuhhmNariUPSSslbZPUJ+nmY3z/5Wzm62ckPSzpzLrvhiQ9nS3rm+VyC83MRmnlOwUkdQF3AlcC/cAmSevHzGz9FNAbEQcl/Tbwx8B/yr47FBEX5c3nFpqZjRLAYEzLteRwCdAXEdsj4ihwH7B6VL6IRyPiYLb6GJB8J7kLmpk1aGGXcymwo269P9s2nhuBB+vWeyRtlvSYpE83S+Yup5mNFsfV5VwoaXPd+tpslurjJul6oBf4aN3mMyNip6SzgUckPRsRL453DBc0MxvlOCd43BsRvRN8vxM4o259WbZtFElXAF8FPhoRv3iQNiJ2Zv/dLulH1N5dMm5Bc5fTzBoMZ620ZksOm4AVkpZLmgFcC4warZR0MXAXsCoidtdtXyBpZvZ5IbVJZSd8q5xbaGY2SisneIyIQUk3AQ8BXcC6iNgi6XZgc0SsB74GzAX+qjazP69ExCrgfcBdkoapNb7uaPbe33IL2tAQejt9SpiePT3JsYeWzU2OBZjz5BvpwT0zC+U+Mj99sr2ui9KnPQIoOs/fws37k2OHe4r9es5/dl9y7I5PpU9zBfDev3wlOfbo2em546UWTB+EGBxuXectm5Z/w5htt9Z9vmKcuH8CfuV4crmFZmYN/OiTmVVDeD40M6sIvyTFzCplyha07FmtzcDOiLi6+CmZ2WQKxFALBwXK1IoW2peArcBJLTiWmbWBTh0UKFSGJS0DPgV8qzWnY2aTLaKlN9aWqmgL7U+BrwDzxttB0hpgDUBPV7F7wcysHNGGxSqP5BaapKuB3RHxxET7RcTaiOiNiN4Z02alpjOz0uRrnVWthXYpsErSJ4Ee4CRJfxER17fm1Mxssky5FlpE3BIRyyLiLGoPnD7iYmbW+SJgaFi5lnbj+9DMrEGnjnK2pKBFxI+AH7XiWGY2uYLO7XK6hWZmY7TnBf88XNDMrEHEZJ9BmnILWgRx9GhyuA4PJsfOeSJ9fiqAfVcsT4499SevFsp96uN70oP3v1UotwrO5da35ozmO43jnP/9UqHc+y4/Kzl28ROHC+UuMqfZvvelz/s3uKU1jyy5y2lmlVAb5Zy6z3KaWcW4y2lmleEup5lVQiAXNDOrjg7tcbqgmdkYAdGGjzXl4YJmZg3c5TSzyvAop5lVgp/lNLPqCKBDC1pn3g5sZidURL4lD0krJW2T1Cfp5mN8/2VJz0t6RtLDks6s++4GSS9kyw3NcrmgmdkYIobzLU2PVHvN5Z3AJ4DzgesknT9mt6eA3oh4P3A/8MdZ7CnAbcCvAZcAt0laMFE+FzQzaxQ5l+YuAfoiYntEHAXuA1aPShXxaEQczFYfA5Zlnz8ObIyI/RHxBrARWDlRMl9DM7PR4rgGBRZK2ly3vjYi1tatLwV21K33U2txjedG4MEJYpdOdDKlFrShuTN559+fkxz/1vL0013009nJsQCnPLk/PbjgGPj+3oXJsT1vTthCb2qwp1gj/py1O5rvNI5YUOzd1af87dbk2IMfPrdQ7jnPv5Ycu6RvKDn2pTfTp+caJf+v7N6I6G1FSknXA73AR1OP4S6nmR2Dci5N7QTqJ8Vblm0bnU26AvgqsCoijhxPbD0XNDNrNJxzaW4TsELSckkzqL0hbn39DpIuBu6iVsx21331EHCVpAXZYMBV2bZx+RqamY3WwvvQImJQ0k3UClEXsC4itki6HdgcEeuBrwFzgb+SBPBKRKyKiP2S/pBaUQS4PSImvPbjgmZmDVr56FNEbAA2jNl2a93nKyaIXQesy5urUJdT0smS7pf0M0lbJX24yPHMrE207raNUhVtof0Z8IOIuCbrHxcbSjSz9tChjz4lFzRJ84HLgM8BZDfNtWjM2Mwmk9qw9ZVHkS7ncmAP8B1JT0n6lqQ5Y3eStEbSZkmbB4+8WyCdmZUiBMM5lzZTpKB1Ax8AvhERFwPvAg0PnkbE2ojojYje7pkN9c7M2lGHXkMrUtD6gf6IeDxbv59agTOzTjfVClpEvAbskHRetuly4PmWnJWZTa4OLWhFRzm/CNyTjXBuB/5L8VMys0nVwRM8FipoEfE0tYdJzaxCOnWU008KmFkjFzQzqwq30HKILnH45K7k+APL8z3efyxL/uFI850mMpSee2jR/EKpZ+8ZTI6d9cKeQrk5XPDPbcb09NjB9HnBAKLA31nRf9CHz3lPcmz3gfT70+ONFv2TnorX0Mysgtp0BDMPFzQza+SCZmZVofTe+qRyQTOzRm6hmVkVKDzKaWZV4lFOM6sMt9DMrCrc5TSzagiPcppZlbiFZmaV4YJmZlXRqdfQCr2X08ysnbigmVmjFk7BLWmlpG2S+iQ1vEhJ0mWSnpQ0KOmaMd8NSXo6W9Y3y+Uup5mN1sJRTkldwJ3AldRerLRJ0vqIqH//yCvU3u/7B8c4xKGIuChvvlIL2rSjwbwd6XM9DXfPSM99OH1OMQC60huz037+eqHUQ6ctT46NIvORAcOnzisU/9a5c5NjT37mzUK5D3/kvOY7jaPntWLvkD28eHZybNcb6blVcA65X2jdNbRLgL6I2A4g6T5gNXUvVIqIl7PvCpdRdznNbBTxb89zNluAhSMvEs+WNWMOtxTYUbfen23Lqyc77mOSPt1sZ3c5zaxR/hba3og4kS9KOjMidko6G3hE0rMR8eJ4O7uFZmaj5Wyd5by1YydwRt36smxbvlOJ2Jn9dzvwI+DiifYvVNAk/Z6kLZKek3SvpJ4ixzOzNjGcc2luE7BC0vLs/b3XAk1HKwEkLZA0M/u8ELiUJi8zTy5okpYCvwP0RsQFQFd2smbW4VrVQouIQeAm4CFgK/C9iNgi6XZJqwAkfVBSP/AZ4C5JW7Lw9wGbJf0UeBS4Y8zoaIOi19C6gVmSBoDZwKsFj2dm7aCFTwpExAZgw5htt9Z93kStKzo27p+AXzmeXMkttKxv+yfU7iHZBbwVET8cu5+kNSMjIAMDxYbCzawEeW+qbcPHo4p0ORdQu59kOXA6MEfS9WP3i4i1EdEbEb3Tp89JP1MzK00LBwVKVWRQ4ArgpYjYExEDwPeBj7TmtMxsUnVoC63INbRXgA9Jmg0cAi4HNrfkrMxsUnXqBI9FrqE9DtwPPAk8mx1rbYvOy8wmSwdfQys0yhkRtwG3tehczKwNKFs6kR99MrNGbdj6ysMFzcwatOMIZh6lFjQNB93vpE8fpEifPogj6XkB9lx2WnLsSS8Xm4Jnz4Xpf00DcxYWyr1g02uF4uc/8EJyrOYWu81nxnM/S44d+PUPFMod3QU6ba/vTY8dLDhN1ggXNDOrBL/GzswqxS00M6sKX0Mzs+pwQTOzqnALzcyqIcg7eWPbcUEzs1FGXpLSiVzQzKyRC5qZVYWiMyuaC5qZjdamM2nk4YJmZg18Dc3MKsOPPplZdbiFZmaV0KYvQMnDBc3MGrmgNRfTxMBJ6XOaDc5Kn2NqaGGxOclOejl9PrWZT20vlPuUU85Njj3p6d2Fckd3V6H4N/5j+rxip/z0jUK5+XdLk0NnvrSnUOqhX1qcHKsF89MTHyr29wWdfWNtkdfYmVlFaThyLbmOJa2UtE1Sn6Sbj/H9ZZKelDQo6Zox390g6YVsuaFZLnc5zWy0Ft6HJqkLuBO4EugHNklaHxHP1+32CvA54A/GxJ5C7SVMvdkZPZHFjtt0dwvNzBpoON+SwyVAX0Rsj4ijwH3A6vodIuLliHiGxkfiPw5sjIj9WRHbCKycKFnTgiZpnaTdkp6r23aKpI1ZM3CjpAW5fjQz6wz538u5UNLmumXNmCMtBXbUrfdn2/I47tg8LbTv0lgVbwYejogVwMPZuplVhCLfAuyNiN66ZVJfNt60oEXEj4H9YzavBu7OPt8NfLq1p2VmkyaAiHxLczuBM+rWl2XbTkhs6jW0xRGxK/v8GjDuGLWkNSPN0YGBdxPTmVmZWngNbROwQtJySTOAa4H1OU/jIeAqSQuyy1pXZdvGVXhQICImHBOJiLUjzdHp04u9Z9HMTryR+9BydjknFBGDwE3UCtFW4HsRsUXS7ZJWAUj6oKR+4DPAXZK2ZLH7gT+kVhQ3Abdn28aVetvG65KWRMQuSUuAYndvmln7yN+dzHm42ABsGLPt1rrPm6h1J48Vuw5YlzdXagttPTByk9sNwN8kHsfM2lCrWmhly3Pbxr3APwPnSeqXdCNwB3ClpBeAK7J1M6uK/LdttJWmXc6IuG6cry5v8bmYWZtox9ZXHn70ycxGC2CoMyuaC5qZNXALLYdp7xxi5k+2JMef3vee5NijZ56aHAvQ89Le5Nh4T7Hcu1anT1307uIlhXIv2HakUPzCv3spOfbw+9Kn/wGY+WRfcuzRC88ulHv2tvSB/4Fl6b8vsbdF/6T91iczqwq30MysGtp0BDMPFzQzG0WAPChgZlXhN6ebWTW4y2lm1dHaZznL5IJmZg08ymlm1eEWmplVQniU08yqpDPrmQuamTXybRtmVh0uaGZWCUHjK387hAuamY0iwl1OM6uQ4c5sopVb0GbOQGce8+UuuRxcviA5tqf/QHIswNGl6bmn7y6W+8w/70qO7dm+q/lOExhYcnKh+H0fOys59qTthwrlPvShc5NjZ/38zUK537ngtPTcuwq8v3a4BS0rdznNrErc5TSz6nBBM7Nq6NyH0/O8l3OdpN2Snqvb9jVJP5P0jKS/lnTyCT1LMyvPyFuf8iw5SFopaZukPkk3H+P7mZL+Mvv+cUlnZdvPknRI0tPZ8s1mufK8Of27wMox2zYCF0TE+4F/BW7JcRwz6xCKyLU0PY7UBdwJfAI4H7hO0vljdrsReCMizgH+J/A/6r57MSIuypbPN8vXtKBFxI+B/WO2/TAiBrPVx4D0oUszaz8R+ZbmLgH6ImJ7RBwF7gNWj9lnNXB39vl+4HJJSjntPC20Zn4LeLAFxzGzdhDUbv/IszS3FNhRt96fbTvmPllD6S1g5F1+yyU9JenvJf2HZskKDQpI+iowCNwzwT5rgDUAPd0nFUlnZqU4rkGBhZI2162vjYi1LTqRXcB7I2KfpF8F/q+kX46It8cLSC5okj4HXA1cHjH+T5/9cGsB5s9a0plDJ2ZTTf6Ctjcieif4fidwRt36smzbsfbpl9QNzAf2ZXXlSO104glJLwLnApsZR1KXU9JK4CvAqog4mHIMM2tTAQwN51ua2wSskLRc0gzgWmD9mH3WAzdkn68BHomIkLQoG1RA0tnACmD7RMmattAk3Qv8OrWmZT9wG7VRzZnAxuza3WN5RiDMrBMERGuefYqIQUk3AQ8BXcC6iNgi6XZgc0SsB74N/LmkPmoDkNdm4ZcBt0saoPYw1ucjYn9jln/TtKBFxHXH2Pzt3D+RmXWeFt5YGxEbgA1jtt1a9/kw8JljxD0APHA8ufykgJmNNjLK2YFc0MysUYc++lRuQZOI6ekph3qS7rWrpT58JDkWYHDO3OTYGQODzXeawPD09J97cFGxW2Wm7y825jNz/vTk2O43i+WOGem3WeqdYlMXDc4+tflO4ziyaHZybHS34tZSXNDMrCIiYGhoss8iiQuamTVyC83MKsMFzcyqIfdzmm3HBc3MRguIFt1YWzYXNDNrlO+xprbjgmZmo0X4NXZmViEeFDCzqgi30MysGjr3rU8uaGY2mh9ON7OqCCD86JOZVUK0boLHsrmgmVmDcJfTzCqjQ1tomuCFTa1PJu0Bfj7BLguBvSWdjnM7dxVznxkRi4ocQNIPqJ1jHnsjYmWRfK1UakFrRtLmJq/Ecm7ndm4bV4umtzQzm3wuaGZWGe1W0Fr1Cnnndm7nnoLa6hqamVkR7dZCMzNL5oJmZpXRFgVN0kpJ2yT1Sbq5xLxnSHpU0vOStkj6Ulm5686hS9JTkv625LwnS7pf0s8kbZX04RJz/1725/2cpHsl9ZzgfOsk7Zb0XN22UyRtlPRC9t8FJeb+Wvbn/oykv5Z08onIPRVNekGT1AXcCXwCOB+4TtL5JaUfBH4/Is4HPgR8ocTcI74EbC05J8CfAT+IiF8CLizrHCQtBX4H6I2IC4Au4NoTnPa7wNibP28GHo6IFcDD2XpZuTcCF0TE+4F/BW45QbmnnEkvaMAlQF9EbI+Io8B9wOoyEkfEroh4Mvt8gNo/6qVl5AaQtAz4FPCtsnJmeecDlwHfBoiIoxHxZomn0A3MktQNzAZePZHJIuLHwP4xm1cDd2ef7wY+XVbuiPhhRAxmq48By05E7qmoHQraUmBH3Xo/JRaVEZLOAi4GHi8x7Z8CXwHKfnBuObAH+E7W3f2WpDllJI6IncCfAK8Au4C3IuKHZeQeY3FE7Mo+vwYsnoRzAPgt4MFJyl057VDQJp2kucADwO9GxNsl5bwa2B0RT5SRb4xu4APANyLiYuBdTlyXa5TsWtVqakX1dGCOpOvLyD2eqN27VPr9S5K+Su2yxz1l566qdihoO4Ez6taXZdtKIWk6tWJ2T0R8v6y8wKXAKkkvU+tmf0zSX5SUux/oj4iR1uj91ApcGa4AXoqIPRExAHwf+EhJueu9LmkJQPbf3WUml/Q54GrgP4dvBm2Zdihom4AVkpZLmkHtAvH6MhJLErXrSFsj4utl5BwREbdExLKIOIvaz/xIRJTSUomI14Adks7LNl0OPF9GbmpdzQ9Jmp39+V/O5AyKrAduyD7fAPxNWYklraR2qWFVRBwsK+9UMOkFLbs4ehPwELVf7O9FxJaS0l8KfJZa6+jpbPlkSbkn2xeBeyQ9A1wE/PcykmatwvuBJ4Fnqf0OntBHgSTdC/wzcJ6kfkk3AncAV0p6gVqr8Y4Sc/8vYB6wMfud++aJyD0V+dEnM6uMSW+hmZm1iguamVWGC5qZVYYLmplVhguamVWGC5qZVYYLmplVxv8HyNEaWJo6pVsAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "resize qk with abs max\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "resize input + resize linear\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "resize qk by linear\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "print(att.attention_head_size)\n",
    "\n",
    "FACTOR = 2\n",
    "\n",
    "att.lowdim = False\n",
    "att.lowdim_factor = FACTOR\n",
    "att.update_lowdim()\n",
    "\n",
    "def downsize_linear(linear):\n",
    "    factor = FACTOR\n",
    "    s = linear.weight.shape\n",
    "    w = F.adaptive_avg_pool2d(linear.weight.view(1, 1, s[0], s[1]), output_size = (s[0] // factor, s[1])).view(s[0] // factor, s[1])\n",
    "    s = linear.bias.shape\n",
    "    b = F.adaptive_avg_pool1d(linear.bias.view(1,1,s[0]), (s[0] // factor,)).view(s[0]//factor)\n",
    "    l = nn.Linear(w.shape[1], w.shape[0])\n",
    "    l = l.to(w.device)\n",
    "    l.weight = nn.Parameter(w)\n",
    "    l.bias = nn.Parameter(b)\n",
    "    print(linear.weight.shape, 'to', l.weight.shape)\n",
    "    return l\n",
    "\n",
    "low_key = downsize_linear(att.key)\n",
    "low_query = downsize_linear(att.query)\n",
    "\n",
    "def _pool2(x):\n",
    "    a = torch.empty_like(x)\n",
    "    a = a[:a.shape[0]//2]\n",
    "    for i in range(len(a)):\n",
    "        if abs(x[i*2].item()) > abs(x[i*2+1]):\n",
    "            a[i] = x[i*2]\n",
    "        else:\n",
    "            a[i] = x[i*2+1]\n",
    "    return a\n",
    "\n",
    "def pool2(x):\n",
    "    s = x.shape\n",
    "    x = x.reshape(-1, s[-1])\n",
    "    a = torch.empty((x.shape[0], x.shape[1] // 2), device=x.device)\n",
    "    for i in range(x.shape[0]):\n",
    "        a[i,:] = _pool2(x[i])\n",
    "    a = a.view(*(s[:-1]+(s[-1]//2,)))\n",
    "    return a\n",
    "\n",
    "def resize(x, factor=FACTOR, method='absmax'):\n",
    "    if method == 'nearest':\n",
    "        return F.interpolate(x, (model.config.hidden_size // factor,))\n",
    "    elif method == 'avg':\n",
    "        return F.adaptive_avg_pool1d(x, output_size = (x.shape[-1] // FACTOR, ))\n",
    "    elif method == 'max':\n",
    "        return F.adaptive_max_pool1d(x, output_size = (x.shape[-1] // FACTOR, ))\n",
    "    elif method == 'absmax':\n",
    "        if factor == 2:\n",
    "            return pool2(x)\n",
    "        elif factor == 4:\n",
    "            return pool2(pool2(x))\n",
    "\n",
    "i = 0\n",
    "N = tokenized_result.input_ids.shape[1]\n",
    "C = 64\n",
    "\n",
    "a_k = att.key(hidden_states)[:,:,i*C:i*C+C]\n",
    "a_q = att.query(hidden_states)[:,:,i*C:i*C+C]\n",
    "\n",
    "b_k = lowdim_apply(att.key, resize(hidden_states), True, FACTOR, 'avg')[:,:,(64//FACTOR)*i:(64//FACTOR)*i+64//FACTOR]\n",
    "b_q = lowdim_apply(att.query, resize(hidden_states), True, FACTOR, 'avg')[:,:,(64//FACTOR)*i:(64//FACTOR)*i+64//FACTOR]\n",
    "\n",
    "d_k = low_key(hidden_states)[:,:,(64//FACTOR)*i:(64//FACTOR)*i+64//FACTOR]\n",
    "d_q = low_query(hidden_states)[:,:,(64//FACTOR)*i:(64//FACTOR)*i+64//FACTOR]\n",
    "\n",
    "a = torch.zeros((N,N), dtype=torch.float32)\n",
    "b = torch.zeros((N,N), dtype=torch.float32)\n",
    "c = torch.zeros((N,N), dtype=torch.float32)\n",
    "d = torch.zeros((N,N), dtype=torch.float32)\n",
    "for j in range(N):\n",
    "    for k in range(N):\n",
    "        a[k, j] = torch.sum(a_k[0,j] * a_q[0,k]).item() / math.sqrt(att.attention_head_size)\n",
    "        b[k, j] = torch.sum(b_k[0,j] * b_q[0,k]).item() / \\\n",
    "            math.sqrt(att.attention_head_size // FACTOR)\n",
    "        c[k, j] = torch.sum(resize(a_k[0,j].view(1,1,-1)) * resize(a_q[0,k].view(1,1,-1))).item() / \\\n",
    "            math.sqrt(att.attention_head_size // FACTOR)\n",
    "        d[k, j] = torch.sum(d_k[0,j] * d_q[0,k]).item() / \\\n",
    "            math.sqrt(att.attention_head_size // FACTOR)\n",
    "\n",
    "a = a + attention_mask.cpu()[0,i]\n",
    "a = torch.nn.functional.softmax(a, dim=-1)\n",
    "b = b + attention_mask.cpu()[0,i]\n",
    "b = torch.nn.functional.softmax(b, dim=-1)\n",
    "c = c + attention_mask.cpu()[0,i]\n",
    "c = torch.nn.functional.softmax(c, dim=-1)\n",
    "d = d + attention_mask.cpu()[0,i]\n",
    "d = torch.nn.functional.softmax(d, dim=-1)\n",
    "\n",
    "print('real')\n",
    "plt.imshow(real_model.encoder.layer[0].attention.self.forward(\n",
    "    hidden_states,\n",
    "    attention_mask = attention_mask,\n",
    "    output_attentions = True,\n",
    ")[1].detach().cpu().numpy()[0,i,:,:])    #original\n",
    "plt.colorbar()\n",
    "plt.show()\n",
    "\n",
    "print('original')\n",
    "plt.imshow(a.detach().numpy())  #original\n",
    "plt.colorbar()\n",
    "plt.show()\n",
    "\n",
    "print('resize qk with abs max')\n",
    "plt.imshow(c.detach().numpy())  #resize vector\\\n",
    "plt.colorbar()\n",
    "plt.show()\n",
    "\n",
    "print('resize input + resize linear')\n",
    "plt.imshow(b.detach().numpy())  #resize linear + vector\n",
    "plt.colorbar()\n",
    "plt.show()\n",
    "\n",
    "print('resize qk by linear')\n",
    "plt.imshow(d.detach().numpy())  #resize linear + vector\n",
    "plt.colorbar()\n",
    "plt.show()\n",
    "\n",
    "def plot_linear_resize(linear_method='avg', resize_method='avg'):\n",
    "    print('linear', linear_method, 'input', resize_method, 'factor', FACTOR)\n",
    "\n",
    "    b_k = lowdim_apply(att.key, resize(hidden_states, method=resize_method), True, FACTOR, linear_method)[:,:,(64//FACTOR)*i:(64//FACTOR)*i+64//FACTOR]\n",
    "    b_q = lowdim_apply(att.query, resize(hidden_states, method=resize_method), True, FACTOR, linear_method)[:,:,(64//FACTOR)*i:(64//FACTOR)*i+64//FACTOR]\n",
    "    b = torch.zeros((N,N), dtype=torch.float32)\n",
    "    for j in range(N):\n",
    "        for k in range(N):\n",
    "            b[k, j] = torch.sum(b_k[0,j] * b_q[0,k]).item() / \\\n",
    "                math.sqrt(att.attention_head_size // FACTOR)\n",
    "    b = b + attention_mask.cpu()[0,i]\n",
    "    b = torch.nn.functional.softmax(b, dim=-1)\n",
    "\n",
    "    print('resize input + resize linear')\n",
    "    plt.imshow(b.detach().numpy())  #resize linear + vector\n",
    "    plt.colorbar()\n",
    "    plt.show()\n",
    "# plot_linear_resize('avg', 'avg')\n",
    "# plot_linear_resize('avg', 'max')\n",
    "# plot_linear_resize('avg', 'nearest')\n",
    "# plot_linear_resize('max', 'avg')\n",
    "# plot_linear_resize('max', 'max')\n",
    "# plot_linear_resize('max', 'nearest')\n",
    "# plot_linear_resize('nearest', 'avg')\n",
    "# plot_linear_resize('nearest', 'max')\n",
    "# plot_linear_resize('nearest', 'nearest')\n",
    "\n",
    "def plot_qk_resize(resize_method = 'avg'):\n",
    "    print('resize attention input qk. method:', resize_method)\n",
    "    a_k = att.key(hidden_states)[:,:,i*C:i*C+C]\n",
    "    a_q = att.query(hidden_states)[:,:,i*C:i*C+C]\n",
    "    c = torch.zeros((N,N), dtype=torch.float32)\n",
    "    for j in range(N):\n",
    "        for k in range(N):\n",
    "            c[k, j] = torch.sum(resize(a_k[0,j].view(1,1,-1), method=resize_method) * resize(a_q[0,k].view(1,1,-1), method=resize_method)).item() / \\\n",
    "                math.sqrt(att.attention_head_size // FACTOR)\n",
    "    c = c + attention_mask.cpu()[0,i]\n",
    "    c = torch.nn.functional.softmax(c, dim=-1)\n",
    "    print('resize qk with abs max')\n",
    "    plt.imshow(c.detach().numpy())  #resize vector\\\n",
    "    plt.colorbar()\n",
    "    plt.show()\n",
    "\n",
    "# plot_qk_resize('avg')\n",
    "# plot_qk_resize('max')\n",
    "# plot_qk_resize('nearest')\n",
    "\n",
    "# plot_linear_resize('avg', 'absmax')\n",
    "# plot_linear_resize('max', 'absmax')\n",
    "# plot_linear_resize('nearest', 'absmax')\n",
    "\n",
    "# # plot_linear_resize('absmax', 'absmax')\n",
    "# # plot_linear_resize('absmax', 'avg')\n",
    "# # plot_linear_resize('absmax', 'max')\n",
    "# # plot_linear_resize('absmax', 'nearest')\n",
    "\n",
    "# plot_qk_resize('absmax')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "att.value.weight.shape\n",
    "import time"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SelfAttention.forward: hidden_states, attention_mask backuped\n",
      "SelfAttention.forward: expected\n",
      "SelfAttention.forward: attention_probs.shape torch.Size([3, 12, 14, 14])\n",
      "SelfAttention.forward: last_attention_probs is cloned from attention_probs\n",
      "1.029902696609497\n"
     ]
    }
   ],
   "source": [
    "from torch.nn import functional as F\n",
    "t = time.time()\n",
    "att.lowdim = True\n",
    "att.lowdim_linear = False\n",
    "att.lowdim_qk = False\n",
    "att.lowdim_qk_linear = True\n",
    "att.lowdim_factor = 2\n",
    "att.lowdim_mode = 'nearest_'\n",
    "att.lowdim_qk_linear_mode = 'absmax'\n",
    "att.update_lowdim()\n",
    "lowdim_result = att.forward(\n",
    "    hidden_states = hidden_states,\n",
    "    attention_mask = attention_mask,\n",
    "    output_attentions = True,\n",
    ")\n",
    "att.lowdim = False\n",
    "print(time.time() - t)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "text/plain": [
       "(torch.Size([3, 12, 14, 14]), torch.Size([3, 12, 14, 14]))"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "plt.imshow(normal_result[1][0,0].cpu().detach().numpy())\n",
    "plt.colorbar()\n",
    "plt.show()\n",
    "plt.imshow(lowdim_result[1][0,0].cpu().detach().numpy())\n",
    "plt.colorbar()\n",
    "plt.show()\n",
    "# plt.imshow(real_result[1][0,0].cpu().detach().numpy())\n",
    "# plt.colorbar()\n",
    "# plt.show()\n",
    "plt.imshow(normal_result[1][0,3].cpu().detach().numpy())\n",
    "plt.colorbar()\n",
    "plt.show()\n",
    "plt.imshow(lowdim_result[1][0,3].cpu().detach().numpy())\n",
    "plt.colorbar()\n",
    "plt.show()\n",
    "normal_result[1].shape, lowdim_result[1].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "12"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.config.num_attention_heads"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([3, 14, 768])"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "normal_result[0].shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/20 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([8, 512])\n",
      "8 False 64\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 20/20 [00:04<00:00,  4.32it/s]\n",
      "100%|██████████| 500/500 [00:31<00:00, 15.68it/s]\n",
      "100%|██████████| 20/20 [00:00<00:00, 78.05it/s]\n",
      " 29%|██▉       | 147/500 [00:09<00:21, 16.16it/s]\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "\u001b[1;32m<ipython-input-12-5d177b213268>\u001b[0m in \u001b[0;36m<module>\u001b[1;34m\u001b[0m\n\u001b[0;32m     76\u001b[0m \u001b[1;31m# exam(4, False)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     77\u001b[0m \u001b[1;31m# exam(4, True)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m---> 78\u001b[1;33m \u001b[0mexam\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;36m8\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;32mFalse\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m     79\u001b[0m \u001b[0mexam\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;36m8\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;32mTrue\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     80\u001b[0m \u001b[0mexam\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;36m16\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;32mFalse\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m<ipython-input-12-5d177b213268>\u001b[0m in \u001b[0;36mexam\u001b[1;34m(factor, using_amp)\u001b[0m\n\u001b[0;32m     64\u001b[0m         \u001b[0mnon\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mrun\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mmodel\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mtokens\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mt\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mfactor\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mfactor\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mlowdim\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;32mFalse\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0musing_amp\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0musing_amp\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     65\u001b[0m         \u001b[0mlow\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mrun\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mmodel\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mtokens\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mt\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mfactor\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mfactor\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mlowdim\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;32mTrue\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0musing_amp\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0musing_amp\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mn\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;36m20\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m---> 66\u001b[1;33m         \u001b[0mlow\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mrun\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mmodel\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mtokens\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mt\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mfactor\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mfactor\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mlowdim\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;32mTrue\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0musing_amp\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0musing_amp\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m     67\u001b[0m         \u001b[0mspeedups\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mappend\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mnon\u001b[0m \u001b[1;33m/\u001b[0m \u001b[0mlow\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     68\u001b[0m     \u001b[0mprint\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;34m'exam: factor'\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mfactor\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;34m'amp'\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0musing_amp\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m<ipython-input-12-5d177b213268>\u001b[0m in \u001b[0;36mrun\u001b[1;34m(model, tokens, factor, lowdim, using_amp, n)\u001b[0m\n\u001b[0;32m     45\u001b[0m     \u001b[1;32mfor\u001b[0m \u001b[0mk\u001b[0m \u001b[1;32min\u001b[0m \u001b[0mtqdm\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mtqdm\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mrange\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mn\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     46\u001b[0m         \u001b[1;32mwith\u001b[0m \u001b[0mtorch\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mno_grad\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mautocast\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0menabled\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0musing_amp\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m---> 47\u001b[1;33m             \u001b[0mmodel\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m     48\u001b[0m             \u001b[1;31m# for i in range(len(model.encoder.layer)):\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     49\u001b[0m             \u001b[1;31m#     model.encoder.layer[i].attention.self.forward(\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\nn\\modules\\module.py\u001b[0m in \u001b[0;36m_call_impl\u001b[1;34m(self, *input, **kwargs)\u001b[0m\n\u001b[0;32m   1100\u001b[0m         if not (self._backward_hooks or self._forward_hooks or self._forward_pre_hooks or _global_backward_hooks\n\u001b[0;32m   1101\u001b[0m                 or _global_forward_hooks or _global_forward_pre_hooks):\n\u001b[1;32m-> 1102\u001b[1;33m             \u001b[1;32mreturn\u001b[0m \u001b[0mforward_call\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m   1103\u001b[0m         \u001b[1;31m# Do not call functions when jit is used\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   1104\u001b[0m         \u001b[0mfull_backward_hooks\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mnon_full_backward_hooks\u001b[0m \u001b[1;33m=\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m<ipython-input-1-11bafc6e467b>\u001b[0m in \u001b[0;36mforward\u001b[1;34m(self, input_ids, attention_mask, token_type_ids, position_ids, head_mask, inputs_embeds, encoder_hidden_states, encoder_attention_mask, past_key_values, use_cache, output_attentions, output_hidden_states, return_dict)\u001b[0m\n\u001b[0;32m    704\u001b[0m             \u001b[0mpast_key_values_length\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mpast_key_values_length\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    705\u001b[0m         )\n\u001b[1;32m--> 706\u001b[1;33m         encoder_outputs = self.encoder(\n\u001b[0m\u001b[0;32m    707\u001b[0m             \u001b[0membedding_output\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    708\u001b[0m             \u001b[0mattention_mask\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mextended_attention_mask\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\nn\\modules\\module.py\u001b[0m in \u001b[0;36m_call_impl\u001b[1;34m(self, *input, **kwargs)\u001b[0m\n\u001b[0;32m   1100\u001b[0m         if not (self._backward_hooks or self._forward_hooks or self._forward_pre_hooks or _global_backward_hooks\n\u001b[0;32m   1101\u001b[0m                 or _global_forward_hooks or _global_forward_pre_hooks):\n\u001b[1;32m-> 1102\u001b[1;33m             \u001b[1;32mreturn\u001b[0m \u001b[0mforward_call\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m   1103\u001b[0m         \u001b[1;31m# Do not call functions when jit is used\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   1104\u001b[0m         \u001b[0mfull_backward_hooks\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mnon_full_backward_hooks\u001b[0m \u001b[1;33m=\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m<ipython-input-1-11bafc6e467b>\u001b[0m in \u001b[0;36mforward\u001b[1;34m(self, hidden_states, attention_mask, head_mask, encoder_hidden_states, encoder_attention_mask, past_key_values, use_cache, output_attentions, output_hidden_states, return_dict)\u001b[0m\n\u001b[0;32m    492\u001b[0m                 \u001b[1;32mraise\u001b[0m \u001b[0mException\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;34m'removed'\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    493\u001b[0m             \u001b[1;32melse\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 494\u001b[1;33m                 layer_outputs = layer_module(\n\u001b[0m\u001b[0;32m    495\u001b[0m                     \u001b[0mhidden_states\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    496\u001b[0m                     \u001b[0mattention_mask\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\nn\\modules\\module.py\u001b[0m in \u001b[0;36m_call_impl\u001b[1;34m(self, *input, **kwargs)\u001b[0m\n\u001b[0;32m   1100\u001b[0m         if not (self._backward_hooks or self._forward_hooks or self._forward_pre_hooks or _global_backward_hooks\n\u001b[0;32m   1101\u001b[0m                 or _global_forward_hooks or _global_forward_pre_hooks):\n\u001b[1;32m-> 1102\u001b[1;33m             \u001b[1;32mreturn\u001b[0m \u001b[0mforward_call\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m   1103\u001b[0m         \u001b[1;31m# Do not call functions when jit is used\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   1104\u001b[0m         \u001b[0mfull_backward_hooks\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mnon_full_backward_hooks\u001b[0m \u001b[1;33m=\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m<ipython-input-1-11bafc6e467b>\u001b[0m in \u001b[0;36mforward\u001b[1;34m(self, hidden_states, attention_mask, head_mask, encoder_hidden_states, encoder_attention_mask, past_key_value, output_attentions)\u001b[0m\n\u001b[0;32m    444\u001b[0m             \u001b[0moutputs\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mself_attention_outputs\u001b[0m\u001b[1;33m[\u001b[0m\u001b[1;36m1\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m]\u001b[0m  \u001b[1;31m# add self attentions if we output attention weights\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    445\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 446\u001b[1;33m         layer_output = apply_chunking_to_forward(\n\u001b[0m\u001b[0;32m    447\u001b[0m             \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mfeed_forward_chunk\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mchunk_size_feed_forward\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mseq_len_dim\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mattention_output\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    448\u001b[0m         )\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\transformers\\modeling_utils.py\u001b[0m in \u001b[0;36mapply_chunking_to_forward\u001b[1;34m(forward_fn, chunk_size, chunk_dim, *input_tensors)\u001b[0m\n\u001b[0;32m   2368\u001b[0m         \u001b[1;32mreturn\u001b[0m \u001b[0mtorch\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mcat\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0moutput_chunks\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mdim\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mchunk_dim\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   2369\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m-> 2370\u001b[1;33m     \u001b[1;32mreturn\u001b[0m \u001b[0mforward_fn\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0minput_tensors\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m",
      "\u001b[1;32m<ipython-input-1-11bafc6e467b>\u001b[0m in \u001b[0;36mfeed_forward_chunk\u001b[1;34m(self, attention_output)\u001b[0m\n\u001b[0;32m    452\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    453\u001b[0m     \u001b[1;32mdef\u001b[0m \u001b[0mfeed_forward_chunk\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mattention_output\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 454\u001b[1;33m         \u001b[0mintermediate_output\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mintermediate\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mattention_output\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m    455\u001b[0m         \u001b[0mlayer_output\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0moutput\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mintermediate_output\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mattention_output\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    456\u001b[0m         \u001b[1;32mreturn\u001b[0m \u001b[0mlayer_output\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\nn\\modules\\module.py\u001b[0m in \u001b[0;36m_call_impl\u001b[1;34m(self, *input, **kwargs)\u001b[0m\n\u001b[0;32m   1100\u001b[0m         if not (self._backward_hooks or self._forward_hooks or self._forward_pre_hooks or _global_backward_hooks\n\u001b[0;32m   1101\u001b[0m                 or _global_forward_hooks or _global_forward_pre_hooks):\n\u001b[1;32m-> 1102\u001b[1;33m             \u001b[1;32mreturn\u001b[0m \u001b[0mforward_call\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m   1103\u001b[0m         \u001b[1;31m# Do not call functions when jit is used\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   1104\u001b[0m         \u001b[0mfull_backward_hooks\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mnon_full_backward_hooks\u001b[0m \u001b[1;33m=\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m<ipython-input-1-11bafc6e467b>\u001b[0m in \u001b[0;36mforward\u001b[1;34m(self, hidden_states)\u001b[0m\n\u001b[0;32m    381\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    382\u001b[0m     \u001b[1;32mdef\u001b[0m \u001b[0mforward\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mhidden_states\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 383\u001b[1;33m         \u001b[0mhidden_states\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mdense\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mhidden_states\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m    384\u001b[0m         \u001b[0mhidden_states\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mintermediate_act_fn\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mhidden_states\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    385\u001b[0m         \u001b[1;32mreturn\u001b[0m \u001b[0mhidden_states\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\nn\\modules\\module.py\u001b[0m in \u001b[0;36m_call_impl\u001b[1;34m(self, *input, **kwargs)\u001b[0m\n\u001b[0;32m   1100\u001b[0m         if not (self._backward_hooks or self._forward_hooks or self._forward_pre_hooks or _global_backward_hooks\n\u001b[0;32m   1101\u001b[0m                 or _global_forward_hooks or _global_forward_pre_hooks):\n\u001b[1;32m-> 1102\u001b[1;33m             \u001b[1;32mreturn\u001b[0m \u001b[0mforward_call\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m   1103\u001b[0m         \u001b[1;31m# Do not call functions when jit is used\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   1104\u001b[0m         \u001b[0mfull_backward_hooks\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mnon_full_backward_hooks\u001b[0m \u001b[1;33m=\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m[\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\nn\\modules\\linear.py\u001b[0m in \u001b[0;36mforward\u001b[1;34m(self, input)\u001b[0m\n\u001b[0;32m    101\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    102\u001b[0m     \u001b[1;32mdef\u001b[0m \u001b[0mforward\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0minput\u001b[0m\u001b[1;33m:\u001b[0m \u001b[0mTensor\u001b[0m\u001b[1;33m)\u001b[0m \u001b[1;33m->\u001b[0m \u001b[0mTensor\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 103\u001b[1;33m         \u001b[1;32mreturn\u001b[0m \u001b[0mF\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mlinear\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mweight\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mbias\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m    104\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    105\u001b[0m     \u001b[1;32mdef\u001b[0m \u001b[0mextra_repr\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m)\u001b[0m \u001b[1;33m->\u001b[0m \u001b[0mstr\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\nn\\functional.py\u001b[0m in \u001b[0;36mlinear\u001b[1;34m(input, weight, bias)\u001b[0m\n\u001b[0;32m   1846\u001b[0m     \u001b[1;32mif\u001b[0m \u001b[0mhas_torch_function_variadic\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mweight\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mbias\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   1847\u001b[0m         \u001b[1;32mreturn\u001b[0m \u001b[0mhandle_torch_function\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mlinear\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m(\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mweight\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mbias\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mweight\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mbias\u001b[0m\u001b[1;33m=\u001b[0m\u001b[0mbias\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m-> 1848\u001b[1;33m     \u001b[1;32mreturn\u001b[0m \u001b[0mtorch\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_C\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_nn\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mlinear\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0minput\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mweight\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mbias\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m   1849\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m   1850\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "import tqdm\n",
    "def set_print(model, v):\n",
    "    for layer in model.encoder.layer:\n",
    "        layer.attention.self.print = v\n",
    "def set_lowdim(model, v, factor=2, qk_linear_mode='avg'):\n",
    "    for layer in model.encoder.layer:\n",
    "        if v != layer.attention.self.lowdim:\n",
    "            if v:\n",
    "                layer.attention.self.lowdim = True\n",
    "                layer.attention.self.lowdim_linear = False\n",
    "                layer.attention.self.lowdim_qk = False\n",
    "                layer.attention.self.lowdim_qk_linear = True\n",
    "                layer.attention.self.lowdim_factor = factor\n",
    "                layer.attention.self.lowdim_qk_linear_mode = qk_linear_mode\n",
    "                layer.attention.self.update_lowdim()\n",
    "            else:\n",
    "                layer.attention.self.lowdim = False\n",
    "BATCH_SIZE=8\n",
    "text = [\n",
    "    (\"[CLS]Bert Model with two heads on top as done during the pretraining: \"+\n",
    "    \"a masked language modeling head and a next sentence prediction (classification) head.\"+\n",
    "    \"This model inherits from PreTrainedModel. Check the superclass documentation for the \"+\n",
    "    \"generic methods the library implements for all its model (such as downloading or saving,\"+\n",
    "    \" resizing the input embeddings, pruning heads etc.)\"+\n",
    "    \"This model is also a PyTorch torch.nn.Module subclass. Use it as a regular PyTorch Module\"+\n",
    "    \" and refer to the PyTorch documentation for all matter related to general usage and behavior.\") * 4]*BATCH_SIZE\n",
    "\n",
    "data = tokenizer(text, padding=True, truncation=True, return_tensors='pt')\n",
    "data = data.input_ids.cuda()\n",
    "print(data.shape)\n",
    "\n",
    "set_print(model, False)\n",
    "set_lowdim(model, False)\n",
    "model(data)\n",
    "hidden = model.encoder.layer[0].attention.self.last_hidden_states\n",
    "mask = model.encoder.layer[0].attention.self.last_attention_mask\n",
    "\n",
    "N = 500\n",
    "def run(model, tokens=512, factor=2, lowdim=False, using_amp=False, n=N):\n",
    "    input = data[:, :tokens]\n",
    "    set_lowdim(model, lowdim, factor=factor)\n",
    "    from torch.cuda.amp import autocast\n",
    "    model.eval()\n",
    "    t = time.time()\n",
    "    for k in tqdm.tqdm(range(n)):\n",
    "        with torch.no_grad(), autocast(enabled=using_amp):\n",
    "            model(input)\n",
    "            # for i in range(len(model.encoder.layer)):\n",
    "            #     model.encoder.layer[i].attention.self.forward(\n",
    "            #         hidden_states = hidden,\n",
    "            #         attention_mask = mask,\n",
    "            #         output_attentions = True,\n",
    "            #     )\n",
    "    took = time.time() - t\n",
    "    set_lowdim(model, False, factor=factor)\n",
    "    return took\n",
    "\n",
    "def exam(factor=2, using_amp=False):\n",
    "    tokens = [64, 128, 256, 512]\n",
    "    speedups = []\n",
    "    for t in tokens:\n",
    "        print(factor, using_amp, t)\n",
    "        non = run(model, tokens=t, factor=factor, lowdim=False, using_amp=using_amp, n=20)\n",
    "        non = run(model, tokens=t, factor=factor, lowdim=False, using_amp=using_amp)\n",
    "        low = run(model, tokens=t, factor=factor, lowdim=True, using_amp=using_amp, n=20)\n",
    "        low = run(model, tokens=t, factor=factor, lowdim=True, using_amp=using_amp)\n",
    "        speedups.append(non / low)\n",
    "    print('exam: factor', factor, 'amp', using_amp)\n",
    "    plt.plot(tokens, speedups)\n",
    "    #plt.axis('equal')\n",
    "    plt.grid()\n",
    "    plt.show()\n",
    "\n",
    "# exam(2, False)\n",
    "# exam(2, True)\n",
    "# exam(4, False)\n",
    "# exam(4, True)\n",
    "exam(8, False)\n",
    "exam(8, True)\n",
    "exam(16, False)\n",
    "exam(16, True)\n",
    "exam(32, False)\n",
    "exam(32, True)\n",
    "print('done')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 120000/120000 [00:00<00:00, 213523.84it/s]\n",
      "100%|██████████| 7600/7600 [00:00<00:00, 58016.52it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Classification Dataset Stat.: name:AG_NEWS, nclass:5, max_len:1012, avg_len:236.477525, count:120000\n",
      "Classification Dataset Stat.: name:AG_NEWS, nclass:5, max_len:892, avg_len:235.2992105263158, count:7600\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trainer.__init__: Model initialized. model = bert\n",
      "Trainer.load: Loading...\n"
     ]
    }
   ],
   "source": [
    "from trainer.classification import Trainer\n",
    "trainer = Trainer()\n",
    "trainer.load()\n",
    "bert = trainer.model.bert\n",
    "fc = trainer.model.classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(tensor([1, 2, 4, 1, 3, 4, 3, 3, 3, 2, 1, 1, 1, 4, 4, 2, 3, 3, 3, 4, 1, 3, 2, 3,\n",
      "        2, 3, 4, 4, 4, 2, 1, 4, 1, 4, 4, 2, 2, 1, 1, 2, 4, 1, 4, 3, 4, 2, 2, 3,\n",
      "        4, 4, 2, 1, 1, 2, 2, 3, 4, 2, 3, 4, 3, 4, 3, 1], device='cuda:0'), tensor([1, 2, 4, 1, 3, 4, 3, 3, 3, 2, 1, 1, 1, 4, 4, 2, 3, 3, 3, 4, 1, 3, 2, 3,\n",
      "        2, 3, 4, 1, 4, 2, 1, 4, 1, 4, 4, 2, 2, 1, 1, 2, 4, 1, 4, 3, 4, 2, 2, 3,\n",
      "        4, 4, 2, 1, 1, 2, 2, 3, 4, 2, 3, 4, 3, 4, 3, 1], device='cuda:0'))\n",
      "0.3320026397705078\n",
      "(tensor([1, 2, 4, 1, 3, 4, 3, 3, 3, 2, 1, 1, 1, 4, 4, 2, 3, 3, 3, 4, 1, 3, 2, 3,\n",
      "        2, 3, 4, 4, 4, 2, 1, 4, 1, 4, 4, 2, 2, 1, 1, 2, 4, 1, 4, 3, 4, 2, 2, 3,\n",
      "        4, 4, 2, 1, 1, 2, 2, 3, 4, 2, 3, 4, 3, 4, 3, 1], device='cuda:0'), tensor([1, 2, 4, 1, 3, 4, 3, 3, 3, 2, 1, 1, 1, 4, 4, 2, 3, 3, 3, 4, 1, 3, 2, 3,\n",
      "        2, 3, 4, 1, 4, 2, 1, 4, 1, 4, 4, 2, 2, 1, 1, 2, 4, 1, 4, 3, 4, 2, 2, 3,\n",
      "        4, 4, 2, 1, 1, 2, 2, 3, 4, 2, 3, 4, 3, 4, 3, 1], device='cuda:0'))\n",
      "0.03699970245361328\n",
      "(tensor([1, 2, 4, 1, 3, 4, 3, 3, 3, 2, 1, 2, 1, 4, 4, 2, 3, 3, 3, 4, 1, 3, 2, 3,\n",
      "        2, 3, 4, 4, 4, 2, 1, 4, 1, 4, 3, 2, 2, 2, 1, 2, 4, 1, 4, 3, 4, 2, 2, 3,\n",
      "        3, 4, 2, 1, 1, 2, 2, 3, 4, 2, 3, 4, 3, 4, 3, 1], device='cuda:0'), tensor([1, 2, 4, 1, 3, 4, 3, 3, 3, 2, 1, 1, 1, 4, 4, 2, 3, 3, 3, 4, 1, 3, 2, 3,\n",
      "        2, 3, 4, 1, 4, 2, 1, 4, 1, 4, 4, 2, 2, 1, 1, 2, 4, 1, 4, 3, 4, 2, 2, 3,\n",
      "        4, 4, 2, 1, 1, 2, 2, 3, 4, 2, 3, 4, 3, 4, 3, 1], device='cuda:0'))\n",
      "0.03500223159790039\n"
     ]
    }
   ],
   "source": [
    "lowbert = BertModel.from_pretrained(trainer.model.bert_model_name)\n",
    "lowbert = lowbert.to(trainer.device)\n",
    "lowbert.eval()\n",
    "lowbert.load_state_dict(bert.state_dict())\n",
    "set_print(lowbert, False)\n",
    "\n",
    "batch = trainer.get_batch()\n",
    "\n",
    "def eval(model, batch=batch, fc=fc):\n",
    "    for _ in range(1):\n",
    "        lm_output = model(\n",
    "            input_ids = batch.input_ids, \n",
    "            attention_mask = batch.attention_masks, \n",
    "            output_hidden_states=True,\n",
    "            output_attentions=True,\n",
    "        )\n",
    "        last_hidden = lm_output.last_hidden_state[:,0,:]\n",
    "        x = fc(last_hidden)\n",
    "    return torch.argmax(x, dim=-1), batch.labels\n",
    "\n",
    "t = time.time()\n",
    "print(eval(bert))\n",
    "print(time.time()-t)\n",
    "t = time.time()\n",
    "print(eval(lowbert))\n",
    "print(time.time()-t)\n",
    "\n",
    "set_lowdim(lowbert, True, factor=16)\n",
    "t = time.time()\n",
    "print(eval(lowbert))\n",
    "print(time.time()-t)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [00:00<00:00, 21.29it/s]\n",
      "100%|██████████| 500/500 [00:09<00:00, 54.67it/s]\n",
      "100%|██████████| 10/10 [00:00<00:00, 60.24it/s]\n",
      "  0%|          | 0/500 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(0.93428125, 2.3367702960968018)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:08<00:00, 60.67it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(0.8609375, 2.321326971054077)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "def accuracy(model, N=500, fc=fc):\n",
    "    trainer.seed()\n",
    "    acc = 0\n",
    "    took = 0\n",
    "    for _ in tqdm.tqdm(range(N)):\n",
    "        batch = trainer.get_batch(test=True)\n",
    "        with torch.no_grad():\n",
    "            t = time.time()\n",
    "            output, label = eval(model, batch=batch, fc=fc)\n",
    "            took += time.time() - t\n",
    "            acc += torch.mean((output == label) * 1.0).item()\n",
    "    return acc/N, took\n",
    "\n",
    "accuracy(bert, N=10)\n",
    "print(accuracy(bert))\n",
    "\n",
    "set_lowdim(lowbert, True, factor=16)\n",
    "accuracy(lowbert, N=10)\n",
    "print(accuracy(lowbert))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[5625] loss:0.46360746026039124, acc:0.90625, test_loss:0.3110387921333313, test_acc:0.890625\n",
      "[15] loss:0.29329419136047363, acc:0.875, test_loss:0.2374582439661026, test_acc:0.9375\n",
      "[30] loss:0.2513085603713989, acc:0.921875, test_loss:0.1530284881591797, test_acc:0.96875\n",
      "[45] loss:0.3470487892627716, acc:0.90625, test_loss:0.32501739263534546, test_acc:0.890625\n",
      "[60] loss:0.07794637978076935, acc:0.96875, test_loss:0.27406245470046997, test_acc:0.875\n",
      "[75] loss:0.33097997307777405, acc:0.921875, test_loss:0.275108277797699, test_acc:0.9375\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "\u001b[1;32m<ipython-input-126-3372104c9da3>\u001b[0m in \u001b[0;36m<module>\u001b[1;34m\u001b[0m\n\u001b[0;32m      5\u001b[0m \u001b[1;32mwhile\u001b[0m \u001b[0msteps\u001b[0m \u001b[1;33m<\u001b[0m \u001b[1;36m2000\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m      6\u001b[0m     \u001b[0mbatch\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mtrainer\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mget_batch\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m----> 7\u001b[1;33m     \u001b[0mtrainer\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0moptimize_step\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mbatch\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m      8\u001b[0m     \u001b[1;32mif\u001b[0m \u001b[0msteps\u001b[0m \u001b[1;33m%\u001b[0m \u001b[1;36m15\u001b[0m \u001b[1;33m==\u001b[0m \u001b[1;36m0\u001b[0m\u001b[1;33m:\u001b[0m \u001b[0mtrainer\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mreport\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mbatch\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m      9\u001b[0m     \u001b[0msteps\u001b[0m \u001b[1;33m+=\u001b[0m \u001b[1;36m1\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32mf:\\Library\\discrete_edge_learning\\trainer\\classification.py\u001b[0m in \u001b[0;36moptimize_step\u001b[1;34m(self, batch)\u001b[0m\n\u001b[0;32m     55\u001b[0m             \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mloss\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0moutput\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mmodel\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mbatch\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mreturn_output\u001b[0m\u001b[1;33m=\u001b[0m\u001b[1;32mTrue\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     56\u001b[0m         \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mscaler\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mscale\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mloss\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mbackward\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m---> 57\u001b[1;33m         \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mscaler\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mstep\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0moptimizer\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m     58\u001b[0m         \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mscaler\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mupdate\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     59\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\cuda\\amp\\grad_scaler.py\u001b[0m in \u001b[0;36mstep\u001b[1;34m(self, optimizer, *args, **kwargs)\u001b[0m\n\u001b[0;32m    336\u001b[0m         \u001b[1;32massert\u001b[0m \u001b[0mlen\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0moptimizer_state\u001b[0m\u001b[1;33m[\u001b[0m\u001b[1;34m\"found_inf_per_device\"\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m)\u001b[0m \u001b[1;33m>\u001b[0m \u001b[1;36m0\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;34m\"No inf checks were recorded for this optimizer.\"\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    337\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 338\u001b[1;33m         \u001b[0mretval\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m_maybe_opt_step\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0moptimizer\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0moptimizer_state\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m*\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m    339\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    340\u001b[0m         \u001b[0moptimizer_state\u001b[0m\u001b[1;33m[\u001b[0m\u001b[1;34m\"stage\"\u001b[0m\u001b[1;33m]\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mOptState\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mSTEPPED\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\cuda\\amp\\grad_scaler.py\u001b[0m in \u001b[0;36m_maybe_opt_step\u001b[1;34m(self, optimizer, optimizer_state, *args, **kwargs)\u001b[0m\n\u001b[0;32m    283\u001b[0m         \u001b[0mretval\u001b[0m \u001b[1;33m=\u001b[0m \u001b[1;32mNone\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    284\u001b[0m         \u001b[1;32mif\u001b[0m \u001b[1;32mnot\u001b[0m \u001b[0msum\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mv\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mitem\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m \u001b[1;32mfor\u001b[0m \u001b[0mv\u001b[0m \u001b[1;32min\u001b[0m \u001b[0moptimizer_state\u001b[0m\u001b[1;33m[\u001b[0m\u001b[1;34m\"found_inf_per_device\"\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mvalues\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 285\u001b[1;33m             \u001b[0mretval\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0moptimizer\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mstep\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m    286\u001b[0m         \u001b[1;32mreturn\u001b[0m \u001b[0mretval\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    287\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\optim\\optimizer.py\u001b[0m in \u001b[0;36mwrapper\u001b[1;34m(*args, **kwargs)\u001b[0m\n\u001b[0;32m     86\u001b[0m                 \u001b[0mprofile_name\u001b[0m \u001b[1;33m=\u001b[0m \u001b[1;34m\"Optimizer.step#{}.step\"\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mformat\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mobj\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m__class__\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m__name__\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     87\u001b[0m                 \u001b[1;32mwith\u001b[0m \u001b[0mtorch\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mautograd\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mprofiler\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mrecord_function\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mprofile_name\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m---> 88\u001b[1;33m                     \u001b[1;32mreturn\u001b[0m \u001b[0mfunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m     89\u001b[0m             \u001b[1;32mreturn\u001b[0m \u001b[0mwrapper\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     90\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\autograd\\grad_mode.py\u001b[0m in \u001b[0;36mdecorate_context\u001b[1;34m(*args, **kwargs)\u001b[0m\n\u001b[0;32m     26\u001b[0m         \u001b[1;32mdef\u001b[0m \u001b[0mdecorate_context\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     27\u001b[0m             \u001b[1;32mwith\u001b[0m \u001b[0mself\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0m__class__\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m---> 28\u001b[1;33m                 \u001b[1;32mreturn\u001b[0m \u001b[0mfunc\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m*\u001b[0m\u001b[0margs\u001b[0m\u001b[1;33m,\u001b[0m \u001b[1;33m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m     29\u001b[0m         \u001b[1;32mreturn\u001b[0m \u001b[0mcast\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mF\u001b[0m\u001b[1;33m,\u001b[0m \u001b[0mdecorate_context\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     30\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\optim\\adam.py\u001b[0m in \u001b[0;36mstep\u001b[1;34m(self, closure)\u001b[0m\n\u001b[0;32m    131\u001b[0m                     \u001b[0mstate_steps\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0mappend\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mstate\u001b[0m\u001b[1;33m[\u001b[0m\u001b[1;34m'step'\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    132\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m--> 133\u001b[1;33m             F.adam(params_with_grad,\n\u001b[0m\u001b[0;32m    134\u001b[0m                    \u001b[0mgrads\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m    135\u001b[0m                    \u001b[0mexp_avgs\u001b[0m\u001b[1;33m,\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;32m~\\anaconda3\\envs\\tensorflow\\lib\\site-packages\\torch\\optim\\_functional.py\u001b[0m in \u001b[0;36madam\u001b[1;34m(params, grads, exp_avgs, exp_avg_sqs, max_exp_avg_sqs, state_steps, amsgrad, beta1, beta2, lr, weight_decay, eps)\u001b[0m\n\u001b[0;32m     92\u001b[0m             \u001b[0mdenom\u001b[0m \u001b[1;33m=\u001b[0m \u001b[1;33m(\u001b[0m\u001b[0mmax_exp_avg_sqs\u001b[0m\u001b[1;33m[\u001b[0m\u001b[0mi\u001b[0m\u001b[1;33m]\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0msqrt\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m \u001b[1;33m/\u001b[0m \u001b[0mmath\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0msqrt\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mbias_correction2\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0madd_\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0meps\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     93\u001b[0m         \u001b[1;32melse\u001b[0m\u001b[1;33m:\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[1;32m---> 94\u001b[1;33m             \u001b[0mdenom\u001b[0m \u001b[1;33m=\u001b[0m \u001b[1;33m(\u001b[0m\u001b[0mexp_avg_sq\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0msqrt\u001b[0m\u001b[1;33m(\u001b[0m\u001b[1;33m)\u001b[0m \u001b[1;33m/\u001b[0m \u001b[0mmath\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0msqrt\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0mbias_correction2\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m.\u001b[0m\u001b[0madd_\u001b[0m\u001b[1;33m(\u001b[0m\u001b[0meps\u001b[0m\u001b[1;33m)\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0m\u001b[0;32m     95\u001b[0m \u001b[1;33m\u001b[0m\u001b[0m\n\u001b[0;32m     96\u001b[0m         \u001b[0mstep_size\u001b[0m \u001b[1;33m=\u001b[0m \u001b[0mlr\u001b[0m \u001b[1;33m/\u001b[0m \u001b[0mbias_correction1\u001b[0m\u001b[1;33m\u001b[0m\u001b[1;33m\u001b[0m\u001b[0m\n",
      "\u001b[1;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "trainer.model.bert = lowbert\n",
    "trainer.model.train()\n",
    "trainer.optimizer = torch.optim.Adam(trainer.model.parameters(), lr=5e-5)\n",
    "steps = 0\n",
    "while steps < 2000:\n",
    "    batch = trainer.get_batch()\n",
    "    trainer.optimize_step(batch)\n",
    "    if steps % 15 == 0: trainer.report(batch)\n",
    "    steps += 1\n",
    "    trainer.steps = steps\n",
    "trainer.model.eval()\n",
    "acc, took = accuracy(lowbert)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 120000/120000 [00:00<00:00, 215436.01it/s]\n",
      "100%|██████████| 7600/7600 [00:00<00:00, 211066.30it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Classification Dataset Stat.: name:AG_NEWS, nclass:5, max_len:1012, avg_len:236.477525, count:120000\n",
      "Classification Dataset Stat.: name:AG_NEWS, nclass:5, max_len:892, avg_len:235.2992105263158, count:7600\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trainer.__init__: Model initialized. model = bert\n",
      "Trainer.load: Loading...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:09<00:00, 55.31it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 1000/1000 [00:50<00:00, 19.66it/s]\n",
      "100%|██████████| 500/500 [00:08<00:00, 58.70it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 1000/1000 [00:49<00:00, 20.04it/s]\n",
      "100%|██████████| 500/500 [00:08<00:00, 59.02it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 1000/1000 [00:50<00:00, 19.71it/s]\n",
      "100%|██████████| 500/500 [00:08<00:00, 59.84it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 1000/1000 [00:49<00:00, 20.05it/s]\n",
      "100%|██████████| 500/500 [00:08<00:00, 59.19it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 1000/1000 [00:51<00:00, 19.45it/s]\n",
      "100%|██████████| 500/500 [00:08<00:00, 57.07it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 1000/1000 [00:51<00:00, 19.41it/s]\n",
      "100%|██████████| 500/500 [00:08<00:00, 60.25it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 1000/1000 [00:50<00:00, 19.71it/s]\n",
      "100%|██████████| 500/500 [00:08<00:00, 59.57it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 1000/1000 [00:51<00:00, 19.51it/s]\n",
      "100%|██████████| 500/500 [00:08<00:00, 57.88it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYAAAAD4CAYAAADlwTGnAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8rg+JYAAAACXBIWXMAAAsTAAALEwEAmpwYAAAxoklEQVR4nO3dd3wVZdr/8c+VAqEEAkmoAQIYeqihg4CggqJUFQQWXVfWdVVcF11Yf8+uD6uP7rP6uGJHRVCaCKioqGsBKdKCSFV6SQClGVoIadfvjxlMiEAScsKc5Fzv12tenDNzz5zroOSbmXvmvkVVMcYYE3iCvC7AGGOMNywAjDEmQFkAGGNMgLIAMMaYAGUBYIwxASrE6wIKIyoqSmNjY70uwxhjSpS1a9ceUdXovOtLVADExsaSmJjodRnGGFOiiMjeC623S0DGGBOgLACMMSZAWQAYY0yAKlF9AMaYwJORkUFycjJpaWlel+L3wsLCiImJITQ0tEDtLQCMMX4tOTmZ8PBwYmNjERGvy/FbqsrRo0dJTk6mfv36BdrHLgEZY/xaWloakZGR9sM/HyJCZGRkoc6ULACMMX7PfvgXTGH/ngIiAFZ8MoM185/lTHqW16UYY4zfCIg+gLANb9MsNZFhG8Jpm9CFER3r0iC6otdlGWOMpwLiDKD1H98iqFwlJpV9mZnf7OCaZ75m5Our+HTTQTKzsr0uzxhjPBEQZwBSsRqhA1+gzuzhJHZbw5thv2Hmqn3cM/1balQKY3iHugzrUIfqlcK8LtUY46cGDhxIUlISaWlpjB07luzsbHbu3Mm//vUvAKZOnUpiYiIvvPAC//jHP5g+fTrR0dHUqVOHdu3aMW7cOI+/wa8FRAAA0OQGaDOKCmte4L47buCeHr1YtPUwb6/cy7NfbGPSV9u5vnl1RnasR+eGdseBMf7ovz/czJYDJ3x6zGa1KvH3m5rn227KlClUrVqVM2fO0L59e7788ku6du36SwC88847PProo6xZs4Z58+axfv16MjIyaNu2Le3atfNpzb4SOAEA0PdJ2L0E3vs9IX9YzrXNqnNts+rsOXKamav3MScxiYUbf6RhdAVGdKzHkHYxVC5XsAcqjDGl26RJk3jvvfcASEpKYvfu3TRo0ICVK1cSFxfHDz/8QNeuXXnuuecYMGAAYWFhhIWFcdNNN3lc+cUFVgCUDYfBk+HNfvDpBBjwAgCxURX46w1NeejaRny04SDTV+5l4kdb+N/PfmBAq9qM6lyPFrUre1y8MaYgv6kXh8WLF/PFF1+wYsUKypcvT8+ePUlLS2PYsGHMmTOHJk2aMGjQoBJ35SAgOoHPU7cTdH0Q1r0NPyw8b1NYaDBD28Xw/h+78tH93RjYujYfrN9P/+eXMeDF5cxdm0xaht1KakygOX78OFWqVKF8+fL88MMPrFy5EoBBgwbxwQcfMGvWLIYNGwZA165d+fDDD0lLS+PUqVN89NFHXpZ+SYEXAAA9J0CNeFhwP5w6fMEmLWpX5qkhLVn11z78/aZmnEzLYNy76+n05Jc88fEW9hw5fYWLNsZ4pW/fvmRmZtK0aVPGjx9Pp06dAKhSpQpNmzZl7969dOjQAYD27dtz880307JlS/r160d8fDyVK/vnFQRRVa9rKLCEhAT12YQwh76HV3vAVb1h2EzI59RNVVmx8yjTV+3ls80/IcD4fk24q1v9EnfaZ0xJ8v3339O0aVOvyyiUU6dOUbFiRVJTU7n66quZPHkybdu2vSKffaG/LxFZq6oJedsG5hkAQLWm0Ocx2LrQuRyUDxGhy1VRvDSiHd+Mv4ZrmlTj8Y+/557pazl+JqP46zXGlBhjxoyhdevWtG3bliFDhlyxH/6FFVidwHl1vAe2fQKfjIfYblC1QYF2q14pjFdHteONZbt56pMfuOn5Zbw0oq11FBtjAJg5c6bXJRRI4J4BAAQFwcCXISgE3rsHsgvewSsi/K57A975fSfSM7MZ/PI3zFi1l5J0Sc0YE9gCOwAAKsfAjc9A0ipY/u9C796uXlU+fqAbnRpE8uh7m/jTO99x+mym7+s0xhgfswAAiB8KzQfDov+Bg+sLvXtkxbJMvaM9f762EQvWH2DAi8vZ/tPJYijUGGN8xwIAnDuAbnwGKkTD/DGQcabQhwgKEu7vHcf0uzqSkprOzS8s5711ycVQrDHG+IYFwDnlqzpPBh/+Ab6ceNmH6XJVFB8/0J34mMr86Z31TJi/0R4eM6aUqlixZA8rX6AAEJG+IrJVRHaIyPgLbK8nIl+KyAYRWSwiMbnWfysi34nIZhG5J9c+i91jfucu1Xz3tS7TVX2gwxhY+RLsWnzZh6leKYyZv+vIPT0aMmv1Pga/9A17j9qDY8YY/5JvAIhIMPAi0A9oBgwXkWZ5mj0NvKWqLYGJwJPu+oNAZ1VtDXQExotIrVz7jVDV1u5yqGhfxUf6/DdExsH798KZlMs+TEhwEOP7NeGN0QnsTzlD/0nL+HTTQd/VaYy5ogYOHEi7du1o3rw5kydP/mX9n/70J5o3b07v3r05fNgZWWDSpEk0a9aMli1b/jJExGOPPcbo0aPp3r079erVY/78+TzyyCPEx8fTt29fMjKc54kmTpxI+/btadGiBWPGjEFVyczMpH379ixevBiACRMm8Oijjxb5O+X7JLCIdAYeU9Xr3fcTAFT1yVxtNgN9VTVJnMdij6tqpTzHiQTWAZ1U9YCILAbGqWqBH+316ZPAl7L/W3jjWqdjeMhrRT5c0rFU7pv5LeuTj/PbrvUZ368JZULs6psxBXHek62fjIcfN/r2A2rEQ7+n8m127Nix84aD/vrrr4mKimL69OmMGDGCiRMncujQIV544QVq1arF7t27KVu2LCkpKURERPDYY4/xxRdfsGjRIrZs2ULnzp2ZN28e/fr1Y9CgQYwePZqBAwf+8jkAo0aN4tZbb+Wmm25i8+bNDB06lOeff56HH36YVatWUaZMmV/V6esngWsDSbneJ7vrclsPDHZfDwLC3R/4iEgdEdngHuOfqnog135vupd//ksuMp6CiIwRkUQRSTyXrsWudlvo8RfYOAc2zSvy4epULc+cezpzR5dYpizfzbDJKziQUviOZmOMdyZNmkSrVq3o1KkTSUlJbN++naCgIG677TYARo4cybJlywBo2bIlI0aMYPr06YSE5Dxv269fP0JDQ4mPjycrK4u+ffsCEB8fz549ewBYtGgRHTt2JD4+nq+++orNmzcD0Lx5c0aNGkX//v2ZMmXKBX/4F5avngQeB7wgIncAS4D9QBaAqiYBLd1LP++LyFxV/Qnn8s9+EQkH5gGjgLfyHlhVJwOTwTkD8FG9+ev2EGz7DD56COp2hkq18t/nEsqGBPPYzc1JiK3C+HkbuXHSUp69rTU9G3vf9WFMiVGA39SLw8WGg87r3O+xH3/8MUuWLOHDDz/kiSeeYONG56ylbNmyAAQFBREaGvpL+6CgIDIzM0lLS+Pee+8lMTGROnXq8Nhjj533ORs3biQiIoJDh3xzxbwgZwD7gTq53se4636hqgdUdbCqtgEeddel5G0DbAK6u+/3u3+eBGYCHS7vKxST4BBn7oCsdKc/INs3cwf3b1mLBfd1pXqlMO6cuoZn/rOVrGx7etgYf3ax4aCzs7OZO3cu4Az/0K1bN7Kzs0lKSqJXr17885//5Pjx45w6dapAn3Puh31UVBSnTp365dgA8+fP59ixYyxZsoT777+flJSUIn+vggTAGiBOROqLSBlgGLAgdwMRiRKRc8eaAExx18eISDn3dRWgG7BVREJEJMpdHwr0xwkH/xLZEK5/AnYtgjWv++ywDaIr8t69XbmlXQzPf7WDka+v4tDJX/82YYzxDxcbDrpChQqsXr2aFi1a8NVXX/G3v/2NrKwsRo4cSXx8PG3atOGBBx4gIiKiQJ8TERHB3XffTYsWLbj++utp3749AEeOHGH8+PG8/vrrNGrUiPvuu4+xY8cW+XsVaDhoEbkB+DcQDExR1SdEZCKQqKoLRGQozp0/inMJ6I+qelZErgWecdcL8IKqThaRCm67UPeYXwAPqeolb5i/Yp3AuanCzFudqSR/vxSiG/n08O8mJvFfH2wiPCyU54e3oVODSJ8e35iSriQOB+2lwnQCB+58AIVx8id4qRNE1IXffQHBvp0n+IcfT3Dv9G/Zc/Q0f76uMX/o0ZCgIJtjwBiwACgsmw/A18Krw03PwcHv4Ov/9fnhm9SoxIL7u3FDfE3+9dlW7pq2hp9Pp/v8c4wxJjcLgIJqdjO0uh2WPg1Ja3x++IplQ3h+eBsmDmjOsh1H6P/8Mtbt+9nnn2NMSVSSrlR4qbB/TxYAhdHvKagUA++NgbMF69UvDBHhN51jmXtPFwBufXUFby7fbf/zm4AWFhbG0aNH7d9BPlSVo0ePEhYWVuB9rA+gsPYsg6n9od1o57JQMUlJTefPc9bz5Q+HuDG+Jk8NiSc8zLd9D8aUBBkZGSQnJ1/wvntzvrCwMGJiYggNPf9nxcX6AAJ7SsjLEdsNutwP30xyzgauHpfvhPKXI6J8GV77TQKvLd3F/362lc0HjvPSiHY0q1Up/52NKUVCQ0OpX7++12WUSnYJ6HL0/ju0Gg6LHocvHnNuFS0GQUHC73s0ZNbdnUhNz2LQS8t5Z80+OxU2xviEBcDlCA6BAS9Bwl3ONJILH/bZk8IX0qF+VRaO7U5CbBX+Mm8j497dQGq6TTtpjCkaC4DLFRTkzCLW5X5Y8xosuA+yiu+HclTFsrz124480DuO+euSGfjicnYc8n1HtDEmcFgAFIUIXPsP6PlX+G4GzLsLMovv/v3gIOGhaxsx7c4OHDmVzoAXlrFg/YH8dzTGmAuwACgqEej5F7jucdjyPswZBRnFe7fC1Y2i+fiBbjSpWYkHZq3j/72/kbOZNu2kMaZwLAB8pcv90P9ZZwjpmbcUy3MCudWsXI7ZYzpxd/f6TF+5j6EvryDpWGqxfqYxpnSxAPClhN/CoFecZwWmDy7SlJIFERocxKM3NmPyqHbsOXqaGyct5fMtPxXrZxpjSg8LAF9rNQxumepMKzntJjh9tNg/8rrmNfj4/u7UjSzP3W8l8uTC78nIKr67kowxpYMFQHFoNgCGz4Ij22DqDXDyx2L/yLqR5Zl7TxdGdqrLq0t2cftrK23aSWPMJVkAFJe4a2HEXEhJgil9IWVfsX9kWGgwjw+M57lhrdl84AQ9n17M3z/YxI/H7RF6Y8yv2VhAxS1pDcwYAmXCYfQCZ5axK/Gxx1J54asdzPs2mSARhnWowx96NqRm5XJX5PONMf7DJoTx0sEN8PYgkCD4zQdQvdkV++ikY6m8tHgH7yY6QXBr+xju7XkVtSIsCIwJFBYAXju8Fd4aAJlpMHI+1G57RT/eCYKdzF2bBMCtCXW4t9dV1LYgMKbUswDwB8d2w1s3Q+rPMOJdqNf5ipeQ/HMqLy/eyZxEJwhuSajDvT0bElOl/BWvxRhzZVgA+Ivj+50QOHEAhs2Ehr08KWN/yhleXryDOWuSUZSh7ZxLQ3WqWhAYU9pYAPiTU4ecPoEj2+DWt6BxP89KOZByhpcX7+SdNUlkqxMEf+xlQWBMaVKkSeFFpK+IbBWRHSIy/gLb64nIlyKyQUQWi0hMrvXfish3IrJZRO7JtU87EdnoHnOSSDHMquKvKlaD0R9C9RbwzkjYNM+zUmpFlOMfA1vw9SM9GdGxLvPX7afX04v5y9wN7DtqQ0sYU5rlewYgIsHANuBaIBlYAwxX1S252rwLfKSq00TkGuBOVR0lImXczzgrIhWBTUAXVT0gIquBB4BVwEJgkqp+cqlaSs0ZwDlpJ2DWMNj7Ddz8PLQd5XVF/Hg8jVe+3snM1fvIylYGt6nNfddcRb3ICl6XZoy5TEU5A+gA7FDVXaqaDswGBuRp0wz4yn296Nx2VU1X1bPu+rLnPk9EagKVVHWlOgn0FjCwcF+pFAir5Dws1vAaZz6BVa96XRE1Kofx2M3NWfpIL0Z1qseC9Qe45pmvGffuevYcOe11ecYYHypIANQGknK9T3bX5bYeGOy+HgSEi0gkgIjUEZEN7jH+qaoH3P2T8zkm7v5jRCRRRBIPHz5cgHJLmDLlnWEjmvSHTx6Bpc94XREA1SvlBMHozrF8uP4Avf/vax6a8x27LQiMKRV8NRTEOKCHiKwDegD7gSwAVU1S1ZbAVcBoEalemAOr6mRVTVDVhOjoaB+V62dCyjoDyMXfAl9OdBY/6ZyvVimMv93UjKV/6cUdXWJZuPEgvZ9ZzEPvfMeuwzYjmTElWUgB2uwH6uR6H+Ou+4X7W/1gAPda/xBVTcnbRkQ2Ad2B5e5xLnrMgBMcCoNehdDyzllAeir0fdKZcMYPVAsP47/6N+OeHg2ZvGQnb6/cy/vf7efmVrW475o4rqpW0esSjTGFVJAzgDVAnIjUdzt1hwELcjcQkSgROXesCcAUd32MiJRzX1cBugFbVfUgcEJEOrl3//wG+MAn36gkCwqGm56DTvfCqpdhwf2Q7V8zfUWHl+XRG5ux9JFr+F33Bny2+SeuffZrxs5ex45DJ70uzxhTCPmeAahqpojcB3wGBANTVHWziEwEElV1AdATeFJEFFgC/NHdvSnwjLtegKdVdaO77V5gKlAO+MRdjAhc/z9QpgIs+RdknHEmmQkO9bqy80SHl+WvNzRlzNUNeG3pLt5esZcF6w/Qv2UtHrjmKuKqh3tdojEmH/YgmD9b9ix88Rg0vhGGToHQMK8ruqijp87y2tLdvLViD2cysrgxviYP9I6jkQWBMZ6zJ4FLqtWvwcJx0KCXM3REGf9+QvfY6XReX7qLad/sITUjixvia/LANXE0rmFBYIxXLABKsnUznOcE6nSC299xnh/wcz+fTuf1ZbuY9s1eTp3N5Ib4GjzQO44mNfy/dmNKGwuAkm7TfJh/N9SId4aTLl/V64oKJCU1nTeW7ebN5Xs4dTaTfi2cIGha04LAmCvFAqA02PopzPmNM6vYqPchvFCPVHgqJTWdKW4QnDybyfXNq/NA7zia16rsdWnGlHoWAKXFrsUwaziE13SmmKwck+8u/uR4agZTlu9myvLdnEzL5LpmThC0qG1BYExxsQAoTfathBm3QFgEjP4AqjbwuqJCO34mgzeX7+aNZU4Q9GlanQf7WBAYUxwsAEqbA+vg7cEQXMaZZ7haE68ruizHz2Qwdfke3li2ixNpmfRpWo2xvRsRH2NBYIyvWACURoe+d+YZzs50OoZrtfa6ost2Ii2Dacv38Pqy3Rw/k8E1TaoxtnccrepEeF2aMSWeBUBpdXSnEwJpJ2DkXKjTweuKiuRkWgbTvnGCICU1g16NoxnbpxGtLQiMuWwWAKVZSpIzz/DJn5yhpRv08LqiIjt1NtMJgqW7+Dk1gx6NohnbJ462dat4XZoxJY4FQGl38kd4ayAc2wW3vQ2Nrve6Ip84dTaTt1bs4bUlThBc3Siasb3jaFfPgsCYgrIACASnj8L0QfDTFhjyOjQf6HVFPnP6bCZvr9zL5CW7OHY6ne5xUTzYJ4529UrGA3HGeMkCIFCkHXduEU1eAwNfgVa3eV2RT50+m8l0NwiOnk6n21VRjO0TR/tYCwJjLsYCIJCkn4aZtzmTzQ+bCY37el2Rz6WmZzJj5T5eXbKTI6fS6XpVJGN7N6JDfQsCY/KyAAg0Z0/CtJucW0VHvQ/1OntdUbE4k57FjFV7eeXrXRw5dZbODSIZ2yeOTg0ivS7NGL9hARCITh+BKdfD6cNw5ydQvbnXFRWbM+lZzFy9j1e+3snhk2fp1KAqY3s3onNDCwJjLAAC1c97nRBQhbv+A1XqeV1RsUrLyGLmKicIDp08S4f6VXmwTxydG0QifjK/sjFXmgVAIPtpC7zZF8pHwW8/g4rRXldU7NIyspi9eh8vf72Tn06cpUNsVcb2iaNLQwsCE3gsAALdvpXOE8PRTeCOj6BsYMzQlZaRxTtrknh58U5+PJFGQr0qPNinEV2vsiAwgcMCwDjzCcy+HWK7wYh3IaSs1xVdMWkZWcxJTOKlRU4QtKtXhbG94+geF2VBYEo9CwDj+G4WvH8PNBsAQ9+EoGCvK7qizmZmMScxmZcW7eDg8TTa1I3gwT6NuNqCwJRiFwuAoALu3FdEtorIDhEZf4Ht9UTkSxHZICKLRSTGXd9aRFaIyGZ322259pkqIrtF5Dt3aV2E72cKqvVwuO5x2PIBLHzY6RwOIGVDghnVqR6LH+7J4wNb8NPxNEZPWc2gl75h8dZDlKRfiIwpqnzPAEQkGNgGXAskA2uA4aq6JVebd4GPVHWaiFwD3Kmqo0SkEaCqul1EagFrgaaqmiIiU9195ha0WDsD8KH//Bd8Mwl6ToCev8r0gJGemc3ctcm8uGgH+1PO0KpOBA/2jqNn42g7IzClRlHOADoAO1R1l6qmA7OBAXnaNAO+cl8vOrddVbep6nb39QHgEFD6b0EpCa6dCK1HwOInYfVrXlfjmTIhQdzesS6LxvXkycHxHD11ljunrmHgi8v56oef7IzAlGohBWhTG0jK9T4Z6JinzXpgMPAcMAgIF5FIVT16roGIdADKADtz7feEiPwN+BIYr6pn8364iIwBxgDUrVu3AOWaAhGBmyZB6lHnUlD5SGgx2OuqPFMmJIjhHeoytF0M879N5vmvdvDbqYlUCgshJLhAV0qNKVbz/tCF+lEVfHrMggRAQYwDXhCRO4AlwH4g69xGEakJvA2MVtVsd/UE4EecUJgM/AWYmPfAqjrZ3U5CQoL9OuZLwSFOR/D0wTB/DJSrAg17eV2Vp0KDg7itfV0Gt43h/XX72ZB83OuSjAGgQlnf37BRkADYD9TJ9T7GXfcL9/LOYAARqQgMUdUU930l4GPgUVVdmWufg+7LsyLyJk6ImCutTHlnEpk3b4R3RsLoD6F2W6+r8lxocBC3JNThloQ6+Tc2poQqyLntGiBOROqLSBlgGLAgdwMRiRKRc8eaAExx15cB3gPeytvZ654VIE5P20BgUxG+hymKclVg5DwoXxVmDIUjO7yuyBhzBeQbAKqaCdwHfAZ8D8xR1c0iMlFEbnab9QS2isg2oDrwhLv+VuBq4I4L3O45Q0Q2AhuBKOBxH30nczkq1YSR7wECbw+CEwe8rsgYU8zsQTBzvgPrYGp/iKgLdy50zg6MMSVakR4EMwGkVhtnEpmjO2DmMEhP9boiY0wxsQAwv9agBwx+DZJWwbt3QFaG1xUZY4qBBYC5sOYD4canYftnsOB+yM7OdxdjTMniq+cATGnU/ndw+igs/h+oEOWMIWSMKTUsAMyl9XjEmVLym+ehQjR0Het1RcYYH7EAMJcmAv3+CalH4PO/ObOKtRnhdVXGGB+wADD5CwqGQa/CmZ+d/oDyVaFxP6+rMsYUkXUCm4IJKQu3TYeaLZ07g/au8LoiY0wRWQCYgisbDiPmQuUYmHUb/LTZ64qMMUVgAWAKp0IUjHoPQsvD24Ph571eV2SMuUwWAKbwIurCyPmQecYZN+jUYa8rMsZcBgsAc3mqN4Pb5ziDxs0YCmdPel2RMaaQLADM5avbCW6dBj9uhNkjIPNXE7oZY/yYBYApmkbXw4AXYffXMP9uyM7Kfx9jjF+wADBF13q4M0zElg9g4TgoQUOMGxPI7EEw4xtd7neGjFj+HFSoBr0meF2RMSYfFgDGd/r8tzN43NdPObeLdrjb64qMMZdgAWB8RwRueg7OHIOFD0P5SGgx2OuqjDEXYX0AxreCQ2DoFOcOofljYOcirysyxlyEBYDxvdByMHwWRDVybg/d/63XFRljLqBAASAifUVkq4jsEJHxF9heT0S+FJENIrJYRGLc9a1FZIWIbHa33ZZrn/oisso95jsiUsZ3X8t4rlwVGDkPKkQ6D4od2eF1RcaYPPINABEJBl4E+gHNgOEi0ixPs6eBt1S1JTAReNJdnwr8RlWbA32Bf4tIhLvtn8CzqnoV8DNwVxG/i/E3lWrCqPcBcYaMOHHA64qMMbkU5AygA7BDVXepajowGxiQp00z4Cv39aJz21V1m6pud18fAA4B0SIiwDXAXHefacDAInwP468iG8LIuU7H8NuDIfWY1xUZY1wFCYDaQFKu98nuutzWA+du9xgEhItIZO4GItIBKAPsBCKBFFXNvMQxz+03RkQSRSTx8GEbdKxEqtUGhs2EYzth1jBIT/W6ImMMvusEHgf0EJF1QA9gP/DLmAAiUhN4G7hTVbMLc2BVnayqCaqaEB0d7aNyzRXXoAcMfg2SVjsTymRleF2RMQGvIAGwH6iT632Mu+4XqnpAVQerahvgUXddCoCIVAI+Bh5V1ZXuLkeBCBEJudgxTSnUfCDc+Axs/8yZWjK7UL8LGGN8rCABsAaIc+/aKQMMAxbkbiAiUSJy7lgTgCnu+jLAezgdxOeu96OqitNXMNRdNRr4oChfxJQQ7e+Cnn+F9bPgi795XY0xAS3fAHCv098HfAZ8D8xR1c0iMlFEbnab9QS2isg2oDrwhLv+VuBq4A4R+c5dWrvb/gI8JCI7cPoE3vDRdzL+rscj0P5u+OZ5Z+wgY4wnREvQyI0JCQmamJjodRnGF7KzYd5dsHk+DHgJ2ozwuiJjSi0RWauqCXnX21hAxhtBQTDoVTjzs9MfUL4qNO7ndVXGBBQbCsJ4J6QM3PY21Gzl3Bm0d4XXFRkTUCwAjLfKhsOId6FyDMy8DX7a7HVFxgQMCwDjvQpRMOo9KFPeeVr4571eV2RMQLAAMP4hoq4TAplpzrhBp+ypb2OKmwWA8R/VmsLtc5xB42YMgbQTXldkTKlmAWD8S92OcOs0+HETvDMCMs96XZExpZYFgPE/ja6HAS/C7iUw/27Izsp/H2NMoVkAGP/Uejhc9wRs+QAWjoMS9MCiMSWFPQhm/FeX++D0YVj+b6hQDXpN8LoiY0oVCwDj3/o8BqePwNdPObeLdrjb64qMKTUsAIx/E4GbnnNmFFv4MJSPhBaD89/PGJMv6wMw/i84BIZOgbqdYP4Y2LnI64qMKRUsAEzJEFoOhs+GqEYwewTsX+t1RcaUeBYApuQoFwGj5kOFSJhxCxzZ7nVFxpRoFgCmZAmvAaPeB8QZMuLEAa8rMqbEsgAwJU9kQxg5D86kOIPHpR7zuiJjSiQLAFMy1WoNw2bAsZ0waxikp3pdkTEljgWAKbka9IAhr0PSamdCmawMrysypkSxADAlW7MB0P//YPtnztSS2dleV2RMiWEPgpmSL+G3ztPCi55wnha+7nGvKzKmRCjQGYCI9BWRrSKyQ0TGX2B7PRH5UkQ2iMhiEYnJte1TEUkRkY/y7DNVRHaLyHfu0rrI38YErqsfhg5j4JvnYflzXldjTImQ7xmAiAQDLwLXAsnAGhFZoKpbcjV7GnhLVaeJyDXAk8Aod9u/gPLA7y9w+IdVdW5RvoAxgDNkRN9/QupR+PxvzpARbUZ6XZUxfq0gZwAdgB2quktV04HZwIA8bZoBX7mvF+XerqpfAid9UKsxlxYUBANfgQa9YMEDsPUTrysyxq8VJABqA0m53ie763JbD5wboWsQEC4ikQU49hPuZaNnRaTshRqIyBgRSRSRxMOHbZ5Yk4+QMnDb21CzlXNn0N4VXldkjN/y1V1A44AeIrIO6AHsB/KbxmkC0ARoD1QF/nKhRqo6WVUTVDUhOjraR+WaUq1sOIx4FyrXgZm3OdNLGmN+pSABsB+ok+t9jLvuF6p6QFUHq2ob4FF3XcqlDqqqB9VxFngT51KTMb5RIcoZN6hMBZg+BH7e43VFxvidggTAGiBOROqLSBlgGLAgdwMRiRKRc8eaAEzJ76AiUtP9U4CBgP2aZnwroq4TAplpzrhBp+wSojG55RsAqpoJ3Ad8BnwPzFHVzSIyUURudpv1BLaKyDagOvDEuf1FZCnwLtBbRJJF5Hp30wwR2QhsBKIAu3nb+F61pnD7HDhxEGYMgbQTXldkjN8QLUGTbSckJGhiYqLXZZiSaNt/nDGDYrvCiLkQcsF7DowplURkraom5F1vQ0GYwNDoOhj4EuxeAvPvhuz87lEwpvSzADCBo9UwuO4J2PIBLBwHJejs15jiYGMBmcDS5T44fRiW/xsqVINeE7yuyBjPWACYwNPnMUg9Al8/5dwu2uFurysyxhMWACbwiED/55yZxBY+DOWrQoshXldlzBVnfQAmMAWHwNApULczzP897Pwq/32MKWUsAEzgCi0Hw2dBdGOYPRL2r/W6ImOuKAsAE9jKRTgTzFeIhBm3wJHtXldkzBVjAWBMeA0Y9T5IkDNkxIkDXldkzBVhAWAMQGRD5wnhMynw9mCng9iYUs4CwJhzarWG4TPh2E5n2Ij0VK8rMqZYWQAYk1v9q2HI65C02plQJivD64qMKTYWAMbk1WwA9P8/2P6Z0yew+jU4vNWGjjCljj0IZsyFJPwWMtPhm0nOuEHgDB0R281Z6l8NkVc5D5UZU0JZABhzMZ3ugY6/h2O7YM8y2LMUdi+FzfOd7RVrnB8IVRtYIJgSxQLAmEsRce4QimwI7UY7l4GO7nTC4Nyyaa7TNrxWrkDoDlXqWyAYv2YBYExhiEDUVc6ScKcTCEe254TBrkWwcY7TtlJMnkCI9bR0Y/KyADCmKEQgupGztL/LCYTDW3MCYcfnsGG207Zy3ZwwiO3mzFlsjIcsAIzxJRGo1sRZOtwN2dlw+Ae3D2EJbPsU1s902kbUg9juOYFQOcbb2k3AsTmBjbmSsrPh0JacTuU9yyAtxdlWpX5Oh3JsN6hUy9NSTelxsTmBLQCM8VJ2Nvy0KVcgLIezx51tVRueHwjhNbyt1ZRYRQoAEekLPAcEA6+r6lN5ttcDpgDRwDFgpKomu9s+BToBy1S1f6596gOzgUhgLTBKVdMvVYcFgCn1srPgx405gbD3Gzh7wtkWGZfTh1CvG4RX97ZWU2JcdgCISDCwDbgWSAbWAMNVdUuuNu8CH6nqNBG5BrhTVUe523oD5YHf5wmAOcB8VZ0tIq8A61X15UvVYgFgAk52Fhxcn3O5aO8KSD/pbItqfH4gVIz2tlbjt4oSAJ2Bx1T1evf9BABVfTJXm81AX1VNEhEBjqtqpVzbewLjzgWA2+YwUENVM/N+xsVYAJiAl5XpBsKSnEDIOO1si256fiBUiPS2VuM3LhYABbkLqDaQlOt9MtAxT5v1wGCcy0SDgHARiVTVoxc5ZiSQoqqZuY5Z+yKFjwHGANSta7fNmQAXHAIx7Zyl25+cweoOfJcTCN/NgDWvOW2rNc+5w6heV2fuY2Ny8dVtoOOAF0TkDmAJsB/I8sWBVXUyMBmcMwBfHNOYUiM4FOq0d5buf3bGLzqwLicQ1k6DVa8AAtVb5AqELlCuitfVG48VJAD2A3VyvY9x1/1CVQ/gnAEgIhWBIaqacoljHgUiRCTEPQv41TGNMZchpAzU7egsVz8MmWdh/7c5D6YlToGVLwECNeJz7jCq1wXCKntdvbnCChIAa4A4966d/cAw4PbcDUQkCjimqtnABJw7gi5KVVVEFgFDce4EGg18UPjyjTGXFFIW6nV2lh6PQEYa7F+bM7Dd6smw4gVnOswaLd0zhO5QtzOEVcr/+KZEK+htoDcA/8a5DXSKqj4hIhOBRFVdICJDgScBxbkE9EdVPevuuxRoAlTE+c3/LlX9TEQa4Pzwrwqsw7l19Oyl6rBOYGN8LOMMJK9xLhftXuq8zs5wAqFm61yB0AnKhntdrblM9iCYMSZ/6amQvDonEPavdQMhGGq1yelDqNMJylb0ulpTQBYAxpjCSz8NSatyAuHAt5CdCUEhUKvt+YFQprzX1ZqLsAAwxhTd2VOQtDJXIKwDzYKgUKjdLlcgdITQcl5Xa1wWAMYY3zt7EvatzOlUPvgdaDYEl4GY9u58CN2d16FhXlcbsCwAjDHFL+0E7FuREwg/bnADoSzU6ZArEBKcO5TMFWEBYIy58s6kuIGwDHYvcQa6QyEkzA0E9y6j2u2cZxhMsSjKUBDGGHN5ykVA437OAnDmZ2eE093u4HaLnnDWh5RzHl6L7QaxVzt3HFkgFDsLAGPMlVOuCjS50VkAUo/B3uU5gfDV48760PLOswe/BEJrZ9gL41MWAMYY75SvCk1vchaA00fOD4QvJzrry1R0A8G9ZFSzlTMwnikS+xs0xviPClHQbICzAJw6DHuX5QTCF3931pcJd4a3iHVvO63ZCoKCvau7hLIAMMb4r4rR0HyQswCc/On8QNj+H2d92UrOgHbnAqFGvAVCAVgAGGNKjvDq0GKIswCcOOheMnKHv972qbM+rLIzB8K5QKjeAoKCvKvbT1kAGGNKrko1IX6oswAc339+IGxd6KwPi8h5BiG2G1RrZoGABYAxpjSpXBta3uosAClJuTqVl8IPHznry1WF2K7OHUax3aBaUxDxrm6PWAAYY0qviDoQMQxaDXPe/7zXOTPYs8wJhO8/dNaXj3IDwb3LKLpxQASCBYAxJnBUqecsbUaAKqTszelQ3rMUtrjzUlWIzrlkVP9qiLyqVAaCBYAxJjCJQJVYZ2k7ygmEn3efHwib33PaVqzhBkI3JxCqNigVgWABYIwx4PxAr9rAWdqNdgLh2K6cDuU9S2HTXKdteK1cgdAdqtQvkYFgAWCMMRciApENnSXhTicQju7ICYRdi2DjHKdtpdo5dxjV7w4R9UpEIFgAGGNMQYhAVJyztL/LCYQj29xAWAo7PocNs522levkCYS63tZ+ERYAxhhzOUScu4WiG0OHu51AOPS9e7loifNQ2vqZTtuIujm3nNbvDpVjvK3dZfMBGGNMccjOhkNbcvoP9iyDtBRnW5X6OR3Ksd2gUq1iLaVIE8KISF/gOSAYeF1Vn8qzvR4wBYgGjgEjVTXZ3TYa+H9u08dVdZq7fjFQEzjjbrtOVQ9dqg4LAGNMiZWdDT9tygmEvcsh7bizrWrD8wMhvIZPP/qyA0BEgoFtwLVAMrAGGK6qW3K1eRf4SFWnicg1wJ2qOkpEqgKJQAKgwFqgnar+7AbAOFUt8E90CwBjTKmRneXMkPZLIHwDZ0842yLjci4X1evmjIFUBEWZEawDsENVd7kHmg0MALbkatMMeMh9vQh43319PfC5qh5z9/0c6AvMuozvYIwxpUdQsDPRTa3W0OU+JxAOrs8JhI1zYe2bTtuoxnDrW1CtiU9LKEgA1AaScr1PBjrmabMeGIxzmWgQEC4ikRfZt3au92+KSBYwD+fy0K9OR0RkDDAGoG5d/+xJN8aYIgsKhtptnaXrA5CV6QaC239QuXb+xyjsR/roOOOAHiKyDugB7Aey8tlnhKrGA93dZdSFGqnqZFVNUNWE6OhoH5VrjDF+LjgEYtpBtwdh5FwoG+7zjyhIAOwH6uR6H+Ou+4WqHlDVwaraBnjUXZdyqX1V9dyfJ4GZOJeajDHGXCEFCYA1QJyI1BeRMsAwYEHuBiISJSLnjjUB544ggM+A60SkiohUAa4DPhOREBGJcvcNBfoDm4r+dYwxxhRUvgGgqpnAfTg/zL8H5qjqZhGZKCI3u816AltFZBtQHXjC3fcY8A+cEFkDTHTXlcUJgg3AdzhnBa/58HsZY4zJhz0IZowxpdzFbgO1OdGMMSZAWQAYY0yAsgAwxpgAZQFgjDEBqkR1AovIYWCv13W4ooAjXheRD3+v0d/rA6vRF/y9PvD/GotaXz1V/dWTtCUqAPyJiCReqFfdn/h7jf5eH1iNvuDv9YH/11hc9dklIGOMCVAWAMYYE6AsAC7fZK8LKAB/r9Hf6wOr0Rf8vT7w/xqLpT7rAzDGmABlZwDGGBOgLACMMSZAWQAUkojUEZFFIrJFRDaLyFiva7oQEQkWkXUi8pHXtVyIiESIyFwR+UFEvheRzl7XlJeI/Mn9b7xJRGaJSJjH9UwRkUMisinXuqoi8rmIbHf/rOKHNf7L/e+8QUTeE5EID0u8YI25tv1ZRPTccPVeuFh9InK/+/e4WUT+1xefZQFQeJnAn1W1GdAJ+KOINPO4pgsZizN8t796DvhUVZsArfCzWkWkNvAAkKCqLYBgnLkwvDQVZ07t3MYDX6pqHPCl+95LU/l1jZ8DLVS1JbANZ84QL03l1zUiInVw5izZd6ULymMqeeoTkV44c7G3UtXmwNO++CALgEJS1YOq+q37+iTODy7fT9ZZBCISA9wIvO51LRciIpWBq4E3AFQ13Z1Bzt+EAOVEJAQoDxzwshhVXQIcy7N6ADDNfT0NGHgla8rrQjWq6n/ceUUAVuLMDOiZi/w9AjwLPAJ4emfMRer7A/CUqp512xzyxWdZABSBiMQCbYBVHpeS179x/kfO9riOi6kPHAbedC9TvS4iFbwuKjd3ytKncX4bPAgcV9X/eFvVBVVX1YPu6x9xJmTyZ78FPvG6iLxEZACwX1XXe13LRTQCuovIKhH5WkTa++KgFgCXSUQqAvOAB1X1hNf1nCMi/YFDqrrW61ouIQRoC7zsziN9Gu8vXZzHvZY+ACesagEVRGSkt1Vdmjr3dPvtfd0i8ijOJdQZXteSm4iUB/4K/M3rWi4hBKiKc9n5YWCOiEhRD2oBcBnceYznATNUdb7X9eTRFbhZRPYAs4FrRGS6tyX9SjKQrKrnzpzm4gSCP+kD7FbVw6qaAcwHunhc04X8JCI1Adw/fXJpwNdE5A6cub9HqP89fNQQJ+jXu/9uYoBvRaSGp1WdLxmYr47VOGf3Re6otgAoJDd13wC+V9X/87qevFR1gqrGqGosTqflV6rqV7+5quqPQJKINHZX9Qa2eFjShewDOolIefe/eW/8rKPatQAY7b4eDXzgYS0XJCJ9cS5J3qyqqV7Xk5eqblTVaqoa6/67SQbauv+f+ov3gV4AItIIKIMPRi+1ACi8rsAonN+sv3OXG7wuqgS6H5ghIhuA1sD/eFvO+dyzk7nAt8BGnH8rng4XICKzgBVAYxFJFpG7gKeAa0VkO85Zy1N+WOMLQDjwufvv5RU/rNFvXKS+KUAD99bQ2cBoX5xJ2VAQxhgToOwMwBhjApQFgDHGBCgLAGOMCVAWAMYYE6AsAIwxJkBZABhjTICyADDGmAD1/wHDW0bVayiGFgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "trainer = Trainer()\n",
    "trainer.load()\n",
    "trainer.model.eval()\n",
    "bert = trainer.model.bert\n",
    "fc = trainer.model.classifier\n",
    "acc_bert = accuracy(bert)[0]\n",
    "for interp_mode in ['avg', 'absmax']:\n",
    "    accs = [acc_bert]\n",
    "    for factor in [2, 4, 8, 16]:\n",
    "        lowbert = BertModel.from_pretrained(trainer.model.bert_model_name)\n",
    "        lowbert.to(trainer.device)\n",
    "        lowbert.train()\n",
    "        lowbert.load_state_dict(bert.state_dict())\n",
    "        set_print(lowbert, False)\n",
    "        set_lowdim(lowbert, True, factor=factor, qk_linear_mode=interp_mode)\n",
    "\n",
    "        trainer.model.bert = lowbert\n",
    "        trainer.model.train()\n",
    "        trainer.optimizer = torch.optim.Adam(trainer.model.parameters(), lr=5e-5)\n",
    "        trainer.scaler = torch.cuda.amp.GradScaler()\n",
    "        for steps in tqdm.tqdm(range(1000)):\n",
    "            batch = trainer.get_batch()\n",
    "            trainer.optimize_step(batch)\n",
    "            #if steps % 15 == 0: trainer.report(batch)\n",
    "            trainer.steps = steps\n",
    "        trainer.model.eval()\n",
    "        acc, took = accuracy(lowbert)\n",
    "        accs.append(acc)\n",
    "    plt.plot([1, 2, 4, 8, 16],accs, label=interp_mode)\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 120000/120000 [00:00<00:00, 205128.11it/s]\n",
      "100%|██████████| 7600/7600 [00:00<00:00, 199999.44it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Classification Dataset Stat.: name:AG_NEWS, nclass:5, max_len:1012, avg_len:236.477525, count:120000\n",
      "Classification Dataset Stat.: name:AG_NEWS, nclass:5, max_len:892, avg_len:235.2992105263158, count:7600\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trainer.__init__: Model initialized. model = bert\n",
      "Trainer.load: Loading...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 500/500 [00:08<00:00, 55.64it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 500/500 [00:08<00:00, 57.40it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 500/500 [00:08<00:00, 59.40it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 500/500 [00:08<00:00, 59.35it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 500/500 [00:08<00:00, 59.77it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 500/500 [00:08<00:00, 57.08it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 500/500 [00:08<00:00, 58.35it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 500/500 [00:08<00:00, 58.47it/s]\n",
      "Some weights of the model checkpoint at google/bert_uncased_L-4_H-256_A-4 were not used when initializing BertModel: ['cls.seq_relationship.bias', 'cls.predictions.transform.dense.weight', 'cls.predictions.bias', 'cls.seq_relationship.weight', 'cls.predictions.decoder.bias', 'cls.predictions.decoder.weight', 'cls.predictions.transform.dense.bias', 'cls.predictions.transform.LayerNorm.bias', 'cls.predictions.transform.LayerNorm.weight']\n",
      "- This IS expected if you are initializing BertModel from the checkpoint of a model trained on another task or with another architecture (e.g. initializing a BertForSequenceClassification model from a BertForPreTraining model).\n",
      "- This IS NOT expected if you are initializing BertModel from the checkpoint of a model that you expect to be exactly identical (initializing a BertForSequenceClassification model from a BertForSequenceClassification model).\n",
      "100%|██████████| 500/500 [00:08<00:00, 58.55it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXoAAAD4CAYAAADiry33AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8rg+JYAAAACXBIWXMAAAsTAAALEwEAmpwYAAAyPElEQVR4nO3dd3xUZfb48c9Jp0NIACFAgiIdKaGJYkUBpaPSsWJDXVfXHy6uBddd2659VVRAqYuAioCCAq5fpStI7zUUgUSaENLO74874BBSJjDJnUzO+/WaV2ZumxPKuc997nPPI6qKMcaY4BXidgDGGGMKlyV6Y4wJcpbojTEmyFmiN8aYIGeJ3hhjglyY2wFkFxMTo/Hx8W6HYYwxxcpPP/10SFVjc1oXcIk+Pj6e5cuXux2GMcYUKyKyM7d11nVjjDFBzhK9McYEOUv0xhgT5HzqoxeRTsAbQCjwoaq+mG19bWA0EAukAANVNclrfXlgHfC5qg7zU+zGmCCSnp5OUlISqampbocS0KKiooiLiyM8PNznffJN9CISCrwDdASSgGUiMkNV13lt9irwiap+LCLXAv8EBnmtfx743ueojDElTlJSEuXKlSM+Ph4RcTucgKSqJCcnk5SUREJCgs/7+dJ10xrYoqrbVDUNmAx0z7ZNQ2C+5/0C7/Ui0hKoCsz1OSpjTImTmppK5cqVLcnnQUSoXLlyga96fEn0NYDdXp+TPMu8/QL08rzvCZQTkcoiEgL8C3g8ry8QkaEislxElh88eNC3yI0xQceSfP7O58/IXzdjHweuEpEVwFXAHiATeACY7d1fnxNVHaWqiaqaGBub43j/fJ04lcaS9+5n37Y157W/McYEK19uxu4Banp9jvMsO0NV9+Jp0YtIWaC3qh4WkXbAlSLyAFAWiBCR46o63C/Rezm2dxP1931BqXFT0Y7PIG3vh5BQf3+NMcYUO7606JcBdUUkQUQigL7ADO8NRCTG000D8CTOCBxUdYCq1lLVeJxW/yeFkeQBqiY05usOn/G/jMbI3BEwuhMc2lwYX2WMMcVKvoleVTOAYcAcYD0wRVXXishIEenm2exqYKOIbMK58fpCIcWbpz7XtOadqs/zlDxM1qFN8G57+PENyMp0IxxjTDHUo0cPWrZsSaNGjRg1ahTvvfcef/nLX86sHzt2LMOGOaPEn3/+eerVq8cVV1xBv379ePXVV90KO08SaFMJJiYm6oXUulm/7yhd3/qBQY2jeCbkI9gwE2okQo//QGw9P0ZqjPGn9evX06BBAwCe+3It6/Ye9evxG1YvzzNdG+W7XUpKCtHR0Zw8eZJWrVoxb9482rdvz5YtWwDo3LkzI0aMIDIyknvuuYfFixeTnp5OixYtuPfee3n88TzHnviF95/VaSLyk6om5rR90D0Z2+Ci8gztUIcxq06ysOXr0PsjSNkG710JP7wGmRluh2iMCWBvvvkml112GW3btmX37t1s376dOnXqsHjxYpKTk9mwYQPt27fnxx9/pHv37kRFRVGuXDm6du3qdui5Crjqlf7w8HV1mb16H3/9fA1f/6knUQkdYNaf4dtnYd0Mp3VfpUG+xzHGuMOXlndh+O677/j2229ZtGgRpUuX5uqrryY1NZW+ffsyZcoU6tevT8+ePYvdMNCga9EDRIWH8o+eTdiRfII3522GslXg1nHQZwwc3gnvd4DvX7XWvTHmLEeOHKFSpUqULl2aDRs2sHjxYgB69uzJF198waRJk+jbty8A7du358svvyQ1NZXjx48zc+ZMN0PPU1AmeoDLL4mhT8s4Rn2/jfX7joIINO4FDy6F+jfB/Ofhw2thv427N8Y4OnXqREZGBg0aNGD48OG0bdsWgEqVKtGgQQN27txJ69atAWjVqhXdunWjadOmdO7cmSZNmlChQgU3w89V0N2M9fbb72lc/+//UTO6NNPuv5zQEK/LrXVfwKzH4ORhuOoJuOJRCPW9SJAxxr9yusEY6I4fP07ZsmU5ceIEHTp0YNSoUbRo0aLQv7fE34z1VqlMBE93bcjK3YcZt2jH2SsbdocHljg/F7wAH1wD+1a5EqcxpngaOnQozZo1o0WLFvTu3btIkvz5CMqbsd66XVadaT/v4ZU5G7mhUTWqVyz1x8oylaHPR9CoB8z8s5Psr3wcrnwMwiJci9kYUzxMnDjR7RB8EtQtenAKAL3QozFZCk9/sYYcu6oadIUHl0Dj3vC/Fz2t+1+KPlhjjCkEQZ/oAWpGl+bPHS/l2/UH+GrN/pw3Kh0NvUZB30nw+0EYdQ3M/ztknCraYI0xxs9KRKIHuKN9PI2ql+eZGWs5cjI99w3rd4EHFkPTW+H7V+CTHpB+ssjiNMYYfysxiT4sNIQXezUl+fgpXvp6Q94bl46Gnu9Brw9g1yKYeqeNuTfGFFslJtEDNImrwF1XJDBxyS6Wbk/Jf4emt8JNr8LG2TDzEQiwoajGmKJRtmxZt0O4ICUq0QM82vFSalQsxZPTV3Eqw4eqlq3uhquGw4rxMO+5wg/QGGP8rMQl+tIRYbzQszFbD/7Ou99t9W2nq4dD4l1OUbSFbxdugMYYV2UvU3zao48+SqNGjbjuuus4PeXpm2++ScOGDWnatOmZ0gjPPvssQ4YM4corr6R27dpMnz6dJ554giZNmtCpUyfS0517hCNHjqRVq1Y0btyYoUOHoqpkZGTQqlUrvvvuOwCefPJJRowYccG/U9CPo8/J1fWq0L1Zdf6zYCs3N72IS6qUy3sHEejyCpw4BHNHQJkYuKxv0QRrTEn01XDYv9q/x6zWBDq/mO9mo0ePPqtMce/evfn9999JTEzktddeY+TIkTz33HO8/fbbvPjii2zfvp3IyEgOHz585hhbt25lwYIFrFu3jnbt2jFt2jRefvllevbsyaxZs+jRowfDhg3j6aefBmDQoEHMnDmTrl27MnbsWPr06cNbb73F119/zZIlSy74Vy9xLfrT/nZzQ0pFhPLk9NVkZfnQ9x4S6tycTegAXzwIm+YWfpDGmCKXvUzx5s2bCQkJ4bbbbgNg4MCB/PDDDwA0bdqUAQMGMH78eMLC/mg3d+7cmfDwcJo0aUJmZiadOnUCoEmTJuzYsQOABQsW0KZNG5o0acL8+fNZu3YtAI0aNWLQoEHcfPPNjB49moiIC39406cWvYh0At4AQoEPVfXFbOtr40wfGAukAANVNUlEmgHvAuVxJgt/QVX/e8FR+0FM2UhG3NSAJ6auYvKy3fRvUyv/ncIioe9EGHsTTBkMQ2ZAzdaFH6wxJY0PLe/CkFuZ4uxOlymeNWsW33//PV9++SUvvPACq1c7VyGRkZEAhISEEB4efmb7kJAQMjIySE1N5YEHHmD58uXUrFmTZ5999qzvWb16NRUrVuTAgQN++b3ybdGLSCjwDtAZaAj0E5GG2TZ7FWc+2KbASOCfnuUngMGq2gjoBLwuIhX9Erkf3NIyjnZ1KvPPr9Zz4Oi5f5k5iiwHA6ZB+Ytgwi1wYH3hBmmMKTK5lSnOyspi6tSpgFP24IorriArK4vdu3dzzTXX8NJLL3HkyBGOHz/u0/ecTuoxMTEcP378zLEBpk+fTkpKCt9//z0PPfTQWV1C58uXrpvWwBZV3aaqacBkoHu2bRoC8z3vF5xer6qbVHWz5/1e4ABOqz8giAj/6NWEUxlZPPvlWt93LBsLgz5zWvjjesHh3YUXpDGmyORWprhMmTIsXbqUxo0bM3/+fJ5++mkyMzMZOHAgTZo0oXnz5jz88MNUrFjRp++pWLEi99xzD40bN+bGG2+kVatWABw6dIjhw4fz4YcfcumllzJs2DAeeeSRC/698i1TLCJ9gE6qerfn8yCgjaoO89pmIrBEVd8QkV7ANCBGVZO9tmkNfAw0UtWsbN8xFBgKUKtWrZY7d+684F+sIN5ZsIVX5mzkg8GJdGxY1fcd96+BMV2gXFW442unSJox5rwUxzLFbnGrTPHjwFUisgK4CtiD0yd/OoCLgHHAHdmTPICqjlLVRFVNjI0t+gb/0A51qFe1HE9/sYZjqXmUR8iuWmPoPxkO74KJt8Ap3y7bjDGmKPmS6PcANb0+x3mWnaGqe1W1l6o2B0Z4lh0GEJHywCxghKou9kfQ/hYeGsI/ezdh/9FU/jV3U8F2rn25M0Xh3pXODdqMtEKJ0RhjzpcviX4ZUFdEEkQkAugLzPDeQERiROT0sZ7EGYGDZ/vPcG7UTiWAtahVicFta/Pxoh2s2PVbwXau3wW6vgFb58Hn90PWORctxhgfBNqMd4HofP6M8k30qpoBDAPmAOuBKaq6VkRGikg3z2ZXAxtFZBNQFXjBs/xWoANwu4is9LyaFTjKIvKXTvWpVj6KJ6evJj2zgMm6xSC4/llYMxXmPGl1cYwpoKioKJKTky3Z50FVSU5OJioqqkD7BfWcsefjm3W/cs8ny/nLjfV48JpLCrazKswZAYvfgWv/Bh0eL5wgjQlC6enpJCUl5Thu3fwhKiqKuLg4wsPPnuM6r5uxJbIEQl46NqxK58bVeGPeZro0uYiEmDK+7ywCN/zdKZUw/3mnVELL2wstVmOCSXh4OAkJCW6HEZRKbAmEvDzXrRGRYSGM+Gx1wS8jQ0Kg+ztwSUeY+Sis/7JwgjTGGB9Zos9BlfJRDO9cn4Vbk5m87DwehgoNh1s/hhotYepdsOMH/wdpjDE+skSfi36tatGuTmVGfLaayUt3FfwAEWWg/xSoFA+T+sG+VX6P0RhjfGGJPhchIcKHQxLpcGksw6ev5vVvNxW8G6d0NAyaDpHlYXxvSNlWOMEaY0weLNHnoUxkGB8MTqRPyzhe/3Yzf/1sNRkFHXZZIc5J9lnpTl2cY78WTrDGGJMLS/T5CA8N4ZU+TRl2zSVMWrqb+8b/xMk0H6Yg9BZbDwZMheO/woTekHqkcII1xpgcWKL3gYjw+I31eL5HY+ZtOED/DxeT8nsBSx3EJcJt45yyxpP6Q7qNFTbGFA1L9AUwqG1t3h3QknV7j9LnvYXsTjlRsANccj30eA92/gDT7oKsAl4ZGGPMebBEX0CdGldjwt1tSD6eRq93F7J2bwG7YZreAp1egg0znXH2AfZksjEm+FiiPw+J8dFMva8d4SHCbe8v5scthwp2gLb3wZWPwc8fw4IX8t/eGGMugCX681S3ajmmP9CeuEqluH3MUr5YuSf/nbxd+zdoMRi+fwWWvF84QRpjDJboL0i1ClH89952tKxdiUcmr+SD7wswTl4EbnoN6t8MXz0BqwO6irMxphizRH+BKpQK5+M7W3NT04t4YfZ6np+5jqwsH/vdQ8Og90dQ+wr47D7YMq9wgzXGlEiW6P0gMiyUt/o25872CXz0w3YenryCUxk+jqgJj4J+EyG2Pvx3ECT9VLjBGmNKHJ8SvYh0EpGNIrJFRIbnsL62iMwTkVUi8p2IxHmtGyIimz2vIf4MPpCEhAh/u7kBf+1Sn5mr9jFk9FKO+jr/bFQFGDgNysbChD5wsIDTGRpjTB7yTfQiEgq8A3QGGgL9RKRhts1exZkusCkwEvinZ99o4BmgDdAaeEZEKvkv/MAiIgztcDGv39aMn3b+xq3vLWL/ER8fjCpXFQZ9BiGhMK4nHCngzV1jjMmFLy361sAWVd2mqmnAZKB7tm0aAvM97xd4rb8R+EZVU1T1N+AboNOFhx3YejSvwZjbW7M75QS9313IlgPHfNsxuo7Tsk89AuN7wYmUwg3UGFMi+JLoawDeRdmTPMu8/QL08rzvCZQTkco+7ouIDBWR5SKy/ODBg77GHtCuqBvDf+9tR1pmFr3fXcTyHT4m7Ysuc/rsU7bBxNsgrYBP3xpjTDb+uhn7OHCViKwArgL2AD4/36+qo1Q1UVUTY2Nj/RSS+xrXqMD0+y+ncpkIBny4hDlr9/u2Y0IHZzTOnuXw6RDI9LGv3xhjcuBLot8D1PT6HOdZdoaq7lXVXqraHBjhWXbYl32DXc3o0ky9/3IaVi/P/eN/Yvzinb7t2LAb3PRv2DwXvhgGWQUsj2yMMR6+JPplQF0RSRCRCKAvMMN7AxGJEZHTx3oSGO15Pwe4QUQqeW7C3uBZVqJEl4lg4t1tuaZeFZ76fA2vztno2yQmiXfANU/Bqsnwzd+sLo4x5rzkm+hVNQMYhpOg1wNTVHWtiIwUkW6eza4GNorIJqAq8IJn3xTgeZyTxTJgpGdZiVMqIpT3B7Wkb6uavL1gC09MXUW6L5OYdHgcWg+FRW/Dj28UfqDGmKAjBZ4er5AlJibq8uXL3Q6j0Kgqb8zbzOvfbubqerH8Z0ALSkeE5b1TVhZMvxvWTIPu70DzgUUTrDGm2BCRn1Q1Mad19mRsERMR/nT9pfyzVxO+33SQfqMWc+j4qbx3Cglx6tjXuQZmPAwbZhdNsMaYoGCJ3iX9Wtdi1KBENv56jD7vLmRn8u957xAW4cxQddFlMPUO2LmwaAI1xhR7luhddH3Dqky8py1HTqbT+92FrEo6nPcOkeVgwKfOhOMT+8L+NUUSpzGmeLNE77IWtSox9f7LiQwLpe+oxfxvUz4PjJWJcUolRJSG8b3htx1FEqcxpviyRB8ALo4ty2cPXE585TLcNXYZ035KynuHirVg4HTIOAnjesGxX4smUGNMsWSJPkBUKR/Ff+9tS5s60Tz26S/857steY+1r9oQ+n8Kx/bB2C5wJJ+TgzGmxLJEH0DKRYUz5vbWdG9WnZe/3sgzM9aSmdckJrXaON04xw/A6M6Qsr3ogjXGFBuW6ANMRFgIr93ajKEd6vDJop0Mm/gzqel5lA2q1RYGfwFpx2BMZ6tlb4w5hyX6ABQSIvy1SwP+dnNDvl67n8EfLeXIiTwKm9VoAbfPgqxMJ9nvX110wRpjAp4l+gB21xUJvNWvOSt3H6bPewvZe/hk7htXbQR3fAVhkTD2ZpuS0BhzhiX6AHdz0+qMvbMV+4+k0us/C9m4P49JTGIucZJ9VAX4pLs9VGWMASzRFwuXXxzDlPvaoSh93lvI4m3JuW9cqTbc+TWUv8gZerl1fu7bGmNKBEv0xUSDi8oz/YH2VC0fxeCPljJ79b7cNy5fHW6fDZUvdmapsto4xpRoluiLkRoVSzH1vnY0javAgxN/ZuyPeQynLBsLQ76Eqo1hyiCn8qUxpkSyRF/MVCwdwfi729CxQVWe/XId//xqPb+fysh549LRztDLuNYw7W5YMaFogzXGBASrR19MZWYpz8xYw/jFuwgNERrXqEDbhGja1IkmMT6a8lHhf2ycdgIm94dtC6DLq9D6HvcCN8YUirzq0fuU6EWkE/AGEAp8qKovZltfC/gYqOjZZriqzhaRcOBDoAUQBnyiqv/M67ss0ftOVVm0LZmFW5JZsj2ZX3YfIS0zixBx+vTbJFSmTZ1oWsdHUykiyylvvHE2dBwJ7R9xO3xjjB9dUKIXkVBgE9ARSMKZErCfqq7z2mYUsEJV3xWRhsBsVY0Xkf5AN1XtKyKlgXXA1aq6I7fvs0R//lLTM/l5128s3Z7Ckm0p/LzrN05lONMV1qtajnYJ5Rl66CWqJ82Gq4bD1cNBxOWojTH+kFeiz2cOOwBaA1tUdZvnYJOB7jhJ+zQFynveVwD2ei0vIyJhQCkgDTha4N/A+CQqPJTLL47h8otjADiVkcnqpCMs2Z7C4m3JTPl5P5+k9eelsOPc8r8X+X7tTn5r/xRt6sRQrUKUy9EbYwqLL4m+BrDb63MS0CbbNs8Cc0XkIaAMcL1n+VSck8I+oDTwaE6Tg4vIUGAoQK1atQoQvslLZFgoifFOn/2D11xCemYWa/ceZcnW+sz76XmuOzSJT6Ydol3GEGpGl6VNQjRt6lSmTUI0cZVKIdbaNyYo+JLofdEPGKuq/xKRdsA4EWmMczWQCVQHKgH/JyLfnr46OE1VRwGjwOm68VNMJpvw0BCa1axIs5oV4aqPyZr7NIMXvUnbmqX4V9Qwvln/K596auFXrxB1Jum3qVOZ+MqlLfEbU0z5kuj3ADW9Psd5lnm7C+gEoKqLRCQKiAH6A1+rajpwQER+BBKBbRh3iRByw0iILMul3/2D9xuFkvXkKDYnn2LJ9mSWbEvh/zYf5LMVzl91lXKRtPZq8detUtYSvzHFhC+JfhlQV0QScBJ8X5wE7m0XcB0wVkQaAFHAQc/ya3Fa+GWAtsDr/gndXDARuPr/QXgp+OZvhKSnUu+WsdSrFs/gdvGoKtsO/c6SbSlnkv/MVc4TudFlImgd7wznbJNQmfrVyhESYonfmEDk6/DKLjgJOhQYraoviMhIYLmqzvCMtPkAKItzA/YJVZ0rImWBMUBDQIAxqvpKXt9lo25csvQDmP041LkG+k6AiDLnbKKq7E45yWJP0l+yPZmk35yKmuWjwpwWf0JlWidE06h6ecJC7Xk8Y4rKBY+jL0qW6F20ciJ88SDUbAP9p0BU+Xx32XP4JEs9iX/p9hS2HfodgLKRYbSsXcnT4o+mSY2KRIRZ4jemsFiiN75bMx2m3wPVmsLAaU4ZhQI4cDSVJdud1v7S7Sls+vU4AKXCQ2lRu+KZFn+zmhWJCg8tjN/AmBLJEr0pmI1fwZTBULkuDP4cylY570MlHz/Fsh0pLPa0+NfvP4qqM2Vis5oVaZsQTeuEyrSoXZHSEf4aBGZMyWOJ3hTc1vkwqT9UiHMKo1Wo4ZfDHjmRzrIdKSzdkcKSbcms2XuUzCwlLERoGleBNnWcFn9i7UqU867XY4zJkyV6c352LoQJtzrdN0NmQKV4v3/F8VMZLN/htPaXbE9hVdJh0jOVEIHGNSrQxtPibx0fTYXSlviNyY0lenP+9vzkzFQVXtpJ9jF1C/XrTqY59XqWbHda/Ct2HyYtIwsRqF+tvPMAV0I0rROiqVw2slBjMaY4sURvLsz+NTCuh/N+0OdQrXGRfXVqeia/7D58psX/087fOJmeCUDdKmWd6pwJlWmbEE2V8lavx5RclujNhTu4CT7pBuknYdBnUKOFK2GkZWSxes8RT+JPZvmO3zjumXglIabMmdZ+mzqVqVGxlCsxGuMGS/TGP1K2O8n+xG8w4FOo3c7tiMjIzGLdvqMs3e6M7Fm2I4UjJ9MBiKtUitYJ0bT11OWvFW31ekzwskRv/OfIHifZH90LfSfCxde4HdFZsrKUjb8eY8m2ZJZsd27yJv+eBkC18lGe1r7zBO/FsWUs8ZugYYne+NfxA/BJd0jeCrd+AvU6uR1RrlSVrQePs3hbypkbvAeOnQIgpmwErROcGbiqlI8iMiyEiLAQIsNCiQgLISI0hMjwP35Ghoae+Wx1fUygsURv/O9ECozvBftXQ+8PoVFPtyPyiaqyM/nEmSJtS7ansOfwyQIfJyxEzj0xhIV4LQshIiz0j89nnTRCiQjNvn3oWZ9Pv4/M9cTjHCM8VOyqxAAXPsOUMecqHe08SDXhVph6J6SnQrN+bkeVLxEhPqYM8TFluK2VM8nN/iOpHD6ZRlpGFqcysjw/M898/mOZ8zP7+rSMLNIyz93nyMl0z/aZ5xzjVEYmWX5oY4ngddIIPXNyOPekEXrWiSKnk0tkXicer5NL9quc098Ralc5AcsSvTl/URVg0HSY3B8+vw/ST0Cru9yOqsCqVYhyZSrFjEzPCSLd+2dmwU4u6ZmcOucYZ59cUtOzOHoyI8djnMp0fvqD91VObieSCK8TTmS+J56cr5Qiw0KI8OpGs6uc/FmiNxcmogz0+y98OgRm/dkZfnn5MLejKhbCQkMICw2hdIS7caiq54ok/5PLqVyWOSeezBxOOM72p09eRz1XOacyMnM4yWWR6Y/LHDjrZJH9CuesE0kuXWhnd53lde/m7GNEZjtGoFzlWKI3Fy48Cm4d51S9nDvCadl3+IvTr2ACnoh4EqL71URPX+VcyMklp+3P6Y7zuso5fWV0+hj+vsrJ8USSw8kiMjyEi2PL8ueOl/rlu8+Kw+9HNCVTWAT0/siZrWrBC5D2O1z/rCV7UyCBdpWT48klp3s42U4W3lczOR3D+2R0+ionLTOLwhoc41OiF5FOwBs4M0x9qKovZltfC/gYqOjZZriqzvasawq8D5QHsoBWqprqr1/ABJDQMOj+HyfZ//i607Lv9BKE2IQjpnjxvsop53YwfpBvoheRUOAdoCOQBCwTkRmqus5rs6eAKar6rmdawdlAvIiEAeOBQar6i4hUBtL9/luYwBESAjf92ymCtuhtJ9l3fRNC3O8WMKak8qVF3xrYoqrbAERkMtAd8E70itNiB6gA7PW8vwFYpaq/AKhqsj+CNgFOBG74u3Oj9n8vOTdoe74PoVZm2Bg3+JLoawC7vT4nAW2ybfMsMFdEHgLKANd7ll8KqIjMAWKByar6cvYvEJGhwFCAWrVqFSR+E6hE4Jq/Oi37b59xxtnfMgbCrLSwMUXNX52n/YCxqhoHdAHGiUgIzonkCmCA52dPEbku+86qOkpVE1U1MTY21k8hmYBwxZ+g8yuwcRZM6gtpJ9yOyJgSx5dEvweo6fU5zrPM213AFABVXQREATE4rf/vVfWQqp7A6bt3p76tcU+bodDtbdi6ACb0gVPH3I7ImBLFl0S/DKgrIgkiEgH0BWZk22YXcB2AiDTASfQHgTlAExEp7bkxexVn9+2bkqLFIKcmzq7FTkG0k7+5HZExJUa+iV5VM4BhOEl7Pc7omrUiMlJEunk2ewy4R0R+ASYBt6vjN+DfOCeLlcDPqjqrEH4PUxw06QO3jXMKoY3tCscPuh2RMSWCVa80RW/LtzB5IFSsCYNnQPmL3I7ImGIvr+qV9iSLKXqXXA8DpzmTl4zpBL/tdDsiY4KaJXrjjvj2Tpnjk7/BmC7OJCbGmEJhid64Jy4RhsyEjJMwuhP8avfpjSkMluiNuy5qCnd8BRICY2+CvSvdjsiYoGOJ3rgvth7c+RVElIWPu8KuJW5HZExQsURvAkN0HbhjNpSJhXE9Ydv/3I7ImKBhid4Ejoo1nW6cirVg4q2waa7bERkTFCzRm8BSrircPsvpzpncH9Z94XZExhR7luhN4ClT2XmQqnpz+PQO+OW/bkdkTLFmid4EplIVYdBnUPty+OxeWD7G7YiMKbYs0ZvAFVkWBnwKdTvCzD/Bov+4HZExxZIlehPYwkvBbROgQTeY8yR8/4rbERlT7FiiN4EvLAL6jIGmt8H8v8O3z0GAFeMzJpD5MpWgMe4LDYMe7zkt/B/+7Uw63ulFZ8pCY0yeLNGb4iMkBG5+3ZmHdvF/nGR/8+sQEup2ZMYENJ+6bkSkk4hsFJEtIjI8h/W1RGSBiKwQkVUi0iWH9cdF5HF/BW5KKBG48R/Q4S/w8yfOiJzMDLejMiag5duiF5FQ4B2gI84csMtEZIaqepcafApn5ql3RaQhztyw8V7r/w185beoTckmAtc+5XTjzBsJ6Sehz2gIi3Q7MmMCki8t+tbAFlXdpqppwGSge7ZtFCjveV8B2Ht6hYj0ALYDay84WmO8XfkYdHoJNsx0nqJNP+l2RMYEJF8SfQ1gt9fnJM8yb88CA0UkCac1/xCAiJQF/h/wXF5fICJDRWS5iCw/eNDmETUF0PY+6PombJkHE26BU8fcjsiYgOOv4ZX9gLGqGgd0AcaJSAjOCeA1VT2e186qOkpVE1U1MTY21k8hmRKj5RDo9QHsXOhUvjx52O2IjAkovoy62QPU9Poc51nm7S6gE4CqLhKRKCAGaAP0EZGXgYpAloikqurbFxq4MWdpeguERzm1cT7uCoM+d2rmGGN8atEvA+qKSIKIRAB9gRnZttkFXAcgIg2AKOCgql6pqvGqGg+8DvzDkrwpNA26Qr/JcGgTjO0Cx/a7HZExASHfRK+qGcAwYA6wHmd0zVoRGSki3TybPQbcIyK/AJOA21Xt0UXjgrrXw4CpcHg3jOns/DSmhJNAy8eJiYm6fPlyt8Mwxd3uZTC+N0SVh8FfQOWL3Y7ImEIlIj+pamJO66zWjQlONVvB7V86T8+O6QIHNrgdkTGusURvgtdFl8HtswF1+uz3/eJ2RMa4whK9CW5V6jvz0IaXdkbj7F7mdkTGFDlL9Cb4Vb4Y7pgNpaJhXA/Y8YPbERlTpCzRm5KhYi2482uoEOfcpN38rdsRGVNkLNGbkqNcNbh9FsTUhUl9Yf1MtyMypkhYojclS5kYGDITqjeDKYNh9VS3IzKm0FmiNyVPqYow6DOofTlMu9upa29MELNEb0qmyHLQfwpcch3MeAim3glbF0BWltuRGeN3NpWgKbkiSkPfic7kJSvGwZppUKEWNOvvvCrVdjtCY/zCSiAYA5Ce6kxgsnKC07JHIeEqaD4IGtzszGZlTADLqwSCteiNAafEcZM+zuvwbvhlEqwYD9PvhsgK0KQ3NB8I1Vs4UxkaU4xYi96Y3GRlwc4fnIS/7gvISIUqDZ2E3/Q2ZwSPMQEirxa9JXpjfJF6xOnDXzEe9vwEIWFQr7PTtXPxdRBqF8fGXdZ1Y8yFiqoAiXc6rwPrnYT/y2RY/yWUrQbN+kGzgRBziduRGnMOa9Ebc74y02HTHCfpb54Lmgk12zpdO416OEM4jSkiF1yPXkQ6ichGEdkiIsNzWF9LRBaIyAoRWSUiXTzLO4rITyKy2vPz2gv7VYwJIKHhzoic/pPhz+vg+ufgRDLMGAav1oPPH4SdiyDAGlOm5Mm3RS8iocAmoCOQhDOHbD9VXee1zShghaq+KyINgdmqGi8izYFfVXWviDQG5qhqjby+z1r0plhThaRlnnH50yHtOERfDM0HwGX9oHx1tyM0QepCW/StgS2quk1V04DJQPds2yhQ3vO+ArAXQFVXqOpez/K1QCkRiSzoL2BMsSECNVtDt7fg8U3Q412nmNq8kfBaI5hwi2cET5rbkZoSxJebsTUA7xmWk4A22bZ5FpgrIg8BZYDrczhOb+BnVT2VfYWIDAWGAtSqVcuHkIwpBiLK/PGUbfJW52GslZOcYmqlKztDNJsPhKqN3I7UBDl/1brpB4xV1TigCzBORM4cW0QaAS8B9+a0s6qOUtVEVU2MjY31U0jGBJDKF8N1T8Oja2DANIi/EpZ+AO9eDqOuhmUfwsnDbkdpgpQvLfo9QE2vz3GeZd7uAjoBqOoiEYkCYoADIhIHfAYMVtWtFx6yMcVYSCjUvd55/Z4Mqz91+vNnPQZzRkD9m51WfsJVEGI1B41/+JLolwF1RSQBJ8H3Bfpn22YXcB0wVkQaAFHAQRGpCMwChqvqj36L2phgUKYytL0P2tzrTFy+cgKsmgJrplpxNeNXPo2j9wyXfB0IBUar6gsiMhJYrqozPCNtPgDK4tyYfUJV54rIU8CTwGavw92gqgdy+y4bdWNKtPRU2DjLGZtvxdVMAVgJBGOKI+/iaod3WnE1kydL9MYUZ2cVV5sBGSetuJo5hyV6Y4JF6hHnQawV42HPciuuZs6wombGBIuoCpB4h/PKqbjaZX2dln5MXbcjNQHEWvTGFHeZ6U5RtRXjnSJrVlytRLKuG2NKimO/wqrJTtI/tAnCy0Cjnk7Sr9XWbuAGMUv0xpQ0VlytxLFEb0xJlva7U0htxXjY+SNICFxyvdPKv7QzhEW4HaHxA0v0xhhH8lZYOdF5HdtrxdWCiCV6Y8zZsjKdJ29XjIMNsyArHao3h2YDoEkfKFXJ7QhNAVmiN8bk7kxxtfHw62oIjYQGXa24WjFjid4Y45t9vzgJf9UUSD1sxdWKEUv0xpiCseJqxY49GWuMKZjwKGjc23l5F1ebfrcVVyuGrEVvjPFNVpYzPHPFeM+8t1ZcLZBY140xxr9yK67WbKAzRt+KqxU567oxxviXFVcrVnwaNyUinURko4hsEZHhOayvJSILRGSFiKzyzEh1et2Tnv02isiN/gzeGBMAqjSAG1+AxzZA34lQowUsfAveToSPboSfx8GpY25HWaLl23UjIqHAJqAjkIQzh2w/VV3ntc0oYIWqvuuZVnC2qsZ73k8CWgPVgW+BS1U1M7fvs64bY4JArsXVBkCtdnYDtxBcaNdNa2CLqm7zHGwy0B1Y57WNAuU97ysAez3vuwOTVfUUsF1EtniOt6jAv4UxpvgoVxXaPwKXP+wprjbe6dNfOd6Kq7nAl66bGsBur89JnmXengUGikgSMBt4qAD7IiJDRWS5iCw/ePCgj6EbYwKeCNRsDd3ehMc3Qo/3oNxFMG8kvNYIJtziGcGT5nakQc1fzzb3A8aqahzQBRgnIj4fW1VHqWqiqibGxsb6KSRjTECJKAPN+sEds+Chn+GKP8P+NTBlMPy7Pnz9pPPZ+J0vyXgPUNPrc5xnmbe7gCkAqroIiAJifNzXGFPSVL4YrvsbPLoGBkyD+Cth2YfwXnt4/ypY+gGc/M3tKIOGL4l+GVBXRBJEJALoC8zIts0u4DoAEWmAk+gPerbrKyKRIpIA1AWW+it4Y0wxFxIKda+HWz+GxzZC55edypqzH4dX68HUu5wSDFlZbkdarOV7M1ZVM0RkGDAHCAVGq+paERkJLFfVGcBjwAci8ijOjdnb1RnOs1ZEpuDcuM0AHsxrxI0xpgQrHQ1t7nVe3sXV1kyFCjWdEspWXO282JOxxpjAlWNxtQ6e4mpdrbiaF3sy1hhTPJ1TXG2yM1nK9HusuFoBWIveGFO8WHG1HFlRM2NMcDpdXG3lBOfBrBJcXM26bowxwems4mobnCdvrbjaOaxFb4wJLpnpsHmu07WzaQ5oJtRs6yT8Rj0gspzbERYK67oxxpRMORZX6+Ek/SArrmaJ3hhTsqmeXVwt7VjQFVezRG+MMael/Q7rZjhJf+cPICHOjdvmA+HSzhAW4XaE58USvTHG5CR5K6yc6Ex+fnQPlK7sDNFsNgCqNXY7ugKxRG+MMXnJyoRtC5xW/oZZkJkGFzVzWvlN+kCpSm5HmC9L9MYY46sTKbD6U2cKxF9XQ2ikU26h+UBIuApC/FXd3b8s0RtjzPnwLq6WethTXK2/p7havNvRncUSvTHGXIj0VNg421NcbT6BWFzNnow1xpgLER4FjXs5ryNJsHKS8xRuMSmuZi16Y4w5HwFWXC2vFr1PdxVEpJOIbBSRLSIyPIf1r4nISs9rk4gc9lr3soisFZH1IvKmSACe7owxpqBCQiDhSuj1vjPxedc3nHlx5/wV/lUPJg+AjV9DZobbkebfdSMiocA7QEcgCVgmIjNUdd3pbVT1Ua/tHwKae95fDrQHmnpW/wBcBXznp/iNMcZ9URWg5e3Oy7u42oaZULaq8/Sti8XVfGnRtwa2qOo2VU0DJgPd89i+HzDJ815x5o+NACKBcODX8w/XGGMCXJX6cMPf4c/roe9EqNESFr4FbyfCRzc6wzZPHSvSkHxJ9DWA3V6fkzzLziEitYEEYD6Aqi4CFgD7PK85qro+h/2GishyEVl+8ODBgv0GxhgTiELDof5N0G+Sk/Q7Pg8nU2DGMGfi888fgJ0LnTo8hczfI//7AlNPTwAuIpcADYA4nJPDtSJyZfadVHWUqiaqamJsbKyfQzLGGJeVqwrtH4YHl8Jd3zpP266bAWM6w1st4P/+BUf3FtrX+5Lo9wA1vT7HeZblpC9/dNsA9AQWq+pxVT0OfAW0O59AjTGm2BOBmq2g25vODdwe70G56jBvJLzWCD69o1C+1pdEvwyoKyIJIhKBk8xnZN9IROoDlYBFXot3AVeJSJiIhOPciD2n68YYY0qciDLQrB/cMQseXgFXPgbRCYXyVfmOulHVDBEZBswBQoHRqrpWREYCy1X1dNLvC0zWswfmTwWuBVbj3Jj9WlW/9OtvYIwxxV10Hbj2qUI7vD0wZYwxQeCCH5gyxhhTfFmiN8aYIGeJ3hhjgpwlemOMCXKW6I0xJshZojfGmCBnid4YY4JcwI2jF5GDwE634/ASAxxyO4h8BHqMgR4fBH6MgR4fBH6MgR4fXFiMtVU1x2JhAZfoA42ILM/tIYRAEegxBnp8EPgxBnp8EPgxBnp8UHgxWteNMcYEOUv0xhgT5CzR52+U2wH4INBjDPT4IPBjDPT4IPBjDPT4oJBitD56Y4wJctaiN8aYIGeJ3hhjgpwl+lyISE0RWSAi60RkrYg84nZMORGRUBFZISIz3Y4lJyJSUUSmisgGEVkvIgE1laSIPOr5+10jIpNEJCoAYhotIgdEZI3XsmgR+UZENnt+VgrAGF/x/D2vEpHPRKRiIMXnte4xEVERiXEjNq84coxRRB7y/DmuFZGX/fFdluhzlwE8pqoNgbbAgyLS0OWYcvIIgT094xs4M4vVBy4jgGIVkRrAw0CiqjbGmUGtr7tRATAW6JRt2XBgnqrWBeZ5PrtpLOfG+A3QWFWbApuAJ4s6KC9jOTc+RKQmcAPONKduG0u2GEXkGqA7cJmqNgJe9ccXWaLPharuU9WfPe+P4SSoGu5GdTYRiQNuAj50O5aciEgFoAPwEYCqpqnqYVeDOlcYUEpEwoDSwF6X40FVvwdSsi3uDnzsef8x0KMoY8oupxhVda6qZng+LgbiijywP2LJ6c8Q4DXgCZypTV2VS4z3Ay+q6inPNgf88V2W6H0gIvFAc2CJy6Fk9zrOP9osl+PITQJwEBjj6V76UETKuB3Uaaq6B6fFtAvYBxxR1bnuRpWrqqq6z/N+P1DVzWB8cCfwldtBeBOR7sAeVf3F7VjycClwpYgsEZH/iUgrfxzUEn0+RKQsMA34k6oedTue00TkZuCAqv7kdix5CANaAO+qanPgd9zvcjjD08/dHeeEVB0oIyID3Y0qf+qMiXa9RZobERmB0/U5we1YThOR0sBfgafdjiUfYUA0TnfxX4ApIiIXelBL9HkQkXCcJD9BVae7HU827YFuIrIDmAxcKyLj3Q3pHElAkqqevhKaipP4A8X1wHZVPaiq6cB04HKXY8rNryJyEYDnp18u6f1NRG4HbgYGaGA9pHMxzgn9F8//mTjgZxGp5mpU50oCpqtjKc7V+gXfNLZEnwvPWfQjYL2q/tvteLJT1SdVNU5V43FuIM5X1YBqjarqfmC3iNTzLLoOWOdiSNntAtqKSGnP3/d1BNDN4mxmAEM874cAX7gYS45EpBNOV2I3VT3hdjzeVHW1qlZR1XjP/5kkoIXn32gg+Ry4BkBELgUi8EPFTUv0uWsPDMJpKa/0vLq4HVQx9BAwQURWAc2Af7gbzh88VxpTgZ+B1Tj/H1x/TF5EJgGLgHoikiQidwEvAh1FZDPOlciLARjj20A54BvP/5f3Aiy+gJJLjKOBOp4hl5OBIf64MrISCMYYE+SsRW+MMUHOEr0xxgQ5S/TGGBPkLNEbY0yQs0RvjDFBzhK9McYEOUv0xhgT5P4/NHZMFMYxsCgAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "trainer = Trainer()\n",
    "trainer.load()\n",
    "trainer.model.eval()\n",
    "bert = trainer.model.bert\n",
    "fc = trainer.model.classifier\n",
    "acc_bert = accuracy(bert)[0]\n",
    "for interp_mode in ['avg', 'absmax']:\n",
    "    accs = [acc_bert]\n",
    "    for factor in [2, 4, 8, 16]:\n",
    "        lowbert = BertModel.from_pretrained(trainer.model.bert_model_name)\n",
    "        lowbert.to(trainer.device)\n",
    "        lowbert.train()\n",
    "        lowbert.load_state_dict(bert.state_dict())\n",
    "        set_print(lowbert, False)\n",
    "        set_lowdim(lowbert, True, factor=factor, qk_linear_mode=interp_mode)\n",
    "\n",
    "        trainer.model.bert = lowbert\n",
    "        trainer.model.train()\n",
    "        trainer.optimizer = torch.optim.Adam(trainer.model.parameters(), lr=5e-5)\n",
    "        trainer.scaler = torch.cuda.amp.GradScaler()\n",
    "        steps = 0\n",
    "        # while steps < 1000:\n",
    "        #     batch = trainer.get_batch()\n",
    "        #     trainer.optimize_step(batch)\n",
    "        #     if steps % 15 == 0: trainer.report(batch)\n",
    "        #     steps += 1\n",
    "        #     trainer.steps = steps\n",
    "        trainer.model.eval()\n",
    "        acc, took = accuracy(lowbert)\n",
    "        accs.append(acc)\n",
    "    plt.plot([1,2,4,8,16], accs, label=interp_mode)\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x1e53e9733a0>"
      ]
     },
     "execution_count": 134,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.plot([1,3,4], label='3')\n",
    "plt.plot([5,3,1], label='123')\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.89121875"
      ]
     },
     "execution_count": 138,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "accs[2]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "interpreter": {
   "hash": "f7b3ac0126d0d6fea024471ce24e510948bf6332f7ae1a66cdcb4ee9887514e9"
  },
  "kernelspec": {
   "display_name": "Python 3.8.3 64-bit ('tensorflow': conda)",
   "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.3"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
