{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Empirical Marginal Distribution of Under CLIP Losses\n",
    "\n",
    "In this Jupyter notebook, we create a version of Figure 3 of the manuscript to illustrate the variants of the CLIP loss. Please view the `README.md` file for dependencies."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn.functional as F\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "\n",
    "from miniclip import MiniCLIP\n",
    "\n",
    "import matplotlib as mpl\n",
    "mpl.rcParams['lines.linewidth'] = 4\n",
    "mpl.rcParams['xtick.labelsize'] = 20\n",
    "mpl.rcParams['ytick.labelsize'] = 20\n",
    "mpl.rcParams[\"axes.labelsize\"] = 20\n",
    "mpl.rcParams['legend.fontsize'] = 33\n",
    "mpl.rcParams['axes.titlesize'] = 32\n",
    "mpl.rcParams['pdf.fonttype'] = 42\n",
    "mpl.rcParams['ps.fonttype'] = 42"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The examples are subsets of the [ImageNet-Captions dataset](https://github.com/mlfoundations/imagenet-captions?tab=readme-ov-file) that have been embedded by a CLIP ViT-B/32 model.\n",
    "\n",
    "![imagenet](imagenet.png)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "DEVICE = \"cuda:0\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "color = \"dodgerblue\"\n",
    "color2 = \"lightsalmon\"\n",
    "fontsize = 24\n",
    "width = 1.0"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The following functions take a `logits` Tensor, which is an $m$-by-$m$ contrast matrix generated from the CLIP model given a batch of $m$ data points, and return two versions of the matrix with their rows and columns scaled: one has columns scaled first (akin to $P_\\theta^{(k)})$ in the manucript), and the other has its rows scaled first. The `jointly_centered_loss` represents zero iterations of balancing, and simply renormalizing the entire measure. The `clip_loss` represents one iteration of balancing, and recovers the original CLIP loss by the arguments in Section 2. Finally, `doubly_centered_loss` performs two iterations of the scaling procedure."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_marginals(log_prob1, log_prob2):\n",
    "    prob1 = torch.exp(log_prob1).cpu()\n",
    "    prob2 = torch.exp(log_prob2).cpu()\n",
    "    return prob1.sum(dim=0) / prob1.sum(), prob1.sum(dim=1) / prob1.sum(), prob2.sum(dim=0) / prob2.sum(), prob2.sum(dim=1) / prob2.sum()\n",
    "\n",
    "def jointly_centered_loss(logits):\n",
    "    norm_factor = torch.logsumexp(torch.flatten(logits), dim=0)\n",
    "    log_prob = logits - norm_factor\n",
    "    return log_prob, log_prob\n",
    "\n",
    "def clip_loss(logits):\n",
    "    cx   = F.log_softmax(logits, dim=1)\n",
    "    cy   = F.log_softmax(logits, dim=0)\n",
    "    return cx, cy\n",
    "\n",
    "def doubly_centered_loss(logits):\n",
    "    cx   = F.log_softmax(logits, dim=1)\n",
    "    cy   = F.log_softmax(logits, dim=0)\n",
    "    cycx = F.log_softmax(cx, dim=0)\n",
    "    cxcy = F.log_softmax(cy, dim=1)\n",
    "    return cycx, cxcy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([16, 512])\n",
      "torch.Size([16, 512])\n"
     ]
    }
   ],
   "source": [
    "X, Y = torch.load(\"data/clip_image_embeds.pt\"), torch.load(\"data/clip_text_embeds.pt\")\n",
    "\n",
    "print(X.shape)\n",
    "print(Y.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([16, 16])\n"
     ]
    }
   ],
   "source": [
    "# load model\n",
    "model_cfg = {\n",
    "    \"architecture\": \"miniclip\",\n",
    "    \"in_features_img\": 512,\n",
    "    \"hidden_size_img\": 256,\n",
    "    \"n_layers_img\": 2,\n",
    "    \"in_features_txt\": 512,\n",
    "    \"hidden_size_txt\": 256,\n",
    "    \"n_layers_txt\": 2,\n",
    "    \"out_features\": 128,\n",
    "    \"loss\": \"clip\",\n",
    "}\n",
    "\n",
    "model = MiniCLIP(**model_cfg).to(DEVICE)\n",
    "with torch.no_grad():\n",
    "    loss, logits = model(X.to(DEVICE), Y.to(DEVICE))\n",
    "print(logits.shape)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Because two matrices are returned by each function, we may visualize the marginals as the row and column sums of each matrix. On each iteration, one of the dimensions is balanced to be uniform."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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TzzxhFdt/PGPf1RsbKyksoHpiy++cHmcY+1kVxISEtS9e3dJMnZidFRYWKj33ntP0m8DoSv29u3n88VHH31kvL7lllsqPH7FihWyWq2yWq3q3Lmz26tXdocOHdJ1111nfOaFF15wedwtt9xiTGgffvih8f8TAAQK84Azb8b/2bNnG+P4mDFjPD7H6tWrjc/17dvX5X4CjP+AeYQ98M+ePdu4bVq9enU999xzbq+YuFOvXj21atXK+Nld3eOuXbuqSpUq2rFjh/Ly8lweM2rUKMXGxurbb7/VP/7xD+PBrfz8fD355JM6cuSImjRponvvvdfl50tKSvTtt99Kkm688Uavvofd7t27jb+Tyy+/3KNSbP379zduRefm5mrhwoVuj/311181bNgwnTp1SpLUp08fPf744y6PrVOnjq688kpJ0vHjx/X999979V0AoCLMA7/xdvy/+uqrjdffffedR+coKCjQ3//+d+PniRMnuvz7ZvwHzCOsgf/06dNasWKF8fP999/vdj1kRexhVyobmFypV6+eevbsqfPnz+uzzz5zeUxycrImT56smJgYzZ8/X926dVNqaqpuuOEGffDBB6pZs6bmzJnj9jbzhg0blJ2dreuvv17Nmzf36bt8/fXXxmtPa0vHxsY6Xd158803jfJzjs6dO6cRI0bo8OHDksqupE2bNs1tvWlJ+v3vf2+8tleNAIBAYB5w5u3436FDB2P8Pnr0qE6ePFnhZxYtWmTsD3DbbbeVexeC8R8wh7AG/uXLlxtXWOLj4/2qM5yUlGS8dld9QZJR7eD99993e8x9992nt956Sz179pTFYtHevXtVs2ZN3Xffffrwww911VVXuf3sBx98IElGSTNfON5qtlqtHn+uV69exnrRvLw8o8qDXXFxscaMGaOdO3dKKlurOnv27HLrTV/Yh+3bt3vcHwCoCPOAM2/H/xo1ajjd2ajoKv/x48c1b948SWUPHbu7u+uqD4z/QOUV1rKcH3/8sfH6xhtv9KuqjeOt2fIe8rr22mvVpUsXbdy4Ubt27XI7aHfq1EmdOnXyqg9Hjx7V6tWr1bp1a916661efdaR/WEvqawknacsFovGjh1rTJhLly7VkCFDVL9+fUnSU089pfXr10uSmjZtqvnz5ztNkO449oFbugACiXnAmS/jf8eOHbVv3z5JZaG8R48ebo+dPn26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",
      "text/plain": [
       "<Figure size 800x500 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "m1x, m1y, m2x, m2y = get_marginals(*jointly_centered_loss(logits))\n",
    "\n",
    "sns.set_style(\"white\")\n",
    "    \n",
    "fig, axes = plt.subplots(2, 2, figsize=(8, 5), sharex=True, sharey=\"row\")\n",
    "ticks = torch.arange(len(m1x))\n",
    "\n",
    "axes[0, 0].bar(ticks, m1x, color=color, width=width)\n",
    "axes[0, 1].bar(ticks, m1y, color=color, width=width)\n",
    "axes[1, 0].bar(ticks, m2x, color=color, width=width)\n",
    "axes[1, 1].bar(ticks, m2y, color=color, width=width)\n",
    "\n",
    "ms = [m1x, m1y, m2x, m2y]\n",
    "for i, ax in enumerate([axes[0, 0], axes[0, 1], axes[1, 0], axes[1, 1]]):\n",
    "    ax.set_xticklabels([])\n",
    "    ax.set_yticks([])\n",
    "    ax.set_ylim(0.0, 5 * torch.mean(ms[i]))\n",
    "\n",
    "labelpad = -24.0\n",
    "coords = (0.05, 0.7)\n",
    "axes[0, 0].text(*coords, r\"$P^{(0)}_{\\theta, X}(x)$\", transform=axes[0, 0].transAxes, fontsize=fontsize)\n",
    "axes[0, 1].text(*coords, r\"$P^{(0)}_{\\theta, Y}(y)$\", transform=axes[0, 1].transAxes, fontsize=fontsize)\n",
    "axes[1, 0].text(*coords, r\"$Q^{(0)}_{\\theta, X}(x)$\", transform=axes[1, 0].transAxes, fontsize=fontsize)\n",
    "axes[1, 1].text(*coords, r\"$Q^{(0)}_{\\theta, Y}(y)$\", transform=axes[1, 1].transAxes, fontsize=fontsize)\n",
    "\n",
    "axes[1, 0].set_xlabel(r\"$x$\")\n",
    "axes[1, 1].set_xlabel(r\"$y$\")\n",
    "\n",
    "fig.tight_layout(pad=2.0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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v7TNE0OGDNGiRYuUn5+vhQsX6vbbbw96GwCUL4wDBej/AfhT2FzhT0lJcenkBw8erPfff1933313kRujREVF6YEHHtCSJUtUoUIFSQW3aqdPn17sOa677jpJBbssBpKzo69Zs6bXNY4cOaLNmzdLkm677TbdeOONfmmbJ2644QZ17NhRkrRt2zYdOnQo6G0AUH4wDhSg/wfgb2ER+P/973/r5Zdftt4PHTpUzz33XJEd/NXatGmjkSNHWu//+c9/FtsxNWvWTJL0/fff+9jikn333XeS5NODTmvXrpXD4ZAk3X333X5plzf69OkjqeC29Nq1a0PWDgBmYxz4L/p/AP4W8sCfkZGhSZMmWZ1bx44d9fTTT3tU47e//a0iIyMlFXRMn376aZHHtWvXTpL0n//8x4cWl85Z33k+b3z00UfW6zvvvLPU41evXi273S673a62bduWevXq6NGjuv32262fefHFF4s87s4771RERME/kw8//ND63wkA/IVxwJUn/f+8efOsfnzMmDFun2PDhg3Wz/Xr10/5+fnXHEP/D5gj5IF/3rx5OnnypCSpcuXKeuGFF6wOxl01a9ZUkyZNrPfFrXvcqVMnVahQQXv27FFmZqb3jS5BXl6evv76a0lSly5dvKqxf/9+6++kcePGbi3FNnDgQDVs2FBSweC5ePHiYo/96aefNGLECJ0/f16S1LdvX02YMKHIY6tXr66bb75ZknTmzBl9++23Hn0XACgN48B/edr/33rrrdbrb775xq1zZGVl6S9/+Yv1ftKkSUX+fdP/A+YIaeD/+eeftXr1auv9gw8+qHr16nlVyxl2pYKOqSg1a9ZUz549deXKFf3jH/8o8phTp06pffv21n+TJk0q8s+ff/75In9+69atSktLU4cOHZSQkODVd/n888+t1+6uLR0ZGelydefNN9+0lp8r7NKlSxo1apSOHTsmqeBK2syZM2Wz2Yqt/Ytf/MJ67Vw1AgD8gXHAlaf9f6tWraz++8SJEzp37lypP7NkyRJrf4C77rqrxLsQ9P+AGUIa+FetWmVdYYmOjvZpneG4uDjrdUmrLzhXO3jvvfeK/DwvL0/p6enWf5cuXZIk5efnF/nnV/vggw8kyVrSzBuFbzXb7Xa3f653797WfNHMzExrlQen3NxcjRkzRnv37pUk3XLLLZo3b571sFtxCrdh9+7d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",
      "text/plain": [
       "<Figure size 800x500 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "m1x, m1y, m2x, m2y = get_marginals(*clip_loss(logits))\n",
    "\n",
    "sns.set_style(\"white\")\n",
    "    \n",
    "fig, axes = plt.subplots(2, 2, figsize=(8, 5), sharex=True, sharey=\"row\")\n",
    "ticks = torch.arange(len(m1x))\n",
    "\n",
    "axes[0, 0].bar(ticks, m1x, color=color, width=width)\n",
    "axes[0, 1].bar(ticks, m1y, color=color2, width=width)\n",
    "axes[1, 0].bar(ticks, m2x, color=color2, width=width)\n",
    "axes[1, 1].bar(ticks, m2y, color=color, width=width)\n",
    "\n",
    "ms = [m1x, m1y, m2x, m2y]\n",
    "for i, ax in enumerate([axes[0, 0], axes[0, 1], axes[1, 0], axes[1, 1]]):\n",
    "    ax.set_xticklabels([])\n",
    "    ax.set_yticks([])\n",
    "    ax.set_ylim(0.0, 5 * torch.mean(ms[i]))\n",
    "\n",
    "labelpad = -24.0\n",
    "coords = (0.05, 0.7)\n",
    "axes[0, 0].text(*coords, r\"$P^{(1)}_{\\theta, X}(x)$\", transform=axes[0, 0].transAxes, fontsize=fontsize)\n",
    "axes[0, 1].text(*coords, r\"$P^{(1)}_{\\theta, Y}(y)$\", transform=axes[0, 1].transAxes, fontsize=fontsize)\n",
    "axes[1, 0].text(*coords, r\"$Q^{(1)}_{\\theta, X}(x)$\", transform=axes[1, 0].transAxes, fontsize=fontsize)\n",
    "axes[1, 1].text(*coords, r\"$Q^{(1)}_{\\theta, Y}(y)$\", transform=axes[1, 1].transAxes, fontsize=fontsize)\n",
    "\n",
    "axes[1, 0].set_xlabel(r\"$x$\")\n",
    "axes[1, 1].set_xlabel(r\"$y$\")\n",
    "\n",
    "fig.tight_layout(pad=2.0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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OTnWw74AUBqMA+587f89eXDX4XDo5Zdftn7eTz/9tEcLgtH/A2Yos1t6Tp48ab2uWrWq/t//+38+1zp27JjbNJSuZzuK0r9/f5/3V5jQ0FD98Y9/LHWdS5cuub33Zvl3pzZt2qhnz5765JNPlJmZqblz50rKX6Rm9uzZatOmjVf1qlSpUmwbAcAXjAPufO3/mzVrpmrVqun8+fNKS0tTcnKyGjdu7LbNunXrrF8GWrdurQceeMCj2vT/gBnK9JYepxo1ahQ7S0xJrpxpoTTz35c3vv5cnnrqKWtOaqepU6eqW7du/mgWAJQa40DxPP152Gw2tW3b1np/5W096enp1qJmNptNf/7zn0v1swZw9SmzwO96luDKMwjecDgc1owIUv4qit7MbFPeVK1a1e395cuXfapz4MABt4VUWrRooX79+vlU68o2XNlGAPAF44C70vT/rlc0rrytZ8GCBTpz5oyk/HVXvLl1iv4fMEOZBf7q1atbr9PT032u869//Uvff/+99X7gwIGlaldZi46OdruM68v9kp999pnbqpVS/uwKrvM7e8O1DWFhYT49VwAAV2IccFea/t81xLue4f/hhx/09ttvS8oP608//bRXbaL/B8xQZoHf9cn/s2fP+tTZZ2dnu82w0LRp0xLvyczKytKyZcvUv39/3XzzzerQoYNGjRplzU1f1mw2mxo0aGC9//nnn736/sGDBzVmzBhlZ2dLyj/T5fTaa6/51CbXBWzq169f4FYhAPAF44C70vT/bdu2tWYFOnjwoDWbzvTp063xYPjw4apfv75XbaL/B8xQZoG/ffv21uu8vDzt3r3b6xrz5893m9JtwoQJxS4i9dNPP6l///6aMWOGjh8/rqZNm8rhcGjr1q16+OGHdfDgQa/bcKVp06bJbrdr8ODBRW6Tm5uruLg42e123XXXXdalVqeYmBjr9bFjxzzed3JyskaMGKGLFy9KkoYOHaply5YpOjpakrR//35t2rTJm8ORJB09etR63aJFC6+/DwCFMW0c2Lp1q+x2u2JiYtyuOBTm888/V6tWrRQTE6N//OMf1p/72v9HRUWpefPmkvJvlTp8+LB27typLVu2SJKuvfZaDR8+3JvDkUT/D5iizAJ/ixYtdO2111rvnZccPfXxxx8rISHBej906NBiH0g9f/68Hn30UR0+fFiPPPKIdu7cqXXr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",
      "text/plain": [
       "<Figure size 800x500 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "m1x, m1y, m2x, m2y = get_marginals(*doubly_centered_loss(logits))\n",
    "\n",
    "sns.set_style(\"white\")\n",
    "    \n",
    "fig, axes = plt.subplots(2, 2, figsize=(8, 5), sharex=True, sharey=\"row\")\n",
    "ticks = torch.arange(len(m1x))\n",
    "\n",
    "axes[0, 0].bar(ticks, m1x, color=color2, width=width)\n",
    "axes[0, 1].bar(ticks, m1y, color=color, width=width)\n",
    "axes[1, 0].bar(ticks, m2x, color=color, width=width)\n",
    "axes[1, 1].bar(ticks, m2y, color=color2, width=width)\n",
    "\n",
    "ms = [m1x, m1y, m2x, m2y]\n",
    "for i, ax in enumerate([axes[0, 0], axes[0, 1], axes[1, 0], axes[1, 1]]):\n",
    "    ax.set_xticklabels([])\n",
    "    ax.set_yticks([])\n",
    "    ax.set_ylim(0.0, 5 * torch.mean(ms[i]))\n",
    "\n",
    "labelpad = -24.0\n",
    "coords = (0.05, 0.7)\n",
    "axes[0, 0].text(*coords, r\"$P^{(2)}_{\\theta, X}(x)$\", transform=axes[0, 0].transAxes, fontsize=fontsize)\n",
    "axes[0, 1].text(*coords, r\"$P^{(2)}_{\\theta, Y}(y)$\", transform=axes[0, 1].transAxes, fontsize=fontsize)\n",
    "axes[1, 0].text(*coords, r\"$Q^{(2)}_{\\theta, X}(x)$\", transform=axes[1, 0].transAxes, fontsize=fontsize)\n",
    "axes[1, 1].text(*coords, r\"$Q^{(2)}_{\\theta, Y}(y)$\", transform=axes[1, 1].transAxes, fontsize=fontsize)\n",
    "\n",
    "axes[1, 0].set_xlabel(r\"$x$\")\n",
    "axes[1, 1].set_xlabel(r\"$y$\")\n",
    "\n",
    "fig.tight_layout(pad=2.0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "dl",
   "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.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
