{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "d2fdd1e3",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tau: 0.02, p: 1e-05, weight_decay: 1e-05\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1234658/2151573786.py:120: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  x_data = torch.tensor(x_data).to(device)\n",
      "/tmp/ipykernel_1234658/2151573786.py:121: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  y_data = torch.tensor(y_data).to(device)\n",
      "/tmp/ipykernel_1234658/2151573786.py:122: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  wts = torch.tensor(wts).to(device)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialized model\n",
      "Generated data\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████████| 1000000/1000000 [13:06<00:00, 1272.12it/s]\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import torch\n",
    "import os\n",
    "\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from tqdm import tqdm\n",
    "import itertools\n",
    "from torch.optim.lr_scheduler import CosineAnnealingLR\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "\n",
    "# Full-batch optimization\n",
    "input_dim = 4  # Input dimension\n",
    "hidden_dim = 10  # Hidden layer size\n",
    "output_dim = 1  # Output dimension\n",
    "# num_epochs = 1000000\n",
    "num_epochs = 1000000\n",
    "torch.set_default_dtype(torch.float64)\n",
    "\n",
    "\n",
    "for tau, p, weight_decay, learning_rate in itertools.product([0.02], [1e-5], [1e-5], [0.01]):\n",
    "# for tau, p, weight_decay, learning_rate in itertools.product([1], [1e-2], [1e-2], [0.011]):\n",
    "# for tau, p, weight_decay, learning_rate in itertools.product([0.1], [1e-3], [1e-3], [0.011]):\n",
    "# for tau, p, weight_decay, learning_rate in itertools.product([0.05], [1e-4], [1e-4], [0.011]):\n",
    "# for tau, p, weight_decay, learning_rate in itertools.product([0.02], [1e-5], [1e-7], [0.011]):\n",
    "    print(f\"tau: {tau}, p: {p}, weight_decay: {weight_decay}\")\n",
    "\n",
    "    # Generate dataset\n",
    "    def generate_data():\n",
    "        x = torch.zeros((82, 4))\n",
    "        x[0,0] = 1\n",
    "        x[0,1] = 1\n",
    "        j = 1\n",
    "        for v in itertools.product([0, 1, 2], repeat=4):\n",
    "            x[j,0] = v[0]\n",
    "            x[j,1] = v[1]\n",
    "            x[j,2] = v[2]\n",
    "            x[j,3] = v[3]\n",
    "            j += 1\n",
    "\n",
    "        y = tau * x[:, 0] * x[:, 1] + x[:, 2] * x[:, 3]\n",
    "        y = y.view(-1, 1)  # Reshape y to be a column vector\n",
    "        wts = torch.zeros((82))\n",
    "        wts[0] = 1-p\n",
    "        for i in range(1, 82):\n",
    "            wts[i] = p / 81\n",
    "        return x, y, wts\n",
    "\n",
    "    def compute_fact(model, xs, ys, wts):\n",
    "\n",
    "        outputs = model(xs)\n",
    "        criterion = nn.MSELoss(reduction='none')\n",
    "        loss = torch.sum(criterion(outputs, ys) * wts)\n",
    "        # Compute gradient of loss with respect to weight matrix W in first layer of network\n",
    "        gradients = torch.autograd.grad(outputs=loss, inputs=model.fc1.weight, create_graph=True)[0]\n",
    "        W = model.fc1.weight.data\n",
    "        fact = -W.T @ gradients\n",
    "        return fact\n",
    "\n",
    "    def compute_symmetrized_fact(model, xs, ys, wts):\n",
    "        fact = compute_fact(model, xs, ys, wts)\n",
    "        fact = fact @ fact.T\n",
    "        # matrix square root of fact\n",
    "        U, S, V = torch.svd(fact)\n",
    "        S = torch.sqrt(S)\n",
    "        fact = U @ torch.diag(S) @ V.T\n",
    "        return fact\n",
    "\n",
    "    def compute_agop(model, xs, wts):\n",
    "        agop = None\n",
    "        for j in range(xs.shape[0]):\n",
    "            x = xs[j].unsqueeze(0)\n",
    "            x = x.to(device)\n",
    "            x.requires_grad = True\n",
    "            output = model(x)\n",
    "            grad_outputs = torch.ones_like(output)\n",
    "            gradients = torch.autograd.grad(outputs=output, inputs=x, grad_outputs=grad_outputs, create_graph=True)[0]\n",
    "            outer_product = torch.bmm(gradients.unsqueeze(2), gradients.unsqueeze(1))  # (df/dx)(df/dx)^T\n",
    "            if agop is None:\n",
    "                agop = outer_product.mean(dim=0) * wts[j]\n",
    "            else:\n",
    "                agop += outer_product.mean(dim=0) * wts[j]\n",
    "        return agop\n",
    "\n",
    "\n",
    "    # Define the two-layer network with quadratic activations\n",
    "    class TwoLayerQuadraticNet(nn.Module):\n",
    "        def __init__(self, input_dim, hidden_dim, output_dim):\n",
    "            super(TwoLayerQuadraticNet, self).__init__()\n",
    "            self.fc1 = nn.Linear(input_dim, hidden_dim, bias=False)\n",
    "            self.fc2 = nn.Linear(hidden_dim, output_dim, bias=False)\n",
    "\n",
    "        def forward(self, x):\n",
    "            x = self.fc1(x)\n",
    "            x = x ** 2  # Quadratic activation\n",
    "            x = self.fc2(x)\n",
    "            return x\n",
    "\n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    # Set random seed for reproducibility\n",
    "    torch.manual_seed(0)\n",
    "    np.random.seed(0)\n",
    "    torch.backends.cudnn.deterministic = True\n",
    "    torch.backends.cudnn.benchmark = False\n",
    "\n",
    "\n",
    "    # Initialize model, loss, and optimizer\n",
    "    model = TwoLayerQuadraticNet(input_dim, hidden_dim, output_dim).to(device)\n",
    "    criterion = nn.MSELoss(reduction='none')\n",
    "    optimizer = optim.Adam(model.parameters(), lr=learning_rate, weight_decay=weight_decay)\n",
    "    # optimizer = optim.SGD(model.parameters(), lr=learning_rate, weight_decay=weight_decay)\n",
    "    scheduler = CosineAnnealingLR(optimizer, T_max=num_epochs, eta_min=0)\n",
    "    print('Initialized model')\n",
    "\n",
    "    # Training loop\n",
    "    x_data, y_data, wts = generate_data()\n",
    "    x_data = torch.tensor(x_data).to(device)\n",
    "    y_data = torch.tensor(y_data).to(device)\n",
    "    wts = torch.tensor(wts).to(device)\n",
    "    print('Generated data')\n",
    "\n",
    "    cosine_similarities = {}\n",
    "    cosine_similarities['fact'] = []\n",
    "    cosine_similarities['agop'] = []\n",
    "    cosine_similarities['symmetrized_fact'] = []\n",
    "    epochs = []\n",
    "\n",
    "    for epoch in tqdm(range(num_epochs)):\n",
    "        optimizer.zero_grad()\n",
    "        outputs = model(x_data)\n",
    "        loss = torch.sum(criterion(outputs, y_data) * wts)\n",
    "        # loss = p*\n",
    "        \n",
    "        # criterion(outputs, batch_y)\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        scheduler.step()\n",
    "        \n",
    "        if (epoch + 1) % 1000 == 0:\n",
    "            epochs.append(epoch)\n",
    "        \n",
    "\n",
    "\n",
    "            # Extract the first layer weights\n",
    "            W = model.fc1.weight.data.cpu().numpy()\n",
    "            # Compute W.T @ W\n",
    "            W_T_W = np.dot(W.T, W)\n",
    "            agop = compute_agop(model, x_data, wts)\n",
    "            fact = compute_fact(model, x_data, y_data, wts)\n",
    "            symmetrized_fact = compute_symmetrized_fact(model, x_data, y_data, wts)\n",
    "\n",
    "            # Compute cosine similarity between agop and W.T @ W\n",
    "            agop_flat = agop.detach().cpu().numpy().flatten()\n",
    "            W_T_W_flat = W_T_W.flatten()\n",
    "            agop_flat /= np.linalg.norm(agop_flat)\n",
    "            W_T_W_flat /= np.linalg.norm(W_T_W_flat)\n",
    "            cosine_similarity = np.dot(agop_flat, W_T_W_flat)\n",
    "            cosine_similarities['agop'].append(cosine_similarity)\n",
    "\n",
    "            # Compute cosine similarity between fact and W.T @ W\n",
    "            fact_flat = fact.detach().cpu().numpy().flatten()\n",
    "            fact_flat /= np.linalg.norm(fact_flat)\n",
    "            W_T_W_flat /= np.linalg.norm(W_T_W_flat)\n",
    "            cosine_similarity = np.dot(fact_flat, W_T_W_flat)\n",
    "            cosine_similarities['fact'].append(cosine_similarity)\n",
    "\n",
    "            # Compute cosine similarity between symmetrized fact and W.T @ W\n",
    "            symmetrized_fact_flat = symmetrized_fact.detach().cpu().numpy().flatten()\n",
    "            symmetrized_fact_flat /= np.linalg.norm(symmetrized_fact_flat)\n",
    "            W_T_W_flat /= np.linalg.norm(W_T_W_flat)\n",
    "            cosine_similarity = np.dot(symmetrized_fact_flat, W_T_W_flat)\n",
    "            cosine_similarities['symmetrized_fact'].append(cosine_similarity)\n",
    "\n",
    "\n",
    "    # plt.plot(epochs,cosine_similarities['fact'], label='$FACT$')\n",
    "    # plt.plot(epochs,cosine_similarities['agop'], label='$AGOP$')\n",
    "    # plt.plot(epochs,cosine_similarities['symmetrized_fact'], label='$\\sqrt{FACT \\cdot FACT^T}$')\n",
    "    # plt.xlabel('Epoch')\n",
    "    # plt.ylabel('Cosine Similarity')\n",
    "    # plt.title('Cosine similarity with $W^T W$')\n",
    "    # plt.legend()\n",
    "    # plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "93f3386b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "np.float64(0.0676312079992229)"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "cosine_similarities['agop'][-1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "6bfe1128",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "np.float64(0.9945955103127555)"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\n",
    "cosine_similarities['fact'][-1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "e2ad1c66",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABYYAAAH/CAYAAADjdAluAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAAReRJREFUeJzt3XmUlNW5L+C3GwSawRlBRBARFGdRcToOYBI1mOCsOKNGjYoTN+rJcMwxuTE5J4k4jyAJShxQDCInQUVEjbOCCOoBEaMi4gAyz3X/8JoYpauqu2voYj/PWq6VRb31fW81O72pX+3auyqTyWQC+IeePXvGq6++Wu42vuEHP/hB3HbbbeVuAwAAAIB1QHW5G4DGZNmyZTFlypRyt7FWe+yxR7lbAAAAAGAdIRiGr3j11Vdj1apV5W5jrQTDAAAAABSKYBi+Yp999olMJlPv/4YOHbrW6+6xxx4Num4mk4mePXuW+KcBAAAAwLpKMAwFVNvexEJdAAAAABoTwTAUUG3B8G677VbiTgAAAACgdoJhKJBMJhOTJ09e62OCYQAAAAAaE8EwFMiMGTNi4cKF3/jzJk2axM4771yGjgAAAABg7QTDUCCvvPLKWv982223jZqamhJ3AwAAAAC1EwxDgTh4DgAAAIBKIRiGAnHwHAAAAACVQjAMBSIYBgAAAKBSVGUymUy5myB/N9xwQwwcODBrzZw5c6Jdu3Yl6efMM8+MoUOH5qy75ZZb4pxzzilBRxG//e1v40c/+lHWmsGDB8dFF11UsHt+8MEH0bFjx7U+9tlnn8VGG21UsHutjXGRWznGBUDqRo4cGa+//nrOum9961vxb//2byXoqHRmzJgR06ZNi1mzZsWsWbNi9uzZsXjx4liyZEksWbIkqqqqokWLFlFTUxObbrppbL755rHFFltEjx49YocddojNN9+83C8BgP8v1fls/vz5MWnSpJg1a1b8/e9/j3fffTfmzZsXS5YsiaVLl8ayZctivfXWixYtWkSLFi2ibdu20a5du2jfvn107do1evToEV26dIkmTZqU+6U02KpVq2LSpEnxv//7vzFz5syYOXNmzJs3LxYvXhyLFy+OTCYTrVq1ilatWsWGG24YXbp0ia5du0a3bt1it912i2bNmpX7JcBaNS13A9TNBhtskLNmwYIFJQkA586dG3fffXdetQsWLChyN19YvXp13HDDDVlrNthggzjzzDMLet/aDp7baqutih4KRxgXuZRrXACszcsvv1yyD8Xq4qWXXiro9VasWBE//OEP45NPPslZO2HChJgwYUJB719qM2fOjAcffDAmTJgQzz//fF6vO5t27drF/vvvHwceeGD07ds3unTpUudrfP/734/Zs2c3qI/GqEOHDjF69OhytwEkIqX5bN68eTFmzJh49NFH4/nnn4/p06dHQ9cSNm/ePHr27Bn77rtv7LfffnHwwQfH+uuvX6COi2vGjBlx7733xhNPPBHPPvtsLFmypF7XadGiRey1117Ru3fvOO6446JHjx4F7rRuCv1v0aqqqmjevHk0b9482rRpE+3atYvNN988tt1229hhhx1i++23j6ZNxY+Nlb+ZCrPhhhvmrClV2HbzzTfH8uXL86r9/PPPi9zNF0aNGhXvvvtu1pqzzjorWrduXdD7lvvgOeMiu3KNC4C1WbhwYbz88svlbqPoHn744bzD0YkTJ8bbb78dXbt2LXJXhbVgwYK444474q677qr13wL19dFHH8XIkSNj5MiRMXDgwNhtt93i+OOPjwEDBsRmm22W1zVee+21nPNfJWpo6A5QF+v6fLZy5cq49957484774yJEyfGqlWrCnr95cuXx7PPPhvPPvts/O53v4v11lsv9t9//zjiiCPihBNOiLZt2xb0fg21fPnyGDZsWAwZMiRefPHFglxz2bJl8eSTT8aTTz4ZP//5z2PXXXeNM888M84888yoqakpyD3qotT/Fm3dunXst99+8d3vfjeOO+64aN++fcnuTW72GK4w+QSApQjbli9fHjfffHPe9aUKAAcPHpz18SZNmsSFF15Y8PuWe39h4yK7co0LgJTdeeededdmMpkYNmxY8ZopsDlz5sSgQYNiyy23jEGDBhU8FF6bV199Na644oro2LFjnHDCCSW5JwDr7ny2YMGC+MUvfhGdO3eOU045JcaPH1/wUHhtVq5cGePHj48LL7wwtthii+jXr1/85S9/Kfp9c1mxYkUMHjw4tt566zj33HMLFgqvzaRJk2LgwIHRpUuX+O///u9YtmxZ0e7VGCxatCj++te/xkUXXRQdO3aMo446Kp5//vlyt8X/JxiuMI1lZeiIESPio48+yru+FAHgyy+/HM8880zWmqOPPjo6depU8HtXQjBsXNSuWOMCIFUffvhhnd/k/eEPf4g1a9YUqaPCWLlyZfzud7+L7t27x+9///uSfRvn6z3ce++90bNnz+jXr19MmTKl5D0ApGJdnM/WrFkTt99+e3Tr1i3+4z/+Iz788MOy9bJy5coYPXp0HHbYYbHzzjvH8OHDy/KzmzhxYuyyyy5xySWXlHQLpo8++iguu+yy2HnnnWP8+PElu285rV69OkaNGhV77713HHfccfHBBx+Uu6XkCYYrTD57yZYibMu1AvPrStHTNddck7Pm0ksvLfh9582bV+vXNEsVDBsXtSvXuABI2R//+MdYvXp1nZ7z3nvvxeOPP16kjhpu6tSpsdtuu8X/+T//JxYuXFjudiIiYvTo0bHbbrvFhRdeGPPnzy93OwDrnHVtPpsxY0bstddecfbZZ8fcuXPL3c6/mDJlSpx66qmxyy67xNixY0tyz5UrV8bAgQPjoIMOijfffLMk91yb6dOnx8EHHxznnnturFixomx9lNr9998fO+20U8n+vlk7wXCFaQwrQx9//PF47bXX6vScYgeAH374Ydx3331Za/bZZ5/Ya6+9Cn7v2lYLb7bZZtGhQ4eC329tjIu1K+e4AEhZXb52+1VDhw4tcCeFMWzYsOjVq1dMnTq13K18w+rVq+P666+Pa6+9ttytAKxz1qX57K677orddtut4IfNFtrrr78effv2Lfp7xTlz5kTv3r3jhhtuaPABe4Vy6623xoEHHrhOHhxbm3nz5sXhhx9epy0pKSyHz1WYNm3aRJMmTbJ+alnsX6D5rMD8umL3dOONN8bKlSuz1hRrVegrr7yy1j8v1WrhCOOiNuUcFwCp+tvf/hZvvfVWvZ770EMPxfz58/P6wLNULrvssvjv//7vOj+vSZMmsdNOO8W+++4bu+22W3Tp0iW22mqr2GCDDaJly5bRvHnzWLJkSSxatCg++eSTmDFjRkyfPj1eeumlmDhxYp22ZoqIRvOmFmBdsS7NZz/+8Y/j6quvrtdzO3XqFHvuuWfsueee0bVr1+jUqVNsscUW0bJly6ipqYn11lsvFi9eHIsWLYp58+bFzJkz4+23346pU6fGM888E2+++Wa95qhizmtvv/129O7dO9577706Pa+mpib233//6NOnT2y//fbRvXv32GSTTaJNmzZRXV0dCxcujM8++yymT58e06ZNiwkTJsSECRNi0aJFed/jueeei169esX48eOje/fudX1pFSmTycT5558fNTU1cfrpp5e7nfRkqDgbbbRRJiJq/e9HP/pR0e791ltvZaqqqrLef23/denSpWg9LV26NLPppptmvf9WW22VWbVqVVHuf+KJJ671nv/+7/9elPvVxrj4V+UeFwC1eeKJJ3L+frzyyivL3Wa9nXnmmXWeD77630033VTul5DJZDKZ1atXZ84555w697/ffvtlbr755szcuXMbdP/XX38987Of/SzTtWvXvO5b6jHTuXPnrP107ty5pP0AFNq6MJ+tXr06c/bZZ9e59x49emR++ctfZqZOndrgHj799NPMiBEjMkceeWSmpqYm7x7mzZvX8B/AWkyfPj3TsWPHOv089thjj8yQIUMyixcvrvP9li1blhkxYkRm//33r9M9O3TokHnrrbeK8BMo/L9FV69enfn0008z06dPz0ycODHz61//OtO3b99MixYt6vSamzdvnnnxxReL8pqpna0kKlCu/WSLuQpz8ODB9frkrpg93XXXXfHJJ59krbnwwgujSZMmRbl/uQ+e+5Jx8a/KPS4AUrRkyZKcW/jk+r1b36/tFtoFF1wQt956a971hxxySDzzzDPx9NNPx7nnnhtt27Zt0P132GGHuOqqq2LGjBnx6KOPRp8+fRp0PQDyt67MZ+edd17cdtttedfvv//+8T//8z8xderU+MlPfhLbb799g3vYeOONo3///vHggw/G3Llz44Ybbohtt922wdetjzlz5kSfPn3i/fffz6u+a9eucf/998eLL74YZ5xxRrRs2bLO92zevHn0798/Jk6cGGPHjs37Zzp79uzo06dPRWwrUV1dHRtvvHFss802sf/++8fll18eY8aMiVmzZsXll18erVu3zus6y5cvj9NPPz1WrVpV5I75KsFwBcr1dZRi7SX72WefxR/+8IdaH6+urn04FXN/2+uuuy7r4+uvv36ceeaZRbn3kiVLav16UamDYePiX5VzXACkauTIkVkPZquqqoqf/exnWa/x4osvln0v38GDB+e911379u3jvvvui7/85S+x7777FqWfb33rW/H444/Hs88+G7169SrKPQD4p3VhPvvNb36T9wec7du3jxEjRsTEiRPj0EMPjaqqqqL01Lp16zj//PPjjTfeiNGjR8eOO+5YlPuszYoVK+Koo47Ke/uIAQMGxOTJk+OYY44pWA+HHXZYvPzyy3HBBRfkVf/BBx/EUUcdFcuXLy9YD6XUrl27+PWvfx0vv/xy3n/XU6dOjZtuuqnInfFVguEKlCsALNYqzNtuuy2WLFlS6+PHH398rY+tWrUq63Pr6/HHH48pU6ZkrTnzzDNj/fXXL/i9IyJee+21WLNmzTf+vE2bNtG1a9ei3LM2xsU/lXtcAKQq12E7+++/fwwaNChatWrVoOsU07hx42LQoEF51e63337x6quvxrHHHlvkrr6w9957x3PPPRdDhgxpNPtWAqyLKn0+GzNmTPz7v/97XrUHHXRQTJo0Kfr371/krv6pqqoqvve978XkyZNjyJAh0a5du6Lf84ILLohnn302Z111dXXccMMNMXTo0Jx/v/XRokWLuP7662PYsGHRtGnuY7+ef/75OP/88wveRyl17949nn/++dhvv/3yqv/tb39r1XAJCYYrUDlWhq5cuTJuuOGGWh9v3rx5/PjHP856jWIEk4MHD876eJMmTeLCCy8s+H2/VNvBc7vuumvRPmWtjXHxT+UeFwApmjlzZkycODFrzamnnhqtW7eOI488MmvdXXfdVZY3BPPnz48zzjhjrR/6ft1RRx0V48ePj/bt25egs3+qqqqKM844IyZPnpz3GywA8lfp89knn3wSZ511Vl5b/Z188snx2GOPlSSYXZvq6uo444wzYtq0aXHiiScW7T5jx46N22+/PWddVVVVDB8+vCRB7GmnnRb3339/XlsbDhkyJMaOHVv0noqpZcuWMXLkyNhiiy1y1r733nsxevToEnRFhGC4IpVjL9n7778/Pvjgg1ofP+mkk3KemFnovmbMmBGPPPJI1pojjzwyttpqq4Le96tq21+4Z8+eRbtnbYyLLzSGcQGQojvvvDPrm9Campp/rKw97bTTsl5r7ty5OX+XF8OFF16YdV77Ut++feOee+6JZs2alaCrtevUqVM8+eST8aMf/ahsPQCsiyp9PjvnnHPio48+yll30kknxR/+8IdGcebKxhtvHHfffXeMGjUq5/vaulq4cGGce+65edX+9re/LWpA/XVHHHFE3HjjjXnVnnvuuVm3N6kE7du3z/v1PvDAA0Xuhi8JhitQOVaGXnPNNVkfv/TSS6NZs2bRvHnzWmsKHQBee+21OT8FveSSSwp6z69rLAfPRRgXX2oM4wIgNWvWrMm633xERL9+/f6xhU+fPn2iY8eOWetLfWjPU089FcOHD89Zt91228U999wT6623Xgm6yq5JkybxX//1X3HLLbc0in4AKl2lz2ePPvpoPPjggznrevXqFUOGDMl6Hkw5HHHEEfHSSy8VdO/hX/ziF3ntK3zKKafEpZdeWrD75uucc87JK7h+77334j//8z9L0FFxff/734+dd945Z924ceNK0A0RguGKVOq9ZJ9++ul46aWXan38kEMOiR122CEiIuuerYXs6/PPP49hw4Zlrdlrr72KdghMxBf7477++utrfawxBsPGxReKPS4AUvT444/nfNN16qmn/uN/V1dXx0knnZS1/pFHHom5c+cWpL985LMXY4sWLeKBBx7I+3TtUjnnnHN86AlQAJU8n2Uymbjiiity1m2wwQbxwAMPZF28U07bbLNNPPfccwWZa+fMmZN168MvdenSJa+6Yvnd736X85u2ERE33XRTzJkzpwQdFU9VVVWcd955Oes++eSTePvtt0vQEYLhCpQrACz01wt+//vfZ338q5+qlSoAvOOOO2LRokVZa4r9BmnatGlrPR20efPmsf322xf13mtjXDSOcQGQolyH67Rv3z6+853v/Muf5fr67apVq/JawVsIjzzySDzzzDM566644oqyzPH5aNmyZblbAKh4lTyfjRw5stYzcL7ql7/8Zc5VzuXWqlWrvA5my+VXv/pVLF26NGfdjTfeWNaDyVu2bBk333xzzrqlS5fG1VdfXYKOiqt379551b322mtF7oQIwXBFyrXnzpo1a3KGY/maOXNm/PnPf6718Z122ulfJsZSBICrV6/O+Wlep06d4phjjinI/WpT2zYSO+64Y0EmsboyLhrHuABIzfz58+Ohhx7KWnPiiSd+Yw/DHj16xB577JH1eaX6+m2urZEiIjp37hyXX355CboBoBwqfT679tprc9bstNNOea3WXBcsWLAghgwZkrPuO9/5Thx22GEl6Ci7Pn36xOGHH56zbsiQIUXZJrKUunfvHptvvnnOunfffbcE3SAYrkC5VoZGFC5su+6667KezP31PXhKEQA+9NBDMWvWrKw1AwcOLPom+rV9GluObSQijIvGMi4AUjNixIhYtmxZ1pqvfu02nz//0tSpU+PFF1+sd2/5mDFjRowfPz5n3WWXXRYtWrQoai8AlE8lz2evv/563t98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Wxn617/+NaZNm1br4+ecc060atWqXr3VNwBcvnx51i0MIiL69u0b3bt3r1dfKTAuACi0XF+7raqqipNPPrko9y7moT35vJmeNWtWRZxcDkBu6+J8lk8w+M4778Tq1avrfO1Klc/8/uabb5agk4Z54403ctYUMxiu60HIy5cvjwULFsQHH3wQr7zySowZMyauueaaOOuss2LnnXcWCjdiguEKlms/2Wxh2zXXXFPrY+utt169v5YZkT0AzBZKjhgxIubOnZv12pdcckm9+0qFcQFAoUybNi1eeOGFrDUHHXRQdOrUqSj379u3b2yyySZZa+6+++56fQ11u+22y1mzcuXKnF9HBaDxW1fns2233TZnzfLly+Ptt9+u03UrWY8ePXLW5BO6lls+PW6//fYl6IR1na0kKtiGG26Y9WshtYVt06ZNi7/+9a+1Pu+EE05o0F419V0Zeu2112a97i677BJ9+vSpd1+pMC4AKJR8Dul54403Yo899ihaD7lW7H7yySfx8MMPx9FHH12n63bq1Ck23XTTnAfMvfbaa954AVS4dXU+69KlS2y44YYxf/78rHVTp05N5huWu+22W86a+fPnx8yZM2PrrbcuQUd1N3PmzLwOaM/ntUIuguEKlms/2dp+kWTbQzYiYtCgQfXs6AvZVqzW1tOECRNi8uTJWa9rVWh+jAsACmHVqlVx11135aybM2dO2Q9xufPOO+scDEd88Ybq0UcfzVozceLEOOGEE+rbGgBltq7PZz179ozx48dnrXnqqafiyCOPbEhrFaNHjx7RvHnzWL58eda6iRMnNtpgeOLEiTlrampq8vr2E+RiK4kKlisAXNvK0E8++SSGDx9e63P69OlTrwNcvqo+K0NzhZLt27eP/v37N6StZBgXABTC2LFj46OPPip3G3n5y1/+Uq+D4vbee++cNU888UR9WgKgkVjX57NevXrlrElpLmvatGnsueeeOesa88/kySefzFnTq1evaNKkSQm6YV0nGK5g9dlL9pZbbolly5bV+pyGrgqNyB4ALl269BtfoZk5c2Y8/PDDWa95/vnnR7NmzRrcWwqMCwAKIZ+v3TYWq1evjj/+8Y91ft53v/vdnDVvvvlmzJgxoz5tAdAIrOvz2WGHHZazZvLkyTF79uz6tlVxDj300Jw1Y8aMaZQHzK5atSrn++CI/P7eIR+C4QpW15WhK1asiBtvvLHW+h49ehTkl0u2ADDim8HkddddF2vWrKm1vqamJs4999wG95UK4wKAhpo7d2488sgj5W6jToYNG1bn5/Tq1Svatm2bs+6ee+6pR0cAlFsK89m+++4bG220UdaaTCYT9957bwO6qiz5fPD72WefxWOPPVaCbupm/Pjx8emnn+asyyf8hnwIhitYXQPAe+65J+ueSZdccklUVVU1uK+6BIALFizI+QnuKaecEptuummD+0qFcQFAQw0fPjxWrVpV7jbq5M0334xnn322Ts+prq6O733veznrhg8fHplMpr6tAVAmKcxnTZs2zWshTz77LK8rdt111+jUqVPOujvuuKME3dRNPj1ttdVWDd7qEb4kGK5gdT1k7Jprrqm1tm3btnHKKacUoq06BYBDhw6NhQsX1lpbVVUVF198cUH6SoVxAUBD3XnnneVuoV7q0/dZZ52Vs+Z///d/Y/To0fVpCYAySmU+O+OMM3LWvPLKKzFhwoR6dlRZqqqqYsCAATnrHnrooXjnnXdK0FF+3nvvvRg1alTOunz+viFfguEKlmsv2a+uDH3iiSdi0qRJtdaed9550aJFi4L0lW8AuGbNmrj++uuz1h566KHRo0ePgvSVCuMCgIZ44YUXYurUqVlrjj322MhkMiX/b5999sna17333htLly6t0+vdZ599Yscdd8xZ96tf/apO1wWgvFKaz/r06RPdunXLWXf11Vfnfc1KN2DAgKiuzh55rVmzplH9TH7zm9/kXOHepEmTvEJvyJdguILVZWVotlWhLVq0iPPOO69QbUWrVq2yno75ZV+jR4+OmTNnZr3WJZdcUrC+UmFcANAQ+axSOvnkk0vQyTeddNJJWR9fsGBBjBw5ss7XHThwYM6aF154IYYPH17nawNQHinNZ1VVVfHDH/4wZ924ceMqbs/l+urcuXNe20UNHTo0Xn/99RJ0lN1bb70Vt956a866I444Ijp27FiCjkiFYLiC5buX7PTp02PMmDG11p1yyimx2WabFbK1aNOmTc6+Bg8enPUaO+64Y3z7298uZFtJMC4AqK9ly5blPGhtk002KdtJ2Mcff3w0bdo0a019vjY8YMCA2HrrrXPWXXbZZfHZZ5/V+foAlFaK89m5554bHTp0yFk3cODAWLJkSZ2uXal+/vOf5zwvZ/Xq1XHeeedlPfi82DKZTFxwwQU5VwtXVVXFz3/+89I0RTIEwxUsVwC4fPnyWL58eVx77bW1HphSVVVVlNWX2bYN+Pzzz2Py5Mnx5JNPZr2GVaH1Y1wAUF8PPvhgzJ8/P2vNscceG+utt15pGvqaTTfdNA455JCsNRMmTIhZs2bV6brrrbdeXHXVVTnr5syZE6ecckqjO4jub3/7W9x4443lbgOg0UhxPqupqYkrr7wyZ90777yT1/7664Jdd901jj766Jx1Tz31VPzmN78pQUdrd91118Vjjz2Ws+64447La/srqAvBcAXLtZdsRMS7774bw4YNq/Xxww47rCh7teYKAHOtCt1ss81yfr2GtTMuAKivxvy123zvn8lkss5xtenfv3/su+++OevGjh0bP/nJT+p8/WK5//774+CDD46PP/643K0ANBqpzmdnnHFG7LLLLjnr/vSnP8V//dd/1enaleo3v/lNtGzZMmfdf/zHf8S4ceNK0NG/euqpp+Lyyy/PWde6deuyhtesuwTDFSzXytCIiN/+9rexePHiWh+/9NJLC9jRP2ULJ6dPnx5/+tOfsj7/vPPOi+bNmxe6rSQYFwDUx9///vcYP3581pouXbrEfvvtV6KO1q5fv35ZtyaKiBg2bFidV/VWV1fHsGHDoqamJmft1VdfXfbD6JYvXx4XX3xxHHfccbFs2bKy9gLQmKQ8nzVt2jT++Mc/RrNmzXLWXn755VnPnFlXbL311vGLX/wiZ92qVavimGOOiVdffbUEXX3hjTfeiCOOOCKWL1+es/ZXv/pVdO7cuQRdkRrBcAXLJwDM9gnjLrvsEgcffHDhGvqKbCtD//SnP2X9xVfoQ89SY1wAUB/Dhg3Lub/eiSeeWKJualdTUxNHHnlk1pp33303ZyiwNt26dct7Nc5PfvKTGDhwYM79AIvhtddei7333juuvfbakt8boLFLfT7beeed89oeKeKLBUGXXnppWeayUrr44ovz+lbQwoULo0+fPvH0008XvaeXX345DjzwwLzOLjjggAPi/PPPL3pPpEkwXMFatmyZc0+klStX1vrYoEGDCt3SP2QLALP1FPHFCa1t27YtdEvJMC4AqKt8v65a7q/dfimfPupzCF3EF4fynHbaaXnV3nDDDdG7d++YPn16ve5VV/Pnz4+BAwdGz549Y9KkSSW5J0AlMZ994bLLLovjjjsur9prrrkmDjjggJg2bVqd79MQK1euzPkesFCqq6vj/vvvz+twvvnz58e3v/3tuO2224rWz4gRI+Kggw7Kaxuojh07xn333RfV1eI7isPIqnD57Ce7Nh06dIgTTjihwN38U7YAMBeHizWccQFAXUyYMCHeeeedrDW77757bLfddiXqKLuDDz44Nt9886w1Dz74YCxYsKBe17/tttvigAMOyKv26aefjp133jmuvPLKnAcd1dfHH38cP/3pT6NLly5xww03xOrVq4tyH4BKZz77QlVVVfzhD3/Ia5VsRMSzzz4bu+yyS1xyySXx4Ycf1uledbVixYq49dZbo1u3bjF79uyi3uurOnToEKNGjcpra8Jly5bFOeecE/369YuZM2cWrIf3338/TjrppDjppJNi0aJFOetbtGgRo0aNinbt2hWsB/g6wXCFy2fbgLUZOHBgUU9grW8A+J3vfCd22GGHAneTHuMCgLrIZzVSYzr8s7q6OucHmUuXLs25d31tmjVrFn/+859jn332yat+2bJlcdVVV0WnTp3i0ksvLcj+hKtWrYqxY8fGSSedFJ07d47/+3//b9GCZ4B1hfnsn1q0aBEPP/xw7LnnnnnVr1q1KgYPHhydO3eOk08+OcaNG1ewFb2ZTCZeeOGF+OlPfxpdu3aNc889N959992CXLsuevXqFQ888EDe59aMHj06tttuuzjvvPNi6tSp9b7vjBkzYtCgQdG9e/cYMWJEXs9p3rx5jBo1KvbYY4963xfy0bTcDdAw9QkAW7VqFeecc07hm/mK+gaAxTr0LDXGBQD5WrBgQTzwwANZa5o0aRL9+/cvUUf5Ofnkk3MemnPnnXfWe27bcMMN49FHH41+/frF448/ntdzFi5cGNdcc01cc801sc0228TBBx8c++yzT+y2226x1VZb1ToPZjKZmD17dkyfPj1eeumlmDhxYjz11FOCYIA6MJ9908Ybbxzjx4+PI488Mh577LG8nrNy5cq4++674+67744NNtggDj744OjVq1fssccesc0228QWW2wRTZvWHiWtWbMm3n333XjjjTfijTfeiClTpsS4ceOKvhI5X3379o0HHnggjj766LwOfVu5cmXcfPPNcfPNN8eee+4ZhxxySPTu3Tt22GGHWlfyfvrppzFt2rSYMGFCjBs3Lp555pk6HSLYvHnzeOihh+LQQw/N+zlQX4LhClefAHDAgAGx0UYbFb6Zr6hPALj99tvHIYccUoRu0mNcAJCve++9N5YsWZK1pk+fPtG+ffsSdZSfnj17Ro8ePeKNN96oteb555+PN954I3r06FGve7Rq1SoeeeSRuPDCC+u81+CMGTNixowZceutt/7jzzbaaKPYYIMNomXLltG8efNYsmRJLF68OD799NNYunRpvXoE4Avms7Vr3bp1PPLII3HRRRfFLbfcUqfnfv755/Hggw/Ggw8++I8/a9KkSWy22WbRqlWrqKmpiSZNmsTixYtj0aJFsWjRoli8eHHOw//y0a1bt7jyyisbfJ216du3b4wbNy6OPfbYmDt3bt7Pe/HFF+PFF1+MX/7ylxHxxfvbTTbZJFq3bh1VVVWxaNGimDdvXsybN6/evbVr1y5GjhwZ//Zv/1bva0Bd2EqiwtV1L9nq6uq4+OKLi9PMV9QnALSHbOEYFwDka+jQoTlrGsshPV+Xz9eB63sI3ZeaN28et956a9x9993RunXrBl1r3rx5MWvWrJg2bVq8+uqr8dZbb8X777/f4FC4b9++eR+YB7CuMp/VrlmzZnHzzTfHyJEj673t4JdWr14dH374YcyYMSOmTJkSkyZNiunTp8eHH34YCxcubHAo3KpVq/j5z38eU6ZMid69ezfoWtkccMAB8fLLL0evXr3qfY0FCxbEO++8E1OmTInXXnstZs6c2aBQuFevXvHyyy8LhSkpwXCFq+sv9X79+kXXrl2L08xX1DUAbNu2baOdpCuRcQFAPt5888147rnnsta0bNkyjjrqqBJ1VDcnnXRSVFVVZa0ZPnx4QQ5rO/HEE2PatGlx7LHHNvhahbLTTjvFww8/HGPGjIkuXbqUux2AsjGf5efoo4+OadOmxemnn57zfqVWU1MTl156acycOTOuvPLKvPcBboiOHTvGM888E1dffXXU1NQU/X61qampiauvvjqeeeaZ2GKLLcrWB2kSDFe4ugaAgwYNKk4jX1PXAPCHP/xhtGjRokjdpMe4ACAf+aw++v73v9/glbLFstVWW+U8cX3OnDkxduzYgtxvyy23jPvuuy/Gjx8f++23X0GuWR+777573H///TF58uQ4/PDDy9YHQGNhPsvf5ptvHnfeeWe89NJLcdhhh5U9IO7YsWNcddVV8c4778Tvfve72GyzzUp6/6ZNm8YVV1wRr7/+ehx33HFRXV26mKy6ujqOOeaYmDJlSlxxxRVZ926GYhEMV7i6BIC9evUq2ZuYumxl0Lx58zjvvPOK2E16jAsAclm9enUMHz48Z11j/+ZGPv01dDuJr+vdu3c8/fTT8fzzz8fxxx9fkg8xN9hggxgwYEA8//zz8dJLL8UxxxxT9jfzAI2B+ax+evbsGWPHjo0333wzzj///KKfN/NVG2ywQZx00kkxevTomDVrVvzsZz+r9SC3Utl6663j3nvvjWnTpsWAAQOiZcuWRbtXTU1NnHrqqfH666/H/fffX5Jv70JtfBxR4X7605/GT3/603K38Q177bVXnU7dpLCMCwBy+eijj+Lss8/OWlNdXd3oDwA9/vjjY86cOVlrmjVrFplMpuBBaq9eveKee+6JRYsWxZgxY+LBBx+MCRMmxMcff1yQ62+33XZx0EEHxfe+97341re+Fc2aNSvIdQvh4osvjvnz59f6eEP3sATIl/msYbp37x433HBDDB48OCZMmBCjRo2Kxx57LKZPn16w927rrbde7LHHHnHAAQdE7969o3fv3o1qTvuqbbfdNoYOHRrXXXddPPTQQ3HffffFk08+GQsWLGjQddu0aRP7779/HHfccXHUUUdFmzZtCtQxNExVRkoDAAAFM3PmzHj++edj6tSpMWvWrJg1a1bMnj07Fi9eHEuWLImlS5dGVVVVtGjRImpqaqJt27bRoUOH2HLLLaNHjx6x4447xu67717yr9MCwJfmz58fL774Yrz66qsxa9asePfdd+O9996LefPm/WMuW758eTRt2vQf89lGG20Um2++eXTo0CE6deoU22+/feywww7Ro0ePit4icM2aNTF58uR49tln46233oqZM2fGO++884+fxeLFiyOTyUSrVq2iVatWscEGG0SXLl2ia9eu0b1799hnn31i1113jSZNmpT7pcA3CIYBAAAAABJjj2EAAAAAgMQIhgEAAAAAEiMYBgAAAABIjGAYAAAAACAxgmEAAAAAgMQIhgEAAAAAEiMYBgAAAABIjGAYAAAAACAxgmEAAAAAgMQIhgEAAAAAEiMYBgAAAABIjGAYAAAAACAxgmEAAAAAgMQIhgEAAAAAEiMYBgAAAABIjGAYAAAAACAxTfMtPLTdecXsg3pY/ckn5W6Br8tkyt0BVIRH19xf7haScOhOPy13C3zd3M/K3QFAvfzl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crdAPVgxDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJKYqk8lkyt0EAAAAAAClY8UwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBiBMMAAAAAAIkRDAMAAAAAJEYwDAAAAACQGMEwAAAAAEBi/h8UdxYdFppsIwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1500x500 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, axs = plt.subplots(1, 3, figsize=(15, 5))\n",
    "\n",
    "# Plot W^T W\n",
    "im1 = axs[0].imshow(W_T_W, cmap='viridis')\n",
    "axs[0].set_title(\"$W^T W$\")\n",
    "\n",
    "# Plot Fact\n",
    "im2 = axs[1].imshow(fact.detach().cpu().numpy(), cmap='viridis')\n",
    "axs[1].set_title(\"FACT\")\n",
    "\n",
    "# Plot AGOP\n",
    "im3 = axs[2].imshow(agop.detach().cpu().numpy(), cmap='viridis')\n",
    "axs[2].set_title(\"AGOP\")\n",
    "for ax in axs:\n",
    "    ax.title.set_fontsize(64)\n",
    "    ax.axis('off')\n",
    "\n",
    "plt.tight_layout()\n",
    "fig.savefig(\"comparison_plot.pdf\", dpi=300, bbox_inches='tight')\n",
    "plt.show()\n",
    "import pickle\n",
    "pickle.dump({'WTW': W_T_W, 'fact': fact.detach().cpu().numpy(), 'agop': agop.detach().cpu().numpy()}, open('comparison.pkl', 'wb'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b2c86440",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
