{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "d9ffea84",
   "metadata": {},
   "source": [
    "### Embedding Geometry Sandbox\n",
    "\n",
    "Notebook to explore embedding geometry: load or compute embeddings, estimate metrics similar to `embedding_geometry.py`, and approximate volume/capacity using the formulas from `hidden_capacity/plots_aaai.ipynb`."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ccc4abd7",
   "metadata": {},
   "source": [
    "### Load embeddings\n",
    "- `load_embeddings_from_disk`: pick a saved `.npy` embedding matrix.\n",
    "- `embed_dataset`: compute embeddings for a dataset/model using the same helper used in `embedding_geometry.py` (only call when models and GPU/CPU resources are available)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "cf50020e",
   "metadata": {},
   "outputs": [],
   "source": [
    "from pathlib import Path\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "\n",
    "SAMPLED_EMB_DIR = Path('data/embeddings_samples')\n",
    "SAMPLED_LENGTHS = [4, 8,16,32,64,128,256,512,1024, 2048, 4096]\n",
    "eps = 0.125\n",
    "DEFAULT_P = 2\n",
    "\n",
    "MODELS = [\n",
    "    'pythia-160m',\n",
    "    'pythia-410m',\n",
    "    'pythia-1b',\n",
    "    # 'pythia-1.4b',\n",
    "    'Qwen2.5-0.5B',\n",
    "    'Qwen2.5-1.5B',\n",
    "    'Llama-3.2-1B',\n",
    "    'gemma-3-270m',\n",
    "]\n",
    "def load_embeddings_from_disk(model, lengths=SAMPLED_LENGTHS) -> np.ndarray:\n",
    "    # emb = np.load(path)\n",
    "    # print(f\"Loaded embeddings {emb.shape} from {path}\")\n",
    "    # return emb\n",
    "\n",
    "    sample_emb = []\n",
    "    for length in lengths:\n",
    "        sample_path = SAMPLED_EMB_DIR / model / f'embedding_{length}.npy'\n",
    "        if not sample_path.exists():\n",
    "            continue\n",
    "            # raise FileNotFoundError(f'Expected sample file at {sample_path}')\n",
    "        sample_emb_length = np.load(sample_path)\n",
    "        sample_emb.append(sample_emb_length)\n",
    "    \n",
    "    return np.concatenate(sample_emb, axis=0)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ce4c7f33",
   "metadata": {},
   "source": [
    "### Estimate upper bounds and geometry"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "301fda80",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Cumulative rectangle + cone log-volumes for all models\n",
    "from pathlib import Path\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from numpy import arccos, log, exp\n",
    "from math import sin, cos\n",
    "from scipy.special import betainc, beta, gammaln\n",
    "\n",
    "CSV_PATH = Path('models_list.csv')\n",
    "model_table = pd.read_csv(CSV_PATH).replace('--', pd.NA)\n",
    "model_table['name'] = model_table['Link to HF'].map(lambda x: x.split(\"/\")[-1])\n",
    "\n",
    "def l1_l2_linf_radii(emb: np.ndarray):\n",
    "    r1 = float(np.max(np.linalg.norm(emb, ord=1, axis=1)))\n",
    "    r2 = float(np.max(np.linalg.norm(emb, ord=2, axis=1)))\n",
    "    rinf = float(np.max(np.linalg.norm(emb, ord=np.inf, axis=1)))\n",
    "    return r1, r2, rinf\n",
    "\n",
    "def average_pairwise_cosine(emb: np.ndarray):\n",
    "    norms = np.linalg.norm(emb, axis=1, keepdims=True) + 1e-12\n",
    "    X = emb / norms\n",
    "    sim = X @ X.T\n",
    "    triu = sim[np.triu_indices(sim.shape[0], k=1)]\n",
    "    return float(triu.mean()), float(triu.min())\n",
    "\n",
    "def th_slope(r, d, v, eps):\n",
    "    \"\"\"Theoretical slope using L_inf radius.\"\"\"\n",
    "    return float((d * log(1 + 2 * r / eps)) / log(v))\n",
    "\n",
    "def log_frac_cone(min_cos, d, p=2):\n",
    "    cos_sim = float(np.clip(min_cos, -1 + eps, 1 - eps))\n",
    "    theta = arccos(cos_sim)\n",
    "    a = gammaln((p + d) / p) - gammaln((p + 1) / p) - gammaln((p + d - 1) / p)  # no r term\n",
    "    b1 = log(2) - log(d) + (d - 1) * log(sin(theta / 2)) + log(cos(theta / 2))\n",
    "    b2 = log(beta(1 / 2, (d + 1) / 2)) + log(1 - betainc(1 / 2, (d + 1) / 2, cos(theta / 2) ** 2))\n",
    "    m = max(b1, b2)\n",
    "    return float(a + m + log(exp(b1 - m) + exp(b2 - m)))\n",
    "\n",
    "def th_slope_with_cone(r, min_cos, d, v, eps):\n",
    "    \"\"\"Theoretical slope including cone fraction. \"\"\"\n",
    "    return float((d * log(1 + 2 * r / eps) + log_frac_cone(min_cos, d)) / log(v))\n",
    "\n",
    "def th_slope_rect(rect, d, v, eps):\n",
    "    \"\"\"Theoretical slope using the L_inf rectangle.\"\"\"\n",
    "    return float(sum([log(1 + ri / eps) for ri in rect]) / log(v))\n",
    "\n",
    "def stats_from_models(model, d, v, slope_exp, lengths=SAMPLED_LENGTHS):\n",
    "    embs = load_embeddings_from_disk(model, lengths=lengths)\n",
    "    print(f\"  - lengths: {lengths}, embeddings shape: {embs.shape}\")\n",
    "    r1, r2, rinf = l1_l2_linf_radii(embs)\n",
    "    # rect = np.max(np.abs(embs), axis=0) #- np.min(embs, axis=0)\n",
    "    rect = np.max(embs, axis=0) - np.min(embs, axis=0)\n",
    "    \n",
    "    avg_cos, min_cos = average_pairwise_cosine(embs)\n",
    "    epsilon = 2 ** ( log(rinf)/log(2) - 11)\n",
    "    # print(d, rect.shape)\n",
    "    slope =  th_slope(rinf, d, v, epsilon)\n",
    "    slope_cone =  th_slope_with_cone(rinf, min_cos, d, v, epsilon)\n",
    "    slope_rect =  th_slope_rect(rect, d, v, epsilon)\n",
    "    # print(rinf, np.max(rect))\n",
    "    print(f\"    - Theoretical slopes: slope={slope}, slope_cone={slope_cone}, slope_rect={slope_rect}\")\n",
    "    return {\n",
    "        'model': model,\n",
    "        'lengths_max': lengths[-1],\n",
    "        'd': d,\n",
    "        'v': v,\n",
    "        'r1': r1,\n",
    "        'r2': r2,\n",
    "        'rinf': rinf,\n",
    "        'min_cos': min_cos,\n",
    "        'avg_cos': avg_cos,\n",
    "        'theta': float(arccos(min_cos)),\n",
    "        'epsilon': epsilon,\n",
    "        'rect_mean': float(np.mean(rect)),\n",
    "        'th_slope':slope,     \n",
    "        'th_slope_cone':slope_cone,\n",
    "        'th_slope_rect':slope_rect,\n",
    "        'slope_exp':slope_exp,\n",
    "        'ratio_th': slope / slope_exp,\n",
    "        'ratio_th_cone': slope_cone / slope_exp,\n",
    "        'ratio_th_rect': slope_rect / slope_exp,\n",
    "    }\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e3ebea7e",
   "metadata": {},
   "source": [
    "### Plot influence of \\ell on upper bounds "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "cf1208ce",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2197142/193572784.py:5: FutureWarning: Calling int on a single element Series is deprecated and will raise a TypeError in the future. Use int(ser.iloc[0]) instead\n",
      "  d = int(model_table.loc[model_table['name'] == model, 'Input hidden size'])\n",
      "/tmp/ipykernel_2197142/193572784.py:6: FutureWarning: Calling int on a single element Series is deprecated and will raise a TypeError in the future. Use int(ser.iloc[0]) instead\n",
      "  v = int(model_table.loc[model_table['name'] == model, 'Vocabulary size'])\n",
      "/tmp/ipykernel_2197142/193572784.py:7: FutureWarning: Calling float on a single element Series is deprecated and will raise a TypeError in the future. Use float(ser.iloc[0]) instead\n",
      "  slope_exp = float(model_table.loc[model_table['name'] == model, 'Slope exp'])\n",
      "/tmp/ipykernel_2197142/526360734.py:51: RuntimeWarning: divide by zero encountered in log\n",
      "  b2 = log(beta(1 / 2, (d + 1) / 2)) + log(1 - betainc(1 / 2, (d + 1) / 2, cos(theta / 2) ** 2))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Summarizing geometry for model pythia-160m\n",
      "  - lengths: [4], embeddings shape: (1000, 768)\n",
      "768 (768,)\n",
      "52.5 66.1875\n",
      "    - Theoretical slopes: slope=477.76121337489565, slope_cone=463.14507281727793, slope_rect=361.74305256809265\n",
      "  - lengths: [4, 8], embeddings shape: (2000, 768)\n",
      "768 (768,)\n",
      "52.5 66.1875\n",
      "    - Theoretical slopes: slope=477.76121337489565, slope_cone=463.18531453776126, slope_rect=365.57922469263406\n",
      "  - lengths: [4, 8, 16], embeddings shape: (3000, 768)\n",
      "768 (768,)\n",
      "52.5 66.1875\n",
      "    - Theoretical slopes: slope=477.76121337489565, slope_cone=464.49786289339784, slope_rect=367.84789881087056\n",
      "  - lengths: [4, 8, 16, 32], embeddings shape: (4000, 768)\n",
      "768 (768,)\n",
      "63.625 66.1875\n",
      "    - Theoretical slopes: slope=491.38092033043426, slope_cone=478.11756984893645, slope_rect=369.38429808250567\n",
      "  - lengths: [4, 8, 16, 32, 64], embeddings shape: (5000, 768)\n",
      "768 (768,)\n",
      "63.625 67.0\n",
      "    - Theoretical slopes: slope=491.38092033043426, slope_cone=478.11756984893645, slope_rect=369.787386395405\n",
      "  - lengths: [4, 8, 16, 32, 64, 128], embeddings shape: (6000, 768)\n",
      "768 (768,)\n",
      "63.625 67.0\n",
      "    - Theoretical slopes: slope=491.38092033043426, slope_cone=478.11756984893645, slope_rect=370.08585757260215\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256], embeddings shape: (7000, 768)\n",
      "768 (768,)\n",
      "63.625 67.0\n",
      "    - Theoretical slopes: slope=491.38092033043426, slope_cone=478.11756984893645, slope_rect=371.41162437011394\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512], embeddings shape: (8000, 768)\n",
      "768 (768,)\n",
      "63.625 67.0\n",
      "    - Theoretical slopes: slope=491.38092033043426, slope_cone=478.11756984893645, slope_rect=371.56677976861164\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024], embeddings shape: (9000, 768)\n",
      "768 (768,)\n",
      "63.625 67.0\n",
      "    - Theoretical slopes: slope=491.38092033043426, slope_cone=478.11756984893645, slope_rect=371.8067519481776\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048], embeddings shape: (10000, 768)\n",
      "768 (768,)\n",
      "63.625 67.0\n",
      "    - Theoretical slopes: slope=491.38092033043426, slope_cone=478.11756984893645, slope_rect=371.87668463578535\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 768)\n",
      "768 (768,)\n",
      "63.625 67.0\n",
      "    - Theoretical slopes: slope=491.38092033043426, slope_cone=478.11756984893645, slope_rect=372.370836538675\n",
      "Summarizing geometry for model pythia-410m\n",
      "  - lengths: [4], embeddings shape: (1000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646812648669, slope_rect=449.4205112961697\n",
      "  - lengths: [4, 8], embeddings shape: (2000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=454.32454947153406\n",
      "  - lengths: [4, 8, 16], embeddings shape: (3000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=456.0152020753965\n",
      "  - lengths: [4, 8, 16, 32], embeddings shape: (4000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=456.48278812662477\n",
      "  - lengths: [4, 8, 16, 32, 64], embeddings shape: (5000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=456.66626840601094\n",
      "  - lengths: [4, 8, 16, 32, 64, 128], embeddings shape: (6000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=456.7420419727191\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256], embeddings shape: (7000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=457.08243667983015\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512], embeddings shape: (8000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=457.15234681578113\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024], embeddings shape: (9000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=457.2155816613316\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048], embeddings shape: (10000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=457.3032173994947\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 1024)\n",
      "1024 (1024,)\n",
      "54.90625 97.5625\n",
      "    - Theoretical slopes: slope=641.2489127280767, slope_cone=620.6646801858616, slope_rect=457.32549843637196\n",
      "Summarizing geometry for model pythia-1b\n",
      "  - lengths: [4], embeddings shape: (1000, 2048)\n",
      "2048 (2048,)\n",
      "54.09375 93.625\n",
      "    - Theoretical slopes: slope=1279.6807045459168, slope_cone=1254.9330257031606, slope_rect=833.2280597333502\n",
      "  - lengths: [4, 8], embeddings shape: (2000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1293.2330969137993, slope_rect=840.1172300465279\n",
      "  - lengths: [4, 8, 16], embeddings shape: (3000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1293.2330969137993, slope_rect=842.1820597393254\n",
      "  - lengths: [4, 8, 16, 32], embeddings shape: (4000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1293.2330969137993, slope_rect=843.0774507192602\n",
      "  - lengths: [4, 8, 16, 32, 64], embeddings shape: (5000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1293.2330969137993, slope_rect=843.3423875828577\n",
      "  - lengths: [4, 8, 16, 32, 64, 128], embeddings shape: (6000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1294.9455176760887, slope_rect=843.5439091876506\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256], embeddings shape: (7000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1295.1825001944355, slope_rect=843.6978016933698\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512], embeddings shape: (8000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1295.1825001944355, slope_rect=843.814168242243\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024], embeddings shape: (9000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1295.608213921101, slope_rect=843.9202512356345\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048], embeddings shape: (10000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1295.608213921101, slope_rect=844.0249811296204\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 2048)\n",
      "2048 (2048,)\n",
      "65.6875 105.21875\n",
      "    - Theoretical slopes: slope=1316.3784494304778, slope_cone=1295.608213921101, slope_rect=844.0362569579996\n",
      "Summarizing geometry for model Qwen2.5-0.5B\n",
      "  - lengths: [4], embeddings shape: (1000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.0361538345745, slope_rect=407.6563116279601\n",
      "  - lengths: [4, 8], embeddings shape: (2000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.0361538345745, slope_rect=411.832635221772\n",
      "  - lengths: [4, 8, 16], embeddings shape: (3000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.2421363470928, slope_rect=412.8891060698377\n",
      "  - lengths: [4, 8, 16, 32], embeddings shape: (4000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.5743454000431, slope_rect=413.29671597286824\n",
      "  - lengths: [4, 8, 16, 32, 64], embeddings shape: (5000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.5743454000431, slope_rect=413.4528027369404\n",
      "  - lengths: [4, 8, 16, 32, 64, 128], embeddings shape: (6000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.5743454000431, slope_rect=413.5335471484544\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256], embeddings shape: (7000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.5743454000431, slope_rect=413.6014412076037\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512], embeddings shape: (8000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.5743454000431, slope_rect=413.6398694906304\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024], embeddings shape: (9000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.5743454000431, slope_rect=413.70223871675677\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048], embeddings shape: (10000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.5743454000431, slope_rect=413.7173808517087\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 896)\n",
      "896 (896,)\n",
      "309.0 408.0625\n",
      "    - Theoretical slopes: slope=638.786134421487, slope_cone=630.5743454000431, slope_rect=413.7878940909579\n",
      "Summarizing geometry for model Qwen2.5-1.5B\n",
      "  - lengths: [4], embeddings shape: (1000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=586.2999867404976\n",
      "  - lengths: [4, 8], embeddings shape: (2000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=591.5074897245364\n",
      "  - lengths: [4, 8, 16], embeddings shape: (3000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=593.4204733899242\n",
      "  - lengths: [4, 8, 16, 32], embeddings shape: (4000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=594.0925386336559\n",
      "  - lengths: [4, 8, 16, 32, 64], embeddings shape: (5000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=594.2743670271462\n",
      "  - lengths: [4, 8, 16, 32, 64, 128], embeddings shape: (6000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=594.4468645915312\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256], embeddings shape: (7000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=594.5204471824597\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512], embeddings shape: (8000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=594.5528595361948\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024], embeddings shape: (9000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=594.8426881841127\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048], embeddings shape: (10000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=594.8583214153333\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 1536)\n",
      "1536 (1536,)\n",
      "153.625 281.1875\n",
      "    - Theoretical slopes: slope=1005.1227614341087, slope_cone=994.2202125409264, slope_rect=594.9579730331927\n",
      "Summarizing geometry for model Llama-3.2-1B\n",
      "  - lengths: [4], embeddings shape: (1000, 2048)\n",
      "2048 (2048,)\n",
      "40.9375 58.96875\n",
      "    - Theoretical slopes: slope=1129.391409414858, slope_cone=1081.6617400363807, slope_rect=776.2385001714356\n",
      "  - lengths: [4, 8], embeddings shape: (2000, 2048)\n",
      "2048 (2048,)\n",
      "40.9375 58.96875\n",
      "    - Theoretical slopes: slope=1129.391409414858, slope_cone=1081.6617389192156, slope_rect=786.3071926270675\n",
      "  - lengths: [4, 8, 16], embeddings shape: (3000, 2048)\n",
      "2048 (2048,)\n",
      "41.0625 59.09375\n",
      "    - Theoretical slopes: slope=1129.9214653857578, slope_cone=1082.1917948901155, slope_rect=788.906231779512\n",
      "  - lengths: [4, 8, 16, 32], embeddings shape: (4000, 2048)\n",
      "2048 (2048,)\n",
      "41.0625 59.09375\n",
      "    - Theoretical slopes: slope=1129.9214653857578, slope_cone=1082.1917948901155, slope_rect=790.0428080488093\n",
      "  - lengths: [4, 8, 16, 32, 64], embeddings shape: (5000, 2048)\n",
      "2048 (2048,)\n",
      "41.0625 59.09375\n",
      "    - Theoretical slopes: slope=1129.9214653857578, slope_cone=1082.1917960072806, slope_rect=790.710024565726\n",
      "  - lengths: [4, 8, 16, 32, 64, 128], embeddings shape: (6000, 2048)\n",
      "2048 (2048,)\n",
      "41.0625 59.09375\n",
      "    - Theoretical slopes: slope=1129.9214653857578, slope_cone=1082.1917948901155, slope_rect=791.0995939904863\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256], embeddings shape: (7000, 2048)\n",
      "2048 (2048,)\n",
      "41.0625 59.09375\n",
      "    - Theoretical slopes: slope=1129.9214653857578, slope_cone=1082.1917948901155, slope_rect=791.4606846614756\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512], embeddings shape: (8000, 2048)\n",
      "2048 (2048,)\n",
      "41.0625 59.09375\n",
      "    - Theoretical slopes: slope=1129.9214653857578, slope_cone=1082.1917948901155, slope_rect=791.615366638954\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024], embeddings shape: (9000, 2048)\n",
      "2048 (2048,)\n",
      "41.0625 59.09375\n",
      "    - Theoretical slopes: slope=1129.9214653857578, slope_cone=1082.1917948901155, slope_rect=791.717491616033\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048], embeddings shape: (10000, 2048)\n",
      "2048 (2048,)\n",
      "41.0625 59.09375\n",
      "    - Theoretical slopes: slope=1129.9214653857578, slope_cone=1082.1917948901155, slope_rect=791.7685541045727\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 2048)\n",
      "2048 (2048,)\n",
      "41.0625 59.09375\n",
      "    - Theoretical slopes: slope=1129.9214653857578, slope_cone=1082.1917948901155, slope_rect=791.8803934087397\n",
      "Summarizing geometry for model gemma-3-270m\n",
      "  - lengths: [4], embeddings shape: (1000, 640)\n",
      "640 (640,)\n",
      "9.8125 17.296875\n",
      "    - Theoretical slopes: slope=259.6899821574081, slope_cone=247.02608126364927, slope_rect=238.085206668414\n",
      "  - lengths: [4, 8], embeddings shape: (2000, 640)\n",
      "640 (640,)\n",
      "10.1640625 17.296875\n",
      "    - Theoretical slopes: slope=261.48442246916414, slope_cone=249.22026150506636, slope_rect=240.93470939796495\n",
      "  - lengths: [4, 8, 16], embeddings shape: (3000, 640)\n",
      "640 (640,)\n",
      "10.1640625 17.296875\n",
      "    - Theoretical slopes: slope=261.48442246916414, slope_cone=249.22026150506636, slope_rect=242.61092660977906\n",
      "  - lengths: [4, 8, 16, 32], embeddings shape: (4000, 640)\n",
      "640 (640,)\n",
      "10.1640625 17.429688\n",
      "    - Theoretical slopes: slope=261.48442246916414, slope_cone=249.22026150506636, slope_rect=243.8106677982961\n",
      "  - lengths: [4, 8, 16, 32, 64], embeddings shape: (5000, 640)\n",
      "640 (640,)\n",
      "10.1640625 17.78125\n",
      "    - Theoretical slopes: slope=261.48442246916414, slope_cone=249.22026150506636, slope_rect=244.59498451147036\n",
      "  - lengths: [4, 8, 16, 32, 64, 128], embeddings shape: (6000, 640)\n",
      "640 (640,)\n",
      "10.7421875 18.460938\n",
      "    - Theoretical slopes: slope=264.3052601297954, slope_cone=252.0410991656976, slope_rect=245.29480744770018\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256], embeddings shape: (7000, 640)\n",
      "640 (640,)\n",
      "10.7421875 18.460938\n",
      "    - Theoretical slopes: slope=264.3052601297954, slope_cone=252.0410991656976, slope_rect=245.85677520593708\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512], embeddings shape: (8000, 640)\n",
      "640 (640,)\n",
      "10.7421875 18.859375\n",
      "    - Theoretical slopes: slope=264.3052601297954, slope_cone=252.0410991656976, slope_rect=246.38925428379991\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024], embeddings shape: (9000, 640)\n",
      "640 (640,)\n",
      "10.7421875 18.859375\n",
      "    - Theoretical slopes: slope=264.3052601297954, slope_cone=252.0410991656976, slope_rect=246.7436859145999\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048], embeddings shape: (9000, 640)\n",
      "640 (640,)\n",
      "10.7421875 18.859375\n",
      "    - Theoretical slopes: slope=264.3052601297954, slope_cone=252.0410991656976, slope_rect=246.7436859145999\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (9000, 640)\n",
      "640 (640,)\n",
      "10.7421875 18.859375\n",
      "    - Theoretical slopes: slope=264.3052601297954, slope_cone=252.0410991656976, slope_rect=246.7436859145999\n"
     ]
    }
   ],
   "source": [
    "embedding_stats = {}\n",
    "for model in MODELS:\n",
    "    embedding_stats[model] = []\n",
    "\n",
    "    d = int(model_table.loc[model_table['name'] == model, 'Input hidden size'])\n",
    "    v = int(model_table.loc[model_table['name'] == model, 'Vocabulary size'])\n",
    "    slope_exp = float(model_table.loc[model_table['name'] == model, 'Slope exp'])\n",
    "\n",
    "    print(f\"Summarizing geometry for model {model}\")\n",
    "    for k in range(1, len(SAMPLED_LENGTHS)+1):\n",
    "        lengths = SAMPLED_LENGTHS[:k]\n",
    "\n",
    "        stats = stats_from_models(model, d, v, slope_exp, lengths=lengths)\n",
    "        embedding_stats[model].append(stats)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "54283651",
   "metadata": {},
   "outputs": [
    {
     "ename": "TypeError",
     "evalue": "list indices must be integers or slices, not str",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mTypeError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[5], line 6\u001b[0m\n\u001b[1;32m      4\u001b[0m fig, axes \u001b[38;5;241m=\u001b[39m plt\u001b[38;5;241m.\u001b[39msubplots(nrows, ncols, figsize\u001b[38;5;241m=\u001b[39m(\u001b[38;5;241m4\u001b[39m \u001b[38;5;241m*\u001b[39m ncols, \u001b[38;5;241m3\u001b[39m \u001b[38;5;241m*\u001b[39m nrows), squeeze\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m, sharex\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m)\n\u001b[1;32m      5\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m ax, model \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mzip\u001b[39m(axes\u001b[38;5;241m.\u001b[39mravel(), models):\n\u001b[0;32m----> 6\u001b[0m     ax\u001b[38;5;241m.\u001b[39mplot([x[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mlengths_max\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;28;01mfor\u001b[39;00m x \u001b[38;5;129;01min\u001b[39;00m \u001b[43membedding_stats\u001b[49m\u001b[43m[\u001b[49m\u001b[43mmodel\u001b[49m\u001b[43m]\u001b[49m][:\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m2\u001b[39m], [x[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mth_slope\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;28;01mfor\u001b[39;00m x \u001b[38;5;129;01min\u001b[39;00m embedding_stats[model]][:\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m2\u001b[39m]    , linestyle\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m-\u001b[39m\u001b[38;5;124m'\u001b[39m, marker\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mo\u001b[39m\u001b[38;5;124m'\u001b[39m, markersize\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m4\u001b[39m, label\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mBall\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m      7\u001b[0m     ax\u001b[38;5;241m.\u001b[39mplot([x[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mlengths_max\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;28;01mfor\u001b[39;00m x \u001b[38;5;129;01min\u001b[39;00m embedding_stats[model]][:\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m2\u001b[39m], [x[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mth_slope_rect\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;28;01mfor\u001b[39;00m x \u001b[38;5;129;01min\u001b[39;00m embedding_stats[model]][:\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m2\u001b[39m], linestyle\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m:\u001b[39m\u001b[38;5;124m'\u001b[39m, marker\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mo\u001b[39m\u001b[38;5;124m'\u001b[39m, markersize\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m4\u001b[39m, label\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mEllipsoid\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m      8\u001b[0m     ax\u001b[38;5;241m.\u001b[39mplot([x[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mlengths_max\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;28;01mfor\u001b[39;00m x \u001b[38;5;129;01min\u001b[39;00m embedding_stats[model]][:\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m2\u001b[39m], [x[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mth_slope_cone\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;28;01mfor\u001b[39;00m x \u001b[38;5;129;01min\u001b[39;00m embedding_stats[model]][:\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m2\u001b[39m], linestyle\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m--\u001b[39m\u001b[38;5;124m'\u001b[39m, marker\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mo\u001b[39m\u001b[38;5;124m'\u001b[39m, markersize\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m4\u001b[39m, label\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mCone\u001b[39m\u001b[38;5;124m'\u001b[39m)\n",
      "\u001b[0;31mTypeError\u001b[0m: list indices must be integers or slices, not str"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA+AAAALmCAYAAADCCf7UAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8ekN5oAAAACXBIWXMAAA9hAAAPYQGoP6dpAABasElEQVR4nO39f3CV5Z0//r+SYE50aiIuS/ixsax2rW1VsCBptIxjJ9vM6NDlj06z2gGW8cfaUkfJ7FYQJLW2hLXqsFNjGamunXnXQuuon05hYm22TMeaHaZAZuwKOhYtbKeJsF0SGmsiyf39I1+TppwAJ5A7J8fHY+b8kbvXdc71MumTeeacnFOUJEkSAAAAwLgqnugDAAAAwAeBAg4AAAApUMABAAAgBQo4AAAApEABBwAAgBQo4AAAAJACBRwAAABSoIADAABAChRwAAAASIECDgAAACnIuYD/4he/iMWLF8esWbOiqKgonn/++VPu2blzZ3zyk5+MTCYTH/nIR+Kpp54aw1EB8pt8BMhOPgIMyrmA9/T0xNy5c6O5ufm01r/55ptx4403xvXXXx/t7e1x9913x6233hovvPBCzocFyGfyESA7+QgwqChJkmTMm4uK4rnnnoslS5aMuuaee+6J7du3x69//euha//4j/8YR48ejZaWlrE+NEBek48A2clH4INsyng/QFtbW9TW1o64VldXF3ffffeoe3p7e6O3t3fo64GBgfjDH/4Qf/VXfxVFRUXjdVSgQCVJEseOHYtZs2ZFcXH+vPWFfAQmmnwEGN14ZOS4F/COjo6orKwcca2ysjK6u7vjT3/6U5x77rkn7Glqaor7779/vI8GfMAcOnQo/uZv/maijzFEPgL5Qj4CjO5sZuS4F/CxWLNmTTQ0NAx93dXVFRdddFEcOnQoysvLJ/BkwGTU3d0dVVVVcf7550/0Uc6YfATOJvkIMLrxyMhxL+AzZsyIzs7OEdc6OzujvLw8628vIyIymUxkMpkTrpeXlwtQYMzy7SWI8hHIF/IRYHRnMyPH/Y99ampqorW1dcS1F198MWpqasb7oQHymnwEyE4+AoUq5wL+xz/+Mdrb26O9vT0iBj8mor29PQ4ePBgRgy//WbZs2dD6O+64Iw4cOBBf/epXY//+/fHYY4/FD3/4w1i1atXZmQAgT8hHgOzkI8CgnAv4r371q7jqqqviqquuioiIhoaGuOqqq2L9+vUREfH73/9+KEwjIv72b/82tm/fHi+++GLMnTs3Hn744fjud78bdXV1Z2kEgPwgHwGyk48Ag87oc8DT0t3dHRUVFdHV1eVveICcFXKGFPJswPgr5Awp5NmAdIxHjuTPBz4CAABAAVPAAQAAIAUKOAAAAKRAAQcAAIAUKOAAAACQAgUcAAAAUqCAAwAAQAoUcAAAAEiBAg4AAAApUMABAAAgBQo4AAAApEABBwAAgBQo4AAAAJACBRwAAABSoIADAABAChRwAAAASIECDgAAAClQwAEAACAFCjgAAACkQAEHAACAFCjgAAAAkAIFHAAAAFKggAMAAEAKFHAAAABIgQIOAAAAKVDAAQAAIAUKOAAAAKRAAQcAAIAUKOAAAACQAgUcAAAAUjCmAt7c3Bxz5syJsrKyqK6ujl27dp10/aZNm+KjH/1onHvuuVFVVRWrVq2Kd999d0wHBshn8hEgO/kIMIYCvm3btmhoaIjGxsbYs2dPzJ07N+rq6uLtt9/Ouv7pp5+O1atXR2NjY+zbty+eeOKJ2LZtW9x7771nfHiAfCIfAbKTjwCDci7gjzzySNx2222xYsWK+PjHPx6bN2+O8847L5588sms619++eW49tpr4+abb445c+bEZz/72bjppptO+VtPgMlGPgJkJx8BBuVUwPv6+mL37t1RW1s7fAfFxVFbWxttbW1Z91xzzTWxe/fuocA8cOBA7NixI2644YZRH6e3tze6u7tH3ADymXwEyE4+AgybksviI0eORH9/f1RWVo64XllZGfv378+65+abb44jR47Epz/96UiSJI4fPx533HHHSV9C1NTUFPfff38uRwOYUPIRIDv5CDBs3N8FfefOnbFhw4Z47LHHYs+ePfHss8/G9u3b44EHHhh1z5o1a6Krq2vodujQofE+JkDq5CNAdvIRKFQ5PQM+bdq0KCkpic7OzhHXOzs7Y8aMGVn33HfffbF06dK49dZbIyLiiiuuiJ6enrj99ttj7dq1UVx84u8AMplMZDKZXI4GMKHkI0B28hFgWE7PgJeWlsb8+fOjtbV16NrAwEC0trZGTU1N1j3vvPPOCSFZUlISERFJkuR6XoC8JB8BspOPAMNyegY8IqKhoSGWL18eCxYsiIULF8amTZuip6cnVqxYERERy5Yti9mzZ0dTU1NERCxe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cT1nrnKd6y8dP348Of/885Pvfe9743XEMRvLbMePH0+uueaa5Lvf/W6yfPnyvAzPJMl9tu985zvJxRdfnPT19aV1xDHJda6VK1cmn/nMZ0Zca2hoSK699tpxPeeZOp0A/epXv5p84hOfGHGtvr4+qaurG8eTjU2h5mOSFG5GysdhkyUfk+SDkZHyUT7mg0LNSPk4TD6e3IS/BL2vry92794dtbW1Q9eKi4ujtrY22trasu5pa2sbsT4ioq6ubtT1E2Esc/2ld955J95777248MILx+uYYzLW2b7+9a/H9OnT45ZbbknjmGMyltl+/OMfR01NTaxcuTIqKyvj8ssvjw0bNkR/f39axz6lscx1zTXXxO7du4deYnTgwIHYsWNH3HDDDamceTxNhgyJKNx8jCjcjJSPI02GfIyQkX+ukDOkkGf7S/mYjxGFm5HycST5eHJTzuahxuLIkSPR398flZWVI65XVlbG/v37s+7p6OjIur6jo2Pczpmrscz1l+65556YNWvWCd/oiTaW2V566aV44oknor29PYUTjt1YZjtw4ED853/+Z3zxi1+MHTt2xBtvvBFf/vKX47333ovGxsY0jn1KY5nr5ptvjiNHjsSnP/3pSJIkjh8/HnfccUfce++9aRx5XI2WId3d3fGnP/0pzj333Ak62UiFmo8RhZuR8nGkyZCPETLyz8nHiVeo+RhRuBkpH0eSjyc34c+Ak93GjRtj69at8dxzz0VZWdlEH+eMHDt2LJYuXRpbtmyJadOmTfRxzrqBgYGYPn16PP744zF//vyor6+PtWvXxubNmyf6aGdk586dsWHDhnjsscdiz5498eyzz8b27dvjgQcemOijQcFkpHycvGQk+apQ8jGisDNSPn5wTfgz4NOmTYuSkpLo7Owccb2zszNmzJiRdc+MGTNyWj8RxjLX+x566KHYuHFj/OxnP4srr7xyPI85JrnO9pvf/CbeeuutWLx48dC1gYGBiIiYMmVKvPbaa3HJJZeM76FP01i+bzNnzoxzzjknSkpKhq597GMfi46Ojujr64vS0tJxPfPpGMtc9913XyxdujRuvfXWiIi44ooroqenJ26//fZYu3ZtFBdP3t/fjZYh5eXlefPsTkTh5mNE4WakfBxpMuRjhIz8c/Jx4hVqPkYUbkbKx5Hk48lN+PSlpaUxf/78aG1tHbo2MDAQ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ti0aGhqisbEx9uzZE3Pnzo26urp4++23s67fuXNn3HTTTfHzn/882traoqqqKj772c/G7373uzM+PEA+kY8A2clHgEFFSZIkuWyorq6Oq6++Oh599NGIiBgYGIiqqqq48847Y/Xq1afc39/fH1OnTo1HH300li1bdlqP2d3dHRUVFdHV1RXl5eW5HBcgtQyRj8BkIx8BRjceOZLTM+B9fX2xe/fuqK2tHb6D4uKora2Ntra207qPd955J95777248MILR13T29sb3d3dI24A+Uw+AmQnHwGG5VTAjxw5Ev39/VFZWTniemVlZXR0dJzWfdxzzz0xa9asESH8l5qamqKiomLoVlVVlcsxAVInHwGyk48Aw1J9F/SNGzfG1q1b47nnnouysrJR161Zsya6urqGbocOHUrxlADpk48A2clHoJBMyWXxtGnToqSkJDo7O0dc7+zsjBkzZpx070MPPRQbN26Mn/3sZ3HllVeedG0mk4lMJpPL0QAmlHwEyE4+AgzL6Rnw0tLSmD9/frS2tg5dGxgYiNbW1qipqRl134MPPhgPPPBAtLS0xIIFC8Z+WoA8JR8BspOPAMNyegY8IqKhoSGWL18eCxYsiIULF8amTZuip6cnVqxYERERy5Yti9mzZ0dTU1NERPzbv/1brF+/Pp5++umYM2fO0N/6fOhDH4oPfehDZ3EUgIklHwGyk48Ag3Iu4PX19XH48OFYv359dHR0xLx586KlpWXojTUOHjwYxcXDT6x/5zvfib6+vvj85z8/4n4aGxvja1/72pmdHiCPyEeA7OQjwKCcPwd8IvgcR+BMFHKGFPJswPgr5Awp5NmAdEz454ADAAAAY6OAAwAAQAoUcAAAAEiBAg4AAAApUMABAAAgBQo4AAAApEABBwAAgBQo4AAAAJACBRwAAABSoIADAABAChRwAAAASIECDgAAAClQwAEAACAFCjgAAACkQAEHAACAFCjgAAAAkAIFHAAAAFKggAMAAEAKFHAAAABIgQIOAAAAKVDAAQAAIAUKOAAAAKRAAQcAAIAUKOAAAACQAgUcAAAAUqCAAwAAQAoUcAAAAEiBAg4AAAApUMABAAAgBWMq4M3NzTFnzpwoKyuL6urq2LVr10nX/+hHP4rLLrssysrK4oorrogdO3aM6bAA+U4+AmQnHwHGUMC3bdsWDQ0N0djYGHv27Im5c+dGXV1dvP3221nXv/zyy3HTTTfFLbfcEnv37o0lS5bEkiVL4te//vUZHx4gn8hHgOzkI8CgoiRJklw2VFdXx9VXXx2PPvpoREQMDAxEVVVV3HnnnbF69eoT1tfX10dPT0/85Cc/Gbr2qU99KubNmxebN28+rcfs7u6OioqK6OrqivLy8lyOC5BahshHYLKRjwCjG48cyekZ8L6+vti9e3fU1tYO30FxcdTW1kZbW1vWPW1tbSPWR0TU1dWNuh5gMpKPANnJR4BhU3JZfOTIkejv74/KysoR1ysrK2P//v1Z93R0dGRd39HRMerj9Pb2Rm9v79DXXV1dETH4GwiAXL2fHTm+4Ccn8hGYjOQjwOjGIyNzKuBpaWpqivvvv/+E61VVVRNwGqBQ/O///m9UVFRM9DHOiHwExoN8BBjd2czInAr4tGnToqSkJDo7O0dc7+zsjBkzZmTdM2PGjJzWR0SsWbMmGhoahr4+evRofPjDH46DBw9O+n8c/lx3d3dUVVXFoUOHCu5vk8w2ORXqbF1dXXHRRRfFhRdeOG6PIR/PvkL9eSzUuSLMNhnJx8mpUH8eIwp3tkKdK6KwZxuPjMypgJeWlsb8+fOjtbU1lixZEhGDb6LR2toaX/nKV7LuqampidbW1rj77ruHrr344otRU1Mz6uNkMpnIZDInXK+oqCi4b2pERHl5eUHOFWG2yapQZysuHtMnL54W+Th+CvXnsVDnijDbZCQfJ6dC/XmMKNzZCnWuiMKe7WxmZM4vQW9oaIjly5fHggULYuHChbFp06bo6emJFStWRETEsmXLYvbs2dHU1BQREXfddVdcd9118fDDD8eNN94YW7dujV/96lfx+OOPn7UhAPKBfATITj4CDMq5gNfX18fhw4dj/fr10dHREfPmzYuWlpahN8o4ePDgiN8QXHPNNfH000/HunXr4t57742/+7u/i+effz4uv/zyszcFQB6QjwDZyUeA/79kEnj33XeTxsbG5N13353oo5xVhTpXkphtsirU2Qp1riQx22RUqHMlidkmo0KdK0nMNlkV6myFOleSmC1XRUkyjp87AQAAAERExPi94wYAAAAwRAEHAACAFCjgAAAAkAIFHAAAAFKQNwW8ubk55syZE2VlZVFdXR27du066fof/ehHcdlll0VZWVlcccUVsWPHjpROmptc5tqyZUssWrQopk6dGlOnTo3a2tpT/neYSLl+z963devWKCoqiiVLlozvAc9ArrMdPXo0Vq5cGTNnzoxMJhOXXnppXv5M5jrXpk2b4qMf/Wice+65UVVVFatWrYp33303pdOevl/84hexePHimDVrVhQVFcXzzz9/yj07d+6MT37yk5HJZOIjH/lIPPXUU+N+zrEq1HyMKNyMlI/DJks+RhRmRsrHkeRjfijUjJSPw+TjSZy191M/A1u3bk1KS0uTJ598Mvnv//7v5LbbbksuuOCCpLOzM+v6X/7yl0lJSUny4IMPJq+++mqybt265JxzzkleeeWVlE9+crnOdfPNNyfNzc3J3r17k3379iX/9E//lFRUVCT/8z//k/LJTy3X2d735ptvJrNnz04WLVqU/MM//EM6h81RrrP19vYmCxYsSG644YbkpZdeSt58881k586dSXt7e8onP7lc5/r+97+fZDKZ5Pvf/37y5ptvJi+88EIyc+bMZNWqVSmf/NR27NiRrF27Nnn22WeTiEiee+65k64/cOBAct555yUNDQ3Jq6++mnz7299OSkpKkpaWlnQOnINCzcckKdyMlI/DJks+JknhZqR8HCYf80OhZqR8HCYfTy4vCvjChQuTlStXDn3d39+fzJo1K2lqasq6/gtf+EJy4403jrhWXV2d/PM///O4njNXuc71l44fP56cf/75yfe+973xOuKYjWW248ePJ9dcc03y3e9+N1m+fHlehmeS5D7bd77zneTiiy9O+vr60jrimOQ618qVK5PPfOYzI641NDQk11577bie80ydToB+9atfTT7xiU+MuFZfX5/U1dWN48nGplDzMUkKNyPl47DJko9J8sHISPkoH/NBoWakfBwmH09uwl+C3tfXF7t3747a2tqha8XFxVFbWxttbW1Z97S1tY1YHxFRV1c36vqJMJa5/tI777wT7733Xlx44YXjdcwxGetsX//612P69Olxyy23pHHMMRnLbD/+8Y+jpqYmVq5cGZWVlXH55ZfHhg0bor+/P61jn9JY5rrmmmti9+7dQy8xOnDgQOzYsSNuuOGGVM48niZDhkQUbj5GFG5GyseRJkM+RsjIP1fIGVLIs/2lfMzHiMLNSPk4knw8uSln81BjceTIkejv74/KysoR1ysrK2P//v1Z93R0dGRd39HRMW7nzNVY5vpL99xzT8yaNeuEb/RE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",
      "text/plain": [
       "<Figure size 1200x900 with 9 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "models = MODELS\n",
    "ncols = 3\n",
    "nrows = int(np.ceil(len(models) / ncols))\n",
    "fig, axes = plt.subplots(nrows, ncols, figsize=(4 * ncols, 3 * nrows), squeeze=False, sharex=False)\n",
    "for ax, model in zip(axes.ravel(), models):\n",
    "    ax.plot([x['lengths_max'] for x in embedding_stats[model]][:-2], [x['th_slope'] for x in embedding_stats[model]][:-2]    , linestyle='-', marker='o', markersize=4, label='Ball')\n",
    "    ax.plot([x['lengths_max'] for x in embedding_stats[model]][:-2], [x['th_slope_rect'] for x in embedding_stats[model]][:-2], linestyle=':', marker='o', markersize=4, label='Ellipsoid')\n",
    "    ax.plot([x['lengths_max'] for x in embedding_stats[model]][:-2], [x['th_slope_cone'] for x in embedding_stats[model]][:-2], linestyle='--', marker='o', markersize=4, label='Cone')\n",
    "    ax.set_xlabel(r\"max sampled input length $\\ell$ (tokens)\")\n",
    "    ax.set_ylabel(r\"upper bound on slope $C$\")\n",
    "    ax.legend(fontsize=10)\n",
    "    ax.set_title(model.capitalize(), fontsize=10)\n",
    "    ax.spines['top'].set_visible(False)\n",
    "    ax.spines['right'].set_visible(False)\n",
    "    ax.xaxis.set_ticks_position('bottom')\n",
    "    ax.yaxis.set_ticks_position('left')\n",
    "    ax.locator_params(axis='y', nbins=2) \n",
    "    ax.locator_params(axis='x', nbins=3) \n",
    "\n",
    "for ax in axes.ravel()[len(models):]:\n",
    "    ax.axis(\"off\")\n",
    "\n",
    "fig.tight_layout()\n",
    "fig.savefig(\"theoretical_slopes_archi.pdf\", dpi=200)\n",
    "plt.show()\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "06fb4f04",
   "metadata": {},
   "source": [
    "### Estimate statistics on models for all lengths\n",
    "The helpers mirror those in `embedding_geometry.py`. `average_pairwise_cosine` is quadratic in the number of samples, so a down-sample option is provided for large matrices."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "a402d451",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_874845/1032888422.py:4: FutureWarning: Calling int on a single element Series is deprecated and will raise a TypeError in the future. Use int(ser.iloc[0]) instead\n",
      "  d = int(model_table.loc[model_table['name'] == model, 'Input hidden size'])\n",
      "/tmp/ipykernel_874845/1032888422.py:5: FutureWarning: Calling int on a single element Series is deprecated and will raise a TypeError in the future. Use int(ser.iloc[0]) instead\n",
      "  v = int(model_table.loc[model_table['name'] == model, 'Vocabulary size'])\n",
      "/tmp/ipykernel_874845/1032888422.py:6: FutureWarning: Calling float on a single element Series is deprecated and will raise a TypeError in the future. Use float(ser.iloc[0]) instead\n",
      "  slope_exp = float(model_table.loc[model_table['name'] == model, 'Slope exp'])\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Summarizing geometry for model pythia-160m\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 768)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_874845/3282313617.py:36: RuntimeWarning: divide by zero encountered in log\n",
      "  b2 = log(beta(1 / 2, (d + 1) / 2)) + log(1 - betainc(1 / 2, (d + 1) / 2, cos(theta / 2) ** 2))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "    - Theoretical slopes: slope=590.0911122325728, slope_cone=576.827761751075, slope_rect=470.8456632179397\n",
      "Summarizing geometry for model pythia-410m\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 1024)\n",
      "    - Theoretical slopes: slope=786.7881496434303, slope_cone=766.2039171012152, slope_rect=602.3464191466994\n",
      "Summarizing geometry for model pythia-1b\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 2048)\n",
      "    - Theoretical slopes: slope=1573.576299286861, slope_cone=1552.806063777484, slope_rect=1099.7066466362169\n",
      "Summarizing geometry for model Qwen2.5-0.5B\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 896)\n",
      "    - Theoretical slopes: slope=624.6587143040047, slope_cone=616.4469252825608, slope_rect=399.7290791623093\n",
      "Summarizing geometry for model Qwen2.5-1.5B\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 1536)\n",
      "    - Theoretical slopes: slope=1070.8435102354365, slope_cone=1059.9409613422542, slope_rect=660.1373852405293\n",
      "Summarizing geometry for model Llama-3.2-1B\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (11000, 2048)\n",
      "    - Theoretical slopes: slope=1448.358999502273, slope_cone=1400.6293290066305, slope_rect=1108.9366523915658\n",
      "Summarizing geometry for model gemma-3-270m\n",
      "  - lengths: [4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096], embeddings shape: (9000, 640)\n",
      "    - Theoretical slopes: slope=426.679188532633, slope_cone=414.41502756853527, slope_rect=408.99999878001876\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>model</th>\n",
       "      <th>lengths_max</th>\n",
       "      <th>d</th>\n",
       "      <th>v</th>\n",
       "      <th>r1</th>\n",
       "      <th>r2</th>\n",
       "      <th>rinf</th>\n",
       "      <th>min_cos</th>\n",
       "      <th>avg_cos</th>\n",
       "      <th>theta</th>\n",
       "      <th>epsilon</th>\n",
       "      <th>rect_mean</th>\n",
       "      <th>th_slope</th>\n",
       "      <th>th_slope_cone</th>\n",
       "      <th>th_slope_rect</th>\n",
       "      <th>slope_exp</th>\n",
       "      <th>ratio_th</th>\n",
       "      <th>ratio_th_cone</th>\n",
       "      <th>ratio_th_rect</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>pythia-160m</td>\n",
       "      <td>4096</td>\n",
       "      <td>768</td>\n",
       "      <td>50304</td>\n",
       "      <td>7379.292480</td>\n",
       "      <td>271.890717</td>\n",
       "      <td>63.625000</td>\n",
       "      <td>-0.387802</td>\n",
       "      <td>0.917150</td>\n",
       "      <td>1.969042</td>\n",
       "      <td>0.031067</td>\n",
       "      <td>24.322632</td>\n",
       "      <td>590.091112</td>\n",
       "      <td>576.827762</td>\n",
       "      <td>470.845663</td>\n",
       "      <td>90.40</td>\n",
       "      <td>6.527557</td>\n",
       "      <td>6.380838</td>\n",
       "      <td>5.208470</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>pythia-410m</td>\n",
       "      <td>4096</td>\n",
       "      <td>1024</td>\n",
       "      <td>50304</td>\n",
       "      <td>2881.314941</td>\n",
       "      <td>112.619034</td>\n",
       "      <td>54.906250</td>\n",
       "      <td>-0.302617</td>\n",
       "      <td>0.659707</td>\n",
       "      <td>1.878234</td>\n",
       "      <td>0.026810</td>\n",
       "      <td>16.047890</td>\n",
       "      <td>786.788150</td>\n",
       "      <td>766.203917</td>\n",
       "      <td>602.346419</td>\n",
       "      <td>112.40</td>\n",
       "      <td>6.999895</td>\n",
       "      <td>6.816761</td>\n",
       "      <td>5.358954</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>pythia-1b</td>\n",
       "      <td>4096</td>\n",
       "      <td>2048</td>\n",
       "      <td>50304</td>\n",
       "      <td>4175.441895</td>\n",
       "      <td>116.046059</td>\n",
       "      <td>65.687500</td>\n",
       "      <td>-0.612078</td>\n",
       "      <td>0.440558</td>\n",
       "      <td>2.229482</td>\n",
       "      <td>0.032074</td>\n",
       "      <td>11.082831</td>\n",
       "      <td>1573.576299</td>\n",
       "      <td>1552.806064</td>\n",
       "      <td>1099.706647</td>\n",
       "      <td>287.40</td>\n",
       "      <td>5.475213</td>\n",
       "      <td>5.402944</td>\n",
       "      <td>3.826398</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>Qwen2.5-0.5B</td>\n",
       "      <td>4096</td>\n",
       "      <td>896</td>\n",
       "      <td>151936</td>\n",
       "      <td>5010.494141</td>\n",
       "      <td>379.666595</td>\n",
       "      <td>309.000000</td>\n",
       "      <td>-0.620301</td>\n",
       "      <td>0.456711</td>\n",
       "      <td>2.239923</td>\n",
       "      <td>0.150879</td>\n",
       "      <td>36.697079</td>\n",
       "      <td>624.658714</td>\n",
       "      <td>616.446925</td>\n",
       "      <td>399.729079</td>\n",
       "      <td>71.00</td>\n",
       "      <td>8.798010</td>\n",
       "      <td>8.682351</td>\n",
       "      <td>5.629987</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>Qwen2.5-1.5B</td>\n",
       "      <td>4096</td>\n",
       "      <td>1536</td>\n",
       "      <td>151936</td>\n",
       "      <td>3876.548340</td>\n",
       "      <td>204.519714</td>\n",
       "      <td>153.625000</td>\n",
       "      <td>-0.697339</td>\n",
       "      <td>0.508266</td>\n",
       "      <td>2.342475</td>\n",
       "      <td>0.075012</td>\n",
       "      <td>14.284251</td>\n",
       "      <td>1070.843510</td>\n",
       "      <td>1059.940961</td>\n",
       "      <td>660.137385</td>\n",
       "      <td>79.60</td>\n",
       "      <td>13.452808</td>\n",
       "      <td>13.315841</td>\n",
       "      <td>8.293183</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>Llama-3.2-1B</td>\n",
       "      <td>4096</td>\n",
       "      <td>2048</td>\n",
       "      <td>128256</td>\n",
       "      <td>3770.023682</td>\n",
       "      <td>104.561874</td>\n",
       "      <td>41.062500</td>\n",
       "      <td>-0.159910</td>\n",
       "      <td>0.608426</td>\n",
       "      <td>1.731396</td>\n",
       "      <td>0.020050</td>\n",
       "      <td>12.083597</td>\n",
       "      <td>1448.359000</td>\n",
       "      <td>1400.629329</td>\n",
       "      <td>1108.936652</td>\n",
       "      <td>137.80</td>\n",
       "      <td>10.510588</td>\n",
       "      <td>10.164219</td>\n",
       "      <td>8.047436</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>gemma-3-270m</td>\n",
       "      <td>4096</td>\n",
       "      <td>640</td>\n",
       "      <td>262144</td>\n",
       "      <td>1063.375732</td>\n",
       "      <td>50.600838</td>\n",
       "      <td>10.742188</td>\n",
       "      <td>-0.251585</td>\n",
       "      <td>0.002519</td>\n",
       "      <td>1.825114</td>\n",
       "      <td>0.005245</td>\n",
       "      <td>15.242126</td>\n",
       "      <td>426.679189</td>\n",
       "      <td>414.415028</td>\n",
       "      <td>408.999999</td>\n",
       "      <td>44.54</td>\n",
       "      <td>9.579685</td>\n",
       "      <td>9.304334</td>\n",
       "      <td>9.182757</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "          model  lengths_max     d       v           r1          r2  \\\n",
       "0   pythia-160m         4096   768   50304  7379.292480  271.890717   \n",
       "1   pythia-410m         4096  1024   50304  2881.314941  112.619034   \n",
       "2     pythia-1b         4096  2048   50304  4175.441895  116.046059   \n",
       "3  Qwen2.5-0.5B         4096   896  151936  5010.494141  379.666595   \n",
       "4  Qwen2.5-1.5B         4096  1536  151936  3876.548340  204.519714   \n",
       "5  Llama-3.2-1B         4096  2048  128256  3770.023682  104.561874   \n",
       "6  gemma-3-270m         4096   640  262144  1063.375732   50.600838   \n",
       "\n",
       "         rinf   min_cos   avg_cos     theta   epsilon  rect_mean     th_slope  \\\n",
       "0   63.625000 -0.387802  0.917150  1.969042  0.031067  24.322632   590.091112   \n",
       "1   54.906250 -0.302617  0.659707  1.878234  0.026810  16.047890   786.788150   \n",
       "2   65.687500 -0.612078  0.440558  2.229482  0.032074  11.082831  1573.576299   \n",
       "3  309.000000 -0.620301  0.456711  2.239923  0.150879  36.697079   624.658714   \n",
       "4  153.625000 -0.697339  0.508266  2.342475  0.075012  14.284251  1070.843510   \n",
       "5   41.062500 -0.159910  0.608426  1.731396  0.020050  12.083597  1448.359000   \n",
       "6   10.742188 -0.251585  0.002519  1.825114  0.005245  15.242126   426.679189   \n",
       "\n",
       "   th_slope_cone  th_slope_rect  slope_exp   ratio_th  ratio_th_cone  \\\n",
       "0     576.827762     470.845663      90.40   6.527557       6.380838   \n",
       "1     766.203917     602.346419     112.40   6.999895       6.816761   \n",
       "2    1552.806064    1099.706647     287.40   5.475213       5.402944   \n",
       "3     616.446925     399.729079      71.00   8.798010       8.682351   \n",
       "4    1059.940961     660.137385      79.60  13.452808      13.315841   \n",
       "5    1400.629329    1108.936652     137.80  10.510588      10.164219   \n",
       "6     414.415028     408.999999      44.54   9.579685       9.304334   \n",
       "\n",
       "   ratio_th_rect  \n",
       "0       5.208470  \n",
       "1       5.358954  \n",
       "2       3.826398  \n",
       "3       5.629987  \n",
       "4       8.293183  \n",
       "5       8.047436  \n",
       "6       9.182757  "
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\n",
    "embedding_stats = []\n",
    "for model in MODELS:\n",
    "\n",
    "    d = int(model_table.loc[model_table['name'] == model, 'Input hidden size'])\n",
    "    v = int(model_table.loc[model_table['name'] == model, 'Vocabulary size'])\n",
    "    slope_exp = float(model_table.loc[model_table['name'] == model, 'Slope exp'])\n",
    "\n",
    "    print(f\"Summarizing geometry for model {model}\")\n",
    "    stats = stats_from_models(model, d, v, slope_exp, lengths=SAMPLED_LENGTHS)\n",
    "    embedding_stats.append(stats)\n",
    "\n",
    "df_results = pd.DataFrame(embedding_stats)\n",
    "\n",
    "df_results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "93c65995",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>model</th>\n",
       "      <th>d</th>\n",
       "      <th>v</th>\n",
       "      <th>rinf</th>\n",
       "      <th>theta</th>\n",
       "      <th>epsilon</th>\n",
       "      <th>ratio_th</th>\n",
       "      <th>ratio_th_cone</th>\n",
       "      <th>ratio_th_rect</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>pythia-160m</td>\n",
       "      <td>768</td>\n",
       "      <td>50304</td>\n",
       "      <td>63.625000</td>\n",
       "      <td>1.969042</td>\n",
       "      <td>0.031067</td>\n",
       "      <td>6.527557</td>\n",
       "      <td>6.380838</td>\n",
       "      <td>5.208470</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>pythia-410m</td>\n",
       "      <td>1024</td>\n",
       "      <td>50304</td>\n",
       "      <td>54.906250</td>\n",
       "      <td>1.878234</td>\n",
       "      <td>0.026810</td>\n",
       "      <td>6.999895</td>\n",
       "      <td>6.816761</td>\n",
       "      <td>5.358954</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>pythia-1b</td>\n",
       "      <td>2048</td>\n",
       "      <td>50304</td>\n",
       "      <td>65.687500</td>\n",
       "      <td>2.229482</td>\n",
       "      <td>0.032074</td>\n",
       "      <td>5.475213</td>\n",
       "      <td>5.402944</td>\n",
       "      <td>3.826398</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>Qwen2.5-0.5B</td>\n",
       "      <td>896</td>\n",
       "      <td>151936</td>\n",
       "      <td>309.000000</td>\n",
       "      <td>2.239923</td>\n",
       "      <td>0.150879</td>\n",
       "      <td>8.798010</td>\n",
       "      <td>8.682351</td>\n",
       "      <td>5.629987</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>Qwen2.5-1.5B</td>\n",
       "      <td>1536</td>\n",
       "      <td>151936</td>\n",
       "      <td>153.625000</td>\n",
       "      <td>2.342475</td>\n",
       "      <td>0.075012</td>\n",
       "      <td>13.452808</td>\n",
       "      <td>13.315841</td>\n",
       "      <td>8.293183</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>Llama-3.2-1B</td>\n",
       "      <td>2048</td>\n",
       "      <td>128256</td>\n",
       "      <td>41.062500</td>\n",
       "      <td>1.731396</td>\n",
       "      <td>0.020050</td>\n",
       "      <td>10.510588</td>\n",
       "      <td>10.164219</td>\n",
       "      <td>8.047436</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>gemma-3-270m</td>\n",
       "      <td>640</td>\n",
       "      <td>262144</td>\n",
       "      <td>10.742188</td>\n",
       "      <td>1.825114</td>\n",
       "      <td>0.005245</td>\n",
       "      <td>9.579685</td>\n",
       "      <td>9.304334</td>\n",
       "      <td>9.182757</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "          model     d       v        rinf     theta   epsilon   ratio_th  \\\n",
       "0   pythia-160m   768   50304   63.625000  1.969042  0.031067   6.527557   \n",
       "1   pythia-410m  1024   50304   54.906250  1.878234  0.026810   6.999895   \n",
       "2     pythia-1b  2048   50304   65.687500  2.229482  0.032074   5.475213   \n",
       "3  Qwen2.5-0.5B   896  151936  309.000000  2.239923  0.150879   8.798010   \n",
       "4  Qwen2.5-1.5B  1536  151936  153.625000  2.342475  0.075012  13.452808   \n",
       "5  Llama-3.2-1B  2048  128256   41.062500  1.731396  0.020050  10.510588   \n",
       "6  gemma-3-270m   640  262144   10.742188  1.825114  0.005245   9.579685   \n",
       "\n",
       "   ratio_th_cone  ratio_th_rect  \n",
       "0       6.380838       5.208470  \n",
       "1       6.816761       5.358954  \n",
       "2       5.402944       3.826398  \n",
       "3       8.682351       5.629987  \n",
       "4      13.315841       8.293183  \n",
       "5      10.164219       8.047436  \n",
       "6       9.304334       9.182757  "
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_results[['model','d','v','rinf','theta','epsilon','ratio_th','ratio_th_cone','ratio_th_rect']]\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "31688a7d",
   "metadata": {},
   "source": [
    "### Create rectangle coordinate txt for each model "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "df99443b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Saved rectangle for pythia-160m to data/rects_inf/pythia-160m.txt\n",
      "Saved rectangle for pythia-410m to data/rects_inf/pythia-410m.txt\n",
      "Saved rectangle for pythia-1b to data/rects_inf/pythia-1b.txt\n",
      "Saved rectangle for pythia-1.4b to data/rects_inf/pythia-1.4b.txt\n",
      "Saved rectangle for Qwen2.5-0.5B to data/rects_inf/Qwen2.5-0.5B.txt\n",
      "Saved rectangle for Qwen2.5-1.5B to data/rects_inf/Qwen2.5-1.5B.txt\n",
      "Saved rectangle for Llama-3.2-1B to data/rects_inf/Llama-3.2-1B.txt\n",
      "Saved rectangle for gemma-3-270m to data/rects_inf/gemma-3-270m.txt\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "\n",
    "# Create directory if it doesn't exist\n",
    "os.makedirs('rects_inf', exist_ok=True)\n",
    "\n",
    "# Save rectangles for each model\n",
    "for model in MODELS:\n",
    "    embs = load_embeddings_from_disk(model, lengths=SAMPLED_LENGTHS)\n",
    "\n",
    "    # Create min/max pairs for each dimension\n",
    "    rect_min = embs.min(axis=0)\n",
    "    rect_max = embs.max(axis=0)\n",
    "\n",
    "    # Save to file\n",
    "    output_path = f'data/rects_inf/{model}.txt'\n",
    "    with open(output_path, 'w') as f:\n",
    "        for i, (min_val, max_val) in enumerate(zip(rect_min, rect_max)):\n",
    "            f.write(f\"{min_val:.6f} {max_val:.6f}\\n\")\n",
    "    \n",
    "    print(f\"Saved rectangle for {model} to {output_path}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "75dbea33",
   "metadata": {},
   "source": [
    "### Estimate median convolution slope for each model "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "a22e23db",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-12.78589114 -25.68060793 -38.55165293 -51.41821671 -64.28341159]\n",
      "[-14.68301113 -29.22819553 -43.77271853 -58.31411373 -72.85354972]\n",
      "[-14.80884088 -29.5682738  -44.27889617 -58.99042751 -73.70024238]\n",
      "[-17.03438638 -33.25784241 -49.78140343 -66.31063815 -82.82055669]\n",
      "[-17.03438638 -34.06877287 -50.98537599 -67.86552789 -84.72565809]\n",
      "[-16.11809565 -31.87154802 -47.65317143 -63.4539057  -79.24736727]\n",
      "[-12.76468893 -25.72918762 -38.74346485 -51.77965872]\n",
      "Model\t-log(slope)\t-log(vocab_size)\tR^2\n",
      "EleutherAI/pythia-160m\t-12.87\t-10.83\t1.00\n",
      "EleutherAI/pythia-1b\t-14.72\t-10.83\t1.00\n",
      "EleutherAI/pythia-410m\t-14.54\t-10.83\t1.00\n",
      "Qwen/Qwen2.5-0.5B\t-16.48\t-11.93\t1.00\n",
      "Qwen/Qwen2.5-1.5B\t-16.90\t-11.93\t1.00\n",
      "google/gemma-3-270m\t-13.01\t-12.48\t1.00\n",
      "meta-llama/Llama-3.2-1B\t-15.79\t-11.76\t1.00\n"
     ]
    }
   ],
   "source": [
    "import csv\n",
    "from collections import defaultdict\n",
    "from pathlib import Path\n",
    "from math import log\n",
    "import numpy as np\n",
    "path = Path(\"/home/mario/codes/llm_vis/figures/median_iterations/median_by_iteration.txt\")\n",
    "\n",
    "# Collect data per model\n",
    "by_model = defaultdict(list)\n",
    "with path.open(newline=\"\") as f:\n",
    "    reader = csv.DictReader(f)\n",
    "    for row in reader:\n",
    "        model = row[\"model\"]\n",
    "        x = float(row[\"iteration\"])\n",
    "        y = float(row[\"median\"])\n",
    "        by_model[model].append((x, y))\n",
    "\n",
    "# Simple linear regression: slope = cov(x,y)/var(x)\n",
    "slopes = []\n",
    "for model, pts in by_model.items():\n",
    "    xs = [p[0] for p in pts]\n",
    "    ys = [p[1] for p in pts]\n",
    "    x = np.array(xs)\n",
    "    y = np.log(np.array(ys))\n",
    "    print(y[:5])\n",
    "    A = np.vstack([x, np.ones(len(x))]).T\n",
    "    m, c = np.linalg.lstsq(A, y, rcond=None)[0]\n",
    "    r2 = 1 - (np.sum((y - (m*x + c))**2) / np.sum((y - y.mean())**2))\n",
    "    slopes.append((model, m, c, r2))\n",
    "\n",
    "vs = [50304, 50304, 50304, 151936, 151936,262144,128256]\n",
    "slopes.sort(key=lambda t: t[0])\n",
    "print(\"Model\\t-log(slope)\\t-log(vocab_size)\\tR^2\")\n",
    "for model, m, b, n in slopes:\n",
    "    print(f\"{model}\\t{m:.2f}\\t{-log(vs.pop(0)):.2f}\\t{r2:.2f}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eaef8b7d",
   "metadata": {},
   "source": [
    "| Model | Slope Conv | Slope Vocab\n",
    "| - | - | -\n",
    "| EleutherAI/pythia-160m\t| -12.87 |\t-10.83\n",
    "| EleutherAI/pythia-1b\t    | -14.71 |\t-10.83\n",
    "| EleutherAI/pythia-410m\t| -14.54 |\t-10.83\n",
    "| Qwen/Qwen2.5-0.5B\t        | -16.50 |\t-11.93\n",
    "| Qwen/Qwen2.5-1.5B\t        | -16.87 |\t-11.93\n",
    "| google/gemma-3-270m\t    | -13.03 |\t-12.48\n",
    "| meta-llama/Llama-3.2-1B\t| -15.79 |\t-11.76\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "4bb316d3",
   "metadata": {},
   "outputs": [],
   "source": [
    "df_results['log_slope_conv'] = pd.Series([ -12.87, -14.71, -14.54, -16.50, -16.87, -13.03, -15.79 ])\n",
    "# df_results['log_slope_conv'] = [ -m for model, m, b, r2 in slopes ]\n",
    "\n",
    "def th_slope_conv_rect(slope_rect, v, log_slope_conv):\n",
    "    \"\"\"Theoretical slope using rectangle with convolution.\"\"\"\n",
    "    return float(slope_rect * log(v) / log_slope_conv)\n",
    "\n",
    "df_results['th_slope_conv_rect'] = df_results.apply(\n",
    "    lambda row: th_slope_conv_rect(row['th_slope_rect'], row['v'], -row['log_slope_conv']), \n",
    "    axis=1)\n",
    "df_results['ratio_conv_rect'] = df_results['th_slope_conv_rect'] / df_results['slope_exp']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a28d1948",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>model</th>\n",
       "      <th>ratio_th</th>\n",
       "      <th>ratio_th_cone</th>\n",
       "      <th>ratio_th_rect</th>\n",
       "      <th>ratio_conv_rect</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>pythia-160m</td>\n",
       "      <td>6.53</td>\n",
       "      <td>6.38</td>\n",
       "      <td>5.21</td>\n",
       "      <td>4.38</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>pythia-410m</td>\n",
       "      <td>7.00</td>\n",
       "      <td>6.82</td>\n",
       "      <td>5.36</td>\n",
       "      <td>3.94</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>pythia-1b</td>\n",
       "      <td>5.48</td>\n",
       "      <td>5.40</td>\n",
       "      <td>3.83</td>\n",
       "      <td>2.85</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>Qwen2.5-0.5B</td>\n",
       "      <td>8.80</td>\n",
       "      <td>8.68</td>\n",
       "      <td>5.63</td>\n",
       "      <td>4.07</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>Qwen2.5-1.5B</td>\n",
       "      <td>13.45</td>\n",
       "      <td>13.32</td>\n",
       "      <td>8.29</td>\n",
       "      <td>5.87</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>Llama-3.2-1B</td>\n",
       "      <td>10.51</td>\n",
       "      <td>10.16</td>\n",
       "      <td>8.05</td>\n",
       "      <td>7.26</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>gemma-3-270m</td>\n",
       "      <td>9.58</td>\n",
       "      <td>9.30</td>\n",
       "      <td>9.18</td>\n",
       "      <td>7.26</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "          model  ratio_th  ratio_th_cone  ratio_th_rect  ratio_conv_rect\n",
       "0   pythia-160m      6.53           6.38           5.21             4.38\n",
       "1   pythia-410m      7.00           6.82           5.36             3.94\n",
       "2     pythia-1b      5.48           5.40           3.83             2.85\n",
       "3  Qwen2.5-0.5B      8.80           8.68           5.63             4.07\n",
       "4  Qwen2.5-1.5B     13.45          13.32           8.29             5.87\n",
       "5  Llama-3.2-1B     10.51          10.16           8.05             7.26\n",
       "6  gemma-3-270m      9.58           9.30           9.18             7.26"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_results[['model', 'ratio_th','ratio_th_cone','ratio_th_rect','ratio_conv_rect']]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0117bf10",
   "metadata": {},
   "source": [
    "| model\t        | ratio_th\t| ratio_th_cone\t| ratio_th_rect\t| ratio_conv_rect\n",
    "| - | - | - | - | -\n",
    "| pythia-160m\t| 5.44\t    | 5.29\t        | 4.12\t        | 3.46\n",
    "| pythia-410m\t| 5.71\t    | 5.52\t        | 4.07\t        | 2.99\n",
    "| pythia-1b\t    | 4.58\t    | 4.51\t        | 2.94\t        | 2.19\n",
    "| Qwen2.5-0.5B\t| 9.00\t    | 8.88\t        | 5.83\t        | 4.21\n",
    "| Qwen2.5-1.5B\t| 12.63\t    | 12.49\t        | 7.47\t        | 5.29\n",
    "| Llama-3.2-1B\t| 8.20\t    | 7.85\t        | 5.75\t        | 5.19\n",
    "| gemma-3-270m\t| 5.93\t    | 5.66\t        | 5.54\t        | 4.38\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b7e125b5",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "id": "0cf85616",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# Data from the table\n",
    "data = {\n",
    "    'pythia-160m': [5.44, 5.29, 4.12, 3.46],\n",
    "    'pythia-410m': [5.71, 5.52, 4.07, 2.99],\n",
    "    'pythia-1b': [4.58, 4.51, 2.94, 2.19],\n",
    "    'Qwen2.5-0.5B': [9.00, 8.88, 5.83, 4.21],\n",
    "    'Qwen2.5-1.5B': [12.63, 12.49, 7.47, 5.29],\n",
    "    'Llama-3.2-1B': [8.20, 7.85, 5.75, 5.19],\n",
    "    'gemma-3-270m': [5.93, 5.66, 5.54, 4.38],\n",
    "}\n",
    "\n",
    "# colors = [\"#7294D4\", \"#C27D86\", \"#88A07A\", \"#7D8491\"]\n",
    "colors =[\"#7593b3\", \"#71b481\", \"#b66a67\",\"#bb9f6b\", ]\n",
    "\n",
    "ratio_names = ['Theorem', 'Cone', 'Ellipsoid', 'Ellipsoid + Conv']\n",
    "models = list(data.keys())\n",
    "x = np.arange(len(models))\n",
    "width = 0.2\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(4, 5))\n",
    "\n",
    "for i, ratio_name in enumerate(ratio_names):\n",
    "    values = [data[model][i] for model in models]\n",
    "    ax.bar(x + i * width, values, width, label=ratio_name, color=colors[i])\n",
    "\n",
    "ax.set_xlabel('Model', fontsize=10)\n",
    "ax.set_ylabel(r'Ratio ($\\downarrow$)', fontsize=10)\n",
    "# ax.set_title('Comparison of Ratios Across Models', fontsize=10)\n",
    "ax.set_xticks(x + width * 1.5)\n",
    "ax.set_xticklabels(models, rotation=90, ha='right')\n",
    "ax.set_yticks(np.arange(0, 11, 5))\n",
    "ax.legend(fontsize=10, frameon=False)\n",
    "ax.spines['top'].set_visible(False)\n",
    "ax.spines['right'].set_visible(False)\n",
    "ax.grid(axis='y', alpha=0.3)\n",
    "\n",
    "fig.tight_layout()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "78bc6ec0",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "llm_vis",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.23"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
