{
 "cells": [
  {
   "cell_type": "code",
   "id": "e0d26e28608724a3",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-10-02T16:06:51.213636Z",
     "start_time": "2025-10-02T16:06:49.160923Z"
    }
   },
   "source": [
    "import sys\n",
    "import os\n",
    "sys.path.insert(0, os.path.abspath(os.path.join(os.getcwd(), '..')))\n",
    "import time\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "print(f\"Using device: {device}\")\n",
    "\n",
    "L = 7 # degree level of dyadic grid\n",
    "m = 2**L - 1 # grid size\n",
    "\n",
    "batch_size = 64\n",
    "in_features  = 512\n",
    "out_features = 10\n",
    "\n",
    "x = torch.rand(batch_size, in_features, device=device) # (B, in_features)\n",
    "print(f\"x shape: {x.shape}\")"
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using device: cuda\n",
      "x shape: torch.Size([64, 512])\n"
     ]
    }
   ],
   "execution_count": 1
  },
  {
   "cell_type": "code",
   "id": "58dbea14953815d9",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-10-02T16:06:51.594905Z",
     "start_time": "2025-10-02T16:06:51.228117Z"
    }
   },
   "source": [
    "from linear_operator.utils.cholesky import psd_safe_cholesky  # from gpytorch's linear_operator\n",
    "from csgp.design_class import HyperbolicCrossDesign\n",
    "from csgp.layers.kernels import LaplaceL1Kernel\n",
    "\n",
    "@torch.no_grad()  # drop this if you want gradients through everything\n",
    "def inv_cholesky_transpose(K: torch.Tensor, jitter: float = 1e-6) -> torch.Tensor:\n",
    "    \"\"\"\n",
    "    Returns U = L^{-T}, where K = L L^T (L lower-triangular), so K^{-1} = U U^T.\n",
    "    Works with batched SPD matrices (..., n, n).\n",
    "    \"\"\"\n",
    "    n = K.size(-1)\n",
    "    I = torch.eye(n, dtype=K.dtype, device=K.device)\n",
    "    # Stable Cholesky of K (lower-triangular)\n",
    "    L = psd_safe_cholesky(K, jitter=jitter)  # (..., n, n), lower-triangular\n",
    "    # Solve L^T U = I  ->  U = L^{-T} (upper-triangular)\n",
    "    U = torch.linalg.solve_triangular(L.transpose(-1, -2), I, upper=True)\n",
    "    return U  # U is upper-triangular and K^{-1} = U @ U.transpose(-1, -2)\n",
    "\n",
    "dyadic_design = HyperbolicCrossDesign(dyadic_sort=True, return_neighbors=True)(deg=L, input_lb=0, input_ub=1)\n",
    "design_points = dyadic_design.points.reshape(-1, 1).to(device)  # [m, 1] size tensor\n",
    "print(f\"Dyadic grid points shape: {dyadic_design.points.shape}\")\n",
    "\n",
    "covar = LaplaceL1Kernel(lengthscale=1.).to(device)(design_points)\n",
    "L_inv_T = inv_cholesky_transpose(covar)\n",
    "print(f\"L_inv_T shape: {L_inv_T.shape}\")\n",
    "\n",
    "h = LaplaceL1Kernel(lengthscale=1.).to(device)(x.unsqueeze(-1), design_points)\n",
    "phi = torch.matmul(h, L_inv_T)\n",
    "print(f\"Phi(x) shape: {phi.shape}\")"
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dyadic grid points shape: torch.Size([127])\n",
      "L_inv_T shape: torch.Size([127, 127])\n",
      "Phi(x) shape: torch.Size([64, 512, 127])\n"
     ]
    }
   ],
   "execution_count": 2
  },
  {
   "cell_type": "code",
   "id": "d106f6bcae1a01d1",
   "metadata": {
    "collapsed": true,
    "ExecuteTime": {
     "end_time": "2025-10-02T16:06:51.878572Z",
     "start_time": "2025-10-02T16:06:51.810455Z"
    }
   },
   "source": [
    "def dyadic_psi(x: torch.Tensor, L: int, sigma: float = 1.0, ell_c: float = 1.0):\n",
    "    \"\"\"\n",
    "    Batched dyadic nonzero indices.\n",
    "\n",
    "    Args\n",
    "    ----\n",
    "    x : (...,) tensor with values in [0, 1].\n",
    "        Works with any number of leading batch dims.\n",
    "    L : int, number of dyadic levels (total columns m = 2^L - 1).\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    idx : (..., L) long tensor\n",
    "        0-based global column indices in dyadic order for each level (DC is level 1).\n",
    "        The returned shape matches the leading shape of x, with an extra trailing dim of size L.\n",
    "    \"\"\"\n",
    "    if x.ndim == 0:\n",
    "        x = x.unsqueeze(0)  # promote scalar to shape (1,)\n",
    "    device, dtype = x.device, x.dtype\n",
    "\n",
    "    # 2^s for s=1..L\n",
    "    pow2 = torch.pow(2, torch.arange(1, L+1, device=device, dtype=torch.int64))  # (L,)\n",
    "\n",
    "    # k_s = ceil(2^s * x) clamped to [1, 2^s - 1]\n",
    "    ks = torch.ceil(x[..., None] * pow2.to(x.dtype)).to(torch.int64)  # (..., L)\n",
    "    ks = torch.clamp(ks, min=1)\n",
    "    ks_max = (pow2 - 1)  # (L,)\n",
    "    ks = torch.minimum(ks, ks_max)  # (..., L)\n",
    "\n",
    "    # r_s^(odd): force to be odd (right endpoint index made odd)\n",
    "    # if ks even -> ks-1, else ks\n",
    "    rs = ks - ((ks & 1) == 0).to(torch.int64) # (..., L), odd in {1,3,...,2^s-1}\n",
    "\n",
    "    # position within level s: t_s in {1,...,2^{s-1}}\n",
    "    ts = (rs + 1) // 2  # (..., L)\n",
    "\n",
    "    # offsets: number of columns before level s (0-based indexing)\n",
    "    offsets = (pow2 // 2) - 1  # (L,)\n",
    "\n",
    "    # global 0-based indices: J_s = offset(s) + (t_s - 1)\n",
    "    idx = offsets + (ts - 1)  # (..., L)\n",
    "\n",
    "    u = HyperbolicCrossDesign(dyadic_sort=True, return_neighbors=True)(deg=L, input_lb=0, input_ub=1).points.to(device) # (2^L-1,)\n",
    "    view_shape = (1,) * x.dim() + (u.shape[0],)      # (1,1,...,1, 2^L-1)\n",
    "    u_selected = torch.gather(u.view(view_shape).expand(*x.shape, -1), dim=-1, index=idx) # (..., L)\n",
    "\n",
    "    delta = torch.abs(x.unsqueeze(-1) - u_selected) # |x - m2^{-l}|\n",
    "    # pow2_f = (1.0 / pow2).view(view_shape).expand(*x.shape, -1)  # (..., L), 2^{-l} for l=1..L\n",
    "    pow2_f = (1.0 / pow2).to(x.dtype)\n",
    "\n",
    "    psi =  torch.sqrt(2 / torch.sinh(pow2_f * 2 * ell_c)) * torch.sinh(ell_c * (pow2_f - delta))\n",
    "\n",
    "    return idx, psi  # (N, L), long\n",
    "\n",
    "bsize, fsize = x.shape[0], x.shape[1]\n",
    "idx, psi = dyadic_psi(x, L)\n",
    "print(f\"Nonzero idx shape: {idx.shape}\")\n",
    "print(f\"Compact Psi(x) shape: {psi.shape}\") # (B, in, L)"
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Nonzero idx shape: torch.Size([64, 512, 7])\n",
      "Compact Psi(x) shape: torch.Size([64, 512, 7])\n"
     ]
    }
   ],
   "execution_count": 3
  },
  {
   "cell_type": "code",
   "id": "9e5e48e7bf445352",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-10-02T16:06:52.005244Z",
     "start_time": "2025-10-02T16:06:52.000587Z"
    }
   },
   "source": [
    "# --------- Quick timing ----------\n",
    "def time_once(fn):\n",
    "    # warmup\n",
    "    for _ in range(5): _ = fn()\n",
    "    if device == \"cuda\":\n",
    "        start = torch.cuda.Event(True); end = torch.cuda.Event(True)\n",
    "        start.record(); _ = fn(); end.record()\n",
    "        torch.cuda.synchronize()\n",
    "        return start.elapsed_time(end)  # ms\n",
    "    else:\n",
    "        t0 = time.perf_counter(); _ = fn(); t1 = time.perf_counter()\n",
    "        return (t1 - t0) * 1000.0\n",
    "\n",
    "# Dense multiplication\n",
    "@torch.inference_mode()\n",
    "def dense_linear(x, weight, num_samples=1):\n",
    "    bsize = x.shape[0]\n",
    "    x = x.view(bsize, -1)\n",
    "    weight = weight.unsqueeze(0).expand(bsize, -1, -1)\n",
    "    output_ = []\n",
    "    for _ in range(num_samples):\n",
    "        noise = torch.randn_like(weight, device=weight.device)\n",
    "        output = torch.matmul(weight * noise, x.unsqueeze(-1)).squeeze(-1)\n",
    "        output_.append(output)\n",
    "    res = torch.mean(torch.stack(output_), dim=0)\n",
    "    return res\n",
    "\n",
    "# Sparse inference based on compact support\n",
    "@torch.inference_mode()\n",
    "def sparse_linear(x, weight, idx, num_samples=1):\n",
    "    bsize = x.shape[0]\n",
    "    fsize = x.shape[1]\n",
    "    osize = weight.shape[0]\n",
    "    # w_batch = weight.unsqueeze(0).expand(bsize, -1, -1).view(bsize, fsize, -1, osize)\n",
    "    # idx_batch = idx.unsqueeze(-1).expand(*idx.shape, osize)\n",
    "    # w_selected = torch.gather(w_batch, dim=-2, index=idx_batch).view(bsize, osize, -1)\n",
    "    \n",
    "    idx = idx.unsqueeze(-3).expand(*idx.shape[:-2], osize, -1, -1) #(bsize, fsize, L) --> (bsize, osize, fsize, L)\n",
    "    w_batch = weight.unsqueeze(0).expand(bsize, -1, -1).view(bsize, osize, fsize, -1)\n",
    "    w_selected = torch.gather(w_batch, dim=-1, index=idx).view(bsize, osize, -1) #(bsize, osize, fsize*L)\n",
    "    \n",
    "    x = x.view(bsize, -1)\n",
    "    output_ = []\n",
    "    for _ in range(num_samples):\n",
    "        noise = torch.randn_like(w_selected, device=weight.device)\n",
    "        output = torch.matmul(w_selected * noise, x.unsqueeze(-1)).squeeze(-1)\n",
    "        output_.append(output)\n",
    "    res = torch.mean(torch.stack(output_), dim=0)\n",
    "    return res"
   ],
   "outputs": [],
   "execution_count": 4
  },
  {
   "cell_type": "code",
   "id": "52beb2dcf3a6def2",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-10-02T16:06:52.139051Z",
     "start_time": "2025-10-02T16:06:52.098175Z"
    }
   },
   "source": [
    "# Define the weight matrix of the linear layer\n",
    "weight = torch.randn(out_features, in_features*m, device=device)\n",
    "print(f\"whole weight shape: {weight.shape}\")\n",
    "\n",
    "out_phi = dense_linear(phi, weight, num_samples=10)\n",
    "print(f\"phi(x) shape: {out_phi.shape}\")\n",
    "\n",
    "out_psi = sparse_linear(psi, weight, idx, num_samples=10)\n",
    "print(f\"psi(x) shape: {out_psi.shape}\")\n",
    "\n",
    "t_sparse = time_once(lambda: sparse_linear(psi, weight, idx, num_samples=1))\n",
    "t_dense = time_once(lambda: dense_linear(phi, weight, num_samples=1))\n",
    "\n",
    "print(f\"Dense computing phi(x): {t_dense:.3f} ms\")\n",
    "print(f\"Sparse computing psi(x): {t_sparse:.3f} ms\")"
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "whole weight shape: torch.Size([10, 65024])\n",
      "phi(x) shape: torch.Size([64, 10])\n",
      "psi(x) shape: torch.Size([64, 10])\n",
      "Dense computing phi(x): 1.144 ms\n",
      "Sparse computing psi(x): 0.063 ms\n"
     ]
    }
   ],
   "execution_count": 5
  },
  {
   "cell_type": "code",
   "id": "c4079e86",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-10-02T16:06:52.292395Z",
     "start_time": "2025-10-02T16:06:52.288048Z"
    }
   },
   "source": [
    "def dyadic_to_dense(vals, idx, m):\n",
    "    \"\"\"\n",
    "    Convert dyadic sparse representation to dense.\n",
    "\n",
    "    Args\n",
    "    ----\n",
    "    vals : (..., fsize, m) tensor\n",
    "        Values at the dyadic non-zero indices.\n",
    "    idx : (..., fsize, m) long tensor\n",
    "        0-based global column indices in dyadic order for each level (DC is level 1).\n",
    "        The leading shape of idx must match that of vals.\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    dense : (..., m) tensor\n",
    "        Dense representation.\n",
    "    \"\"\"\n",
    "    dense = torch.zeros(*vals.shape[:-1], m, device=vals.device, dtype=vals.dtype)\n",
    "    dense.scatter_(-1, idx, vals)\n",
    "    return dense\n",
    "\n",
    "def dense_gp(x, L, weight):\n",
    "    idx, psi = dyadic_psi(x, L)\n",
    "    m = 2**L - 1\n",
    "    dense_psi = dyadic_to_dense(psi, idx, m) #(bsize, fsize, m)\n",
    "    \n",
    "    bsize = x.shape[0]\n",
    "    psi = dense_psi.view(bsize, -1) #(bsize, fsize*m)\n",
    "    weight = weight.unsqueeze(0).expand(bsize, -1, -1)\n",
    "    \n",
    "    noise = torch.randn_like(weight, device=weight.device)\n",
    "    output = torch.matmul(weight * noise, psi.unsqueeze(-1)).squeeze(-1)\n",
    "\n",
    "    return output\n",
    "\n",
    "def sparse_gp(x, L, weight):\n",
    "    idx, psi = dyadic_psi(x, L)\n",
    "    bsize = x.shape[0]\n",
    "    fsize = x.shape[1]\n",
    "    osize = weight.shape[0]\n",
    "    \n",
    "    idx = idx.unsqueeze(-3).expand(*idx.shape[:-2], osize, -1, -1) #(bsize, fsize, L) --> (bsize, osize, fsize, L)\n",
    "    w_batch = weight.unsqueeze(0).expand(bsize, -1, -1).view(bsize, osize, fsize, -1)\n",
    "    w_selected = torch.gather(w_batch, dim=-1, index=idx).view(bsize, osize, -1) #(bsize, osize, fsize*L)\n",
    "    \n",
    "    psi = psi.view(bsize, -1)\n",
    "    \n",
    "    noise = torch.randn_like(w_selected, device=weight.device)\n",
    "    output = torch.matmul(w_selected * noise, psi.unsqueeze(-1)).squeeze(-1)\n",
    "\n",
    "    return output\n",
    "\n",
    "\n",
    "def deep_inference(x, L, weight, num_layers=1, use_sparse: bool = True):\n",
    "    if use_sparse:\n",
    "        for _ in range(num_layers):\n",
    "            x = sparse_gp(x, L, weight)\n",
    "        return x\n",
    "    else:\n",
    "        for _ in range(num_layers):\n",
    "            x = dense_gp(x, L, weight)\n",
    "        return x"
   ],
   "outputs": [],
   "execution_count": 6
  },
  {
   "cell_type": "code",
   "id": "1e54d012",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-10-02T16:14:40.370485Z",
     "start_time": "2025-10-02T16:14:37.188866Z"
    }
   },
   "source": [
    "# select 3 random seeds\n",
    "seeds = [0, 42, 2024]\n",
    "batch_size = 64\n",
    "out_features = 128\n",
    "in_features = 128\n",
    "num_mc = 10\n",
    "num_layers = 3\n",
    "L=7\n",
    "m = 2**L - 1\n",
    "x = torch.rand(batch_size, in_features, device=device) # (B, in_features)\n",
    "\n",
    "\n",
    "\n",
    "sparse_times = []\n",
    "dense_times = []\n",
    "for seed in seeds:\n",
    "    torch.manual_seed(seed)\n",
    "    weight = torch.randn(out_features, in_features*m, device=device)\n",
    "    t_sparse = 0\n",
    "    t_dense = 0\n",
    "    for _ in range(num_mc): # average over 10 runs\n",
    "        t_sparse += time_once(lambda: deep_inference(x, L, weight, num_layers=num_layers, use_sparse=True))\n",
    "        t_dense += time_once(lambda: deep_inference(x, L, weight, num_layers=num_layers, use_sparse=False))\n",
    "    t_sparse /= num_mc\n",
    "    t_dense /= num_mc\n",
    "    sparse_times.append(t_sparse)\n",
    "    dense_times.append(t_dense)\n",
    "    print(f\"Seed {seed}: Dense {t_dense:.3f} ms, Sparse {t_sparse:.3f} ms\")\n",
    "\n",
    "print(f\"Avg Dense: {sum(dense_times)/len(dense_times):.3f} ms\")\n",
    "print(f\"Std Dense: {torch.std(torch.tensor(dense_times)):.3f} ms\")\n",
    "print(\"-----\")\n",
    "print(f\"Avg Sparse: {sum(sparse_times)/len(sparse_times):.3f} ms\")\n",
    "print(f\"Std Sparse: {torch.std(torch.tensor(sparse_times)):.3f} ms\")"
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Seed 0: Dense 12.764 ms, Sparse 2.205 ms\n",
      "Seed 42: Dense 13.939 ms, Sparse 2.384 ms\n",
      "Seed 2024: Dense 13.935 ms, Sparse 2.234 ms\n",
      "Avg Dense: 13.546 ms\n",
      "Std Dense: 0.677 ms\n",
      "-----\n",
      "Avg Sparse: 2.275 ms\n",
      "Std Sparse: 0.096 ms\n"
     ]
    }
   ],
   "execution_count": 15
  },
  {
   "cell_type": "code",
   "id": "8652c83b",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2025-10-02T16:06:55.260148Z",
     "start_time": "2025-10-02T16:06:54.878669Z"
    }
   },
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "\n",
    "suptitlesize = 20\n",
    "titlesize = 18\n",
    "labelsize = 14\n",
    "legendsize = 12\n",
    "ticksize = 12\n",
    "\n",
    "layers = [1, 2, 3, 4]\n",
    "dense_time_mean = [4.129, 8.198, 12.406, 16.745]  # example mean times in ms\n",
    "dense_time_std = [0.114, 0.108, 0.209, 0.271]     # example std deviations in ms\n",
    "sparse_time_mean = [0.617, 1.332, 1.953, 2.669]  # example mean times in ms\n",
    "sparse_time_std = [0.013, 0.081, 0.128, 0.110]     # example std deviations in ms\n",
    "\n",
    "plt.figure(figsize=(4, 3))\n",
    "sns.set(style=\"whitegrid\")\n",
    "plt.errorbar(layers, dense_time_mean, yerr=dense_time_std, label='Dense', fmt='-o')\n",
    "plt.errorbar(layers, sparse_time_mean, yerr=sparse_time_std, label='Sparse (ours)', fmt='-o')\n",
    "plt.yscale('log')\n",
    "plt.xlabel('Number of Layers', fontsize=labelsize)\n",
    "plt.ylabel('Time (ms, log scale)', fontsize=labelsize)\n",
    "# plt.title(f'Time Analysis (Batch Size={args.batch_size}, In Features={args.in_features}, Out Features={args.out_features}, Samples={args.samples})')\n",
    "plt.legend(fontsize=legendsize)\n",
    "plt.tight_layout()\n",
    "plt.savefig('./time_analysis_layers.pdf', bbox_inches='tight')\n",
    "plt.show()"
   ],
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<Figure size 400x300 with 1 Axes>"
      ],
      "image/png": "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"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "execution_count": 8
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "dgp",
   "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.13.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
