{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "import torch\n",
    "\n",
    "path = 'pkl/grad_norm_dict.pkl'\n",
    "\n",
    "with open(path, 'rb') as f:\n",
    "    grad_norm_dict = pickle.load(f) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "def find_top_sum(tensor, target_percentage):\n",
    "    sorted_values = torch.sort(tensor.view(-1), descending=True).values\n",
    "    target_sum = target_percentage * tensor.sum()\n",
    "\n",
    "    cumulative_sum = torch.cumsum(sorted_values, dim=0)\n",
    "    count = torch.sum(cumulative_sum < target_sum).item() + 1\n",
    "    return count"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "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": [
    "# for key in grad_norm_dict.keys():\n",
    "#     a = torch.concat(grad_norm_dict[key], dim=0)\n",
    "#     a.sort(dim=-1, descending=True)\n",
    "key = 100\n",
    "\n",
    "a = torch.vstack(grad_norm_dict[key])\n",
    "a = a / a.sum(dim=-1, keepdim=True)\n",
    "a_0 = a[11]\n",
    "index = a_0.argsort(descending=True)\n",
    "a_sorted = a[:,index]\n",
    "\n",
    "percentile = []\n",
    "for i in range(12):\n",
    "    percentile.append(find_top_sum(a[i], 0.95))\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "z = a_sorted.T.cpu().numpy()\n",
    "\n",
    "z = -np.log(z)\n",
    "\n",
    "plt.figure(figsize=(6, 4))\n",
    "plt.rcParams.update({'font.size': 24})\n",
    "\n",
    "plt.xlim(11, 0)\n",
    "plt.xticks(np.arange(0, 12, 2), np.arange(0, 12, 2))\n",
    "plt.xticks(np.arange(0, 12, 1))\n",
    "\n",
    "plt.ylim(0,31)\n",
    "plt.yticks(np.arange(0, 32, 5), np.arange(0, 32, 5))\n",
    "plt.yticks(np.arange(0, 32,1))\n",
    "\n",
    "plt.imshow(z, cmap='Oranges_r', interpolation='nearest', aspect=0.25, vmin=0, vmax=10)\n",
    "\n",
    "plt.xlabel('Layer Index')\n",
    "plt.ylabel('Datum Index')\n",
    "\n",
    "# cbar = plt.colorbar()\n",
    "# cbar.set_label('grad norm proportion', rotation=270, labelpad=15, fontsize=18)\n",
    "# # cbar.cmap.set_under('white')\n",
    "# cbar.set_ticks(np.arange(0, 11, 2))\n",
    "# cbar.set_ticklabels([1] + [f\"1e-{i}\" for i in range(2,11, 2)], fontsize=18)\n",
    "\n",
    "#add an arrow at the title position\n",
    "plt.annotate('Backpropagation', xy=(0.5, 1.125), xycoords='axes fraction', ha='center', va='center')\n",
    "plt.annotate('', xy=(0.1, 1.05), xycoords='axes fraction', xytext=(0.9, 1.05), textcoords='axes fraction', arrowprops=dict(arrowstyle='<-', color='black', lw=2))\n",
    "\n",
    "#plot percentile of grad norm as a short horizontal line from 0 to 11\n",
    "line = [None] * 12\n",
    "for i in range(12):\n",
    "    # plt.axhline(y=percentile[i], color='black', linestyle='-',xmin=1-i/12, xmax=1-(i+1)/12)\n",
    "    line[i] = plt.axhline(y=percentile[i], color='black', linestyle='-',xmin=1-i/12, xmax=1-(i+1)/12)\n",
    "\n",
    "# connect the horizontal lines\n",
    "# for i in range(11):\n",
    "#     first = line[i].get_xydata()[1]\n",
    "#     second = line[i+1].get_xydata()[0]\n",
    "#     if first[1] != second[1]:\n",
    "#         # plt.axline(line[i].get_xydata()[1], line[i+1].get_xydata()[0], color='black', linestyle='-')\n",
    "#         plt.axvline(x=12-first[0]*12, color='black', linestyle='-', ymin=first[1]/32, ymax=second[1]/32)\n",
    "\n",
    "# for i in range(11):\n",
    "#     plt.axvline(x=i+0.1, color='black', linestyle='-', ymin=percentile[i]/32+0.4/32, ymax=percentile[i+1]/32+0.4/32)\n",
    "\n",
    "# line the position of percentile\n",
    "# for i in range(12):\n",
    "#     plt.axhline(y=percentile[i], color='r', linestyle='-')\n",
    "# plt.colorbar()\n",
    "plt.savefig(f'norm_{key}.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([[ 1.        , 13.        ],\n",
       "        [ 0.91666667, 13.        ]]),\n",
       " array([[ 0.91666667, 13.        ],\n",
       "        [ 0.83333333, 13.        ]]))"
      ]
     },
     "execution_count": 104,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "line[0].get_xydata(), line[1].get_xydata()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "sample",
   "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.9.12"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
