{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Import Libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import torch\n",
    "import torch.autograd as autograd         \n",
    "from torch import Tensor                 \n",
    "import torch.nn as nn                    \n",
    "import torch.optim as optim             \n",
    "import time\n",
    "from pyDOE import lhs  \n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.ticker\n",
    "\n",
    "#Set default dtype to float32\n",
    "torch.set_default_dtype(torch.float)\n",
    "\n",
    "#PyTorch random number generator\n",
    "torch.manual_seed(1234)\n",
    "\n",
    "# Random number generators in other libraries\n",
    "np.random.seed(1234)\n",
    "\n",
    "# Device configuration\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "\n",
    "print(device)\n",
    "\n",
    "if device == 'cuda': print(torch.cuda.get_device_name()) "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# *Data Prep*\n",
    "\n",
    "Training and Testing data is prepared from the solution file"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6603b176",
   "metadata": {},
   "outputs": [],
   "source": [
    "N_f = 10000  \n",
    "N_b = 1000   \n",
    "\n",
    "theta = 2 * np.pi * np.random.rand(N_f)\n",
    "r = np.sqrt(np.random.rand(N_f))\n",
    "x_f = r * np.cos(theta)\n",
    "y_f = r * np.sin(theta)\n",
    "X_f = np.stack([x_f, y_f], axis=1)\n",
    "\n",
    "theta_b = 2 * np.pi * np.random.rand(N_b)\n",
    "x_b = np.cos(theta_b)\n",
    "y_b = np.sin(theta_b)\n",
    "X_b = np.stack([x_b, y_b], axis=1)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Test Data\n",
    "\n",
    "We prepare the test data to compare against the solution produced by the PINN."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cb125bf4",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "N_mesh = 201   \n",
    "x = np.linspace(-1, 1, N_mesh)\n",
    "y = np.linspace(-1, 1, N_mesh)\n",
    "X, Y = np.meshgrid(x, y)\n",
    "\n",
    "X_flat, Y_flat = X.flatten(), Y.flatten()\n",
    "mask = X_flat**2 + Y_flat**2 <= 1\n",
    "X_in = X_flat[mask]\n",
    "Y_in = Y_flat[mask]\n",
    "\n",
    "X_u_test = np.hstack((\n",
    "    X_in[:, None],   \n",
    "    Y_in[:, None]\n",
    "))\n",
    "\n",
    "u_true = (1 - X_in**2 - Y_in**2)[:, None]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "12930ed0",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "def trainingdata(N_u, N_f):\n",
    "   \n",
    "    theta_b = 2 * np.pi * np.random.rand(N_u)\n",
    "    x_b = np.cos(theta_b)\n",
    "    y_b = np.sin(theta_b)\n",
    "    X_u_train = np.hstack([x_b[:, None], y_b[:, None]])    \n",
    "    u_train = np.zeros_like(x_b)[:, None]                  #\n",
    "\n",
    "    X_f = []\n",
    "    count = 0\n",
    "    while count < N_f:\n",
    "        \n",
    "        pts = 2 * lhs(2, N_f) - 1     \n",
    "        mask = np.sum(pts**2, axis=1) < 1\n",
    "        pts_in = pts[mask]\n",
    "        X_f.append(pts_in)\n",
    "        count += pts_in.shape[0]\n",
    "    X_f_train = np.vstack(X_f)[:N_f]   \n",
    "\n",
    "    X_f_train_all = np.vstack([X_f_train, X_u_train])\n",
    "\n",
    "    return X_f_train_all, X_u_train, u_train"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Physics Informed Neural Network"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Sequentialmodel(nn.Module):\n",
    "    \n",
    "    def __init__(self, layers):\n",
    "        super().__init__()\n",
    "    \n",
    "        assert len(layers) == 3, \"Layers must follow [input_dim, m, output_dim] structure\"\n",
    "    \n",
    "        self.activation = nn.Tanh()\n",
    "    \n",
    "        self.loss_function = nn.MSELoss(reduction='mean')\n",
    "    \n",
    "        self.w_layer = nn.Linear(layers[0], layers[1], bias=False)\n",
    "    \n",
    "        self.a_layer = nn.Linear(layers[1], layers[2], bias=False)\n",
    "    \n",
    "        # Use Xavier/Glorot initialization suitable for Tanh activation\n",
    "        with torch.no_grad():\n",
    "            nn.init.xavier_normal_(self.w_layer.weight, gain=nn.init.calculate_gain('tanh'))\n",
    "            # For the output weights (a_layer) use Xavier uniform\n",
    "            nn.init.xavier_uniform_(self.a_layer.weight)\n",
    "\n",
    "        '''\n",
    "        with torch.no_grad():\n",
    "            # LeCun Normal initialization for the hidden layer\n",
    "            fan_in = self.w_layer.weight.size(1)\n",
    "            nn.init.normal_(self.w_layer.weight, mean=0.0, std=np.sqrt(1.0 / fan_in))\n",
    "            \n",
    "            # LeCun Normal initialization for the output layer\n",
    "            fan_in_out = self.a_layer.weight.size(1)\n",
    "            nn.init.normal_(self.a_layer.weight, mean=0.0, std=np.sqrt(1.0 / fan_in_out))\n",
    "        '''\n",
    "        \n",
    "        ''' \n",
    "        self.w_layer.weight.data.normal_(mean=0.0, std=1.0)  \n",
    "\n",
    "        self.a_layer.weight.data.uniform_(-1.0, 1.0)  \n",
    "        '''\n",
    "\n",
    "        self.iter = 0\n",
    "\n",
    "    def forward(self, x):\n",
    "     \n",
    "        x = self.w_layer(x)        \n",
    "        x = self.activation(x)      \n",
    "        x = self.a_layer(x)                 \n",
    "        return x                   \n",
    "\n",
    "    def calc_source(self, x, y):\n",
    "        u_exact = 1 - x**2 - y**2\n",
    "        A = 1 + u_exact**2\n",
    "        \n",
    "        f = (\n",
    "            4 * A\n",
    "            - 8 * (x**2 + y**2) * (1 - x**2 - y**2)\n",
    "            + u_exact\n",
    "            + u_exact**3\n",
    "        )\n",
    "        return f\n",
    "\n",
    "    def loss(self, x_to_train_f):\n",
    "        x_f = x_to_train_f[:, [0]]\n",
    "        y_f = x_to_train_f[:, [1]]\n",
    "        \n",
    "        g = x_to_train_f.clone().detach()\n",
    "        g.requires_grad = True\n",
    "        u = self.forward(g)  \n",
    "\n",
    "        u_x_y = autograd.grad(u, g, torch.ones_like(u), retain_graph=True, create_graph=True)[0]\n",
    "        u_x = u_x_y[:, [0]]\n",
    "        u_y = u_x_y[:, [1]]\n",
    "\n",
    "        u_xx = autograd.grad(u_x, g, torch.ones_like(u_x), retain_graph=True, create_graph=True)[0][:, [0]]\n",
    "        u_yy = autograd.grad(u_y, g, torch.ones_like(u_y), retain_graph=True, create_graph=True)[0][:, [1]]\n",
    "\n",
    "        A = 1 + u**2\n",
    "        A_x = 2 * u * u_x\n",
    "        A_y = 2 * u * u_y\n",
    "\n",
    "        # -div((1+u^2)*grad u) = -[A_x * u_x + A * u_xx + A_y * u_y + A * u_yy]\n",
    "        div = (A_x * u_x + A * u_xx) + (A_y * u_y + A * u_yy)\n",
    "        div = div\n",
    "       \n",
    "        res = -div + u + u**3\n",
    "\n",
    "        f = self.calc_source(x_f, y_f)\n",
    "\n",
    "        pde_f = res - f\n",
    "\n",
    "        loss_f = self.loss_function(pde_f, torch.zeros_like(pde_f))\n",
    "        return loss_f"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b5f94d60",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "\n",
    "def assign_params(model, param_list):\n",
    "    for p, src in zip(model.parameters(), param_list):\n",
    "        p.data.copy_(src.data)\n",
    "\n",
    "\n",
    "def IGD_PINN(PINN, X_u_train, u_train, X_f_train, theta_init, K0, K1, gamma, eta):\n",
    "    print(\"IGD_PINN called\", flush=True)    \n",
    "    for p in PINN.w_layer.parameters():\n",
    "        p.requires_grad_(False)\n",
    "    for p in PINN.a_layer.parameters():\n",
    "        p.requires_grad_(True)\n",
    "    theta_n = [p.clone() for p in theta_init]\n",
    "    \n",
    "    n = 0\n",
    "    loss_list = []\n",
    "\n",
    "    while n < K0:\n",
    "        assign_params(PINN, theta_n)\n",
    "  \n",
    "        optimizer_inner = torch.optim.LBFGS(PINN.a_layer.parameters(), \n",
    "                                          lr=gamma,        \n",
    "                                          max_iter=K1,     \n",
    "                                          max_eval=K1*2, \n",
    "                                          history_size=100)\n",
    "        \n",
    "        def closure():\n",
    "            optimizer_inner.zero_grad()\n",
    "            \n",
    "            param_now = list(PINN.parameters())\n",
    "            loss_quad = 0.5 * sum([(a - b).pow(2).sum() for a, b in zip(param_now, theta_n)])\n",
    "            \n",
    "            loss_main = PINN.loss(X_f_train)\n",
    "            \n",
    "            total_loss = loss_quad + eta * loss_main\n",
    "            total_loss.backward()\n",
    "            return total_loss\n",
    "        \n",
    "        optimizer_inner.step(closure)\n",
    "        theta_n = [p.clone().detach() for p in PINN.parameters()]\n",
    "        loss = PINN.loss(X_f_train)\n",
    "\n",
    "        loss_current = loss.item()\n",
    "        loss_list.append(loss_current)\n",
    "    \n",
    "        if n % 1000 == 0:\n",
    "            loss = PINN.loss(X_f_train)\n",
    "            print(f'Iteration {n}, loss : {loss.item():.8e}')\n",
    "\n",
    "        n += 1\n",
    "\n",
    "    assign_params(PINN, theta_n)\n",
    "    return [p.clone().detach() for p in PINN.parameters()], loss_list"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Main"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sequentialmodel(\n",
      "  (activation): Tanh()\n",
      "  (loss_function): MSELoss()\n",
      "  (w_layer): Linear(in_features=2, out_features=1000, bias=False)\n",
      "  (a_layer): Linear(in_features=1000, out_features=1, bias=False)\n",
      ")\n",
      "IGD_PINN called\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "c:\\Users\\user\\anaconda3\\envs\\python_31015\\lib\\site-packages\\torch\\autograd\\graph.py:769: UserWarning: Attempting to run cuBLAS, but there was no current CUDA context! Attempting to set the primary context... (Triggered internally at C:\\actions-runner\\_work\\pytorch\\pytorch\\builder\\windows\\pytorch\\aten\\src\\ATen\\cuda\\CublasHandlePool.cpp:135.)\n",
      "  return Variable._execution_engine.run_backward(  # Calls into the C++ engine to run the backward pass\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Iteration 0, loss : 2.57438755e+01\n",
      "Iteration 100, loss : 2.57426281e+01\n",
      "Iteration 200, loss : 2.57426281e+01\n",
      "Iteration 300, loss : 2.57426243e+01\n",
      "Iteration 400, loss : 2.57426243e+01\n",
      "Iteration 500, loss : 2.57426243e+01\n",
      "Iteration 600, loss : 2.57426243e+01\n",
      "Iteration 700, loss : 2.57426243e+01\n",
      "Iteration 800, loss : 2.57426243e+01\n",
      "Iteration 900, loss : 2.57426243e+01\n",
      "Iteration 1000, loss : 2.57426243e+01\n",
      "Iteration 1100, loss : 2.57426243e+01\n",
      "Iteration 1200, loss : 2.57426243e+01\n",
      "Iteration 1300, loss : 2.57426243e+01\n",
      "Iteration 1400, loss : 2.57426243e+01\n",
      "Iteration 1500, loss : 2.57426224e+01\n",
      "Iteration 1600, loss : 2.57426224e+01\n",
      "Iteration 1700, loss : 2.57426224e+01\n",
      "Iteration 1800, loss : 2.57426224e+01\n",
      "Iteration 1900, loss : 2.57426224e+01\n",
      "Iteration 2000, loss : 2.57426224e+01\n",
      "Iteration 2100, loss : 2.57426224e+01\n",
      "Iteration 2200, loss : 2.57426186e+01\n",
      "Iteration 2300, loss : 2.57426186e+01\n",
      "Iteration 2400, loss : 2.57426186e+01\n",
      "Iteration 2500, loss : 2.57426186e+01\n",
      "Iteration 2600, loss : 2.57426186e+01\n",
      "Iteration 2700, loss : 2.57426186e+01\n",
      "Iteration 2800, loss : 2.57426186e+01\n",
      "Iteration 2900, loss : 2.57426186e+01\n",
      "Iteration 3000, loss : 2.57426186e+01\n",
      "Iteration 3100, loss : 2.57426186e+01\n",
      "Iteration 3200, loss : 2.57426186e+01\n",
      "Iteration 3300, loss : 2.57426186e+01\n",
      "Iteration 3400, loss : 2.57426186e+01\n",
      "Iteration 3500, loss : 2.57426186e+01\n",
      "Iteration 3600, loss : 2.57426186e+01\n",
      "Iteration 3700, loss : 2.57426167e+01\n",
      "Iteration 3800, loss : 2.57426167e+01\n",
      "Iteration 3900, loss : 2.57426167e+01\n",
      "Iteration 4000, loss : 2.57426167e+01\n",
      "Iteration 4100, loss : 2.57426167e+01\n",
      "Iteration 4200, loss : 2.57426167e+01\n",
      "Iteration 4300, loss : 2.57426128e+01\n",
      "Iteration 4400, loss : 2.57426128e+01\n",
      "Iteration 4500, loss : 2.57426128e+01\n",
      "Iteration 4600, loss : 2.57426128e+01\n",
      "Iteration 4700, loss : 2.57426128e+01\n",
      "Iteration 4800, loss : 2.57426128e+01\n",
      "Iteration 4900, loss : 2.57426128e+01\n",
      "Iteration 5000, loss : 2.57426128e+01\n",
      "Iteration 5100, loss : 2.57426128e+01\n",
      "Iteration 5200, loss : 2.57426128e+01\n",
      "Iteration 5300, loss : 2.57426128e+01\n",
      "Iteration 5400, loss : 2.57426128e+01\n",
      "Iteration 5500, loss : 2.57426128e+01\n",
      "Iteration 5600, loss : 2.57426090e+01\n",
      "Iteration 5700, loss : 2.57426090e+01\n",
      "Iteration 5800, loss : 2.57426090e+01\n",
      "Iteration 5900, loss : 2.57426090e+01\n",
      "Iteration 6000, loss : 2.57426090e+01\n",
      "Iteration 6100, loss : 2.57426090e+01\n",
      "Iteration 6200, loss : 2.57426090e+01\n",
      "Iteration 6300, loss : 2.57426090e+01\n",
      "Iteration 6400, loss : 2.57426071e+01\n",
      "Iteration 6500, loss : 2.57426071e+01\n",
      "Iteration 6600, loss : 2.57426071e+01\n",
      "Iteration 6700, loss : 2.57426071e+01\n",
      "Iteration 6800, loss : 2.57426071e+01\n",
      "Iteration 6900, loss : 2.57426071e+01\n",
      "Iteration 7000, loss : 2.57426071e+01\n",
      "Iteration 7100, loss : 2.57426071e+01\n",
      "Iteration 7200, loss : 2.57426071e+01\n",
      "Iteration 7300, loss : 2.57426071e+01\n",
      "Iteration 7400, loss : 2.57426071e+01\n",
      "Iteration 7500, loss : 2.57426071e+01\n",
      "Iteration 7600, loss : 2.57426071e+01\n",
      "Iteration 7700, loss : 2.57426033e+01\n",
      "Iteration 7800, loss : 2.57426033e+01\n",
      "Iteration 7900, loss : 2.57426033e+01\n",
      "Iteration 8000, loss : 2.57426033e+01\n",
      "Iteration 8100, loss : 2.57426014e+01\n",
      "Iteration 8200, loss : 2.57426014e+01\n",
      "Iteration 8300, loss : 2.57426014e+01\n",
      "Iteration 8400, loss : 2.57426014e+01\n",
      "Iteration 8500, loss : 2.57426014e+01\n",
      "Iteration 8600, loss : 2.57426014e+01\n",
      "Iteration 8700, loss : 2.57426014e+01\n",
      "Iteration 8800, loss : 2.57426014e+01\n",
      "Iteration 8900, loss : 2.57426014e+01\n",
      "Iteration 9000, loss : 2.57426014e+01\n",
      "Iteration 9100, loss : 2.57426014e+01\n",
      "Iteration 9200, loss : 2.57426014e+01\n",
      "Iteration 9300, loss : 2.57426014e+01\n",
      "Iteration 9400, loss : 2.57426014e+01\n",
      "Iteration 9500, loss : 2.57425976e+01\n",
      "Iteration 9600, loss : 2.57425976e+01\n",
      "Iteration 9700, loss : 2.57425976e+01\n",
      "Iteration 9800, loss : 2.57425957e+01\n",
      "Iteration 9900, loss : 2.57425957e+01\n"
     ]
    },
    {
     "data": {
      "image/png": "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p6eHpUuXws/PDwqFQtPu6emJ9PR0rfrvdOzYMTz44IOa93FxcQCAqKgopKSkAACmTZuGn3/+GW+88QZKS0vxwAMPYM+ePc0eFukJbt/+KMDZoomIiHRFJgS/t4Fadvvp4KqqKpiamnbZuENe24XGJoHDrwain5lRl41LRERE7f/81vk9gURERETU/RgCiYiIiCSIIZCIiIhIghgCiYiIiCSIIZCIiIhIghgCiYiIiCSIIZCIiIhIghgCqdvJINN1CURERJLHEEg6w2nKiYiIdIchkIiIiEiCGAKJiIiIJIghkIiIiEiCGAKJiIiIJIghkIiIiEiCGAKJiIiIJIghkIiIiEiCGAKJiIiIJIghkLrf/74whHNFExER6Q5DIBEREZEEMQQSERERSRBDIBEREZEEMQQSERERSRBDIBEREZEEMQQSERERSRBDIBEREZEEMQQSERERSRBDIHW7/80VDSE4XTQREZGuMAQSERERSRBDIBEREZEEMQQSERERSRBDIBEREZEEMQQSERERSRBDIBEREZEEMQQSERERSRBDIBEREZEEMQRSt5P9b7ZozhVNRESkOwyBRERERBLEEEhEREQkQQyBEjJlyhRYWFjg8ccf13UpREREpGMMgRISExODL774QtdlEBERUQ/AECghAQEBMDEx0XUZRERE1APoPAQmJCRg7NixMDExgY2NDSIiIlBQUNDmOosXL4ZMJtN6ubq6apY7Ozs3Wy6TyaBUKpuNtWzZMshkMsTGxmq1JyUlwcPDA6ampjA1NYWvry92797dbP2ffvoJTz31FKysrGBkZISRI0fi2LFjnTsYrThw4ADCw8Nhb28PmUyG7du3t9gvMTERzs7OMDQ0hI+PD44ePdqldRAREdEfh85DYFZWFpRKJY4cOQKVSoXGxkZMmjQJdXV1ba7n7u6OkpISzevgwYOaZdnZ2VrLVCoVAGDq1KlaY2RnZ2P16tXw8PBoNr6DgwOWLVuGnJwcHDt2DIGBgZg8eTJOnTql6XPt2jWMHz8evXr1wu7du3H69Gn885//hIWFRat1Hzp0CI2Njc3aT58+jbKyshbXqaurg6enJxITE1sdNzU1FXFxcYiPj0dubi48PT0REhKC8vLyVtchIiIiCRM9THl5uQAgsrKyWu0THx8vPD092z1mTEyMGDRokFCr1Zq2mpoaMWTIEKFSqYS/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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "N_u = 500 #Total number of data points for 'u'\n",
    "N_f = 10000 #Total number of collocation points \n",
    "\n",
    "# Training data\n",
    "X_f_train_np_array, X_u_train_np_array, u_train_np_array = trainingdata(N_u,N_f)\n",
    "\n",
    "'Convert to tensor and send to GPU'\n",
    "X_f_train = torch.from_numpy(X_f_train_np_array).float().to(device)\n",
    "X_u_train = torch.from_numpy(X_u_train_np_array).float().to(device)\n",
    "u_train = torch.from_numpy(u_train_np_array).float().to(device)\n",
    "X_u_test_tensor = torch.from_numpy(X_u_test).float().to(device)\n",
    "u = torch.from_numpy(u_true).float().to(device)\n",
    "f_hat = torch.zeros(X_f_train.shape[0],1).to(device)\n",
    "\n",
    "layers = np.array([2, 100, 1])\n",
    "\n",
    "PINN = Sequentialmodel(layers)\n",
    "       \n",
    "PINN.to(device)\n",
    "\n",
    "'Neural Network Summary'\n",
    "\n",
    "print(PINN)\n",
    "\n",
    "params = list(PINN.parameters())\n",
    "\n",
    "'''Optimization'''\n",
    "\n",
    "theta_init = [p.data.clone() for p in PINN.parameters()]\n",
    "K0 = 10000       \n",
    "K1 = 40       \n",
    "gamma = 0.1     \n",
    "eta = 0.5     \n",
    "\n",
    "final_params, loss_list = IGD_PINN(PINN, X_u_train, u_train, X_f_train, theta_init, K0, K1, gamma, eta)\n",
    "\n",
    "plt.plot(loss_list)\n",
    "plt.yscale('log')\n",
    "plt.xlabel('Iteration')\n",
    "plt.ylabel('Loss')\n",
    "plt.title('Loss (only train a_layer)')\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "c594ff78",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total grad norm: 2.4363e-04\n"
     ]
    }
   ],
   "source": [
    "\n",
    "PINN.zero_grad()\n",
    "loss = PINN.loss(X_f_train)\n",
    "loss.backward()\n",
    "\n",
    "total_norm = 0.0\n",
    "for param in PINN.parameters():\n",
    "    if param.grad is not None:\n",
    "        param_norm = param.grad.norm().item()\n",
    "        total_norm += param_norm ** 2\n",
    "total_norm = total_norm ** 0.5\n",
    "print(f\"Total grad norm: {total_norm:.4e}\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "python_31015",
   "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.10.15"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
