{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {},
   "outputs": [],
   "source": [
    "# the following codes demonstrate about how syre improves trainablity \n",
    "# by shifting eigenvalue distribution of FCN trained on MNIST dataset\n",
    "import torch\n",
    "from torch import nn\n",
    "from torch.autograd import Variable\n",
    "import torchvision\n",
    "\n",
    "import torch.optim as optim\n",
    "import numpy as np\n",
    "import torch.nn.functional as F\n",
    "import numpy as np\n",
    "\n",
    "device = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")\n",
    "\n",
    "class BiasedLinear(nn.Module):\n",
    "    def __init__(self,in_dim,out_dim,bias=True):\n",
    "        super(BiasedLinear,self).__init__()\n",
    "        self.weight0=nn.Linear(in_dim,out_dim).weight.data.to(device)\n",
    "        self.weight=nn.Parameter(self.weight0*0)\n",
    "        self.biase0=nn.Linear(in_dim,out_dim).bias.data.to(device)\n",
    "        if bias:\n",
    "            self.biase=nn.Parameter(self.biase0*0)\n",
    "        else:\n",
    "            self.biase=self.biase0*0\n",
    "            self.biase0=self.biase0*0\n",
    "\n",
    "    def forward(self,x):\n",
    "        # syre: add a statistical bias\n",
    "        return F.linear(x,self.weight+self.weight0,self.biase+self.biase0)\n",
    "\n",
    "\n",
    "activation=F.tanh\n",
    "# standard FCN\n",
    "class FCN(nn.Module):\n",
    "    def __init__(self,in_dim,n_hidden_1,out_dim):\n",
    "        super(FCN,self).__init__()\n",
    "        self.layer1 = nn.Linear(in_dim,n_hidden_1)\n",
    "        self.layer2 = nn.Linear(n_hidden_1,out_dim)\n",
    "\n",
    "    def forward(self,x):\n",
    "        x = activation(self.layer1(x))\n",
    "        x = self.layer2(x)\n",
    "        return x\n",
    "\n",
    "# standard FCN with syre \n",
    "class FCN_syre(nn.Module):\n",
    "    def __init__(self,in_dim,n_hidden_1,out_dim):\n",
    "        super(FCN_syre,self).__init__()\n",
    "        self.layer1 = BiasedLinear(in_dim,n_hidden_1)\n",
    "        self.layer2 = BiasedLinear(n_hidden_1,out_dim)\n",
    "\n",
    "    def forward(self,x):\n",
    "        x = activation(self.layer1(x))\n",
    "        x = self.layer2(x)\n",
    "        return x\n",
    "    \n",
    "def setup_seed(seed):\n",
    "    torch.manual_seed(seed)\n",
    "    torch.cuda.manual_seed_all(seed)\n",
    "    np.random.seed(seed)\n",
    "    torch.backends.cudnn.deterministic = True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [],
   "source": [
    "# standard training procedure of MNIST\n",
    "setup_seed(0)\n",
    "batch_size_train = 64\n",
    "batch_size_test = 100\n",
    "train_loader = torch.utils.data.DataLoader(\n",
    "    torchvision.datasets.MNIST('./dataset/', train=True, download=True,\n",
    "                               transform=torchvision.transforms.Compose([\n",
    "                                   torchvision.transforms.ToTensor(),\n",
    "                                   torchvision.transforms.Normalize(\n",
    "                                       (0.1307,), (0.3081,))\n",
    "                               ])),\n",
    "    batch_size=batch_size_train, shuffle=True)\n",
    "test_loader = torch.utils.data.DataLoader(\n",
    "    torchvision.datasets.MNIST('./dataset/', train=False, download=True,\n",
    "                               transform=torchvision.transforms.Compose([\n",
    "                                   torchvision.transforms.ToTensor(),\n",
    "                                   torchvision.transforms.Normalize(\n",
    "                                       (0.1307,), (0.3081,))\n",
    "                               ])),\n",
    "    batch_size=batch_size_test, shuffle=True)\n",
    "\n",
    "\n",
    "epoches = 4\n",
    "criterion = nn.CrossEntropyLoss()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {},
   "outputs": [],
   "source": [
    "def run(weight_decay,syre):\n",
    "    if syre:\n",
    "        model = FCN_syre(28*28,300,10)\n",
    "    else:\n",
    "        model = FCN(28*28,300,10)\n",
    "    optimizer = optim.SGD(model.parameters(),weight_decay=weight_decay,lr=0.01)\n",
    "    train_loss_list=[]\n",
    "    train_acc_list=[]\n",
    "    test_loss_list=[]\n",
    "    test_acc_list=[]\n",
    "\n",
    "    for epoch in range(epoches):\n",
    "        # training\n",
    "        train_loss = 0\n",
    "        train_acc = 0    \n",
    "        for img,label in train_loader:\n",
    "            img = Variable(img.view(img.size(0),-1))\n",
    "            label = Variable(label)\n",
    "            output = model(img)\n",
    "            loss = criterion(output,label)\n",
    "            if epoch > 0:\n",
    "                optimizer.zero_grad()\n",
    "                loss.backward()\n",
    "                optimizer.step()\n",
    "            train_loss += loss.data\n",
    "            _,pred = output.max(1)\n",
    "            num_correct = (pred==label).sum().item()\n",
    "            acc = num_correct / img.shape[0]\n",
    "            train_acc += acc\n",
    "\n",
    "        train_loss_list.append(train_loss/len(train_loader))\n",
    "        train_acc_list.append(train_acc/len(train_loader))\n",
    "        print('epoch: {}, Train Loss: {:.6f}, Train Acc: {:.6f}'.format(epoch+1, train_loss_list[-1], train_acc_list[-1]))\n",
    "\n",
    "        # testing\n",
    "        model.eval()\n",
    "        eval_loss = 0\n",
    "        eval_acc = 0\n",
    "        for img,label in test_loader:\n",
    "            img = Variable(img.view(img.size(0),-1))\n",
    "            label = Variable(label)\n",
    "            output = model(img)\n",
    "            loss = criterion(output,label)\n",
    "            eval_loss += loss.data*img.size(0)\n",
    "            _ , pred = torch.max(output,1)\n",
    "            num_correct = (pred==label).sum().item()\n",
    "            eval_acc += num_correct \n",
    "\n",
    "        test_loss_list.append(eval_loss/batch_size_test/len(test_loader))\n",
    "        test_acc_list.append(eval_acc/batch_size_test/len(test_loader))\n",
    "        print(\"Test Loss:{:.6f},Acc:{:.6f}\".format(test_loss_list[-1],test_acc_list[-1]))\n",
    "\n",
    "    # calculate rank\n",
    "    train_loader2 = torch.utils.data.DataLoader(\n",
    "    torchvision.datasets.MNIST('./dataset/', train=True, download=True,\n",
    "                               transform=torchvision.transforms.Compose([\n",
    "                                   torchvision.transforms.ToTensor(),\n",
    "                                   torchvision.transforms.Normalize(\n",
    "                                       (0.1307,), (0.3081,))\n",
    "                               ])),\n",
    "    batch_size=1000, shuffle=True)\n",
    "    img,label = next(iter(train_loader2))\n",
    "    img = Variable(img.view(img.size(0),-1))\n",
    "    # calculate the eigenvalue of the first layer output\n",
    "    if syre:\n",
    "        out = (model.layer1.weight+model.layer1.weight0)@img.T\n",
    "    else:\n",
    "        out = model.layer1.weight@img.T\n",
    "    cov = np.corrcoef(out.detach().numpy())\n",
    "        \n",
    "    return train_acc_list, test_acc_list, cov"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "epoch: 1, Train Loss: 2.309378, Train Acc: 0.114622\n",
      "Test Loss:2.315925,Acc:0.110900\n",
      "epoch: 2, Train Loss: 1.093217, Train Acc: 0.801323\n",
      "Test Loss:1.010211,Acc:0.832100\n",
      "epoch: 3, Train Loss: 1.028576, Train Acc: 0.822012\n",
      "Test Loss:1.009970,Acc:0.826200\n",
      "epoch: 4, Train Loss: 1.028627, Train Acc: 0.821012\n",
      "Test Loss:1.009573,Acc:0.836000\n",
      "epoch: 1, Train Loss: 2.312532, Train Acc: 0.083872\n",
      "Test Loss:2.311544,Acc:0.080400\n",
      "epoch: 2, Train Loss: 1.741598, Train Acc: 0.608775\n",
      "Test Loss:1.752680,Acc:0.551900\n",
      "epoch: 3, Train Loss: 1.761499, Train Acc: 0.538430\n",
      "Test Loss:1.751691,Acc:0.549200\n",
      "epoch: 4, Train Loss: 1.759311, Train Acc: 0.531933\n",
      "Test Loss:1.748071,Acc:0.550100\n"
     ]
    }
   ],
   "source": [
    "# This is an extreme case with very large weight decay such that the difference is easily observable\n",
    "weight_decay = 1\n",
    "train_acc_syre, test_acc_syre, cov_syre=run(weight_decay,True)\n",
    "train_acc, test_acc, cov=run(weight_decay,False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 41,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x1ec05942640>"
      ]
     },
     "execution_count": 41,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# The results indicate that syre significantly improves the trainability\n",
    "plt.plot(train_acc,\"b-\",label='vanilla train')\n",
    "plt.plot(test_acc,\"b--\",label='vanilla test')\n",
    "plt.plot(train_acc_syre, \"r-\", label=r'syre train')\n",
    "plt.plot(test_acc_syre,\"r--\",label='syre test')\n",
    "plt.xticks([0,1,2,3])\n",
    "plt.xlabel(\"epoch\")\n",
    "plt.ylabel(\"acc.\")\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# eigenvalue distribution of standard FCN\n",
    "MIN,MAX=0.00001,10\n",
    "plt.hist(np.linalg.eig(cov)[0], bins=np.append(0,10 ** np.linspace(np.log10(MIN), np.log10(MAX), 50))) \n",
    "plt.gca().set_xscale(\"log\")\n",
    "plt.xlabel(\"eigenvalues\")\n",
    "plt.ylabel(\"number of eigenvalues\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# eigenvalue distribution of syre, where we can observe that it clearly shifts to the right\n",
    "MIN,MAX=0.00001,10\n",
    "plt.hist(np.linalg.eig(cov_syre)[0], bins=np.append(0,10 ** np.linspace(np.log10(MIN), np.log10(MAX), 50))) \n",
    "plt.gca().set_xscale(\"log\")\n",
    "plt.xlabel(\"eigenvalues\")\n",
    "plt.ylabel(\"number of eigenvalues\")\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
