{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/users/xz231/.local/lib/python3.6/site-packages/tqdm/auto.py:22: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/usr/lib/python3.6/_collections_abc.py:841: MatplotlibDeprecationWarning: Support for setting the 'text.latex.preamble' or 'pgf.preamble' rcParam to a list of strings is deprecated since 3.3 and will be removed two minor releases later; set it to a single string instead.\n",
      "  self[key] = other[key]\n"
     ]
    }
   ],
   "source": [
    "%matplotlib inline\n",
    "\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "# import torch.linalg\n",
    "from utils import FakeDL\n",
    "import algos.pyhessian as pyhessian \n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "from utils import kp_2d\n",
    "import matplotlib as mpl\n",
    "from tqdm import tqdm\n",
    "\n",
    "# mpl.use('Agg')\n",
    "plt.style.use('ggplot')\n",
    "plt.rc('font', **{'family': 'serif', 'serif': ['Computer Modern']})\n",
    "plt.rc('text', usetex=True)\n",
    "params= {'text.latex.preamble' : [r'\\usepackage{amsmath}',r'\\usepackage{amssymb}', r'\\usepackage{bm}']}\n",
    "plt.rcParams.update(params)\n",
    "\n",
    "\n",
    "dev = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "print(dev)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_sharpness_traj(ss, eos, fig_size, exp_name):\n",
    "\n",
    "    plt.figure(figsize=fig_size)\n",
    "    plt.axhline(eos, label=r\"EoS Threshold ($2/\\eta$)\", linestyle='--', color='black')\n",
    "    plt.scatter(np.arange(len(ss)), ss, label=\"Numerical Sharpness\", s=7, zorder=100)\n",
    "    plt.legend(loc=1)\n",
    "    plt.xlabel('Iterations')\n",
    "    plt.ylabel('Sharpness')\n",
    "    plt.tight_layout()\n",
    "    plt.savefig(f\"./figs/{exp_name}_sharpness.pdf\")\n",
    "    plt.show()\n",
    "\n",
    "def plot_loss_traj(losses, fig_size, exp_name, logy=False):\n",
    "\n",
    "    plt.figure(figsize=fig_size)\n",
    "    plt.plot(np.arange(len(losses)), losses, label=\"Loss\", zorder=100)\n",
    "    plt.legend(loc=1)\n",
    "    plt.xlabel('Iterations')\n",
    "    plt.ylabel('Loss')\n",
    "    if logy:\n",
    "        plt.yscale('log')\n",
    "    plt.tight_layout()\n",
    "    plt.savefig(f\"./figs/{exp_name}_loss.pdf\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 74,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_dataset_single_dirc_scaled(n, d):\n",
    "    # assert n >= d\n",
    "    X = np.sqrt(n) * torch.qr(torch.normal(torch.zeros(n, d)))[0]\n",
    "    inc_rate = torch.diag(torch.Tensor([1] + [0] * (d-1)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_single_dirc(n, d):\n",
    "    X = torch.qr(torch.normal(torch.zeros(n, d)))[0]\n",
    "    inc_rate = torch.diag(torch.Tensor([1] + [0] * (d-1)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_multi_dirc(n, d):\n",
    "    X = torch.qr(torch.normal(torch.zeros(n, d)))[0]\n",
    "    inc_rate = torch.diag(torch.Tensor(np.linspace(1, 0, d)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_factorization(n, d):\n",
    "    # assert n >= d\n",
    "    X = torch.eye(d)\n",
    "    inc_rate = torch.diag(torch.Tensor(np.linspace(1, 0, d)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_factorization_single(n, d):\n",
    "    # assert n >= d\n",
    "    X = torch.eye(d)\n",
    "\n",
    "    inc_rate = torch.diag(torch.Tensor([1] + [1e-5] * (d-1)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_factorization_single_scaled(n, d):\n",
    "    # assert n >= d\n",
    "    X = torch.eye(d) * np.sqrt(n)\n",
    "    inc_rate = torch.diag(torch.Tensor([1] + [1e-5] * (d-1)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_scalar_factorization(n, d):\n",
    "    # assert n >= d\n",
    "    X = torch.eye(1)\n",
    "    Y = torch.eye(1)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_factorization(n, d):\n",
    "    # assert n >= d\n",
    "    X = torch.eye(d)\n",
    "    inc_rate = torch.diag(torch.Tensor(np.linspace(1, 0, d)))\n",
    "    A = torch.eye(d) * np.sqrt(d) * inc_rate\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)\n",
    "\n",
    "def get_dataset_isotropic_factorization(n, d):\n",
    "    # assert n >= d\n",
    "    X = torch.eye(d)\n",
    "    A = torch.eye(d)\n",
    "    Y = torch.matmul(X, A)\n",
    "    return X.to(dev), Y.to(dev)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "class LinearNet(nn.Module):\n",
    "    def __init__(self, L, d):\n",
    "        super(LinearNet, self).__init__()\n",
    "        self.L = L\n",
    "        self.layers = []\n",
    "        for i in range(L):\n",
    "            self.layers.append(nn.Linear(d, d, bias=False))\n",
    "            self.add_module(\"W{}\".format(i+1), self.layers[-1])\n",
    "\n",
    "    def forward(self, x):\n",
    "        for i in range(self.L):\n",
    "            x = self.layers[i](x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "class LinearNetSF(nn.Module):\n",
    "    def __init__(self, L, d):\n",
    "        super(LinearNetSF, self).__init__()\n",
    "        self.L = L\n",
    "        self.layers = []\n",
    "        self.layers.append(nn.Linear(1, d, bias=False))\n",
    "        self.add_module(\"W{}\".format(1), self.layers[-1])\n",
    "        self.layers.append(nn.Linear(d, 1, bias=False))\n",
    "        self.add_module(\"W{}\".format(2), self.layers[-1])\n",
    "\n",
    "    def forward(self, x):\n",
    "        for i in range(self.L):\n",
    "            x = self.layers[i](x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 75,
   "metadata": {},
   "outputs": [],
   "source": [
    "class IsotropicFactorizationNet(nn.Module):\n",
    "    def __init__(self, L, d):\n",
    "        super(IsotropicFactorizationNet, self).__init__()\n",
    "        self.L = L\n",
    "        self.layers = []\n",
    "        self.layers.append(nn.Linear(d, 1, bias=False))\n",
    "        self.add_module(\"W{}\".format(1), self.layers[-1])\n",
    "        self.layers.append(nn.Linear(1, d, bias=False))\n",
    "        self.add_module(\"W{}\".format(2), self.layers[-1])\n",
    "\n",
    "    def forward(self, x):\n",
    "        for i in range(self.L):\n",
    "            x = self.layers[i](x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "class LinearReLUNet(nn.Module):\n",
    "    def __init__(self, L, d):\n",
    "        super(LinearReLUNet, self).__init__()\n",
    "        self.L = L\n",
    "        self.layers = []\n",
    "        for i in range(L):\n",
    "            self.layers.append(nn.Linear(d, d, bias=False))\n",
    "            self.add_module(\"W{}\".format(i+1), self.layers[-1])\n",
    "\n",
    "    def forward(self, x):\n",
    "        for i in range(self.L):\n",
    "            x = self.layers[i](x)\n",
    "            if i != self.L - 1:\n",
    "                x = nn.functional.relu(x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_eigeninfo(net, dl, topn, criterion):\n",
    "    # Computing the top n eigenvectors using Power Iteration Method.\n",
    "    net.eval()\n",
    "    hessian_comp = pyhessian.hessian(net, criterion, dataloader=dl, cuda=False if dev==\"cpu\" else True)\n",
    "    eigenvals, eigenvecs = hessian_comp.eigenvalues(top_n=topn, tol=1e-6, maxIter=10000)\n",
    "    # print(eigenvals)\n",
    "    net.train()\n",
    "    return eigenvals, eigenvecs\n",
    "\n",
    "def compute_hvp(net, dl, topn, criterion, v):\n",
    "    # Computing the top n eigenvectors using Power Iteration Method.\n",
    "    net.eval()\n",
    "    hessian_comp = pyhessian.hessian(net, criterion, dataloader=dl, cuda=False if dev==\"cpu\" else True)\n",
    "    # print(\"conducting HVP with v{}\".format(v.shape))\n",
    "    val, hvp = hessian_comp.dataloader_hv_product(v)\n",
    "    return val, hvp\n",
    "\n",
    "def eigenthings_tensor_utils(t, device=None, out_device='cpu', symmetric=False, topn=-1):\n",
    "    t = t.to(device)\n",
    "    if topn >= 0:\n",
    "        _, eigenvals, eigenvecs = torch.svd_lowrank(t, q=min(topn, t.size()[0], t.size()[1]))\n",
    "        eigenvecs.transpose_(0, 1)\n",
    "    else:\n",
    "        if symmetric:\n",
    "            eigenvals, eigenvecs = torch.symeig(t, eigenvectors=True) # pylint: disable=no-member\n",
    "            eigenvals = eigenvals.flip(0)\n",
    "            eigenvecs = eigenvecs.transpose(0, 1).flip(0)\n",
    "        else:\n",
    "            _, eigenvals, eigenvecs = torch.svd(t, compute_uv=True) # pylint: disable=no-member\n",
    "            eigenvecs = eigenvecs.transpose(0, 1)\n",
    "    return eigenvals, eigenvecs\n",
    "\n",
    "\n",
    "def quadratic_loss(Y_hat, Y):\n",
    "    \"\"\"Batched quadratic loss\"\"\"\n",
    "    diff = torch.square(Y - Y_hat)\n",
    "    loss = diff.sum() * 0.5\n",
    "    if len(Y_hat.shape) != 1:\n",
    "        loss\n",
    "    return loss"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_Wprod_singular_space(Ws):\n",
    "    W_prod = Ws[0].clone()\n",
    "    for W in Ws[1:]:\n",
    "        W_prod = W.matmul(W_prod)\n",
    "\n",
    "    u_prod, s_prod,v_prod = np.linalg.svd(W_prod)\n",
    "    return W_prod, u_prod, s_prod, v_prod"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 49  Loss: 0.0328561 Sharpness: 11.7682: 100%|██████████| 50/50 [00:18<00:00,  2.74it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 288x216 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 288x288 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "seed, eval_start, eval_end = 0, 0, 50\n",
    "freq = 1\n",
    "d = 10 # layer width\n",
    "n = 10 # # of samples\n",
    "L = 2 # # of layers\n",
    "exp_name = 'ReLU_d10l2n10_eos50_init5-0.1_singledirc'\n",
    "sharp = 40\n",
    "eta = 2/sharp\n",
    "\n",
    "preset_scales = [5, 0.1]\n",
    "\n",
    "torch.manual_seed(seed)\n",
    "hessian_topn = 1\n",
    "\n",
    "X, Y = get_dataset_single_dirc(n, d)\n",
    "X, Y = X.to(dev), Y.to(dev)\n",
    "\n",
    "epochs = eval_end\n",
    "hessian_eval_epochs = list(range(eval_start, eval_end, freq))\n",
    "\n",
    "dataloader = FakeDL(X, Y, dev)\n",
    "loss_record = []\n",
    "eigenvals_record = []\n",
    "eigenvecs_record = []\n",
    "\n",
    "weight_record = []\n",
    "weight_prod_record = []\n",
    "Y_record = []\n",
    "\n",
    "net = LinearReLUNet(L, d).to(dev)\n",
    "for i, layer in enumerate(net.layers):\n",
    "    nn.init.xavier_normal_(layer.weight, gain=1)\n",
    "    layer.weight.data *= preset_scales[i]\n",
    "\n",
    "# criterion = nn.MSELoss(reduction='mean')\n",
    "optimizer = torch.optim.SGD(net.parameters(), lr=eta)\n",
    "\n",
    "trange = tqdm(range(epochs))\n",
    "for epoch in trange:  # loop over the dataset multiple times\n",
    "    optimizer.zero_grad()\n",
    "    \n",
    "    Y_hat = net(X)\n",
    "    loss = quadratic_loss(Y_hat, Y)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "    \n",
    "    loss_record.append(loss.item())\n",
    "\n",
    "    if epoch in hessian_eval_epochs:\n",
    "        eigenvals, eigenvecs = compute_eigeninfo(net, dataloader, hessian_topn, quadratic_loss)\n",
    "        eigenvals_record.append(eigenvals)\n",
    "        eigenvecs_record.append(eigenvecs)\n",
    "\n",
    "        Ws = [W.weight.data.clone() for W in net.layers]\n",
    "        W_prod = Ws[0].clone()\n",
    "        for W in Ws[1:]:\n",
    "            W_prod = W.matmul(W_prod)\n",
    "        \n",
    "        weight_record.append(Ws)\n",
    "        weight_prod_record.append(W_prod)\n",
    "        Y_record.append(Y_hat)\n",
    "\n",
    "    trange.set_description_str(\"Epoch {}  Loss: {:.6g} Sharpness: {:.6g}\".format(epoch, loss.item(), eigenvals_record[-1][0] if len(eigenvals_record) > 0 else -1))\n",
    "\n",
    "f, (ax1, ax2) = plt.subplots(2, 1, sharex=True, figsize=(4, 3))\n",
    "ax1.set_ylabel('Loss')\n",
    "ax1.plot(np.arange(len(loss_record)), np.array(loss_record))\n",
    "ax1.set_yscale('log')\n",
    "\n",
    "sharpness_record = [v[0] for v in eigenvals_record]\n",
    "ax2.scatter(np.arange(len(sharpness_record)), sharpness_record, s=5)\n",
    "ax2.axhline(2/eta, label=r\"EoS Threshold ($2/\\eta$)\", linestyle='--', color='black')\n",
    "ax2.legend(loc=1)\n",
    "\n",
    "\n",
    "ax2.set_ylabel('Numerical Sharpness')\n",
    "ax2.set_xlabel('Iterations')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f\"./figs/Shallow_{exp_name}.pdf\", bbox_inches='tight', pad_inches=0)\n",
    "plt.show()\n",
    "\n",
    "f, (ax1, ax2) = plt.subplots(2, 1, sharex=True, figsize=(4, 4))\n",
    "ax1.set_ylabel('Loss')\n",
    "ax1.plot(np.arange(len(loss_record)), np.array(loss_record))\n",
    "ax1.set_yscale('log')\n",
    "\n",
    "sharpness_record = [v[0] for v in eigenvals_record]\n",
    "ax2.scatter(np.arange(len(sharpness_record)), sharpness_record, s=5)\n",
    "ax2.axhline(2/eta, label=r\"EoS Threshold ($2/\\eta$)\", linestyle='--', color='black')\n",
    "ax2.legend(loc=1)\n",
    "\n",
    "\n",
    "ax2.set_ylabel('Sharpness')\n",
    "ax2.set_xlabel('Iterations')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f\"./figs/Intro_{exp_name}.pdf\", bbox_inches='tight', pad_inches=0)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 49  Loss: 0.681322 Sharpness: 39.0267: 100%|██████████| 50/50 [00:39<00:00,  1.28it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 288x216 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 288x288 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "seed, eval_start, eval_end = 0, 0, 50\n",
    "freq = 1\n",
    "d = 10 # layer width\n",
    "n = 10 # # of samples\n",
    "L = 2 # # of layers\n",
    "exp_name = 'ReLU_d10l2n10_eos50_init5-0.1_multidirc'\n",
    "sharp = 50\n",
    "eta = 2/sharp\n",
    "\n",
    "preset_scales = [5, 0.1]\n",
    "\n",
    "torch.manual_seed(seed)\n",
    "hessian_topn = 1\n",
    "\n",
    "X, Y = get_dataset_multi_dirc(n, d)\n",
    "X, Y = X.to(dev), Y.to(dev)\n",
    "\n",
    "epochs = eval_end\n",
    "hessian_eval_epochs = list(range(eval_start, eval_end, freq))\n",
    "\n",
    "dataloader = FakeDL(X, Y, dev)\n",
    "loss_record = []\n",
    "eigenvals_record = []\n",
    "eigenvecs_record = []\n",
    "\n",
    "weight_record = []\n",
    "weight_prod_record = []\n",
    "Y_record = []\n",
    "\n",
    "net = LinearReLUNet(L, d).to(dev)\n",
    "for i, layer in enumerate(net.layers):\n",
    "    nn.init.xavier_normal_(layer.weight, gain=1)\n",
    "    layer.weight.data *= preset_scales[i]\n",
    "\n",
    "# criterion = nn.MSELoss(reduction='mean')\n",
    "optimizer = torch.optim.SGD(net.parameters(), lr=eta)\n",
    "\n",
    "trange = tqdm(range(epochs))\n",
    "for epoch in trange:  # loop over the dataset multiple times\n",
    "    optimizer.zero_grad()\n",
    "    \n",
    "    Y_hat = net(X)\n",
    "    loss = quadratic_loss(Y_hat, Y)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "    \n",
    "    loss_record.append(loss.item())\n",
    "\n",
    "    if epoch in hessian_eval_epochs:\n",
    "        eigenvals, eigenvecs = compute_eigeninfo(net, dataloader, hessian_topn, quadratic_loss)\n",
    "        eigenvals_record.append(eigenvals)\n",
    "        eigenvecs_record.append(eigenvecs)\n",
    "\n",
    "        Ws = [W.weight.data.clone() for W in net.layers]\n",
    "        W_prod = Ws[0].clone()\n",
    "        for W in Ws[1:]:\n",
    "            W_prod = W.matmul(W_prod)\n",
    "        \n",
    "        weight_record.append(Ws)\n",
    "        weight_prod_record.append(W_prod)\n",
    "        Y_record.append(Y_hat)\n",
    "\n",
    "    trange.set_description_str(\"Epoch {}  Loss: {:.6g} Sharpness: {:.6g}\".format(epoch, loss.item(), eigenvals_record[-1][0] if len(eigenvals_record) > 0 else -1))\n",
    "\n",
    "f, (ax1, ax2) = plt.subplots(2, 1, sharex=True, figsize=(4, 3))\n",
    "ax1.set_ylabel('Loss')\n",
    "ax1.plot(np.arange(len(loss_record)), np.array(loss_record))\n",
    "ax1.set_yscale('log')\n",
    "\n",
    "sharpness_record = [v[0] for v in eigenvals_record]\n",
    "ax2.scatter(np.arange(len(sharpness_record)), sharpness_record, s=5)\n",
    "ax2.axhline(2/eta, label=r\"EoS Threshold ($2/\\eta$)\", linestyle='--', color='black')\n",
    "ax2.legend(loc=1)\n",
    "\n",
    "\n",
    "ax2.set_ylabel('Sharpness')\n",
    "ax2.set_xlabel('Iterations')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f\"./figs/Shallow_{exp_name}.pdf\", bbox_inches='tight', pad_inches=0)\n",
    "plt.show()\n",
    "\n",
    "f, (ax1, ax2) = plt.subplots(2, 1, sharex=True, figsize=(4, 4))\n",
    "ax1.set_ylabel('Loss')\n",
    "ax1.plot(np.arange(len(loss_record)), np.array(loss_record))\n",
    "ax1.set_yscale('log')\n",
    "\n",
    "sharpness_record = [v[0] for v in eigenvals_record]\n",
    "ax2.scatter(np.arange(len(sharpness_record)), sharpness_record, s=5)\n",
    "ax2.axhline(2/eta, label=r\"EoS Threshold ($2/\\eta$)\", linestyle='--', color='black')\n",
    "ax2.legend(loc=1)\n",
    "\n",
    "\n",
    "ax2.set_ylabel('Sharpness')\n",
    "ax2.set_xlabel('Iterations')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f\"./figs/Intro_{exp_name}.pdf\", bbox_inches='tight', pad_inches=0)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 49  Loss: 7.10543e-15 Sharpness: 31.6251: 100%|██████████| 50/50 [00:00<00:00, 62.95it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 288x216 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "seed, eval_start, eval_end = 0, 0, 50\n",
    "freq = 1\n",
    "d = 10 # layer width\n",
    "n = 1 # # of samples\n",
    "L = 2 # # of layers\n",
    "exp_name = 'Scalar_Factorization_d10l2n1_eos40_init5-0.1'\n",
    "sharp = 40\n",
    "eta = 2/sharp\n",
    "\n",
    "preset_scales = [5, 0.1]\n",
    "\n",
    "torch.manual_seed(seed)\n",
    "hessian_topn = 1\n",
    "\n",
    "X, Y = get_dataset_scalar_factorization(n, d)\n",
    "X, Y = X.to(dev), Y.to(dev)\n",
    "\n",
    "epochs = eval_end\n",
    "hessian_eval_epochs = list(range(eval_start, eval_end, freq))\n",
    "\n",
    "dataloader = FakeDL(X, Y, dev)\n",
    "loss_record = []\n",
    "eigenvals_record = []\n",
    "eigenvecs_record = []\n",
    "\n",
    "weight_record = []\n",
    "weight_prod_record = []\n",
    "Y_record = []\n",
    "\n",
    "net = LinearNetSF(L, d).to(dev)\n",
    "for i, layer in enumerate(net.layers):\n",
    "    nn.init.xavier_normal_(layer.weight, gain=1)\n",
    "    layer.weight.data *= preset_scales[i]\n",
    "\n",
    "# criterion = nn.MSELoss(reduction='mean')\n",
    "optimizer = torch.optim.SGD(net.parameters(), lr=eta)\n",
    "\n",
    "trange = tqdm(range(epochs))\n",
    "for epoch in trange:  # loop over the dataset multiple times\n",
    "    optimizer.zero_grad()\n",
    "    \n",
    "    Y_hat = net(X)\n",
    "    loss = quadratic_loss(Y_hat, Y)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "    \n",
    "    loss_record.append(loss.item())\n",
    "\n",
    "    if epoch in hessian_eval_epochs:\n",
    "        eigenvals, eigenvecs = compute_eigeninfo(net, dataloader, hessian_topn, quadratic_loss)\n",
    "        eigenvals_record.append(eigenvals)\n",
    "        eigenvecs_record.append(eigenvecs)\n",
    "\n",
    "        Ws = [W.weight.data.clone() for W in net.layers]\n",
    "        W_prod = Ws[0].clone()\n",
    "        for W in Ws[1:]:\n",
    "            W_prod = W.matmul(W_prod)\n",
    "        \n",
    "        weight_record.append(Ws)\n",
    "        weight_prod_record.append(W_prod)\n",
    "        Y_record.append(Y_hat)\n",
    "\n",
    "    trange.set_description_str(\"Epoch {}  Loss: {:.6g} Sharpness: {:.6g}\".format(epoch, loss.item(), eigenvals_record[-1][0] if len(eigenvals_record) > 0 else -1))\n",
    "\n",
    "f, (ax1, ax2) = plt.subplots(2, 1, sharex=True, figsize=(4, 3))\n",
    "ax1.set_ylabel('Loss')\n",
    "ax1.plot(np.arange(len(loss_record)), np.array(loss_record))\n",
    "ax1.set_yscale('log')\n",
    "\n",
    "sharpness_record = [v[0] for v in eigenvals_record]\n",
    "ax2.scatter(np.arange(len(sharpness_record)), sharpness_record, s=5)\n",
    "ax2.axhline(2/eta, label=r\"EoS Threshold ($2/\\eta$)\", linestyle='--', color='black')\n",
    "ax2.legend(loc=1)\n",
    "\n",
    "\n",
    "ax2.set_ylabel('Sharpness')\n",
    "ax2.set_xlabel('Iterations')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f\"./figs/Shallow_{exp_name}.pdf\", bbox_inches='tight', pad_inches=0)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 88,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 29  Loss: 4.5 Sharpness: 31.5512: 100%|██████████| 30/30 [00:06<00:00,  4.32it/s]    \n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 288x216 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "seed, eval_start, eval_end = 0, 0, 30\n",
    "freq = 1\n",
    "d = 10 # layer width\n",
    "n = 10 # # of samples\n",
    "L = 2 # # of layers\n",
    "exp_name = 'IsotropicFactorization_d10l2n10_eos50_init5-0.1_multidirc'\n",
    "sharp = 40\n",
    "eta = 2/sharp\n",
    "\n",
    "preset_scales = [5, 0.1]\n",
    "\n",
    "torch.manual_seed(seed)\n",
    "hessian_topn = 1\n",
    "\n",
    "X, Y = get_dataset_isotropic_factorization(n, d)\n",
    "X, Y = X.to(dev), Y.to(dev)\n",
    "\n",
    "epochs = eval_end\n",
    "hessian_eval_epochs = list(range(eval_start, eval_end, freq))\n",
    "\n",
    "dataloader = FakeDL(X, Y, dev)\n",
    "loss_record = []\n",
    "eigenvals_record = []\n",
    "eigenvecs_record = []\n",
    "\n",
    "weight_record = []\n",
    "weight_prod_record = []\n",
    "Y_record = []\n",
    "\n",
    "net = IsotropicFactorizationNet(L, d).to(dev)\n",
    "for i, layer in enumerate(net.layers):\n",
    "    nn.init.xavier_normal_(layer.weight, gain=1)\n",
    "    layer.weight.data *= preset_scales[i]\n",
    "\n",
    "# criterion = nn.MSELoss(reduction='mean')\n",
    "optimizer = torch.optim.SGD(net.parameters(), lr=eta)\n",
    "\n",
    "trange = tqdm(range(epochs))\n",
    "for epoch in trange:  # loop over the dataset multiple times\n",
    "    optimizer.zero_grad()\n",
    "    \n",
    "    Y_hat = net(X)\n",
    "    loss = quadratic_loss(Y_hat, Y)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "    \n",
    "    loss_record.append(loss.item())\n",
    "\n",
    "    if epoch in hessian_eval_epochs:\n",
    "        eigenvals, eigenvecs = compute_eigeninfo(net, dataloader, hessian_topn, quadratic_loss)\n",
    "        eigenvals_record.append(eigenvals)\n",
    "        eigenvecs_record.append(eigenvecs)\n",
    "\n",
    "        Ws = [W.weight.data.clone() for W in net.layers]\n",
    "        W_prod = Ws[0].clone()\n",
    "        for W in Ws[1:]:\n",
    "            W_prod = W.matmul(W_prod)\n",
    "        \n",
    "        weight_record.append(Ws)\n",
    "        weight_prod_record.append(W_prod)\n",
    "        Y_record.append(Y_hat)\n",
    "\n",
    "    trange.set_description_str(\"Epoch {}  Loss: {:.6g} Sharpness: {:.6g}\".format(epoch, loss.item(), eigenvals_record[-1][0] if len(eigenvals_record) > 0 else -1))\n",
    "\n",
    "f, (ax1, ax2) = plt.subplots(2, 1, sharex=True, figsize=(4, 3))\n",
    "ax1.set_ylabel('Loss - 4.5')\n",
    "ax1.plot(np.arange(len(loss_record)), np.array(loss_record) - 4.5)\n",
    "ax1.set_yscale('log')\n",
    "\n",
    "sharpness_record = [v[0] for v in eigenvals_record]\n",
    "ax2.scatter(np.arange(len(sharpness_record)), sharpness_record, s=5)\n",
    "ax2.axhline(2/eta, label=r\"EoS Threshold ($2/\\eta$)\", linestyle='--', color='black')\n",
    "ax2.legend(loc=1)\n",
    "\n",
    "\n",
    "ax2.set_ylabel('Sharpness')\n",
    "ax2.set_xlabel('Iterations')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f\"./figs/Shallow_{exp_name}.pdf\", bbox_inches='tight', pad_inches=0)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "31f2aee4e71d21fbe5cf8b01ff0e069b9275f58929596ceb00d14d90e3e16cd6"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
