{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "import sys, os\n",
    "import numpy as np\n",
    "import cupy as cp \n",
    "from cupyx.time import repeat\n",
    "import ot\n",
    "import ot.gpu\n",
    "from ot.datasets import make_1D_gauss as gauss\n",
    "import matplotlib.pyplot as plt\n",
    "from timeit import timeit\n",
    "from time import time\n",
    "import csv\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Generate data\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def save(C, nrows, ncols, filename):\n",
    "    assert C.flags['F_CONTIGUOUS']\n",
    "    output_file = open(filename, 'wb')\n",
    "    C.tofile(output_file)\n",
    "    output_file.close()\n",
    "\n",
    "def readfval(filename, dtype=np.float32):\n",
    "    f = [];\n",
    "    with open(filename) as csvfile:\n",
    "        csvReader = csv.reader(csvfile, delimiter=\",\")\n",
    "        next(csvReader)  # skip the header\n",
    "        for row in csvReader:\n",
    "            f.append(float(row[0]))\n",
    "    return np.asarray(f, dtype=dtype)\n",
    "\n",
    "def readoutput(filename):\n",
    "    with open(filename) as csvfile:\n",
    "        k = []; t = []; r = []; f = [];\n",
    "        csvReader = csv.reader(csvfile, delimiter=\",\")\n",
    "        next(csvReader)  # skip the header\n",
    "        for row in csvReader:\n",
    "            k.append(int(row[0]))\n",
    "            t.append(float(row[1]))\n",
    "            r.append(float(row[2]))\n",
    "            f.append(float(row[3]))\n",
    "    return k, t, r, f\n",
    "\n",
    "def two_dimensional_gaussian_ot(m, n, filename=None):\n",
    "    d = 2\n",
    "    mu_s = np.random.normal(0.0, 1.0, (d,)) # Gaussian mean\n",
    "    A_s = np.random.rand(d, d)\n",
    "    cov_s = np.dot(A_s, A_s.transpose()) # Gaussian covariance matrix\n",
    "    mu_t = np.random.normal(5.0, 10.0, (d,))\n",
    "    A_t = np.random.rand(d, d)\n",
    "    cov_t = np.dot(A_t, A_t.transpose())\n",
    "    xs = ot.datasets.make_2D_samples_gauss(m, mu_s, cov_s).astype(np.float32)\n",
    "    xt = ot.datasets.make_2D_samples_gauss(n, mu_t, cov_t).astype(np.float32)\n",
    "    p, q = np.ones((m,)).astype(np.float32) / m, np.ones((n,)).astype(np.float32) / n  \n",
    "    C = np.array(ot.dist(xs, xt)).astype(np.float32)\n",
    "    C /= C.max()\n",
    "    if filename is not None:\n",
    "        save(np.array(C, order='F'), m, n, filename)\n",
    "    return m, n, C, p, q"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Single Experiment"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Problem data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "m, n, C, p, q = two_dimensional_gaussian_ot(512, 512, \"data/cmatrix\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "femd = ot.emd2(p, q, C, numItermax=2000000)  # LP solver"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Drot on the same matrix C"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0, 0.5, 'Primal Residual')"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAmoAAAHgCAYAAAAVEUFcAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAgAElEQVR4nOzdd3hUZfr/8feTRiqB0FvoIiBNoqgoKDZcQdG118WCrGvXVbf+dL+urqtr11VssIoFK6LYEREFKdIFAYNAaAkt1PTn98eTBiYhZWbOlM/ruuY6Z05mztxhYef2KfdtrLWIiIiISPCJ8joAEREREamaEjURERGRIKVETURERCRIKVETERERCVJK1ERERESClBI1ERERkSAV43UA/tC8eXPbqVMnr8MQEREROaT58+dvtda2qOpnYZmoderUiXnz5nkdhoiIiMghGWPWVvczTX2KiIiIBKmwStSMMSONMeNyc3O9DkVERESkwcIqUbPWTrHWjklNTfU6FBEREZEGC8s1alUpLCwkKyuLvLw8r0Pxi/j4eNq3b09sbKzXoYiIiIiPREyilpWVRUpKCp06dcIY43U4PmWtZdu2bWRlZdG5c2evwxEREREfCaupz5rWqOXl5dGsWbOwS9IAjDE0a9YsbEcLRUREIlVYJWqHWqMWjklamXD+3URERCJVWCVqwS46Opr+/fvTu3dv+vXrxyOPPEJJSUmd7jF9+nS+++47P0UoIiIiwSRi1qgFg4SEBBYuXAhAdnY2l1xyCbm5udx7770HvK6oqIiYmKr/p5k+fTrJyckcd9xxfo9XREREvKURNY+0bNmScePG8dRTT2GtZfz48Zx//vmMHDmS0047je3btzNq1Cj69u3LMcccw+LFi/nll1949tlnefTRR+nfvz/ffPON17+GiIiI+FFYjagZY0YCI7t161bzC+/xU521e+pWaLdLly6UlJSQnZ0NwKxZs1i8eDFpaWnceOONDBgwgPfff59p06ZxxRVXsHDhQsaOHUtycjJ33HGHP34DERERCSJhNaIWigVvrbXl56eeeippaWkAzJw5k8svvxyAYcOGsW3bNtRxQUREJLKE1YhardVx5MtfMjMziY6OpmXLlgAkJSWV/6xyAldGOztFREQiS1iNqIWSnJwcxo4dyw033FBlAjZkyBAmTpwI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\n",
      "text/plain": [
       "<Figure size 720x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Remember to run DROT before executing this cell\n",
    "\n",
    "filenames = [ 'output/' + 'drot_512_0.001953.csv' ];\n",
    "labels = ['Drot']\n",
    "colors = ['C0', 'C1']\n",
    "markers = ['', '']\n",
    "fig1 = plt.figure(figsize=(10, 8))\n",
    "\n",
    "optval = 0\n",
    "\n",
    "for i in range(len(filenames)):\n",
    "    k, t, r, f = readoutput(filenames[i])\n",
    "\n",
    "    plt.plot(k, [res for res in r], color=colors[i+1], marker=markers[i], label=labels[i], linewidth=2)\n",
    "    plt.plot(k, [abs(fval - optval) for fval in f], color=colors[i], marker=markers[i], label=labels[i], linewidth=2)\n",
    "\n",
    "plt.yscale('log')\n",
    "plt.legend()\n",
    "plt.xlabel(\"Iteration\")\n",
    "plt.ylabel(\"Primal Residual\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Performance profile"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def multi_experiment(m, n, max_iters, accuracies, skregs, alpha=1.0, ntests=20, verbskip=1):\n",
    "    num_accuracies = accuracies.shape[0]\n",
    "    num_algs = skregs.shape[0] + 1\n",
    "    outs = np.zeros([num_algs, 1, num_accuracies, ntests])\n",
    "    optvals = []\n",
    "    if not os.path.isdir('data'):\n",
    "            os.mkdir('data')\n",
    "    \n",
    "    for test_idx in range(ntests):\n",
    "        print(\"\\n *** Experiment\", test_idx+1, \"of\", ntests, \"***\")\n",
    "        \n",
    "        filename = \"data/cmatrix_\" + str(m) + '_test_' + str(test_idx)\n",
    "        m, n, C, p, q = two_dimensional_gaussian_ot(m, n, filename)\n",
    "        assert(C.dtype==np.float32)\n",
    "\n",
    "        optval = ot.emd2(p, q, C, numItermax=2_000_000)\n",
    "        optvals.append(optval)\n",
    "        \n",
    "        skout = []                        \n",
    "        for reg in skregs:\n",
    "            skout.append(sinkhorn_knopp(p, q, C, reg, numItermax=1000, stopThr=1e-4))\n",
    "        \n",
    "        for sk_idx, xval in enumerate(skout):\n",
    "            outs[sk_idx+1, 0, :, test_idx] = abs(np.sum(xval*C) - optvals[-1]) / optvals[-1]\n",
    "\n",
    "    file_name = 'dims_' + str(m) + '_test_' + str(ntests)\n",
    "    np.save('output/'+file_name + '.npy', outs)\n",
    "    return file_name, optvals\n",
    "\n",
    "def profile(dir, accuracies, optvals, labels, colors):         \n",
    "    outs = np.load(dir)\n",
    "    (num_algs, num_objs_computed, num_accuracies, ntests) = outs.shape\n",
    "    performance_ratio = np.zeros((num_algs, num_accuracies))\n",
    "\n",
    "#     Read DROT's data\n",
    "    filename = 'output/drot_' + str(m) + '_ntests_' + str(ntests) + '.csv'\n",
    "    fval = readfval(filename)\n",
    "    for test_idx in range(ntests):\n",
    "            outs[0, 0, :, test_idx] = abs(fval[test_idx] - optvals[test_idx]) / optvals[test_idx]\n",
    "    \n",
    "    for alg_idx in range(num_algs):\n",
    "        for acc_idx in range(num_accuracies):\n",
    "            performance_ratio[alg_idx, acc_idx] = np.sum((outs[alg_idx, 0, acc_idx, :] <= accuracies[acc_idx])) / ntests\n",
    "\n",
    "    fig = plt.figure()        \n",
    "    for alg_idx in range(num_algs):\n",
    "        plt.plot(accuracies, performance_ratio[alg_idx, :], color=colors[alg_idx], label=labels[alg_idx], linewidth=2.5)\n",
    "  \n",
    "    ylabel = r'Performance ratio'\n",
    "    plt.xlabel(r'Accuracy')\n",
    "    plt.ylabel(ylabel)\n",
    "    plt.xscale('log')\n",
    "    plt.legend()\n",
    "        \n",
    "    return fig"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "m, n = 512, 512\n",
    "max_iters = 1000\n",
    "accuracies = np.logspace(-4.5, -1, num=15)\n",
    "skregs = np.array([1e-4, 1e-3, 5e-3, 1e-2, 5e-2, 1e-1])\n",
    "\n",
    "file_name, optvals = multi_experiment(m, n, max_iters, accuracies, skregs, alpha=2, ntests=100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "labels = ['DROT', 'SK1', 'SK2', 'SK3', 'SK4', 'Sk5', 'SK6']\n",
    "colors = ['C0', 'C1', 'C2', 'C3', 'C4', 'C5', 'C6']\n",
    "dir = \"output/\" + file_name  + '.npy'\n",
    "fig = profile(dir, accuracies, optvals, labels, colors)\n",
    "\n",
    "# fig.tight_layout()\n",
    "# fig.savefig('figures/'+ file_name + '_mean_10_f32.eps', format='eps')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Runtime performance"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Generate data for DROT and runtime for Sinkhorn"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def gen_and_run(dims, ntests=10, eps=1e-14):\n",
    "    times = np.zeros((len(dims), ntests))\n",
    "    for idx, n in enumerate(dims):\n",
    "        print(\"\\n *** m = n =\", n, \"***\")\n",
    "        m, n, C, p, q = two_dimensional_gaussian_ot(n, n, \"data/cmatrix_\" + str(n))\n",
    "        for test in range(ntests):\n",
    "            _, log = sinkhorn_knopp(p, q, C, reg=5e-2, numItermax=100, stopThr=eps, log=True)\n",
    "            times[idx, test] = log['time']\n",
    "\n",
    "    if not os.path.isdir('output'):\n",
    "        os.mkdir('output')\n",
    "    np.save('output/sinkhorn_runtime.npy', times)\n",
    "    return times\n",
    "\n",
    "def loadruntime(dims, ntests=1):\n",
    "    times = np.zeros((len(dims), ntests))\n",
    "    for idx, n in enumerate(dims):\n",
    "        for test in range(ntests):\n",
    "            k, t, _, _ = readoutput('output/drot_runtime_' + str(n) + '_test_' + str(test) + '.csv')\n",
    "            times[idx, test] = t[-1]\n",
    "#         print(k[-1])\n",
    "    return np.asarray(times)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Run "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      " *** m = n = 100 ***\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "\n",
      " *** m = n = 200 ***\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "\n",
      " *** m = n = 500 ***\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "\n",
      " *** m = n = 1000 ***\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "\n",
      " *** m = n = 2000 ***\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "\n",
      " *** m = n = 5000 ***\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "\n",
      " *** m = n = 10000 ***\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "\n",
      " *** m = n = 20000 ***\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n",
      "Terminated at iteration 100\n"
     ]
    }
   ],
   "source": [
    "dims = [100, 200, 500, 1000, 2000, 5000, 10000, 20000]\n",
    "sktimes = gen_and_run(dims, ntests=10, eps=1e-14)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0, 0.5, 'Time [ms]')"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAnEAAAHsCAYAAAC9nfs4AAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAgAElEQVR4nOzdd3RVVfrG8e/OTU8gQEIPEJBeE8CKveOADgoqoI4zlnGsiB11sI+9IFiYGX+2kaJiwd5AURSlhC69hRoCJCGk3/37Y4fQhUCSc2/u81mLNfuck5z7IDB51zl779dYaxERERGR4BLmdQARERERqTgVcSIiIiJBSEWciIiISBBSESciIiIShFTEiYiIiAQhFXEiIiIiQSjc6wBeSEpKsikpKV7HEBERETmoGTNmbLbW1t/7fEgWcSkpKUyfPt3rGCIiIiIHZYxZtb/zep0qIiIiEoRUxImIiIgEIRVxIiIiIkEoJOfE7U9xcTEZGRkUFBR4HaXaRUdHk5ycTEREhNdRRERE5BCFVBFnjOkL9G3duvU+1zIyMqhVqxYpKSkYY6o/nEestWRlZZGRkUHLli29jiMiIiKHKKRep1prJ1prr01ISNjnWkFBAYmJiSFVwAEYY0hMTAzJJ5AiIiLBLKSKuIMJtQJup1D9fYuIiAQzFXEBxOfzkZqaSqdOnejWrRvPPvssfr8fgMmTJ5OQkEBaWhrt27fn9ttv3+N7P/zwQ7p27Ur79u3p0qULH374IQA33HADqampdOzYkZiYGFJTU0lNTeW9996r9t+fiIiIVJ6QmhMX6GJiYkhPTwdg06ZNDBo0iOzsbB588EEATjrpJD755BPy8/NJS0ujX79+9OrVi9mzZ3P77bfz9ddf07JlS1asWMFZZ51Fq1atGDVqFAArV66kT58+5fcXERGR4KYncQGqQYMGjB49mpEjR2Kt3ePazidqa9euBeDpp59m2LBh5QsTWrZsyT333MNTTz1V7blFRESkeqiIC2CtWrXC7/ezadOmPc5v3bqVJUuWcPLJJwMwf/58evToscfX9OzZk/nz51dbVhEREaleep26Hw9OnM+CdTmVft+OTWozvG+nCn3P7k/hpkyZQteuXVm0aBF33303jRo1Kv+avRcn7O+ciIiI1Bwq4vZjwbocpq3Y4nUMli9fjs/no0GDBixcuLB8TtzixYs58cQT6devX/lCiOnTp9O1a9fy7505cyYdO3b0ML2IiIhUJRVx+9GxSW3P75uZmcl1113HjTfeuM8TtbZt23LPPffwxBNPMGbMGG6//XYGDBjA6aefTkpKCitXruSxxx7TClQREZEaTEXcflT0lWdlyc/PJzU1leLiYsLDw7n88ssZOnTofr/2uuuu4+mnn2bFihWkpqbyxBNP0LdvX4qLi4mIiODJJ58kNTW1mn8HIiIiUl3M3isfQ0HPnj3t9OnT9zi3cOFCOnTo4FEi74X6719ERCRQGWNmWGt77n1eq1NFREREKmrjfJj+Gnj4MEyvU0VEREQqorgA3r8GNs2Hpd/CxW9BWPU/F9OTOBEREZGK+PYhV8AB1G3pSQEHKuJEREREDt2y7+AX19KSxDZw+v2eRVERJyIiInIodmyBD6934/AouOi/EBHlWRwVcSIiIiIHYy1MvBly17vjE2+DJt08jaQiLoA8+uijdOrUia5du5Kamsq0adM49dRT2bkdysqVK2nTpg1ffvmlx0lFRERCzKy3YeFEN252LJxyp7d50OrUgPHzzz/zySefMHPmTKKioti8eTNFRUXl1zMyMjjnnHN45plnOOecczxMKiIiEmKylsHnd7lxTF0Y8DoEQH9yFXEBYv369SQlJREV5d6tJyUllV/bsGEDV1xxBY888gjnn3++VxFFRERCT2kxTLgWivPcce+noHYTbzOV0evUAHH22WezZs0a2rZty/XXX8/3339ffu2KK67gxhtvZMCAAR4mFBERCUE/PAVry7o8dbgAugbOz2I9idufz++GDXMr/76NukDvx/d7KT4+nhkzZjBlyhQmTZrEJZdcwuOPu68988wzeeutt7jyyiuJjY2t/FwiIiKyr9XTXBEHkNAM+r28x2W/3xIW5t1rVRVx+7NhLqz6sdo/1ufzceqpp3LqqafSpUsX3njjDQDuvPNO3n77bQYMGMBHH31EeLj+2ERERKpUYS5MuAasH8LC4cJ/Q2Rc+eUPZmXw7vQMnr8klQa1oz2JqGpgfxp1qfb7Llq0iLCwMNq0aQNAeno6LVq0YN68eQA899xzDBo0iKuuuorXX38dEwATKkVERGqsz++Cbavc+NjrocXx5ZeWZ27n3g/msaOolMH/mcaXQ0725Imcirj9OcArz6q0fft2brrpJrZt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\n",
      "text/plain": [
       "<Figure size 720x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Remember to run ./bin/runtime before executing this cell\n",
    "\n",
    "dims = [100, 200, 500, 1000, 2000, 5000, 10000, 20000]\n",
    "labels = ['DROT', 'SK']\n",
    "colors = ['C0', 'C1']\n",
    "\n",
    "times = loadruntime(dims, ntests=10) / 100\n",
    "sktimes = np.load('output/sinkhorn_runtime.npy') * 1000 / 100\n",
    "\n",
    "fig = plt.figure(figsize=(10, 8))\n",
    "plt.plot(dims, np.median(times, axis=1), color=colors[0], label=labels[0], linewidth=2.5)\n",
    "plt.plot(dims, np.median(sktimes, axis=1), color=colors[1], label=labels[1], linewidth=2.5)\n",
    "plt.fill_between(dims, np.quantile(times, 0.95, axis=1), np.quantile(times, 0.05, axis=1), color=colors[0], alpha=.25)  \n",
    "plt.fill_between(dims, np.quantile(sktimes, 0.95, axis=1), np.quantile(sktimes, 0.05, axis=1), color=colors[1], alpha=.25)  \n",
    "\n",
    "plt.yscale('log')\n",
    "plt.xscale('log')\n",
    "plt.legend()\n",
    "plt.xlabel(r'Dimension $m=n$', fontsize=18)\n",
    "plt.ylabel(\"Time [ms]\", fontsize=18)\n",
    "\n",
    "# fig.tight_layout()\n",
    "# fig.savefig('figures/runtime.eps', format='eps')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Mics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def sinkhorn_knopp(a, b, M, reg, numItermax=1000, stopThr=1e-9,\n",
    "                   verbose=False, log=False, to_numpy=True, **kwargs):\n",
    "    a = cp.asarray(a)\n",
    "    b = cp.asarray(b)\n",
    "    M = cp.asarray(M)\n",
    "\n",
    "    if len(a) == 0:\n",
    "        a = cp.ones((M.shape[0],), dtype=M.dtype) / M.shape[0]\n",
    "    if len(b) == 0:\n",
    "        b = cp.ones((M.shape[1],), dtype=M.dtype) / M.shape[1]\n",
    "\n",
    "    # init data\n",
    "    Nini = len(a)\n",
    "    Nfin = len(b)\n",
    "\n",
    "    if log:\n",
    "        log = {'err': [], 'time': []}\n",
    "\n",
    "    start = time()\n",
    "\n",
    "    u = cp.ones(Nini, dtype=M.dtype) / Nini\n",
    "    v = cp.ones(Nfin, dtype=M.dtype) / Nfin\n",
    "\n",
    "    K = cp.empty(M.shape, dtype=M.dtype)\n",
    "    cp.divide(M, -reg, out=K)\n",
    "    cp.exp(K, out=K)\n",
    "\n",
    "    tmp1 = cp.empty(a.shape, dtype=M.dtype)\n",
    "    tmp2 = cp.empty(b.shape, dtype=M.dtype)\n",
    "\n",
    "    Kp = (1 / a).reshape(-1, 1) * K\n",
    "     \n",
    "    cpt = 0  # CPU\n",
    "    err = 1\n",
    "    while (err > stopThr and cpt < numItermax):\n",
    "        uprev = u\n",
    "        vprev = v\n",
    "\n",
    "        KtransposeU = cp.dot(K.T, u)\n",
    "        v = cp.divide(b, KtransposeU)\n",
    "        u = 1. / cp.dot(Kp, v)\n",
    "\n",
    "        if (cp.any(KtransposeU == 0) or\n",
    "                cp.any(cp.isnan(u)) or cp.any(cp.isnan(v)) or\n",
    "                cp.any(cp.isinf(u)) or cp.any(cp.isinf(v))):\n",
    "            # come back to previous solution and quit loop\n",
    "            print('Warning: numerical errors at iteration', cpt)\n",
    "            u = uprev\n",
    "            v = vprev\n",
    "            break\n",
    "        if cpt % 10 == 0:\n",
    "            # we can speed up the process by checking for the error only all\n",
    "            # the 10th iterations\n",
    "\n",
    "            # compute right marginal tmp2= (diag(u)Kdiag(v))^T1\n",
    "            tmp1 = cp.einsum('i,ij,j->i', u, K, v)\n",
    "            tmp2 = cp.einsum('i,ij,j->j', u, K, v)\n",
    "            err = cp.sqrt(cp.linalg.norm(tmp1 - a)**2 \n",
    "                          + cp.linalg.norm(tmp2 - b)**2)  # violation of marginal\n",
    "\n",
    "            if log:\n",
    "                log['err'].append(err)\n",
    "#                 log['time'].append(err)\n",
    "\n",
    "        cpt += 1\n",
    "\n",
    "    end = time()\n",
    "    if log:\n",
    "        log['u'] = u\n",
    "        log['v'] = v\n",
    "        log['time'] = end - start\n",
    "    print(\"Terminated at iteration\",cpt)\n",
    "    res = u.reshape((-1, 1)) * K * v.reshape((1, -1))\n",
    "#     res = cp.einsum('ik,ij,jk,ij->k', u, K, v, M)\n",
    "    if to_numpy:\n",
    "        res = to_np(res)\n",
    "    if log:\n",
    "        return res, log\n",
    "    else:\n",
    "        return res\n",
    "\n",
    "def sinkhorn(a, b, M, reg, numItermax=1000,\n",
    "                   stopThr=1e-9, verbose=False, log=False, **kwargs):\n",
    "\n",
    "    if len(a) == 0:\n",
    "        a = np.full((M.shape[0],), 1.0 / M.shape[0], type_as=M)\n",
    "    if len(b) == 0:\n",
    "        b = np.full((M.shape[1],), 1.0 / M.shape[1], type_as=M)\n",
    "\n",
    "    # init data\n",
    "    dim_a = len(a)\n",
    "    dim_b = len(b)\n",
    "\n",
    "    if len(b.shape) > 1:\n",
    "        n_hists = b.shape[1]\n",
    "    else:\n",
    "        n_hists = 0\n",
    "\n",
    "    if log:\n",
    "        log = {'err': []}\n",
    "\n",
    "    # we assume that no distances are null except those of the diagonal of\n",
    "    # distances\n",
    "    if n_hists:\n",
    "        u = np.ones((dim_a, n_hists), dtpye=M.dtype) / dim_a\n",
    "        v = np.ones((dim_b, n_hists), dtpye=M.dtype) / dim_b\n",
    "    else:\n",
    "        u = np.ones(dim_a).astype(M.dtype) / dim_a\n",
    "        v = np.ones(dim_b).astype(M.dtype) / dim_b\n",
    "\n",
    "    K = np.exp(M / (-reg))\n",
    "\n",
    "    Kp = (1 / a).reshape(-1, 1) * K\n",
    "    cpt = 0\n",
    "    err = 1\n",
    "    while (err > stopThr and cpt < numItermax):\n",
    "        uprev = u\n",
    "        vprev = v\n",
    "\n",
    "        KtransposeU = np.dot(K.T, u)\n",
    "        v = b / KtransposeU\n",
    "        u = 1. / np.dot(Kp, v)\n",
    "\n",
    "        if (np.any(KtransposeU == 0)\n",
    "                or np.any(np.isnan(u)) or np.any(np.isnan(v))\n",
    "                or np.any(np.isinf(u)) or np.any(np.isinf(v))):\n",
    "            # we have reached the machine precision\n",
    "            # come back to previous solution and quit loop\n",
    "            print('Warning: numerical errors at iteration', cpt)\n",
    "            u = uprev\n",
    "            v = vprev\n",
    "            break\n",
    "        if cpt % 10 == 0:\n",
    "            # we can speed up the process by checking for the error only all\n",
    "            # the 10th iterations\n",
    "            if n_hists:\n",
    "                tmp2 = np.einsum('ik,ij,jk->jk', u, K, v)\n",
    "            else:\n",
    "                # compute right marginal tmp2= (diag(u)Kdiag(v))^T1\n",
    "                tmp1 = np.einsum('i,ij,j->i', u, K, v)\n",
    "                tmp2 = np.einsum('i,ij,j->j', u, K, v)\n",
    "            err = np.linalg.norm(tmp2 - b)  # violation of marginal\n",
    "            if log:\n",
    "                log['err'].append(err)\n",
    "\n",
    "            if verbose:\n",
    "                if cpt % 2 == 0:\n",
    "                    print(\n",
    "                        '{:5s}|{:12s}'.format('It.', 'Err') + '\\n' + '-' * 19)\n",
    "                print('{:5d}|{:8e}|'.format(cpt, err))\n",
    "        cpt = cpt + 1\n",
    "    if log:\n",
    "        log['u'] = u\n",
    "        log['v'] = v\n",
    "\n",
    "    if n_hists:  # return only loss\n",
    "        res = np.einsum('ik,ij,jk,ij->k', u, K, v, M)\n",
    "        if log:\n",
    "            return res, log\n",
    "        else:\n",
    "            return res\n",
    "\n",
    "    else:  # return OT matrix\n",
    "\n",
    "        if log:\n",
    "            return u.reshape((-1, 1)) * K * v.reshape((1, -1)), log\n",
    "        else:\n",
    "            return u.reshape((-1, 1)) * K * v.reshape((1, -1))\n",
    "\n",
    "def to_gpu(*args):\n",
    "    \"\"\" Upload numpy arrays to GPU and return them\"\"\"\n",
    "    if len(args) > 1:\n",
    "        return (cp.asarray(x) for x in args)\n",
    "    else:\n",
    "        return cp.asarray(args[0])\n",
    "\n",
    "\n",
    "def to_np(*args):\n",
    "    \"\"\" convert GPU arras to numpy and return them\"\"\"\n",
    "    if len(args) > 1:\n",
    "        return (cp.asnumpy(x) for x in args)\n",
    "    else:\n",
    "        return cp.asnumpy(args[0])    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Mics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def two_dimensional_gaussian_ot_f64(m, n):\n",
    "    mu_s = np.array([0, 0.5])\n",
    "    cov_s = np.array([[1, 0.5], [0.5, 1]])\n",
    "    mu_t = np.array([4, 10])\n",
    "    cov_t = np.array([[1, 0.85], [0.85, 1]])\n",
    "    xs = ot.datasets.make_2D_samples_gauss(m, mu_s, cov_s)\n",
    "    xt = ot.datasets.make_2D_samples_gauss(n, mu_t, cov_t)\n",
    "    p, q = np.ones((m,)) / m, np.ones((n,)) / n  \n",
    "    C = np.array(ot.dist(xs, xt))\n",
    "    C /= C.max()\n",
    "    return m, n, C, p, q\n",
    "m, n, C, p, q = two_dimensional_gaussian_ot_f64(4000, 4000)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "reg = 0.01\n",
    "# x_sk = sinkhorn(p, q, C, reg, numItermax=1000, stopThr=1e-14, verbose=False)\n",
    "\n",
    "x_gpu = sinkhorn_knopp(p, q, C, reg, numItermax=1000, stopThr=1e-14)\n",
    "# %timeit x_sk = ot.sinkhorn(p, q, C, reg, stopThr=1e-14)\n",
    "# d_sk = np.sum(x_sk*C) \n",
    "# d_gpu = cp.sum(x_gpu*C)\n",
    "\n",
    "# femd, d_sk, d_gpu\n",
    "# def run_sk(p, q, C, reg):\n",
    "#     sinkhorn_knopp(p, q, C, reg, numItermax=1000, stopThr=1e-14, to_numpy=False)\n",
    "# print(repeat(run_sk, (p, q, C, 1e-2), n_repeat=1))      "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "C_ = cp.asarray(C)\n",
    "p_ = cp.asarray(p)\n",
    "q_ = cp.asarray(q)\n",
    "\n",
    "print(C_.device)\n",
    "\n",
    "def run_gemv(A, x):\n",
    "    return A.T.dot(x), A.dot(x)\n",
    "\n",
    "print(repeat(run_gemv, (C_, p_), n_repeat=1))  \n",
    "\n",
    "def run_exp(M, x, reg=1e-2):\n",
    "    K = cp.empty(M.shape, dtype=M.dtype)\n",
    "    cp.divide(M, -reg, out=K)\n",
    "    cp.exp(K, out=K)\n",
    "    o = K.T.dot(x)\n",
    "\n",
    "print(repeat(run_exp, (C_, p_, 1e-2), n_repeat=1))     \n",
    "\n",
    "def run_setup(a, b, M, reg):\n",
    "#     a = cp.asarray(a)\n",
    "#     b = cp.asarray(b)\n",
    "#     M = cp.asarray(M)\n",
    "\n",
    "    if len(a) == 0:\n",
    "        a = cp.ones((M.shape[0],), dtype=M.dtype) / M.shape[0]\n",
    "    if len(b) == 0:\n",
    "        b = cp.ones((M.shape[1],), dtype=M.dtype) / M.shape[1]\n",
    "\n",
    "    # init data\n",
    "    Nini = len(a)\n",
    "    Nfin = len(b)\n",
    "\n",
    "    u = cp.ones(Nini, dtype=M.dtype) / Nini\n",
    "    v = cp.ones(Nfin, dtype=M.dtype) / Nfin\n",
    "\n",
    "    K = cp.empty(M.shape, dtype=M.dtype)\n",
    "    cp.divide(M, -reg, out=K)\n",
    "    cp.exp(K, out=K)\n",
    "    \n",
    "    tmp1 = cp.empty(a.shape, dtype=M.dtype)\n",
    "    tmp2 = cp.empty(b.shape, dtype=M.dtype)\n",
    "\n",
    "    Kp = (1 / a).reshape(-1, 1) * K\n",
    "    \n",
    "    uprev = u\n",
    "    vprev = v\n",
    "    \n",
    "    KtransposeU = cp.dot(K.T, u)\n",
    "    v = b / KtransposeU\n",
    "    u = 1. / cp.dot(Kp, v)\n",
    "\n",
    "print(repeat(run_setup, (p_, q_, C_, 1e-2), n_repeat=1))        \n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.7.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
