{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "import sys, os\n",
    "# sys.path.append(\"../\")\n",
    "import numpy as np\n",
    "import cupy as cp\n",
    "import scipy as sp\n",
    "import numpy.linalg as nla\n",
    "from cupyx.time import repeat\n",
    "import matplotlib.pyplot as plt\n",
    "import ot\n",
    "import ot.gpu\n",
    "from ot.datasets import make_1D_gauss as gauss\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "from timeit import timeit\n",
    "import csv\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Generate data\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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",
    "def readcsv(filename):\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=np.float32)\n",
    "\n",
    "# def two_dimensional_gaussian_ot_64(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, -.8], [-.8, 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), order='F')\n",
    "#     C /= C.max()\n",
    "#     C32 = np.array(C, order='F').astype(np.float32)\n",
    "#     save(C32, m, n, \"../gpu/data/cmatrix\")\n",
    "#     return m, n, C, p, q\n",
    "\n",
    "def two_dimensional_gaussian_ot(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, -.8], [-.8, 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,)).astype(np.float32) / m, np.ones((n,)).astype(np.float32) / n  \n",
    "    C = np.array(ot.dist(xs, xt), order='F').astype(np.float32)\n",
    "    C /= C.max()\n",
    "    return m, n, C, p, q"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 224,
   "metadata": {},
   "outputs": [],
   "source": [
    "m, n, C, p, q = two_dimensional_gaussian_ot_64(1000, 1000)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Compute EMD for the different losses\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 225,
   "metadata": {},
   "outputs": [],
   "source": [
    "femd = ot.emd2(p, q, C.copy(), numItermax=100000)  # direct computation of EMD"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Single run"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 155,
   "metadata": {
    "collapsed": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0, 0.5, 'Primal Residual')"
      ]
     },
     "execution_count": 155,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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LcouIiKSgloWvPUpEGlX8zpqiXGRlKiJiIhESy1qCaH5Wp+JcY2UqImIiERJS0glBnV9ioiIpKCWY9T8iUPa12xR9gS5SErUREREotSiRS1BkoBUo/IcIiIiKSjyO19LSAVTeHKWKJcoqRI1LSElIiJ+0MoEiSH8MiXKMl9JlahpCSkREfGD6qglBnV9ioiIpCDVUUsM4XXUEiRPU6ImIiISrRZrfSZKFpDCEmUcoRI1ERGRKLUco5YYSUCqsdbvCLpOiZqIiEiUWoxR07erxIj+lERERKKkFjXpLkrUREREohTZopYo459Sjbo+RUREUpBmfUp3UaImIiISpcgGtEQppirBp0RNREQkSpFdner6lFhRoiYiIhKllisT+BOHJB8laiIiIlFqOUZNmVoQWRJvNoESNRERkShF5mWqoyaxkuF3AB0xxpwAXIqL9RBr7TSfQxIREWnGoDFqiSDyOiWCbs35jTH3GWO2G2OWReyfYYxZZYxZY4y5tb1zWGvftNZeBzwL/K074xURETkQ4/r3pG/PbACK8rI4eEC+zxFJa75w5GDS0wznTh7idyidZmw3Vn8zxpwI7AMetNZO9PalA6uB04ASYAFwMZAO/DziFFdZa7d7z3sM+Iq1dk9Hr1tcXGwXLlwYs/chIiLSkfoGS219A5npaaRrNkFgha5RkBhjFllri1t7rFu7Pq21bxhjRkTsngKssdau9YJ7FJhprf05cGZr5zHGDAPKO5OkiYiI+CE9zZCelu53GNKBoCVpHfEj2sHAxrDtEm9fe64G7m/vAGPMLGPMQmPMwtLS0ihDFBEREfGfH4laa+3B7fa/Wmt/aK19p4NjZltri621xf369YsqQBEREZEg8CNRKwGGhm0PATb7EIeIiIhIoPmRqC0AxhpjRhpjsoCLgH/H4sTGmLOMMbPLy8tjcToRERERX3V3eY5HgHeB8caYEmPM1dbaOuBrwFxgBfCYtXZ5LF7PWjvHWjuroKAgFqcTERER8VV3z/q8uI39zwHPdedri4iIiCS6xJqj2gF1fYqIiEgySapETV2fIiIikkySKlETERERSSZK1EREREQCKqkSNY1RExERkWSSVImaxqiJiIhIMkmqRE1EREQkmShRExEREQmopErUNEZNREREkklSJWoaoyYiIiLJJKkSNREREZFkYqy1fscQc8aYUmB9N79M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\n",
      "text/plain": [
       "<Figure size 720x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "filenames = [ '../gpu/output/' + 'drot_5000_0.000010.csv' ];\n",
    "labels = ['Drot']\n",
    "colors = ['C0', 'C1']\n",
    "markers = ['', '']\n",
    "fig1 = plt.figure(figsize=(10, 8))\n",
    "\n",
    "optval = femd\n",
    "\n",
    "for i in range(len(filenames)):\n",
    "    k = []; t = []; r = []; f = [];\n",
    "    with open(filenames[i]) as csvfile:\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",
    "\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",
    "\n",
    "plt.yscale('log')\n",
    "plt.legend()\n",
    "plt.xlabel(\"Iteration\")\n",
    "plt.ylabel(\"Primal Residual\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 228,
   "metadata": {
    "collapsed": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0, 0.5, 'Primal Residual')"
      ]
     },
     "execution_count": 228,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "filenames = [ '../gpu/output/' + 'drot_1000_0.000500.csv' ];\n",
    "labels = ['Drot']\n",
    "colors = ['C0', 'C1']\n",
    "markers = ['', '']\n",
    "fig1 = plt.figure(figsize=(10, 8))\n",
    "\n",
    "for i in range(len(filenames)):\n",
    "    k = []; t = []; r = []; g = [];\n",
    "    with open(filenames[i]) as csvfile:\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",
    "            g.append(float(row[3]))\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(gap-optval) for gap in g], color=colors[i], marker=markers[i], label=labels[i], linewidth=2)\n",
    "\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": 33,
   "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",
    "    \n",
    "    for test_idx in range(ntests):\n",
    "        print(\"\\n *** Experiment\", test_idx+1, \"of\", ntests, \"***\")\n",
    "\n",
    "        m, n, C, p, q = two_dimensional_gaussian_ot(m, n)\n",
    "        filename = \"data/cmatrix_\" + str(m) + '_test_' + str(test_idx)\n",
    "        assert(C.dtype==np.float32)\n",
    "        save(C, m, n, filename)\n",
    "\n",
    "        x0 = np.array(np.outer(p, q), order = 'F')\n",
    "        step = alpha / (m+n)\n",
    "        \n",
    "        C_ = C.copy()\n",
    "        G0 = ot.emd(p, q, C_, numItermax=1_000_000)\n",
    "        optval = np.sum(G0 * C_)\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, fval in enumerate(skout):\n",
    "            outs[sk_idx+1, 0, :, test_idx] = abs(np.sum(fval*C_) - optval) / optval\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) + '_' + str(ntests) + '.csv'\n",
    "#     drout = readcsv(filename)\n",
    "#     for test_idx in range(drout.shape[0]):\n",
    "#             outs[0, 0, :, test_idx] = abs(np.sum(drout[test_idx]*C) - optval) / optval\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'Final accuracy')\n",
    "    plt.ylabel(ylabel)\n",
    "    plt.xscale('log')\n",
    "    plt.legend()\n",
    "        \n",
    "    return fig"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {
    "collapsed": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      " *** Experiment 1 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 2 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 3 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 4 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 5 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 6 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 7 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 8 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 9 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 10 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 11 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 12 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 13 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 14 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 15 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 16 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 17 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 18 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 19 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 20 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 21 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 22 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 23 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 24 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 25 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 26 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 27 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 28 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 29 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 30 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 31 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 32 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 33 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 34 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 35 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 36 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 37 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 38 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 39 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 40 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 41 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 42 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 43 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 44 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 45 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 46 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 47 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 48 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 49 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "\n",
      " *** Experiment 50 of 50 ***\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n",
      "Warning: numerical errors at iteration 0\n"
     ]
    }
   ],
   "source": [
    "m, n = 500, 500\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=50)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "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"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Compute Sinkhorn for the different losses\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "NumIter:  10\n"
     ]
    }
   ],
   "source": [
    "reg = 0.01\n",
    "# x_sk = ot.sinkhorn(p, q, C.copy(), reg)\n",
    "# x_gpu = ot.gpu.sinkhorn(p, q, C, reg)\n",
    "\n",
    "x_gpu = sinkhorn_knopp(p, q, C.copy(), reg, numItermax=1000, stopThr=1e-4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "651 ms ± 3.25 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)\n"
     ]
    }
   ],
   "source": [
    "%timeit x_sk = ot.sinkhorn(p, q, C, reg, stopThr=1e-4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(0.39248813228914514, 0.40523145572242397, 0.39830318)"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# d_sk = np.sum(x_sk*C) \n",
    "d_gpu = cp.sum(x_gpu*C)\n",
    "\n",
    "d_emd, d_sk, d_gpu"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.032468047782629135"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "(d_sk-d_emd)/d_emd"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "NumIter:  10\n",
      "NumIter:  10\n",
      "NumIter:  10\n",
      "NumIter:  10\n",
      "NumIter:  10\n",
      "NumIter:  10\n",
      "NumIter:  10\n",
      "NumIter:  10\n",
      "NumIter:  10\n",
      "NumIter:  10\n",
      "NumIter:  10\n",
      "run_sk              :    CPU:75640.623 us     GPU-0:79221.024 us\n"
     ]
    }
   ],
   "source": [
    "def run_sk(p, q, C, reg):\n",
    "    sinkhorn_knopp(p, q, C, reg, numItermax=1000, stopThr=1e-4, 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": [
    "%timeit sinkhorn_knopp(p, q, C, reg, numItermax=100, stopThr=1e-4, to_numpy=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "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",
    "   \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': []}\n",
    "\n",
    "    u = cp.ones(Nini, dtype=M.dtype) / Nini\n",
    "    v = cp.ones(Nfin, dtype=M.dtype) / Nfin\n",
    "\n",
    "    # Next 3 lines equivalent to K= cp.exp(-M/reg), but faster to compute\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",
    "            # 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",
    "\n",
    "            # compute right marginal tmp2= (diag(u)Kdiag(v))^T1\n",
    "            tmp1 = cp.sum(u[:, None] * K * v[None, :], 1)\n",
    "            tmp2 = cp.sum(u[:, None] * K * v[None, :], 0)\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",
    "\n",
    "        cpt += 1\n",
    "        \n",
    "    if log:\n",
    "        log['u'] = u\n",
    "        log['v'] = v\n",
    "\n",
    " \n",
    "    res = u.reshape((-1, 1)) * K * v.reshape((1, -1))\n",
    "    if to_numpy:\n",
    "        res = to_np(res)\n",
    "    if log:\n",
    "        return res, log\n",
    "    else:\n",
    "        return res\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "def euclidean_distances(a, b, squared=False, to_numpy=True):\n",
    "    \"\"\"\n",
    "    Compute the pairwise euclidean distance between matrices a and b.\n",
    "    If the input matrix are in numpy format, they will be uploaded to the\n",
    "    GPU first which can incur significant time overhead.\n",
    "    Parameters\n",
    "    ----------\n",
    "    a : np.ndarray (n, f)\n",
    "        first matrix\n",
    "    b : np.ndarray (m, f)\n",
    "        second matrix\n",
    "    to_numpy : boolean, optional (default True)\n",
    "        If true convert back the GPU array result to numpy format.\n",
    "    squared : boolean, optional (default False)\n",
    "        if True, return squared euclidean distance matrix\n",
    "    Returns\n",
    "    -------\n",
    "    c : (n x m) np.ndarray or cupy.ndarray\n",
    "        pairwise euclidean distance distance matrix\n",
    "    \"\"\"\n",
    "\n",
    "    a, b = to_gpu(a, b)\n",
    "\n",
    "    a2 = np.sum(np.square(a), 1)\n",
    "    b2 = np.sum(np.square(b), 1)\n",
    "\n",
    "    c = -2 * np.dot(a, b.T)\n",
    "    c += a2[:, None]\n",
    "    c += b2[None, :]\n",
    "\n",
    "    if not squared:\n",
    "        np.sqrt(c, out=c)\n",
    "    if to_numpy:\n",
    "        return to_np(c)\n",
    "    else:\n",
    "        return c\n",
    "\n",
    "\n",
    "def dist(x1, x2=None, metric='sqeuclidean', to_numpy=True):\n",
    "    \"\"\"Compute distance between samples in x1 and x2 on gpu\n",
    "    Parameters\n",
    "    ----------\n",
    "    x1 : np.array (n1,d)\n",
    "        matrix with n1 samples of size d\n",
    "    x2 : np.array (n2,d), optional\n",
    "        matrix with n2 samples of size d (if None then x2=x1)\n",
    "    metric : str\n",
    "        Metric from 'sqeuclidean', 'euclidean',\n",
    "    Returns\n",
    "    -------\n",
    "    M : np.array (n1,n2)\n",
    "        distance matrix computed with given metric\n",
    "    \"\"\"\n",
    "    if x2 is None:\n",
    "        x2 = x1\n",
    "    if metric == \"sqeuclidean\":\n",
    "        return euclidean_distances(x1, x2, squared=True, to_numpy=to_numpy)\n",
    "    elif metric == \"euclidean\":\n",
    "        return euclidean_distances(x1, x2, squared=False, to_numpy=to_numpy)\n",
    "    else:\n",
    "        raise NotImplementedError\n",
    "\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": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "m, n = 4, 20\n",
    "A = np.random.randn(m, n)\n",
    "e = np.ones(n)\n",
    "f = np.ones(m)\n",
    "yy = np.random.randn(m,1)\n",
    "xx = np.random.randn(n)\n",
    "assert np.allclose(A - np.outer(yy, e) - np.outer(f, xx), A - yy - xx)\n",
    "D = np.outer(yy, e) + np.outer(f, xx)\n",
    "[np.allclose(D[i, j], yy[i] + xx[j]) for i, j in zip(range(yy.size), range(xx.size))]\n",
    "\n",
    "np.outer(yy, e)\n",
    "np.outer(f, xx)\n",
    "D = np.outer(yy, e) + np.outer(f, xx)\n",
    "[np.allclose(D[i, j], yy[i] + xx[j]) for i, j in zip(range(yy.size), range(xx.size))]\n",
    "i, j = 3, 4\n",
    "\n",
    "D[i, j], yy[i] + xx[j]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_max(X):\n",
    "    return cp.asarray(X)\n",
    "run_max(C)\n",
    "\n",
    "print(repeat(run_max, (C,), n_repeat=1))  \n",
    "def run_gemv(A, x):\n",
    "    return A.T.dot(x), A.dot(x)\n",
    "\n",
    "C_ = run_max(C)\n",
    "p_ = cp.asarray(p)\n",
    "\n",
    "print(C_.device)\n",
    "\n",
    "print(repeat(run_gemv, (C_, p_), n_repeat=1))  \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=20))        \n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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
}
