{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import copy\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from torch.utils.data import Dataset, DataLoader\n",
    "from torchvision import datasets, transforms\n",
    "import time\n",
    "import random\n",
    "import pickle\n",
    "\n",
    "%matplotlib inline\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.lines as lines\n",
    "from mpl_toolkits.mplot3d import Axes3D\n",
    "\n",
    "from pyhessian.hessian import hessian # Hessian computation\n",
    "\n",
    "from sharpness import vectorize, gather_grads, get_hessian_measures, calculate_IGS_largemodel\n",
    "from utils import *"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "os.environ[\"CUDA_DEVICE_ORDER\"] = \"PCI_BUS_ID\"\n",
    "os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"0\""
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "tags": []
   },
   "source": [
    "# Load MNIST data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "60000\n"
     ]
    }
   ],
   "source": [
    "dataset = 'mnist'\n",
    "batch_size = 128\n",
    "test_batch_size = 500\n",
    "data_aug = False\n",
    "cutout = False\n",
    "use_hess_loader = True\n",
    "hess_batch_size = 1024\n",
    "num_classes = 10\n",
    "train_loader, test_loader = get_data(dataset=dataset,\n",
    "                                         train_bs=batch_size,\n",
    "                                         test_bs=test_batch_size,\n",
    "                                         data_augmentation=data_aug,\n",
    "                                         normalization=True,\n",
    "                                         shuffle=True,\n",
    "                                         cutout=cutout, \n",
    "                                         model = 'lenet'\n",
    "                                        )    \n",
    "print(len(train_loader.dataset))\n",
    "hessian_loader, test_hessian_loader = get_data(dataset, train_bs=hess_batch_size, test_bs=hess_batch_size, \n",
    "                                 data_augmentation = data_aug, \n",
    "                                 normalization = True, \n",
    "                                 shuffle = False,\n",
    "                                 cutout=cutout, \n",
    "                                 model = 'lenet')\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "LeNet5(\n",
      "  (conv1): Conv2d(1, 6, kernel_size=(5, 5), stride=(1, 1))\n",
      "  (pool): MaxPool2d(kernel_size=2, stride=2, padding=0, dilation=1, ceil_mode=False)\n",
      "  (conv2): Conv2d(6, 16, kernel_size=(5, 5), stride=(1, 1))\n",
      "  (fc1): Linear(in_features=400, out_features=120, bias=True)\n",
      "  (fc2): Linear(in_features=120, out_features=84, bias=True)\n",
      "  (fc3): Linear(in_features=84, out_features=10, bias=True)\n",
      ")\n",
      "# params: 61706\n"
     ]
    }
   ],
   "source": [
    "model = get_model('lenet',\n",
    "                  dataset=dataset,\n",
    "                  num_classes=num_classes\n",
    "                 )\n",
    "model = model.cuda()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load trained models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2\n"
     ]
    }
   ],
   "source": [
    "loss_val = 0.01\n",
    "file_list = []\n",
    "bs_list = []\n",
    "lr_list = []\n",
    "save_folder = \"./trained_model_cor_exp_MNIST/\"\n",
    "files = list(os.listdir(save_folder))\n",
    "files = [os.path.join(save_folder, f) for f in files] \n",
    "files.sort(key=lambda x: os.path.getmtime(x))\n",
    "for filename in files:\n",
    "    if len(filename.split('lenet')) > 1 and len(filename.split('labelsmoothing')) == 1:\n",
    "        if filename.endswith(\".pth\") and len(filename.split('loss'+str(loss_val)))>1: \n",
    "             file_list.append(filename)\n",
    "             lr_list.append(float(filename.split('lr')[1].split('_')[0]))\n",
    "             bs_list.append(int(filename.split('bs')[1].split('_')[0]))\n",
    "print(len(file_list))\n",
    "# file_list_red = []\n",
    "# bs_list_red = []\n",
    "# lr_list_red = []\n",
    "# startidx = 16\n",
    "# endidx = 25\n",
    "# for i in range(startidx, endidx):\n",
    "#     file_list_red.append(file_list[i])\n",
    "#     bs_list_red.append(bs_list[i])\n",
    "#     lr_list_red.append(lr_list[i])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_file_list = len(file_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "criterion = torch.nn.CrossEntropyLoss()\n",
    "criterion_alldata = torch.nn.CrossEntropyLoss(reduction = 'none')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# change keys for models\n",
    "def remove_module_in_keys(state_dict):\n",
    "    for key in list(state_dict.keys()):\n",
    "        if 'module.' in key:\n",
    "            state_dict[key.replace('module.', '')] = state_dict.pop(key)\n",
    "    state_dict.keys()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Calculate m-IGS values"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### NOTE: Precomputed m-IGS values for some trained models are saved in 'numerics_MNIST_lenet_*.pickle' in 'trained_model_cor_exp_MNIST' folder"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "128 128\n",
      "128 128\n"
     ]
    }
   ],
   "source": [
    "train_loader_dict_mIGS = {}\n",
    "test_loader_dict_mIGS = {}\n",
    "bs_set = [128]\n",
    "for bs in bs_set:\n",
    "    train_loader_bs, test_loader_bs = get_data(dataset=dataset,\n",
    "                                         train_bs=int(bs),\n",
    "                                         test_bs=int(bs),\n",
    "                                         data_augmentation=data_aug,\n",
    "                                         normalization=True,\n",
    "                                         shuffle=True,\n",
    "                                         cutout=cutout, \n",
    "                                         model = 'lenet'\n",
    "                                        )    \n",
    "    train_loader_dict_mIGS[str(bs)] = train_loader_bs\n",
    "    test_loader_dict_mIGS[str(bs)] = test_loader_bs\n",
    "    for i, (X, y) in enumerate(train_loader_bs):\n",
    "        print(bs, X.shape[0])\n",
    "        break\n",
    "    for i, (X, y) in enumerate(test_loader_bs):\n",
    "        print(bs, X.shape[0])\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "# calculate m-sharpness \n",
    "def calculate_m_IGS2(model, train_loader, L, V, criterion, eps = 0, num_trials = 1, expand=False):\n",
    "    # model: the model to calculate m-IGS on\n",
    "    # train_loader: dataloader to use in calculating m-IGS\n",
    "    # L: considered top eigenvalues of the FIM\n",
    "    # V: considered top eigenvectors of the FIM\n",
    "    # criterion: the loss function\n",
    "    # eps (float): regularization term for pseudoinverse of the FIM\n",
    "    # num_trials (int): the number of trials to calculate m-IGS (random for different shuffling of mini-batches)\n",
    "    # expand (bool): to provide more outputs or not\n",
    "    \n",
    "    dims = len(L)\n",
    "    IGS_dims_set = []\n",
    "    avg_grad_info_sqnorm_dims_set = []\n",
    "    trC_set = []\n",
    "    avg_grad_sqnorm_set = []\n",
    "    for t in range(num_trials):\n",
    "        tempIGS_dims = []\n",
    "        tempavg_grad_info_sqnorm_dims = []\n",
    "        trC = 0\n",
    "        for i in range(dims):\n",
    "            tempIGS_dims.append(0)\n",
    "            tempavg_grad_info_sqnorm_dims.append(0)\n",
    "\n",
    "        # compute IGSs for each mini-batch and get their average \n",
    "        for k, (X, y) in enumerate(train_loader):\n",
    "            model.zero_grad()\n",
    "            X = X.cuda()\n",
    "            y = y.cuda()\n",
    "            output = model(X)\n",
    "            loss = criterion(output, y)\n",
    "            loss.backward()\n",
    "            J = vectorize([gather_grads(model)])\n",
    "            if k == 0:\n",
    "                g = torch.zeros_like(J)\n",
    "            g += J * X.shape[0]\n",
    "            trC += (J**2).sum().item() * X.shape[0]\n",
    "            # save norms along eigenvectors for each eigenvector\n",
    "            for i in range(dims):\n",
    "                Jv = (J*V[i]).sum()\n",
    "                temp = (Jv*Jv/(L[i]+eps)).sum().item() * X.shape[0]\n",
    "                tempIGS_dims[i] += temp\n",
    "                \n",
    "        g /= len(train_loader.dataset)\n",
    "        for i in range(dims):\n",
    "            gv = (g*V[i]).sum()\n",
    "            temp3 = (gv*gv/(L[i]+eps)).sum().item()\n",
    "            tempavg_grad_info_sqnorm_dims[i] += temp3\n",
    "            \n",
    "        # accumulate norms along each eigenvectors\n",
    "        IGS_dims = []\n",
    "        avg_grad_info_sqnorm_dims = []\n",
    "        for i in range(dims):\n",
    "            if i == 0:\n",
    "                IGS_dims.append(tempIGS_dims[i]/len(train_loader.dataset))\n",
    "                avg_grad_info_sqnorm_dims.append(tempavg_grad_info_sqnorm_dims[i])\n",
    "            else:\n",
    "                IGS_dims.append(tempIGS_dims[i]/len(train_loader.dataset) + IGS_dims[-1])\n",
    "                avg_grad_info_sqnorm_dims.append(tempavg_grad_info_sqnorm_dims[i] + avg_grad_info_sqnorm_dims[-1])\n",
    "        IGS_dims_set.append(IGS_dims)\n",
    "        avg_grad_info_sqnorm_dims_set.append(avg_grad_info_sqnorm_dims)\n",
    "        trC_set.append(trC/len(train_loader.dataset))\n",
    "        avg_grad_sqnorm_set.append(g.norm().item()**2)\n",
    "    if expand:\n",
    "        return IGS_dims_set, trC_set, avg_grad_sqnorm_set, avg_grad_info_sqnorm_dims_set\n",
    "    return IGS_dims_set, trC_set\n",
    "\n",
    "def get_m_IGS_for_model(model, Xeig, yeig, bs_set, loader_dict_mIGS, criterion, tol, top_n, \n",
    "                        exact_fisher=False, expand=False):\n",
    "    # model: the model to calculate m-IGS on\n",
    "    # Xeig, yeig: data and label to calculate the FIM and m-IGS for\n",
    "    # bs_set: list of batch sizes to consider\n",
    "    # loader_dict_mIGS: dictionary of dataloaders to calculate m-IGS\n",
    "    # criterion: the loss function\n",
    "    # tol: tolerance for the power iterations to get eigenvalues/eigenvectors of the FIM\n",
    "    # top_n: the number of top eigenvalues to obtain from the power iterations\n",
    "    # exact_fisher (bool): to calculate the exact FIM or not\n",
    "    # expand (bool): to provide more outputs or not\n",
    "    \n",
    "    # calculate IGS and get top eigenvalues/eigenvectors of the FIM\n",
    "    start_time = time.time()\n",
    "    IGS_dims = []\n",
    "    while len(IGS_dims) == 0:\n",
    "        IGS_dims, L_F, V_F, spurious_dim_F, trF, trC, trFC_dims, \\\n",
    "        avg_grad_info_sqnorm_dims, L_F_gram, L_C_gram = calculate_IGS_largemodel(model, Xeig, Xeig, yeig, \n",
    "                                                                criterion, tol, top_n, exact_fisher=exact_fisher, eps=0,\n",
    "                                                                     expand=True)\n",
    "    print('time for IGS: ', time.time() - start_time)\n",
    "    \n",
    "    # calculate m-IGS\n",
    "    start_time = time.time()\n",
    "    m_IGS_dims_bs = []\n",
    "    trC_bs = []\n",
    "    avg_grad_sqnorm_bs = []\n",
    "    m_avg_grad_info_sqnorm_bs = []\n",
    "    dims = len(IGS_dims)\n",
    "    for bs in bs_set:\n",
    "        temp_start_time = time.time()\n",
    "        cur_m_IGS_dims, cur_trC_set, cur_avg_grad_sqnorm_set, cur_avg_grad_info_sqnorm_dims_set \\\n",
    "        = calculate_m_IGS2(model, loader_dict_mIGS[str(bs)], \n",
    "                                         L_F[:dims], V_F[:dims], criterion, num_trials=1, expand=True)\n",
    "        m_IGS_dims_bs.append(cur_m_IGS_dims)\n",
    "        m_avg_grad_info_sqnorm_bs.append(cur_avg_grad_info_sqnorm_dims_set)\n",
    "        trC_bs.append(cur_trC_set)\n",
    "        avg_grad_sqnorm_bs.append(cur_avg_grad_sqnorm_set)\n",
    "        print('bs = ', bs, ', time: ', time.time() - temp_start_time)\n",
    "    print('time for m-IGS: ', time.time() - start_time)\n",
    "    if expand:\n",
    "        return L_F, L_F_gram, m_IGS_dims_bs, m_avg_grad_info_sqnorm_bs, trC_bs, avg_grad_sqnorm_bs, trF\n",
    "    return L_F, L_F_gram, m_IGS_dims_bs, m_avg_grad_info_sqnorm_bs "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "2"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(file_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "startidx = 0\n",
    "endidx = len(file_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--------- filename = ./trained_model_cor_exp_MNIST/mnist_lenet_SGD_baseline_lr0.05_mom0.0_wd0.0_bs128_seed1_20220415153031_loss0.01.pth ---------\n",
      "time for loss and acc:  3.8975882530212402\n",
      "time for hessian measures:  0.3713209629058838\n",
      "time for IGS:  9.202679872512817\n",
      "bs =  128 , time:  3.6839704513549805\n",
      "time for m-IGS:  3.6842262744903564\n",
      "time for hessian measures:  0.15859365463256836\n",
      "time for IGS:  7.776372194290161\n",
      "bs =  128 , time:  0.9998109340667725\n",
      "time for m-IGS:  1.0012741088867188\n",
      "--------- filename = ./trained_model_cor_exp_MNIST/mnist_lenet_SGD_baseline_lr0.002_mom0.0_wd0.0_bs1024_seed2_20220427065421_loss0.01_postepoch0.pth ---------\n",
      "time for loss and acc:  4.031113624572754\n",
      "time for hessian measures:  0.22839951515197754\n",
      "time for IGS:  7.863954544067383\n",
      "bs =  128 , time:  3.426474094390869\n",
      "time for m-IGS:  3.426726818084717\n",
      "time for hessian measures:  0.18496108055114746\n",
      "time for IGS:  7.8021461963653564\n",
      "bs =  128 , time:  1.0181488990783691\n",
      "time for m-IGS:  1.0185585021972656\n"
     ]
    }
   ],
   "source": [
    "for dataidx in range(1):\n",
    "    \n",
    "    for i, (Xeig, yeig) in enumerate(hessian_loader):\n",
    "        Xeig = Xeig.cuda()\n",
    "        yeig = yeig.cuda()\n",
    "        if i == dataidx:\n",
    "            break\n",
    "    \n",
    "    for i, (test_Xeig, test_yeig) in enumerate(test_hessian_loader):\n",
    "        test_Xeig = test_Xeig.cuda()\n",
    "        test_yeig = test_yeig.cuda()\n",
    "        if i == dataidx:\n",
    "            break\n",
    "    \n",
    "    train_loss_list4 = []\n",
    "    train_acc_list4 = []\n",
    "    test_loss_list4 = []\n",
    "    test_acc_list4 = []\n",
    "    \n",
    "    m_IGS_dims_bs_list4 = []\n",
    "    m_avg_grad_info_sqnorm_bs_list4 = []\n",
    "    L_F_list4 = []\n",
    "    L_F_gram_list4 = []\n",
    "    test_m_IGS_dims_bs_list4 = []\n",
    "    test_m_avg_grad_info_sqnorm_bs_list4 = []\n",
    "    test_L_F_list4 = []\n",
    "    test_L_F_gram_list4 = []\n",
    "\n",
    "    trC_bs_list4 = []\n",
    "    avg_grad_sqnorm_bs_list4 = []\n",
    "    test_trC_bs_list4 = []\n",
    "    test_avg_grad_sqnorm_bs_list4 = []\n",
    "\n",
    "    top_H_eigval_list4 = []\n",
    "    trace_H_list4 = []\n",
    "    trF_list4 = []\n",
    "    test_top_H_eigval_list4 = []\n",
    "    test_trace_H_list4 = []\n",
    "    test_trF_list4 = []\n",
    "\n",
    "    bs_set4 = [128]\n",
    "    tol=5e-5\n",
    "    top_n = 50\n",
    "    for i in range(startidx, endidx):\n",
    "\n",
    "        filename = file_list[i]\n",
    "        start_time = time.time()\n",
    "        print('--------- filename = {:s} ---------'.format(filename))\n",
    "        \n",
    "        state_dict = torch.load(filename)\n",
    "        remove_module_in_keys(state_dict)\n",
    "\n",
    "        model.load_state_dict(state_dict)\n",
    "        model.eval()\n",
    "        \n",
    "        cur_acc, avg_loss = test(model, train_loader, criterion)\n",
    "        train_loss_list4.append(avg_loss)\n",
    "        train_acc_list4.append(cur_acc)\n",
    "\n",
    "        cur_acc, avg_loss = test(model, test_loader, criterion)\n",
    "        test_loss_list4.append(avg_loss)\n",
    "        test_acc_list4.append(cur_acc)\n",
    "        print('time for loss and acc: ', time.time() - start_time)\n",
    "        \n",
    "\n",
    "        # calculate Hessian measures (for train data)\n",
    "        start_time = time.time()\n",
    "        top_n_h = 1\n",
    "        top_eigenvalues, top_eigenvector, traceH = get_hessian_measures(model, criterion, \n",
    "                                                                        Xeig, yeig, \n",
    "                                                                        top_n_h, tol=1e-2)\n",
    "        top_H_eigval_list4.append(top_eigenvalues[0])\n",
    "        trace_H_list4.append(np.array(traceH).mean())\n",
    "        print('time for hessian measures: ', time.time() - start_time)\n",
    "\n",
    "        # calculate m-IGS (for train data)\n",
    "        L_F, L_F_gram, m_IGS_dims_bs, m_avg_grad_info_sqnorm_bs, trC_bs, avg_grad_sqnorm_bs, trF \\\n",
    "        = get_m_IGS_for_model(model, Xeig, yeig, bs_set4, train_loader_dict_mIGS, criterion, tol, top_n, \n",
    "                            exact_fisher=False, expand=True)\n",
    "\n",
    "        L_F_list4.append(L_F)\n",
    "        L_F_gram_list4.append(L_F_gram)\n",
    "        m_IGS_dims_bs_list4.append(m_IGS_dims_bs)\n",
    "        m_avg_grad_info_sqnorm_bs_list4.append(m_avg_grad_info_sqnorm_bs)\n",
    "        trC_bs_list4.append(trC_bs)\n",
    "        avg_grad_sqnorm_bs_list4.append(avg_grad_sqnorm_bs)\n",
    "        trF_list4.append(trF)\n",
    "        \n",
    "        \n",
    "        # calculate Hessian measures (for test data)\n",
    "        start_time = time.time()\n",
    "        top_n_h = 1\n",
    "        top_eigenvalues, top_eigenvector, traceH = get_hessian_measures(model, criterion, \n",
    "                                                                        test_Xeig, test_yeig, \n",
    "                                                                        top_n_h, tol=1e-2)\n",
    "        test_top_H_eigval_list4.append(top_eigenvalues[0])\n",
    "        test_trace_H_list4.append(np.array(traceH).mean())\n",
    "        print('time for hessian measures: ', time.time() - start_time)\n",
    "\n",
    "        # calculate m-IGS (for test data)\n",
    "        test_L_F, test_L_F_gram, test_m_IGS_dims_bs, test_m_avg_grad_info_sqnorm_bs, \\\n",
    "        test_trC_bs, test_avg_grad_sqnorm_bs, test_trF \\\n",
    "        = get_m_IGS_for_model(model, test_Xeig, test_yeig, bs_set4, test_loader_dict_mIGS, criterion, tol, top_n, \n",
    "                            exact_fisher=False, expand=True)\n",
    "\n",
    "        test_L_F_list4.append(test_L_F)\n",
    "        test_L_F_gram_list4.append(test_L_F_gram)\n",
    "        test_m_IGS_dims_bs_list4.append(test_m_IGS_dims_bs)\n",
    "        test_m_avg_grad_info_sqnorm_bs_list4.append(test_m_avg_grad_info_sqnorm_bs)\n",
    "        test_trC_bs_list4.append(test_trC_bs)\n",
    "        test_avg_grad_sqnorm_bs_list4.append(test_avg_grad_sqnorm_bs)\n",
    "        test_trF_list4.append(test_trF)\n",
    "\n",
    "    res = {\n",
    "        'file_list': file_list,\n",
    "        'bs_set4': bs_set4,\n",
    "\n",
    "        \n",
    "        'train_loss_list4': train_loss_list4,\n",
    "        'train_acc_list4': train_acc_list4,\n",
    "        'test_loss_list4': test_loss_list4,\n",
    "        'test_acc_list4': test_acc_list4,\n",
    "        \n",
    "        # to calculate m-IGS\n",
    "        'L_F_list4': L_F_list4,\n",
    "        'L_F_gram_list4': L_F_gram_list4,\n",
    "        'm_IGS_dims_bs_list4': m_IGS_dims_bs_list4,\n",
    "        'm_avg_grad_info_sqnorm_bs_list4': m_avg_grad_info_sqnorm_bs_list4,\n",
    "        'test_L_F_list4': test_L_F_list4,\n",
    "        'test_L_F_gram_list4': test_L_F_gram_list4,\n",
    "        'test_m_IGS_dims_bs_list4': test_m_IGS_dims_bs_list4,\n",
    "        'test_m_avg_grad_info_sqnorm_bs_list4': test_m_avg_grad_info_sqnorm_bs_list4,\n",
    "\n",
    "        'trC_bs_list4': trC_bs_list4,\n",
    "        'avg_grad_sqnorm_bs_list4': avg_grad_sqnorm_bs_list4,\n",
    "        'test_trC_bs_list4': test_trC_bs_list4,\n",
    "        'test_avg_grad_sqnorm_bs_list4': test_avg_grad_sqnorm_bs_list4,\n",
    "\n",
    "        'top_H_eigval_list4': top_H_eigval_list4,\n",
    "        'trace_H_list4': trace_H_list4,\n",
    "        'trF_list4': trF_list4,\n",
    "        'test_top_H_eigval_list4': test_top_H_eigval_list4,\n",
    "        'test_trace_H_list4': test_trace_H_list4,\n",
    "        'test_trF_list4': test_trF_list4,\n",
    "\n",
    "    }\n",
    "\n",
    "\n",
    "    filename = save_folder+'numerics_MNIST_lenet_loss0.01_mIGS_'+str(startidx)+'_'+str(endidx)+'_dataidx'+str(dataidx)+'.pickle'\n",
    "    with open(filename, 'wb') as handle:\n",
    "        pickle.dump(res, handle, protocol=pickle.HIGHEST_PROTOCOL)\n",
    "    \n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Average results from different choices of subsamples for F"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Load precomputed results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "res_dataidx = []\n",
    "for j in range(5):\n",
    "    filename2 = save_folder+'numerics_MNIST_lenet_loss0.01_averaging_20220427_multi_mIGS_0_20_dataidx'+str(j)+'.pickle'\n",
    "    with open(filename2, 'rb') as handle:\n",
    "        cur_res2 = pickle.load(handle)\n",
    "        \n",
    "    filename = save_folder+'numerics_MNIST_lenet_loss0.01_averaging_20220415_multi_mIGS_0_60_dataidx'+str(j)+'.pickle'\n",
    "    with open(filename, 'rb') as handle:\n",
    "        cur_res = pickle.load(handle)\n",
    "    \n",
    "    for key in cur_res2.keys():\n",
    "        cur_res2[key] += cur_res[key]\n",
    "        \n",
    "    res_dataidx.append(cur_res2)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "def gather_m_IGS_vals_trF(m_IGS_dims_bs_list, L_F_gram_list, trF_threshold):\n",
    "    # set the number of eigenvalues to consider in calculating m-IGS based on thresholding tr(F)\n",
    "    # m_IGS_dims_bs_list: multi-hierarchical list of accumulated m-IGS values with respect to the number of eigenvalues with list hierarchy size: (the number of files (or results) X the number of considered batch sizes for m-IGS X the number of trials)\n",
    "    # L_F_gram_list: list of the eigenvalues of the gram matrix of the FIM (in ascending order)\n",
    "    # trF_threshold (float): the ratio of the threshold to tr(F), e.g., 0.95\n",
    "    IGS_vals_bs_list = []\n",
    "    thres_violate_list = []\n",
    "    idx_list = []\n",
    "    # iteration for each file\n",
    "    for i in range(len(m_IGS_dims_bs_list)):\n",
    "        cur_IGS_vals_bs = []\n",
    "        #print(len(m_IGS_dims_bs_list[i][0][0]))\n",
    "        for j in range(len(m_IGS_dims_bs_list[i][0][0])):\n",
    "            if np.array(L_F_gram_list[i][::-1][:j]).sum() > trF_threshold*np.array(L_F_gram_list[i]).sum():\n",
    "                break\n",
    "            elif j == len(m_IGS_dims_bs_list[i][0][0])-1:\n",
    "                thres_violate_list.append(i)\n",
    "        idx = j\n",
    "        idx_list.append(j)\n",
    "        #print(i, idx)\n",
    "        # iteration for each batch size\n",
    "        for k in range(len(m_IGS_dims_bs_list[i])):\n",
    "            cur_IGS_vals_bs.append(m_IGS_dims_bs_list[i][k][0][idx]) # 0 is for the trial number\n",
    "        IGS_vals_bs_list.append(cur_IGS_vals_bs)\n",
    "        \n",
    "    return IGS_vals_bs_list, thres_violate_list, idx_list\n",
    "\n",
    "def get_top_L_F_lists(L_F_lists):\n",
    "    top_L_F_list = []\n",
    "    for L_F_list in L_F_lists:\n",
    "        top_L_F_list.append([L_F[0] for L_F in L_F_list])\n",
    "    return top_L_F_list\n",
    "\n",
    "def get_top_L_F_gram_lists(L_F_gram_lists):\n",
    "    top_L_F_gram_list = []\n",
    "    for L_F_gram_list in L_F_gram_lists:\n",
    "        top_L_F_gram_list.append([L_F[-1] for L_F in L_F_gram_list])\n",
    "    return top_L_F_gram_list\n",
    "\n",
    "top_L_F_np_dataidx = []\n",
    "top_test_L_F_np_dataidx = []\n",
    "\n",
    "trF_np_dataidx = []\n",
    "test_trF_np_dataidx = []\n",
    "\n",
    "IGS_vals_bs_list_np_dataidx = []\n",
    "corrected_IGS_vals_bs_list_np_dataidx = []\n",
    "test_IGS_vals_bs_list_np_dataidx = []\n",
    "corrected_test_IGS_vals_bs_list_np_dataidx = []\n",
    "\n",
    "traceH_lists_np_dataidx = []\n",
    "test_traceH_lists_np_dataidx = []\n",
    "\n",
    "trF_threshold = 0.95\n",
    "\n",
    "for j in range(5):\n",
    "    \n",
    "    top_L_F_dataidx = [L_F[0] for L_F in res_dataidx[j]['L_F_list4']]\n",
    "    top_test_L_F_dataidx = [L_F[0] for L_F in res_dataidx[j]['test_L_F_list4']]\n",
    "    \n",
    "    trF_np_dataidx.append(np.array(res_dataidx[j]['L_F_gram_list4']).sum(axis=1))\n",
    "    test_trF_np_dataidx.append(np.array(res_dataidx[j]['test_L_F_gram_list4']).sum(axis=1))\n",
    "    \n",
    "    cur_m_IGS_dims_bs_list4 = res_dataidx[j]['m_IGS_dims_bs_list4']\n",
    "    cur_m_avg_grad_info_sqnorm_bs_list4 = res_dataidx[j]['m_avg_grad_info_sqnorm_bs_list4']\n",
    "    cur_L_F_gram_list4 = res_dataidx[j]['L_F_gram_list4']\n",
    "    cur_test_m_IGS_dims_bs_list4 = res_dataidx[j]['test_m_IGS_dims_bs_list4']\n",
    "    cur_test_m_avg_grad_info_sqnorm_bs_list4 = res_dataidx[j]['test_m_avg_grad_info_sqnorm_bs_list4']\n",
    "    cur_test_L_F_gram_list4 = res_dataidx[j]['test_L_F_gram_list4']\n",
    "\n",
    "    traceH_lists_np_dataidx.append(np.array(res_dataidx[j]['trace_H_list4']))\n",
    "    test_traceH_lists_np_dataidx.append(np.array(res_dataidx[j]['test_trace_H_list4']))\n",
    "        \n",
    "    top_L_F_np_dataidx.append(np.array(top_L_F_dataidx))\n",
    "    top_test_L_F_np_dataidx.append(np.array(top_test_L_F_dataidx))\n",
    "        \n",
    "    # for averaged param\n",
    "    IGS_vals_bs_list, thres_violate_list2, idx_list2 \\\n",
    "    = gather_m_IGS_vals_trF(cur_m_IGS_dims_bs_list4, cur_L_F_gram_list4, trF_threshold)\n",
    "    avg_grad_vals_bs_list, thres_violate_list2, idx_list2 \\\n",
    "    = gather_m_IGS_vals_trF(cur_m_avg_grad_info_sqnorm_bs_list4, cur_L_F_gram_list4, trF_threshold)\n",
    "    test_IGS_vals_bs_list, test_thres_violate_list2, test_idx_list2 \\\n",
    "    = gather_m_IGS_vals_trF(cur_test_m_IGS_dims_bs_list4, cur_test_L_F_gram_list4, trF_threshold)\n",
    "    test_avg_grad_vals_bs_list, test_thres_violate_list2, test_idx_list2 \\\n",
    "    = gather_m_IGS_vals_trF(cur_test_m_avg_grad_info_sqnorm_bs_list4, cur_test_L_F_gram_list4, trF_threshold)\n",
    "\n",
    "    \n",
    "    IGS_vals_bs_list_np = np.array(IGS_vals_bs_list)\n",
    "    corrected_IGS_vals_bs_list_np = np.array(IGS_vals_bs_list) - np.array(avg_grad_vals_bs_list)\n",
    "\n",
    "    test_IGS_vals_bs_list_np = np.array(test_IGS_vals_bs_list)\n",
    "    corrected_test_IGS_vals_bs_list_np = np.array(test_IGS_vals_bs_list) - np.array(test_avg_grad_vals_bs_list)\n",
    "      \n",
    "    IGS_vals_bs_list_np_dataidx.append(IGS_vals_bs_list_np)\n",
    "    corrected_IGS_vals_bs_list_np_dataidx.append(corrected_IGS_vals_bs_list_np)\n",
    "    test_IGS_vals_bs_list_np_dataidx.append(test_IGS_vals_bs_list_np)\n",
    "    corrected_test_IGS_vals_bs_list_np_dataidx.append(corrected_test_IGS_vals_bs_list_np)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "bs_list = []\n",
    "lr_list = []\n",
    "\n",
    "for filename in res_dataidx[0]['file_list']:\n",
    "    lr_list.append(float(filename.split('lr')[1].split('_')[0]))\n",
    "    bs_list.append(int(filename.split('bs')[1].split('_')[0]))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Scatter plots for generalization gap vs. IGS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "60\n",
      "0.5782834564815124 0.6605987162958026\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.5782834564815124 0.6605987162958026\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.6602642155804493\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plot_idx = np.array(lr_list) >= 0.002\n",
    "print(plot_idx.sum())\n",
    "\n",
    "g_gap_list_np = np.array(res_dataidx[0]['test_loss_list4']) - np.array(res_dataidx[0]['train_loss_list4'])\n",
    "\n",
    "IGS_vals_bs_list_np_dataidx_mean = np.array(IGS_vals_bs_list_np_dataidx).mean(axis=0)[:,0]\n",
    "test_IGS_vals_bs_list_np_dataidx_mean = np.array(test_IGS_vals_bs_list_np_dataidx).mean(axis=0)[:,0]\n",
    "\n",
    "corrected_IGS_vals_bs_list_np_dataidx_mean = np.array(corrected_IGS_vals_bs_list_np_dataidx).mean(axis=0)[:,0]\n",
    "corrected_test_IGS_vals_bs_list_np_dataidx_mean = np.array(corrected_test_IGS_vals_bs_list_np_dataidx).mean(axis=0)[:,0]\n",
    "\n",
    "traceH_list_np_dataidx_mean = np.array(traceH_lists_np_dataidx).mean(axis=0)\n",
    "test_traceH_list_np_dataidx_mean = np.array(test_traceH_lists_np_dataidx).mean(axis=0)\n",
    "\n",
    "plt.figure()\n",
    "plt.scatter(g_gap_list_np[plot_idx], test_IGS_vals_bs_list_np_dataidx_mean[plot_idx], label='IGS_b')\n",
    "plt.scatter(g_gap_list_np[plot_idx], corrected_test_IGS_vals_bs_list_np_dataidx_mean[plot_idx], label='IGS_b-C')\n",
    "plt.xlabel('Generalization gap', fontsize=18)\n",
    "plt.ylabel('IGS_b', fontsize=18)\n",
    "print(np.corrcoef(g_gap_list_np[plot_idx], test_IGS_vals_bs_list_np_dataidx_mean[plot_idx])[0,1],\n",
    "     np.corrcoef(g_gap_list_np[plot_idx], corrected_test_IGS_vals_bs_list_np_dataidx_mean[plot_idx])[0,1])\n",
    "plt.show()\n",
    "\n",
    "\n",
    "# convert m-IGS to IGS\n",
    "Ntest = 10000\n",
    "gamma = (Ntest-128)/(Ntest-1)\n",
    "ratio = 128/gamma\n",
    "\n",
    "test_IGS_vals_mean = corrected_test_IGS_vals_bs_list_np_dataidx_mean * ratio + np.array(test_avg_grad_vals_bs_list)[:,0]\n",
    "corrected_test_IGS_vals_mean = corrected_test_IGS_vals_bs_list_np_dataidx_mean * ratio\n",
    "plt.figure()\n",
    "plt.scatter(g_gap_list_np[plot_idx], test_IGS_vals_mean[plot_idx], label='IGS')\n",
    "plt.scatter(g_gap_list_np[plot_idx], corrected_test_IGS_vals_mean[plot_idx], label='IGS-C')\n",
    "plt.xlabel('Generalization gap', fontsize=18)\n",
    "plt.ylabel('IGS', fontsize=18)\n",
    "print(np.corrcoef(g_gap_list_np[plot_idx], test_IGS_vals_bs_list_np_dataidx_mean[plot_idx])[0,1],\n",
    "     np.corrcoef(g_gap_list_np[plot_idx], corrected_test_IGS_vals_bs_list_np_dataidx_mean[plot_idx])[0,1])\n",
    "plt.show()\n",
    "\n",
    "plt.figure()\n",
    "plt.scatter(g_gap_list_np[plot_idx], test_IGS_vals_mean[plot_idx], c='k')\n",
    "plt.xlabel('Generalization gap', fontsize=18)\n",
    "plt.ylabel('IGS', fontsize=18)\n",
    "print(np.corrcoef(g_gap_list_np[plot_idx], test_IGS_vals_mean[plot_idx])[0,1])\n",
    "plt.xticks([0.0275,  0.0350,  0.0425], fontsize=18)\n",
    "plt.yticks(fontsize=18)\n",
    "#plt.savefig('./figs_for_paper/IGS_gen_gap_MNIST.pdf', format='pdf', bbox_inches='tight')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "tags": []
   },
   "source": [
    "#### Scatter plots for generalization gap vs. tr(H)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.6080519255063889\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure()\n",
    "plt.scatter(g_gap_list_np[plot_idx], traceH_list_np_dataidx_mean[plot_idx], c='k')\n",
    "plt.ylabel(r'Tr($H$)', fontsize=18)\n",
    "plt.xlabel('generalization gap', fontsize=18)\n",
    "plt.xticks([0.0275,  0.0350,  0.0425], fontsize=18)\n",
    "plt.yticks(fontsize=18)\n",
    "#plt.legend()\n",
    "print(np.corrcoef(g_gap_list_np[plot_idx], traceH_list_np_dataidx_mean[plot_idx])[0,1])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.6.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
