{
 "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 CIFAR data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "50000\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n"
     ]
    }
   ],
   "source": [
    "dataset = 'cifar10'\n",
    "model_name = 'lenet'\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 = model_name\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",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "LeNet5(\n",
      "  (conv1): Conv2d(3, 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: 62006\n"
     ]
    }
   ],
   "source": [
    "model = get_model(model_name,\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": [
      "1\n"
     ]
    }
   ],
   "source": [
    "loss_val = 0.1\n",
    "file_list = []\n",
    "bs_list = []\n",
    "lr_list = []\n",
    "save_folder = \"./trained_model_cor_exp_CIFAR10/\"\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 = 0\n",
    "endidx = len(file_list)\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": [
    "criterion = torch.nn.CrossEntropyLoss()\n",
    "criterion_alldata = torch.nn.CrossEntropyLoss(reduction = 'none')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "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": {
    "tags": []
   },
   "source": [
    "# Calculate m-IGS values"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### NOTE: Precomputed m-IGS values for some trained models are saved in 'numerics_lenet_*.pickle' in 'trained_model_cor_exp_CIFAR10' folder"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "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",
    "    \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",
    "            \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_m_avg_grad_info_sqnorm_dims \\\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_m_avg_grad_info_sqnorm_dims)\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, \\\n",
    "    trC_bs, avg_grad_sqnorm_bs, trF\n",
    "    return L_F, L_F_gram, m_IGS_dims_bs, m_avg_grad_info_sqnorm_bs \n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "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 = model_name\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": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--------- filename = ./trained_model_cor_exp_CIFAR10/cifar10_lenet_SGD_baseline_lr0.0001_mom0.9_wd0.0005_bs1024_seed1_20220413164644_loss0.1.pth ---------\n",
      "time for loss and acc:  5.531251907348633\n",
      "time for hessian measures:  0.3016073703765869\n",
      "time for IGS:  64.83677005767822\n",
      "bs =  128 , time:  7.571581602096558\n",
      "time for m-IGS:  7.5717902183532715\n",
      "time for hessian measures:  0.24822068214416504\n",
      "time for IGS:  64.75696897506714\n",
      "bs =  128 , time:  2.031064510345459\n",
      "time for m-IGS:  2.0316977500915527\n"
     ]
    }
   ],
   "source": [
    "tol=5e-5\n",
    "top_n = 1000\n",
    "bs_set4 = [128]\n",
    "\n",
    "for dataidx in range(0,1):\n",
    "    for i, (Xeig, yeig) in enumerate(hessian_loader):\n",
    "        Xeig = Xeig.cuda()\n",
    "        yeig = yeig.cuda()\n",
    "        if i == dataidx:\n",
    "            break\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",
    "    train_loss_list = []\n",
    "    test_loss_list = []\n",
    "    train_acc_list = []\n",
    "    test_acc_list = []\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",
    "    top_H_eigval_list4 = []\n",
    "    trace_H_list4 = []\n",
    "    trC_bs_list4 = []\n",
    "    avg_grad_sqnorm_bs_list4 = []\n",
    "    trF_list4 = []\n",
    "    test_top_H_eigval_list4 = []\n",
    "    test_trace_H_list4 = []\n",
    "    test_trC_bs_list4 = []\n",
    "    test_avg_grad_sqnorm_bs_list4 = []\n",
    "    test_trF_list4 = []\n",
    "    \n",
    "    \n",
    "\n",
    "    for i in range(startidx, endidx):\n",
    "        filename = file_list[i]\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",
    "        start_time = time.time()\n",
    "        cur_acc, avg_loss = test(model, train_loader, criterion)\n",
    "        train_loss_list.append(avg_loss)\n",
    "        train_acc_list.append(cur_acc)\n",
    "\n",
    "        cur_acc, avg_loss = test(model, test_loader, criterion)\n",
    "        test_loss_list.append(avg_loss)\n",
    "        test_acc_list.append(cur_acc)\n",
    "        print('time for loss and acc: ', time.time() - start_time)\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, \n",
    "                              tol, top_n, 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",
    "        \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",
    "        test_top_eigenvalues, test_top_eigenvector, test_traceH = get_hessian_measures(model, \n",
    "                                                                                       criterion, \n",
    "                                                                        test_Xeig, test_yeig, \n",
    "                                                                        top_n_h, tol=1e-2)\n",
    "        test_top_H_eigval_list4.append(test_top_eigenvalues[0])\n",
    "        test_trace_H_list4.append(np.array(test_traceH).mean())\n",
    "        \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, 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, \n",
    "                              criterion, tol, top_n, 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",
    "        \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",
    "        'bs_set4': bs_set4,\n",
    "        'train_loss_list': train_loss_list,\n",
    "        'test_loss_list': test_loss_list,\n",
    "        'train_acc_list': train_acc_list,\n",
    "        'test_acc_list': test_acc_list,\n",
    "        \n",
    "        'file_list_red': file_list_red,\n",
    "        'bs_list_red': bs_list_red,\n",
    "        'lr_list_red': lr_list_red,\n",
    "        \n",
    "        'top_H_eigval_list4': top_H_eigval_list4,\n",
    "        'trace_H_list4': trace_H_list4,\n",
    "        'test_top_H_eigval_list4': test_top_H_eigval_list4,\n",
    "        'test_trace_H_list4': test_trace_H_list4,\n",
    "\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",
    "        'L_F_list4': L_F_list4,\n",
    "        'L_F_gram_list4': 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",
    "        'test_L_F_list4': test_L_F_list4,\n",
    "        'test_L_F_gram_list4': test_L_F_gram_list4,\n",
    "        \n",
    "        'trC_bs_list4': trC_bs_list4,\n",
    "        'avg_grad_sqnorm_bs_list4': avg_grad_sqnorm_bs_list4,\n",
    "        'trF_list4': trF_list4,\n",
    "        'test_trC_bs_list4': test_trC_bs_list4,\n",
    "        'test_avg_grad_sqnorm_bs_list4': test_avg_grad_sqnorm_bs_list4,\n",
    "        'test_trF_list4': test_trF_list4,\n",
    "    }\n",
    "    filename = save_folder+'numerics_lenet_'+str(startidx)+'_'+str(endidx)+'_dataidx'+str(dataidx)+'.pickle'\n",
    "    with open(filename, 'wb') as handle:\n",
    "        pickle.dump(res, handle, protocol=pickle.HIGHEST_PROTOCOL)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "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": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "res_dataidx = []\n",
    "for dataidx in range(0,5):\n",
    "    filenames = []\n",
    "    filenames.append(save_folder+'numerics_lenet_0_8_dataidx'+str(dataidx)+'.pickle')\n",
    "    filenames.append(save_folder+'numerics_lenet_8_16_dataidx'+str(dataidx)+'.pickle')\n",
    "    filenames.append(save_folder+'numerics_lenet_16_25_dataidx'+str(dataidx)+'.pickle')\n",
    "    filenames.append(save_folder+'numerics_lenet_25_34_dataidx'+str(dataidx)+'.pickle')\n",
    "    filenames.append(save_folder+'numerics_lenet_34_42_dataidx'+str(dataidx)+'.pickle')\n",
    "    filenames.append(save_folder+'numerics_lenet_42_50_dataidx'+str(dataidx)+'.pickle')\n",
    "    filenames.append(save_folder+'numerics_lenet_50_58_dataidx'+str(dataidx)+'.pickle')\n",
    "    \n",
    "    for i, filename in enumerate(filenames):\n",
    "        if i == 0:\n",
    "            with open(filename, 'rb') as handle:\n",
    "                res_curidx = pickle.load(handle)\n",
    "        else:\n",
    "            with open(filename, 'rb') as handle:\n",
    "                res = pickle.load(handle)\n",
    "                \n",
    "            if dataidx == 0:\n",
    "                res_curidx['train_loss_list'] += res['train_loss_list']\n",
    "                res_curidx['test_loss_list'] += res['test_loss_list']\n",
    "                res_curidx['train_acc_list'] += res['train_acc_list']\n",
    "                res_curidx['test_acc_list'] += res['test_acc_list']\n",
    "                \n",
    "                res_curidx['file_list_red'] += res['file_list_red']\n",
    "                res_curidx['bs_list_red'] += res['bs_list_red']\n",
    "                res_curidx['lr_list_red'] += res['lr_list_red']\n",
    "            \n",
    "            res_curidx['top_H_eigval_list4'] += res['top_H_eigval_list4']\n",
    "            res_curidx['trace_H_list4'] += res['trace_H_list4']\n",
    "            res_curidx['test_top_H_eigval_list4'] += res['test_top_H_eigval_list4']\n",
    "            res_curidx['test_trace_H_list4'] += res['test_trace_H_list4']\n",
    "\n",
    "            res_curidx['m_IGS_dims_bs_list4'] += res['m_IGS_dims_bs_list4']\n",
    "            res_curidx['m_avg_grad_info_sqnorm_bs_list4'] += res['m_avg_grad_info_sqnorm_bs_list4']\n",
    "            res_curidx['L_F_list4'] += res['L_F_list4']\n",
    "            res_curidx['L_F_gram_list4'] += res['L_F_gram_list4']\n",
    "            res_curidx['test_m_IGS_dims_bs_list4'] += res['test_m_IGS_dims_bs_list4']\n",
    "            res_curidx['test_m_avg_grad_info_sqnorm_bs_list4'] += res['test_m_avg_grad_info_sqnorm_bs_list4']\n",
    "            res_curidx['test_L_F_list4'] += res['test_L_F_list4']\n",
    "            res_curidx['test_L_F_gram_list4'] += res['test_L_F_gram_list4']\n",
    "\n",
    "            res_curidx['trC_bs_list4'] += res['trC_bs_list4']\n",
    "            res_curidx['avg_grad_sqnorm_bs_list4'] += res['avg_grad_sqnorm_bs_list4']\n",
    "            res_curidx['trF_list4'] += res['trF_list4']\n",
    "            res_curidx['test_trC_bs_list4'] += res['test_trC_bs_list4']\n",
    "            res_curidx['test_avg_grad_sqnorm_bs_list4'] += res['test_avg_grad_sqnorm_bs_list4']\n",
    "            res_curidx['test_trF_list4'] += res['test_trF_list4']\n",
    "    res_dataidx.append(res_curidx)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "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",
    "    \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_dataidx = []\n",
    "#top_test_L_F_dataidx = []\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",
    "        \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",
    "    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",
    "    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": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "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_lists_np_dataidx_mean = np.array(traceH_lists_np_dataidx).mean(axis=0)\n",
    "test_traceH_lists_np_dataidx_mean = np.array(test_traceH_lists_np_dataidx).mean(axis=0)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "tags": []
   },
   "source": [
    "#### Scatter plots for generalization gap vs. IGS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.6226342217263964 0.6096647879656568\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.609793427521536 0.6096647879656568\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.609793427521536\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(res_dataidx[0]['lr_list_red']) >= 0.001\n",
    "#print(plot_idx)\n",
    "\n",
    "g_gap = np.array(res_dataidx[0]['test_loss_list']) - np.array(res_dataidx[0]['train_loss_list'])\n",
    "\n",
    "plt.figure()\n",
    "plt.scatter(g_gap[plot_idx], test_IGS_vals_bs_list_np_dataidx_mean[plot_idx], label='IGS_b')\n",
    "plt.scatter(g_gap[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",
    "plt.legend()\n",
    "print(np.corrcoef(g_gap[plot_idx], test_IGS_vals_bs_list_np_dataidx_mean[plot_idx])[0,1],\n",
    "     np.corrcoef(g_gap[plot_idx], corrected_test_IGS_vals_bs_list_np_dataidx_mean[plot_idx])[0,1])\n",
    "plt.show()\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[plot_idx], test_IGS_vals_mean[plot_idx], label='IGS')\n",
    "plt.scatter(g_gap[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",
    "plt.legend()\n",
    "print(np.corrcoef(g_gap[plot_idx], test_IGS_vals_mean[plot_idx])[0,1],\n",
    "     np.corrcoef(g_gap[plot_idx], corrected_test_IGS_vals_mean[plot_idx])[0,1])\n",
    "plt.show()\n",
    "\n",
    "print(np.corrcoef(g_gap[plot_idx], test_IGS_vals_mean[plot_idx])[0,1])\n",
    "plt.figure()\n",
    "plt.scatter(g_gap[plot_idx], test_IGS_vals_mean[plot_idx], c='k')\n",
    "plt.xlabel('Generalization gap', fontsize=18)\n",
    "plt.ylabel('IGS', fontsize=18)\n",
    "plt.xticks([2.5, 2.7, 2.9, 3.1], fontsize=18)\n",
    "plt.yticks(fontsize=18)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "tags": []
   },
   "source": [
    "#### Scatter plots for generalization gap vs. tr(H)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.543552453891878\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[plot_idx], traceH_lists_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([2.5, 2.7, 2.9, 3.1], fontsize=18)\n",
    "plt.yticks(fontsize=18)\n",
    "#plt.legend()\n",
    "print(np.corrcoef(g_gap[plot_idx], traceH_lists_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
}
