{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "51d11caf-0971-4324-b63b-819b714a9c3c",
   "metadata": {},
   "source": [
    "# Modular Addition"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "80f249f1-6985-4c73-86cd-04e1adac3e8d",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import random\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm import tqdm\n",
    "import pickle"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9fd05577-db56-4d0a-bb93-1d0b48cecaf6",
   "metadata": {},
   "source": [
    "## Dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "f19bd1ad-9e8f-4720-b317-afe13fafae88",
   "metadata": {},
   "outputs": [],
   "source": [
    "def one_hot(p):\n",
    "    \"\"\"One-hot encode an integer value in R^p.\"\"\"\n",
    "    vec = np.zeros(p)\n",
    "    vec[0] = 1\n",
    "    return vec\n",
    "\n",
    "def generate_template(p, magnitude, exponent):\n",
    "    weight = magnitude * np.power(np.arange(1, p), -exponent)  # Power-law singular values\n",
    "    template = np.ones(p)  # Base term (DC component)\n",
    "    for freq in range(1, p):\n",
    "        template += weight[freq-1] * np.cos(np.arange(p) * freq / p * 2 * np.pi)\n",
    "    return template / p\n",
    "\n",
    "def generate_fixed_template(p):\n",
    "    # Generate template array from Fourier spectrum\n",
    "    spectrum = np.zeros(p, dtype=complex)\n",
    "    \n",
    "    # Set only three frequencies with specific amplitudes\n",
    "    spectrum[1] = 10 # Positive frequency\n",
    "    spectrum[-1] = 10  # Negative frequency (conjugate)\n",
    "    spectrum[3] = 5 # Second frequency\n",
    "    spectrum[-3] =  5  # Its conjugate\n",
    "    spectrum[5] = 2.5  # Third frequency \n",
    "    spectrum[-5] = 2.5  # Its conjugate\n",
    "    \n",
    "    # Generate signal from spectrum\n",
    "    template = np.fft.ifft(spectrum).real\n",
    "\n",
    "    return template\n",
    "\n",
    "def ModularAdditionDataset(p, template):\n",
    "    # Initialize data arrays\n",
    "    X = np.zeros((p * p, 2, p))  # Shape: (p^2, 2, p)\n",
    "    Y = np.zeros((p * p, p))     # Shape: (p^2, p)\n",
    "    \n",
    "    # Generate the dataset\n",
    "    idx = 0\n",
    "    for a in range(p):\n",
    "        for b in range(p):\n",
    "            q = (a + b) % p  # a + b mod p\n",
    "            X[idx, 0, :] = np.roll(template, a)\n",
    "            X[idx, 1, :] = np.roll(template, b)\n",
    "            Y[idx, :] = np.roll(template, q)\n",
    "            idx += 1\n",
    "            \n",
    "    return X, Y"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7a0ecbbd-ceaf-4bef-af4a-13a22fa70063",
   "metadata": {},
   "source": [
    "## Architecture"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "2cf22b7d-49e7-445b-8742-2e75cd1fa55a",
   "metadata": {},
   "outputs": [],
   "source": [
    "class TwoLayerNet(nn.Module):\n",
    "    def __init__(self, p, hidden_size, init_scale=1.0):\n",
    "        super(TwoLayerNet, self).__init__()\n",
    "        \n",
    "        # Store dimensions\n",
    "        self.p = p\n",
    "        self.hidden_size = hidden_size\n",
    "        self.init_scale = init_scale\n",
    "        \n",
    "        # Initialize parameters \n",
    "        self.U = nn.Parameter(self.init_scale * torch.randn(hidden_size, p) / np.sqrt(2*p))\n",
    "        self.V = nn.Parameter(self.init_scale * torch.randn(hidden_size, p) / np.sqrt(2*p))\n",
    "        self.W = nn.Parameter(self.init_scale * torch.randn(hidden_size, p) / np.sqrt(p))\n",
    "\n",
    "    def forward(self, x):\n",
    "        x1 = x[:, :self.p] @ self.U.T\n",
    "        x2 = x[:, self.p:] @ self.V.T\n",
    "        x_out = ((x1 + x2)**2) @ self.W        \n",
    "        return x_out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "32f3992e",
   "metadata": {},
   "outputs": [],
   "source": [
    "seed = 30  # or any integer you like\n",
    "random.seed(seed)\n",
    "np.random.seed(seed)\n",
    "torch.manual_seed(seed)\n",
    "torch.cuda.manual_seed_all(seed)  # if using GPU\n",
    "\n",
    "# Data Generation using the new function\n",
    "p = 20  # Modulus\n",
    "\n",
    "# Get base vector\n",
    "template = generate_fixed_template(p)\n",
    "\n",
    "# Mean center template\n",
    "template -= np.mean(template)\n",
    "\n",
    "# Generate dataset using numpy\n",
    "X, Y = ModularAdditionDataset(p, template)\n",
    "X_train = torch.tensor(X, dtype=torch.float32).view(-1, 2 * p).cuda()\n",
    "Y_train = torch.tensor(Y, dtype=torch.float32).cuda()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "4a08195a",
   "metadata": {},
   "outputs": [],
   "source": [
    "lr = 0.01\n",
    "mom = 0.9\n",
    "init_scale=2e-3\n",
    "hidden_size = 6 * 3\n",
    "\n",
    "costmin_lr = 0.01\n",
    "utilmax_lr = 10"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f7e7336b-5c6e-48af-a357-2b2c877f6168",
   "metadata": {},
   "source": [
    "## Optimization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1035f81c-e877-4655-8640-4e4c3d323af8",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 200000/200000 [05:21<00:00, 621.16it/s]\n"
     ]
    }
   ],
   "source": [
    "def train(model, X_train, Y_train, optimizer, nsteps=100):\n",
    "    model.train()\n",
    "    loss_history = {}\n",
    "    param_history = {}\n",
    "\n",
    "    for t in tqdm(range(nsteps)):\n",
    "        optimizer.zero_grad()\n",
    "        residual = (Y_train - model(X_train))\n",
    "        loss = (residual**2).mean()\n",
    "        loss_history[t] = loss.item()\n",
    "        if loss.item() > 1e5:\n",
    "            assert False\n",
    "        if t % nsteps//100 == 0:\n",
    "            param_history[t] = {\n",
    "                \"U\": model.U.detach().cpu().clone(),\n",
    "                \"V\": model.V.detach().cpu().clone(),\n",
    "\n",
    "                \"W\": model.W.detach().cpu().clone()\n",
    "            }            \n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        \n",
    "    return loss_history, param_history # Return loss history for plotting\n",
    "\n",
    "model = TwoLayerNet(p=p, hidden_size=hidden_size,\n",
    "                    init_scale=init_scale).cuda()\n",
    "\n",
    "optimizer = optim.SGD(model.parameters(), lr=lr, momentum=mom)\n",
    "nsteps = 200_000\n",
    "loss_history, param_history = train(model, X_train, Y_train, optimizer, nsteps=nsteps)\n",
    "\n",
    "with open(\"loss_history.pickle\", \"wb\") as f:\n",
    "    pickle.dump(loss_history, f)\n",
    "\n",
    "with open(\"param_history.pickle\", \"wb\") as f:\n",
    "    pickle.dump(param_history, f)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "062a4393",
   "metadata": {},
   "source": [
    "## AGF"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "fe647614",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(\"loss_history.pickle\", \"rb\") as f:\n",
    "    loss_history = pickle.load(f)\n",
    "with open(\"param_history.pickle\", \"rb\") as f:\n",
    "    param_history = pickle.load(f)\n",
    "\n",
    "if loss_history==None or param_history==None:\n",
    "    print(\"Must first generate true GD trajectory\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "190e63c5",
   "metadata": {},
   "outputs": [],
   "source": [
    "class ModsumSubNetwork(nn.Module):\n",
    "    \n",
    "    def __init__(self, d_in, d_out, init_scale):\n",
    "        super().__init__()\n",
    "        assert d_in%2 == 0\n",
    "        self.p = d_in // 2\n",
    "        self.u = nn.Linear(self.p, 1, bias=False)\n",
    "        self.v = nn.Linear(self.p, 1, bias=False)\n",
    "        self.w = nn.Linear(1, d_out, bias=False)\n",
    "        with torch.no_grad():\n",
    "            self.w.weight.mul_(init_scale)\n",
    "            self.u.weight.mul_(init_scale)\n",
    "            self.v.weight.mul_(init_scale)\n",
    "        self.active = False\n",
    "        self.util_acc = 0\n",
    "        self.c_a = 1/self.get_norm() - 1\n",
    "        \n",
    "        self.normalize()\n",
    "        \n",
    "    def get_norm(self):\n",
    "        sqnorm = lambda x: torch.linalg.norm(x.weight)**2\n",
    "        norm = torch.sqrt(sqnorm(self.w) + sqnorm(self.u) + sqnorm(self.v))\n",
    "        return norm\n",
    "    \n",
    "    def reinitialize(self, u, v, w):\n",
    "        with torch.no_grad():\n",
    "            self.u.weight.copy_(u)\n",
    "            self.v.weight.copy_(v)\n",
    "            self.w.weight.copy_(w)\n",
    "        self.c_a = 1/self.get_norm() - 1\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x1 = x[:, :self.p]\n",
    "        x2 = x[:, self.p:]\n",
    "        return self.w((self.u(x1) + self.v(x2))**2)\n",
    "    \n",
    "    def normalize(self):\n",
    "        norm = self.get_norm()\n",
    "        with torch.no_grad():\n",
    "            self.w.weight.div_(norm)\n",
    "            self.u.weight.div_(norm)\n",
    "            self.v.weight.div_(norm)\n",
    "    \n",
    "    def utility_step(self, x, residual, learning_rate):\n",
    "        f_i = self(x)\n",
    "        util = torch.einsum('nd,nd->n',  f_i, residual).mean()\n",
    "        self.util_acc += 3 * learning_rate * util.item()\n",
    "        norm_th = 1/(1 + self.c_a - self.util_acc)\n",
    "        \n",
    "        util.backward()\n",
    "        with torch.no_grad():\n",
    "            self.w.weight += norm_th * learning_rate * self.w.weight.grad\n",
    "            self.u.weight += norm_th * learning_rate * self.u.weight.grad\n",
    "            self.v.weight += norm_th * learning_rate * self.v.weight.grad\n",
    "            self.w.weight.grad.zero_()\n",
    "            self.u.weight.grad.zero_()\n",
    "            self.v.weight.grad.zero_()\n",
    "            self.normalize()\n",
    "\n",
    "\n",
    "class ModsumNetwork(nn.Module):\n",
    "    \n",
    "    def __init__(self, d_in, d_out, init_scale, width=100):\n",
    "        super().__init__()\n",
    "        neurons = [ModsumSubNetwork(d_in, d_out, init_scale) for _ in range(width)]\n",
    "        self.neurons = nn.ModuleList(neurons)\n",
    "        self.set_mode(\"utilmax\")\n",
    "    \n",
    "    def load_init(self, U, V, W):\n",
    "        for i, n in enumerate(self.neurons):\n",
    "            u, v, w = U[i], V[i], W[i][:, None]\n",
    "            n.reinitialize(u, v, w)\n",
    "            \n",
    "    def set_mode(self, mode):\n",
    "        if mode not in [\"utilmax\", \"costmin\"]:\n",
    "            raise ValueError(\"mode must be utilmax or costmin\")\n",
    "        self.mode = mode\n",
    "        for neuron in self.neurons:\n",
    "            grad_on = (mode==\"utilmax\") ^ neuron.active\n",
    "            for param in neuron.parameters():\n",
    "                param.requires_grad = grad_on\n",
    "    \n",
    "    def forward(self, x):\n",
    "        if not np.any([n.active for n in self.neurons]):\n",
    "            return 0\n",
    "        else:\n",
    "            outputs = torch.stack([neuron(x) for neuron in self.neurons if neuron.active], dim=0)\n",
    "            return torch.sum(outputs, dim=0)\n",
    "\n",
    "\n",
    "def train_agf(X_train, Y_train, init_sz=1e-3, agf_steps=5, from_init=None,\n",
    "              utilmax_lr=1, costmin_lr=1, costmin_maxiter=2_000, costmin_grad_thresh=1e-1):\n",
    "    d_in, d_out = X_train.shape[-1], Y_train.shape[-1]\n",
    "    if from_init:\n",
    "        U, V, W = from_init[\"U\"], from_init[\"V\"], from_init[\"W\"]\n",
    "        assert d_in == U.shape[1]*2\n",
    "        assert d_out == W.shape[1]\n",
    "        width = U.shape[0]\n",
    "        net = ModsumNetwork(d_in, d_out, init_sz, width=width).cuda()\n",
    "        net.load_init(U, V, W)\n",
    "    else:\n",
    "        net = ModsumNetwork(d_in, d_out, init_sz, width=agf_steps).cuda()\n",
    "    X_train.requires_grad = False\n",
    "    \n",
    "    def update_results(results, t):\n",
    "        results[\"t\"].append(t)\n",
    "        residual = (Y_train - net(X_train))\n",
    "        residual = residual.detach()\n",
    "        results[\"residuals\"].append(residual)\n",
    "        loss = (residual**2).mean().item()\n",
    "        results[\"losses\"].append(loss)\n",
    "        results[\"models\"].append(net.state_dict())\n",
    "        \n",
    "    results = {\n",
    "        \"t\": [],\n",
    "        \"residuals\": [],\n",
    "        \"losses\": [],\n",
    "        \"models\": [],\n",
    "    }\n",
    "    t = 0\n",
    "    update_results(results, t)\n",
    "    for _ in tqdm(range(agf_steps)):\n",
    "        residual = (1/d_out) * 2*(Y_train - net(X_train))\n",
    "        residual = residual.detach()\n",
    "        iters = 0\n",
    "        while net.mode == \"utilmax\":\n",
    "            for n in net.neurons:\n",
    "                if n.active:\n",
    "                    continue\n",
    "                n.utility_step(X_train, residual, utilmax_lr)\n",
    "                if n.util_acc > n.c_a:\n",
    "                    n.active = True\n",
    "                    net.set_mode(\"costmin\")\n",
    "                    break\n",
    "            iters += 1\n",
    "        t += iters\n",
    "            \n",
    "        optimizer = torch.optim.SGD(net.parameters(), lr=costmin_lr, momentum=0.9)\n",
    "        iters = 0\n",
    "        while net.mode == \"costmin\":\n",
    "            optimizer.zero_grad(set_to_none=False)\n",
    "            residual = (Y_train - net(X_train))\n",
    "            loss = (residual**2).mean()\n",
    "            loss.backward()\n",
    "            if iters%100 == 0:\n",
    "                with torch.no_grad():\n",
    "                    grad_norm = np.sqrt(sum([torch.norm(p.grad)**2 for p in net.parameters()]).item())\n",
    "                rel_grad = grad_norm/loss.item()\n",
    "                if rel_grad < costmin_grad_thresh:\n",
    "                    net.set_mode(\"utilmax\")\n",
    "                elif rel_grad*costmin_lr > 1:\n",
    "                    raise ValueError\n",
    "            if iters > costmin_maxiter:\n",
    "                net.set_mode(\"utilmax\")\n",
    "            optimizer.step()\n",
    "            iters += 1\n",
    "        update_results(results, t)\n",
    "        print(f\"loss: {loss.item():.5f}\")\n",
    "    \n",
    "    return results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fdf989d0",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  9%|▉         | 1/11 [00:04<00:44,  4.41s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.59962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 18%|█▊        | 2/11 [00:07<00:30,  3.39s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.41270\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 27%|██▋       | 3/11 [00:09<00:23,  2.92s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.28552\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|███▋      | 4/11 [00:14<00:27,  3.93s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.24469\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|████▌     | 5/11 [00:20<00:27,  4.52s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "loss: 0.15730\n"
     ]
    }
   ],
   "source": [
    "results = train_agf(X_train, Y_train, init_sz=init_scale, agf_steps=11, from_init=param_history[0],\n",
    "                    utilmax_lr=utilmax_lr, costmin_lr=costmin_lr, costmin_grad_thresh=3.8e-2)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3ed32d33",
   "metadata": {},
   "source": [
    "## Plots"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "6d2ef745",
   "metadata": {},
   "outputs": [],
   "source": [
    "def smart_tick_formatter(x, pos):\n",
    "    abs_x = abs(x)\n",
    "\n",
    "    if x == 0:\n",
    "        return \"0\"\n",
    "    elif abs_x >= 1e4 or abs_x < 1e-3:\n",
    "        return f\"{x:.1e}\"  # scientific notation\n",
    "    elif x == int(x):\n",
    "        return f\"{x:.0f}\"\n",
    "    elif 1 <= x and x <= 10:\n",
    "        return f\"{x:.1f}\"\n",
    "    else:\n",
    "        return f\"{x:.2f}\"\n",
    "\n",
    "def style_axes(ax, numyticks=5, numxticks=5):\n",
    "    # formatter = FuncFormatter(smart_tick_formatter)\n",
    "\n",
    "    ax.tick_params(axis=\"y\", which=\"both\", bottom=True, top=False,\n",
    "                   labelbottom=True, left=True, right=False,\n",
    "                   labelleft=True, direction='out', length=7, width=1.5, pad=8, labelsize=24)\n",
    "    # ax.yaxis.set_major_formatter(formatter)\n",
    "\n",
    "    ax.tick_params(axis=\"x\", which=\"both\", bottom=True, top=False,\n",
    "                   labelbottom=True, left=True, right=False,\n",
    "                   labelleft=True, direction='out', length=7, width=1.5, pad=8, labelsize=24)\n",
    "    # ax.xaxis.set_major_formatter(formatter)\n",
    "\n",
    "\n",
    "    if ax.get_yscale() == 'linear':\n",
    "        ax.ticklabel_format(style='sci', axis='y', scilimits=(-2, 2))\n",
    "    \n",
    "    if ax.get_xscale() == 'linear':\n",
    "        ax.ticklabel_format(style='sci', axis='x', scilimits=(-2, 2))\n",
    "\n",
    "    ax.xaxis.offsetText.set_fontsize(20)\n",
    "    ax.grid()\n",
    "\n",
    "    for spine in [\"top\", \"right\"]:\n",
    "        ax.spines[spine].set_visible(False)\n",
    "    for spine in [\"left\", \"bottom\"]:\n",
    "        ax.spines[spine].set_linewidth(3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "cdf80908",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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g67rtufSZYqbmFLVnoV6e7RhALHzonkpf9nEi9vncogYv4gxpxtz5I7rTtPOnTvR2dmJv/71r+jr60N1dbW57Gc/+5n5/Pzzz7ct39HRAQCWckSJKC0uynQIMdzElKr2LNRbVhSyfwMnuUsJL26fyyFi/CLGnG7MkTuuu5pvfvObAQChUAiXX345WltbMT4+jltuuQV//OMfIUkSdu3ahaamppiyuq5j7969kCSJk1sSpZhfdugc8fI5IqKEuB5pet/73oevfOUr6OnpwZNPPolNmzbFvOfGG2+0LfvUU09hfHwckiThjDPOcBsKEcXBeZqIiNxxPdJUUFCA//mf/0FhYaE5y/fCfwDwzne+E//f//f/2Zb97//+b/O50+E7IichVTX/8wo3MaWqPQt1+iT77hFv2JsaXtw+l0PE+EWMOd2YI3eSctOps88+G/v378c3v/lNPPPMMxgbG8P69evx93//9/jABz5gW2ZwcBD33nsvAKCwsNDxRHEiJ/uPtJiXzp6yvTnT4QBwF1Oq2rNQb++k/dQCnHIgNby4fS6HiPGLGHO6MUfuJO1OnfX19bj99tsTfn95ebl55RwRpZ7icE6TwQN0REQJ4e3NKa54HdtMd3pLigqhqhoURc5oHJHcxJSq9izUawQ02+USr55LCS9un8shYvwixpxuzJE77DRRXIWFhZkOwVHTurWZDiGGm5hS1Z6Fekem5gAciH0DO00p4cXtczlEjF/EmNONOXInrZ2mv/3tb3j44YfR19eHyspK7N69G2eddVY6QyBatQpy7L/uPA+ciCgxrjtN4+Pj+OhHPwoA2LVrFz71qU/Zvu+GG27Ad7/7Xctr//Zv/4b3ve99+PGPfwxZ5lChF01MTDguCwaDtvNvkTf5ZR9y/T7MhKwnfvPqOSKixLjuND3++OO49957IUkSLrroItv3/PCHP8R3vvMd22X33nsvSkpKHJdTZjndiBng/QJFVJjjx0xo1vKazj4TEVFCXHeafv/73wMAZFnGW9/61pjlmqbhS1/6EoDwCacnnXQSLr74YnR0dOBXv/oVDMPAf/zHf+AjH/kItm7d6jYcWkVa2rugahoUWcbGhvpMhwPAXUypak9kvUW5CgYmojtN7DWlghe3z+UQMX4RY0435sgd152mF154AQCwbds2lJaWxix/9NFH0d3dDUmScP755+P/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n5nsupNRn9s6uJ/yrug8x/vhzxPBiYiIiBLAThMRERFRAthpIiIiIkoAO01ERERECWCniYiIiCgB7DQRERERJYDzNBERERElgCNNRERERAlgp4mIiIgoAew0ERERESWAnSYiIiKiBLDTRERERJQAdpqIiIiIEsBOE1GSdXZ2ori4GJIkQZIktLW1ZTokIlrlvvSlL5n7JKf/brrppkyH6XlKpgMgyjbXXnstxsfHMx0GEVGMqqoqbN682XZZY2NjeoMREDtNREl0991343//93/xjne8Aw888ECmwyEisrj00kvxk5/8JNNhCIuH54iSpKenBzfeeCMaGxvx5S9/OdPhEBFRknGkiShJrrvuOoyMjODnP/85CgoKMh0OERElGUeayNM0TcPLL7+MH/3oR7juuutw+umnIxAImCcuXnDBBSuue25uDvfccw92796NhoYG5Obmora2Fueccw6++c1vYmBgIOG67rvvPvzmN7/B+973PlxyySUrjomIxCTKvuqll17Ce9/7Xlx44YV4y1vegptvvhkvvPDCimNbdQwij3rwwQeN/Px8A4Djf+eff/6K6j548KBx6qmnxq27qqrK+O1vf7tkXf39/UZFRYVRUVFhBINBwzAMo7W11ayntbV1RTESkRhE2Fd98YtfjFvH1VdfbUxPT68oxtWEI03kWSMjI5iamkp6vV1dXbjooovw4osvAgAkScL555+PD3zgA3jzm9+MvLw8AEB/fz/e9ra34bHHHotb38c+9jEMDAzg9ttvR0VFRdLjJSJvE2FfVVNTg0996lN49tln0dfXh5mZGRw8eBCf/exnIcsyfvKTn+CDH/xg0tuQdTLdayNycvfddxsAjOrqauNNb3qT8S//8i/Gnj17jE984hOufr2dd955ZvmGhgbjpZdesiwPBoPGRRddZL6nrKzMGB4etq3rwQcfNAAYr3/96y2vc6SJaPUQYV8Vz3333WfW8eyzzy67/GrCThN5Vk9Pj9He3h7zeuQw83J3RL/97W/NsoFAwHj55Zdt3zcxMWFs2LDBfO/NN98c856hoSGjpqbGKCgoiOkYsdNEtHp4fV+ViDPPPNMAYNx4440rKr9a8PAceVZNTQ3Wr1+f1DrvuOMO8/lVV12FnTt32r6voKDAMm3Af/7nf0JVVct7Pv3pT6O3txf/+q//yknhiFYxr++rEnHuuecCAI4ePbrssqsJO020akxMTFiO+b///e+P+/53vvOdKCoqAgAMDQ3hySeftCx//vnnAQBf/epXUVNTY/nvjDPOMN93xhlnoKamBl/4wheS1RQiymLJ3lclIhAIAABCodCyy64m7DTRqvHss89idnYWQPjXWWTHxk5OTg7OOuss89+PP/647fuCwSD6+vos/0VeAjwwMIC+vj6MjY0loRVElO1Sta+KZ9++fQCAdevWLbvsasJOE60aBw8eNJ/v3LkTirL03K6nnXaabXkA2Lt3L4zweYEx/7W2tprva21thWEYuP322903goiyXrL3VUvZu3cv/u///g8A8PrXv35ZZVcbdppo1Th8+LD5vKGhIaEykecpHDp0KOkxERFFS/a+av/+/bj22mvNqQsi/fa3v8Wll14KTdNw6qmn4u1vf/sKo14deBsVWjUGBwfN59XV1QmVqampMZ8PDQ0lPSYiomjJ3leFQiHcdddduOuuu1BaWoqmpib4/X60trYiGAwCAHbt2oWHHnoIPh/HUuJhp4lWjYmJCfP5wqRwS4l8X2R5IqJUSfa+qrGxEf/6r/+KP/3pTzh48CBaWlowPT2N0tJSvO51r8O73vUuXHXVVcjJyUlOA7IYO020aszMzJjPF64UWUrkTmR6ejrhz2psbIRhGIkHR0Q0L9n7qtLSUvzTP/1TcoJb5TgOR6tGbm6u+Xxubi6hMgtXsACJ/+IjInKD+yrvYqeJVo3CwkLzeaKjRpHviyxPRJQq3Fd5FztNtGqUl5ebz/v6+hIq09vbaz4vKytLekxERNG4r/Iudppo1Whubjaft7e3J1Smo6PDfL5169akx0REFI37Ku9ip4lWjW3btpnP9+3bl9D9mV544QXb8kREqcJ9lXex00SrxjnnnGNeYTI5OWneO87J7Ows/vSnP5n/vvDCC1MaHxERwH2Vl7HTRKtGYWEhLrroIvPfP/nJT+K+/4EHHsD4+DgAYM2aNTjvvPNSGR4REQDuq7yMnSZaVT7ykY+Yz++++27s37/f9n1TU1P4whe+YP77Qx/6UEL3fyIiSgbuq7yJnSZaVd74xjfiNa95DYDw/CdvetObzLt7LxgcHMTb3vY2HDt2DED4SpTPfvazaY+ViFYv7qu8STI4bTF52O7du9Hd3W15rbe317wMt6CgAJs2bYopt2fPHtTV1dnW2dXVhTPPPBM9PT0AAJ/Ph/PPPx8bNmxAMBjEo48+iqmpKQCAoij43//9X8tQORFRNO6rVgd2msjTGhsbE77kNlJraysaGxsdlx86dAhXXHEF9u7d6/ieyspK3H333XjjG9+47M8notWF+6rVgQc+aVXaunUr/vznP+P+++/Hfffdh/3796Ovrw+lpaXYsGED3v72t+Oaa65BRUVFpkMlolWM+ypv4UgTERERUQJ4IjgRERFRAthpIiIiIkoAO01ERERECWCniYiIiCgB7DQRERERJYCdJiIiIqIEsNNERERElAB2moiIiIgSwE4TERERUQLYaSIiIiJKADtNRERERAlgp4mIiIgoAew0ERERESWAnSYiIsIFF1wASZIgSRL+8Ic/ZDocIk9ip4mIskLkH/3l/nf11VdnOnwiEgA7TUREREQJUDIdABFRsp1xxhk488wzE37/WWedlcJoiChbsNNERFln9+7d+NKXvpTpMIgoy/DwHBEREVEC2GkiIiIiSgA7TURENhobG82r69ra2gAAhw8fxo033ogdO3agpKQExcXF2LlzJz73uc+hu7t7WfUPDAzglltuwfnnn4/a2lrk5OSgoqICp556Kj796U/jwIEDy465r68Pt956Ky6++GKsX78eeXl5yMvLw/r163HppZfi1ltvNduSiKGhIXz961/HGWecgYqKCuTl5WHDhg34wAc+gFdeeSWhOkKhEO6991684x3vwIYNG1BYWIicnBzU1dVh165deOMb34jbbrst4fqIMsogIsoC559/vgHAAGB88YtfdF1fQ0ODWV9ra6tx1113GTk5OeZr0f+VlJQY//M//5NQ3T/60Y+MkpISx7oAGLIsGzfccIOhquqS9WmaZvzLv/yLkZ+fH7dOAIbP5zP2798fU0dk/p544gnj6aefNtauXRs3vh/84Adx4zp8+LCxbdu2JWNa+O/o0aMJ5Y8oU3giOBHREh566CF84hOfAADU1tbivPPOQ2FhIY4ePYpnnnkGmqZhdHQUV1xxBQKBAN7ylrc41vXNb34Tn/70p81/5+Tk4Pzzz8f69esxPDyMJ554AkNDQ9A0Dbfffjva29vxq1/9CpIk2danaRre9a534cEHHzRfCwQCOPvss9HY2AhFUdDb24sXXngBPT090HUdc3Nzcdv7yiuv4Oabb8bExASqqqrwmte8BuXl5Thx4gQef/xxTE9PQ9M0fPjDH8ZJJ52Es88+O6aO8fFxvO51r0NnZycAwOfz4dRTT8W2bdtQWFiIqakpnDhxAi+99BIGBgbixkPkGZnutRERJUMqR5oCgYDh8/mMb3zjG4amaZb3HTp0yDj55JPN91ZUVBj9/f22dT777LOGLMvme9/whjcYPT09lvfMzMwYn/70py0jMLfddptjnJ/97Gct7/3Yxz5mDA4O2r73z3/+s3HllVcar7zySsyyyPzl5OQYsiwbt912mxEKhSzv6+joME466STzva997WttP+vf//3fzfds377dOHTokO37dF03/vKXvxjXXXed0dHR4dhOIi+QDMMw0ttNIyJKvgsuuAB//OMfASx/nqYvf/nLKCsrs7zW2NiI9vZ289+33HILPvvZz9qWDwaDOPnkk9HT0wMA+PSnP41bb7015n3nn38+nnzySQDhuaH++Mc/IhAI2Nb5iU98At/5zncAAMXFxejq6kJRUZHlPUeOHMG2bdug6zoA4Gtf+xpuuummRJocIzJ/APCf//mfuPbaa23f+8orr2DXrl0wDAOSJOHEiROora21vOed73wnfvWrXwEAHnnkEbzuda9bUVxEnpLhThsRUVJEjpQs97/W1taY+iJHmjZs2BAz4hLtrrvusow2Rb//wIEDls/829/+Fre+iYkJo6Kiwnz/97///Zj3fPjDHzaXn3XWWYau60snykFk/nbu3Lnk+88880zz/Q8//HDM8osvvthcvnfv3hXHReQlvHqOiGgJ73nPe6Ao8U8Bvfzyy5GTkwMgfGVc9NVgTzzxhPn85JNPxmmnnRa3voKCAlxxxRW25Rf87//+r/n8Yx/7mON5T8v1rne9a8n3nHrqqeZzuyvy1q9fbz6/8847kxIXUaax00REWeeLX/wiDMNI+L/Gxsa49SVym5XCwkKcdNJJ5r9ffPFFy/LIf5977rkJtSPyfS+88IJlWV9fn6Wz8trXvjahOhOxc+fOJd9TXl5uPh8dHY1Z/vd///fm8x/84Ac47bTT8O1vfxuHDh1KTpBEGcBOExHREiJHTeJZt26d+TwYDFqWRf67oaEhofoiO3PRV5j19fWZzxfmPUqWkpKSJd/j9/vN56FQKGb5xRdfjE9+8pPmv1988UXccMMN2LZtGyorK/G2t70Nt99+u3l1HZEI2GkiIlpCfn5+Qu8rKCgwn4+Pj1uWTUxM2L5vpfVF/ruwsDCh+hKVrMN83/rWt/DQQw/hnHPOsbw+MDCA3/zmN/jkJz+JhoYGXHbZZZaT7om8ip0mIqIlTE1NJfS+yclJ83n0lW6RHZvI9620vsh/R3bIvObNb34znnnmGXR1deHee+/Fhz70IWzfvt1cbhgGHnjgAbzqVa/CkSNHMhgp0dLYaSIiWkJHR8ey31dRUWFZVllZuez6Ikdfouurrq42n8/OzprTHXjV2rVr8d73vhff//73sX//fnR2duIrX/mKOZo2ODiIG2+8McNREsXHThMR0RKee+65Jd8zMTFhuWIu+uq4yKvNnn322YQ+95lnnnGsr7q62nLO0+OPP55QnV5RX1+Pf/7nf8Zdd91lvvb73/8es7OzGYyKKD52moiIlnDfffdBVdUl37Nwe5KKigrLlXQAcOGFF5rPX3zxRbz00ktx65uensb9999vW37BpZdeaj6/4447YAg4V/Gb3vQm83koFMLQ0FAGoyGKj50mIqIlHD9+HLfddpvj8oGBAXzxi180/3311VfHzOu0detWnHfeeea/r7/+eturzhZ8/vOfR39/P4DwjODvec97Yt5zww03wOcL78afe+45fP3rX0+sQWmQ6P3kIg9V+ny+mJnZibyEnSYioiUEAgF87nOfw2233WbesmTBkSNHcPHFF5vnFJWXl1tuyBvplltugSzLAICnnnoKl112mdkxWjA3N4d/+qd/snTSvvjFL9peIbdlyxb84z/+o/nvm2++Gddff73jaM1f//pXXH311di/f38CrXbn7LPPxhVXXIE9e/Y43iD44MGDuPLKK81/X3TRReYEoUReFH+KWyIiAe3ZsyfhkQ4gPKWA3b3iFtx666244YYb8KlPfQrf+ta3cN5556GwsBBHjx79/9u7Y5DUojiO47/XlcLA2yZBWOZqRGvREjWG0lJ7griIzi1OSUNjQ2VDo80ODQU2FIggtJmSuIQ0SG4uJadNnq/38vB6ET2+n/Ue/vd/z/Tj3HvP0fX1tXq9niTJcRzlcjn5/f7f1llcXNTe3l4/VBUKBU1PT2tlZUWBQECdTkdXV1cDvW9sbAzsd/SrbDaru7s7FQoFSdLBwYGOj4+1tLSkYDAoj8ejx8dHVSqVfrBLp9PWc/O3np+flc/nlc/n5fV6NT8/r1AoJNd11el01Gg0VKlU+uO9Xq/29/c/vS/gQ77k8BYA+Mc+cvbcxMTEm3o/nz3XbDbN0dGRGR0d/WMN13XN2dmZVa8nJyfGdd13e3Icx6RSKfPy8jK0Xq/XMzs7O2ZsbGzoszqOY6rV6rvzVywWh94zk8n0x2cymTfX5+bmrOd/dnbW3Nzc2Ewd8KVYaQIAC/F4XMvLyzo8PNTl5aUeHh5kjNHMzIzW19eVTCY1NTVlVSsWiykajSqXy+n8/Fz1el1PT0/y+XwKBAJaW1vT9vb2wH5G7xkZGdHu7q4SiYROT091cXGh+/t7tdtteTwe+f1+hcNhra6uamtry7rPj7i9vVWpVFKxWFS5XFatVlOr1VK329X4+LgmJye1sLCgSCSizc1NXsvhW/hhzDf83QIAPlkwGOzvk9RsNoeeTwfg/8eH4AAAABYITQAAABYITQAAABYITQAAABYITQAAABYITQAAABbYcgAAAMACK00AAAAWCE0AAAAWCE0AAAAWCE0AAAAWCE0AAAAWCE0AAAAWCE0AAAAWCE0AAAAWCE0AAAAWXgGGGYrfpVidkQAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 600x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def get_power(points):\n",
    "    p = len(points)\n",
    "    num_coefficients = (p // 2) + 1\n",
    "    \n",
    "    # Perform FFT and calculate power spectrum\n",
    "    ft = np.fft.fft(points) # Could consider using np.fft.rfft which is designed for real valued input.\n",
    "    power = np.abs(ft[:num_coefficients])**2 / p\n",
    "    \n",
    "    # Double power for frequencies strictly between 0 and Nyquist (Nyquist is not doubled if p is even)\n",
    "    if p % 2 == 0:  # p is even, Nyquist frequency at index num_coefficients - 1\n",
    "        power[1:num_coefficients - 1] *= 2\n",
    "    else:  # p is odd, no Nyquist frequency\n",
    "        power[1:] *= 2\n",
    "\n",
    "    # Confirm the power sum approximates the squared norm of points\n",
    "    total_power = np.sum(power)\n",
    "    norm_squared = np.linalg.norm(points)**2\n",
    "    if not np.isclose(total_power, norm_squared, rtol=1e-3):\n",
    "        print(f\"Warning: Total power {total_power:.3f} does not match norm squared {norm_squared:.3f}\")\n",
    "\n",
    "    return np.arange(num_coefficients), power\n",
    "\n",
    "fig, ax = plt.subplots(1, 1, figsize=(6, 6))\n",
    "ax.plot(list(loss_history.values()), lw=4)\n",
    "\n",
    "for lossval in results[\"losses\"]:\n",
    "    ax.axhline(lossval, alpha=0.3, ls=\":\", color=\"xkcd:slate\", zorder=-4, lw=2)\n",
    "\n",
    "f = utilmax_lr / (lr/(1-mom))\n",
    "for t in results[\"t\"]:\n",
    "    ax.axvline(f*t, alpha=0.3, ls=\":\", color=\"xkcd:slate\", zorder=-4, lw=2)\n",
    "\n",
    "ax.step(f*np.array(results[\"t\"]), results[\"losses\"], where=\"post\",\n",
    "        lw=2, color=\"xkcd:tangerine\")\n",
    "\n",
    "# === Compute power spectrum of template ===\n",
    "freq, power = get_power(template)\n",
    "valid = power > 1e-20\n",
    "freq, power = freq[valid], power[valid]\n",
    "sorted_idx = np.argsort(-power)\n",
    "freq, power = freq[sorted_idx], power[sorted_idx]\n",
    "\n",
    "alpha_values = [np.sum(power[k:]) for k in range(len(power))]\n",
    "coef = 1 / p\n",
    "for k, alpha in enumerate(alpha_values):\n",
    "    ax.axhline(y=coef * alpha, color='black', linestyle='--', linewidth=2, zorder=-2)\n",
    "\n",
    "ax.set_xscale(\"log\")\n",
    "ax.set_yscale(\"log\")\n",
    "ax.set_xlim(7e3, 2e5)\n",
    "ax.set_xlabel('Epochs', fontsize=24)\n",
    "ax.set_ylabel('Train Loss', fontsize=24)\n",
    "\n",
    "style_axes(ax)\n",
    "plt.tight_layout()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "c436b2a6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([     0.,   8800.,  11700.,  13200.,  15600.,  16400.,  32200.,\n",
       "        46600.,  66000., 116500.])"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "f*np.array(results[\"t\"])"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "win-research",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
