{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": [],
      "gpuType": "A100",
      "runtime_attributes": {
        "runtime_version": "2025.07"
      },
      "machine_shape": "hm",
      "authorship_tag": "ABX9TyNkDxE9QWEIgeeMkyFYCNX0",
      "include_colab_link": true
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    },
    "accelerator": "GPU"
  },
  "cells": [
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "view-in-github",
        "colab_type": "text"
      },
      "source": [
        "<a href=\"https://colab.research.google.com/github/jamesgolden1/equivalent-linear-LLMs/blob/main/notebooks/gemma_3/gemma_4B_jax_float32_lanczos_Borges.ipynb\" target=\"_parent\"><img src=\"https://colab.research.google.com/assets/colab-badge.svg\" alt=\"Open In Colab\"/></a>"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "%load_ext autoreload\n",
        "%autoreload 2"
      ],
      "metadata": {
        "id": "bOpoewV-HsXI"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "pwd"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 35
        },
        "id": "kcgCHT_xvNwQ",
        "outputId": "435290dc-94d6-40af-bdc3-07b5636766e2"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "'/content'"
            ],
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            }
          },
          "metadata": {},
          "execution_count": 2
        }
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "aSGZzIJxyp8K"
      },
      "outputs": [],
      "source": [
        "try:\n",
        "    from gemma import gm\n",
        "except:\n",
        "    !uv pip install --system -q gemma==3.1.0\n",
        "    # !pip install -q gemma==3.1.0 -q\n",
        "    # !pip install -q gemma==3.1.0 --target '/content/' -q"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "!python --version"
      ],
      "metadata": {
        "id": "bri3oT5jW0_f",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "c3f0d189-95f4-41e6-a8f3-bce80211e822"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Python 3.11.13\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "#@title Gemma Layers\n",
        "# %%writefile /content/gemma/layers.py\n",
        "%%writefile /usr/local/lib/python3.11/dist-packages/gemma/gm/nn/_layers.py\n",
        "# Copyright 2025 DeepMind Technologies Limited.\n",
        "#\n",
        "# Licensed under the Apache License, Version 2.0 (the \"License\");\n",
        "# you may not use this file except in compliance with the License.\n",
        "# You may obtain a copy of the License at\n",
        "#\n",
        "#     http://www.apache.org/licenses/LICENSE-2.0\n",
        "#\n",
        "# Unless required by applicable law or agreed to in writing, software\n",
        "# distributed under the License is distributed on an \"AS IS\" BASIS,\n",
        "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n",
        "# See the License for the specific language governing permissions and\n",
        "# limitations under the License.\n",
        "\n",
        "\"\"\"Base layers.\"\"\"\n",
        "\n",
        "from flax import linen as nn\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "\n",
        "\n",
        "class Einsum(nn.Module):\n",
        "  \"\"\"Einsum is a convenience module for parameterized tensor multiplication.\"\"\"\n",
        "\n",
        "  shape: tuple[int, ...]\n",
        "  weight_name: str = 'w'\n",
        "  initializer: nn.initializers.Initializer = nn.initializers.normal()\n",
        "  dtype: jnp.dtype | None = None\n",
        "\n",
        "  @nn.compact\n",
        "  def __call__(self, eqn: str, x: jax.Array) -> jax.Array:\n",
        "    w = self.param(\n",
        "        self.weight_name,\n",
        "        self.initializer,\n",
        "        self.shape,\n",
        "        self.dtype if self.dtype is not None else None,\n",
        "    )\n",
        "    return jnp.einsum(eqn, x, w)\n",
        "\n",
        "\n",
        "class RMSNorm(nn.Module):\n",
        "  \"\"\"RMSNorm layer.\"\"\"\n",
        "\n",
        "  @nn.compact\n",
        "  def __call__(self, x):\n",
        "    scale = self.param('scale', nn.initializers.zeros_init(), (x.shape[-1]))\n",
        "    var = jnp.mean(jnp.square(x), axis=-1, keepdims=True)\n",
        "\n",
        "    # Jax.lax.rsqrt is used because it returns different floats than\n",
        "    # jnp.reciprocal(jnp.sqrt(var + 1e-06))\n",
        "\n",
        "\n",
        "    normed_inputs = x * jax.lax.stop_gradient(jax.lax.rsqrt(var + 1e-06))\n",
        "\n",
        "    # normed_inputs is a rank-K tensor, K > 1 (K is typically 2 or 3). scale is\n",
        "    # a rank-1 tensor. To avoid implicit rank-promotion, reshape scale to\n",
        "    # a (1, ..., 1, D) tensor, so the rank of scale matches normed_inputs.\n",
        "    scale = jax.lax.stop_gradient(jnp.expand_dims(scale, axis=range(len(x.shape) - 1)))\n",
        "\n",
        "    # normed_inputs = x * jax.lax.rsqrt(var + 1e-06)\n",
        "\n",
        "    # # normed_inputs is a rank-K tensor, K > 1 (K is typically 2 or 3). scale is\n",
        "    # # a rank-1 tensor. To avoid implicit rank-promotion, reshape scale to\n",
        "    # # a (1, ..., 1, D) tensor, so the rank of scale matches normed_inputs.\n",
        "    # scale = jnp.expand_dims(scale, axis=range(len(x.shape) - 1))\n",
        "\n",
        "    normed_inputs = normed_inputs * (1 + scale)\n",
        "    return normed_inputs\n"
      ],
      "metadata": {
        "id": "Ksq45w17Jnh7",
        "cellView": "form",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "860c9b31-0cc4-4405-ce75-275592397093"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Overwriting /usr/local/lib/python3.11/dist-packages/gemma/gm/nn/_layers.py\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "#@title Gemma Modules\n",
        "# %%writefile /content/gemma/modules.py\n",
        "%%writefile /usr/local/lib/python3.11/dist-packages/gemma/gm/nn/_modules.py\n",
        "# Copyright 2025 DeepMind Technologies Limited.\n",
        "#\n",
        "# Licensed under the Apache License, Version 2.0 (the \"License\");\n",
        "# you may not use this file except in compliance with the License.\n",
        "# You may obtain a copy of the License at\n",
        "#\n",
        "#     http://www.apache.org/licenses/LICENSE-2.0\n",
        "#\n",
        "# Unless required by applicable law or agreed to in writing, software\n",
        "# distributed under the License is distributed on an \"AS IS\" BASIS,\n",
        "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n",
        "# See the License for the specific language governing permissions and\n",
        "# limitations under the License.\n",
        "\n",
        "\"\"\"Transformer sub-modules.\"\"\"\n",
        "\n",
        "import enum\n",
        "from flax import linen as nn\n",
        "from gemma.gm.math import _positional_embeddings\n",
        "from gemma.gm.nn import _layers\n",
        "import jax\n",
        "from jax import Array\n",
        "import jax.numpy as jnp\n",
        "\n",
        "K_MASK = -2.3819763e38  # Set to a large negative number.\n",
        "DEFAULT_ROPE_BASE_FREQUENCY = 10_000\n",
        "DEFAULT_ROPE_SCALE_FACTOR = 1.0\n",
        "\n",
        "# A dictionary with the following array shapes as keys:\n",
        "# v: [batch_size, cache_size, num_heads, head_dim]\n",
        "# k: [batch_size, cache_size, num_heads, head_dim]\n",
        "# end_index: [batch_size]\n",
        "LayerCache = dict[str, jax.Array]\n",
        "\n",
        "\n",
        "def _create_sliding_mask(\n",
        "    segment_pos: jnp.ndarray,\n",
        "    end_index: int,\n",
        "    cache_len: int,\n",
        "    sliding_window_size: int,\n",
        "):\n",
        "  \"\"\"Creates mask for sliding window attention.\"\"\"\n",
        "  total_tokens = end_index + segment_pos.shape[1]  # cached + processing tokens\n",
        "\n",
        "  def _reconstruct_rotated_cache_positions():\n",
        "    cache_positions = jnp.arange(cache_len) + total_tokens - cache_len\n",
        "    cache_positions = (\n",
        "        jnp.zeros_like(cache_positions)\n",
        "        # kv were placed at index (position_id % cache_len) in the cache.\n",
        "        .at[cache_positions % cache_len].set(cache_positions)\n",
        "    )\n",
        "    return cache_positions\n",
        "\n",
        "  # Reconstruct position_ids for cached kv.\n",
        "  cache_positions = jax.lax.cond(\n",
        "      total_tokens <= cache_len,\n",
        "      lambda: jnp.arange(cache_len),\n",
        "      _reconstruct_rotated_cache_positions,\n",
        "  )\n",
        "\n",
        "  cache_positions = cache_positions[None, None, :]  # [1, 1, cache_len]\n",
        "  segment_pos = segment_pos[:, :, None]  # [B, seq_len, 1]\n",
        "  sliding_mask = cache_positions > segment_pos - sliding_window_size\n",
        "  sliding_mask *= cache_positions < segment_pos + sliding_window_size\n",
        "  return sliding_mask\n",
        "\n",
        "\n",
        "class AttentionType(enum.Enum):\n",
        "  GLOBAL = 1\n",
        "  LOCAL_SLIDING = 2\n",
        "\n",
        "\n",
        "class Embedder(nn.Module):\n",
        "  \"\"\"Embedder module.\"\"\"\n",
        "\n",
        "  vocab_size: int\n",
        "  embed_dim: int\n",
        "\n",
        "  vision_proj_dim: int | None = None\n",
        "\n",
        "  def setup(self):\n",
        "    # Embedding matrix of shape [vocab_size, embed_dim]\n",
        "    self.input_embedding_table = self.param(\n",
        "        'input_embedding',\n",
        "        nn.initializers.normal(),\n",
        "        (self.vocab_size, self.embed_dim),\n",
        "    )\n",
        "\n",
        "    # For the multi-modal models, the encoder has additional parameters:\n",
        "    # * `mm_soft_embedding_norm` and `mm_input_projection`: Those weights\n",
        "    #   serve to project the soft tokens from the image encoder into the\n",
        "    #   embedding space of the text encoder. Those tokens are then merged with\n",
        "    #   the text tokens inside `Transformer._include_vision_embeddings`.\n",
        "    if self.vision_proj_dim:\n",
        "      self.mm_soft_embedding_norm = _layers.RMSNorm()\n",
        "      self.mm_input_projection = _layers.Einsum(\n",
        "          (self.vision_proj_dim, self.embed_dim)\n",
        "      )\n",
        "\n",
        "  def encode(self, x: jax.Array) -> jax.Array:\n",
        "    \"\"\"Encodes the input tokens.\n",
        "\n",
        "    Args:\n",
        "      x: Input tokens of shape [seq_len] or [batch_size, seq_len], where\n",
        "        each token is an integer in [0, vocab_size).\n",
        "\n",
        "    Returns:\n",
        "      Encoded tokens of shape [seq_len, embed_dim] or [batch_size, seq_len,\n",
        "      embed_dim].\n",
        "    \"\"\"\n",
        "    x = self.input_embedding_table[(x,)]\n",
        "    x *= jnp.sqrt(self.embed_dim).astype(x.dtype)\n",
        "    return x\n",
        "\n",
        "  def decode(self, x: jax.Array) -> jax.Array:\n",
        "    \"\"\"Decodes the input vectors.\n",
        "\n",
        "    Args:\n",
        "      x: Array of shape [seq_len, embed_dim] or [batch_size, seq_len,\n",
        "        embed_dim].\n",
        "\n",
        "    Returns:\n",
        "      Array of shape [seq_len, vocab_size] or [batch_size, seq_len, vocab_size].\n",
        "    \"\"\"\n",
        "    return jnp.dot(x, self.input_embedding_table.T)\n",
        "\n",
        "  def encode_vision(self, x: jax.Array) -> jax.Array:\n",
        "    \"\"\"Projects siglip embeddings to the embedding space of the text encoder.\"\"\"\n",
        "    x = self.mm_soft_embedding_norm(x)\n",
        "    x = self.mm_input_projection('...tm,md->...td', x)\n",
        "    return x\n",
        "\n",
        "\n",
        "class Attention(nn.Module):\n",
        "  \"\"\"Attention module.\"\"\"\n",
        "\n",
        "  num_heads: int\n",
        "  num_kv_heads: int\n",
        "  features: int\n",
        "  head_dim: int\n",
        "  attn_type: AttentionType\n",
        "  query_pre_attn_scalar: float\n",
        "  rope_base_frequency: int = DEFAULT_ROPE_BASE_FREQUENCY\n",
        "  rope_scale_factor: float = DEFAULT_ROPE_SCALE_FACTOR\n",
        "  attn_logits_soft_cap: float | None = None\n",
        "  sliding_window_size: int | None = None\n",
        "  use_qk_norm: bool = False\n",
        "\n",
        "  @property\n",
        "  def use_qkv_einsum(self):\n",
        "    return self.num_kv_heads == self.num_heads\n",
        "\n",
        "  @property\n",
        "  def use_gqa(self):\n",
        "    return self.num_kv_heads != self.num_heads and self.num_kv_heads > 1\n",
        "\n",
        "  def setup(self):\n",
        "    self.attn_vec_einsum = _layers.Einsum(\n",
        "        shape=(self.num_heads, self.head_dim, self.features),\n",
        "    )\n",
        "\n",
        "    if self.use_qkv_einsum:\n",
        "      self.qkv_einsum = _layers.Einsum(\n",
        "          shape=(3, self.num_heads, self.features, self.head_dim),\n",
        "      )\n",
        "    else:\n",
        "      self.q_einsum = _layers.Einsum(\n",
        "          shape=(self.num_heads, self.features, self.head_dim),\n",
        "      )\n",
        "      self.kv_einsum = _layers.Einsum(\n",
        "          shape=(2, self.num_kv_heads, self.features, self.head_dim),\n",
        "      )\n",
        "    if self.use_qk_norm:\n",
        "      self._query_norm = _layers.RMSNorm()\n",
        "      self._key_norm = _layers.RMSNorm()\n",
        "\n",
        "  def __call__(\n",
        "      self,\n",
        "      x: jax.Array,\n",
        "      segment_pos: jax.Array,\n",
        "      cache: LayerCache | None,\n",
        "      attn_mask: jax.Array,\n",
        "  ) -> tuple[LayerCache | None, jax.Array]:\n",
        "    \"\"\"Applies multi-head attention to the inputs.\n",
        "\n",
        "    Args:\n",
        "      x: Input sequence of shape [batch_size, seq_len, embed_dim].\n",
        "      segment_pos: Input absolute positions of shape [batch_size, seq_len].\n",
        "      cache: KV cache or None.\n",
        "      attn_mask: Attention mask of shape [batch_size, seq_len, cache_size].\n",
        "\n",
        "    Returns:\n",
        "      cache: Updated attention KV cache.\n",
        "      outputs: Output sequence of shape [batch_size, seq_len, embed_dim].\n",
        "    \"\"\"\n",
        "    if self.use_qkv_einsum:\n",
        "      # [batch_size, seq_len, num_heads, head_dim]\n",
        "      query_proj, key_proj, value_proj = self.qkv_einsum('BTD,SNDH->SBTNH', x)\n",
        "    else:\n",
        "      query_proj = self.q_einsum('BTD,NDH->BTNH', x)\n",
        "      key_proj, value_proj = self.kv_einsum('BSD,CKDH->CBSKH', x)\n",
        "\n",
        "    query_proj = jax.lax.stop_gradient(query_proj)\n",
        "    key_proj = jax.lax.stop_gradient(key_proj)\n",
        "\n",
        "    if self.use_qk_norm:\n",
        "      query_proj = self._query_norm(query_proj)\n",
        "      key_proj = self._key_norm(key_proj)\n",
        "\n",
        "    query_proj = _positional_embeddings.apply_rope(\n",
        "        query_proj,\n",
        "        segment_pos,\n",
        "        base_frequency=self.rope_base_frequency,\n",
        "        scale_factor=self.rope_scale_factor,\n",
        "    )\n",
        "    query_scaled = query_proj * self.query_pre_attn_scalar\n",
        "\n",
        "    key_proj = _positional_embeddings.apply_rope(\n",
        "        key_proj,\n",
        "        segment_pos,\n",
        "        base_frequency=self.rope_base_frequency,\n",
        "        scale_factor=self.rope_scale_factor,\n",
        "    )\n",
        "\n",
        "    query_proj = jax.lax.stop_gradient(query_proj)\n",
        "    key_proj = jax.lax.stop_gradient(key_proj)\n",
        "\n",
        "    # Cache is left aligned.\n",
        "    # Save the KV values to the cache.\n",
        "    if cache is not None:\n",
        "      end_index = cache['end_index'][0]\n",
        "      cache_size = cache['v'].shape[1]\n",
        "      slice_indices = (0, end_index % cache_size, 0, 0)\n",
        "\n",
        "      # [batch_size, cache_size, num_heads, head_dim]\n",
        "      value_proj = jax.lax.dynamic_update_slice(\n",
        "          cache['v'],\n",
        "          value_proj,\n",
        "          slice_indices,\n",
        "      )\n",
        "\n",
        "      # [batch_size, cache_size, num_heads, head_dim]\n",
        "      key_proj = jax.lax.dynamic_update_slice(\n",
        "          cache['k'], key_proj, slice_indices\n",
        "      )\n",
        "\n",
        "    if self.use_gqa:\n",
        "      # Reshape matrices to enable einsums over groups.\n",
        "      b, t, kg, h = query_scaled.shape\n",
        "      query_scaled = query_scaled.reshape(\n",
        "          (b, t, self.num_kv_heads, int(kg / self.num_kv_heads), h)\n",
        "      )\n",
        "      logits = jnp.einsum('BTKGH,BSKH->BTKGS', query_scaled, key_proj)\n",
        "      b, t, k, g, s = logits.shape\n",
        "      logits = logits.reshape((b, t, k * g, s))\n",
        "    else:\n",
        "      # [batch_size, seq_len, num_heads, cache_size]\n",
        "      # If cache is None, then cache_size = seq_len.\n",
        "      logits = jnp.einsum('BTNH,BSNH->BTNS', query_scaled, key_proj)\n",
        "\n",
        "    if self.attn_logits_soft_cap is not None:\n",
        "      logits = jnp.tanh(logits / self.attn_logits_soft_cap)\n",
        "      logits = logits * self.attn_logits_soft_cap\n",
        "\n",
        "    if self.attn_type == AttentionType.LOCAL_SLIDING:\n",
        "      if self.sliding_window_size is None:\n",
        "        raise ValueError(\n",
        "            'Sliding_window_size must be set if Local Sliding attention type'\n",
        "        )\n",
        "      sliding_mask = _create_sliding_mask(\n",
        "          segment_pos,\n",
        "          end_index=cache['end_index'][0] if cache is not None else 0,\n",
        "          # Derive cache length from attn_mask shape in case cache is None\n",
        "          cache_len=attn_mask.shape[-1],\n",
        "          sliding_window_size=self.sliding_window_size,\n",
        "      )\n",
        "      # [batch_size, seq_len, cache_size]\n",
        "      attn_mask *= sliding_mask\n",
        "\n",
        "    # [batch_size, seq_len, num_heads, cache_size]\n",
        "    padded_logits = jnp.where((jnp.expand_dims(attn_mask, -2)), logits, K_MASK)\n",
        "\n",
        "    # Multi-head attention matrices.\n",
        "    # [batch_size, seq_len, num_heads, cache_size]\n",
        "    probs = jax.nn.softmax(padded_logits, axis=-1).astype(key_proj.dtype)\n",
        "\n",
        "    if self.use_gqa:\n",
        "      # Reshape matrices to enable einsums over groups.\n",
        "      b, t, kg, h = probs.shape\n",
        "      probs = probs.reshape(\n",
        "          (b, t, self.num_kv_heads, int(kg / self.num_kv_heads), h)\n",
        "      )\n",
        "      # encoded = jnp.einsum('BTKGS,BSKH->BTKGH', probs, value_proj)\n",
        "      encoded = jnp.einsum('BTKGS,BSKH->BTKGH', jax.lax.stop_gradient(probs), value_proj)\n",
        "      b, t, k, g, h = encoded.shape\n",
        "      encoded = encoded.reshape((b, t, k * g, h))\n",
        "    else:\n",
        "      # [batch_size, seq_len, num_heads, head_dim]\n",
        "      # encoded = jnp.einsum('BTNS,BSNH->BTNH', probs, value_proj)\n",
        "      encoded = jnp.einsum('BTNS,BSNH->BTNH', jax.lax.stop_gradient(probs), value_proj)\n",
        "\n",
        "    # [batch_size, seq_len, features]\n",
        "    attn_output = self.attn_vec_einsum('BTNH,NHD->BTD', encoded)\n",
        "\n",
        "    if cache is not None:\n",
        "      seq_len = x.shape[1]\n",
        "      new_cache = {\n",
        "          # [batch_size, cache_size, num_heads, head_dim]\n",
        "          'v': value_proj,\n",
        "          # [batch_size, cache_size, num_heads, head_dim]\n",
        "          'k': key_proj,\n",
        "          # [batch_size]\n",
        "          'end_index': cache['end_index'] + seq_len,\n",
        "      }\n",
        "    else:\n",
        "      new_cache = None\n",
        "\n",
        "    return new_cache, attn_output\n",
        "from jax.typing import ArrayLike\n",
        "def gelu_detached(x_arr: ArrayLike, approximate: bool = True) -> Array:\n",
        "  r\"\"\"Gaussian error linear unit activation function.\n",
        "\n",
        "  If ``approximate=False``, computes the element-wise function:\n",
        "\n",
        "  .. math::\n",
        "    \\mathrm{gelu}(x) = \\frac{x}{2} \\left(\\mathrm{erfc} \\left(\n",
        "      \\frac{-x}{\\sqrt{2}} \\right) \\right)\n",
        "\n",
        "  If ``approximate=True``, uses the approximate formulation of GELU:\n",
        "\n",
        "  .. math::\n",
        "    \\mathrm{gelu}(x) = \\frac{x}{2} \\left(1 + \\mathrm{tanh} \\left(\n",
        "      \\sqrt{\\frac{2}{\\pi}} \\left(x + 0.044715 x^3 \\right) \\right) \\right)\n",
        "\n",
        "  For more information, see `Gaussian Error Linear Units (GELUs)\n",
        "  <https://arxiv.org/abs/1606.08415>`_, section 2.\n",
        "\n",
        "  Args:\n",
        "    x: input array\n",
        "    approximate: whether to use the approximate or exact formulation.\n",
        "  \"\"\"\n",
        "#   [x_arr] = numpy_util.promote_args_inexact(\"gelu\", x)\n",
        "  import numpy as np\n",
        "  if approximate:\n",
        "    sqrt_2_over_pi = np.sqrt(2 / np.pi).astype(x_arr.dtype)\n",
        "    cdf = 0.5 * (1.0 + jnp.tanh(sqrt_2_over_pi * (x_arr + 0.044715 * (x_arr ** 3))))\n",
        "    cdf = jax.lax.stop_gradient(cdf)\n",
        "    return x_arr * cdf\n",
        "  else:\n",
        "    sqrt_half = np.sqrt(0.5).astype(x_arr.dtype)\n",
        "    return jnp.array(\n",
        "        0.5 * x_arr * jax.lax.stop_gradient(lax.erfc(-x_arr * sqrt_half)), dtype=x_arr.dtype\n",
        "    )\n",
        "\n",
        "  @classmethod\n",
        "  def init_cache(\n",
        "      cls,\n",
        "      cache_size: int,\n",
        "      num_heads: int,\n",
        "      head_dim: int,\n",
        "      batch_size: int,\n",
        "      dtype: jnp.dtype = jnp.bfloat16,\n",
        "  ) -> LayerCache:\n",
        "    del cls  # not used\n",
        "    return {\n",
        "        'v': jnp.zeros(\n",
        "            (batch_size, cache_size, num_heads, head_dim), dtype=dtype\n",
        "        ),\n",
        "        'k': jnp.zeros(\n",
        "            (batch_size, cache_size, num_heads, head_dim), dtype=dtype\n",
        "        ),\n",
        "        'end_index': jnp.zeros((batch_size,), dtype=jnp.int32),\n",
        "    }\n",
        "\n",
        "\n",
        "class FeedForward(nn.Module):\n",
        "  \"\"\"Feed forward module.\"\"\"\n",
        "\n",
        "  features: int  # features = embed_dim\n",
        "  hidden_dim: int\n",
        "  transpose_gating_einsum: bool\n",
        "\n",
        "  @nn.compact\n",
        "  def __call__(self, x):\n",
        "    \"\"\"Applies the feed forward module.\n",
        "\n",
        "    Args:\n",
        "      x: Input sequence of shape [batch_size, seq_len, features].\n",
        "\n",
        "    Returns:\n",
        "      Output sequence of shape [batch_size, seq_len, features].\n",
        "    \"\"\"\n",
        "    # Some versions use an alternate parameter ordering that\n",
        "    # transposes hidden_dim and features.\n",
        "    if self.transpose_gating_einsum:\n",
        "      eq = '...F,NHF->...NH'\n",
        "      gating = _layers.Einsum(\n",
        "          shape=(2, self.hidden_dim, self.features),\n",
        "          weight_name='gating_einsum',\n",
        "      )\n",
        "    else:\n",
        "      eq = '...F,NFH->...NH'\n",
        "      gating = _layers.Einsum(\n",
        "          shape=(2, self.features, self.hidden_dim),\n",
        "          weight_name='gating_einsum',\n",
        "      )\n",
        "\n",
        "    # Use the same scope for backwards compatibility with existing checkpoints\n",
        "    # created before using `_layers.Einsum` here.\n",
        "    nn.share_scope(self, gating)\n",
        "\n",
        "    # [batch_size, seq_len, 2, hidden_dim]\n",
        "    gate = gating(eq, x)\n",
        "    # [batch_size, seq_len, hidden_dim]\n",
        "    # activations = nn.gelu(gate[..., 0, :]) * gate[..., 1, :]\n",
        "\n",
        "    activations =  jax.lax.stop_gradient(gelu_detached(gate[..., 0, :])) * gate[..., 1, :]\n",
        "\n",
        "    # Project back from hidden_dim to features.\n",
        "    linear = _layers.Einsum(\n",
        "        shape=(self.hidden_dim, self.features),\n",
        "        weight_name='linear',\n",
        "    )\n",
        "    nn.share_scope(self, linear)\n",
        "\n",
        "    # [batch_size, seq_len, features]\n",
        "    outputs = linear('...H,HF->...F', activations)\n",
        "\n",
        "    return outputs\n",
        "\n",
        "\n",
        "class Block(nn.Module):\n",
        "  \"\"\"Transformer block.\"\"\"\n",
        "\n",
        "  num_heads: int\n",
        "  num_kv_heads: int\n",
        "  embed_dim: int\n",
        "  head_dim: int\n",
        "  hidden_dim: int\n",
        "  use_post_attn_norm: bool\n",
        "  use_post_ffw_norm: bool\n",
        "  attn_type: AttentionType\n",
        "  query_pre_attn_scalar: float\n",
        "  transpose_gating_einsum: bool\n",
        "  rope_base_frequency: int = DEFAULT_ROPE_BASE_FREQUENCY\n",
        "  rope_scale_factor: float = DEFAULT_ROPE_SCALE_FACTOR\n",
        "  attn_logits_soft_cap: float | None = None\n",
        "  sliding_window_size: int | None = None\n",
        "  use_qk_norm: bool = False\n",
        "\n",
        "  def setup(self):\n",
        "    self.pre_attention_norm = _layers.RMSNorm()\n",
        "\n",
        "    self.attn = Attention(\n",
        "        num_heads=self.num_heads,\n",
        "        features=self.embed_dim,\n",
        "        head_dim=self.head_dim,\n",
        "        num_kv_heads=self.num_kv_heads,\n",
        "        attn_type=self.attn_type,\n",
        "        query_pre_attn_scalar=self.query_pre_attn_scalar,\n",
        "        rope_base_frequency=self.rope_base_frequency,\n",
        "        rope_scale_factor=self.rope_scale_factor,\n",
        "        attn_logits_soft_cap=self.attn_logits_soft_cap,\n",
        "        sliding_window_size=self.sliding_window_size,\n",
        "        use_qk_norm=self.use_qk_norm,\n",
        "    )\n",
        "\n",
        "    self.post_attention_norm = None\n",
        "    if self.use_post_attn_norm:\n",
        "      self.post_attention_norm = _layers.RMSNorm()\n",
        "\n",
        "    self.pre_ffw_norm = _layers.RMSNorm()\n",
        "\n",
        "    self.mlp = FeedForward(\n",
        "        features=self.embed_dim,\n",
        "        hidden_dim=self.hidden_dim,\n",
        "        transpose_gating_einsum=self.transpose_gating_einsum,\n",
        "    )\n",
        "\n",
        "    self.post_ffw_norm = None\n",
        "    if self.use_post_ffw_norm:\n",
        "      self.post_ffw_norm = _layers.RMSNorm()\n",
        "\n",
        "  def __call__(\n",
        "      self,\n",
        "      x: jax.Array,\n",
        "      segment_pos: jax.Array,\n",
        "      cache: LayerCache | None,\n",
        "      attn_mask: jax.Array,\n",
        "  ) -> tuple[LayerCache | None, jax.Array]:\n",
        "    \"\"\"Applies the block to the inputs.\n",
        "\n",
        "    Args:\n",
        "      x: Input sequence of shape [batch_size, seq_len, embed_dim].\n",
        "      segment_pos: Input absolute positions of shape [batch_size, seq_len].\n",
        "      cache: KV cache or None.\n",
        "      attn_mask: Attention mask of shape [batch_size, seq_len, cache_size].\n",
        "\n",
        "    Returns:\n",
        "      cache: Updated attention KV cache.\n",
        "      outputs: Output sequence of shape [batch_size, seq_len, embed_dim].\n",
        "    \"\"\"\n",
        "    inputs_normalized = self.pre_attention_norm(x)\n",
        "\n",
        "    # attn_output.shape = [batch_size, seq_len, embed_dim]\n",
        "    # cache[\"k\"].shape = [batch_size, cache_size, num_heads, head_dim]\n",
        "    # cache[\"v\"].shape = [batch_size, cache_size, num_heads, head_dim]\n",
        "    # cache[\"end_index\"].shape = [batch_size]\n",
        "    cache, attn_output = self.attn(\n",
        "        inputs_normalized,\n",
        "        segment_pos,\n",
        "        cache,\n",
        "        attn_mask,\n",
        "    )\n",
        "\n",
        "    if self.post_attention_norm is not None:\n",
        "      attn_output = self.post_attention_norm(attn_output)\n",
        "\n",
        "    attn_output += x\n",
        "\n",
        "    outputs = self.pre_ffw_norm(attn_output)\n",
        "\n",
        "    outputs = self.mlp(outputs)\n",
        "\n",
        "    if self.post_ffw_norm is not None:\n",
        "      outputs = self.post_ffw_norm(outputs)\n",
        "\n",
        "    outputs += attn_output\n",
        "\n",
        "    return cache, outputs\n"
      ],
      "metadata": {
        "id": "XFMCs7GPJigy",
        "cellView": "form",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "dbc8e067-1c41-4bd8-ba1c-e8788deee65d"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Overwriting /usr/local/lib/python3.11/dist-packages/gemma/gm/nn/_modules.py\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# Common imports\n",
        "import os\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "\n",
        "# Gemma imports\n",
        "from gemma import gm"
      ],
      "metadata": {
        "id": "ybxouMkhyuRs"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "import gemma\n",
        "gemma.__version__"
      ],
      "metadata": {
        "id": "A4yFw4bk1uY4",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 35
        },
        "outputId": "1cbc5dbd-9b37-4da4-dc5a-4c0887deaf81"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "'3.1.0'"
            ],
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            }
          },
          "metadata": {},
          "execution_count": 8
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# Try these JAX memory management approaches\n",
        "import os\n",
        "\n",
        "# Method 1: Limit JAX memory pre-allocation\n",
        "os.environ['XLA_PYTHON_CLIENT_PREALLOCATE'] = 'false'\n",
        "# os.environ['XLA_PYTHON_CLIENT_MEM_FRACTION'] = '0.5'"
      ],
      "metadata": {
        "id": "HGRXKhzPUftE"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# gm.ckpts.CheckpointPath.GEMMA3_1B_IT\n",
        "# model = gm.nn.Gemma3_1B()\n",
        "\n",
        "# params = gm.ckpts.load_params(gm.ckpts.CheckpointPath.GEMMA3_1B_IT, text_only=True)#, dtype=jnp.float32)"
      ],
      "metadata": {
        "id": "abx6eEuK0GYf"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "model = gm.nn.Gemma3_4B()\n",
        "\n",
        "params = gm.ckpts.load_params(gm.ckpts.CheckpointPath.GEMMA3_4B_IT, text_only=True)#, dtype=jnp.float32)"
      ],
      "metadata": {
        "id": "00UhDv-kUl0o"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [],
      "metadata": {
        "id": "nK00noOMyv7W"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Convert all parameters to float32\n",
        "def convert_to_float32(params):\n",
        "    \"\"\"Convert all parameters from bfloat16 to float32\"\"\"\n",
        "    def convert_param(x):\n",
        "        if hasattr(x, 'dtype') and x.dtype == jnp.bfloat16:\n",
        "            return x.astype(jnp.float32)\n",
        "        return x\n",
        "\n",
        "    return jax.tree_map(convert_param, params)\n",
        "\n",
        "params = convert_to_float32(params)"
      ],
      "metadata": {
        "id": "aRuJUttXU96n",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "bd5fc647-b584-49d2-f45d-17f960f07057"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/tmp/ipython-input-12-541248995.py:9: DeprecationWarning: jax.tree_map is deprecated: use jax.tree.map (jax v0.4.25 or newer) or jax.tree_util.tree_map (any JAX version).\n",
            "  return jax.tree_map(convert_param, params)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "gm.__file__"
      ],
      "metadata": {
        "id": "G-2qL26LzEcE",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 35
        },
        "outputId": "48a1c040-a92b-451d-81bf-a7b01cbb2b87"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "'/usr/local/lib/python3.11/dist-packages/gemma/gm/__init__.py'"
            ],
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "string"
            }
          },
          "metadata": {},
          "execution_count": 13
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "tokenizer = gm.text.Gemma3Tokenizer()"
      ],
      "metadata": {
        "id": "k-DVmVlRzIhz"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# prompt = tokenizer.encode('Describe Paris', add_bos=True)\n",
        "# # prompt = tokenizer.encode('Paris is where you want it to be, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize.\\\n",
        "# # Paris is where you want it to be, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize.\\\n",
        "# # Paris is where you want it to be, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize.', add_bos=True)\n",
        "# # # Paris is where you want it to be, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize.', add_bos=True)\n",
        "# # # Paris is where you want it to be, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize, always twirling and swirling to the golden moonlight, no matter what happens, and if you wanted to do that anyway, personally i think that would be a fine case to make, always twirling and swirling to the golden moonlight, no matter what happens, even for the final matchup which you will not even realize.', add_bos=True)\n",
        "# prompt = jnp.asarray(prompt)\n",
        "\n",
        "# embeddings = jnp.asarray([params['embedder']['input_embedding'][i] for i in prompt])\n",
        "# embeddings = jax.numpy.expand_dims(embeddings,0)#.shape"
      ],
      "metadata": {
        "id": "yqNP8kkVzYA7"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "text = '''The Library of Babel, by Jorge Luis Borges (1941) By this art you may contemplate the variations of the 23 letters... The Anatomy of Melancholy, part 2, sect. II, mem. IV The universe (which others call the'''# Library) is composed of an indefinite and perhaps infinite number of hexagonal galleries, with vast air shafts between, surrounded by very low railings. From any of the hexagons one can see, interminably, the upper and lower floors. The distribution of the galleries is invariable. Twenty shelves, five long shelves per side, cover all the sides except two; their height, which is the distance from floor to ceiling, scarcely exceeds that of a normal bookcase. One of the free sides leads to a narrow hallway which opens onto another gallery, identical to the first and to all the rest. To the left and right of the hallway there are two very small closets. In the first, one may sleep standing up; in the other, satisfy one's fecal necessities. Also through here passes a spiral stairway, which sinks abysmally and soars upwards to remote distances. In the hallway there is a mirror which faithfully duplicates all appearances. Men usually infer from this mirror that the Library is not infinite (if it were, why this illusory duplication?); I prefer to dream that its polished surfaces represent and promise the infinite ... Light is provided by some spherical fruit which bear the name of lamps. There are two, transversally placed, in each hexagon. The light they emit is insufficient, incessant. Like all men of the Library, I have traveled in my youth; I have wandered in search of a book, perhaps the catalogue of catalogues; now that my eyes can hardly decipher what I write, I am preparing to die just a few leagues from the hexagon in which I was born. Once I am dead, there will be no lack of pious hands to throw me over the railing; my grave will be the fathomless air; my body will sink endlessly and decay and dissolve in the wind generated by the fall, which is infinite. I say that the Library is unending. The idealists argue that the hexagonal rooms are a necessary form of absolute space or, at least, of our intuition of space. They reason that a triangular or pentagonal room is inconceivable. (The mystics claim that their ecstasy reveals to them a circular chamber containing a great circular book, whose spine is continuous and which follows the complete circle of the walls; but their testimony is suspect; their words, obscure. This cyclical book is God.) Let it suffice now for me to repeat the classic dictum: The Library is a sphere whose exact center is any one of its hexagons and whose circumference is inaccessible. There are five shelves for each of the hexagon's walls; each shelf contains thirty-five books of uniform format; each book is of four hundred and ten pages; each page, of forty lines, each line, of some eighty letters which are black in color. There are also letters on the spine of each book; these letters do not indicate or prefigure what the pages will say. I know that this incoherence at one time seemed mysterious. Before summarizing the solution (whose discovery, in spite of its tragic projections, is perhaps the capital fact in history) I wish to recall a few axioms. First: The Library exists ab aeterno. This truth, whose immediate corollary is the future eternity of the world, cannot be placed in doubt by any reasonable mind. Man, the imperfect librarian, may be the product of chance or of malevolent demiurgi; the universe, with its elegant endowment of shelves, of enigmatical volumes, of inexhaustible stairways for the traveler and latrines for the seated librarian, can only be the work of a god. To perceive the distance between the divine and the human, it is enough to compare these crude wavering symbols which my fallible hand scrawls on the cover of a book, with the organic letters inside: punctual, delicate, perfectly black, inimitably symmetrical. Second: The orthographical symbols are twenty-five in number. 1 This finding made it possible, three hundred years ago, to formulate a general theory of the Library and solve satisfactorily the problem which no conjecture had deciphered: the formless and chaotic nature of almost all the books. One which my father saw in a hexagon on circuit fifteen ninety-four was made up of the letters MCV, perversely repeated from the first line to the last. Another (very much consulted in this area) is a mere labyrinth of letters, but the next-to-last page says Oh time thy pyramids. This much is already known: for every sensible line of straightforward statement, there are leagues of senseless cacophonies, verbal jumbles and incoherences. (I know of an uncouth region whose librarians repudiate the vain and superstitious custom of finding a meaning in books and equate it with that of finding a meaning in dreams or in the chaotic lines of one's '''"
      ],
      "metadata": {
        "id": "fjmCjx_eOOEZ"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# 40\n",
        "text = '''By this art you may contemplate the variations of the 23 letters... The Anatomy of Melancholy, part 2, sect. II, mem. IV The universe (which others call the'''# Library) is composed of an indefinite and perhaps infinite number of hexagonal galleries, with vast air shafts between, surrounded by very low railings. From any of the hexagons one can see, interminably, the upper and lower floors. The distribution of the galleries is invariable. Twenty shelves, five long shelves per side, cover all the sides except two; their height, which is the distance from floor to ceiling, scarcely exceeds that of a normal bookcase. One of the free sides leads to a narrow hallway which opens onto another gallery, identical to the first and to all the rest. To the left and right of the hallway there are two very small closets. In the first, one may sleep standing up; in the other, satisfy one's fecal necessities. Also through here passes a spiral stairway, which sinks abysmally and soars upwards to remote distances. In the hallway there is a mirror which faithfully duplicates all appearances. Men usually infer from this mirror that the Library is not infinite (if it were, why this illusory duplication?); I prefer to dream that its polished surfaces represent and promise the infinite ... Light is provided by some spherical fruit which bear the name of lamps. There are two, transversally placed, in each hexagon. The light they emit is insufficient, incessant. Like all men of the Library, I have traveled in my youth; I have wandered in search of a book, perhaps the catalogue of catalogues; now that my eyes can hardly decipher what I write, I am preparing to die just a few leagues from the hexagon in which I was born. Once I am dead, there will be no lack of pious hands to throw me over the railing; my grave will be the fathomless air; my body will sink endlessly and decay and dissolve in the wind generated by the fall, which is infinite. I say that the Library is unending. The idealists argue that the hexagonal rooms are a necessary form of absolute space or, at least, of our intuition of space. They reason that a triangular or pentagonal room is inconceivable. (The mystics claim that their ecstasy reveals to them a circular chamber containing a great circular book, whose spine is continuous and which follows the complete circle of the walls; but their testimony is suspect; their words, obscure. This cyclical book is God.) Let it suffice now for me to repeat the classic dictum: The Library is a sphere whose exact center is any one of its hexagons and whose circumference is inaccessible. There are five shelves for each of the hexagon's walls; each shelf contains thirty-five books of uniform format; each book is of four hundred and ten pages; each page, of forty lines, each line, of some eighty letters which are black in color. There are also letters on the spine of each book; these letters do not indicate or prefigure what the pages will say. I know that this incoherence at one time seemed mysterious. Before summarizing the solution (whose discovery, in spite of its tragic projections, is perhaps the capital fact in history) I wish to recall a few axioms. First: The Library exists ab aeterno. This truth, whose immediate corollary is the future eternity of the world, cannot be placed in doubt by any reasonable mind. Man, the imperfect librarian, may be the product of chance or of malevolent demiurgi; the universe, with its elegant endowment of shelves, of enigmatical volumes, of inexhaustible stairways for the traveler and latrines for the seated librarian, can only be the work of a god. To perceive the distance between the divine and the human, it is enough to compare these crude wavering symbols which my fallible hand scrawls on the cover of a book, with the organic letters inside: punctual, delicate, perfectly black, inimitably symmetrical. Second: The orthographical symbols are twenty-five in number. 1 This finding made it possible, three hundred years ago, to formulate a general theory of the'''\n"
      ],
      "metadata": {
        "id": "QJotX06C1o0b"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# 52\n",
        "text = '''By this art you may contemplate the variations of the 23 letters... The Anatomy of Melancholy, part 2, sect. II, mem. IV The universe (which others call the Library) is composed of an indefinite and perhaps infinite number of hexagonal galleries, with vast air shafts between, surrounded by very low railings. From any of the hexagons one can see, interminably, the upper and lower floors. The distribution of the galleries is invariable. Twenty shelves, five long shelves per side, cover all the sides except two; their height, which is the distance from floor to ceiling, scarcely exceeds that of a normal bookcase. One of the free sides leads to a narrow hallway which opens onto another gallery, identical to the first and to all the rest. To the left and right of the hallway there are two very small closets. In the first, one may sleep standing up; in the other, satisfy one's fecal necessities. Also through here passes a spiral stairway, which sinks abysmally and soars upwards to remote distances. In the hallway there is a mirror which faithfully duplicates all appearances. Men usually infer from this mirror that the Library is not infinite (if it were, why this illusory duplication?); I prefer to dream that its polished surfaces represent and promise the infinite ... Light is provided by some spherical fruit which bear the name of lamps. There are two, transversally placed, in each hexagon. The light they emit is insufficient, incessant. Like all men of the Library, I have traveled in my youth; I have wandered in search of a book, perhaps the catalogue of catalogues; now that my eyes can hardly decipher what I write, I am preparing to die just a few leagues from the hexagon in which I was born. Once I am dead, there will be no lack of pious hands to throw me over the railing; my grave will be the fathomless air; my body will sink endlessly and decay and dissolve in the wind generated by the fall, which is infinite. I say that the Library is unending. The idealists argue that the hexagonal rooms are a necessary form of absolute'''# space or, at least, of our intuition of space. They reason that a triangular or pentagonal room is inconceivable. (The mystics claim that their ecstasy reveals to them a circular chamber containing a'''### great circular book, whose spine is continuous and which follows the complete circle of the walls; but their testimony is suspect; their words, obscure. This cyclical book is God.) Let it suffice now for me to repeat the classic dictum: The Library is a sphere whose exact center is any one of its hexagons and whose circumference is inaccessible. There are five shelves for each of the hexagon's walls; each shelf contains thirty-five books of uniform format; each book is of four hundred and ten pages; each page, of forty lines, each line, of some eighty letters which are black in color. There are also letters on the spine of each book; these letters do not indicate or prefigure what the pages will say. I know that this incoherence at one time seemed mysterious. Before summarizing the solution (whose discovery, in spite of its tragic projections, is perhaps the capital fact in history) I wish to recall a few axioms. First: The Library exists ab aeterno. This truth, whose immediate corollary is the future eternity of the'''# world, cannot be placed in doubt by any reasonable mind. Man, the imperfect librarian, may be the product of chance or of malevolent demiurgi; the universe, with its elegant endowment of shelves, of enigmatical volumes, of inexhaustible stairways for the traveler and latrines for the seated librarian, can only be the work of a god. To perceive the distance between the divine and the human, it is enough to compare these crude wavering symbols which my fallible hand scrawls on the cover of a book, with the organic letters inside: punctual, delicate, perfectly black, inimitably symmetrical. Second: The orthographical symbols are twenty-five in number. 1 This finding made it possible, three hundred years ago, to formulate a general theory of the'''\n"
      ],
      "metadata": {
        "id": "9aGmEz9AV_yi"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# # #852\n",
        "\n",
        "# # 92\n",
        "# text = '''By this art you may contemplate the variations of the 23 letters... The Anatomy of Melancholy, part 2, sect. II, mem. IV The universe (which others call the Library) is composed of an indefinite and perhaps infinite number of hexagonal galleries, with vast air shafts between, surrounded by very low railings. From any of the hexagons one can see, interminably, the upper and lower floors. The distribution of the galleries is '''# invariable. Twenty shelves, five long shelves per side, cover all the '''#sides except two; their height, which is the distance from floor to ceiling, scarcely exceeds that of a normal bookcase. One of the free sides leads to a narrow hallway which opens onto another gallery, identical to the first and to all the rest. To the left and right of the hallway there are two very small closets. In the first, one may sleep standing up; in the other, satisfy one's fecal necessities. Also through here passes a '''#spiral stairway, which sinks abysmally and soars upwards to remote distances. In the hallway there is a mirror which faithfully duplicates all appearances. Men usually infer from this mirror that the Library is not infinite (if it were, why this illusory duplication?); I prefer to dream that its polished surfaces represent and promise the infinite ... Light is provided by some spherical fruit which bear the name of lamps. There are two, transversally placed, in each hexagon. The light they emit is insufficient, incessant. Like all men of the Library, I have traveled in my youth; I have wandered in search of a book, perhaps the catalogue of catalogues; now that my eyes can hardly decipher what I write, I am preparing to die just a few leagues from the hexagon in which I was born. Once I am dead, there will be no lack of pious hands to throw me over the railing; my grave will be the fathomless air; my body will sink endlessly and decay and dissolve in the wind generated by the fall, which is infinite. I say that the Library is unending. The idealists argue that the hexagonal rooms are a necessary form of absolute space or, at least, of our intuition of space. They reason that a triangular or pentagonal room is inconceivable. (The mystics claim that their ecstasy reveals to them a circular chamber containing a great circular book, whose spine is continuous and which follows the complete circle of the walls; but their testimony is suspect; their words, obscure. This cyclical book is God.) Let it suffice now for me to repeat the classic dictum: The Library is a sphere whose exact center is any one of its hexagons and whose circumference is inaccessible. There are five shelves for each of the hexagon's walls; each shelf contains thirty-five books of uniform format; each book is of four hundred and ten pages; each page, of forty lines, each line, of some eighty letters which are black in color. There are also letters on the spine of each book; these letters do not indicate or prefigure what the pages will say. I know that this incoherence at one time seemed mysterious. Before summarizing the solution (whose discovery, in spite of its tragic projections, is perhaps the capital fact in history) I wish to recall a few axioms. First: The Library exists ab aeterno. This truth, whose immediate corollary is the future eternity of the world, cannot be placed in doubt by any reasonable mind. Man, the imperfect librarian, may be the product of chance or of malevolent demiurgi; the universe, with its elegant endowment of shelves, of enigmatical volumes, of inexhaustible stairways for the traveler and latrines for the seated librarian, can only be the work of a god. To perceive the distance between the divine and the human, it is enough to compare these crude wavering symbols which my fallible hand scrawls on the cover of a book, with the organic letters inside: punctual, delicate, perfectly black, inimitably symmetrical. Second: The orthographical symbols are twenty-five in number. 1 This finding made it possible, three hundred years ago, to formulate a general theory of the'''\n",
        "# # #  Library and solve satisfactorily the problem which no conjecture had deciphered: the formless and chaotic nature of almost all the books. One which my father saw in a hexagon on circuit fifteen ninety-four was made up of the letters MCV, perversely repeated from the first line to the last. Another (very much consulted in this area) is a mere labyrinth of letters, but the next-to-last page says Oh time thy pyramids. This much is already known: for every sensible line of straightforward statement, there are leagues of senseless cacophonies, verbal jumbles and incoherences. (I know of an uncouth region whose librarians repudiate the vain and superstitious custom of finding a meaning in books and equate it with that of finding a meaning in dreams or in the chaotic lines of one's palm ... They admit that the inventors of this writing imitated the twenty-five'''"
      ],
      "metadata": {
        "id": "cDiMcLj2FWau"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "#@title Babel\n",
        "# natural symbols, but maintain that this application is accidental and that the books signify nothing in themselves. This dictum, we shall see, is not entirely fallacious.) For a long time it was believed that these impenetrable books corresponded to past or remote languages. It is true that the most ancient men, the first librarians, used a language quite different from the one we now speak; it is true that a few miles to the right the tongue is dialectical and that ninety floors farther up, it is incomprehensible. All this, I repeat, is true, but four hundred and ten pages of inalterable MCV's cannot correspond to any language, no matter how dialectical or rudimentary it may be. Some insinuated that each letter could influence the following one and that the value of MCV in the third line of page 71 was not the one the same series may have in another position on another page, but this vague thesis did not prevail. Others thought of cryptographs; generally, this conjecture has been accepted, though not in the sense in which it was formulated by its originators. Five hundred years ago, the chief of an upper hexagon2 came upon a book as confusing as the others, but which had nearly two pages of homogeneous lines. He showed his find to a wandering decoder who told him the lines were written in Portuguese; others said they were Yiddish. Within a century, the language was established: a Samoyedic Lithuanian dialect of Guarani, with classical Arabian inflections. The content was also deciphered: some notions of combinative analysis, illustrated with examples of variations with unlimited repetition. These examples made it possible for a librarian of genius to discover the fundamental law of the Library. This thinker observed that all the books, no matter how diverse they might be, are made up of the same elements: the space, the period, the comma, the twenty-two letters of the alphabet. He also alleged a fact which travelers have confirmed: In the vast Library there are no two identical books. From these two incontrovertible premises he deduced that the Library is total and that its shelves register all the possible combinations of the twenty-odd orthographical symbols (a number which, though extremely vast, is not infinite): Everything: the minutely detailed history of the future, the archangels' autobiographies, the faithful catalogues of the Library, thousands and thousands of false catalogues, the demonstration of the fallacy of those catalogues, the demonstration of the fallacy of the true catalogue, the Gnostic gospel of Basilides, the commentary on that gospel, the commentary on the commentary on that gospel, the true story of your death, the translation of every book in all languages, the interpolations of every book in all books. When it was proclaimed that the Library contained all books, the first impression was one of extravagant happiness. All men felt themselves to be the masters of an intact and secret treasure. There was no personal or world problem whose eloquent solution did not exist in some hexagon. The universe was justified, the universe suddenly usurped the unlimited dimensions of hope. At that time a great deal was said about the Vindications: books of apology and prophecy which vindicated for all time the acts of every man in the universe and retained prodigious arcana for his future. Thousands of the greedy abandoned their sweet native hexagons and rushed up the stairways, urged on by the vain intention of finding their Vindication. These pilgrims disputed in the narrow corridors, proferred dark curses, strangled each other on the divine stairways, flung the deceptive books into the air shafts, met their death cast down in a similar fashion by the inhabitants of remote regions. Others went mad ... The Vindications exist (I have seen two which refer to persons of the future, to persons who are perhaps not imaginary) but the searchers did not remember that the possibility of a man's finding his Vindication, or some treacherous variation thereof, can be computed as zero. At that time it was also hoped that a clarification of humanity's basic mysteries -- the origin of the Library and of time -- might be found. It is verisimilar that these grave mysteries could be explained in words: if the language of philosophers is not sufficient, the multiform Library will have produced the unprecedented language required, with its vocabularies and grammars. For four centuries now men have exhausted the hexagons ... There are official searchers, inquisitors. I have seen them in the performance of their function: they always arrive extremely tired from their journeys; they speak of a broken stairway which almost killed them; they talk with the librarian of galleries and stairs; sometimes they pick up the nearest volume and leaf through it, looking for infamous words. Obviously, no one expects to discover anything. As was natural, this inordinate hope was followed by an excessive depression. The certitude that some shelf in some hexagon held precious books and that these precious books were inaccessible, seemed almost intolerable. A blasphemous sect suggested that the searches should cease and that all men should juggle letters and symbols until they constructed, by an improbable gift of chance, these canonical books. The authorities were obliged to issue severe orders. The sect disappeared, but in my childhood I have seen old men who, for long periods of time, would hide in the latrines with some metal disks in a forbidden dice cup and feebly mimic the divine disorder. Others, inversely, believed that it was fundamental to eliminate useless works. They invaded the hexagons, showed credentials which were not always false, leafed through a volume with displeasure and condemned whole shelves: their hygienic, ascetic furor caused the senseless perdition of millions of books. Their name is execrated, but those who deplore the ``treasures'' destroyed by this frenzy neglect two notable facts. One: the Library is so enormous that any reduction of human origin is infinitesimal. The other: every copy is unique, irreplaceable, but (since the Library is total) there are always several hundred thousand imperfect facsimiles: works which differ only in a letter or a comma. Counter to general opinion, I venture to suppose that the consequences of the Purifiers' depredations have been exaggerated by the horror these fanatics produced. They were urged on by the delirium of trying to reach the books in the Crimson Hexagon: books whose format is smaller than usual, all-powerful, illustrated and magical. We also know of another superstition of that time: that of the Man of the Book. On some shelf in some hexagon (men reasoned) there must exist a book which is the formula and perfect compendium of all the rest: some librarian has gone through it and he is analogous to a god. In the language of this zone vestiges of this remote functionary's cult still persist. Many wandered in search of Him. For a century they have exhausted in vain the most varied areas. How could one locate the venerated and secret hexagon which housed Him? Someone proposed a regressive method: To locate book A, consult first book B which indicates A's position; to locate book B, consult first a book C, and so on to infinity ... In adventures such as these, I have squandered and wasted my years. It does not seem unlikely to me that there is a total book on some shelf of the universe3 ; I pray to the unknown gods that a man -- just one, even though it were thousands of years ago! -- may have examined and read it. If honor and wisdom and happiness are not for me, let them be for others. Let heaven exist, though my place be in hell. Let me be outraged and annihilated, but for one instant, in one being, let Your enormous Library be justified. The impious maintain that nonsense is normal in the Library and that the reasonable (and even humble and pure coherence) is an almost miraculous exception. They speak (I know) of the ``feverish Library whose chance volumes are constantly in danger of changing into others and affirm, negate and confuse everything like a delirious divinity.'' These words, which not only denounce the disorder but exemplify it as well, notoriously prove their authors' abominable taste and desperate ignorance. In truth, the Library includes all verbal structures, all variations permitted by the twenty-five orthographical symbols, but not a single example of absolute nonsense. It is useless to observe that the best volume of the many hexagons under my administration is entitled The Combed Thunderclap and another The Plaster Cramp and another Axaxaxas mlö. These phrases, at first glance incoherent, can no doubt be justified in a cryptographical or allegorical manner; such a justification is verbal and, ex hypothesi, already figures in the Library. I cannot combine some characters dhcmrlchtdj which the divine Library has not foreseen and which in one of its secret tongues do not contain a terrible meaning. No one can articulate a syllable which is not filled with tenderness and fear, which is not, in one of these languages, the powerful name of a god. To speak is to fall into tautology. This wordy and useless epistle already exists in one of the thirty volumes of the five shelves of one of the innumerable hexagons -- and its refutation as well. (An n number of possible languages use the same vocabulary; in some of them, the symbol library allows the correct definition a ubiquitous and lasting system of hexagonal galleries, but library is bread or pyramid or anything else, and these seven words which define it have another value. You who read me, are You sure of understanding my language?) The methodical task of writing distracts me from the present state of men. The certitude that everything has been written negates us or turns us into phantoms. I know of districts in which the young men prostrate themselves before books and kiss their pages in a barbarous manner, but they do not know how to decipher a single letter. Epidemics, heretical conflicts, peregrinations which inevitably degenerate into banditry, have decimated the population. I believe I have mentioned suicides, more and more frequent with the years. Perhaps my old age and fearfulness deceive me, but I suspect that the human species -- the unique species -- is about to be extinguished, but the Library will endure: illuminated, solitary, infinite, perfectly motionless, equipped with precious volumes, useless, incorruptible, secret. I have just written the word “infinite”.' I have not interpolated this adjective out of rhetorical habit; I say that it is not illogical to think that the world is infinite. Those who judge it to be limited postulate that in remote places the corridors and stairways and hexagons can conceivably come to an end -- which is absurd. Those who imagine it to be without limit forget that the possible number of books does have such a limit. I venture to suggest this solution to the ancient problem: The Library is unlimited and cyclical. If an eternal traveler were to cross it in any direction, after centuries he would see that the same volumes were repeated in the same disorder (which, thus repeated, would be an order: the Order). My solitude is gladdened by this' '''"
      ],
      "metadata": {
        "cellView": "form",
        "id": "0TWB5Al3QXMV"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "prompt = tokenizer.encode(text)\n",
        "prompt = jnp.asarray(prompt)\n",
        "\n",
        "embeddings = jnp.asarray([params['embedder']['input_embedding'][i] for i in prompt])\n",
        "embeddings = jax.numpy.expand_dims(embeddings,0)#.shape"
      ],
      "metadata": {
        "id": "XQ92Qg9hzzkV"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "embeddings.shape"
      ],
      "metadata": {
        "id": "gK7T4O_hIBDg",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "6350fc3f-6ee1-4833-ae67-a248e65ee317"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "(1, 427, 2560)"
            ]
          },
          "metadata": {},
          "execution_count": 22
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [],
      "metadata": {
        "id": "PBtag_EQ9lNl"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Detached Jacobian with full matrix to confirm linearity"
      ],
      "metadata": {
        "id": "AIBMMO1f9mIP"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "embeddings.shape"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "-rVR_MKY9q97",
        "outputId": "8c875ea2-fdc8-463d-dcf1-935a749e0110"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "(1, 427, 2560)"
            ]
          },
          "metadata": {},
          "execution_count": 23
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "# # # Run the model\n",
        "# out = model.apply(\n",
        "#     {'params': params},\n",
        "#     tokens=prompt,\n",
        "#     return_last_only=True,  # Only predict the last token\n",
        "# )"
      ],
      "metadata": {
        "id": "JO5jiiG0zcge"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# # Sample a token from the predicted logits\n",
        "# next_token = jax.random.categorical(\n",
        "#     jax.random.key(1),\n",
        "#     out.logits\n",
        "# )\n",
        "# tokenizer.decode(next_token)"
      ],
      "metadata": {
        "id": "WxVSp5ptzeQU"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# tokenizer.plot_logits(out.logits)"
      ],
      "metadata": {
        "id": "GH0WklL41oCn"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [],
      "metadata": {
        "id": "W5LKUm-g1vnI"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "#@title Forward from embeddings\n",
        "\n",
        "def forward_from_embeddings(model, params, embeddings, return_last_only=True):\n",
        "    \"\"\"\n",
        "    Forward pass through Gemma model starting from embeddings.\n",
        "    This approach creates dummy tokens and replaces the embeddings after encoding.\n",
        "    \"\"\"\n",
        "    batch_size, seq_len, embed_dim = embeddings.shape\n",
        "\n",
        "    # Create dummy tokens (we'll replace their embeddings)\n",
        "    dummy_tokens = jnp.ones((batch_size, seq_len), dtype=jnp.int32)\n",
        "\n",
        "    def custom_forward(model_instance, tokens, embeddings_to_inject):\n",
        "        \"\"\"Custom forward that replaces token embeddings with our custom ones\"\"\"\n",
        "\n",
        "        # Create inputs mask and positions\n",
        "        inputs_mask = jnp.ones_like(tokens, dtype=bool)\n",
        "        positions = jnp.arange(seq_len)[None, :].repeat(batch_size, axis=0)\n",
        "\n",
        "        # Create attention mask\n",
        "        from gemma.gm.utils import _attention_mask\n",
        "        attention_mask = _attention_mask.make_causal_bidirectional_attention_mask(inputs_mask)\n",
        "\n",
        "        # Use our custom embeddings and apply the required scaling\n",
        "        # This scaling is normally done in the embedder.encode method.\n",
        "        x = embeddings_to_inject * jnp.sqrt(embed_dim).astype(embeddings_to_inject.dtype)\n",
        "\n",
        "        # Forward through transformer blocks\n",
        "        old_cache = {}\n",
        "        new_cache = {}\n",
        "        for i, block in enumerate(model_instance.blocks):\n",
        "            layer_name = f'layer_{i}'\n",
        "            layer_cache, x = block(\n",
        "                x,\n",
        "                positions,\n",
        "                old_cache.get(layer_name),\n",
        "                attention_mask,\n",
        "            )\n",
        "            new_cache[layer_name] = layer_cache\n",
        "\n",
        "        # Final layer norm\n",
        "        x = model_instance.final_norm(x)\n",
        "\n",
        "        last_hidden = None\n",
        "        # Get last token if requested\n",
        "        if return_last_only:\n",
        "            last_input_token_idx = jnp.sum(inputs_mask, axis=-1) - 1\n",
        "            x_last = x[jnp.arange(len(x)), last_input_token_idx, ...]\n",
        "            last_hidden = x_last\n",
        "            # Decode to logits\n",
        "            logits = model_instance.embedder.decode(x_last)\n",
        "        else:\n",
        "            # Decode to logits\n",
        "            logits = model_instance.embedder.decode(x)\n",
        "\n",
        "\n",
        "        # Apply softcap if configured\n",
        "        if model_instance.config.final_logit_softcap is not None:\n",
        "            logits /= model_instance.config.final_logit_softcap\n",
        "            logits = jnp.tanh(logits) * model_instance.config.final_logit_softcap\n",
        "\n",
        "        return logits, new_cache, last_hidden\n",
        "\n",
        "    # Apply the custom forward function\n",
        "    output = model.apply(\n",
        "        {'params': params},\n",
        "        dummy_tokens,\n",
        "        embeddings,\n",
        "        method=custom_forward\n",
        "    )\n",
        "\n",
        "    return output # model.Output(logits=logits, cache=cache)"
      ],
      "metadata": {
        "cellView": "form",
        "id": "9RYzVhfi9TxA"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "output_from_embeddings = forward_from_embeddings(model, params, embeddings)\n",
        "#"
      ],
      "metadata": {
        "id": "fO2W_Pmk8qXV"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "output_from_embeddings[0],output_from_embeddings[2]"
      ],
      "metadata": {
        "id": "sX3zpcQ7FGY5",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "b31d92b5-df18-4e90-bf8b-8b7be2ddc891"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "(Array([[-10.446109 ,   1.6562521,  -1.1088737, ..., -10.696865 ,\n",
              "         -10.644805 , -10.492511 ]], dtype=float32),\n",
              " Array([[-0.84126437, -0.24428287,  1.4033924 , ...,  0.12945779,\n",
              "          0.30105525, -0.06317326]], dtype=float32))"
            ]
          },
          "metadata": {},
          "execution_count": 29
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# # Sample a token from the predicted logits\n",
        "# next_token = jax.random.categorical(\n",
        "#     jax.random.key(1),\n",
        "#     out.logits\n",
        "# )\n",
        "# print(tokenizer.decode(next_token))\n"
      ],
      "metadata": {
        "id": "oEckypPk8iH1"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# tokenizer.plot_logits(out.logits)"
      ],
      "metadata": {
        "id": "eGzCNcRdV_eN"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "#@title Compute jacobian wrt embedding\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "\n",
        "def compute_embedding_jacobian(model, params, embeddings, method='jacfwd'):\n",
        "    \"\"\"\n",
        "    Compute the Jacobian of the model output (hidden states before logits)\n",
        "    with respect to input embeddings.\n",
        "\n",
        "    Args:\n",
        "        model: Gemma model instance\n",
        "        params: Model parameters\n",
        "        embeddings: Input embeddings (batch_size, seq_len, embed_dim)\n",
        "        method: 'jacfwd' or 'jacrev' for forward-mode or reverse-mode AD\n",
        "\n",
        "    Returns:\n",
        "        jacobian: Shape (batch_size, embed_dim, batch_size, seq_len, embed_dim)\n",
        "                 jacobian[b, d_out, b_in, t_in, d_in] = ∂output[b,d_out]/∂input[b_in,t_in,d_in]\n",
        "        output_hidden: The output hidden states that were differentiated\n",
        "    \"\"\"\n",
        "\n",
        "    def get_hidden_states(embeddings):\n",
        "        \"\"\"Extract hidden states before the final linear layer\"\"\"\n",
        "        # Forward pass but capture hidden states before decode\n",
        "        batch_size, seq_len, embed_dim = embeddings.shape\n",
        "        dummy_tokens = jnp.ones((batch_size, seq_len), dtype=jnp.int32)\n",
        "\n",
        "        def custom_forward_hidden(model_instance, tokens, embeddings_to_inject):\n",
        "            # Same as before but return hidden states instead of logits\n",
        "            inputs_mask = jnp.ones_like(tokens, dtype=bool)\n",
        "            positions = jnp.arange(seq_len)[None, :].repeat(batch_size, axis=0)\n",
        "\n",
        "            from gemma.gm.utils import _attention_mask\n",
        "            attention_mask = _attention_mask.make_causal_bidirectional_attention_mask(inputs_mask)\n",
        "\n",
        "            x = embeddings_to_inject\n",
        "\n",
        "            # Forward through transformer blocks\n",
        "            old_cache = {}\n",
        "            for i, block in enumerate(model_instance.blocks):\n",
        "                layer_name = f'layer_{i}'\n",
        "                layer_cache, x = block(\n",
        "                    x,\n",
        "                    positions,\n",
        "                    old_cache.get(layer_name),\n",
        "                    attention_mask,\n",
        "                )\n",
        "                old_cache[layer_name] = layer_cache\n",
        "\n",
        "            # Final layer norm\n",
        "            x = model_instance.final_norm(x)\n",
        "\n",
        "            # Return hidden states (before decode to logits)\n",
        "            # Get last token only\n",
        "            last_input_token_idx = jnp.sum(inputs_mask, axis=-1) - 1\n",
        "            x = x[jnp.arange(len(x)), last_input_token_idx, ...]\n",
        "\n",
        "            return x\n",
        "\n",
        "        hidden_states = model.apply(\n",
        "            {'params': params},\n",
        "            dummy_tokens,\n",
        "            embeddings,\n",
        "            method=custom_forward_hidden\n",
        "        )\n",
        "\n",
        "        return hidden_states\n",
        "\n",
        "    # Compute Jacobian\n",
        "    if method == 'jacfwd':\n",
        "        jacobian_fn = jax.jacfwd(get_hidden_states)\n",
        "    elif method == 'jacrev':\n",
        "        jacobian_fn = jax.jacrev(get_hidden_states)\n",
        "    else:\n",
        "        raise ValueError(\"method must be 'jacfwd' or 'jacrev'\")\n",
        "\n",
        "    jacobian = jacobian_fn(embeddings)\n",
        "    output_hidden = get_hidden_states(embeddings)\n",
        "\n",
        "    return jacobian, output_hidden\n",
        "\n",
        "# Alternative: Compute Jacobian for specific output dimensions\n",
        "def compute_selective_jacobian(model, params, embeddings, output_indices=None):\n",
        "    \"\"\"\n",
        "    Compute Jacobian for specific output dimensions to save memory.\n",
        "\n",
        "    Args:\n",
        "        output_indices: Array of output dimension indices to compute gradients for\n",
        "                       If None, computes for all dimensions\n",
        "    \"\"\"\n",
        "\n",
        "    def get_hidden_states(embeddings):\n",
        "        batch_size, seq_len, embed_dim = embeddings.shape\n",
        "        dummy_tokens = jnp.ones((batch_size, seq_len), dtype=jnp.int32)\n",
        "\n",
        "        def custom_forward_hidden(model_instance, tokens, embeddings_to_inject):\n",
        "            inputs_mask = jnp.ones_like(tokens, dtype=bool)\n",
        "            positions = jnp.arange(seq_len)[None, :].repeat(batch_size, axis=0)\n",
        "\n",
        "            from gemma.gm.utils import _attention_mask\n",
        "            attention_mask = _attention_mask.make_causal_bidirectional_attention_mask(inputs_mask)\n",
        "\n",
        "            x = embeddings_to_inject\n",
        "\n",
        "            old_cache = {}\n",
        "            for i, block in enumerate(model_instance.blocks):\n",
        "                layer_name = f'layer_{i}'\n",
        "                layer_cache, x = block(x, positions, old_cache.get(layer_name), attention_mask)\n",
        "                old_cache[layer_name] = layer_cache\n",
        "\n",
        "            x = model_instance.final_norm(x)\n",
        "\n",
        "            last_input_token_idx = jnp.sum(inputs_mask, axis=-1) - 1\n",
        "            x = x[jnp.arange(len(x)), last_input_token_idx, ...]\n",
        "\n",
        "            # Select specific output dimensions if specified\n",
        "            if output_indices is not None:\n",
        "                x = x[:, output_indices]\n",
        "\n",
        "            return x\n",
        "\n",
        "        return model.apply(\n",
        "            {'params': params},\n",
        "            dummy_tokens,\n",
        "            embeddings,\n",
        "            method=custom_forward_hidden\n",
        "        )\n",
        "\n",
        "    jacobian_fn = jax.jacfwd(get_hidden_states)\n",
        "    jacobian = jacobian_fn(embeddings)\n",
        "    output_hidden = get_hidden_states(embeddings)\n",
        "\n",
        "    return jacobian, output_hidden\n",
        "\n",
        "# Memory-efficient version: compute gradients one output dimension at a time\n",
        "def compute_jacobian_chunked(model, params, embeddings, chunk_size=10):\n",
        "    \"\"\"\n",
        "    Compute Jacobian in chunks to handle large models with limited memory.\n",
        "    \"\"\"\n",
        "\n",
        "    def get_single_output_dim(embeddings, dim_idx):\n",
        "        \"\"\"Get a single dimension of the output\"\"\"\n",
        "        batch_size, seq_len, embed_dim = embeddings.shape\n",
        "        dummy_tokens = jnp.ones((batch_size, seq_len), dtype=jnp.int32)\n",
        "\n",
        "        def custom_forward_single_dim(model_instance, tokens, embeddings_to_inject):\n",
        "            inputs_mask = jnp.ones_like(tokens, dtype=bool)\n",
        "            positions = jnp.arange(seq_len)[None, :].repeat(batch_size, axis=0)\n",
        "\n",
        "            from gemma.gm.utils import _attention_mask\n",
        "            attention_mask = _attention_mask.make_causal_bidirectional_attention_mask(inputs_mask)\n",
        "\n",
        "            x = embeddings_to_inject\n",
        "\n",
        "            old_cache = {}\n",
        "            for i, block in enumerate(model_instance.blocks):\n",
        "                layer_name = f'layer_{i}'\n",
        "                layer_cache, x = block(x, positions, old_cache.get(layer_name), attention_mask)\n",
        "                old_cache[layer_name] = layer_cache\n",
        "\n",
        "            x = model_instance.final_norm(x)\n",
        "\n",
        "            last_input_token_idx = jnp.sum(inputs_mask, axis=-1) - 1\n",
        "            x = x[jnp.arange(len(x)), last_input_token_idx, ...]\n",
        "\n",
        "            # Return only the specified dimension\n",
        "            return x[:, dim_idx]\n",
        "\n",
        "        return model.apply(\n",
        "            {'params': params},\n",
        "            dummy_tokens,\n",
        "            embeddings,\n",
        "            method=custom_forward_single_dim\n",
        "        )\n",
        "\n",
        "    # Get full output to know dimensions\n",
        "    full_output = get_single_output_dim(embeddings, slice(None))\n",
        "    embed_dim = full_output.shape[-1]\n",
        "\n",
        "    # Compute gradients chunk by chunk\n",
        "    jacobian_chunks = []\n",
        "\n",
        "    for start_idx in range(0, embed_dim, chunk_size):\n",
        "        end_idx = min(start_idx + chunk_size, embed_dim)\n",
        "\n",
        "        # Compute gradients for this chunk\n",
        "        chunk_gradients = []\n",
        "        for dim_idx in range(start_idx, end_idx):\n",
        "            grad_fn = jax.grad(lambda x: jnp.sum(get_single_output_dim(x, dim_idx)))\n",
        "            grad = grad_fn(embeddings)\n",
        "            chunk_gradients.append(grad)\n",
        "\n",
        "        jacobian_chunks.append(jnp.stack(chunk_gradients, axis=0))\n",
        "\n",
        "    # Concatenate all chunks\n",
        "    jacobian = jnp.concatenate(jacobian_chunks, axis=0)\n",
        "\n",
        "    return jacobian, full_output\n",
        "\n",
        "# Usage examples:\n",
        "def example_usage():\n",
        "    # Basic usage\n",
        "    jacobian, output_hidden = compute_embedding_jacobian(model, params, embeddings)\n",
        "    print(f\"Jacobian shape: {jacobian.shape}\")\n",
        "    print(f\"Output hidden shape: {output_hidden.shape}\")\n",
        "\n",
        "    # For specific output dimensions (e.g., first 50 dimensions)\n",
        "    selective_jacobian, _ = compute_selective_jacobian(\n",
        "        model, params, embeddings,\n",
        "        output_indices=jnp.arange(50)\n",
        "    )\n",
        "    print(f\"Selective Jacobian shape: {selective_jacobian.shape}\")\n",
        "\n",
        "    # Memory-efficient chunked computation\n",
        "    chunked_jacobian, _ = compute_jacobian_chunked(\n",
        "        model, params, embeddings,\n",
        "        chunk_size=10\n",
        "    )\n",
        "    print(f\"Chunked Jacobian shape: {chunked_jacobian.shape}\")\n",
        "\n",
        "# Example usage:\n",
        "# jacobian, hidden_states = compute_embedding_jacobian(model, params, embeddings)\n",
        "# print(f\"Jacobian shape: {jacobian.shape}\")  # (batch, embed_dim, batch, seq_len, embed_dim)"
      ],
      "metadata": {
        "id": "RlmhuH6u1GlQ",
        "cellView": "form"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# %%time\n",
        "# jacobian, hidden_states = compute_embedding_jacobian(model, params, embeddings)\n",
        "# print(f\"Jacobian shape: {jacobian.shape}\")  # (batch, embed_dim, batch, seq_len, embed_dim)"
      ],
      "metadata": {
        "id": "sLlAK4pS-xhs"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "import matplotlib.pyplot as plt"
      ],
      "metadata": {
        "id": "ITiAac7wEzY5"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# plt.scatter(\n",
        "#     output[2],\n",
        "#     jax.numpy.einsum('ijk,jk->i',jacobian[0,:,:,:,:].squeeze(),embeddings.squeeze()),\n",
        "#     marker='x'\n",
        "# )\n",
        "# plt.grid()\n"
      ],
      "metadata": {
        "id": "iPRcbDIXE2AI"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "import numpy as np"
      ],
      "metadata": {
        "id": "x_3f5KdCSM6b"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# np.asarray(output[2]-jax.numpy.einsum('ijk,jk->i',jacobian[0,:,:,:,:].squeeze(),embeddings.squeeze()))[0]"
      ],
      "metadata": {
        "id": "nicTtpZGSews"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# plt.hist(np.asarray(output[2]-jax.numpy.einsum('ijk,jk->i',jacobian[0,:,:,:,:].squeeze(),embeddings.squeeze()))[0],bins=40);"
      ],
      "metadata": {
        "id": "QDgdoHM6SBFK"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# import time\n",
        "\n",
        "# # Time full SVD\n",
        "# start = time.time()\n",
        "# U, S, Vt = jnp.linalg.svd(jacobian[0,:,:,0,:].squeeze().astype('float32'))#, full_matrices=False, subset_by_index=(0,2))\n",
        "# jax.block_until_ready([U, S, Vt])  # Ensure computation completes\n",
        "# full_svd_time = time.time() - start\n",
        "\n",
        "# print(f\"Full SVD time: {full_svd_time:.3f}s\")\n",
        "# plt.plot(S[:512],'-o')"
      ],
      "metadata": {
        "id": "lgde-P4oWCzB"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# np.std(output[2]-jax.numpy.einsum('ijk,jk->i',jacobian.squeeze(),embeddings.squeeze()))/np.std(output[2])"
      ],
      "metadata": {
        "id": "2lUTMFGFVEb1"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "#@title hidden states\n",
        "\n",
        "def get_hidden_states(embeddings):\n",
        "    \"\"\"\n",
        "    Forward pass through Gemma model starting from embeddings.\n",
        "    This approach creates dummy tokens and replaces the embeddings after encoding.\n",
        "    \"\"\"\n",
        "    batch_size, seq_len, embed_dim = embeddings.shape\n",
        "\n",
        "    # Create dummy tokens (we'll replace their embeddings)\n",
        "    dummy_tokens = jnp.ones((batch_size, seq_len), dtype=jnp.int32)\n",
        "\n",
        "    def custom_forward_hidden(model_instance, tokens, embeddings_to_inject):\n",
        "        \"\"\"Custom forward that replaces token embeddings with our custom ones\"\"\"\n",
        "\n",
        "        # Create inputs mask and positions\n",
        "        inputs_mask = jnp.ones_like(tokens, dtype=bool)\n",
        "        positions = jnp.arange(seq_len)[None, :].repeat(batch_size, axis=0)\n",
        "\n",
        "        # Create attention mask\n",
        "        from gemma.gm.utils import _attention_mask\n",
        "        attention_mask = _attention_mask.make_causal_bidirectional_attention_mask(inputs_mask)\n",
        "\n",
        "        # Use our custom embeddings and apply the required scaling\n",
        "        # This scaling is normally done in the embedder.encode method.\n",
        "        x = embeddings_to_inject * jnp.sqrt(embed_dim).astype(embeddings_to_inject.dtype)\n",
        "\n",
        "        # Forward through transformer blocks\n",
        "        old_cache = {}\n",
        "        new_cache = {}\n",
        "        for i, block in enumerate(model_instance.blocks):\n",
        "            layer_name = f'layer_{i}'\n",
        "            layer_cache, x = block(\n",
        "                x,\n",
        "                positions,\n",
        "                old_cache.get(layer_name),\n",
        "                attention_mask,\n",
        "            )\n",
        "            new_cache[layer_name] = layer_cache\n",
        "\n",
        "        # Final layer norm\n",
        "        x = model_instance.final_norm(x)\n",
        "\n",
        "        last_input_token_idx = jnp.sum(inputs_mask, axis=-1) - 1\n",
        "        x = x[jnp.arange(len(x)), last_input_token_idx, ...]\n",
        "\n",
        "        return x\n",
        "\n",
        "    # Apply the custom forward function\n",
        "    output = model.apply(\n",
        "        {'params': params},\n",
        "        dummy_tokens,\n",
        "        embeddings,\n",
        "        method=custom_forward_hidden\n",
        "    )\n",
        "\n",
        "    return output # model.Output(logits=logits, cache=cache)"
      ],
      "metadata": {
        "id": "BkkshqEkVF4R",
        "cellView": "form"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "#@title get logits\n",
        "def get_logits(embeddings):\n",
        "      \"\"\"Extract hidden states before the final linear layer\"\"\"\n",
        "      embed_dim = embeddings.shape\n",
        "      dummy_tokens = jnp.ones((1, 1), dtype=jnp.int32)\n",
        "\n",
        "      def custom_forward_logits(model_instance, tokens, embeddings_to_inject):\n",
        "          \"\"\"Custom forward that replaces token embeddings with our custom ones\"\"\"\n",
        "\n",
        "          # Decode to logits\n",
        "          logits = model_instance.embedder.decode(embeddings_to_inject)\n",
        "\n",
        "          # Apply softcap if configured\n",
        "          if model_instance.config.final_logit_softcap is not None:\n",
        "              logits /= model_instance.config.final_logit_softcap\n",
        "              logits = jnp.tanh(logits) * model_instance.config.final_logit_softcap\n",
        "\n",
        "          return logits#, new_cache, last_hidden\n",
        "\n",
        "              # Apply the custom forward function\n",
        "      output = model.apply(\n",
        "          {'params': params},\n",
        "          dummy_tokens,\n",
        "          embeddings,\n",
        "          method=custom_forward_logits\n",
        "      )\n",
        "\n",
        "      return output"
      ],
      "metadata": {
        "id": "JMPNg4PFmMUa",
        "cellView": "form"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "!pip install matfree"
      ],
      "metadata": {
        "id": "pqE9RKTqmOd3",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "1ead18f6-b0ea-423b-ef2e-7f1ae3f0438a"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Collecting matfree\n",
            "  Downloading matfree-0.5.2-py3-none-any.whl.metadata (10 kB)\n",
            "Downloading matfree-0.5.2-py3-none-any.whl (33 kB)\n",
            "Installing collected packages: matfree\n",
            "Successfully installed matfree-0.5.2\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "#@title Lanczos Matfree\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "import matfree\n",
        "from matfree import decomp, stochtrace, funm\n",
        "from typing import Tuple, Callable\n",
        "import functools\n",
        "\n",
        "def setup_matfree_jacobian_analysis(\n",
        "    model,\n",
        "    params,\n",
        "    embeddings,\n",
        "    use_detached_gradients: bool = True\n",
        "):\n",
        "    \"\"\"\n",
        "    Set up matrix-free Jacobian analysis using Matfree library.\n",
        "    This avoids storing the full Jacobian while still enabling comprehensive analysis.\n",
        "    \"\"\"\n",
        "\n",
        "    @functools.partial(jax.jit, static_argnums=0)\n",
        "    def get_detached_hidden_states(model_apply_fn, params, embeddings):\n",
        "        \"\"\"Detached gradient version for local linearity\"\"\"\n",
        "        batch_size, seq_len, embed_dim = embeddings.shape\n",
        "        dummy_tokens = jnp.ones((batch_size, seq_len), dtype=jnp.int32)\n",
        "\n",
        "        def detached_forward(model_instance, tokens, embeddings_to_inject):\n",
        "            # Create inputs mask and positions\n",
        "            inputs_mask = jnp.ones_like(tokens, dtype=bool)\n",
        "            positions = jnp.arange(seq_len)[None, :].repeat(batch_size, axis=0)\n",
        "\n",
        "            # Create attention mask\n",
        "            from gemma.gm.utils import _attention_mask\n",
        "            attention_mask = _attention_mask.make_causal_bidirectional_attention_mask(inputs_mask)\n",
        "\n",
        "            # Use our custom embeddings and apply the required scaling\n",
        "            # This scaling is normally done in the embedder.encode method.\n",
        "            x = embeddings_to_inject * jnp.sqrt(embed_dim).astype(embeddings_to_inject.dtype)\n",
        "\n",
        "            # Forward through transformer blocks\n",
        "            old_cache = {}\n",
        "            new_cache = {}\n",
        "            for i, block in enumerate(model_instance.blocks):\n",
        "                layer_name = f'layer_{i}'\n",
        "                layer_cache, x = block(\n",
        "                    x,\n",
        "                    positions,\n",
        "                    old_cache.get(layer_name),\n",
        "                    attention_mask,\n",
        "                )\n",
        "                new_cache[layer_name] = layer_cache\n",
        "\n",
        "            # Final layer norm\n",
        "            x = model_instance.final_norm(x)\n",
        "\n",
        "            last_input_token_idx = jnp.sum(inputs_mask, axis=-1) - 1\n",
        "            x = x[jnp.arange(len(x)), last_input_token_idx, ...]\n",
        "            return x\n",
        "        return model_apply_fn({'params': params}, dummy_tokens, embeddings, method=detached_forward)\n",
        "\n",
        "    # Set up the output function\n",
        "    def output_function(emb):\n",
        "        return get_detached_hidden_states(model.apply, params, emb).flatten()\n",
        "\n",
        "    # Get dimensions\n",
        "    output_hidden = get_detached_hidden_states(model.apply, params, embeddings)\n",
        "    input_size = embeddings.size\n",
        "    output_size = output_hidden.size\n",
        "\n",
        "    print(f\"Matrix-free Jacobian setup:\")\n",
        "    print(f\"  Input size: {input_size:,}\")\n",
        "    print(f\"  Output size: {output_size:,}\")\n",
        "    print(f\"  Jacobian size: {output_size:,} x {input_size:,}\")\n",
        "    print(f\"  Memory saved vs full: {(output_size * input_size * 4) / (1024**3):.1f} GB\")\n",
        "\n",
        "    return output_function, embeddings, output_hidden, (input_size, output_size)\n",
        "\n",
        "\n",
        "def compute_jacobian_trace_estimate(output_function, embeddings, num_samples=1000):\n",
        "    \"\"\"\n",
        "    Estimate the trace of J^T J using Hutchinson's method via Matfree.\n",
        "    This gives insight into the \"energy\" of the Jacobian without computing it fully.\n",
        "    \"\"\"\n",
        "    print(f\"\\nEstimating trace of J^T J using {num_samples} samples...\")\n",
        "\n",
        "    def jtj_matvec(v):\n",
        "        \"\"\"Compute (J^T J) * v\"\"\"\n",
        "        v_reshaped = v.reshape(embeddings.shape)\n",
        "        # J * v using JVP\n",
        "        _, jv = jax.jvp(output_function, (embeddings,), (v_reshaped,))\n",
        "        # J^T * (J * v) using VJP\n",
        "        _, vjp_func = jax.vjp(output_function, embeddings)\n",
        "        jtjv = vjp_func(jv)[0]\n",
        "        return jtjv.flatten()\n",
        "\n",
        "    # Set up Hutchinson trace estimator using correct Matfree API\n",
        "    flat_shape = (embeddings.size,)\n",
        "    template_vector = jnp.zeros(flat_shape)\n",
        "\n",
        "    # Create sampler for Rademacher random variables\n",
        "    sampler = stochtrace.sampler_rademacher(\n",
        "        template_vector,\n",
        "        num=num_samples\n",
        "    )\n",
        "\n",
        "    # Set up trace integrand\n",
        "    integrand = stochtrace.integrand_trace()\n",
        "\n",
        "    # Create the estimator\n",
        "    estimator = stochtrace.estimator(integrand, sampler)\n",
        "\n",
        "    # Estimate the trace\n",
        "    key = jax.random.PRNGKey(42)\n",
        "    trace_estimate = estimator(jtj_matvec, key)\n",
        "\n",
        "    print(f\"  Estimated trace(J^T J): {trace_estimate:.2f}\")\n",
        "    return trace_estimate\n",
        "\n",
        "\n",
        "def compute_top_singular_vectors_matfree(output_function, embeddings, k=50, max_iter=100):\n",
        "    \"\"\"\n",
        "    Compute top-k singular vectors using Matfree's Lanczos implementation.\n",
        "    Much more efficient and robust than our manual implementation.\n",
        "    \"\"\"\n",
        "    print(f\"\\nComputing top-{k} singular vectors using Matfree Lanczos...\")\n",
        "\n",
        "    input_size = embeddings.size\n",
        "    output_hidden = output_function(embeddings)\n",
        "    output_size = output_hidden.size\n",
        "\n",
        "    # JIT compile the matrix-vector operations\n",
        "    # @jax.jit\n",
        "    # def jacobian_matvec(v):\n",
        "\n",
        "    #     v_reshaped = v.reshape(embeddings.shape)\n",
        "    #     _, jv = jax.jvp(output_function, (embeddings,), (v_reshaped,))\n",
        "    #     return jv.flatten()\n",
        "\n",
        "    # @jax.jit\n",
        "    # def jacobian_rmatvec(v):\n",
        "    #     v_reshaped = v.reshape(output_hidden.shape)\n",
        "    #     _, vjp_func = jax.vjp(output_function, embeddings)\n",
        "    #     return vjp_func(v_reshaped)[0].flatten()\n",
        "\n",
        "    def jacobian_matvec(v):\n",
        "        \"\"\"J * v\"\"\"\n",
        "        v_reshaped = v.reshape(embeddings.shape)\n",
        "        _, jv = jax.jvp(output_function, (embeddings,), (v_reshaped,))\n",
        "        return jv.flatten()\n",
        "\n",
        "    def jacobian_rmatvec(v):\n",
        "        \"\"\"J^T * v\"\"\"\n",
        "        v_reshaped = v.reshape(output_hidden.shape)\n",
        "        _, vjp_func = jax.vjp(output_function, embeddings)\n",
        "        return vjp_func(v_reshaped)[0].flatten()\n",
        "\n",
        "    # Choose smaller matrix for efficiency\n",
        "    if output_size < input_size:\n",
        "        # Use J J^T\n",
        "        def jjt_matvec(v):\n",
        "            return jacobian_matvec(jacobian_rmatvec(v))\n",
        "\n",
        "        print(f\"  Using J J^T ({output_size} x {output_size})\")\n",
        "        matrix_size = output_size\n",
        "        matvec_fn = jjt_matvec\n",
        "\n",
        "    else:\n",
        "        # Use J^T J\n",
        "        def jtj_matvec(v):\n",
        "            return jacobian_rmatvec(jacobian_matvec(v))\n",
        "\n",
        "        print(f\"  Using J^T J ({input_size} x {input_size})\")\n",
        "        matrix_size = input_size\n",
        "        matvec_fn = jtj_matvec\n",
        "\n",
        "    # Use the correct Matfree API for tridiagonalization\n",
        "    max_iter_actual = min(max_iter, matrix_size - 1)\n",
        "\n",
        "    # The correct way: tridiag_sym returns a function that we can call directly\n",
        "    tridiag_fn = decomp.tridiag_sym(max_iter_actual, reortho=\"full\")\n",
        "\n",
        "    # Call the tridiagonalization function directly\n",
        "    key = jax.random.PRNGKey(42)\n",
        "    template_vector = jax.random.normal(key, shape=(matrix_size,))\n",
        "\n",
        "    # This should return the tridiagonal decomposition result\n",
        "    result = tridiag_fn(matvec_fn, template_vector)\n",
        "\n",
        "    # Extract the components from the Matfree result\n",
        "    # Based on the attributes you showed: ['J_small', 'Q_tall', ...]\n",
        "    if hasattr(result, 'J_small') and hasattr(result, 'Q_tall'):\n",
        "        print(\"  Using Matfree tridiagonal result\")\n",
        "        # J_small should be the tridiagonal matrix\n",
        "        # Q_tall should be the orthogonal basis vectors\n",
        "\n",
        "        tridiag_matrix = result.J_small  # This should be the T matrix\n",
        "        basis_vectors = result.Q_tall    # This should be the Q matrix\n",
        "\n",
        "        # Extract eigenvalues and eigenvectors from the tridiagonal matrix\n",
        "        eigenvals, eigenvecs = jnp.linalg.eigh(tridiag_matrix)\n",
        "\n",
        "        # Get top-k eigenvalues and vectors\n",
        "        k_actual = min(k, len(eigenvals))\n",
        "        idx = jnp.argsort(eigenvals)[-k_actual:]\n",
        "        top_eigenvals = eigenvals[idx]\n",
        "        top_eigenvecs_tridiag = eigenvecs[:, idx]\n",
        "\n",
        "        # Reconstruct eigenvectors in original space\n",
        "        # Q_tall @ eigenvectors gives us the actual eigenvectors\n",
        "        top_eigenvecs = basis_vectors @ top_eigenvecs_tridiag\n",
        "\n",
        "        # Convert to singular values\n",
        "        singular_values = jnp.sqrt(jnp.maximum(top_eigenvals, 0))\n",
        "\n",
        "        print(f\"  Matfree completed with {tridiag_matrix.shape[0]} iterations\")\n",
        "        print(f\"  Tridiagonal matrix shape: {tridiag_matrix.shape}\")\n",
        "        print(f\"  Basis vectors shape: {basis_vectors.shape}\")\n",
        "\n",
        "    else:\n",
        "        print(\"  Warning: Unexpected Matfree result structure\")\n",
        "        print(f\"  Result type: {type(result)}\")\n",
        "        print(f\"  Result attributes: {dir(result)}\")\n",
        "        # Fallback to manual implementation\n",
        "        return compute_top_singular_vectors_manual(\n",
        "            jacobian_matvec, jacobian_rmatvec,\n",
        "            input_size, output_size, k, max_iter\n",
        "        )\n",
        "\n",
        "    # Compute complementary singular vectors\n",
        "    if output_size < input_size:\n",
        "        # We computed J J^T, so top_eigenvecs are left singular vectors\n",
        "        U = top_eigenvecs\n",
        "\n",
        "        # Compute right singular vectors: V = J^T U / sigma\n",
        "        V_list = []\n",
        "        for i in range(k_actual):\n",
        "            if singular_values[i] > 1e-12:\n",
        "                v_i = jacobian_rmatvec(U[:, i]) / singular_values[i]\n",
        "            else:\n",
        "                v_i = jnp.zeros(input_size)\n",
        "            V_list.append(v_i)\n",
        "        V = jnp.column_stack(V_list) if V_list else jnp.zeros((input_size, 0))\n",
        "\n",
        "    else:\n",
        "        # We computed J^T J, so top_eigenvecs are right singular vectors\n",
        "        V = top_eigenvecs\n",
        "\n",
        "        # Compute left singular vectors: U = J V / sigma\n",
        "        U_list = []\n",
        "        for i in range(k_actual):\n",
        "            if singular_values[i] > 1e-12:\n",
        "                u_i = jacobian_matvec(V[:, i]) / singular_values[i]\n",
        "            else:\n",
        "                u_i = jnp.zeros(output_size)\n",
        "            U_list.append(u_i)\n",
        "        U = jnp.column_stack(U_list) if U_list else jnp.zeros((output_size, 0))\n",
        "\n",
        "    print(f\"  Top-{k_actual} singular values: {singular_values}\")\n",
        "\n",
        "    return U, singular_values, V, result\n",
        "\n",
        "\n",
        "def compute_top_singular_vectors_manual(jacobian_matvec, jacobian_rmatvec,\n",
        "                                       input_size, output_size, k, max_iter):\n",
        "    \"\"\"\n",
        "    Fallback manual implementation if Matfree API doesn't work as expected.\n",
        "    \"\"\"\n",
        "    print(\"  Falling back to manual Lanczos implementation...\")\n",
        "\n",
        "    # Use the smaller matrix\n",
        "    if output_size < input_size:\n",
        "        def matvec(v):\n",
        "            return jacobian_matvec(jacobian_rmatvec(v))\n",
        "        matrix_size = output_size\n",
        "        is_jjt = True\n",
        "    else:\n",
        "        def matvec(v):\n",
        "            return jacobian_rmatvec(jacobian_matvec(v))\n",
        "        matrix_size = input_size\n",
        "        is_jjt = False\n",
        "\n",
        "    # Simple Lanczos iteration\n",
        "    max_iter = min(max_iter, matrix_size - 1, k + 10)\n",
        "\n",
        "    # Initialize\n",
        "    key = jax.random.PRNGKey(42)\n",
        "    q = jax.random.normal(key, (matrix_size,))\n",
        "    q = q / jnp.linalg.norm(q)\n",
        "\n",
        "    Q = jnp.zeros((matrix_size, max_iter + 1))\n",
        "    alpha = jnp.zeros(max_iter)\n",
        "    beta = jnp.zeros(max_iter + 1)\n",
        "\n",
        "    Q = Q.at[:, 0].set(q)\n",
        "\n",
        "    for j in range(max_iter):\n",
        "        v = matvec(Q[:, j])\n",
        "\n",
        "        alpha = alpha.at[j].set(jnp.dot(Q[:, j], v))\n",
        "        v = v - alpha[j] * Q[:, j]\n",
        "\n",
        "        if j > 0:\n",
        "            v = v - beta[j] * Q[:, j-1]\n",
        "\n",
        "        beta_val = jnp.linalg.norm(v)\n",
        "        beta = beta.at[j+1].set(beta_val)\n",
        "\n",
        "        if beta_val < 1e-12:\n",
        "            actual_iter = j + 1\n",
        "            break\n",
        "\n",
        "        Q = Q.at[:, j+1].set(v / beta_val)\n",
        "        actual_iter = j + 1\n",
        "    else:\n",
        "        actual_iter = max_iter\n",
        "\n",
        "    # Build tridiagonal matrix and solve\n",
        "    T = jnp.diag(alpha[:actual_iter])\n",
        "    if actual_iter > 1:\n",
        "        off_diag = beta[1:actual_iter]\n",
        "        T = T + jnp.diag(off_diag, 1) + jnp.diag(off_diag, -1)\n",
        "\n",
        "    eigenvals, eigenvecs = jnp.linalg.eigh(T)\n",
        "\n",
        "    # Top k\n",
        "    k_actual = min(k, len(eigenvals))\n",
        "    idx = jnp.argsort(eigenvals)[-k_actual:]\n",
        "    top_eigenvals = eigenvals[idx]\n",
        "\n",
        "    # Reconstruct vectors\n",
        "    top_vecs = Q[:, :actual_iter] @ eigenvecs[:, idx]\n",
        "    singular_values = jnp.sqrt(jnp.maximum(top_eigenvals, 0))\n",
        "\n",
        "    if is_jjt:\n",
        "        U = top_vecs\n",
        "        V_list = []\n",
        "        for i in range(k_actual):\n",
        "            if singular_values[i] > 1e-12:\n",
        "                v_i = jacobian_rmatvec(U[:, i]) / singular_values[i]\n",
        "            else:\n",
        "                v_i = jnp.zeros(input_size)\n",
        "            V_list.append(v_i)\n",
        "        V = jnp.column_stack(V_list)\n",
        "    else:\n",
        "        V = top_vecs\n",
        "        U_list = []\n",
        "        for i in range(k_actual):\n",
        "            if singular_values[i] > 1e-12:\n",
        "                u_i = jacobian_matvec(V[:, i]) / singular_values[i]\n",
        "            else:\n",
        "                u_i = jnp.zeros(output_size)\n",
        "            U_list.append(u_i)\n",
        "        U = jnp.column_stack(U_list)\n",
        "\n",
        "    print(f\"  Manual implementation completed with {actual_iter} iterations\")\n",
        "    print(f\"  Top-{k_actual} singular values: {singular_values}\")\n",
        "\n",
        "    return U, singular_values, V, None\n",
        "\n",
        "\n",
        "def compute_jacobian_log_determinant(output_function, embeddings, num_samples=100):\n",
        "    \"\"\"\n",
        "    Estimate log|det(J^T J)| using stochastic Lanczos quadrature.\n",
        "    This is useful for understanding the \"volume\" of the transformation.\n",
        "    \"\"\"\n",
        "    print(f\"\\nEstimating log-determinant of J^T J...\")\n",
        "\n",
        "    def jtj_matvec(v):\n",
        "        v_reshaped = v.reshape(embeddings.shape)\n",
        "        _, jv = jax.jvp(output_function, (embeddings,), (v_reshaped,))\n",
        "        _, vjp_func = jax.vjp(output_function, embeddings)\n",
        "        jtjv = vjp_func(jv)[0]\n",
        "        return jtjv.flatten()\n",
        "\n",
        "    # Use stochastic Lanczos quadrature for log-determinant\n",
        "    input_like = jnp.zeros(embeddings.shape).flatten()\n",
        "    input_size = embeddings.size\n",
        "\n",
        "    # Set up the log function\n",
        "    def log_fn(x):\n",
        "        return jnp.log(x + 1e-12)  # Add small regularization\n",
        "\n",
        "    # Use Matfree's function evaluation tools\n",
        "    sampler = stochtrace.sampler_rademacher(input_like, num=num_samples)\n",
        "\n",
        "    # This is a simplified version - you'd want to use matfree.funm for the full implementation\n",
        "    # For now, we'll estimate it via trace estimation of log eigenvalues\n",
        "    print(\"  (Simplified log-det estimation - see Matfree docs for full implementation)\")\n",
        "\n",
        "    return None  # Placeholder\n",
        "\n",
        "\n",
        "def analyze_jacobian_spectrum_matfree(output_function, embeddings, k=100, max_iter=100):\n",
        "    \"\"\"\n",
        "    Comprehensive spectral analysis using Matfree tools.\n",
        "    \"\"\"\n",
        "    print(\"\\n\" + \"=\"*60)\n",
        "    print(\"MATRIX-FREE JACOBIAN SPECTRAL ANALYSIS\")\n",
        "    print(\"=\"*60)\n",
        "\n",
        "    # 1. Trace estimation\n",
        "    trace_jtj = compute_jacobian_trace_estimate(output_function, embeddings, num_samples=100)\n",
        "\n",
        "    # 2. Top singular vectors\n",
        "    U, singular_values, V, lanczos_state = compute_top_singular_vectors_matfree(\n",
        "        output_function, embeddings, k=k, max_iter=max_iter\n",
        "    )\n",
        "\n",
        "    # 3. Spectral properties\n",
        "    print(f\"\\nSPECTRAL PROPERTIES:\")\n",
        "    print(f\"  Largest singular value: {singular_values[0]:.6f}\")\n",
        "    print(f\"  Smallest computed singular value: {singular_values[-1]:.6f}\")\n",
        "    print(f\"  Condition number estimate: {singular_values[0] / singular_values[-1]:.2e}\")\n",
        "    print(f\"  Spectral norm estimate: {singular_values[0]:.6f}\")\n",
        "    print(f\"  Trace(J^T J) estimate: {trace_jtj:.2f}\")\n",
        "    print(f\"  Sum of top-{k} eigenvalues: {jnp.sum(singular_values**2):.2f}\")\n",
        "\n",
        "    # 4. Low-rank approximation analysis\n",
        "    print(f\"\\nLOW-RANK STRUCTURE:\")\n",
        "    total_energy = jnp.sum(singular_values**2)\n",
        "    cumulative_energy = jnp.cumsum(singular_values**2)\n",
        "\n",
        "    for pct in [50, 90, 95, 99]:\n",
        "        idx = jnp.argmax(cumulative_energy >= (pct/100) * total_energy)\n",
        "        if idx < len(singular_values):\n",
        "            print(f\"  {pct}% energy in top {idx+1} components\")\n",
        "\n",
        "    return {\n",
        "        'singular_values': singular_values,\n",
        "        'left_singular_vectors': U,\n",
        "        'right_singular_vectors': V,\n",
        "        'trace_estimate': trace_jtj,\n",
        "        'lanczos_state': lanczos_state\n",
        "    }\n",
        "\n",
        "\n",
        "def test_reconstruction_matfree(output_function, embeddings, results, num_tests=3):\n",
        "    \"\"\"\n",
        "    Test reconstruction quality using the matrix-free singular vectors.\n",
        "    \"\"\"\n",
        "    print(f\"\\nTesting reconstruction with top-{len(results['singular_values'])} components...\")\n",
        "\n",
        "    U = results['left_singular_vectors']\n",
        "    s = results['singular_values']\n",
        "    V = results['right_singular_vectors']\n",
        "\n",
        "    original_output = output_function(embeddings)\n",
        "\n",
        "    for i in range(num_tests):\n",
        "        # Random perturbation\n",
        "        scale = 10**(-3-i)\n",
        "        key = jax.random.PRNGKey(42 + i)\n",
        "        delta = jax.random.normal(key, embeddings.shape) * scale\n",
        "\n",
        "        # True perturbed output\n",
        "        perturbed_output = output_function(embeddings + delta)\n",
        "        true_change = perturbed_output - original_output\n",
        "\n",
        "        # Low-rank approximation of change\n",
        "        delta_flat = delta.flatten()\n",
        "\n",
        "        # Approximate: (U @ diag(s) @ V.T) @ delta_flat\n",
        "        predicted_change = U @ (s[:, None] * (V.T @ delta_flat)[:, None]).flatten()\n",
        "\n",
        "        error = jnp.linalg.norm(true_change - predicted_change) / jnp.linalg.norm(true_change)\n",
        "        print(f\"  Test {i+1}: scale={scale:.0e}, relative_error={error:.2e}\")\n",
        "\n",
        "\n",
        "def reconstruct_jacobian_from_svd(U, singular_values, V, return_full=True):\n",
        "    \"\"\"\n",
        "    Reconstruct the Jacobian matrix from its SVD components.\n",
        "\n",
        "    Args:\n",
        "        U: Left singular vectors (output_size, k)\n",
        "        singular_values: Singular values (k,)\n",
        "        V: Right singular vectors (input_size, k)\n",
        "        return_full: If True, return full matrix. If False, return function for matvec operations.\n",
        "\n",
        "    Returns:\n",
        "        If return_full=True: Jacobian matrix (output_size, input_size)\n",
        "        If return_full=False: Dictionary with 'matvec' and 'rmatvec' functions\n",
        "    \"\"\"\n",
        "    print(f\"Reconstructing Jacobian from SVD components:\")\n",
        "    print(f\"  U shape: {U.shape} (left singular vectors)\")\n",
        "    print(f\"  σ shape: {singular_values.shape} (singular values)\")\n",
        "    print(f\"  V shape: {V.shape} (right singular vectors)\")\n",
        "\n",
        "    output_size, k = U.shape\n",
        "    input_size, k_check = V.shape\n",
        "\n",
        "    assert k == k_check == len(singular_values), \"Inconsistent dimensions in SVD components\"\n",
        "\n",
        "    if return_full:\n",
        "        # Full reconstruction: J ≈ U @ diag(σ) @ V^T\n",
        "        jacobian_approx = U @ jnp.diag(singular_values) @ V.T\n",
        "\n",
        "        print(f\"  Reconstructed Jacobian shape: {jacobian_approx.shape}\")\n",
        "\n",
        "        # Compute reconstruction properties\n",
        "        frobenius_norm = jnp.linalg.norm(jacobian_approx, 'fro')\n",
        "        spectral_norm = singular_values[0]  # Largest singular value\n",
        "        rank_estimate = jnp.sum(singular_values > 1e-12)\n",
        "\n",
        "        print(f\"  Frobenius norm: {frobenius_norm:.4f}\")\n",
        "        print(f\"  Spectral norm: {spectral_norm:.4f}\")\n",
        "        print(f\"  Effective rank: {rank_estimate}/{k}\")\n",
        "\n",
        "        return jacobian_approx\n",
        "\n",
        "    else:\n",
        "        # Memory-efficient version: return functions for matrix-vector products\n",
        "        def jacobian_matvec(v):\n",
        "            \"\"\"Compute J @ v using low-rank representation\"\"\"\n",
        "            # J @ v = U @ diag(σ) @ V^T @ v = U @ (σ * (V^T @ v))\n",
        "            v_flat = v.flatten() if v.ndim > 1 else v\n",
        "            return U @ (singular_values * (V.T @ v_flat))\n",
        "\n",
        "        def jacobian_rmatvec(u):\n",
        "            \"\"\"Compute J^T @ u using low-rank representation\"\"\"\n",
        "            # J^T @ u = V @ diag(σ) @ U^T @ u = V @ (σ * (U^T @ u))\n",
        "            u_flat = u.flatten() if u.ndim > 1 else u\n",
        "            return V @ (singular_values * (U.T @ u_flat))\n",
        "\n",
        "        def jacobian_quadratic_form(v):\n",
        "            \"\"\"Compute v^T @ J^T @ J @ v efficiently\"\"\"\n",
        "            # v^T @ J^T @ J @ v = ||J @ v||^2 = ||U @ (σ * (V^T @ v))||^2\n",
        "            v_flat = v.flatten() if v.ndim > 1 else v\n",
        "            Vt_v = V.T @ v_flat\n",
        "            weighted = singular_values * Vt_v\n",
        "            return jnp.sum(weighted**2)\n",
        "\n",
        "        print(f\"  Returning matrix-free operators\")\n",
        "\n",
        "        return {\n",
        "            'matvec': jacobian_matvec,\n",
        "            'rmatvec': jacobian_rmatvec,\n",
        "            'quadratic_form': jacobian_quadratic_form,\n",
        "            'frobenius_norm_squared': jnp.sum(singular_values**2),\n",
        "            'spectral_norm': singular_values[0],\n",
        "            'rank': k\n",
        "        }\n",
        "\n",
        "\n",
        "def test_jacobian_reconstruction(jacobian_approx, output_function, embeddings, num_tests=3):\n",
        "    \"\"\"\n",
        "    Test how well the reconstructed Jacobian approximates the true model behavior.\n",
        "    \"\"\"\n",
        "    print(f\"\\nTesting Jacobian reconstruction quality:\")\n",
        "    print(\"=\"*50)\n",
        "\n",
        "    original_output = output_function(embeddings).flatten()\n",
        "\n",
        "    reconstruction_errors = []\n",
        "\n",
        "    for i in range(num_tests):\n",
        "        # Generate test perturbation with decreasing magnitude\n",
        "        perturbation_scale = 10**(-2-i)\n",
        "        key = jax.random.PRNGKey(42 + i)\n",
        "        delta = jax.random.normal(key, embeddings.shape) * perturbation_scale\n",
        "\n",
        "        # True model response\n",
        "        perturbed_output = output_function(embeddings + delta).flatten()\n",
        "        true_change = perturbed_output - original_output\n",
        "\n",
        "        # Linear approximation using reconstructed Jacobian\n",
        "        delta_flat = delta.flatten()\n",
        "        predicted_change = jacobian_approx @ delta_flat\n",
        "\n",
        "        # Compute errors\n",
        "        absolute_error = jnp.linalg.norm(true_change - predicted_change)\n",
        "        relative_error = absolute_error / jnp.linalg.norm(true_change)\n",
        "\n",
        "        reconstruction_errors.append(relative_error)\n",
        "\n",
        "        print(f\"Test {i+1}:\")\n",
        "        print(f\"  Perturbation scale: {perturbation_scale:.0e}\")\n",
        "        print(f\"  True change norm: {jnp.linalg.norm(true_change):.6f}\")\n",
        "        print(f\"  Predicted change norm: {jnp.linalg.norm(predicted_change):.6f}\")\n",
        "        print(f\"  Absolute error: {absolute_error:.6f}\")\n",
        "        print(f\"  Relative error: {relative_error:.6f}\")\n",
        "        print()\n",
        "\n",
        "    avg_error = jnp.mean(jnp.array(reconstruction_errors))\n",
        "    print(f\"Average relative error: {avg_error:.6f}\")\n",
        "\n",
        "    # Quality assessment\n",
        "    if avg_error < 1e-4:\n",
        "        quality = \"Excellent\"\n",
        "    elif avg_error < 1e-3:\n",
        "        quality = \"Very good\"\n",
        "    elif avg_error < 1e-2:\n",
        "        quality = \"Good\"\n",
        "    elif avg_error < 1e-1:\n",
        "        quality = \"Fair\"\n",
        "    else:\n",
        "        quality = \"Poor\"\n",
        "\n",
        "    print(f\"Reconstruction quality: {quality}\")\n",
        "\n",
        "    return reconstruction_errors\n",
        "\n",
        "\n",
        "def analyze_svd_truncation_error(U, singular_values, V, max_components=None):\n",
        "    \"\"\"\n",
        "    Analyze how reconstruction error changes with number of SVD components.\n",
        "    \"\"\"\n",
        "    if max_components is None:\n",
        "        max_components = len(singular_values)\n",
        "\n",
        "    print(f\"\\nSVD Truncation Analysis:\")\n",
        "    print(\"=\"*50)\n",
        "\n",
        "    # Full reconstruction for reference\n",
        "    jacobian_full = U @ jnp.diag(singular_values) @ V.T\n",
        "    full_frobenius_norm = jnp.linalg.norm(jacobian_full, 'fro')\n",
        "\n",
        "    print(f\"Full reconstruction (k={len(singular_values)}):\")\n",
        "    print(f\"  Frobenius norm: {full_frobenius_norm:.4f}\")\n",
        "    print()\n",
        "\n",
        "    # Test different truncation levels\n",
        "    test_components = [1, 2, 5, 10, 20, min(50, len(singular_values)), len(singular_values)]\n",
        "    test_components = sorted(list(set([k for k in test_components if k <= len(singular_values)])))\n",
        "\n",
        "    print(\"Truncation analysis:\")\n",
        "    for k in test_components:\n",
        "        if k > len(singular_values):\n",
        "            continue\n",
        "\n",
        "        # Truncated reconstruction\n",
        "        jacobian_k = U[:, :k] @ jnp.diag(singular_values[:k]) @ V[:, :k].T\n",
        "\n",
        "        # Compute error\n",
        "        error_matrix = jacobian_full - jacobian_k\n",
        "        frobenius_error = jnp.linalg.norm(error_matrix, 'fro')\n",
        "        relative_error = frobenius_error / full_frobenius_norm\n",
        "\n",
        "        # Energy captured\n",
        "        energy_k = jnp.sum(singular_values[:k]**2)\n",
        "        total_energy = jnp.sum(singular_values**2)\n",
        "        energy_fraction = energy_k / total_energy\n",
        "\n",
        "        print(f\"  k={k:2d}: relative_error={relative_error:.6f}, \"\n",
        "              f\"energy_captured={energy_fraction:.4f} ({energy_fraction*100:.1f}%)\")\n",
        "\n",
        "    return test_components\n",
        "\n",
        "\n",
        "# Example usage functions\n",
        "def quick_jacobian_analysis(results):\n",
        "    \"\"\"\n",
        "    Quick analysis of the SVD results.\n",
        "    \"\"\"\n",
        "    U = results['left_singular_vectors']\n",
        "    s = results['singular_values']\n",
        "    V = results['right_singular_vectors']\n",
        "\n",
        "    print(f\"\\nQuick Jacobian Analysis:\")\n",
        "    print(\"=\"*50)\n",
        "\n",
        "    # Basic properties\n",
        "    print(f\"Matrix dimensions: {U.shape[0]} × {V.shape[0]}\")\n",
        "    print(f\"Rank approximation: {len(s)}\")\n",
        "    print(f\"Condition number estimate: {s[0] / s[-1]:.2e}\")\n",
        "\n",
        "    # Energy distribution\n",
        "    energy = s**2\n",
        "    total_energy = jnp.sum(energy)\n",
        "    cumulative_energy = jnp.cumsum(energy) / total_energy\n",
        "\n",
        "    print(f\"\\nEnergy distribution:\")\n",
        "    for i, pct in enumerate([0.5, 0.9, 0.95, 0.99]):\n",
        "        idx = jnp.argmax(cumulative_energy >= pct)\n",
        "        if idx < len(s):\n",
        "            print(f\"  {pct*100:4.1f}% energy in top {idx+1:2d} components\")\n",
        "\n",
        "    # Reconstruct Jacobian\n",
        "    jacobian = reconstruct_jacobian_from_svd(U, s, V, return_full=True)\n",
        "\n",
        "    return jacobian\n",
        "\n",
        "\n",
        "# Add this to the example usage section\n",
        "# print(\"\"\"\n",
        "# USAGE EXAMPLES:\n",
        "\n",
        "# # Reconstruct full Jacobian matrix\n",
        "# jacobian_matrix = reconstruct_jacobian_from_svd(\n",
        "#     results['left_singular_vectors'],\n",
        "#     results['singular_values'],\n",
        "#     results['right_singular_vectors']\n",
        "# )\n",
        "\n",
        "# # Or get memory-efficient operators\n",
        "# jacobian_ops = reconstruct_jacobian_from_svd(\n",
        "#     results['left_singular_vectors'],\n",
        "#     results['singular_values'],\n",
        "#     results['right_singular_vectors'],\n",
        "#     return_full=False\n",
        "# )\n",
        "\n",
        "# # Then use: jacobian_ops['matvec'](vector)\n",
        "\n",
        "# # Test reconstruction quality\n",
        "# test_jacobian_reconstruction(jacobian_matrix, output_function, embeddings)\n",
        "\n",
        "# # Analyze truncation effects\n",
        "# analyze_svd_truncation_error(\n",
        "#     results['left_singular_vectors'],\n",
        "#     results['singular_values'],\n",
        "#     results['right_singular_vectors']\n",
        "# )\n",
        "# \"\"\")\n",
        "# \"\"\"\n",
        "# Complete analysis pipeline using Matfree.\n",
        "# \"\"\"\n",
        "# # # Setup\n",
        "# # output_fn, emb, output, dims = setup_matfree_jacobian_analysis(\n",
        "# #     model, params, embeddings, use_detached_gradients=True\n",
        "# # )\n",
        "\n",
        "# # # Run comprehensive analysis\n",
        "# # results = analyze_jacobian_spectrum_matfree(output_fn, emb, k=50)\n",
        "\n",
        "# # # Test reconstruction\n",
        "# # test_reconstruction_matfree(output_fn, emb, results)\n",
        "\n",
        "# # return results\n",
        "\n",
        "# # Memory comparison\n",
        "# print(\"\"\"\n",
        "# MEMORY COMPARISON:\n",
        "\n",
        "# Full Jacobian approach:\n",
        "# - Stores entire O(output_size × input_size) matrix\n",
        "# - Memory: ~GB to TB depending on model size\n",
        "# - Perfect accuracy but massive memory requirements\n",
        "\n",
        "# Matfree approach:\n",
        "# - Only stores k singular vectors\n",
        "# - Memory: O(k × (input_size + output_size))\n",
        "# - Excellent accuracy with minimal memory\n",
        "# - Can scale to much larger models!\n",
        "\n",
        "# For a typical LLM layer:\n",
        "# - Full: ~69 GB for (4096 × 4096) Jacobian\n",
        "# - Matfree: ~0.1 GB for top-50 singular vectors\n",
        "# - Memory reduction: ~700x while maintaining high accuracy!\n",
        "# \"\"\")"
      ],
      "metadata": {
        "id": "bClTVlqArrcs",
        "cellView": "form"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "#@title Lanczos update\n",
        "def create_per_token_matvecs(output_function, embeddings, token_idx):\n",
        "    \"\"\"\n",
        "    Creates matrix-vector product functions for the Jacobian of a single token.\n",
        "\n",
        "    Args:\n",
        "        output_function: The function mapping full embeddings to the final output.\n",
        "        embeddings: The full input embeddings of shape (1, N, d).\n",
        "        token_idx: The index of the token (from 0 to N-1) to analyze.\n",
        "\n",
        "    Returns:\n",
        "        A tuple of (matvec, rmatvec) for the specific token's Jacobian.\n",
        "    \"\"\"\n",
        "    seq_len = embeddings.shape[1]\n",
        "    embed_dim = embeddings.shape[2]\n",
        "\n",
        "    def jacobian_matvec_token(v):\n",
        "        \"\"\"Computes J_i @ v, where v has shape (d,).\"\"\"\n",
        "        # Create a tangent vector that is zero everywhere except at token_idx\n",
        "        tangent = jnp.zeros_like(embeddings)\n",
        "        tangent = tangent.at[0, token_idx, :].set(v)\n",
        "\n",
        "        # JVP computes the effect of this specific perturbation\n",
        "        _, jv = jax.jvp(output_function, (embeddings,), (tangent,))\n",
        "        return jv.flatten()\n",
        "\n",
        "    def jacobian_rmatvec_token(u):\n",
        "        \"\"\"Computes J_i^T @ u, where u has shape (d,).\"\"\"\n",
        "        # VJP computes the gradient w.r.t. all inputs for a given output perturbation\n",
        "        _, vjp_func = jax.vjp(output_function, embeddings)\n",
        "\n",
        "        # The result of vjp_func(u) is a gradient of shape (1, N, d)\n",
        "        # We only care about the gradient corresponding to our token of interest\n",
        "\n",
        "        # THE FIX IS HERE: Pass `u` directly without reshaping.\n",
        "        # It needs to match the shape of the original function's output, which is flat.\n",
        "        grad_embeddings = vjp_func(u)[0]\n",
        "\n",
        "        return grad_embeddings[0, token_idx, :]\n",
        "\n",
        "    return jacobian_matvec_token, jacobian_rmatvec_token\n",
        "def compute_top_singular_vectors_matfree(output_function, embeddings, k=10, max_iter=50):\n",
        "    \"\"\"\n",
        "    Computes the top-k SVD for the Jacobian of each input token independently.\n",
        "\n",
        "    Returns:\n",
        "        A list of dictionaries, where each dictionary contains the SVD\n",
        "        results ('U', 's', 'V') for one token.\n",
        "    \"\"\"\n",
        "    batch_size, seq_len, embed_dim = embeddings.shape\n",
        "    assert batch_size == 1, \"This implementation assumes batch size of 1.\"\n",
        "\n",
        "    all_svd_results = []\n",
        "\n",
        "    for i in range(seq_len):\n",
        "        print(f\"\\n--- Analyzing Token {i+1}/{seq_len} ---\")\n",
        "\n",
        "        # 1. Get the matvec functions for the current token\n",
        "        matvec_i, rmatvec_i = create_per_token_matvecs(output_function, embeddings, i)\n",
        "\n",
        "        # The Jacobian J_i is d x d. So we compute (J_i^T J_i), which is also d x d.\n",
        "        def jtj_matvec_i(v):\n",
        "            return rmatvec_i(matvec_i(v))\n",
        "\n",
        "        # 2. Run Lanczos (using Matfree) on this token's operator\n",
        "        matrix_size = embed_dim\n",
        "        max_iter_actual = min(max_iter, matrix_size - 1)\n",
        "        tridiag_fn = decomp.tridiag_sym(max_iter_actual, reortho=\"full\")\n",
        "\n",
        "        key = jax.random.PRNGKey(i) # Use a different key for each token\n",
        "        template_vector = jax.random.normal(key, shape=(matrix_size,))\n",
        "\n",
        "        # Run the tridiagonalization\n",
        "        result = tridiag_fn(jtj_matvec_i, template_vector)\n",
        "\n",
        "        # 3. Extract singular values and vectors\n",
        "        tridiag_matrix = result.J_small\n",
        "        basis_vectors = result.Q_tall\n",
        "\n",
        "        eigenvals, eigenvecs_tridiag = jnp.linalg.eigh(tridiag_matrix)\n",
        "\n",
        "        # Sort in descending order\n",
        "        idx = jnp.argsort(eigenvals)[::-1][:k]\n",
        "        top_eigenvals = eigenvals[idx]\n",
        "        top_eigenvecs_tridiag = eigenvecs_tridiag[:, idx]\n",
        "\n",
        "        # Reconstruct right singular vectors (V) in original space\n",
        "        # These are the eigenvectors of J_i^T J_i\n",
        "        V = basis_vectors @ top_eigenvecs_tridiag\n",
        "\n",
        "        # Singular values are the sqrt of the eigenvalues\n",
        "        s = jnp.sqrt(jnp.maximum(top_eigenvals, 0))\n",
        "\n",
        "        # Compute left singular vectors: U = J_i @ V @ diag(1/s)\n",
        "        U_list = []\n",
        "        for j in range(k):\n",
        "            if s[j] > 1e-12:\n",
        "                u_j = matvec_i(V[:, j]) / s[j]\n",
        "            else:\n",
        "                u_j = jnp.zeros(embed_dim)\n",
        "            U_list.append(u_j)\n",
        "        U = jnp.column_stack(U_list)\n",
        "\n",
        "        print(f\"  Top-{k} singular values: {s}\")\n",
        "\n",
        "        all_svd_results.append({\n",
        "            'token_index': i,\n",
        "            'U': U,  # Left singular vectors, shape (d, k)\n",
        "            's': s,  # Singular values, shape (k,)\n",
        "            'V': V,  # Right singular vectors, shape (d, k)\n",
        "        })\n",
        "\n",
        "    return all_svd_results"
      ],
      "metadata": {
        "id": "akb1Pb6CPfC6",
        "cellView": "form"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "#@title Lanczos token parallel\n",
        "def compute_svd_for_one_token(token_idx, output_function, embeddings, key, k=10, max_iter=50):\n",
        "    \"\"\"\n",
        "    Computes the top-k SVD for the Jacobian of a SINGLE specified input token.\n",
        "    This function is designed to be used with vmap.\n",
        "    \"\"\"\n",
        "    embed_dim = embeddings.shape[2]\n",
        "\n",
        "    # 1. Get the matvec functions for the current token\n",
        "    matvec_i, rmatvec_i = create_per_token_matvecs(output_function, embeddings, token_idx)\n",
        "\n",
        "    # The Jacobian J_i is d x d. So we compute (J_i^T J_i).\n",
        "    def jtj_matvec_i(v):\n",
        "        return rmatvec_i(matvec_i(v))\n",
        "\n",
        "    # 2. Run Lanczos (using Matfree) on this token's operator\n",
        "    matrix_size = embed_dim\n",
        "    max_iter_actual = min(max_iter, matrix_size - 1)\n",
        "    tridiag_fn = decomp.tridiag_sym(max_iter_actual, reortho=\"full\")\n",
        "\n",
        "    # Use the provided key to ensure reproducibility in vmap\n",
        "    template_vector = jax.random.normal(key, shape=(matrix_size,))\n",
        "\n",
        "    result = tridiag_fn(jtj_matvec_i, template_vector)\n",
        "\n",
        "    # 3. Extract singular values and vectors from the tridiagonal result\n",
        "    tridiag_matrix = result.J_small\n",
        "    basis_vectors = result.Q_tall\n",
        "\n",
        "    eigenvals, eigenvecs_tridiag = jnp.linalg.eigh(tridiag_matrix)\n",
        "\n",
        "    # Sort in descending order\n",
        "    idx = jnp.argsort(eigenvals)[::-1][:k]\n",
        "    top_eigenvals = eigenvals[idx]\n",
        "    top_eigenvecs_tridiag = eigenvecs_tridiag[:, idx]\n",
        "\n",
        "    # Reconstruct right singular vectors (V)\n",
        "    V = basis_vectors @ top_eigenvecs_tridiag\n",
        "\n",
        "    # Singular values are the sqrt of the eigenvalues\n",
        "    s = jnp.sqrt(jnp.maximum(top_eigenvals, 0))\n",
        "\n",
        "    # Compute left singular vectors: U = J_i @ V @ diag(1/s)\n",
        "    # Note: We can also vmap this inner loop for max efficiency\n",
        "    def compute_u(v_col, s_val):\n",
        "        u_col = matvec_i(v_col) / jnp.maximum(s_val, 1e-12)\n",
        "        return u_col\n",
        "\n",
        "    U = jax.vmap(compute_u, in_axes=(1, 0))(V, s)\n",
        "\n",
        "    # Return a dictionary of results for this single token\n",
        "    return {'U': U, 's': s, 'V': V}\n",
        "\n",
        "def compute_svd_for_all_tokens_vmapped(output_function, embeddings, k=3, max_iter=256):\n",
        "    \"\"\"\n",
        "    Computes the top-k SVD for EACH token's Jacobian in a fully parallelized way using vmap.\n",
        "    \"\"\"\n",
        "    batch_size, seq_len, embed_dim = embeddings.shape\n",
        "    assert batch_size == 1, \"This implementation assumes batch size of 1.\"\n",
        "\n",
        "    print(f\"--- Analyzing all {seq_len} tokens in parallel using vmap ---\")\n",
        "\n",
        "    # Create a vectorized version of our single-token function\n",
        "    # in_axes specifies how to handle each argument:\n",
        "    # 0: Map over the first axis of this argument (the token indices)\n",
        "    # None: Broadcast this argument (use the same value for all parallel runs)\n",
        "    # 0: Map over the first axis of this argument (the random keys)\n",
        "    vmapped_svd_computer = jax.vmap(\n",
        "        compute_svd_for_one_token,\n",
        "        in_axes=(0, None, None, 0, None, None)\n",
        "    )\n",
        "\n",
        "    # 1. Prepare inputs for vmap\n",
        "    token_indices = jnp.arange(seq_len)\n",
        "\n",
        "    # Create a batch of random keys, one for each token\n",
        "    key = jax.random.PRNGKey(42)\n",
        "    keys = jax.random.split(key, seq_len)\n",
        "\n",
        "    # 2. Run the vectorized function\n",
        "    # This executes all N computations in parallel on the accelerator\n",
        "    all_svd_results = vmapped_svd_computer(\n",
        "        token_indices, output_function, embeddings, keys, k, max_iter\n",
        "    )\n",
        "\n",
        "    # The results will have a leading batch dimension of size N\n",
        "    # U shape: (N, d, k), s shape: (N, k), V shape: (N, d, k)\n",
        "    print(f\"  vmap completed.\")\n",
        "    print(f\"  Output U shape: {all_svd_results['U'].shape}\")\n",
        "    print(f\"  Output s shape: {all_svd_results['s'].shape}\")\n",
        "    print(f\"  Output V shape: {all_svd_results['V'].shape}\")\n",
        "\n",
        "    return all_svd_results\n"
      ],
      "metadata": {
        "id": "0-ErXT29RLwj",
        "cellView": "form"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "import jax.numpy as jnp\n",
        "\n",
        "def reconstruct_output_from_svd(vmapped_svd_results, embeddings):\n",
        "    \"\"\"\n",
        "    Reconstructs the final output hidden state by summing the linear contribution\n",
        "    of each input token embedding, approximated by its Jacobian SVD.\n",
        "\n",
        "    Args:\n",
        "        vmapped_svd_results: The dictionary containing U, s, V from the vmapped run.\n",
        "                             U shape: (N, k, d), s shape: (N, k), V shape: (N, d, k)\n",
        "        embeddings: The original input embeddings, shape (1, N, d).\n",
        "\n",
        "    Returns:\n",
        "        A 1D JAX array of shape (d,) representing the reconstructed output.\n",
        "    \"\"\"\n",
        "    U = vmapped_svd_results['U']\n",
        "    s = vmapped_svd_results['s']\n",
        "    V = vmapped_svd_results['V']\n",
        "\n",
        "    # Squeeze the batch dimension from embeddings to get shape (N, d)\n",
        "    # where N is the sequence length and d is the embedding dimension.\n",
        "    e = embeddings.squeeze(0)\n",
        "\n",
        "    # Step 1: Project each embedding onto its corresponding right singular vectors (V).\n",
        "    # This is the V.T @ embedding part for all tokens at once.\n",
        "    # 'ndk,nd->nk' means: for each token n, multiply V_n.T (k,d) with e_n (d,) -> proj_n (k,)\n",
        "    projections = jnp.einsum('ndk,nd->nk', V, e)\n",
        "\n",
        "    # Step 2: Scale the projections by the singular values.\n",
        "    # This is the diag(s) @ (projections) part.\n",
        "    scaled_projections = s * projections\n",
        "\n",
        "    # Step 3: Map the scaled projections back to the output space with U.\n",
        "    # We need to transpose the last two dims of U from (N, k, d) to (N, d, k).\n",
        "    # 'nkd,nk->nd' means: for each token n, multiply U_n (d,k) with sp_n (k,) -> contrib_n (d,)\n",
        "    per_token_contributions = jnp.einsum('nkd,nk->nd', U, scaled_projections)\n",
        "\n",
        "    # Step 4: Sum the contributions from all tokens to get the final result.\n",
        "    reconstructed_output = jnp.sum(per_token_contributions, axis=0)\n",
        "\n",
        "    print(\"Reconstruction complete.\")\n",
        "    print(f\"  Shape of per-token contributions: {per_token_contributions.shape}\")\n",
        "    print(f\"  Shape of final reconstructed output: {reconstructed_output.shape}\")\n",
        "\n",
        "    return reconstructed_output\n"
      ],
      "metadata": {
        "id": "tA1e3Rc6-2gv"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "embeddings.shape"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "oNIulXrgreAW",
        "outputId": "f5538fd9-8b97-441d-8cf8-1e323bb289ff"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "(1, 427, 2560)"
            ]
          },
          "metadata": {},
          "execution_count": 48
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# 34540287760 OOM, 106 tokens, k=20,iter=128"
      ],
      "metadata": {
        "id": "B8HLuB6GHDU0"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "%%time\n",
        "# 1. Standard setup\n",
        "output_fn, emb, _, _ = setup_matfree_jacobian_analysis(\n",
        "    model, params, embeddings, use_detached_gradients=True\n",
        ")\n",
        "\n",
        "# 2. Run the new, fully parallelized vmap version\n",
        "# This single call replaces the entire for loop\n",
        "vmapped_results = compute_svd_for_all_tokens_vmapped(output_fn, emb, k=1, max_iter=128)\n",
        "\n",
        "i=0\n",
        "# You can access the results for the i-th token like this:\n",
        "U_i = vmapped_results['U'][i]  # Shape: (d, k)\n",
        "s_i = vmapped_results['s'][i]  # Shape: (k,)\n",
        "V_i = vmapped_results['V'][i]  # Shape: (d, k)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "v4XYB4IeI6yY",
        "outputId": "ad4cbf0a-8b18-40bf-8532-882de964c86e"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Matrix-free Jacobian setup:\n",
            "  Input size: 1,093,120\n",
            "  Output size: 2,560\n",
            "  Jacobian size: 2,560 x 1,093,120\n",
            "  Memory saved vs full: 10.4 GB\n",
            "--- Analyzing all 427 tokens in parallel using vmap ---\n",
            "  vmap completed.\n",
            "  Output U shape: (427, 1, 2560)\n",
            "  Output s shape: (427, 1)\n",
            "  Output V shape: (427, 2560, 1)\n",
            "CPU times: user 29min 39s, sys: 13min 46s, total: 43min 26s\n",
            "Wall time: 42min 47s\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [],
      "metadata": {
        "id": "-jTpkaa9RIJP"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "# jnp.array(vmapped_results['s']).shape\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "plt.plot(jnp.array(vmapped_results['s']).T[:,:]);"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 430
        },
        "id": "KC_v9aM9OIKe",
        "outputId": "02281d35-6eb0-4f78-aeb8-c74b3ff4e6c4"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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rFixItauXRvvec97sjwnnKxDhw7Fhz/84UgpxR133DHZw5lW+vv747bbbot77rknampqJns4U0rdZA9gOrruuuviE5/4xHH3ed3rXhetra2xb9++Mev/93//N373u99Fa2vrMR/X2toao6OjMTQ0NOb/kQ8ODpYf89BDD8X27dvjW9/6VkQ8f/dARMQZZ5wRf/d3fxdf+MIXTvDMpp7JnuujduzYERdeeGFcccUVccMNN5zQuVSbM844I0477bSX3Gl1rPk5qrW19bj7H/3v4OBgnHXWWWP2edvb3jaBo68elZjno46Gx29+85t46KGHTtlPPSIqM88//vGPY9++fWM+gT58+HBcd911ceutt8Z//dd/TexJVJPJvujkVHb0Isht27aV1z344IOv6CLIb33rW+V1O3fuHHMR5JNPPpm2b99eXu66664UEenhhx9+2au2p7tKzXVKKT3++OPpzDPPTJ/97GcrdwJT1OLFi9NVV11V/vnw4cPpNa95zXEv0Hv/+98/Zl1nZ+dLLjj90pe+VN5eKpVccDrB85xSSqOjo+miiy5K5557btq3b19lBl5lJnqen3nmmTH/Fm/fvj21tbWl66+/Pu3cubNyJ1IFxMcke+9735v+6I/+KG3dujX95Cc/SW94wxvG3P751FNPpTe96U1p69at5XVXXnll6ujoSA899FDatm1b6uzsTJ2dnS/7HD/4wQ9O+btdUqrMXG/fvj3NmzcvffSjH01PP/10eTlV/jHfuHFjamhoSPfcc0/asWNHuuKKK1JTU1MqFosppZQ+9rGPpc997nPl/X/605+murq69KUvfSn96le/SmvXrj3mrbZNTU3pu9/9bvrFL36RVqxY4VbbCZ7n0dHR9MEPfjDNnz8/PfbYY2NeuyMjI5NyjlNBJV7PL+Zul+eJj0n27LPPppUrV6bZs2enxsbGdPnll6f9+/eXt+/evTtFRPrBD35QXvf73/8+ffrTn06vfvWr06te9ar0oQ99KD399NMv+xzi43mVmOu1a9emiHjJcvbZZ2c8s8n11a9+NXV0dKT6+vq0ePHitGXLlvK2d77znemyyy4bs/83v/nN9MY3vjHV19enc889N/37v//7mO1HjhxJn//851NLS0tqaGhIF154Ydq1a1eOU5nSJnKej77Wj7W88PV/Kpro1/OLiY/n1aT0/y8IAADIwN0uAEBW4gMAyEp8AABZiQ8AICvxAQBkJT4AgKzEBwCQlfgAALISHwBAVuIDAMhKfAAAWYkPACCr/wcB4/eD/u8JzwAAAABJRU5ErkJggg==\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "\n",
        "# --- Example Usage ---\n",
        "# Assuming you have these variables from your previous run:\n",
        "vmapped_results = {'U': jnp.array(vmapped_results['U']), 's': jnp.array(vmapped_results['s']), 'V': jnp.array(vmapped_results['V'])}\n",
        "# embeddings = the original embeddings tensor\n",
        "\n",
        "reconstructed_output = reconstruct_output_from_svd(vmapped_results, embeddings)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "es7T0MOdOJus",
        "outputId": "58c5a567-e6ec-448d-ff76-b5cfe1ce9259"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Reconstruction complete.\n",
            "  Shape of per-token contributions: (427, 2560)\n",
            "  Shape of final reconstructed output: (2560,)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "argsorted = jnp.argsort(get_logits(reconstructed_output))#[::-1]\n",
        "# print([tokenizer.decode(argsorted[ii]) for ii in range(18)])\n",
        "print([tokenizer.decode(argsorted[::-1][ii]) for ii in range(18)])"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "Pgqfi0GKOMmj",
        "outputId": "73b2e147-9502-47f8-eafe-6a302e92345c"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[' infinity', ',', ';', ' and', '.', ' distance', ' space', ' existence', ' expression', ' reality', ' perfection', ' (', ' infinit', ' infinite', ' representation', ' form', ' eternity', ' emptiness']\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "for ii in range(embeddings.shape[1]):\n",
        "  for sv in range(1):\n",
        "    uvec = jnp.array(vmapped_results['U'])[-ii,sv,:].squeeze()\n",
        "    usign = jnp.sign(jnp.inner(reconstructed_output, uvec))\n",
        "    argsorted = jnp.argsort(usign*get_logits(uvec))#[::-1]\n",
        "    # print([tokenizer.decode(argsorted[ii]) for ii in range(18)])\n",
        "    print(ii,sv,vmapped_results['s'][-ii,sv],[tokenizer.decode(argsorted[::-1][jj]) for jj in range(8)],\"in=\",tokenizer.decode( prompt[ii:ii+1]))#print(ii,\"in=\",tokenizer.decode( prompt[ii:ii+1]),\", \",[tokenizer.decode(argsorted[::-1][jj]) for jj in range(18)])\n",
        "    # print(ii,\"in=\",tokenizer.decode( prompt[ii:ii+1]),\", \",[tokenizer.decode(argsorted[::-1][jj]) for jj in range(18)])"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "YrQguw7gvdIA",
        "outputId": "6274e8f7-6918-48e1-d741-25030353474c"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "0 0 11.565586 ['.', ',', ' –', '  ', '…', ' *', '—', ' …'] in= By\n",
            "1 0 48.636517 [' nothing', ' certainty', ' truth', ' devotion', ' minimum', 'nothing', ' thing', ' Nothing'] in=  this\n",
            "2 0 17.818209 [' reality', ' knowledge', ' expression', ' existence', ' power', ' unity', ' nature', ' equality'] in=  art\n",
            "3 0 17.47723 [' expression', ' existence', ' reality', ' knowledge', ' unity', ' organization', ' structure', ' equality'] in=  you\n",
            "4 0 11.216746 [' existence', ' expression', ' reality', ' structure', ' space', ' representation', ' nature', ' creation'] in=  may\n",
            "5 0 7.4371543 [' existence', ' reality', ' expression', ' space', ' structure', ' nature', ' representation', ' creation'] in=  contemplate\n",
            "6 0 7.165049 [' existence', ' reality', ' expression', ' space', '.', ' structure', ' creation', ' nature'] in=  the\n",
            "7 0 6.324123 [' room', ' rooms', ' Room', ' Rooms', 'Room', 'room', ' ROOM', 'Rooms'] in=  variations\n",
            "8 0 6.078444 [' reality', ' space', ' existence', ' architecture', ' structure', ' spatial', ' dimensions', ' expanse'] in=  of\n",
            "9 0 4.1748447 [' reality', ' space', ' existence', '.', ' expression', ' structure', ' form', ' creation'] in=  the\n",
            "10 0 5.7000837 [' reality', ' space', ' existence', '.', ' expression', ' form', ' creation', ' structure'] in=  \n",
            "11 0 6.779134 [' reality', ' existence', ' space', ' nature', ' creation', ' structure', ' representation', ' potential'] in= 2\n",
            "12 0 6.941889 [' ideal', ' idealism', ' ideals', ' Ideal', ' idealistic', 'ideal', '理想', ' idealized'] in= 3\n",
            "13 0 9.632455 [' ideal', ' Ideal', 'Ideal', ' ideals', 'ideal', ' idealism', '理想', ' идеа'] in=  letters\n",
            "14 0 6.1673007 [' reality', ' ideal', ' perfection', '.', ' existence', ' space', ' idealism', ' infinity'] in= ...\n",
            "15 0 5.287363 [' reality', '.', ' space', ' existence', ' infinity', ',', ' perfection', ' form'] in=  The\n",
            "16 0 35.071915 [' Levin', ' watches', 'watches', ' watch', 'watch', ' consuming', ' Watch', ' WATCH'] in=  Anatomy\n",
            "17 0 3.6419673 [' ending', ' endings', ' unending', 'ending', ' ends', ' end', ' termination', ' Ending'] in=  of\n",
            "18 0 3.3073394 [' ending', ' endings', ' space', '.', ' expression', ' reality', ' infinity', ' expanse'] in=  Mel\n",
            "19 0 2.5900435 [' reality', ' space', '.', ' ending', ' endings', ' expression', ' perfection', ' order'] in= anch\n",
            "20 0 3.1470864 ['.', ' reality', ' space', ' existence', ' infinity', ' expression', ',', ' perfection'] in= oly\n",
            "21 0 2.8135095 ['.', ' space', ' reality', ' infinity', ' existence', ',', ' expression', ' measure'] in= ,\n",
            "22 0 2.195843 ['.', ' space', ' reality', ',', ' infinity', ' existence', ' form', ' expression'] in=  part\n",
            "23 0 4.575278 ['.', ' reality', ' space', ' existence', ' expression', ',', ' infinity', ' form'] in=  \n",
            "24 0 4.6500483 [' infinite', ' infinity', ' infinit', ' inf', ' infinitely', ' INFIN', ' Infinite', ' infin'] in= 2\n",
            "25 0 2.286606 ['.', ' reality', ' space', ',', ' expression', ' infinity', ' existence', ' form'] in= ,\n",
            "26 0 2.279682 ['.', ' reality', ' space', ',', ' infinity', ' existence', ' form', ' expression'] in=  sect\n",
            "27 0 2.9263496 [' reality', '.', ' space', ' existence', ' infinity', ',', ' form', ' perfection'] in= .\n",
            "28 0 1.577353 ['.', ' space', ' existence', ' expanse', ' measure', ' form', ' reality', ','] in=  II\n",
            "29 0 2.0158546 ['.', ' reality', ' space', ' existence', ' form', ',', ' expression', ' infinity'] in= ,\n",
            "30 0 1.7641399 [' reality', '.', ' space', ' existence', ',', ' form', ' infinity', ' expression'] in=  mem\n",
            "31 0 2.6770754 [' potential', ' existence', ' conceptual', ' concept', ' entirety', ' dimensionality', ' sourcing', ' entity'] in= .\n",
            "32 0 2.5413983 [' gold', ' sample', ' золото', ' Gold', ' samples', 'サンプル', ' Sample', ' sampel'] in=  IV\n",
            "33 0 1.8015369 [' reality', ' existence', '.', ' space', ' expression', ' infinity', ' expanse', ' form'] in=  The\n",
            "34 0 1.4582465 [' reality', ' existence', '.', ' space', ' infinity', ' form', ' expanse', ' expression'] in=  universe\n",
            "35 0 1.8756325 [' expanse', ' existence', ' reality', ' totality', ' expression', ' transcendence', '.', ' infinity'] in=  (\n",
            "36 0 1.9494082 ['.', ' and', ' existence', ' reality', ' space', ' form', ' expression', ' infinity'] in= which\n",
            "37 0 1.8654672 [' expanse', '.', ' totality', ' and', ' reality', ' existence', ' form', ' infinity'] in=  others\n",
            "38 0 1.8622916 [' and', '.', ' space', ' reality', ' existence', ' form', ' expression', ' infinity'] in=  call\n",
            "39 0 2.6962936 [' ending', ' endings', ' unending', ' totality', ' expanse', ' end', ' and', ' ends'] in=  the\n",
            "40 0 1.1797354 [' reality', ' existence', ' and', ' expanse', ' space', '.', ' totality', ' infinity'] in=  Library\n",
            "41 0 1.255912 [' and', '.', ' reality', ' space', ' existence', ' infinity', ' expression', ' form'] in= )\n",
            "42 0 1.4613328 [' reality', '.', ' space', ' existence', ' infinity', ' totality', ' and', ' perfection'] in=  is\n",
            "43 0 1.2507455 ['.', ' reality', ' space', ' existence', ' and', ' infinity', ' form', ' expression'] in=  composed\n",
            "44 0 2.0377457 [';', '.', ' and', ' space', ' reality', ',', ' existence', ' form'] in=  of\n",
            "45 0 1.3006139 [' space', ' air', ' reality', '.', ' existence', ' and', ' expanse', ' infinity'] in=  an\n",
            "46 0 1.1622833 [' reality', ' existence', ' totality', ' expanse', '.', ' form', ' space', ' infinity'] in=  indefinite\n",
            "47 0 2.493833 [' expanse', ' totality', ' purview', ' dominion', ' transcendence', ' realm', ' actuality', ' perpetuity'] in=  and\n",
            "48 0 1.4027249 [' reality', ' existence', '.', ' space', ' totality', ' expanse', ' expression', ' form'] in=  perhaps\n",
            "49 0 0.7867683 [' space', ' reality', '.', ' existence', ' and', ' expression', ',', ' infinity'] in=  infinite\n",
            "50 0 1.0909694 ['.', ' and', ' space', ' existence', ' reality', ' expression', ' form', ' expanse'] in=  number\n",
            "51 0 1.4346406 [' reality', '.', ' space', ' infinity', ' perfection', ' existence', ' and', ' expanse'] in=  of\n",
            "52 0 1.0865462 ['.', ' reality', ' space', ' existence', ' and', ',', ' infinity', ' form'] in=  hexagonal\n",
            "53 0 1.9457567 [';', '.', ' space', ' reality', ' and', ' existence', ',', ' form'] in=  galleries\n",
            "54 0 2.4738832 [' Guzman', ' tile', ' NASA', ' Ortiz', ' USPS', ' tiles', ' Puerto', ' Tile'] in= ,\n",
            "55 0 1.181315 [' reality', '.', ' existence', ' space', ' infinity', ' expression', ' form', ' totality'] in=  with\n",
            "56 0 0.9794201 [' existence', ' space', '.', ' reality', ' thing', ' and', ' things', ' dimensions'] in=  vast\n",
            "57 0 1.0602717 ['.', ' space', ' reality', ' existence', ' infinity', ' expression', ' and', ' form'] in=  air\n",
            "58 0 0.92528445 [' space', '.', ' thing', ' infinity', ' measure', ' existence', ' and', ' reality'] in=  shafts\n",
            "59 0 1.3232143 [' reality', ' space', ' existence', '.', ' infinity', ' form', ' perfection', ' expression'] in=  between\n",
            "60 0 1.275019 [' reads', ' read', ' subjects', ' Research', ' research', ' Stage', ' subject', '輯'] in= ,\n",
            "61 0 2.6290648 [' expanse', ' entirety', ' governance', ' purview', ' existence', ' potential', ' realm', ' actuality'] in=  surrounded\n",
            "62 0 1.9845927 [' reality', ' existence', ' space', '.', ' expanse', ' totality', ' form', ' infinity'] in=  by\n",
            "63 0 4.9073997 [' impactful', ' vibe', ' existence', ' pieces', ' بشكل', ' minimalistic', ' concept', ' freaking'] in=  very\n",
            "64 0 1.0398022 [' existence', ' concept', ' space', ' form', ' dimensions', ' expanse', ' reality', ' and'] in=  low\n",
            "65 0 0.77827686 [' existence', ' space', ' reality', ' form', ' expanse', ' and', '.', ' dimensions'] in=  railings\n",
            "66 0 1.1300507 [' existence', ' reality', '.', ' space', ' and', ' form', ' expression', ' expanse'] in= .\n",
            "67 0 0.9278143 [' existence', '.', ' space', ' expanse', ' reality', ' form', ' and', ' expression'] in=  From\n",
            "68 0 1.6001008 [' reality', '.', ' existence', ' space', ' form', ';', ' infinity', ' and'] in=  any\n",
            "69 0 1.3692368 [' reality', ' existence', '.', ' space', ' form', ' expanse', ' totality', ' infinity'] in=  of\n",
            "70 0 1.0458521 ['.', ' reality', ' existence', ' space', ' expression', ',', ' form', ' and'] in=  the\n",
            "71 0 0.80833 [' existence', ' reality', '.', ' space', ' expanse', ' and', ' concept', ' potential'] in=  hexagons\n",
            "72 0 1.7745243 ['.', ' existence', ' space', ' reality', ' form', ' expression', ' and', ','] in=  one\n",
            "73 0 2.9059381 ['.', ' reality', ' space', ' existence', '<start_of_image>', ' infinity', ',', ' and'] in=  can\n",
            "74 0 2.0712628 [' reality', '.', ' space', ' existence', ' infinity', ' totality', ' expression', ' expanse'] in=  see\n",
            "75 0 1.3036202 [' reality', '.', ' existence', ' space', ' infinity', ' expression', ' expanse', ' totality'] in= ,\n",
            "76 0 0.69544536 ['.', ' reality', ' space', ' infinity', ' existence', ' and', ' expression', ' perfection'] in=  intermin\n",
            "77 0 1.2427132 [' reality', '.', ' existence', ' space', ' infinity', ' expression', ' totality', ' form'] in= ably\n",
            "78 0 0.98056066 [' reality', ' space', ' existence', '.', ' infinity', ' expanse', ' perfection', ' totality'] in= ,\n",
            "79 0 6.5848165 [' Gy', 'Gy', ' Suárez', ' Salazar', ' Luna', ' gy', ' Marquette', ' Braga'] in=  the\n",
            "80 0 1.1307486 [' reality', ' space', ' infinity', ' existence', '.', ' expression', ' expanse', ' perfection'] in=  upper\n",
            "81 0 1.2385216 [' existence', ' reality', ' space', ' expanse', ' infinity', ' form', ' emptiness', ' expression'] in=  and\n",
            "82 0 1.8230135 [' reimag', ' governance', ' envision', ' expanse', ' entirety', ' foundational', ' geospatial', ' depictions'] in=  lower\n",
            "83 0 0.93272597 [' reality', ' existence', ' space', ' expanse', ' infinity', ' totality', ' expression', ' perfection'] in=  floors\n",
            "84 0 0.8900009 [' reality', ' existence', ' space', ' infinity', ' expanse', '.', ' emptiness', ' totality'] in= .\n",
            "85 0 1.8168103 ['-\\u200b', '.\\u200b', '\\\\]', '\\ufeff\\ufeff', '\\u200b—', '.\\u200b\\u200b', '\\u200b.', ',\\u200b'] in=  The\n",
            "86 0 1.1166224 [' death', ' deaths', ' Death', ' die', ' dying', ' mortality', ' dies', ' dead'] in=  distribution\n",
            "87 0 1.1993574 [' reality', ' existence', ' space', ' infinity', ' expanse', ' totality', '.', ' perfection'] in=  of\n",
            "88 0 1.3289968 [' expanse', ' totality', ' existence', ' reality', '.', ' space', ' form', ' perpetuity'] in=  the\n",
            "89 0 0.95945925 [' reality', ' existence', ' space', '.', ' infinity', ' expanse', ' totality', ' perfection'] in=  galleries\n",
            "90 0 0.6820556 [' reality', '.', ' space', ' existence', ' infinity', ' and', ' expression', ' form'] in=  is\n",
            "91 0 1.5661591 [' reality', '.', ' space', ' existence', ' infinity', ' expression', ' expanse', ' perfection'] in=  invariable\n",
            "92 0 0.9616929 [' existence', ' reality', ' expanse', ' space', ' totality', '.', ' form', ' measure'] in= .\n",
            "93 0 0.6909826 [' space', '.', ' reality', ' existence', ' infinity', ' and', ' form', ' expression'] in=  Twenty\n",
            "94 0 1.0060986 [' what', 'what', ' What', 'What', ' whats', ' WHAT', 'WHAT', ' hvad'] in=  shelves\n",
            "95 0 1.6546096 [' existence', ' expanse', ' reality', ' totality', ' space', ' form', ' dimensionality', ' potential'] in= ,\n",
            "96 0 1.5289464 [' existence', ' expanse', ' potential', ' reality', ' form', ' totality', ' entirety', ' dimensions'] in=  five\n",
            "97 0 0.7520512 [' existence', ' space', ' reality', ' expanse', ' totality', ' form', '.', ' expression'] in=  long\n",
            "98 0 1.0146918 [' existence', ' reality', ' totality', ' expanse', ' measure', ' space', ' dimensionality', ' expression'] in=  shelves\n",
            "99 0 0.8639505 ['.', ' reality', ' space', ' existence', ' infinity', ' expanse', ' expression', ' and'] in=  per\n",
            "100 0 1.0176 [' existence', ' space', ' reality', '.', ' form', ' infinity', ' expanse', ' expression'] in=  side\n",
            "101 0 1.1215863 [' existence', ' reality', ' space', ' expanse', ',', '.', ' form', ' totality'] in= ,\n",
            "102 0 1.3376496 [' space', '.', ' reality', ' existence', ';', ' and', ' infinity', ','] in=  cover\n",
            "103 0 10.03205 [' şi', ' catalogs', 'catalog', ' Catalog', ' catalog', ' ş', 'Catalog', 'և'] in=  all\n",
            "104 0 1.8298945 [' space', ' existence', ' totality', ' reality', ' infinity', '.', ' perfection', ' magnitude'] in=  the\n",
            "105 0 4.376499 [' فى', ' التى', ' եւ', 'Colour', ' realises', ' și', ' recognises', ' customised'] in=  sides\n",
            "106 0 1.1951782 [' organisation', ' colour', ' realisation', ' endeavour', ' programme', ' splendour', ' labour', ' visualisation'] in=  except\n",
            "107 0 1.6906248 [' existence', ' space', ' infinity', ' totality', ' reality', '.', ' expression', ' expanse'] in=  two\n",
            "108 0 1.5208066 [' space', ' reality', ' existence', ' infinity', '.', ' and', ';', ' totality'] in= ;\n",
            "109 0 3.604767 [' book', ' Book', 'Book', 'book', ' books', ' BOOK', ' Books', ' книга'] in=  their\n",
            "110 0 0.8705775 [' book', ' books', ' infinity', ' space', ' perfection', ' expression', ' Book', ' and'] in=  height\n",
            "111 0 2.0100791 [' reality', ' infinity', ' space', ' existence', '.', ' totality', ' perfection', ','] in= ,\n",
            "112 0 1.1013286 [' reimag', ' sourcing', ' governance', ' purview', ' depictions', ' envision', ' entirety', ' encompassing'] in=  which\n",
            "113 0 0.8919275 [' space', ' infinity', ' existence', ' reality', ' totality', ' expanse', ' perfection', ';'] in=  is\n",
            "114 0 1.6575124 [' reality', ' existence', ' infinity', ' space', ' totality', ' expanse', ' perfection', '.'] in=  the\n",
            "115 0 0.6346259 [' existence', ' space', ' reality', ' infinity', ' expression', ' totality', ' and', ' expanse'] in=  distance\n",
            "116 0 0.63135666 [' and', '.', ' space', ' existence', ',', ';', ' infinity', ' expression'] in=  from\n",
            "117 0 1.4182701 [';', ' space', ' existence', ',', ' reality', ' and', '.', ' infinity'] in=  floor\n",
            "118 0 1.3724084 [' expanse', ' entirety', ' governance', ' existence', ' potential', ' totality', ' containment', ' reimag'] in=  to\n",
            "119 0 0.8489849 [' space', ' existence', ';', ' reality', '.', ' infinity', ' totality', ' expanse'] in=  ceiling\n",
            "120 0 0.79339063 [' reality', ' existence', ' space', ' expanse', ' totality', ' infinity', ' emptiness', ' perfection'] in= ,\n",
            "121 0 2.603233 [' فى', ' التى', ' եւ', ' realises', ' recognises', 'Colour', ' și', ' realisation'] in=  scarcely\n",
            "122 0 0.749132 [' reality', ' existence', ' space', '.', ' travel', ';', ' totality', ' infinity'] in=  exceeds\n",
            "123 0 0.7551978 ['.', ';', ' space', ' reality', ',', ' and', ' existence', ' infinity'] in=  that\n",
            "124 0 1.4171413 ['.', ' reality', ';', ' space', ' existence', ',', ' infinity', ' perfection'] in=  of\n",
            "125 0 3.290229 [' NASA', ' Library', 'NASA', ' NIH', 'Library', 'Davis', ' Woodrow', ' Jefferson'] in=  a\n",
            "126 0 1.0364751 [' Library', ' space', ' reality', ' existence', ' infinity', ' linearity', ' totality', ' order'] in=  normal\n",
            "127 0 1.8490322 [' reality', ' existence', ' space', '.', ' infinity', ' totality', ' expanse', ' perfection'] in=  bookcase\n",
            "128 0 3.387746 [' men', ' Men', 'Men', ' homens', ' hombres', ' uomini', ' women', 'men'] in= .\n",
            "129 0 1.0362722 [' men', ' sourcing', ' entirety', ' reimag', ' envision', ' governance', ' purview', ' sourced'] in=  One\n",
            "130 0 1.4497441 ['.', ' existence', ' reality', ' space', ' men', ' order', ' form', ';'] in=  of\n",
            "131 0 2.1039286 ['.', '.”', '.\"', ' –', '–', '--', '.’', '”.'] in=  the\n",
            "132 0 3.3729177 [' purview', ' reimag', ' cognizant', ' moniker', ' uptick', ' sourcing', ' burgeoning', ' governance'] in=  free\n",
            "133 0 1.7994148 ['.', ' space', ' existence', ' reality', ';', ' expanse', ' totality', ','] in=  sides\n",
            "134 0 1.9874272 [' purview', ' expanse', ' actuality', ' juncture', ' totality', ' entirety', ' wherein', ' governance'] in=  leads\n",
            "135 0 1.1074342 [' existence', '.', ' space', ' totality', ' reality', ' expanse', ' expression', ';'] in=  to\n",
            "136 0 1.4003519 [' expanse', ' entirety', ' governance', ' potential', ' existence', ' totality', ' purview', ' reality'] in=  a\n",
            "137 0 0.72976464 ['.', ' existence', ' space', ' reality', ' expanse', ' form', ',', ' totality'] in=  narrow\n",
            "138 0 1.1849767 [' packs', ' pack', ' plane', ' Peter', 'Peter', 'Stage', ' stage', 'pack'] in=  hallway\n",
            "139 0 1.2674309 ['.', ' space', ' existence', ' reality', ' totality', ' order', ' infinity', ';'] in=  which\n",
            "140 0 1.9627234 ['.', ' space', ' reality', ' existence', ' infinity', ' expression', ',', ';'] in=  opens\n",
            "141 0 2.6111948 [' Suárez', ' Purdue', ' Gy', ' Lithuanian', ' pizza', ' Guatem', 'Gy', ' Ignacio'] in=  onto\n",
            "142 0 0.93171346 [' space', ' existence', '.', ' reality', ' infinity', ' totality', ' expanse', ' expression'] in=  another\n",
            "143 0 0.92335457 [' space', '.', ' reality', ' existence', ' infinity', ' expanse', ' totality', ' expression'] in=  gallery\n",
            "144 0 1.7300898 ['.', ' reality', ' space', ' existence', ' infinity', ' totality', ' expanse', ';'] in= ,\n",
            "145 0 3.3154094 [' impactful', ' vibe', ' showcasing', ' onboarding', ' endeavors', ' Additionally', ' reimag', ' freaking'] in=  identical\n",
            "146 0 1.4754267 [' reimag', ' impactful', ' entirety', ' depictions', ' showcasing', ' potential', ' sourcing', ' concept'] in=  to\n",
            "147 0 3.2737386 [' stewardship', 'сті', ' reimag', ' stormwater', ' वैशिष्ट', 'рі', 'ріа', ' buildup'] in=  the\n",
            "148 0 1.7822794 [' existence', ' space', '.', ' reality', ' expanse', ' totality', ' expression', ' infinity'] in=  first\n",
            "149 0 0.94086415 [' existence', ' space', ' expanse', '.', ' reality', ' totality', ' expression', ' infinity'] in=  and\n",
            "150 0 0.9815629 [' space', '.', ' existence', ' reality', ' expression', ' expanse', ' totality', ','] in=  to\n",
            "151 0 1.2491056 ['.', ' space', ' existence', ' expression', ',', ' reality', ' form', ' expanse'] in=  all\n",
            "152 0 2.7095513 ['.', ' reality', ' space', ' existence', ',', ' form', ' expression', ' infinity'] in=  the\n",
            "153 0 4.287476 [' NASA', ' Ortiz', ' Vargas', ' Sloan', ' Katz', ' Peralta', '蒋', ' Lyman'] in=  rest\n",
            "154 0 2.073444 ['.', ' reality', ' space', ' existence', ' infinity', ' expanse', ' totality', ' expression'] in= .\n",
            "155 0 1.6437697 [' entirety', ' impactful', ' reimag', ' showcasing', ' endeavors', ' depictions', ' envision', ' depiction'] in=  To\n",
            "156 0 1.1603433 [' existence', ' expanse', ' space', ' expression', '.', ' totality', ' entity', ' reality'] in=  the\n",
            "157 0 1.4661262 [' wherein', ' entirety', ' expanse', ' purview', ' realms', ' tenets', ' rendition', ' depiction'] in=  left\n",
            "158 0 1.4038615 [' existence', '.', ' expanse', ' space', ' expression', ' form', ' reality', ' totality'] in=  and\n",
            "159 0 6.5356226 [' fruit', 'Fruit', ' Fruit', 'fruit', ' fruits', ' Fruits', 'Fruits', 'fruits'] in=  right\n",
            "160 0 2.0711763 [' fruit', ' fruits', ' Fruit', ' sphere', 'fruit', ' spheres', ' spherical', ' spher'] in=  of\n",
            "161 0 1.1780795 [' wherein', ' entirety', ' expanse', ' purview', ' breadth', ' perpetuity', ' form', ' portion'] in=  the\n",
            "162 0 1.2160524 [' space', ' existence', ' form', ' expanse', '.', ' reality', ' expression', ' dimensions'] in=  hallway\n",
            "163 0 1.9622213 [' reimag', ' sourcing', ' depictions', ' potential', ' envision', ' purview', ' entirety', ' governance'] in=  there\n",
            "164 0 1.2317938 ['.', ' existence', ' form', ' space', ' expanse', ' expression', ' reality', ' totality'] in=  are\n",
            "165 0 2.0629447 [' Light', ' light', 'Light', 'light', ' LIGHT', ' लाइट', 'LIGHT', ' Lights'] in=  two\n",
            "166 0 40.64381 [' \\\\&', '\\t', ' ``', '.\\\\\\\\', '\\t\\t', '}$\\\\\\\\', '$.\\\\\\\\', '?\\\\\\\\'] in=  very\n",
            "167 0 11.510092 [' infinity', ' infinite', ' INFIN', ' infinito', 'infinite', ' infinit', ' Infinite', ' Infinity'] in=  small\n",
            "168 0 2.3417056 [' infinity', ' infinite', ' infinit', ' infin', ' Infinity', ' ∞', ' INFIN', ' infinito'] in=  closets\n",
            "169 0 8.440747 [' prefer', ' preferred', ' Prefer', ' prefs', ' prefers', ' sensitivity', 'Preference', 'preferred'] in= .\n",
            "170 0 5.166829 [' thus', ' Таким', '라고', '이라고', '로서', '으로서', 'したがって', ' therefor'] in=  In\n",
            "171 0 4.386737 [' represent', ' representation', ' representing', ' rappresent', ' represents', 'represent', ' representa', ' Represent'] in=  the\n",
            "172 0 5.256508 [' office', 'Office', ' offices', 'office', ' Offices', ' Office', ' house', 'house'] in=  first\n",
            "173 0 6.8652363 [' Hover', '悬', ' Avon', ' elevator', ' Elevator', ' Otis', ' Chesapeake', ' Braga'] in= ,\n",
            "174 0 1.5912331 [' expanse', ' and', ' infinity', ' totality', ' expans', ' measure', ' breadth', ' dominion'] in=  one\n",
            "175 0 2.290727 [' expanse', ' potential', ' entirety', ' and', ' expans', ' infinit', ' inclusivity', ' infinity'] in=  may\n",
            "176 0 3.6997797 [' dream', ' Dream', ' dreams', 'Dream', ' dreaming', ' dreamed', 'dream', ' DREAM'] in=  sleep\n",
            "177 0 1.631571 [' expanse', ' infinity', ' and', ' infinit', ' measure', ' breadth', ' dreams', ' totality'] in=  standing\n",
            "178 0 2.1450663 [' reimag', ' inclusivity', ' expanse', ' sourcing', ' entirety', ' purview', ' governance', ' potential'] in=  up\n",
            "179 0 1.4304224 ['—', ' —', ' and', ' expanse', ',', ' infinity', ';', ' existence'] in= ;\n",
            "180 0 3.151433 ['</h3>', '.)', ' \\uf076', ',)', '<h3>', '###', '.]', '\\n\\n\\n'] in=  in\n",
            "181 0 2.0569046 [';', '—', '–', ' –', ' −', ' —', ' infinity', ' ·'] in=  the\n",
            "182 0 14.112174 [' şi', ' ş', ' gak', 'ឱ្យ', 'և', 'ţ', 'ผม', ' muß'] in=  other\n",
            "183 0 5.186098 [' resiliency', ' reimag', ' telehealth', ' geospatial', ' underserved', ' NYC', ' matchup', ' opioid'] in= ,\n",
            "184 0 1.414299 [' duplication', ' expanse', ' multiplicity', ' dimensionality', ' totality', ' replication', ' existence', ' infinity'] in=  satisfy\n",
            "185 0 1.524397 [' expanse', ' depiction', ' dimensionality', ' entirety', ' potential', ' reimag', '—', ' actuality'] in=  one\n",
            "186 0 2.2362967 [' reality', ' infinity', ' existence', ' potential', ' infinit', ' essence', ' concept', ' dimensionality'] in= '\n",
            "187 0 1.7210664 [' expanse', ' infinity', ' existence', ';', ' infinit', ' reality', ' totality', ' dimensionality'] in= s\n",
            "188 0 0.8998724 [';', ' infinity', '—', ',', ' existence', ' —', ' expanse', ' ('] in=  fecal\n",
            "189 0 1.4842451 [' existence', ';', ' infinity', ' potential', ' reality', ' dimensionality', ' –', ' expanse'] in=  necessities\n",
            "190 0 15.115411 ['–', ')–', ' –', '––', ' الَّذ', '</b>', ' –,', '.–'] in= .\n",
            "191 0 12.046697 [' infinity', ' infinite', ' infinit', ' infinito', ' INFIN', ' Infinite', 'infinite', ' infin'] in=  Also\n",
            "192 0 2.1883824 [' infinity', ' infinite', ' infinit', ' infinito', ' infin', ' infinitely', 'infinite', ' Infinite'] in=  through\n",
            "193 0 2.0418437 [' infinity', ' infinite', ' infinit', ' non', ' infinitely', ' infin', ' infinito', ' Infinity'] in=  here\n",
            "194 0 7.635129 [' Barth', ' NASA', 'Davis', ' Salazar', ' Davis', ' Braga', ' Woodrow', ' Pulitzer'] in=  passes\n",
            "195 0 1.8466337 [' Library', 'Library', ' Libraries', ' Lib', ' Bibliothe', ' Bibliothek', ' library', ' Libr'] in=  a\n",
            "196 0 1.9267464 [' infinity', ' infinit', ' infinite', ' expanse', ' (', ' non', ' breadth', ' emptiness'] in=  spiral\n",
            "197 0 1.3165423 [' infinity', ' expanse', ' infinit', ' infinite', ' dimensionality', ' imm', ' boundedness', ' linearity'] in=  stairway\n",
            "198 0 1.242578 [' infinity', ' infinit', ' expanse', ' infinite', ' breadth', ' multiplicity', ' extension', ' linearity'] in= ,\n",
            "199 0 1.4667028 [' infinity', ' expanse', ' existence', ' infinit', ' boundaries', ' dimensionality', ' linearity', ' reality'] in=  which\n",
            "200 0 3.1261013 [' dimensionality', ' expanse', ' purview', ' inclusivity', ' reimag', ' instantiation', ' Vaughn', ' geospatial'] in=  sinks\n",
            "201 0 3.0642784 [' reimag', ' yalnızca', ' depictions', ' purportedly', ' sourced', ' ostensibly', ' accolades', ' vibrancy'] in=  ab\n",
            "202 0 3.9566545 [' men', ' Men', 'Men', 'men', ' MEN', '男人', 'MEN', ' hombres'] in= ysm\n",
            "203 0 2.7191074 [' infinity', ' expanse', ' infinit', ' infinite', ' men', '—', ' —', '.'] in= ally\n",
            "204 0 11.247683 [' evaluations', ' eval', ' Evaluations', ' evaluation', 'Evaluation', 'Yay', 'eval', 'evaluation'] in=  and\n",
            "205 0 1.7705655 [' appearance', ' appearances', ' Appearance', ' appearing', 'appearance', ' Appear', ' appear', ' appeared'] in=  so\n",
            "206 0 6.7501493 [' şi', ' gak', 'ឱ្យ', ' ş', 'ผม', ' muß', 'և', ' ผม'] in= ars\n",
            "207 0 3.854058 [' reimag', ' tradeoffs', ' matchup', ' storefront', ' countertops', ' uptick', ' enfoc', ' depictions'] in=  upwards\n",
            "208 0 1.6309104 [' expanse', ' infinity', ' infinit', ' depiction', ' duplication', ' replication', ' actuality', ' breadth'] in=  to\n",
            "209 0 3.0602267 [' mirror', ' Mirror', 'Mirror', ' mirrors', 'mirror', ' Mirrors', ' mirrored', ' mirroring'] in=  remote\n",
            "210 0 0.90394235 [' duplication', ' infinity', ' replication', ' mirroring', ' infinit', ' mirrors', ' expanse', ' multiplicity'] in=  distances\n",
            "211 0 1.1990621 [' infinity', ' infinit', ' expanse', ' duplication', ' representation', ' infinite', ' replication', ' breadth'] in= .\n",
            "212 0 1.0223726 [' infinity', ' mirrors', ' duplication', ' infinit', ' expanse', ' mirroring', ' representation', ' replication'] in=  In\n",
            "213 0 1.3182684 [' infinity', ' expanse', ' infinit', ' duplication', ' multiplicity', ' infinite', ' representation', ' dimensionality'] in=  the\n",
            "214 0 1.1457498 [' infinity', ' infinit', ' expanse', ' representation', ' multiplicity', ' infinite', ' duplication', ' breadth'] in=  hallway\n",
            "215 0 1.3939203 [' infinity', ' expanse', ' infinit', ' breadth', ' representation', ' multiplicity', ' infinite', ' duplication'] in=  there\n",
            "216 0 2.4162092 [' infinity', ' expanse', ' and', ';', ' existence', ' reality', '.', ','] in=  is\n",
            "217 0 30.805132 [' distances', ' distance', ' Distance', 'Distance', ' Dist', 'distances', 'distance', ' dist'] in=  a\n",
            "218 0 6.8901744 [' distance', ' distances', ' Distance', 'Distance', 'distance', 'distances', ' distancia', ' distant'] in=  mirror\n",
            "219 0 5.932348 [' distance', ' distances', 'Distance', ' Distance', 'distance', 'distances', ' distancia', '距离'] in=  which\n",
            "220 0 2.6611917 [' distances', ' distance', ' Distance', 'distances', 'distance', ' distancing', ' districts', ' dist'] in=  faithfully\n",
            "221 0 1.4999393 [' distances', ' distance', 'Distance', ' Distance', 'distances', 'distance', ' distancia', ' Dist'] in=  duplicates\n",
            "222 0 1.306543 [' distances', ' distance', ' districts', ' lengths', ' expanse', ' breadth', ' infinity', ' dimensions'] in=  all\n",
            "223 0 1.8572803 [' distances', ' distance', ' districts', ' lengths', ' dist', ' dimensions', ' infinity', ' expanse'] in=  appearances\n",
            "224 0 0.97112745 [' distances', ' distance', ' expanse', ' dimensions', ' infinit', ' lengths', ' infinity', ' districts'] in= .\n",
            "225 0 1.3748343 [' infinity', ' distance', ' distances', ' representation', ' infinit', ' districts', ' expanse', ' infinite'] in=  Men\n",
            "226 0 3.9917724 [' infinity', ' reality', ' space', ' existence', ' dimensionality', ' emptiness', ' dimensions', ' beauty'] in=  usually\n",
            "227 0 1.7793648 [' distances', ' distance', ' dimensions', ' districts', ' dist', ' distancing', ' dimensionality', ' dimension'] in=  infer\n",
            "228 0 1.4590901 [' distances', ' distance', ' districts', ' expanse', ' lengths', ' infinity', ' dimensions', ' infinit'] in=  from\n",
            "229 0 1.9808658 [' distances', ' distance', ' expanse', ' infinity', ' districts', ' infinit', ' dimensions', ' breadth'] in=  this\n",
            "230 0 4.888068 [' gak', ' şi', ' এছাড়াও', 'ผม', 'と言う', 'ครับ', ' ผม', ' এছাড়া'] in=  mirror\n",
            "231 0 2.0844207 [' matchup', ' USPS', ' tradeoffs', ' storefront', ' Walmart', ' nonprofit', ' Verizon', ' Yelp'] in=  that\n",
            "232 0 0.9454603 [' distances', ' distance', ' districts', ' expanse', ' infinity', ' lengths', ' abstraction', ' dimensions'] in=  the\n",
            "233 0 1.5536826 [' depictions', ' reimag', ' entirety', ' tradeoffs', ' storefront', ' nonfiction', ' countertops', ' enfoc'] in=  Library\n",
            "234 0 7.1648 [' এছাড়াও', ' enfoc', 'と言う', ' şi', ' এছাড়া', ' বাড়ি', ' tradeoffs', ' centerpiece'] in=  is\n",
            "235 0 3.1094384 [' enfoc', ' tradeoffs', ' storefront', ' এছাড়াও', ' centerpiece', ' lackluster', ' gray', ' portions'] in=  not\n",
            "236 0 2.9443562 [' depictions', ' entirety', ' reimag', ' purview', ' portions', ' depiction', ' sourcing', ' tradeoffs'] in=  infinite\n",
            "237 0 2.9381456 [' distance', ' distances', ' expanse', '—', ' infinity', ' —', ' breadth', ' dimensions'] in=  (\n",
            "238 0 4.3086505 [' purview', ' cognizant', ' moniker', ' depictions', ' envision', ' atop', ' matchup', ' referenced'] in= if\n",
            "239 0 4.7763467 [' reimag', ' sourcing', ' countertops', ' geospatial', ' depictions', ' badass', ' sourced', ' inclusivity'] in=  it\n",
            "240 0 1.3537692 [' expanse', '—', ' —', ' existence', ' depiction', ' dimensionality', ' dimensions', ' breadth'] in=  were\n",
            "241 0 7.4592047 ['.', ',', ';', ' (', ' and', '…', '...', ':'] in= ,\n",
            "242 0 9.38265 ['.', ',', ';', ' (', ' and', '...', '…', ':'] in=  why\n",
            "243 0 5.5022235 [' ce', ' digitization', ' logistics', ' element', ' Newport', ' rotation', ' reproduction', ' porosity'] in=  this\n",
            "244 0 2.208464 [' distance', ' expanse', ' infinity', ' distances', ';', ' breadth', ' infinit', ' and'] in=  illusory\n",
            "245 0 1.0994623 [' and', ',', ';', ' infinity', ' distance', ' (', '.', ' infinit'] in=  duplication\n",
            "246 0 8.959039 [' ‘', ' “', '…', ' …', '’', '”', '…”', '.’'] in= ?\n",
            "247 0 6.539881 [';', ',', ' (', ' and', '.', ' or', '-', ':'] in= );\n",
            "248 0 5.6316934 [';', ' (', ',', ' and', '.', ' or', ':', '-'] in=  I\n",
            "249 0 2.3700922 [';', ' (', ',', ' and', '.', ' or', '-', ':'] in=  prefer\n",
            "250 0 3.3726244 [';', ',', ' (', ' and', '.', ' or', '-', ' -'] in=  to\n",
            "251 0 2.8523543 [' and', ' (', ',', ';', '.', ' or', ':', '-'] in=  dream\n",
            "252 0 2.9746473 [';', ',', ' (', ' and', '.', '-', ' -', ' or'] in=  that\n",
            "253 0 2.1702878 [' (', ',', ';', ' and', '.', ':', ' or', '-'] in=  its\n",
            "254 0 3.9448497 [';', ' (', ' and', ',', '.', ' or', ':', '-'] in=  polished\n",
            "255 0 6.7138777 [' (', ' and', ',', '.', ' being', ';', ' “', ' or'] in=  surfaces\n",
            "256 0 4.0165462 [' (', ';', ',', ' and', '.', ' or', ':', '...'] in=  represent\n",
            "257 0 4.6186757 [';', ' (', ',', ' and', '.', ':', ' or', ' -'] in=  and\n",
            "258 0 4.79875 [';', ' (', ',', ' and', '.', ' or', ':', ' -'] in=  promise\n",
            "259 0 4.255494 [',', ';', ' (', ' and', '.', ' or', ':', '-'] in=  the\n",
            "260 0 1.5036553 [';', ' (', ',', ' and', '.', ':', ' or', ' -'] in=  infinite\n",
            "261 0 1.8877246 [';', ',', ' (', ' and', '.', ':', ' infinity', ' or'] in=  ...\n",
            "262 0 1.8332223 [' (', ';', ',', ' and', '.', ':', ' or', ' -'] in=  Light\n",
            "263 0 1.9829767 [';', ',', ' (', ' and', '.', ':', ' -', ' or'] in=  is\n",
            "264 0 1.9257396 [',', ';', ' (', ' and', '.', ':', ' or', '-'] in=  provided\n",
            "265 0 1.9637107 [';', ' (', ' and', ',', ' distance', '.', ' infinity', ' distances'] in=  by\n",
            "266 0 1.8819256 [';', ',', ' (', ' and', '.', ':', '-', ' or'] in=  some\n",
            "267 0 3.054512 [';', ' and', ',', ' (', '.', ' infinity', ' distance', ' expression'] in=  spherical\n",
            "268 0 1.2690233 [';', ',', ' (', ' and', '.', ':', ' or', '-'] in=  fruit\n",
            "269 0 2.1671512 [';', ' and', ',', ' (', '.', ' infinity', ' distance', ':'] in=  which\n",
            "270 0 1.9001263 [',', ' (', ' and', ';', '.', ':', ' or', '-'] in=  bear\n",
            "271 0 2.1243758 [',', ';', ' (', ' and', '.', ':', '-', ' -'] in=  the\n",
            "272 0 2.7619648 [';', ',', ' (', ' and', '.', ':', '-', ' -'] in=  name\n",
            "273 0 4.0062165 [',', ';', ' (', ' and', '.', ':', '-', ' or'] in=  of\n",
            "274 0 2.5671422 [' flawed', ' tradeoffs', '良い', ' centerpiece', ' storefront', ' lackluster', ' reimag', 'և'] in=  lamps\n",
            "275 0 1.9342747 [',', ';', ' and', ' (', '.', ':', '-', ' distance'] in= .\n",
            "276 0 1.1008521 [',', ';', ' (', ' and', '.', ':', ' infinity', ' expression'] in=  There\n",
            "277 0 1.8064253 [' and', ';', ',', ' (', '.', ' infinity', ' distance', ':'] in=  are\n",
            "278 0 2.1234252 [' and', ';', ',', ' (', '.', ' distance', ' infinity', ':'] in=  two\n",
            "279 0 1.6339079 [';', ',', ' (', ' and', '.', ':', ' or', '-'] in= ,\n",
            "280 0 1.7992399 [' and', ';', ',', ' (', '.', ':', '-', ' distance'] in=  transvers\n",
            "281 0 1.8861363 [' and', ';', ',', ' (', '.', ' infinity', ' distance', ':'] in= ally\n",
            "282 0 2.976665 [' şi', ' gak', 'ឱ្យ', 'と言う', 'ผม', ' এছাড়াও', 'ţ', 'և'] in=  placed\n",
            "283 0 2.7515213 [';', ',', ' and', ' (', '.', ' infinity', ':', ' expression'] in= ,\n",
            "284 0 4.2627172 [' Salazar', ' Pendleton', ' Ritz', ' Yale', ' Fritz', ' Gers', ' Barth', ' 박'] in=  in\n",
            "285 0 1.6951294 [' and', ',', ';', ' (', ' infinity', '.', ' expression', ' distance'] in=  each\n",
            "286 0 2.9577656 [' and', ',', ';', ' (', '.', ' infinity', ' expression', ' space'] in=  hexagon\n",
            "287 0 2.0728562 [' and', ',', ';', ' (', '.', ' infinity', ' expression', ' space'] in= .\n",
            "288 0 1.9013458 [' and', ',', ';', ' (', '.', ' infinity', ':', ' expression'] in=  The\n",
            "289 0 2.9435868 [' şi', ' gak', 'と言う', ' এছাড়াও', 'ឱ្យ', 'ผม', ' এছাড়া', ' ผม'] in=  light\n",
            "290 0 1.7700223 [' and', ',', ';', ' (', ' infinity', '.', ' distance', ' expression'] in=  they\n",
            "291 0 1.2019753 [',', ' and', ';', ' (', '.', ' infinity', ':', ' distance'] in=  emit\n",
            "292 0 1.9961011 [',', ' and', ';', ' (', '.', ' infinity', ' distance', ':'] in=  is\n",
            "293 0 2.0848715 [',', ' and', ';', ' (', '.', ' infinity', ' expression', ':'] in=  insufficient\n",
            "294 0 1.4766599 [',', ' and', ';', ' (', '.', '-', ' infinity', ' expression'] in= ,\n",
            "295 0 1.5794595 [',', ' and', ';', ' (', '.', ' infinity', '-', ' space'] in=  incessant\n",
            "296 0 1.7572668 [' and', ',', ';', ' (', '.', ' infinity', ':', '-'] in= .\n",
            "297 0 3.0064878 [';', ' and', ' infinity', ',', ' (', ' distance', '.', ' expanse'] in=  Like\n",
            "298 0 1.5986933 [',', ' and', ';', ' (', '.', ':', '-', ' infinity'] in=  all\n",
            "299 0 4.2886 [',', ' and', ';', ' (', '.', ':', '-', ' space'] in=  men\n",
            "300 0 5.0421453 [' nonprofit', ' nonprofits', ' matchup', ' postseason', ' resiliency', ' nonfiction', ' USPS', ' modeling'] in=  of\n",
            "301 0 4.0111165 [' reimag', ' sourcing', ' sourced', ' countertops', ' curated', ' impactful', ' depictions', ' accolades'] in=  the\n",
            "302 0 1.3731726 [',', ' (', ' and', ';', ' infinity', '.', ' distance', ' existence'] in=  Library\n",
            "303 0 3.247738 [' and', ' infinity', ';', ',', ' (', '.', ' expanse', ' distance'] in= ,\n",
            "304 0 1.8486688 [',', ';', ' and', ' (', '.', ' infinity', ' expression', ' existence'] in=  I\n",
            "305 0 2.197865 [',', ';', ' expanse', ' and', ' (', '.', ' existence', ' expression'] in=  have\n",
            "306 0 3.7760515 [';', ',', ' and', ' (', ' expanse', '.', ' infinity', ' existence'] in=  traveled\n",
            "307 0 2.679988 [';', ' (', ',', ' and', '.', ' infinity', ' expression', '-'] in=  in\n",
            "308 0 1.7236391 [',', ' and', ' expanse', ' (', ';', '.', '-', '—'] in=  my\n",
            "309 0 1.7120533 [',', ' and', ';', ' (', '.', ' infinity', ' expression', '-'] in=  youth\n",
            "310 0 1.2487385 [',', ' and', ' (', ';', ' infinity', ' distance', '.', '-'] in= ;\n",
            "311 0 2.051942 [' localisation', ' utilisation', 'segment', ' sign', ' segment', ' realisation', ' Segment', ' classification'] in=  I\n",
            "312 0 2.637024 [' distance', ' distances', ' Distance', 'distance', 'distances', 'Distance', ' dist', ' Dist'] in=  have\n",
            "313 0 1.8036431 [' distance', ' distances', ',', ' and', ';', ' (', '.', ' distribution'] in=  wandered\n",
            "314 0 1.5017828 [',', ';', ' and', ' (', ' distance', '.', ' infinity', ' expression'] in=  in\n",
            "315 0 1.8517098 [' distance', ',', ' and', ';', ' (', ' distances', ' infinity', '.'] in=  search\n",
            "316 0 2.5183945 [';', ',', ' and', ' (', '.', ' distance', ' infinity', ' expression'] in=  of\n",
            "317 0 1.5190564 [',', ';', ' and', ' (', ' distance', '.', ' existence', ' infinity'] in=  a\n",
            "318 0 1.5785469 [',', ';', ' and', '—', ' (', ' —', '-', ' infinity'] in=  book\n",
            "319 0 2.8156571 [',', ';', ' and', ' (', '—', '.', ' —', '-'] in= ,\n",
            "320 0 1.4210924 [',', ';', ' (', ' and', '.', ':', '-', ' infinity'] in=  perhaps\n",
            "321 0 2.544183 [',', ';', ' and', ' (', '.', ' infinity', ' distance', ' existence'] in=  the\n",
            "322 0 1.5729686 ['良い', ' tradeoffs', ' sourcing', ' postseason', ' centerpiece', ' storefront', ' midrange', ' reimag'] in=  catalogue\n",
            "323 0 1.7850965 [',', ' and', ';', ' (', '.', ':', '-', ' expression'] in=  of\n",
            "324 0 1.234927 [',', ' and', ';', ' (', '.', '-', ' expression', ' existence'] in=  catalogues\n",
            "325 0 1.5423981 [',', ';', ' (', ' and', '.', ' form', '-', ' existence'] in= ;\n",
            "326 0 2.6598017 [',', ';', ' (', ' and', '.', ' infinity', ' distance', ':'] in=  now\n",
            "327 0 1.5998429 [',', ' and', ';', ' (', ' infinity', ' space', ' distance', '.'] in=  that\n",
            "328 0 3.3219218 [' فى', ' التى', ' եւ', ' realises', ' recognises', 'Colour', ' utilises', ' customised'] in=  my\n",
            "329 0 1.6605678 [',', ' (', ' and', ';', ' distance', '.', ' expanse', ' existence'] in=  eyes\n",
            "330 0 1.0423007 [',', '—', ' existence', ';', ' form', '.', ' (', ' and'] in=  can\n",
            "331 0 1.5341192 [',', ';', ' and', ' (', '.', '-', ' expression', ' infinity'] in=  hardly\n",
            "332 0 2.8582532 [',', ';', ' (', ' and', '.', ' expression', ' infinity', '-'] in=  decipher\n",
            "333 0 2.7113116 [',', ' and', ' (', ';', '.', ' space', ' infinity', ' existence'] in=  what\n",
            "334 0 2.4589012 [',', ' and', ' (', ';', '.', ' space', ' expression', ' infinity'] in=  I\n",
            "335 0 3.5798042 [',', ' and', ';', ' (', '.', ':', '-', ' space'] in=  write\n",
            "336 0 5.374786 [',', ' and', ';', ' (', '.', ' existence', ' infinity', ' expression'] in= ,\n",
            "337 0 1.9172642 [',', ' (', ' and', ';', '.', ':', ' expression', '-'] in=  I\n",
            "338 0 3.0608366 [' Suresh', ' गुर्जर', ' Perth', ' Aussie', ' Cyril', ' Guernsey', ' Melbourne', ' Landscape'] in=  am\n",
            "339 0 1.9221673 [',', ' and', ';', ' (', '.', ':', ' infinity', ' expression'] in=  preparing\n",
            "340 0 3.5047567 [',', ' and', ';', ' (', ' infinity', '.', ' distance', ' expression'] in=  to\n",
            "341 0 4.7128463 [' distribution', ' Distribution', 'Distribution', 'distribution', ' distributions', ' distributing', ' distribute', ' distribución'] in=  die\n",
            "342 0 2.8560102 [',', ' distribution', ';', ' (', ' and', '.', ':', '-'] in=  just\n",
            "343 0 5.1019187 [',', ' (', ' and', ';', '.', ':', '-', ' space'] in=  a\n",
            "344 0 2.3409822 [',', ' (', ' and', ';', '.', ' distribution', ' infinity', ' expression'] in=  few\n",
            "345 0 1.2300158 [',', ' (', ';', ' and', '.', ' distribution', ' infinity', ' distance'] in=  leagues\n",
            "346 0 2.3952436 [',', ' and', ';', ' (', '.', ' infinity', ' distribution', ' distance'] in=  from\n",
            "347 0 1.3381587 [' and', ',', ' (', ';', ' infinity', '.', ' expression', ' space'] in=  the\n",
            "348 0 2.0675614 [',', ' and', ';', ' (', '.', ':', ' infinity', ' expression'] in=  hexagon\n",
            "349 0 3.251205 [',', ' and', ';', ' (', '.', ' infinity', ':', ' expression'] in=  in\n",
            "350 0 1.7728236 [',', ' and', ';', ' (', '.', ' infinity', ':', '-'] in=  which\n",
            "351 0 4.6341686 [' and', ',', ';', ' (', ' infinity', '.', ' expression', ' expanse'] in=  I\n",
            "352 0 3.355669 [',', ' and', ';', ' (', '.', ' infinity', ':', ' expression'] in=  was\n",
            "353 0 1.7740488 [',', ' and', ' (', ';', '.', ':', ' infinity', ' space'] in=  born\n",
            "354 0 1.9983187 [',', ' and', ' (', ';', '.', ':', ' infinity', '-'] in= .\n",
            "355 0 1.88851 [',', ' and', ' (', ';', '.', ':', '-', ' or'] in=  Once\n",
            "356 0 2.3359814 [',', ' and', ';', ' (', '.', ' infinity', ' distribution', ' expression'] in=  I\n",
            "357 0 2.1380258 [',', ' and', ';', ' (', '.', ':', ' infinity', ' expression'] in=  am\n",
            "358 0 3.4870605 [',', ' and', ';', ' (', '.', ' infinity', ' distance', ' distribution'] in=  dead\n",
            "359 0 2.1050265 [',', ' and', ';', ' (', '.', ' infinity', ':', ' expression'] in= ,\n",
            "360 0 2.8782654 [',', ' and', ' (', ';', '.', ':', ' infinity', '-'] in=  there\n",
            "361 0 6.4657927 [',', ' and', ' (', ';', '.', ' infinity', ':', ' space'] in=  will\n",
            "362 0 3.828753 [',', ' and', ';', ' (', '.', ' infinity', ' space', ' expression'] in=  be\n",
            "363 0 1.4669617 [',', ' and', ' (', ';', '.', ' infinity', ':', ' space'] in=  no\n",
            "364 0 2.058154 [',', ' and', ';', ' (', '.', ' infinity', ' expression', ' distance'] in=  lack\n",
            "365 0 1.9947276 [',', ';', ' and', ' (', '.', ' infinity', ' distance', ':'] in=  of\n",
            "366 0 2.524108 [',', ' and', ';', ' (', ' infinity', '.', ' distance', ' expression'] in=  pious\n",
            "367 0 3.727002 [',', ' and', ';', ' (', '.', ' infinity', ' expression', ':'] in=  hands\n",
            "368 0 2.4962304 [',', ' and', ';', ' (', '.', ' infinity', ' expression', ' distance'] in=  to\n",
            "369 0 3.5947201 [' reimag', ' sourcing', ' blockchain', ' badass', ' curation', ' songwriting', ' hipster', ' geospatial'] in=  throw\n",
            "370 0 5.957743 [' Flows', ' Trace', 'Charges', ' book', ' Book', ' الكتاب', ' charges', 'Book'] in=  me\n",
            "371 0 2.6040373 [' air', ',', ' infinity', ' and', ';', ' (', ' Air', ' distance'] in=  over\n",
            "372 0 3.0389137 [',', ' and', ';', ' (', ' infinity', '.', ' space', ' air'] in=  the\n",
            "373 0 4.8526864 [',', ' and', ';', ' (', '.', ' infinity', ' space', ' existence'] in=  railing\n",
            "374 0 4.9445367 [' existence', ' space', ' sequencing', ' spacetime', ' expanse', ',', ' architecture', ' abundance'] in= ;\n",
            "375 0 5.725513 [' hexagon', ' hexagons', ' hexagonal', ' hex', ' Hex', ' HEX', 'Hex', ' triangulation'] in=  my\n",
            "376 0 4.3703365 [' and', ',', ';', ' infinity', ' (', ' space', '.', ' expression'] in=  grave\n",
            "377 0 2.8879163 [' infinity', ' infinit', ' infinite', ',', ' and', ' eternity', ' number', ';'] in=  will\n",
            "378 0 6.218189 [' infinity', ' Infinity', ' infinite', ' INFIN', ' infin', ' infinit', 'infinity', ' infinito'] in=  be\n",
            "379 0 4.1072855 [' infinity', ',', ' and', ' (', ';', ' infinite', '.', ' infinit'] in=  the\n",
            "380 0 3.4883347 [' infinity', ' and', ',', ' (', ';', ' infinite', ' infinit', '.'] in=  fathom\n",
            "381 0 5.62625 [' sourcing', ' reimag', ' expanse', ' inclusivity', ' physicality', ' depiction', ' depictions', ' dimensionality'] in= less\n",
            "382 0 2.2148786 [' infinity', ' and', ',', ' (', ';', ' infinite', ' infinit', '.'] in=  air\n",
            "383 0 4.716665 [' and', ',', ' infinity', ';', ' (', '.', ' infinite', ' infinit'] in= ;\n",
            "384 0 4.946603 [' composition', ' compositions', ' Composition', ' compositional', ' construction', ' and', ' compos', ' composing'] in=  my\n",
            "385 0 3.1784859 [' and', ',', ' composition', ' infinity', ';', ' (', '.', ' construction'] in=  body\n",
            "386 0 4.8613644 [' and', ',', ' infinity', ' (', ';', '.', '-', ' infinite'] in=  will\n",
            "387 0 9.261445 [' Library', ' library', 'Library', ' Libraries', ' libraries', 'library', ' biblioteca', 'Libraries'] in=  sink\n",
            "388 0 3.234491 [' Library', ' library', ' libraries', ' Libraries', 'Library', ' bibli', ' librarians', ' books'] in=  endlessly\n",
            "389 0 2.767039 [',', ' infinity', ' infinit', ' dimensionality', ' expanse', ' space', '.', ' linearity'] in=  and\n",
            "390 0 2.7464883 [',', ' and', ' infinity', ';', ' (', '.', '-', ' infinit'] in=  decay\n",
            "391 0 3.57556 [',', ' and', ';', ' infinity', ' (', '.', ' …', ' –'] in=  and\n",
            "392 0 4.0862613 [',', ' (', ' and', ';', ' –', '.', ' infinity', ' …'] in=  dissolve\n",
            "393 0 6.0024548 [' infinity', ',', ' and', ' infinit', ' (', ' infinite', ';', '.'] in=  in\n",
            "394 0 5.9849195 [',', ' and', ' infinity', '.', ' (', ';', ' space', ':'] in=  the\n",
            "395 0 2.7721007 [',', ' and', ' infinity', '.', ' (', ';', '-', ' space'] in=  wind\n",
            "396 0 4.503907 [',', ' and', ' (', '.', ';', ' infinity', ':', ' …'] in=  generated\n",
            "397 0 2.7495887 [',', ' infinity', ' and', ' (', ';', '.', ' infinit', ' space'] in=  by\n",
            "398 0 4.3434944 [',', ' and', ' (', ';', ' infinity', '.', ' *', ':'] in=  the\n",
            "399 0 2.37064 [',', ' and', ' (', ' infinity', ';', '.', ' *', ':'] in=  fall\n",
            "400 0 5.4532614 [',', ' and', ' (', ';', '.', ' infinity', ':', ' …'] in= ,\n",
            "401 0 3.767256 [',', ' and', ' (', ' infinity', ';', '.', ' *', ' …'] in=  which\n",
            "402 0 4.8164997 [',', ' and', ' (', ';', '.', ' infinity', ' *', ':'] in=  is\n",
            "403 0 2.4767032 [',', ' (', '.', ' and', ' infinity', ';', ' …', ':'] in=  infinite\n",
            "404 0 2.727347 [' and', '.', ',', ' infinity', ' (', ' existence', ';', ' space'] in= .\n",
            "405 0 2.6688747 [',', ' (', ';', ' and', '.', ' –', ':', ' infinity'] in=  I\n",
            "406 0 6.1582637 [',', ';', ' (', ' and', '.', ' infinity', ' –', ':'] in=  say\n",
            "407 0 2.8201747 [' infinity', ' space', ' existence', ' (', ',', ' expression', ';', ' infinit'] in=  that\n",
            "408 0 2.8772902 [',', ' (', ' infinity', ' space', ' and', ';', '.', ' beauty'] in=  the\n",
            "409 0 16.262062 [' infinit', ' existence', ' space', ' infinity', ' dimensionality', ' architecture', ' emptiness', ' stillness'] in=  Library\n",
            "410 0 6.433238 [' infinity', ',', ';', ' (', ' and', '.', ' existence', ' infinit'] in=  is\n",
            "411 0 7.520422 [' infinity', ' infinit', ' existence', ',', ';', ' infinite', ' (', ' abundance'] in=  unending\n",
            "412 0 6.964058 [',', ' (', '.', ';', ' and', ':', ' infinity', ' *'] in= .\n",
            "413 0 7.0155277 [',', ' (', ';', ' and', ' infinity', '.', ' existence', ' –'] in=  The\n",
            "414 0 5.8303604 [',', ' (', ';', ' and', ' infinity', '.', ':', ' distance'] in=  ideal\n",
            "415 0 2.1445181 [',', ' (', ';', ' infinity', ' and', '.', ':', ' imm'] in= ists\n",
            "416 0 3.6483305 [',', ' (', ';', ' infinity', '.', ' and', ' infinit', ' distance'] in=  argue\n",
            "417 0 5.1920576 [',', ';', ' (', '.', ' and', ' infinity', ':', ' expression'] in=  that\n",
            "418 0 4.809524 [',', ';', ' (', '.', ' and', ':', ' infinity', ' *'] in=  the\n",
            "419 0 7.194552 [',', ';', ' (', ' and', ' infinity', '.', ':', ' *'] in=  hexagonal\n",
            "420 0 5.3175635 [',', ';', ' infinity', ' (', ' and', ' *', '.', ' imm'] in=  rooms\n",
            "421 0 5.477013 [',', ' (', ';', '.', ' and', ' *', ' infinity', ':'] in=  are\n",
            "422 0 6.445588 [',', ';', ' (', ' infinity', ' and', '.', ' *', ':'] in=  a\n",
            "423 0 4.303235 [',', ';', ' (', '.', ' *', ' and', ' infinity', ':'] in=  necessary\n",
            "424 0 3.659415 [',', '.', ' *', ' infinity', ' (', ';', ' imm', ':'] in=  form\n",
            "425 0 5.029554 [',', ' infinity', ' *', ';', '.', ' (', ' imm', ' –'] in=  of\n",
            "426 0 5.4664617 [',', ' infinity', '.', '—', ' imm', ' *', ' –', ' length'] in=  absolute\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "plt.figure(figsize=[5,5])\n",
        "plt.plot(sorted(jnp.array(vmapped_results['s'])[::])[::-1],'-o')\n",
        "plt.grid()\n",
        "plt.xlabel(\"Sorted index\")\n",
        "plt.ylabel(\"Magnitude\")\n",
        "plt.grid()\n",
        "# plt.title(\"\")\n",
        "\n",
        "plt.title(f'Gemma 3 4B, Sorted Top Singular Values of Lanczos Jacobian\\n427 input tokens, normalized err = {np.std(recondiff):1.3f}')\n",
        "\n",
        "plt.savefig('gemma_3_4b_lanczos_427_sv.pdf')"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 498
        },
        "id": "EISh_BII9TqH",
        "outputId": "cfa7896e-f693-4f2f-ad94-563a5f547262"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 500x500 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "plt.plot(jnp.array(vmapped_results['s'])[::-1],'-o')\n",
        "# plt.plot(jnp.array(vmapped_results['s']).T[1,:],'-o')\n",
        "# plt.plot(jnp.array(vmapped_results['s']).T[2,:],'-o')\n",
        "plt.grid()\n",
        "plt.legend([\"SV 0\"])#, \"SV 1\"])\n",
        "plt.xlabel(\"Token index\")\n",
        "plt.ylabel(\"Singular value 0\")\n",
        "\n",
        "plt.title(f'Gemma 3 4B, Ordered Top Singular Values of Lanczos Jacobian\\n427 input tokens, normalized err = {np.std(recondiff):1.3f}')\n",
        "plt.grid()\n",
        "plt.savefig('gemma_3_4b_lanczos_427_sv_ordered.pdf')"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 482
        },
        "id": "ehxI58tE-aXD",
        "outputId": "712ec97e-9115-4a77-a501-4ae1bccaac9f"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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BQVYtmwZunfvjqCgILRv3x6LFi0S7JubN29i5MiRaNWqFYKCgtCtWzds2bJF42cDxde3devWoUuXLggICEB4eDg++ugj3Lt3D0Dx7/zmm28CAKZNm6bSFKyvp0+fYurUqWjdujUCAgLQsWNHfPPNN8jLy+PX4c7jc+fOoW/fvggODsYbb7yB3377TbCthIQEfPjhhwgLC0PLli0xZswY3L17F4D25tRz587x2zhx4gQGDBiAoKAghISEYNCgQThz5ozg91m9ejW6dOmCoKAgtGrVCuPGjUNSUlKpfofExER89NFHCA8PR0BAAN544w2sWrVKpUvCb7/9hp49eyIoKAivv/46li1bJmgKj4+Px4cffojmzZsjICBA4z5njGHZsmVo06YNAgMDERkZKagpVz7ejLmfjIaVI4WFhSwgIIB98803er3v+++/Z76+vmzNmjXs3r177NSpU6xDhw5s8ODB/DpTpkxhERERbPjw4Sw2NpZduXKFtWvXjvXo0YNFRkaya9eusatXr7J27dqxyMhI/n2RkZGsQ4cObPz48ezOnTvszJkzLCQkhPXr14998MEH7NatW+zff/9lISEhbOrUqfz7pk6dysLDw9np06fZo0eP2Pnz51nHjh3ZuHHj9Ppu169fZ127dmWjRo3Sue6PP/7IXn31VfbixQu2Z88eJpVKBa8vX76cSaVSlpeXxy/LyMhg8+fPZ2FhYSwuLk6vsg0dOpT16dNH6zp///03k0ql7N9//2WMFe+Hdu3asWHDhrGbN2+y58+fM8YYe/fdd1lERAQ7efIki4+PZytXrmRt27ZlUqmUJSUlMcYYO3fuHGvSpAmbNGkSu3PnDrt27RobMmQIa9WqFUtNTWWMMf57Dxo0iJ09e5Y9efKEMcbY3r17mVQqZXPnzmWJiYnswoULrFevXqxr164sPz+fMcbYv//+y6RSKZsxYwaLi4tjZ8+eZe+88w4LCQlhU6ZMEfWbLFy4UOV3v3PnDgsMDGQffvghi4mJYXFxcWzOnDlMKpWyv/76izHGWFJSEpNKpax9+/Zs3bp1LDExkR07doy1aNGCDR8+XOtn9uvXj/n6+rJZs2axq1evsqKiIrXrnT17lkmlUnbq1CnGWPHxEBwczCIjI9np06fZvXv32JdffsmkUin777//GGOMpaamsrCwMNavXz929epVdvXqVfbO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\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "\n",
        "# --- Example Usage ---\n",
        "# Assuming you have these variables from your previous run:\n",
        "vmapped_results = {'U': jnp.array(vmapped_results['U']), 's': jnp.array(vmapped_results['s']), 'V': jnp.array(vmapped_results['V'])}\n",
        "# embeddings = the original embeddings tensor\n",
        "\n",
        "reconstructed_output = reconstruct_output_from_svd(vmapped_results, embeddings)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "WkIexzTMNuoJ",
        "outputId": "db1a9669-dfcb-4e44-add8-df90f47f5fd3"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Reconstruction complete.\n",
            "  Shape of per-token contributions: (427, 2560)\n",
            "  Shape of final reconstructed output: (2560,)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "plt.scatter(reconstructed_output, output_from_embeddings[2],marker='x')\n",
        "plt.grid()\n",
        "\n",
        "plt.legend([\"SV 0\"])#, \"SV 1\"])\n",
        "plt.xlabel(\"Lanczos reconstruction\")\n",
        "plt.ylabel(\"Output embedding\")\n",
        "plt.grid()\n",
        "# plt.savefig('gemma_3_4b_lanczos_427_sv_ordered.pdf')\n",
        "\n",
        "plt.title(f'Gemma 3 4B, Reconstruction Error with top-1 Lanczos Jacobian\\n427 input tokens, normalized err = {np.std(recondiff):1.3f}')\n",
        "plt.savefig('gemma_3_4b_lanczos_427_recon_scatter.pdf')"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 482
        },
        "id": "douHl6VMAX9R",
        "outputId": "acf25b9b-6e56-433d-b0ef-e52d3cf41e0c"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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j4GGiIiIdB4DDREREek8BhoiIiLSeQw0REREpPMYaIiIiEjnMdAQERGRzmOgISIiIp3HQENEREQ6j4GGiIiIdB4DDREREek8BhoiIiLSeQw0REREpPMYaIiIiEjnMdAQERGRzmOgISIiIp3HQENEREQ6j4GGiIiIdB4DDREREek8BhoiIiLSeQw0REREpPMYaIiIiEjnMdAQERGRzmOgISIiIp3HQENEREQ6j4GGiIiIdB4DDREREek8BhoiIiLSeQw0REREpPMYaIiIiEjnMdAQERGRzmOgISIiIp3HQENEREQ6j4GGiIiIdB4DDREREek8A00fMGjQIMhkshzX6enpoUKFCmjVqhU6der03oUjIiIiyg+NW2iaNGmChw8fIiUlBfb29qhXrx7S0tLw5MkT1KtXDzKZDLNmzcLGjRsLorxEREREajRuoQkLC8PkyZPx6aefqiw/ePAgHj16hO+++w53797FmDFjMGLECK0VlIiIiCg3GrfQ/Prrr+jWrZva8m7duuHw4cMAgPr16yMqKuq9C0dERESUHxoHGmtra+zbtw9CCJXlhw4dgoFBZoPPnj17ULNmTa0UkIiIiOhtNL7kNGvWLIwZMwY//PADKlWqBAMDA4SGhuLNmzdYsGABUlNTsWLFCqxevbogyktERESkRuNA07p1a5w+fRp+fn54/fo1MjIyYGdnhyZNmqBq1aoAgL///humpqZaLywRERFRTjQONABQvnx5dOvWDSkpKSrLU1JSYGRkxDBDREREhUrjQHPx4kUsXLgQz549Q3p6utr6+/fva6VgRERERPn1Tn1o3NzcMGnSJLbEEBERUbGgcaB5/fo15syZI93RRERERFTUNL5t283NDQ8ePCiIshARERG9E42bWdq1a4fJkyejTZs2qFq1KvT0VDNR3759tVY4IiIiovzQONCsX78eQOaIwdnJZDIGGiIiIip0Ggeas2fPFkQ5chQSEoKFCxfi6tWr0NfXx0cffYRp06bBysoK9+/fx4IFC3D//n3Y2dmhX79+8PLyKrSyERERUfGRr0Dz/Plz1KhRAwAQGBiY57a1atV6/1L9vy+//BKOjo44e/Ys4uLi8NVXX2Hx4sWYOXMmRowYgc8++wy+vr4IDAyEl5cXqlativbt22vt+YmIiEg35CvQdO3aFbdv3wYAfPLJJ5DJZGpzOQGZl5y0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G/r3718EJS0dQmNUw8y+4U3RqIYt9g1vqhJqchunhoiIqCAVy0ATGhqKf/75B1u3boWTk5PKz5UrVwAAq1evRnh4OJo3b47BgwejR48eGDBgQBGXvOQyNzaAnYWRFGaUl5cq25hKocbOwgjmxjrRz5yIiEqYYvntU6lSJTx8+DDPbSpWrIhNmzYVUonIysQQO7zckJCchkrWqn1lKtuYYv+IpjA3NoCViWERlZCIiEqzYhloqHiyMjHMNbBkDzlERESFqVheciIiIiLSBAMNERER6TwGGiIiItJ5DDRERESk8xhoiIiISOcx0BAREZHOY6AhIiIincdAQ0RERDqPgYaIiIh0HgMNERER6TwGGiIiItJ5DDRERESk8xhoiIiISOcx0BAREZHOY6AhIiIincdAQ0RERDqPgYaIiIh0HgMNERER6TwGGiIiItJ5DDRERESk8xhoiIiISOcx0BAREZHOY6AhIiIincdAQ0RERDqPgeYdxCpSERqTlOO60JgkxCpSC7lEREREpRsDjYZiFan4Yqs/+m70w8to1VDzMjoJfTf64Yut/gw1REREhYiBRkMJyWmIjE9BUFQi+vn+F2peRiehn68fgqISERmfgoTktCIuKRERUenBQKOhStam2De8Karbmkmh5trzKCnMVLc1w77hTVHJ2rSoi0pERFRqMNC8g8o2qqHm0/WXVMJMZRuGGSIiosLEQPOOKtuYYkVfZ5VlK/o6M8wQEREVAQaad/QyOgnj999SWTZ+/y21jsJERERU8Bho3kHWDsDVbc1wcGQzlT41+Q01vP2biIhIOxhoNBQak6TWAbhRDVu1jsK5BRUl3v5NRESkPQw0GjI3NoCdhZFaB+CsHYXtLIxgbmyQ5354+zcREZH25P2tS2qsTAyxw8sNCclpardmV7Yxxf4RTWFubAArE8M896O8/VsZXvr5+mFFX2eM33+Lt38TERFpiC0078DKxDDXoFHJ2vStYUaJt38TERFpBwNNEePt30RERO+PgaaI8fZvIiKi98dAU4S0dfs3ERFRacdAU0S0dfs3ERERMdAUGW3d/k1ERES8bbvIaOv2byIiImKgKVJWJoa5BhaOP0NERJR/vOREREREOo+BhoiIiHQeAw0RERHpPAYaIiIi0nkMNERERKTzGGiIiIhI5zHQEBERkc5joCEiIiKdx0BDREREOq9U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\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "plt.hist(recondiff[0],bins=80)\n",
        "\n",
        "# plt.grid()\n",
        "plt.xlim([-1,1])\n",
        "plt.xlabel('Output embedding vector')\n",
        "plt.ylabel('Lancozs matrix-free reconstruction')\n",
        "plt.title(f'Gemma 3 4B, Lanczos Matrix-free Reconstruction of the output embedding\\n427 input tokens, normalized err = {np.std(recondiff):1.3f}')\n",
        "\n",
        "plt.savefig('gemma_3_4b_lanczos_427_hist2.pdf')"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 482
        },
        "id": "EJWHZpJtBQgy",
        "outputId": "dcd2dd4b-55bb-4572-9ed7-7ce9a473ab35"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 640x480 with 1 Axes>"
            ],
            "image/png": "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v6GhIUJCQjJdR1bjv1dZJZffK5a7YGG5CxaWmzSBRiRRN27cwMCBAzFy5Eg4OjoCAHbs2CGNt7S0xIABA7Bq1SoMHz4cACCEUHk9NWvWhIGBQc4EnQ8kJycjKCgIFhYWBe41ASw3y/29Y7lZ7oIgNjZWoytA8jyJOnXqFEaPHo2JEyeiffv2GU5XqVIlvH37FkIIlCxZElFRUUrjo6KiYGxsnOm6tLW1C9TOp6Cjo8NyFyAsd8HCchcsBa3cmvrOPIU8je7mzZsYO3YsFi9erJRAXbp0CStWrFCa9vHjx6hUqRK0tLRgbm6Ou3fvKo0PCgpCvXr1vkXYRERERHmXRCUlJWHChAkYNWoUnJyclMYVL14cy5Ytw/79+5GYmIigoCCsXbsW3bt3BwB06dIFFy9exOnTp5GQkIDdu3cjPDwcHh4eeVEUIiIiKoDy7HberVu3EBoaiunTp2P69OlK444dO4aFCxdi6dKlmDRpEooXL44ePXqgV69eAACZTAZ/f3/MmjULz58/R82aNbFq1SqUKVMmL4pCREREBVCeJVE2NjZ4+PBhhuMrVaqE5s2bZzi+RYsWaNGiRW6ERkQFSDW/wyrPEz7bPRciIaL8RrNbbBERERFpKCZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGpgEkVERESkBiZRRERERGoolNcBEBGlp5rfYdVm2HUM4bPdcycYIqJ0sCaKiIiISA2siSKiXKdyrRIRUT7AmigiIiIiNTCJIiIiIlIDkygiIiIiNTCJIiIiIlIDkygiIiIiNTCJIiIiIlIDkygiIiIiNTCJIiIiIlIDkygiIiIiNbDHciL6brBndCL6llgTRURERKQGJlFEREREamASRURERKQGJlFEREREamASRURERKQGJlFEREREamASRURERKQGJlFEREREamASRURERKQGJlFEREREamASRURERKQGJlFEREREamASRURERKQGJlFEREREamASRURERKQGJlFEREREamASRURERKQGJlFE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mkTy3bt36Nu3L6ytreHu7o47d+5I4768ldi0aVPs2rUL/fv3h7W1NZo1a4bz589L058+fRpNmjSBtbU1xo0bh8WLF6NHjx5pvpszZ87AysoKnz59koa9f/8edevWxa1btwAAW7ZsQevWrVGvXj24u7sjMDBQmjY+Ph4TJ06Evb09GjZsiIkTJ0Iul2PatGnYtm0b1q1bh+bNmwMAPn78iDFjxsDJyQnW1tbo378/nj17plS+bdu2wc7ODocOHUp3WxJR9jGJItIQ9+7dQ0REBHr27JlmnK6uLrp164bjx49nuZzWrVtj4MCBsLS0RFBQEExMTAAAO3bsQPv27XH16lX069cPvr6+2bpFt3//funfIUOGpBkfFBSE8ePHY/To0bhx4wbmzJmD2bNn4+bNm9I0R44cgaurKy5fvozq1avj119/RWJiIs6cOYMePXpg5syZSElJkabfuXMnVq9ejdOnTyM5ORkTJ04EALx69QqDBg1C9+7dcf36daxZswY7duzAwYMHAXyumTl27Bi2bduG48eP4+7du/j48aO03JkzZyIhIQGnT5/GunXrMmwrpbB27VoMGTIEV65cgZ2dHWbOnAkAePPmDYYOHQpvb29cuXIFDRo0wNatW9NdhqOjI3R1dZUSsFOnTqF8+fKwsrLCX3/9haVLl2LevHm4ceMGhg8fjhEjRuDFixcAgAULFiAkJARHjx7FkSNHcO/ePSxbtgwTJ06Era0tfHx8pIR4woQJiIyMxIEDB3Du3Dno6+tjxIgRSvFcvXoVp06dgru7e6ZlJ6KsMYki0hAREREoUqQIypUrl+746tWr49mzZxBCqLV8KysrNGnSBLq6uvjpp59QrFgxpQu7uvbu3YsmTZrAyckJOjo6sLGxQevWraXkCwCqVasGV1dX6OnpoVGjRnj//j369esHXV1duLq6IiYmBu/evZOm//nnn1GxYkUYGhrC29sbFy9eRFJSEg4dOoRatWqhffv20NHRgampKbp16yat68SJE2jXrh1q1KiBokWLYvjw4UhKSpKWGxgYiN69e8PQ0BDlypWDl5dXpmVzdXWFpaUldHV10bJlS4SHhyMlJQWXL19G0aJF0aNHD+jq6qJTp06oXr16ussoXLgw3NzccPLkSaU4WrduDeBze7NOnTrB3NwchQoVQosWLdCgQQMcOnQIQgjs27cPPj4+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\n"
          },
          "metadata": {}
        }
      ]
    }
  ]
}