{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "cbb36c4a-28f7-4ebc-a024-b8c2042184c4",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "True"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import math\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torch.optim as optim\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "torch.set_default_dtype(torch.float64)\n",
    "torch.cuda.is_available()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "e7701f0b-f2e1-4938-9f14-31af68c84875",
   "metadata": {},
   "outputs": [],
   "source": [
    "class DeepNetwork(nn.Module):\n",
    "    def __init__(self, layer_sizes, init_scale=1.0):\n",
    "        \"\"\"\n",
    "        Initialize the deep linear network with scaled-down weight initialization.\n",
    "\n",
    "        Parameters:\n",
    "        layer_sizes (list of int): A list containing the sizes of each layer in the network.\n",
    "                                   For example, [input_size, hidden1_size, hidden2_size, ..., output_size].\n",
    "        init_scale (float): A scaling factor for the initial weights of each layer. Default is 1.0 (no scaling).\n",
    "        \"\"\"\n",
    "        super(DeepNetwork, self).__init__()\n",
    "\n",
    "        # Create a list to hold the layers\n",
    "        self.layers = nn.ModuleList()\n",
    "\n",
    "        # Iterate through the layer sizes and create the linear layers\n",
    "        for i in range(len(layer_sizes) - 1):\n",
    "            layer = nn.Linear(layer_sizes[i], layer_sizes[i + 1], bias=False)\n",
    "            \n",
    "            # Scale down the weights of the layer by the specified factor\n",
    "            with torch.no_grad():\n",
    "                layer.weight *= init_scale\n",
    "                if layer.bias is not None:\n",
    "                    layer.bias *= init_scale\n",
    "            \n",
    "            self.layers.append(layer)\n",
    "\n",
    "    def forward(self, x):\n",
    "        \"\"\"\n",
    "        Forward pass through the network.\n",
    "\n",
    "        Parameters:\n",
    "        x (torch.Tensor): Input tensor to the network.\n",
    "\n",
    "        Returns:\n",
    "        torch.Tensor: Output tensor after passing through the network.\n",
    "        \"\"\"\n",
    "        layer_inputs = [x]  # To store the input to each layer\n",
    "        for layer in self.layers:\n",
    "            x = F.relu(layer(x))\n",
    "            layer_inputs.append(x)  # Append the input to the next layer\n",
    "        return x, layer_inputs\n",
    "\n",
    "class MuPDeepNetwork(nn.Module):\n",
    "    def __init__(self, layer_sizes, init_scale=1.0):\n",
    "        \"\"\"\n",
    "        Initialize the deep linear network using μP parameterization.\n",
    "\n",
    "        Parameters:\n",
    "        layer_sizes (list of int): A list containing the sizes of each layer in the network.\n",
    "                                   For example, [input_size, hidden1_size, hidden2_size, ..., output_size].\n",
    "        init_scale (float): A scaling factor for the initialization of weights. Default is 1.0.\n",
    "        \"\"\"\n",
    "        super(MuPDeepNetwork, self).__init__()\n",
    "\n",
    "        # Create a list to hold the layers\n",
    "        self.layers = nn.ModuleList()\n",
    "\n",
    "        # Iterate through the layer sizes and create the linear layers using μP parameterization\n",
    "        for i in range(len(layer_sizes) - 1):\n",
    "            layer = nn.Linear(layer_sizes[i], layer_sizes[i + 1], bias=False)\n",
    "            \n",
    "            # μP initialization: Scale the weights by a factor related to the fan-in\n",
    "            fan_in = layer_sizes[i]\n",
    "            with torch.no_grad():\n",
    "                layer.weight.data.normal_(0, init_scale / math.sqrt(fan_in))  # μP scaling of weights\n",
    "                if layer.bias is not None:\n",
    "                    layer.bias.data.zero_()  # μP recommends no scaling for biases\n",
    "            \n",
    "            self.layers.append(layer)\n",
    "\n",
    "    def forward(self, x):\n",
    "        \"\"\"\n",
    "        Forward pass through the network.\n",
    "\n",
    "        Parameters:\n",
    "        x (torch.Tensor): Input tensor to the network.\n",
    "\n",
    "        Returns:\n",
    "        torch.Tensor, list of torch.Tensor: Output tensor and a list of layer inputs.\n",
    "        \"\"\"\n",
    "        layer_inputs = [x]  # To store the input to each layer\n",
    "        for layer in self.layers:\n",
    "            x = F.relu(layer(x))\n",
    "            layer_inputs.append(x)  # Append the input to the next layer\n",
    "        return x, layer_inputs\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "ae90c815-30b5-42bd-8a54-cbe15a8fc730",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def compute_jacobians(model, x):\n",
    "    \"\"\"\n",
    "    Compute the Jacobian of the network's output with respect to the input to each layer.\n",
    "\n",
    "    Parameters:\n",
    "    model (DeepLinearNetwork): The deep linear network.\n",
    "    x (torch.Tensor): Input tensor to the network.\n",
    "\n",
    "    Returns:\n",
    "    list of torch.Tensor: A list containing the Jacobians of the output with respect to the input of each layer.\n",
    "    \"\"\"\n",
    "    # Enable gradient tracking on the input\n",
    "    optimizer.zero_grad()\n",
    "    x.requires_grad_(True)\n",
    "\n",
    "    # Perform the forward pass and get the layer inputs\n",
    "    output, layer_inputs = model(x)\n",
    "\n",
    "    jacobians = []\n",
    "\n",
    "    # Compute the Jacobian for each layer's input\n",
    "    for i, layer_input in enumerate(layer_inputs):\n",
    "        # Compute the Jacobian of the output with respect to the current layer's input\n",
    "        jacobian = []\n",
    "        for j in range(output.shape[1]):\n",
    "            grad_outputs = torch.zeros_like(output)\n",
    "            grad_outputs[:, j] = 1.0  # Set the gradient of the desired output component to 1\n",
    "            jac = torch.autograd.grad(\n",
    "                outputs=output,\n",
    "                inputs=layer_input,\n",
    "                grad_outputs=grad_outputs,\n",
    "                retain_graph=True,\n",
    "                create_graph=True,\n",
    "                allow_unused=True,\n",
    "            )[0]\n",
    "            jacobian.append(jac)\n",
    "        \n",
    "        jacobian = torch.stack(jacobian, dim=1)\n",
    "        jacobians.append(jacobian.detach().cpu())\n",
    "\n",
    "    return jacobians\n",
    "\n",
    "def compute_jacobians_wrt_loss(model, x, y, loss_fn):\n",
    "    \"\"\"\n",
    "    Compute the Jacobian of the network's output with respect to the input to each layer.\n",
    "\n",
    "    Parameters:\n",
    "    model (DeepLinearNetwork): The deep linear network.\n",
    "    x (torch.Tensor): Input tensor to the network.\n",
    "\n",
    "    Returns:\n",
    "    list of torch.Tensor: A list containing the Jacobians of the output with respect to the input of each layer.\n",
    "    \"\"\"\n",
    "    # Enable gradient tracking on the input\n",
    "    optimizer.zero_grad()\n",
    "    x.requires_grad_(True)\n",
    "\n",
    "    # Perform the forward pass and get the layer inputs\n",
    "    output, layer_inputs = model(x)\n",
    "    loss = loss_fn(output, y)\n",
    "    loss.backward()\n",
    "    return x.grad.detach().cpu()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "fdd4c302-3636-481c-8456-1f356acb1012",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def get_all_layer_svs(model):\n",
    "    eigs = []\n",
    "    with torch.no_grad():\n",
    "        for layer in model.layers:\n",
    "            _, S, _ = torch.linalg.svd(layer.weight.data.detach().cpu(), full_matrices=True)\n",
    "            S, _ = torch.sort(S, descending=True)\n",
    "            eigs.append(S)\n",
    "    return eigs\n",
    "\n",
    "def plot_all_layer_svs(eigs, title='', save_fp=None):\n",
    "    plt.close()\n",
    "    plt.clf()\n",
    "    plt.figure(figsize=(5,4))\n",
    "    \n",
    "    for layer_idx in range(len(eigs)):\n",
    "        plt.plot(range(len(eigs[layer_idx])), eigs[layer_idx], label=f'Layer {layer_idx}')\n",
    "    \n",
    "    plt.ylabel(r'Singular Value')\n",
    "    plt.xlabel(r'Singular Value Index')\n",
    "    plt.legend()\n",
    "    plt.xscale('log')\n",
    "    \n",
    "    if title != '':\n",
    "        plt.title(title)\n",
    "        \n",
    "    if save_fp is not None:\n",
    "        plt.savefig(save_fp, bbox_inches='tight')\n",
    "        \n",
    "    # plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "25903a8b-9c72-4e05-b768-f11e4099d5ca",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "input_size = 10\n",
    "output_size = 5\n",
    "num_samples = 5000\n",
    "\n",
    "num_epochs = 10000\n",
    "hidden_width = 512\n",
    "n_h_layers = 6\n",
    "learning_rate = 5e-3\n",
    "weight_decay = 1e-2\n",
    "init_scale = 1.0\n",
    "batch_size = 128\n",
    "\n",
    "log_freq = num_epochs / 10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "id": "58de375e-0c3c-4105-99d9-b63ace076345",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "X = torch.randn(num_samples, input_size)\n",
    "beta = torch.randn(input_size, output_size)\n",
    "y = X @ beta\n",
    "\n",
    "X_test = torch.randn(num_samples, input_size)\n",
    "y_test = X_test @ beta\n",
    "\n",
    "dataset = torch.utils.data.TensorDataset(X, y)\n",
    "dataloader = torch.utils.data.DataLoader(dataset, \n",
    "                                         batch_size=batch_size, \n",
    "                                         shuffle=True, \n",
    "                                         drop_last=False)\n",
    "\n",
    "test_loader = torch.utils.data.DataLoader(torch.utils.data.TensorDataset(X_test, y_test),\n",
    "                                          batch_size=batch_size,\n",
    "                                          shuffle=False,\n",
    "                                          drop_last=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "id": "a3953727-a5a1-4d2e-9378-1c676b5af9fa",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "layer_sizes = [input_size]\n",
    "for _ in range(n_h_layers):\n",
    "    layer_sizes.append(hidden_width)\n",
    "layer_sizes.append(output_size)\n",
    "\n",
    "model = DeepNetwork(layer_sizes, init_scale=init_scale)\n",
    "model = model.to('cuda')\n",
    "\n",
    "criterion = nn.MSELoss()\n",
    "optimizer = optim.SGD(model.parameters(), lr=learning_rate, weight_decay=weight_decay)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "id": "65252833-208b-40ba-aaf9-58c6ef39c3a7",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAb0AAAGMCAYAAABOCHxMAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAABrtklEQVR4nO3deVxU1fvA8c+dGdYZQEBAUFHcN0zRNC01NXEp08xfmZoblkuLfu2raZumFaVpZmVWJm7kkqmlLeZXwaXMxDRLzVxQTMSdXZZhzu8PZHIEZREchef9eo0w95577nOHiw/n3HPu1ZRSCiGEEKIC0Nk7ACGEEOJWkaQnhBCiwpCkJ4QQosKQpCeEEKLCkKQnhBCiwpCkJ4QQosKQpCeEEKLCkKQnhBCiwpCkJ4QQosKQpHeNnTt38sgjjxAYGIiTkxN+fn60adOGF154wabc/fffz/3332+fIAswZcoUNE0r8/088sgjuLi4kJiYeN0yAwYMwMHBgTNnzhS5Xk3TmDJlys0HaGcrVqygcePGuLi4oGkae/fuzVcmJiYGTdN455138q3r1asXmqbxySef5FvXuXNnvL29Kc5NlG7mvFi4cCGaphETE1No2blz57Jw4cIS7edmvfXWW6xdu7bI5TVN49lnny3RvoYMGULNmjWLtP/o6Gg0TSM6OrrY+ylo27L+Hf/iiy+YPXt2gevKy+8nAEpYrV+/Xul0OtWpUye1bNkyFR0drZYtW6ZeeOEFVbVqVZuy+/fvV/v377dTpPlNnjxZ3Yof57p16xSgPvroowLXJyYmKhcXF9W7d+9i1QuoyZMnl0KE9nP27Fnl4OCgevbsqaKjo9WOHTtUWlpavnI5OTnKw8NDde3aNd9yT09PZTQa1eOPP26zLjMzU7m4uKg+ffoUK6aTJ0+qHTt2FP9glFIREREKULt27Sq0bOPGjVWHDh1KtJ+bZTQa1eDBg4tcHlDPPPNMifZ15MgR9dtvvxVp/0lJSWrHjh0qKSmp2PuJiopSgIqKirIuu5mfZVE8+OCDqkaNGgWu27Fjhzp58mSZ7ftWMtgz4d5upk+fTlBQEBs2bMBg+Pej6devH9OnT7cp26hRo1sd3i2Vnp6Oq6trvuXdu3cnICCABQsWMHr06Hzrly1bxuXLlwkLC7sVYd5W/v77b7Kzsxk4cCAdOnS4bjmdTkf79u2JiorCbDZbz7Xff/+dS5cu8d///pclS5bYbLNz504uX75Mx44dixVTtWrVqFatWvEPRhSodu3aRS7r7u7OPffcU2r7tufPsjSPw96ke/MqFy5coHLlyjYJL49OZ/tRXdu9efz4cTRN491332XWrFkEBQVhMplo06YNv/zyS776PvvsM+rVq4eTkxONGjXiiy++yNd1cr3ukbx9FdadtGLFCkJDQ/H398fFxYWGDRsyceJE0tLSbMoNGTIEk8nEH3/8QWhoKG5ubnTu3LnAOvV6PYMHD2b37t388ccf+dZHRETg7+9P9+7dOXfuHKNHj6ZRo0aYTCZ8fX3p1KkT27Ztu2HccP2unLwut+PHj+c71jZt2mA0GjGZTHTt2pU9e/bYlDl27Bj9+vUjICDA2nXduXPnArsgr/XNN9/Qpk0bXF1dcXNzo0uXLuzYscO6fsiQIdx3330APP7442iadsPu744dO5KammrTdRgdHU1AQADDhw/nzJkzHDhwwGZd3nbFOeaCPsfMzExeeOEFqlSpgqurK+3bt2f37t3UrFmTIUOG5Is1JSWFUaNGUblyZby9venTpw/x8fHW9TVr1mT//v1s2bIFTdPQNM16HlssFt544w3q16+Pi4sLlSpVomnTprz//vvX/WwAMjIyeOGFF2jWrBkeHh54eXnRpk0bvv76a5tymqaRlpbGokWLrPsu7mWHvN+zZcuW8fLLLxMQEIC7uzsPPPAAhw4dsil77e/ojfZf0O9vTEwM/fr1o2bNmri4uFCzZk2eeOIJTpw4UWic1/4s834XCnpd/Rl89NFHtG/fHl9fX4xGI8HBwUyfPp3s7Gxrmfvvv59vv/2WEydO2NRz9XFe2735559/0qtXLzw9PXF2dqZZs2YsWrSoxJ/trSJJ7ypt2rRh586dPP/88+zcudPmpCiqjz76iI0bNzJ79mwiIyNJS0ujR48eJCUlWct8+umnPP300zRt2pTVq1fzyiuv8Prrr5eo7/9GDh8+TI8ePfj888/54YcfGDt2LCtXrqRnz575ymZlZfHwww/TqVMnvv76a15//fXr1jts2DA0TWPBggU2yw8cOMCvv/7K4MGD0ev1XLx4EYDJkyfz7bffEhERQa1atbj//vtL9VjfeustnnjiCRo1asTKlStZsmQJKSkptGvXziZx9OjRg927dzN9+nQ2btzIxx9/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XNMTQpQXd3BHWLlUGj+PYk9ZGDt2LJcuXeKXX37BxcWFH374gUWLFlG3bl2++eabmw5ICCHsTa/PvfvKtT1Zwr7S09MBcHBwKHEdxW7pbd68ma+//pq7774bnU5HjRo16NKlC+7u7oSHh/Pggw+WOBghhLgdGAwGXF1dOXfuHA4ODuh0pXf/S1F8SinS09M5e/YslSpVsv5RUhLFTnppaWnWJy54eXlx7tw56tWrR3BwML/99luJAxFCiNuFpmn4+/sTGxvLiRMn7B2OuKJSpUpUqVLlpuoodtKrX78+hw4dombNmjRr1oxPPvmEmjVrMm/ePPz9/W8qGCGEuF04OjpSt25d6eK8TTg4ONxUCy9PsZPe2LFjOX36NACTJ0+ma9euREZG4ujoyMKFC286ICGEuF3odDqZslDO3PQ8vfT0dP766y8CAwOpXLlyacVVJDJPTwghBNyCeXp5XF1dCQkJudlqhBBCiDJXpKQ3bty4Ilc4a9asEgcjhBBClKUiJb09e/YUqbKrb0AthBBC3G6KlPTkJtNCCCHKA5lxKYQQosIo9kCWjh073rAbc/PmzUWuKzw8nNWrV/PXX3/h4uJC27Zteeedd6hfv35xwxJCCCEKVeyWXrNmzbjrrrusr0aNGpGVlcVvv/1GcHBwserasmULzzzzDL/88gsbN27EbDYTGhpKWlpaccMSQgghClVqz9ObMmUKqampvPvuuyWu49y5c/j6+rJlyxbat29faHmZpyeEEAKKng9K7ZrewIEDWbBgwU3VkZSUBOTe07MgmZmZJCcn27yEEEKIoiq1pLdjx46bul2PUopx48Zx33330aRJkwLLhIeH4+HhYX1Vr169xPsTQghR8RR7IEufPn1s3iulOH36NDExMbz66qslDuTZZ59l3759bN++/bplJk2aZDNRPjk5WRKfEEKIIit20vPw8LB5r9PpqF+/PlOnTiU0NLREQTz33HN88803bN26lWrVql23nJOTE05OTiXahxBCCFHspBcREVFqO1dK8dxzz7FmzRqio6MJCgoqtbqFEEKIa930DadvxjPPPMMXX3zB119/jZubGwkJCUBua9LFxcWeoQkhhCiHij1lwdPTs8DJ6Zqm4ezsTJ06dRgyZAhDhw4tfOfXmeQeERHBkCFDCt1epiwIIYSAMny00Guvvcabb75J9+7dadWqFUop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Q5KeEEKICkOSnhBCiApDkp4QQogKQ5KeKHc0TbM++uVWuv/++xk7duwt329RHD9+HE3T2Lt3r71DKZKaNWsye/Zse4chyiFJeuKOcvbsWUaMGEFgYCBOTk5UqVKFrl27smPHDmuZ06dP0717dztGWXrOnDmDg4MDS5cuLXD9iBEjaNq06S2O6vqmTJlCs2bN7B2GENclSU/cUR599FF+//13Fi1axN9//80333zD/fffz8WLF61lqlSpgpOTkx2jLBmlFGaz2WaZn58fDz74IBEREfnKX758meXLlxMWFnarQhTijidJT9wxEhMT2b59O++88w4dO3akRo0atGrVikmTJvHggw9ay13dvZnXrbd69Wo6duyIq6srd911l03LEOCzzz6jevXquLq68sgjjzBr1iwqVapkXT9kyBB69+5ts83YsWO5//77rxvv0qVLadmyJW5ublSpUoX+/ftz9uxZ6/ro6Gg0TWPDhg20bNkSJycntm3blq+esLAwoqKiOH78uM3yVatWkZGRwcCBA/nhhx+47777qFSpEt7e3jz00EMcPXr0urEtXLjQ5vgA1q5dm+8hquvWraNFixY4OztTq1YtXn/99XyJ+UbyPrd3330Xf39/vL29eeaZZ8jOzraWOXv2LD179sTFxYWgoCAiIyPz1ZOUlMTTTz+Nr68v7u7udOrUid9//x2Ac+fOUaVKFd566y1r+Z07d+Lo6MiPP/5Y5FhFxSBJT9wxTCYTJpOJtWvXkpmZWaxtX375Zf773/+yd+9e6tWrxxNPPGH9z/unn35i5MiRjBkzhr1799KlSxfefPPNm443KyuLadOm8fvvv7N27VpiY2MZMmRIvnITJkwgPDycgwcPFthV2aNHD6pUqcLChQttli9YsIDevXvj7e1NWloa48aNY9euXWzatAmdTscjjzxyU0/63rBhAwMHDuT555/nwIEDfPLJJyxcuLDYn01UVBRHjx4lKiqKRYsWsXDhQptjGTJkCMePH2fz5s2sWrWKuXPn2vxxoJTiwQcfJCEhge+++47du3cTEhJC586duXjxIj4+PixYsIApU6YQExNDamoqAwcOZPTo0YSGhpb4+EU5Zd+HPAhRPKtWrVKenp7K2dlZtW3bVk2aNEn9/vvvNmUAtWbNGqWUUrGxsQpQ8+fPt67fv3+/AtTBgweVUko9/vjj6sEHH7SpY8CAAcrDw8P6fvDgwapXr142ZcaMGaM6dOhgfd+hQwc1ZsyY68b+66+/KkClpKQopf591M3atWsLPe4XX3xR1ahRQ1ksFqWUUseOHVOapqkNGzYUWP7s2bMKUH/88YdS6t/PYc+ePUoppSIiImyOTyml1qxZo67+L6Fdu3bqrbfesimzZMkS5e/vf904J0+erO666y7r+8GDB6saNWoos9lsXfZ///d/6vHHH1dKKXXo0CEFqF9++cW6/uDBgwpQ7733nlJKqU2bNil3d3eVkZFhs6/atWurTz75xPp+9OjRql69emrAgAGqSZMm6vLly9eNU1Rc0tITd5RHH32U+Ph4vvnmG7p27Up0dDQhISH5WkHXuroF5e/vD2BtTRw6dIhWrVrZlL/2fUns2bOHXr16UaNGDdzc3KxdoXFxcTblWrZsWWhdYWFhnDhxgs2bNwO5rbxq1arxwAMPAHD06FH69+9PrVq1cHd3JygoqMB9Fcfu3buZOnWqtYVtMpl46qmnOH36NOnp6UWup3Hjxuj1eut7f39/62d/8OBBDAaDzWfQoEEDm67X3bt3k5qaire3t00ssbGxNl247777LmazmZUrVxIZGYmzs3OJj12UX+X6yemifHJ2dqZLly506dKF1157jeHDhzN58uQCuw7zODg4WL/Pu26V1/WnlMp3LUtd85hJnU6Xb9nV16WulZaWRmhoKKGhoSxduhQfHx/i4uLo2rUrWVlZNmWNRuP1D/aKunXr0q5dOyIiIujYsSOLFi1i6NCh6HS5f7f27NmT6tWr89lnnxEQEIDFYqFJkyb59lWc47FYLLz++uv06dMn3/bFSShXf/aQ+/lf/dnnLbsei8WCv78/0dHR+dZdnRyPHTtGfHw8FouFEydO3FajWsXtQ5KeuOM1atTopublNWjQgF9//dVmWUxMjM17Hx8f/vzzT5tle/fuzfcfep6//vqL8+fP8/bbb1O9evUC6yyusLAwRo0aRa9evfjnn38YOnQoABcuXODgwYN88skntGvXDoDt27ffsC4fHx9SUlJIS0uzJt1r5/CFhIRw6NAh6tSpc1Nx30jDhg0xm83ExMRYW9eHDh0iMTHRJo6EhAQMBgM1a9YssJ6srCwGDBjA448/ToMGDQgLC+OPP/7Az8+vzGIXdybp3hR3jAsXLtCpUyeWLl3Kvn37iI2N5csvv2T69On06tWrxPU+99xzfPfdd8yaNYvDhw/zySef8P3339u0Pjp16kRMTAyLFy/m8OHDTJ48OV8SvFpgYCCOjo588MEHHDt2jG+++YZp06aVOEaA//u//8PBwYERI0bQuXNnawLw9PTE29ubTz/9lCNHjrB582bGjRt3w7pat26Nq6srL730EkeOHOGLL77I10X82muvsXjxYqZMmcL+/fs5ePAgK1as4JVXXrmp47ha/fr16datG0899RQ7d+5k9+7dDB8+HBcXF2uZBx54gDZt2tC7d282bNjA8ePH+fnnn3nllVesf0i8/PLLJCUlMWfOHCZMmEDDhg1lKocokCQ9cccwmUy0bt2a9957j/bt29OkSRNeffVVnnrqKT788MMS13vvvfcyb948Zs2axV133cUPP/zAf/7zH5suvK5du/Lqq68yYcIE7r77blJSUhg0aNB16/Tx8WHhwoV8+eWXNGrUiLfffpt33323xDECuLq60q9fPy5dusSwYcOsy3U6HcuXL2f37t00adKE//znP8yYMeOGdXl5ebF06VK+++47goODWbZsGVOmTLEp07VrV9avX8/GjRu5++67ueeee5g1axY1atS4qeO4VkREBNWrV6dDhw706dPHOjUhj6ZpfPfdd7Rv355hw4ZRr149+vXrx/Hjx/Hz8yM6OprZs2ezZMkS3N3d0el0LFmyhO3bt/Pxxx+Xaqzizqepazv2hRA89dRT/PXXXwXOmxNC3Lnkmp4Q5I7869KlC0ajke+//55FixYxd+5ce4clhChl0tITAnjssceIjo4mJSWFWrVq8dxzzzFy5Eh7hyWEKGWS9IQQQlQYMpBFCCFEhSFJTwghRIUhSU8IIUSFIUlPCCFEhSFJTwghRIUhSU8IIUSFIUlPCCFEhSFJTwghRIUhSU8IIUSF8f9I8Hwqy+FbiwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 500x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "eigs = get_all_layer_svs(model)\n",
    "# plot_all_layer_svs(eigs, title='Singular Values of Weights at Initialization', save_fp='figs/weight_svs_init.pdf')\n",
    "plot_all_layer_svs(eigs, title='Singular Values of Weights at Initialization')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2fa71ddd-6fc2-4c6b-b573-8882afe42a72",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1, Train Loss: 10.161540842896454, Val Loss: 10.269934784247871\n"
     ]
    }
   ],
   "source": [
    "losses = []\n",
    "\n",
    "for epoch in range(num_epochs):\n",
    "    for batchX, batchY in dataloader:\n",
    "        batchX = batchX.to('cuda')\n",
    "        batchY = batchY.to('cuda')\n",
    "        \n",
    "        y_pred, _ = model(batchX)\n",
    "\n",
    "        loss = criterion(y_pred, batchY)\n",
    "        losses.append(loss.item())\n",
    "\n",
    "        optimizer.zero_grad()\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "\n",
    "    if epoch % log_freq == 0:\n",
    "        with torch.no_grad():\n",
    "            total_loss = 0.0\n",
    "            for batchX, batchY in dataloader:\n",
    "                y_pred, _ = model(batchX.to('cuda'))\n",
    "                loss = criterion(y_pred.cpu(), batchY)\n",
    "                total_loss += loss * batchX.shape[0]\n",
    "                \n",
    "                \n",
    "            total_val_loss = 0.0\n",
    "            for batchX, batchY in test_loader:\n",
    "                y_pred, _ = model(batchX.to('cuda'))\n",
    "                loss = criterion(y_pred.cpu(), batchY)\n",
    "                total_val_loss += loss * batchX.shape[0]\n",
    "\n",
    "            print(f'Epoch {epoch+1}, Train Loss: {total_loss / num_samples}, Val Loss: {total_val_loss / num_samples}')\n",
    "\n",
    "            eigs = get_all_layer_svs(model)\n",
    "            # plot_all_layer_svs(eigs, title=f'Singular Values of Weights - Epoch {epoch+1}', save_fp=f'figs/weight_svs_ep_{epoch+1}.pdf')\n",
    "            plot_all_layer_svs(eigs, title=f'Singular Values of Weights - Epoch {epoch+1}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "01474656-d4f6-4e0d-8b2f-56fa3117fe3e",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "nfm = model.layers[0].weight.T @ model.layers[0].weight\n",
    "nfm = nfm.detach().cpu()\n",
    "\n",
    "jacs = compute_jacobians(model, X.to('cuda'))\n",
    "jacs[0] = jacs[0].reshape(-1, input_size)\n",
    "agop = jacs[0].t() @ jacs[0] / num_samples\n",
    "\n",
    "loss_grads = compute_jacobians_wrt_loss(model, X.to('cuda'), y.to('cuda'), criterion)\n",
    "lgop = - loss_grads.t() @ X / (weight_decay*num_samples)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "04f6473c-8abd-4b6f-aac3-b945db63fbb4",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "n_layers = [20, 17, 15, 13, 11, 9, 7, 5, 4, 3, 2, 1]\n",
    "pows = [1./x for x in n_layers]\n",
    "agop_sims = []\n",
    "lgop_sims = []\n",
    "\n",
    "U, S, Vh = torch.linalg.svd(agop)\n",
    "U2, S2, Vh2 = torch.linalg.svd(lgop)\n",
    "\n",
    "for power in pows:\n",
    "    _S = S.clone()\n",
    "    _S2 = S2.clone()\n",
    "    _agop = U @ torch.diag(_S.pow(power)) @ Vh\n",
    "    _lgop = U2 @ torch.diag(_S2.pow(power)) @ Vh2\n",
    "    \n",
    "    sim = F.cosine_similarity(nfm.flatten().unsqueeze(0), _agop.flatten().unsqueeze(0))\n",
    "    agop_sims.append(sim.squeeze())\n",
    "    \n",
    "    sim = F.cosine_similarity(nfm.flatten().unsqueeze(0), _lgop.flatten().unsqueeze(0))\n",
    "    lgop_sims.append(sim.squeeze())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3cb4e284-1182-4f3f-a417-197452f5b739",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "print('agop_sims:')\n",
    "print([float(x) for x in agop_sims])\n",
    "print('\\nlgop_sims:')\n",
    "print([float(x) for x in lgop_sims])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ad5877e7-e011-4a07-9599-bfb46441d6bc",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "plt.plot(np.log(pows), agop_sims, label='M = AGOP')\n",
    "plt.plot(np.log(pows), lgop_sims, label='M = LGOP')\n",
    "plt.xticks(np.log(pows), [f'1/{x}' for x in n_layers], rotation=50)\n",
    "plt.ylabel(r'cos_sim(nfm, $M^{\\alpha})$')\n",
    "plt.xlabel(r'AGOP power $\\alpha$')\n",
    "plt.title(f'{n_h_layers} hidden layers == {n_h_layers+1} weight matrices')\n",
    "plt.grid()\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "badd1f79-e6dd-4041-ba75-ecd6e793ab79",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3d6156fc-29bb-4770-957b-7a959589a6b0",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:.conda-torch_and_jax_conda]",
   "language": "python",
   "name": "conda-env-.conda-torch_and_jax_conda-py"
  },
  "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.18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
