{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/tqdm/auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n"
     ]
    }
   ],
   "source": [
    "import wandb\n",
    "import timm\n",
    "\n",
    "import torch\n",
    "from torchvision import datasets, transforms\n",
    "from torch.utils.data import DataLoader\n",
    "\n",
    "import Positional_Embeddings.AxialRoPE\n",
    "from model_registry import MODEL_REGISTRY\n",
    "\n",
    "dataset = 'CIFAR100'\n",
    "PE = 'Spherical RoPE'\n",
    "base_model = 'ViT-B'\n",
    "\n",
    "data_path = './data/'\n",
    "num_workers = 8\n",
    "batch_size = 1024\n",
    "weight_decay = .02\n",
    "dropout = 0.1\n",
    "cls_token=True\n",
    "lr = 3e-3\n",
    "num_epochs = 400\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Set Default Parameters\n",
    "ViT_Params = {\n",
    "                'patch_size' : 16,\n",
    "                'img_size' : 224,\n",
    "                'in_chans': 3,\n",
    "                'num_classes': 1000,\n",
    "                'embed_dim': 60*12, # 60 is divisible by both 4 and 3 \n",
    "                'depth': 12,\n",
    "                'num_heads': 12,\n",
    "                'global_pool' : '',\n",
    "                'class_token' : False,\n",
    "                'drop_rate' : dropout,\n",
    "                'drop_path_rate' : dropout,\n",
    "                'init_values' : 1.,\n",
    "                }\n",
    "if PE != 'Default':\n",
    "    ViT_Params['pos_embed'] = 'none'\n",
    "attn_args = {\n",
    "                'attn_drop' : 0.1,\n",
    "                'proj_drop' : 0.1,\n",
    "}\n",
    "if base_model == 'ViT-S':\n",
    "    ViT_Params['num_heads'] = 6\n",
    "    ViT_Params['embed_dim'] = 60*6 # 60 is divisible by both 4 and 3 \n",
    "if base_model == 'ViT-B':\n",
    "    ViT_Params['num_heads'] = 12\n",
    "    ViT_Params['embed_dim'] = 60*12 # 60 is divisible by both 4 and 3 \n",
    "\n",
    "kwargs = MODEL_REGISTRY[PE]\n",
    "kwargs['flash_att'] = True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model 'ImageNet' already registered!\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/torch/utils/data/dataloader.py:624: UserWarning: This DataLoader will create 8 worker processes in total. Our suggested max number of worker in current system is 1, which is smaller than what this DataLoader is going to create. Please be aware that excessive worker creation might get DataLoader running slow or even freeze, lower the worker number to avoid potential slowness/freeze if necessary.\n",
      "  warnings.warn(\n"
     ]
    }
   ],
   "source": [
    "import torchvision\n",
    "import torch\n",
    "from torchvision import datasets, transforms\n",
    "from torch.utils.data import DataLoader\n",
    "from Dataloader_funcs import *\n",
    "from Dataloader_funcs.DL_registry import DL_REGISTRY\n",
    "\n",
    "train_loader, val_loader, test_loader = DL_REGISTRY[dataset](data_path, batch_size=batch_size, num_workers=num_workers)\n",
    "\n",
    "\n",
    "if dataset == 'ImageNet':\n",
    "    classes = 1000\n",
    "    patch_size = 16\n",
    "elif dataset == 'ImageNette':\n",
    "    ViT_Params['num_classes']=10\n",
    "    ViT_Params['patch_size'] = 16\n",
    "elif dataset == 'CIFAR100':\n",
    "    ViT_Params['num_classes'] = 100\n",
    "    ViT_Params['patch_size'] = 4\n",
    "    ViT_Params['img_size'] = 32\n",
    "elif dataset == 'CIFAR10':\n",
    "    classes=10\n",
    "else:\n",
    "    # Load training and validation datasets\n",
    "    train_dataset = datasets.CIFAR10(root='~/data/CIFAR/train', train=True, download=True, transform=transform)\n",
    "    val_dataset = datasets.CIFAR10(root='~/data/CIFAR/valid', train=False, download=True, transform=transform)\n",
    "\n",
    "    # DataLoader for batching\n",
    "    train_loader = DataLoader(train_dataset, batch_size=batch_size, shuffle=True, num_workers=num_workers)\n",
    "    val_loader = DataLoader(val_dataset, batch_size=batch_size, shuffle=False, num_workers=num_workers)\n",
    "    classes=10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Clipping input data to the valid range for imshow with RGB data ([0..1] for floats or [0..255] for integers). Got range [-0.448713..1.512129].\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "def imshow(img):\n",
    "    # Convert tensor to numpy for displaying with matplotlib\n",
    "    img = img / 2 + 0.5  # Unnormalize\n",
    "    npimg = img.numpy()\n",
    "    plt.imshow(np.transpose(npimg, (1, 2, 0)))\n",
    "    plt.show()\n",
    "    \n",
    "# Get some random training images\n",
    "data_iter = iter(train_loader)\n",
    "images, labels = next(data_iter)\n",
    "\n",
    "# Show images\n",
    "imshow(torchvision.utils.make_grid(images))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/timm/models/layers/__init__.py:48: FutureWarning: Importing from timm.models.layers is deprecated, please import via timm.layers\n",
      "  warnings.warn(f\"Importing from {__name__} is deprecated, please import via timm.layers\", FutureWarning)\n"
     ]
    },
    {
     "ename": "AttributeError",
     "evalue": "'Spherical_RoPE' object has no attribute 'frequency_x'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mAttributeError\u001b[0m                            Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[6], line 7\u001b[0m\n\u001b[1;32m      5\u001b[0m kwargs[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mViT_kwargs\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;241m=\u001b[39m ViT_Params\n\u001b[1;32m      6\u001b[0m kwargs[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mattn_kwargs\u001b[39m\u001b[38;5;124m'\u001b[39m] \u001b[38;5;241m=\u001b[39m attn_args\n\u001b[0;32m----> 7\u001b[0m model \u001b[38;5;241m=\u001b[39m \u001b[43mCustomPE_ViT\u001b[49m\u001b[43m(\u001b[49m\u001b[43muse_cls_token\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mcls_token\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m      9\u001b[0m \u001b[38;5;28mprint\u001b[39m(ViT_Params[\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mnum_classes\u001b[39m\u001b[38;5;124m'\u001b[39m])\n\u001b[1;32m     10\u001b[0m \u001b[38;5;66;03m# Test with a dummy input\u001b[39;00m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/git/Spherical RoPE/model/custom_vit.py:49\u001b[0m, in \u001b[0;36mCustomPE_ViT.__init__\u001b[0;34m(self, ViT_kwargs, attn_kwargs, pretrained, PE_method, stem_only, num_heads, shared_pe, rot_x, rot_value, flash_att, use_cls_token, use_default)\u001b[0m\n\u001b[1;32m     47\u001b[0m     pe \u001b[38;5;241m=\u001b[39m PE_method(embed_dim\u001b[38;5;241m*\u001b[39mnum_heads, P_x\u001b[38;5;241m=\u001b[39mP_x,P_y\u001b[38;5;241m=\u001b[39mP_y)\n\u001b[1;32m     48\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m---> 49\u001b[0m     pe \u001b[38;5;241m=\u001b[39m \u001b[43mPE_method\u001b[49m\u001b[43m(\u001b[49m\u001b[43membed_dim\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mP_x\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mP_x\u001b[49m\u001b[43m,\u001b[49m\u001b[43mP_y\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mP_y\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     50\u001b[0m \u001b[38;5;66;03m# self.PE_method = pe\u001b[39;00m\n\u001b[1;32m     51\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mmodel\u001b[38;5;241m.\u001b[39mpos_embed \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/git/Spherical RoPE/Positional_Embeddings/SphericalRoPE.py:14\u001b[0m, in \u001b[0;36mSpherical_RoPE.__init__\u001b[0;34m(self, embedding_dim, P_x, P_y, xy_range)\u001b[0m\n\u001b[1;32m     12\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mregister_buffer(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mtheta_d\u001b[39m\u001b[38;5;124m'\u001b[39m, theta_d) \u001b[38;5;66;03m# [1, D/3]\u001b[39;00m\n\u001b[1;32m     13\u001b[0m \u001b[38;5;66;03m# self._init_rotation_matrices_(P_x, P_y, D)\u001b[39;00m\n\u001b[0;32m---> 14\u001b[0m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_init_sin_cos\u001b[49m\u001b[43m(\u001b[49m\u001b[43mP_x\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mP_y\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mD\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     15\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mfrequency_x \u001b[38;5;241m=\u001b[39m theta_d \u001b[38;5;241m*\u001b[39m torch\u001b[38;5;241m.\u001b[39marange(\u001b[38;5;241m0\u001b[39m, P_x)\u001b[38;5;241m.\u001b[39munsqueeze(\u001b[38;5;241m1\u001b[39m)  \u001b[38;5;66;03m# [P_x, 1]\u001b[39;00m\n\u001b[1;32m     16\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mfrequency_y \u001b[38;5;241m=\u001b[39m theta_d \u001b[38;5;241m*\u001b[39m torch\u001b[38;5;241m.\u001b[39marange(\u001b[38;5;241m0\u001b[39m, P_y)\u001b[38;5;241m.\u001b[39munsqueeze(\u001b[38;5;241m1\u001b[39m)\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/git/Spherical RoPE/Positional_Embeddings/SphericalRoPE.py:72\u001b[0m, in \u001b[0;36mSpherical_RoPE._init_sin_cos\u001b[0;34m(self, P_x, P_y, D)\u001b[0m\n\u001b[1;32m     65\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21m_init_sin_cos\u001b[39m(\u001b[38;5;28mself\u001b[39m, P_x, P_y, D):\n\u001b[1;32m     66\u001b[0m   \u001b[38;5;66;03m# P_x : Number of patches in the X direction\u001b[39;00m\n\u001b[1;32m     67\u001b[0m   \u001b[38;5;66;03m# P_y : Number of patches in the Y direction\u001b[39;00m\n\u001b[1;32m     68\u001b[0m   \u001b[38;5;66;03m# D : Hidden/Embedding dimensions\u001b[39;00m\n\u001b[1;32m     70\u001b[0m   \u001b[38;5;28;01massert\u001b[39;00m D \u001b[38;5;241m%\u001b[39m \u001b[38;5;241m3\u001b[39m \u001b[38;5;241m==\u001b[39m \u001b[38;5;241m0\u001b[39m, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mD must be divisible by 3\u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[0;32m---> 72\u001b[0m   thetas_x \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mmatmul(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mi, \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mfrequency_x\u001b[49m)\n\u001b[1;32m     73\u001b[0m   \u001b[38;5;66;03m# P_x x D/3\u001b[39;00m\n\u001b[1;32m     74\u001b[0m   thetas_x \u001b[38;5;241m=\u001b[39m thetas_x\u001b[38;5;241m.\u001b[39mflatten()\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/torch/nn/modules/module.py:1928\u001b[0m, in \u001b[0;36mModule.__getattr__\u001b[0;34m(self, name)\u001b[0m\n\u001b[1;32m   1926\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m name \u001b[38;5;129;01min\u001b[39;00m modules:\n\u001b[1;32m   1927\u001b[0m         \u001b[38;5;28;01mreturn\u001b[39;00m modules[name]\n\u001b[0;32m-> 1928\u001b[0m \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mAttributeError\u001b[39;00m(\n\u001b[1;32m   1929\u001b[0m     \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;132;01m{\u001b[39;00m\u001b[38;5;28mtype\u001b[39m(\u001b[38;5;28mself\u001b[39m)\u001b[38;5;241m.\u001b[39m\u001b[38;5;18m__name__\u001b[39m\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m object has no attribute \u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mname\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m   1930\u001b[0m )\n",
      "\u001b[0;31mAttributeError\u001b[0m: 'Spherical_RoPE' object has no attribute 'frequency_x'"
     ]
    }
   ],
   "source": [
    "from model.custom_vit import CustomPE_ViT\n",
    "import torch\n",
    "\n",
    "# Create a model with the custom rotary embedding\n",
    "kwargs['ViT_kwargs'] = ViT_Params\n",
    "kwargs['attn_kwargs'] = attn_args\n",
    "model = CustomPE_ViT(use_cls_token=cls_token, **kwargs)\n",
    "\n",
    "print(ViT_Params['num_classes'])\n",
    "# Test with a dummy input\n",
    "dummy_input = torch.randn(1, 3, ViT_Params['img_size'], ViT_Params['img_size'])\n",
    "output = model(dummy_input)\n",
    "print(output.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cpu\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "CustomPE_ViT(\n",
       "  (model): VisionTransformer(\n",
       "    (patch_embed): PatchEmbed(\n",
       "      (proj): Conv2d(3, 720, kernel_size=(4, 4), stride=(4, 4))\n",
       "      (norm): Identity()\n",
       "    )\n",
       "    (pos_drop): Dropout(p=0.0, inplace=False)\n",
       "    (patch_drop): Identity()\n",
       "    (norm_pre): Identity()\n",
       "    (blocks): Sequential(\n",
       "      (0): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): Identity()\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): Identity()\n",
       "      )\n",
       "      (1): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.009)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.009)\n",
       "      )\n",
       "      (2): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.018)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.018)\n",
       "      )\n",
       "      (3): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.027)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.027)\n",
       "      )\n",
       "      (4): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.036)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.036)\n",
       "      )\n",
       "      (5): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.045)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.045)\n",
       "      )\n",
       "      (6): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.055)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.055)\n",
       "      )\n",
       "      (7): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.064)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.064)\n",
       "      )\n",
       "      (8): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.073)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.073)\n",
       "      )\n",
       "      (9): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.082)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.082)\n",
       "      )\n",
       "      (10): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.091)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.091)\n",
       "      )\n",
       "      (11): Block(\n",
       "        (norm1): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (attn): Custom_Attention(\n",
       "          (qkv): Linear(in_features=720, out_features=2160, bias=False)\n",
       "          (q_norm): Identity()\n",
       "          (k_norm): Identity()\n",
       "          (attn_drop): Dropout(p=0.1, inplace=False)\n",
       "          (proj): Linear(in_features=720, out_features=720, bias=True)\n",
       "          (proj_drop): Dropout(p=0.1, inplace=False)\n",
       "          (pe): Learned_Axial_RoPE()\n",
       "        )\n",
       "        (ls1): LayerScale()\n",
       "        (drop_path1): DropPath(drop_prob=0.100)\n",
       "        (norm2): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "        (mlp): Mlp(\n",
       "          (fc1): Linear(in_features=720, out_features=2880, bias=True)\n",
       "          (act): GELU(approximate='none')\n",
       "          (drop1): Dropout(p=0.0, inplace=False)\n",
       "          (norm): Identity()\n",
       "          (fc2): Linear(in_features=2880, out_features=720, bias=True)\n",
       "          (drop2): Dropout(p=0.0, inplace=False)\n",
       "        )\n",
       "        (ls2): LayerScale()\n",
       "        (drop_path2): DropPath(drop_prob=0.100)\n",
       "      )\n",
       "    )\n",
       "    (norm): LayerNorm((720,), eps=1e-06, elementwise_affine=True)\n",
       "    (fc_norm): Identity()\n",
       "    (head_drop): Dropout(p=0.1, inplace=False)\n",
       "    (head): Linear(in_features=720, out_features=100, bias=True)\n",
       "  )\n",
       ")"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import torch.optim as optim\n",
    "import torch_optimizer as optimizers\n",
    "\n",
    "import torch.nn as nn\n",
    "from torch.optim.lr_scheduler import CosineAnnealingWarmRestarts, CosineAnnealingLR\n",
    "\n",
    "import torch\n",
    "warmup_steps = 15\n",
    "\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "print(device)\n",
    "\n",
    "# Loss function and optimizer\n",
    "optimizer = optimizers.lamb.Lamb(model.parameters(), lr=lr, weight_decay=weight_decay)\n",
    "# Define the warmup scheduler\n",
    "\n",
    "# Send the model to GPU if available\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "model.to(device)\n",
    "# model = torch.compile(model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "dummy_input = dummy_input.to(device)\n",
    "output = model(dummy_input)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/torch/amp/grad_scaler.py:132: UserWarning: torch.cuda.amp.GradScaler is enabled, but CUDA is not available.  Disabling.\n",
      "  warnings.warn(\n",
      "\u001b[34m\u001b[1mwandb\u001b[0m: Currently logged in as: \u001b[33mchase-geijn\u001b[0m (\u001b[33mchase-geijn-university-of-g-ttingen\u001b[0m) to \u001b[32mhttps://api.wandb.ai\u001b[0m. Use \u001b[1m`wandb login --relogin`\u001b[0m to force relogin\n",
      "\u001b[34m\u001b[1mwandb\u001b[0m: Using wandb-core as the SDK backend.  Please refer to https://wandb.me/wandb-core for more information.\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "Tracking run with wandb version 0.19.6"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Run data is saved locally in <code>/mnt/vast-nhr/home/geijn/u14771/git/Spherical RoPE/wandb/run-20250521_170504-dvm3h9ln</code>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "Syncing run <strong><a href='https://wandb.ai/chase-geijn-university-of-g-ttingen/QuatRo-CIFAR100/runs/dvm3h9ln' target=\"_blank\">Learned Axial RoPE</a></strong> to <a href='https://wandb.ai/chase-geijn-university-of-g-ttingen/QuatRo-CIFAR100' target=\"_blank\">Weights & Biases</a> (<a href='https://wandb.me/developer-guide' target=\"_blank\">docs</a>)<br>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       " View project at <a href='https://wandb.ai/chase-geijn-university-of-g-ttingen/QuatRo-CIFAR100' target=\"_blank\">https://wandb.ai/chase-geijn-university-of-g-ttingen/QuatRo-CIFAR100</a>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       " View run at <a href='https://wandb.ai/chase-geijn-university-of-g-ttingen/QuatRo-CIFAR100/runs/dvm3h9ln' target=\"_blank\">https://wandb.ai/chase-geijn-university-of-g-ttingen/QuatRo-CIFAR100/runs/dvm3h9ln</a>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/torch/amp/grad_scaler.py:132: UserWarning: torch.cuda.amp.GradScaler is enabled, but CUDA is not available.  Disabling.\n",
      "  warnings.warn(\n",
      "Epoch 1/400 [Train]:   0%|          | 0/49 [00:00<?, ?it/s]/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/torch/utils/data/dataloader.py:624: UserWarning: This DataLoader will create 8 worker processes in total. Our suggested max number of worker in current system is 1, which is smaller than what this DataLoader is going to create. Please be aware that excessive worker creation might get DataLoader running slow or even freeze, lower the worker number to avoid potential slowness/freeze if necessary.\n",
      "  warnings.warn(\n",
      "Epoch 1/400 [Train]:   0%|          | 0/49 [00:06<?, ?it/s]\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[9], line 47\u001b[0m\n\u001b[1;32m     45\u001b[0m total \u001b[38;5;241m=\u001b[39m \u001b[38;5;241m0\u001b[39m\n\u001b[1;32m     46\u001b[0m model\u001b[38;5;241m.\u001b[39mtrain()\n\u001b[0;32m---> 47\u001b[0m \u001b[43m\u001b[49m\u001b[38;5;28;43;01mfor\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43minputs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlabels\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mtqdm\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mtqdm\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtrain_loader\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mdesc\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;124;43mf\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[38;5;124;43mEpoch \u001b[39;49m\u001b[38;5;132;43;01m{\u001b[39;49;00m\u001b[43mepoch\u001b[49m\u001b[38;5;241;43m+\u001b[39;49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[38;5;132;43;01m}\u001b[39;49;00m\u001b[38;5;124;43m/\u001b[39;49m\u001b[38;5;132;43;01m{\u001b[39;49;00m\u001b[43mnum_epochs\u001b[49m\u001b[38;5;132;43;01m}\u001b[39;49;00m\u001b[38;5;124;43m [Train]\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m:\u001b[49m\n\u001b[1;32m     48\u001b[0m \u001b[43m    \u001b[49m\u001b[43minputs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlabels\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43m \u001b[49m\u001b[43minputs\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mto\u001b[49m\u001b[43m(\u001b[49m\u001b[43mdevice\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlabels\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mto\u001b[49m\u001b[43m(\u001b[49m\u001b[43mdevice\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     49\u001b[0m \u001b[43m    \u001b[49m\u001b[38;5;66;43;03m# labels_oh = torch.nn.functional.one_hot(labels, num_classes=100).float()\u001b[39;49;00m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/tqdm/std.py:1181\u001b[0m, in \u001b[0;36mtqdm.__iter__\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m   1178\u001b[0m time \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_time\n\u001b[1;32m   1180\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m-> 1181\u001b[0m \u001b[43m    \u001b[49m\u001b[38;5;28;43;01mfor\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mobj\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43miterable\u001b[49m\u001b[43m:\u001b[49m\n\u001b[1;32m   1182\u001b[0m \u001b[43m        \u001b[49m\u001b[38;5;28;43;01myield\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mobj\u001b[49m\n\u001b[1;32m   1183\u001b[0m \u001b[43m        \u001b[49m\u001b[38;5;66;43;03m# Update and possibly print the progressbar.\u001b[39;49;00m\n\u001b[1;32m   1184\u001b[0m \u001b[43m        \u001b[49m\u001b[38;5;66;43;03m# Note: does not call self.update(1) for speed optimisation.\u001b[39;49;00m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/torch/utils/data/dataloader.py:708\u001b[0m, in \u001b[0;36m_BaseDataLoaderIter.__next__\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    705\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_sampler_iter \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m    706\u001b[0m     \u001b[38;5;66;03m# TODO(https://github.com/pytorch/pytorch/issues/76750)\u001b[39;00m\n\u001b[1;32m    707\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_reset()  \u001b[38;5;66;03m# type: ignore[call-arg]\u001b[39;00m\n\u001b[0;32m--> 708\u001b[0m data \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_next_data\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    709\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_num_yielded \u001b[38;5;241m+\u001b[39m\u001b[38;5;241m=\u001b[39m \u001b[38;5;241m1\u001b[39m\n\u001b[1;32m    710\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m (\n\u001b[1;32m    711\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_dataset_kind \u001b[38;5;241m==\u001b[39m _DatasetKind\u001b[38;5;241m.\u001b[39mIterable\n\u001b[1;32m    712\u001b[0m     \u001b[38;5;129;01mand\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_IterableDataset_len_called \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m\n\u001b[1;32m    713\u001b[0m     \u001b[38;5;129;01mand\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_num_yielded \u001b[38;5;241m>\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_IterableDataset_len_called\n\u001b[1;32m    714\u001b[0m ):\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/torch/utils/data/dataloader.py:1458\u001b[0m, in \u001b[0;36m_MultiProcessingDataLoaderIter._next_data\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m   1455\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_process_data(data)\n\u001b[1;32m   1457\u001b[0m \u001b[38;5;28;01massert\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_shutdown \u001b[38;5;129;01mand\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_tasks_outstanding \u001b[38;5;241m>\u001b[39m \u001b[38;5;241m0\u001b[39m\n\u001b[0;32m-> 1458\u001b[0m idx, data \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_get_data\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1459\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_tasks_outstanding \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m=\u001b[39m \u001b[38;5;241m1\u001b[39m\n\u001b[1;32m   1460\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_dataset_kind \u001b[38;5;241m==\u001b[39m _DatasetKind\u001b[38;5;241m.\u001b[39mIterable:\n\u001b[1;32m   1461\u001b[0m     \u001b[38;5;66;03m# Check for _IterableDatasetStopIteration\u001b[39;00m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/torch/utils/data/dataloader.py:1420\u001b[0m, in \u001b[0;36m_MultiProcessingDataLoaderIter._get_data\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m   1416\u001b[0m     \u001b[38;5;66;03m# In this case, `self._data_queue` is a `queue.Queue`,. But we don't\u001b[39;00m\n\u001b[1;32m   1417\u001b[0m     \u001b[38;5;66;03m# need to call `.task_done()` because we don't use `.join()`.\u001b[39;00m\n\u001b[1;32m   1418\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m   1419\u001b[0m     \u001b[38;5;28;01mwhile\u001b[39;00m \u001b[38;5;28;01mTrue\u001b[39;00m:\n\u001b[0;32m-> 1420\u001b[0m         success, data \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_try_get_data\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1421\u001b[0m         \u001b[38;5;28;01mif\u001b[39;00m success:\n\u001b[1;32m   1422\u001b[0m             \u001b[38;5;28;01mreturn\u001b[39;00m data\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/torch/utils/data/dataloader.py:1251\u001b[0m, in \u001b[0;36m_MultiProcessingDataLoaderIter._try_get_data\u001b[0;34m(self, timeout)\u001b[0m\n\u001b[1;32m   1238\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21m_try_get_data\u001b[39m(\u001b[38;5;28mself\u001b[39m, timeout\u001b[38;5;241m=\u001b[39m_utils\u001b[38;5;241m.\u001b[39mMP_STATUS_CHECK_INTERVAL):\n\u001b[1;32m   1239\u001b[0m     \u001b[38;5;66;03m# Tries to fetch data from `self._data_queue` once for a given timeout.\u001b[39;00m\n\u001b[1;32m   1240\u001b[0m     \u001b[38;5;66;03m# This can also be used as inner loop of fetching without timeout, with\u001b[39;00m\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m   1248\u001b[0m     \u001b[38;5;66;03m# Returns a 2-tuple:\u001b[39;00m\n\u001b[1;32m   1249\u001b[0m     \u001b[38;5;66;03m#   (bool: whether successfully get data, any: data if successful else None)\u001b[39;00m\n\u001b[1;32m   1250\u001b[0m     \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m-> 1251\u001b[0m         data \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_data_queue\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mget\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtimeout\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mtimeout\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1252\u001b[0m         \u001b[38;5;28;01mreturn\u001b[39;00m (\u001b[38;5;28;01mTrue\u001b[39;00m, data)\n\u001b[1;32m   1253\u001b[0m     \u001b[38;5;28;01mexcept\u001b[39;00m \u001b[38;5;167;01mException\u001b[39;00m \u001b[38;5;28;01mas\u001b[39;00m e:\n\u001b[1;32m   1254\u001b[0m         \u001b[38;5;66;03m# At timeout and error, we manually check whether any worker has\u001b[39;00m\n\u001b[1;32m   1255\u001b[0m         \u001b[38;5;66;03m# failed. Note that this is the only mechanism for Windows to detect\u001b[39;00m\n\u001b[1;32m   1256\u001b[0m         \u001b[38;5;66;03m# worker failures.\u001b[39;00m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/multiprocessing/queues.py:113\u001b[0m, in \u001b[0;36mQueue.get\u001b[0;34m(self, block, timeout)\u001b[0m\n\u001b[1;32m    111\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m block:\n\u001b[1;32m    112\u001b[0m     timeout \u001b[38;5;241m=\u001b[39m deadline \u001b[38;5;241m-\u001b[39m time\u001b[38;5;241m.\u001b[39mmonotonic()\n\u001b[0;32m--> 113\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_poll\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtimeout\u001b[49m\u001b[43m)\u001b[49m:\n\u001b[1;32m    114\u001b[0m         \u001b[38;5;28;01mraise\u001b[39;00m Empty\n\u001b[1;32m    115\u001b[0m \u001b[38;5;28;01melif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_poll():\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/multiprocessing/connection.py:257\u001b[0m, in \u001b[0;36m_ConnectionBase.poll\u001b[0;34m(self, timeout)\u001b[0m\n\u001b[1;32m    255\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_check_closed()\n\u001b[1;32m    256\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_check_readable()\n\u001b[0;32m--> 257\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_poll\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtimeout\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/multiprocessing/connection.py:440\u001b[0m, in \u001b[0;36mConnection._poll\u001b[0;34m(self, timeout)\u001b[0m\n\u001b[1;32m    439\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21m_poll\u001b[39m(\u001b[38;5;28mself\u001b[39m, timeout):\n\u001b[0;32m--> 440\u001b[0m     r \u001b[38;5;241m=\u001b[39m \u001b[43mwait\u001b[49m\u001b[43m(\u001b[49m\u001b[43m[\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m]\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtimeout\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    441\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mbool\u001b[39m(r)\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/multiprocessing/connection.py:948\u001b[0m, in \u001b[0;36mwait\u001b[0;34m(object_list, timeout)\u001b[0m\n\u001b[1;32m    945\u001b[0m     deadline \u001b[38;5;241m=\u001b[39m time\u001b[38;5;241m.\u001b[39mmonotonic() \u001b[38;5;241m+\u001b[39m timeout\n\u001b[1;32m    947\u001b[0m \u001b[38;5;28;01mwhile\u001b[39;00m \u001b[38;5;28;01mTrue\u001b[39;00m:\n\u001b[0;32m--> 948\u001b[0m     ready \u001b[38;5;241m=\u001b[39m \u001b[43mselector\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mselect\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtimeout\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    949\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m ready:\n\u001b[1;32m    950\u001b[0m         \u001b[38;5;28;01mreturn\u001b[39;00m [key\u001b[38;5;241m.\u001b[39mfileobj \u001b[38;5;28;01mfor\u001b[39;00m (key, events) \u001b[38;5;129;01min\u001b[39;00m ready]\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/selectors.py:415\u001b[0m, in \u001b[0;36m_PollLikeSelector.select\u001b[0;34m(self, timeout)\u001b[0m\n\u001b[1;32m    413\u001b[0m ready \u001b[38;5;241m=\u001b[39m []\n\u001b[1;32m    414\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m--> 415\u001b[0m     fd_event_list \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_selector\u001b[38;5;241m.\u001b[39mpoll(timeout)\n\u001b[1;32m    416\u001b[0m \u001b[38;5;28;01mexcept\u001b[39;00m \u001b[38;5;167;01mInterruptedError\u001b[39;00m:\n\u001b[1;32m    417\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m ready\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Error in callback <bound method _WandbInit._pause_backend of <wandb.sdk.wandb_init._WandbInit object at 0x14cc50400f50>> (for post_run_cell), with arguments args (<ExecutionResult object at 14cd22aa5f50, execution_count=9 error_before_exec=None error_in_exec= info=<ExecutionInfo object at 14cc53615050, raw_cell=\"import tqdm\n",
      "from torch.amp import autocast, GradSc..\" store_history=True silent=False shell_futures=True cell_id=vscode-notebook-cell://ssh-remote%2Bhlrn/user/geijn/u14771/git/Spherical%20RoPE/CIFAR100_Train.ipynb#X11sdnNjb2RlLXJlbW90ZQ%3D%3D> result=None>,),kwargs {}:\n"
     ]
    },
    {
     "ename": "BrokenPipeError",
     "evalue": "[Errno 32] Broken pipe",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mBrokenPipeError\u001b[0m                           Traceback (most recent call last)",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/wandb/sdk/wandb_init.py:555\u001b[0m, in \u001b[0;36m_WandbInit._pause_backend\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m    553\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mbackend\u001b[38;5;241m.\u001b[39minterface \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m    554\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_logger\u001b[38;5;241m.\u001b[39minfo(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mpausing backend\u001b[39m\u001b[38;5;124m\"\u001b[39m)  \u001b[38;5;66;03m# type: ignore\u001b[39;00m\n\u001b[0;32m--> 555\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mbackend\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43minterface\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mpublish_pause\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/wandb/sdk/interface/interface.py:771\u001b[0m, in \u001b[0;36mInterfaceBase.publish_pause\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    769\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21mpublish_pause\u001b[39m(\u001b[38;5;28mself\u001b[39m) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m    770\u001b[0m     pause \u001b[38;5;241m=\u001b[39m pb\u001b[38;5;241m.\u001b[39mPauseRequest()\n\u001b[0;32m--> 771\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_publish_pause\u001b[49m\u001b[43m(\u001b[49m\u001b[43mpause\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/wandb/sdk/interface/interface_shared.py:368\u001b[0m, in \u001b[0;36mInterfaceShared._publish_pause\u001b[0;34m(self, pause)\u001b[0m\n\u001b[1;32m    366\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21m_publish_pause\u001b[39m(\u001b[38;5;28mself\u001b[39m, pause: pb\u001b[38;5;241m.\u001b[39mPauseRequest) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m    367\u001b[0m     rec \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_make_request(pause\u001b[38;5;241m=\u001b[39mpause)\n\u001b[0;32m--> 368\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_publish\u001b[49m\u001b[43m(\u001b[49m\u001b[43mrec\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/wandb/sdk/interface/interface_sock.py:47\u001b[0m, in \u001b[0;36mInterfaceSock._publish\u001b[0;34m(self, record, local)\u001b[0m\n\u001b[1;32m     45\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21m_publish\u001b[39m(\u001b[38;5;28mself\u001b[39m, record: \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mpb.Record\u001b[39m\u001b[38;5;124m\"\u001b[39m, local: Optional[\u001b[38;5;28mbool\u001b[39m] \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m     46\u001b[0m     \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_assign(record)\n\u001b[0;32m---> 47\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_sock_client\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msend_record_publish\u001b[49m\u001b[43m(\u001b[49m\u001b[43mrecord\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/wandb/sdk/lib/sock_client.py:222\u001b[0m, in \u001b[0;36mSockClient.send_record_publish\u001b[0;34m(self, record)\u001b[0m\n\u001b[1;32m    220\u001b[0m server_req \u001b[38;5;241m=\u001b[39m spb\u001b[38;5;241m.\u001b[39mServerRequest()\n\u001b[1;32m    221\u001b[0m server_req\u001b[38;5;241m.\u001b[39mrecord_publish\u001b[38;5;241m.\u001b[39mCopyFrom(record)\n\u001b[0;32m--> 222\u001b[0m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msend_server_request\u001b[49m\u001b[43m(\u001b[49m\u001b[43mserver_req\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/wandb/sdk/lib/sock_client.py:154\u001b[0m, in \u001b[0;36mSockClient.send_server_request\u001b[0;34m(self, msg)\u001b[0m\n\u001b[1;32m    153\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m\u001b[38;5;250m \u001b[39m\u001b[38;5;21msend_server_request\u001b[39m(\u001b[38;5;28mself\u001b[39m, msg: Any) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[0;32m--> 154\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_send_message\u001b[49m\u001b[43m(\u001b[49m\u001b[43mmsg\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/wandb/sdk/lib/sock_client.py:151\u001b[0m, in \u001b[0;36mSockClient._send_message\u001b[0;34m(self, msg)\u001b[0m\n\u001b[1;32m    149\u001b[0m header \u001b[38;5;241m=\u001b[39m struct\u001b[38;5;241m.\u001b[39mpack(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124m<BI\u001b[39m\u001b[38;5;124m\"\u001b[39m, \u001b[38;5;28mord\u001b[39m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mW\u001b[39m\u001b[38;5;124m\"\u001b[39m), raw_size)\n\u001b[1;32m    150\u001b[0m \u001b[38;5;28;01mwith\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_lock:\n\u001b[0;32m--> 151\u001b[0m     \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_sendall_with_error_handle\u001b[49m\u001b[43m(\u001b[49m\u001b[43mheader\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m+\u001b[39;49m\u001b[43m \u001b[49m\u001b[43mdata\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m/mnt/vast-nhr/home/geijn/u14771/micromamba/envs/neuroformer/lib/python3.11/site-packages/wandb/sdk/lib/sock_client.py:130\u001b[0m, in \u001b[0;36mSockClient._sendall_with_error_handle\u001b[0;34m(self, data)\u001b[0m\n\u001b[1;32m    128\u001b[0m start_time \u001b[38;5;241m=\u001b[39m time\u001b[38;5;241m.\u001b[39mmonotonic()\n\u001b[1;32m    129\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m--> 130\u001b[0m     sent \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_sock\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msend\u001b[49m\u001b[43m(\u001b[49m\u001b[43mdata\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    131\u001b[0m     \u001b[38;5;66;03m# sent equal to 0 indicates a closed socket\u001b[39;00m\n\u001b[1;32m    132\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m sent \u001b[38;5;241m==\u001b[39m \u001b[38;5;241m0\u001b[39m:\n",
      "\u001b[0;31mBrokenPipeError\u001b[0m: [Errno 32] Broken pipe"
     ]
    }
   ],
   "source": [
    "import tqdm\n",
    "from torch.amp import autocast, GradScaler\n",
    "from torchvision.transforms import v2\n",
    "import random\n",
    "\n",
    "scaler = GradScaler()\n",
    "from transformers import get_cosine_schedule_with_warmup\n",
    "project_name = 'Spherical_RoPE'\n",
    "# Initialize W&B\n",
    "wandb.init(\n",
    "    project=f\"{project_name}-{dataset}\",\n",
    "    name=f\"{PE}\",\n",
    "    group=f\"{PE}\",\n",
    "    config={\n",
    "            'epochs': num_epochs,\n",
    "            'batch_size': batch_size,\n",
    "            'learning_rate': lr,\n",
    "            'model': base_model,\n",
    "            'dataset': dataset\n",
    "            }\n",
    "\n",
    ")\n",
    "config = wandb.config\n",
    "best_acc = 0.0\n",
    "\n",
    "scaler = GradScaler()\n",
    "\n",
    "scheduler = get_cosine_schedule_with_warmup(\n",
    "    optimizer,\n",
    "    num_warmup_steps=5,\n",
    "    num_training_steps=num_epochs\n",
    ")\n",
    "criterion = nn.CrossEntropyLoss(label_smoothing=.1)\n",
    "cm = v2.CutMix(num_classes=100, alpha=1.)\n",
    "mu = v2.MixUp(num_classes=100, alpha=.8)\n",
    "cmmu = [cm,mu]\n",
    "save_path = f\"./trained_models/{dataset}_{PE}.pth\"\n",
    "os.makedirs(os.path.dirname(save_path), exist_ok=True)\n",
    "\n",
    "for epoch in range(num_epochs):\n",
    "    running_loss = 0.0\n",
    "    correct = 0\n",
    "    total = 0\n",
    "    model.train()\n",
    "    for inputs, labels in tqdm.tqdm(train_loader, desc=f\"Epoch {epoch+1}/{num_epochs} [Train]\"):\n",
    "        inputs, labels = inputs.to(device), labels.to(device)\n",
    "        cm_or_mu = random.choice(cmmu) # Randomly choose between CutMix and MixUp\n",
    "        optimizer.zero_grad()\n",
    "        with autocast('cuda'):\n",
    "            inputs, labels_oh = cm_or_mu(inputs, labels)\n",
    "            # labels_oh = labels # Needed if skipping CutMix/MixUp\n",
    "            outputs = model(inputs)\n",
    "            loss = criterion(outputs, labels_oh)\n",
    "        scaler.scale(loss).backward()\n",
    "        scaler.step(optimizer)\n",
    "        scaler.update()\n",
    "        \n",
    "        for name, param in model.named_parameters():\n",
    "            if param.grad is None:\n",
    "                print(f\"Unused parameter: {name}\")\n",
    "\n",
    "        running_loss += loss\n",
    "        _, predicted = torch.max(outputs, 1)\n",
    "        correct += (predicted == labels).sum()\n",
    "        total += labels.size(0)\n",
    "\n",
    "    epoch_loss = running_loss.item() / total\n",
    "    epoch_accuracy = 100.0 * correct.item() / total\n",
    "    print(f\"Epoch [{epoch+1}/{num_epochs}] | Train Loss: {epoch_loss:.4f} | Accuracy: {epoch_accuracy:.2f}%\")\n",
    "\n",
    "    wandb.log({\n",
    "        'train_loss': epoch_loss,\n",
    "        'train_accuracy': epoch_accuracy,\n",
    "        'lr': scheduler.get_last_lr()[0]\n",
    "    }, step=epoch+1)\n",
    "\n",
    "    # Validation\n",
    "    model.eval()\n",
    "    val_loss = 0.0\n",
    "    val_correct = 0\n",
    "    val_total = 0\n",
    "\n",
    "    with torch.no_grad():\n",
    "        for inputs, labels in tqdm.tqdm(val_loader, desc=f\"Epoch {epoch+1}/{num_epochs} [Val]\"):\n",
    "            inputs, labels = inputs.to(device), labels.to(device)\n",
    "            with autocast('cuda'):\n",
    "                outputs = model(inputs)\n",
    "                loss = criterion(outputs, labels)\n",
    "            val_loss += loss.item()\n",
    "            _, predicted = torch.max(outputs, 1)\n",
    "            val_correct += (predicted == labels).sum().item()\n",
    "            val_total += labels.size(0)\n",
    "\n",
    "    val_accuracy = 100.0 * val_correct / val_total\n",
    "    avg_val_loss = val_loss / val_total\n",
    "\n",
    "    print(f\"Validation Accuracy: {val_accuracy:.2f}% | Val Loss: {avg_val_loss:.4f}\")\n",
    "\n",
    "    wandb.log({\n",
    "        'valid_loss': avg_val_loss,\n",
    "        'val_accuracy': val_accuracy\n",
    "    }, step=epoch+1)\n",
    "\n",
    "    if val_accuracy > best_acc:\n",
    "        best_acc = val_accuracy\n",
    "        torch.save(model.state_dict(), save_path)\n",
    "        wandb.save(save_path)\n",
    "\n",
    "    scheduler.step()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "neuroformer",
   "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.11.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
