{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "collapsed": true,
    "pycharm": {
     "name": "#%%\n"
    }
   },
   "outputs": [],
   "source": [
    "import cvxpy as cp\n",
    "import numpy as np\n",
    "import osqp\n",
    "from scipy import sparse\n",
    "from tqdm import tqdm\n",
    "import matplotlib.pyplot as plt\n",
    "import warnings\n",
    "import torch\n",
    "import pandas as pd\n",
    "from qpth.qp import QPFunction\n",
    "import time\n",
    "warnings.filterwarnings('ignore')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "outputs": [],
   "source": [
    "def osqp_interface(P,q,A,l,u):\n",
    "    prob = osqp.OSQP()\n",
    "    prob.setup(P, q, A, l, u,verbose = False)\n",
    "    t0 = time.time()\n",
    "    res = prob.solve()\n",
    "    return res.x,res.y,time.time() - t0"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "outputs": [],
   "source": [
    "ndim = 20\n",
    "neq = 10\n",
    "nineq = 10"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "outputs": [],
   "source": [
    "P = np.random.random((ndim,ndim))\n",
    "P = P.T@P+(0.0001*np.eye(ndim,ndim))\n",
    "Pm = P.copy()\n",
    "P = sparse.csc_matrix(P)\n",
    "q = np.random.random(ndim)\n",
    "A = np.random.random((neq,ndim))\n",
    "G = np.random.random((nineq,ndim))\n",
    "b = np.random.random(neq)\n",
    "h = G@np.random.random(ndim)\n",
    "osA = np.vstack([G,A])\n",
    "osA = sparse.csc_matrix(osA)\n",
    "l = np.hstack([-np.inf*np.ones(nineq),b])\n",
    "u = np.hstack([h,b])"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "# 1.OSQP Forward"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%% md\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "OSQP Forward Time spent: 0.0001621246337890625\n"
     ]
    }
   ],
   "source": [
    "x_value, y_value, time_spent = osqp_interface(P,q,osA,l,u)\n",
    "print('OSQP Forward Time spent:',time_spent)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "# 2.OSQP Backward"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%% md\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "OSQP Backward Time spent: 5.936622619628906e-05\n"
     ]
    }
   ],
   "source": [
    "lambs = y_value[:nineq] # active set\n",
    "active_set = np.argwhere(lambs>1e-8)\n",
    "bG = G[active_set,:].squeeze()\n",
    "bb = np.zeros(neq)\n",
    "bh = np.zeros(len(active_set))\n",
    "bq = np.ones(ndim)\n",
    "osnewA = np.vstack([bG,A])\n",
    "osnewA = sparse.csc_matrix(osnewA)\n",
    "l_new = np.hstack([bh,bb])\n",
    "u_new = np.hstack([bh,bb])\n",
    "\n",
    "x_grad, y_grad, time_spent_backward = osqp_interface(P,bq,osnewA,l_new,u_new)\n",
    "print('OSQP Backward Time spent:',time_spent_backward)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "# 3.CVXPY Backward"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%% md\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CVXPY Forward Time Spent: 0.06544685363769531\n",
      "CVXPY Backward Time Spent: 0.003671884536743164\n"
     ]
    }
   ],
   "source": [
    "qq = cp.Parameter(ndim)\n",
    "qq.value = q\n",
    "x1 = cp.Variable(ndim)\n",
    "prob = cp.Problem(cp.Minimize((1 / 2) * cp.quad_form(x1, P) + qq.T @ x1),\n",
    "                              [G @ x1 <= h,\n",
    "                               A @ x1 == b])\n",
    "t3 = time.time()\n",
    "prob.solve(requires_grad = True,solver='SCS')\n",
    "print('CVXPY Forward Time Spent:',time.time() - t3)\n",
    "t4 = time.time()\n",
    "prob.backward()\n",
    "t4 = time.time() - t4\n",
    "print('CVXPY Backward Time Spent:',t4)\n",
    "total_time_cvxpy = time.time() - t3"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "qpth Forward Time Spent: 0.8730852603912354\n",
      "qpth Backward Time Spent: 0.005921363830566406\n"
     ]
    }
   ],
   "source": [
    "qpf = QPFunction(verbose=0,maxIter=500)\n",
    "tP, tq, tG, th, tA, tb = [torch.Tensor(x).unsqueeze(0).cuda() for x in [Pm, q, G, h, A, b]]\n",
    "tq.requires_grad = True\n",
    "t6 = time.time()\n",
    "qpth_x_value = qpf(tP, tq, tG, th, tA, tb)\n",
    "print('qpth Forward Time Spent:', time.time() - t6)\n",
    "t7 = time.time()\n",
    "qpth_x_value.backward(torch.ones(1, ndim).cuda())\n",
    "print('qpth Backward Time Spent:', time.time() - t7)\n",
    "qpth_x_grad = tq.grad.squeeze().cpu().numpy()\n",
    "qpth_x_value = qpth_x_value.squeeze().detach().cpu().numpy()\n",
    "total_time_qpth = time.time() - t6"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "# 4. Exact backward (Matrix inverse, Gould et al. 2022)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%% md\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Exact Backward Time Spent 0.0015475749969482422\n"
     ]
    }
   ],
   "source": [
    "KKT_L1 = np.hstack([Pm,G.T,A.T])\n",
    "KKT_L2 = np.hstack([np.diag(lambs)@G, np.diag(G@x_value-h),np.zeros((nineq,neq))])\n",
    "KKT_L3 = np.hstack([A, np.zeros((neq,neq)),np.zeros((neq,nineq))])\n",
    "KKT = np.vstack([KKT_L1,KKT_L2,KKT_L3])\n",
    "t5 = time.time()\n",
    "exact_bb =-(np.linalg.inv(KKT)@np.hstack([np.ones(ndim),np.zeros(nineq),np.zeros(neq)]))[:ndim]\n",
    "t5 = time.time()-t5\n",
    "print(\"Exact Backward Time Spent\", t5)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "# 5. Alt-diff, Sun et al. 2023"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%% md\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "outputs": [],
   "source": [
    "import math\n",
    "import scipy\n",
    "def relu(s):\n",
    "    ss = s\n",
    "    for i in range(len(s)):\n",
    "        if s[i]<0:\n",
    "            ss[i] = 0\n",
    "    return ss\n",
    "\n",
    "def proj(s):\n",
    "    ss = s\n",
    "    for i in range(len(s)):\n",
    "        if s[i]<0:\n",
    "            ss[i] = (ss[i] + math.sqrt(ss[i] ** 2 + 4 * 0.001))/2\n",
    "    return ss\n",
    "\n",
    "def sgn(s,p):\n",
    "    ss = np.zeros(p)\n",
    "    for i in range(len(s)):\n",
    "        if s[i]<=0:\n",
    "            ss[i] = 0\n",
    "        else:\n",
    "            ss[i] = 1\n",
    "    return ss\n",
    "\n",
    "def alt_diff(P,q,A, b, G, h):\n",
    "    n, m, p = len(P), len(A), len(G)\n",
    "    sk = np.zeros(p)\n",
    "    lamb = np.zeros(m)\n",
    "    nu = np.zeros(p)\n",
    "\n",
    "    dxk = np.zeros((m, n))\n",
    "    dsk = np.zeros((p, n))\n",
    "    dlamb = np.zeros((m, n))\n",
    "    dnu = np.zeros((p, n))\n",
    "\n",
    "    rho = 1\n",
    "    M = P + rho * A.T @ A + rho * G.T @ G\n",
    "\n",
    "    begininv = time.time()\n",
    "    R = - np.linalg.inv(P + rho * A.T @ A + rho * G.T @ G)\n",
    "    endinv = time.time()\n",
    "    print(\"The inverse time is\", endinv-begininv)\n",
    "\n",
    "    coef = rho * A.T @ A + rho * G.T @ G\n",
    "\n",
    "    res = [1000, -100]\n",
    "    # thres = 1e-5\n",
    "\n",
    "    begin2 = time.time()\n",
    "    xk = np.ones(n)\n",
    "    thres = 1e-4\n",
    "    M_csr = scipy.sparse.csr_matrix(M)\n",
    "    eta = 0.5\n",
    "    while abs((np.linalg.norm(res[-1]) - np.linalg.norm(res[-2])) / np.linalg.norm(res[-2])) > thres:\n",
    "\n",
    "        xk = R @ (q + A.T @ lamb + G.T @ nu - rho * A.T @ b + rho * G.T @ (sk - h))\n",
    "\n",
    "        dxk = R @ (np.ones(n) + A.T @ dlamb + G.T @ dnu  + rho * G.T @ dsk)\n",
    "        sk = relu(- (1 / rho) * nu - (G @ xk - h))\n",
    "\n",
    "        dsk = (-1 / rho) * sgn(sk,p).reshape(p, 1) @ np.ones((1,n)) * (dnu + rho * G @ dxk)\n",
    "\n",
    "        lamb = lamb + rho * (A @ xk - b)\n",
    "        dlamb = dlamb + rho * (A @ dxk)\n",
    "        nu = nu + rho * (G @ xk + sk - h)\n",
    "        dnu = dnu + rho * (G @ dxk + dsk)\n",
    "        res.append(xk)\n",
    "\n",
    "    y_f = dxk.T[0]\n",
    "    end2 = time.time()\n",
    "    return xk, y_f, endinv-begininv, end2-begin2"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002791881561279297\n",
      "Alt-diff Total Time Spent 0.020316123962402344\n"
     ]
    }
   ],
   "source": [
    "t6 = time.time()\n",
    "xk, dq,_,_ = alt_diff(Pm, q, A, b, G, h)\n",
    "t6 = time.time()-t6\n",
    "print(\"Alt-diff Total Time Spent\", t6)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "# 6.JAXOpt, Blondel et al., 2021"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%% md\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "outputs": [],
   "source": [
    "import jax\n",
    "import time\n",
    "import jaxopt\n",
    "import jax.numpy as jnp\n",
    "# from jaxopt import OSQP\n",
    "from jax import random"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "WARNING:jax._src.lib.xla_bridge:No GPU/TPU found, falling back to CPU. (Set TF_CPP_MIN_LOG_LEVEL=0 and rerun for more info.)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "JAXOpt Backward time spent 5.197052478790283\n"
     ]
    }
   ],
   "source": [
    "def jax_qp_solver(c):\n",
    "    qp = jaxopt.OSQP()\n",
    "    sol = qp.run(params_obj=(Q, c), params_eq=(A, b), params_ineq=(G, h)).params\n",
    "    loss = jnp.dot(sol.primal,jnp.ones(ndim))\n",
    "    return loss\n",
    "key = random.PRNGKey(758493)\n",
    "Q = Pm\n",
    "c = q\n",
    "qp = jaxopt.OSQP()\n",
    "sol = qp.run(params_obj=(Q, c), params_eq=(A, b), params_ineq=(G, h)).params\n",
    "\n",
    "t7 = time.time()\n",
    "jax_gradient = jax.grad(jax_qp_solver)(c)\n",
    "t7 = time.time() - t7\n",
    "print('JAXOpt Backward time spent', t7)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "# 7. Accuracy Analysis"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%% md\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "outputs": [],
   "source": [
    "def cal_cos_accuracy(x_exact,x_approx):\n",
    "    return np.dot(x_exact,x_approx)/(np.linalg.norm(x_exact)*np.linalg.norm(x_approx))"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Forward pass\n",
      "      CVXPY      qpth      BPQP  Alt-diff    JAXOpt\n",
      "0  0.132419  0.132420  0.132419  0.133603  0.132297\n",
      "1 -0.443361 -0.443361 -0.443361 -0.442038 -0.443239\n",
      "2  0.007332  0.007332  0.007332  0.008804  0.007357\n",
      "3  0.037662  0.037663  0.037662  0.036391  0.037604\n",
      "4 -0.121533 -0.121533 -0.121533 -0.122703 -0.121578\n",
      "Backward pass\n",
      "      CVXPY      qpth      BPQP  Alt-diff    JAXOpt     exact\n",
      "0 -0.045199 -0.045218 -0.045215 -0.045214 -0.047433 -0.045215\n",
      "1 -0.088725 -0.088909 -0.088909 -0.088977 -0.075136 -0.088909\n",
      "2  0.073655  0.073632  0.073634  0.073617  0.076985  0.073634\n",
      "3 -0.049887 -0.049774 -0.049775 -0.049736 -0.056122 -0.049774\n",
      "4  0.035862  0.035979  0.035982  0.035976  0.033324  0.035982\n",
      "Time Spent\n",
      "                 0\n",
      "CVXPY     0.069416\n",
      "qpth      0.879650\n",
      "BPQP      0.000221\n",
      "Alt-diff  0.020316\n",
      "JAXOpt    5.197052\n",
      "exact     0.001548\n"
     ]
    },
    {
     "data": {
      "text/plain": "<AxesSubplot:>"
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "text/plain": "<Figure size 432x288 with 1 Axes>",
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXwAAAEQCAYAAACz0c/rAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAAAVrUlEQVR4nO3df7RlZX3f8fdHkGA0oEY0kQFmWENQ1LjQW6yyUn+gBgIDlkYBm7XSiEzRkES7VtLRWmNtEzArtg0RTSeREhsKocQo49Dg8gfBRIwMZqEQoBkHkCEmjCZBjCYIfvvH3lcO15nxMnffs/c5+/1a6665+zm/vmfdO5/7nOd59rNTVUiS5t9j+i5AkjQdBr4kjYSBL0kjYeBL0kgY+JI0Ega+JI3E/n0XsDdPecpTau3atX2XIUkz5cYbb/xKVR2ytH3Qgb927Vq2bdvWdxmSNFOS3LW7dod0JGkkDHxJGgkDX5JGwsCXpJEw8CVpJAYZ+Ek2JNl833339V2KJM2NQQZ+VW2pqo0HH3xw36VI0twY9Dp8SbNl7aatU329Oy84eaqvN+sG2cOXJHXPwJekkTDwJWkkDHxJGgkDX5JGwsCXpJEw8CVpJAx8SRoJA1+SRmJqZ9omeTzwXuAB4NqqunRary1JWmEPP8nFSe5NcvOS9hOT3J5ke5JNbfPpwJVVdQ5w6kpeV5L06K10SOcS4MTJhiT7ARcBJwHHAGclOQZYA9zd3u2hFb6uJOlRWlHgV9V1wN8uaT4O2F5VO6rqAeBy4DRgJ03or/h1JUmP3moE76E83JOHJugPBT4I/Ksk7wO27OnBSTYm2ZZk265du1ahPEkap6lN2lbVPwA/s4z7bQY2AywsLNRq1yVJY7EaPfx7gMMmjte0bcvmFa8kqXurEfg3AEclWZfkAOBM4KpH8wRe8UqSurfSZZmXAdcDRyfZmeTsqnoQOA+4BrgVuKKqbll5qZKklVjRGH5VnbWH9quBq/f1eZNsADasX79+X59CkrTEIJdHOqQjSd0bZOA7aStJ3Rtk4NvDl6TuDTLwJUndm9qJV+rO2k1bp/Zad15w8tReS9LqGmQP3zF8SereIAPfMXxJ6t4gA1+S1D0DX5JGYpCB7xi+JHVvkIHvGL4kdW+QgS9J6p6BL0kjYeBL0kgMMvCdtJWk7g0y8J20laTuDTLwJUndM/AlaSQMfEkaCQNfkkZikIHvKh1J6t4gA99VOpLUvUEGviSpewa+JI2EgS9JI2HgS9JIGPiSNBIGviSNhIEvSSMxyMD3xCtJ6t4gA98TrySpe4MMfElS9wx8SRoJA1+SRsLAl6SRMPAlaSQMfEkaCQNfkkbCwJekkTDwJWkkDHxJGompBX6SI5O8P8mV03pNSdLDlhX4SS5Ocm+Sm5e0n5jk9iTbk2za23NU1Y6qOnslxUqS9t3+y7zfJcB7gA8sNiTZD7gIeAWwE7ghyVXAfsD5Sx7/uqq6d8XVSpL22bICv6quS7J2SfNxwPaq2gGQ5HLgtKo6HzhlXwtKshHYCHD44Yfv69NIkpZYyRj+ocDdE8c727bdSvKDSX4LODbJW/Z0v6raXFULVbVwyCGHrKA8SdKk5Q7prFhVfRU4dzn3TbIB2LB+/frVLUqSRmQlPfx7gMMmjte0bSvmBVAkqXsrCfwbgKOSrEtyAHAmcFU3ZUmSurbcZZmXAdcDRyfZmeTsqnoQOA+4BrgVuKKqbumiKK9pK0ndW+4qnbP20H41cHWnFTXPuwXYsrCwcE7Xzy1JY+XWCpI0EoMMfId0JKl7U1uW+WisdEhn7aatHVe0d3decPJUX0+S9sUge/iSpO4NMvAd0pGk7g0y8D3xSpK6N8jAlyR1z8CXpJEYZOA7hi9J3Rtk4DuGL0ndG2TgS5K6Z+BL0kgY+JI0EoMMfCdtJal7gwx8J20lqXuDDHxJUvcMfEkaCQNfkkbCwJekkTDwJWkkBhn4LsuUpO4NMvBdlilJ3Rtk4EuSumfgS9JIGPiSNBIGviSNhIEvSSNh4EvSSOzfdwG7k2QDsGH9+vV9lyJ1au2mrVN9vTsvOHmqr6dhG2QP33X4ktS9QQa+JKl7Br4kjYSBL0kjYeBL0kgY+JI0Ega+JI2EgS9JI2HgS9JIGPiSNBIGviSNxNT20knyKuBk4CDg/VX10Wm9tiRpmT38JBcnuTfJzUvaT0xye5LtSTbt7Tmq6kNVdQ5wLnDGvpcsSdoXy+3hXwK8B/jAYkOS/YCLgFcAO4EbklwF7Aecv+Txr6uqe9vv39Y+TpI0RcsK/Kq6LsnaJc3HAduragdAksuB06rqfOCUpc+RJMAFwP+tqs+tqGpJ0qO2kknbQ4G7J453tm178nPAy4GfTHLunu6UZGOSbUm27dq1awXlSZImTW3StqouBC5cxv02A5sBFhYWarXrkqSxWEkP/x7gsInjNW3biiXZkGTzfffd18XTSZJYWeDfAByVZF2SA4Azgau6KMorXklS95a7LPMy4Hrg6CQ7k5xdVQ8C5wHXALcCV1TVLatXqiRpJZa7SuesPbRfDVzdaUV4EXNJWg2D3FrBIR1J6t4gA99JW0nq3iAD3x6+JHVvkIEvSeqegS9JIzHIwHcMX5K6N8jAdwxfkro3yMCXJHXPwJekkRhk4DuGL0ndG2TgO4YvSd0bZOBLkrpn4EvSSBj4kjQSgwx8J20lqXuDDHwnbSWpe4MMfElS9wx8SRoJA1+SRsLAl6SRGGTgu0pHkro3yMB3lY4kdW+QgS9J6p6BL0kjYeBL0kgY+JI0Ega+JI2EgS9JI2HgS9JIDDLwPfFKkro3yMD3xCtJ6t4gA1+S1D0DX5JGwsCXpJEw8CVpJPbvuwBJmhVrN22d6uvdecHJnT6fPXxJGgkDX5JGwsCXpJEw8CVpJAx8SRqJqQV+kmcm+a0kVyZ5w7ReV5LUWFbgJ7k4yb1Jbl7SfmKS25NsT7Jpb89RVbdW1bnAa4Dj971kSdK+WG4P/xLgxMmGJPsBFwEnAccAZyU5JslzknxkyddT28ecCmwFru7sHUiSlmVZJ15V1XVJ1i5pPg7YXlU7AJJcDpxWVecDp+zhea4CrkqyFfjfu7tPko3ARoDDDz98OeVJkpZhJWfaHgrcPXG8E3jBnu6c5CXA6cD3sZceflVtBjYDLCws1ArqkyRNmNrWClV1LXDtcu6bZAOwYf369atZkiSNykpW6dwDHDZxvKZtWzEvgCJJ3VtJ4N8AHJVkXZIDgDOBq7opS5LUteUuy7wMuB44OsnOJGdX1YPAecA1wK3AFVV1SxdFeU1bSereclfpnLWH9qtZhSWWVbUF2LKwsHBO188tSWPl1gqSNBKDDHyHdCSpe4MMfFfpSFL3Bhn4kqTuDTLwHdKRpO4NMvAd0pGk7g0y8CVJ3TPwJWkkBhn4juFLUvcGGfiO4UtS9wYZ+JKk7hn4kjQSBr4kjcQgA99JW0nq3iAD30lbSereIANfktQ9A1+SRsLAl6SRMPAlaSQMfEkaiUEGvssyJal7gwx8l2VKUvcGGfiSpO4Z+JI0Ega+JI2EgS9JI7F/3wVIk9Zu2jrV17vzgpOn+npSn+zhS9JIDDLwXYcvSd0bZOC7Dl+SujfIwJckdc/Al6SRMPAlaSQMfEkaCQNfkkbCwJekkUhV9V3DHiXZBdw1xZd8CvCVKb7etM3z+5vn9wa+v1k37fd3RFUdsrRx0IE/bUm2VdVC33Wslnl+f/P83sD3N+uG8v4c0pGkkTDwJWkkDPxH2tx3Aatsnt/fPL838P3NukG8P8fwJWkk7OFL0kgY+JI0Ega+pKlK8gvLaVP3Rhn4Sdb2XYNWJskhSRaSPLHvWrqU5OPtv+/qu5ZV9NO7afs30y5iNSVZt5y2aRvrNW0/luR3gF+vqgf7LkaPTpLXA78KfBFYl2RjVV3Vc1ld+eEkLwJOTXI5kMkbq+pz/ZS1cknOAl5L8zOb/Hn9APC3/VS1av4AeN6StiuB5/dQy3eMNfCPBd4J3JjkvKr6VN8FrYYkpwPvAp5KExwBqqoO6rWwlXsT8Kyq2pXkSOBSYF4C/+3AfwTWAO/mkYFfwMv6KKojnwa+TLPNwLsn2u8HPt9LRR1L8gzgWcDB7f+/RQcBB/ZT1cNGGfhVdT/w5iTPBz6eZCfwbR4OxB/ttcDu/Bqwoapu7buQjj1QVbsAqmpHku/ru6AOfbmqTkry9qp6Z9/FdKmq7qLZG+uFSX4IOI7mj9jtc/RJ+2jgFOCJwIaJ9vuBc/ooaNJo1+EneRnwG8A1wEU0gQ985xdz5iX506o6vu86upbkXuDyiaYzJ4+r6uenXlRHktxYVc9P8rmqWjokMBeSnA38MvAJmk7Wi4F3VtXFvRbWoSQvrKrr+65jqVEGfjs2ugZ4Q1V9oe96ujbxUfLFwA8BHwL+afH2qvpgD2V1JsnuJv2+o6p+d1q1dC3JZ2iGN04Dfn/p7bP8x2xRktuBF1XVV9vjHwQ+XVVH91tZd5L8LvALVfX37fGTgHdX1ev6rGuUQzrAx4APA0ckeeLiD2WOTH6U/AbwyonjAmY68CcDPclBbdvX+quoU6cALwd+HLix51pWy1dphjgW3d+2zZMfncyVqvq7JMf2WA8w3sAHuBnYwfyt8qCqfgYgyfFV9aeTtyWZiyGedt32L9FOhCX5KvD2qro8yWFVdXevBe6jqvoKcHmSW6vqpr7rWSXbgT9L8mGaDshpwOeT/DuAqvqvfRbXkcckeVJV/R1AkiczgLztvYCevAl49pyu8pj0m3z30rDdtc2UJO+gmfD7sara0bYdCfxGkiNoJsfW91fhvkvyS1X1a8Drk3zXeOs8DOnQLKf94sTxh9t/f6CHWlbLu4Hrk/wfmnmKnwR+pd+Sxhv487zKgyQvBF4EHLLYa2odBOzXT1Wd+tfAc6rqHxcb2p/ja4BdNGu9Z9XiiqptvVaxiqrqPyV5Qvv91/uuZzVU1QeS3Ai8tG06var+os+aYLyBvybJhXs6noNe1AHAE2h+vpO9pq/R9DRm3UOTYb+oqr6Z5J5ZHp6rqi3tvzM78bw3Sd4IbAIe3x5/HXhXVb2318JWQVXd0l6mdXHY8fCq+lKfNY018H9xyfFcTY5V1R8Df5zkkqq6q53YrPb8g3lwT5ITqurjk43tUtt7eqqpE0m20Ixr71ZVnTrFcjqV5G00nzxfspuhuCdX1X/ptcAOJTmVZljn6cC9wBE0n96e1WtdI12W+Ziq+vYebpubVTtJFoD/ycO9/PuA11XVTP+BS/IsmnHfP+HhP9YLwPHAqUP46Lyvkry4/fZ0miW1v9cenwX8TVW9uZfCOtAux3zu0k9nSR4H3FRVP9JPZd1LchPNWdEfq6pjk7wU+KmqOrvPuka5eRqwLckLlja2e7TM7F4lu3Ex8MaqWltVa4GfpfkDMNOq6hbg2cB1wNr26zqaifiZDXtoPp21n9COr6ozqmpL+/Va4Mf6rm+Fak9DcUyc+DgnvtWeZ/CYtoP5SZpOSa/GOqTz88DmJJ8F/j3Nx633AjuBf9FnYR17aHKfoKr6kyTzcgr7icCTgY9W1TV9F7MKHp/kyImhj3W0494zbG9DcV/uqabV8vftxPR1wKXt2eH/0HNN4xzSAUiyP/AOml7v14Gzq+qjvRbVsST/HXgccBnNuPAZwD/SDhPM6s6LSd5LMxb6aeAEYEtV/ed+q+pWkhNproO6g2ZZ3xHAxln+Hf0eQ3GntZ/c5kKSx9P8XwvNqrKDgUsXzy7ura4RB/6ZNOtifx94Bc3p7L9YVXOzTWuST7bfLv6QH7HzYlXN5M6LSW6mGQt+KMn3A5+qql63nV0N7XLhZ7SHt1XVP+3t/rMgyYE0y2YXJy//guY8mAfnaAM1khyzdHgxyUuq6tqeSgJGOqST5GM0f31fXlV3tKsHfha4Icm7qmoQV5jvwEdown4x6Jd+P6seqKqHAKrqG0nyvR4wi9qAvynJ5qra2Hc9K9WuQDpv6SZpSU6g2cjw2b0UtjquSPK/aHasPbD9dwF4YZ9FjXXS9qKqOqWq7gCoqm9X1W/SfLR88d4fOlOeD7wB+GGa5WHn0lwL4Ant16x6RpLPt19fmDj+QpK52Fd9id4n+zpyOfDJJG9N8tgkT09yBc3FbPa6Id4MegFwGM2w4w3AX9HkS69G2cMH1idZU1U7Jxur6q9pxtvmxRrgeYvr79stCbZW1U/1WtXKPbPvAqbs3r4L6EJVXZrkIzS93VuBx9IMq/52zd/Y8reAb9LMoR0I3LGnpeDTNNYe/tNp9rn4VJI3Jjmk74JWydOAByaOH2jbZlpV3bX0i2YFxJdqfq5l8OrF76vqxKVtM+wYmn2QPkuzZffTmM+O5w00gb9As5z2rHZfnV6NMvDbk1cOB94GPIdmp74/SvLTSeZpA6cPAJ9N8o62d/9nwCW9VtSBJP88ybVJPpjk2HYS92bgb9rVLfPgLctsmxlJ3k9zsaE3tucVHEuzeuWmJK/c64NnzznAXwJvraovAz8H9L776WhX6UxKsh/NHuQXAEdX1ff3XFJnkjyPh0/Yua6q/rzPerqQZBvwVpqw2AycVFWfSXM90cuqqvd9x/dVkpOAnwBewyMvgHIQcExVHddLYR1I8mbgwsUJ94n259DMq83NOTBJ3kdzMtnLquqZaS6A8tGq+md91jWPH6UelfaX7UyaNepfYcZ7UUu1a+1ncr39Xuy/uB49yTur6jMAVXXbHCzY+SuaNeqn8sg9nu4HZnZbBYCq+m97uOkvgfdMs5YpeEFVPS/Jn8N3LoDy2L6LGmXgJzmKJuTPBB6iWT3wysWzGjV4k5Nf31xy20x/ZK3moic3Jfm9eVqXvlT7qfrHafYIeiXwKeCKXovq1rfa91gA7Txh77+bowx84DbgWuCMqrp5sTHN1aD+uqq+uKcHahCem+RrNOcUPK79nvb4wP7KWrl2meliSCy9uarquVMvqkPt5nCvpRm2+izNUsV1VfWNXgvr3oXAHwJPTfIrNNuSv63fkkY6hp9kK7CpllzAvB3e+dWq2rD7R0qrK80Vu76rmWZN91uq6iemXFJnkuwEvgS8D/hQVd2f5I6qWtdzaauinVM6gebn9/GquvV7PGTVjbWH/9SlYQ9QVV9IsraHeiSgWXK6+H2ai16/Fng1cAfwB33V1ZErgVfRzJc9lIevaTuXquo2mtGEwRjlskzgSXu57XFTq0JaIsmPJPnlJLfRXH/4SzSfxF9aVTM9sVlVbwLW0VwY5CXA7TRDHme0O0tqlY11SOcy4BNV9dtL2l8PvKKqzuinMo1dkm/TTGCeXVXb27YdVXVkv5V1r1218p2J26qa1xMgB2Osgf80mgmVB3jkNq0HAP+y3WJBmrokr6JZPXY88Ec0K8h+Zx7GuZPcz3cP4SzOTH8L+H/Af1i6X766M8rAX9Redmxxh75bquoTfdYjLWr3Uz+Npvf7Mpqzpv9wlvfD35t2CeOzafaMn6ddMwdl1IEvzYL2LM1X0ywjPqHvelZTkn9bVf+j7zrmlYEvSSMx1lU6kjQ6Br4kjYSBL0kjYeBL0kgY+JI0Ev8fS3o2YqJAgckAAAAASUVORK5CYII=\n"
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "print('Forward pass')\n",
    "print(pd.DataFrame({'CVXPY':x1.value, 'qpth': qpth_x_value, 'BPQP': x_value,'Alt-diff': xk,'JAXOpt':sol.primal}).head(5))\n",
    "print('Backward pass')\n",
    "print(pd.DataFrame({'CVXPY':qq.gradient, 'qpth': qpth_x_grad, 'BPQP': x_grad, 'Alt-diff':dq,'JAXOpt':jax_gradient, 'exact': exact_bb}).head(5))\n",
    "print('Time Spent')\n",
    "time_table = pd.Series({'CVXPY':total_time_cvxpy, 'qpth': total_time_qpth, 'BPQP': time_spent+time_spent_backward, 'Alt-diff':t6,'JAXOpt':t7, 'exact': t5})\n",
    "print(time_table.to_frame())\n",
    "np.log(1+time_table.to_frame()).plot.bar(logy=True,legend=False)"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "OSQP vs CVXPY forward dif 0.9999999999999981\n",
      "OSQP vs qpth forward dif 1.000000037440453\n",
      "OSQP vs Exact backward dif 0.9999999999997422\n",
      "CVXPY vs Exact backward dif 0.9999986934226582\n",
      "qpth vs Exact backward dif 1.0000000177948525\n",
      "qpth vs CVXPY backward dif 0.9999987084601358\n",
      "alt-diff vs Exact backward backward dif 0.9999999115412972\n",
      "JAXOpt vs Exact backward backward dif 0.9960119620777163\n"
     ]
    }
   ],
   "source": [
    "print('OSQP vs CVXPY forward dif',cal_cos_accuracy(x_value , x1.value))\n",
    "print('OSQP vs qpth forward dif',cal_cos_accuracy(x_value , qpth_x_value))\n",
    "print('OSQP vs Exact backward dif',cal_cos_accuracy(exact_bb , x_grad))\n",
    "print('CVXPY vs Exact backward dif',cal_cos_accuracy(exact_bb, qq.gradient))\n",
    "print('qpth vs Exact backward dif',cal_cos_accuracy(exact_bb, qpth_x_grad))\n",
    "print('qpth vs CVXPY backward dif',cal_cos_accuracy(qq.gradient, qpth_x_grad))\n",
    "print('alt-diff vs Exact backward backward dif',cal_cos_accuracy(exact_bb,dq))\n",
    "print('JAXOpt vs Exact backward backward dif',cal_cos_accuracy(exact_bb,jax_gradient))"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "# 8. Comparison with state-of-art differentiable QP optimizer"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%% md\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "outputs": [],
   "source": [
    "def QP_instances(ndim,neq,nineq):\n",
    "\n",
    "    P = np.random.randn(ndim,ndim)\n",
    "    P = P.T@P+(0.001*np.eye(ndim,ndim))\n",
    "    Pm = P.copy()\n",
    "    P = sparse.csc_matrix(P)\n",
    "    q = np.random.randn(ndim)\n",
    "    A = np.random.randn(neq,ndim)\n",
    "    G = np.random.randn(nineq,ndim)\n",
    "    b = np.random.randn(neq)\n",
    "    h = np.random.randn(neq)\n",
    "    osA = np.vstack([G,A])\n",
    "    osA = sparse.csc_matrix(osA)\n",
    "    l = np.hstack([-np.inf*np.ones(len(h)),b])\n",
    "    u = np.hstack([h,b])\n",
    "    return P,q,G,b,A,h,osA,l,u,Pm\n",
    "\n",
    "def QP_OSQP_backward(y_value,P,G,A):\n",
    "    nineq,ndim = G.shape\n",
    "    neq = A.shape[0]\n",
    "    lambs = y_value[:nineq] # active set\n",
    "    active_set = np.argwhere(lambs>1e-8)\n",
    "    bG = G[active_set,:].squeeze()\n",
    "    bb = np.zeros(neq)\n",
    "    bh = np.zeros(len(active_set))\n",
    "    bq = np.ones(ndim)\n",
    "    osnewA = np.vstack([bG,A])\n",
    "    osnewA = sparse.csc_matrix(osnewA)\n",
    "    l_new = np.hstack([bh,bb])\n",
    "    u_new = np.hstack([bh,bb])\n",
    "    x_grad, y_grad, time_spent_backward = osqp_interface(P,bq,osnewA,l_new,u_new)\n",
    "    return x_grad, y_grad, time_spent_backward\n",
    "\n",
    "def QP_cvxpy_backward(P,G,A,h,b,q):\n",
    "    nineq,ndim = G.shape\n",
    "    neq = nineq\n",
    "    qq = cp.Parameter(ndim)\n",
    "    qq.value = q\n",
    "    x1 = cp.Variable(ndim)\n",
    "    prob = cp.Problem(cp.Minimize((1 / 2) * cp.quad_form(x1, P) + qq.T @ x1),\n",
    "                                  [G @ x1 <= h,\n",
    "                                   A @ x1 == b])\n",
    "    t3 = time.time()\n",
    "    try:\n",
    "        prob.solve(requires_grad=True, solver='SCS')\n",
    "        time_spent_forward = time.time() - t3\n",
    "        t4 = time.time()\n",
    "        prob.backward()\n",
    "        time_spent_backward = time.time() - t4\n",
    "        return x1.value,prob.value, qq.gradient,time_spent_forward,time_spent_backward\n",
    "    except:\n",
    "        return np.nan*np.zeros(ndim),np.nan,np.nan*np.zeros(ndim),np.nan,np.nan\n",
    "\n",
    "def QP_qpth_evaluate(Pm,G,A,h,b,q):\n",
    "    nineq,ndim = G.shape\n",
    "    neq = nineq\n",
    "    qpf = QPFunction(verbose=0,maxIter=500)\n",
    "    tP, tq, tG, th, tA, tb = [torch.Tensor(x).unsqueeze(0).cuda() for x in [Pm, q, G, h, A, b]]\n",
    "    tq.requires_grad = True\n",
    "    try:\n",
    "        t6 = time.time()\n",
    "        qpth_x_value = qpf(tP, tq, tG, th, tA, tb)\n",
    "        t7 = time.time()\n",
    "        qpth_x_value.backward(torch.ones(1, ndim).cuda())\n",
    "        qpth_x_grad = tq.grad.squeeze().cpu().numpy()\n",
    "        qpth_x_value = qpth_x_value.squeeze().detach().cpu().numpy()\n",
    "        return qpth_x_value,qpth_x_grad,time.time() - t6,time.time() - t7\n",
    "    except:\n",
    "        return np.nan*np.zeros(ndim),np.nan*np.zeros(ndim),np.nan,np.nan\n",
    "\n",
    "def QP_cal_exact_backward(Pm,G,A,y_value,x_value,h):\n",
    "    nineq,ndim = G.shape\n",
    "    neq = A.shape[0]\n",
    "    lambs = y_value[:nineq] # active set\n",
    "    KKT_L1 = np.hstack([Pm,G.T,A.T])\n",
    "    KKT_L2 = np.hstack([np.diag(lambs)@G, np.diag(G@x_value-h),np.zeros((nineq,neq))])\n",
    "    KKT_L3 = np.hstack([A, np.zeros((neq,neq)),np.zeros((neq,nineq))])\n",
    "    KKT = np.vstack([KKT_L1,KKT_L2,KKT_L3])\n",
    "    t5 = time.time()\n",
    "    exact_bb =-(np.linalg.inv(KKT)@np.hstack([np.ones(ndim),np.zeros(nineq),np.zeros(neq)]))[:ndim]\n",
    "    return exact_bb,time.time()-t5\n",
    "\n",
    "\n",
    "\n",
    "def dict_report(stats, key, value):\n",
    "    if key in stats.keys():\n",
    "        stats[key] = np.append(stats[key], value)\n",
    "    else:\n",
    "        stats[key] = value"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|                                                                                                                    | 0/200 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002770423889160156\n",
      "The inverse time is 0.0012254714965820312\n",
      "The inverse time is 0.17031264305114746\n",
      "The inverse time is 2.291799545288086\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|▌                                                                                                           | 1/200 [00:17<59:24, 17.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002353191375732422\n",
      "The inverse time is 0.08327245712280273\n",
      "The inverse time is 0.2660210132598877\n",
      "The inverse time is 2.3468446731567383\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  1%|█                                                                                                           | 2/200 [00:30<49:44, 15.07s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.006010770797729492\n",
      "The inverse time is 0.09507060050964355\n",
      "The inverse time is 0.3768641948699951\n",
      "The inverse time is 2.2898733615875244\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|█▌                                                                                                          | 3/200 [00:43<45:09, 13.76s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002162456512451172\n",
      "The inverse time is 0.16909384727478027\n",
      "The inverse time is 0.3138084411621094\n",
      "The inverse time is 2.431558847427368\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|██▏                                                                                                         | 4/200 [00:56<44:27, 13.61s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002429485321044922\n",
      "The inverse time is 0.12879014015197754\n",
      "The inverse time is 0.316356897354126\n",
      "The inverse time is 2.2481117248535156\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|██▋                                                                                                         | 5/200 [01:09<43:33, 13.40s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002040863037109375\n",
      "The inverse time is 0.045361995697021484\n",
      "The inverse time is 0.33800745010375977\n",
      "The inverse time is 2.210009813308716\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  3%|███▏                                                                                                        | 6/200 [01:24<44:54, 13.89s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001938343048095703\n",
      "The inverse time is 0.02817821502685547\n",
      "The inverse time is 0.27886128425598145\n",
      "The inverse time is 2.2853446006774902\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  4%|███▊                                                                                                        | 7/200 [01:36<42:54, 13.34s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002372264862060547\n",
      "The inverse time is 0.11029672622680664\n",
      "The inverse time is 0.27012085914611816\n",
      "The inverse time is 2.386561870574951\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  4%|████▎                                                                                                       | 8/200 [01:49<42:28, 13.27s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.008338212966918945\n",
      "The inverse time is 0.12830305099487305\n",
      "The inverse time is 0.35295629501342773\n",
      "The inverse time is 2.296135902404785\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  4%|████▊                                                                                                       | 9/200 [02:02<42:07, 13.23s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019311904907226562\n",
      "The inverse time is 0.018635988235473633\n",
      "The inverse time is 0.23768901824951172\n",
      "The inverse time is 2.3826208114624023\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  5%|█████▎                                                                                                     | 10/200 [02:14<39:49, 12.58s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00833582878112793\n",
      "The inverse time is 0.11352157592773438\n",
      "The inverse time is 0.34798693656921387\n",
      "The inverse time is 2.4158732891082764\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  6%|█████▉                                                                                                     | 11/200 [02:24<38:00, 12.07s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.000209808349609375\n",
      "The inverse time is 0.1584641933441162\n",
      "The inverse time is 0.28495216369628906\n",
      "The inverse time is 2.4834237098693848\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  6%|██████▍                                                                                                    | 12/200 [02:36<37:45, 12.05s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0019409656524658203\n",
      "The inverse time is 0.020132780075073242\n",
      "The inverse time is 0.2621583938598633\n",
      "The inverse time is 2.4187231063842773\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  6%|██████▉                                                                                                    | 13/200 [02:50<39:25, 12.65s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.000217437744140625\n",
      "The inverse time is 0.11884760856628418\n",
      "The inverse time is 0.2959010601043701\n",
      "The inverse time is 2.458833694458008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  7%|███████▍                                                                                                   | 14/200 [03:03<38:56, 12.56s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00017189979553222656\n",
      "The inverse time is 0.11475133895874023\n",
      "The inverse time is 0.3143954277038574\n",
      "The inverse time is 2.232158660888672\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  8%|████████                                                                                                   | 15/200 [03:15<38:23, 12.45s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002830028533935547\n",
      "The inverse time is 0.025731325149536133\n",
      "The inverse time is 0.2673344612121582\n",
      "The inverse time is 2.3233225345611572\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  8%|████████▌                                                                                                  | 16/200 [03:24<35:06, 11.45s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00621485710144043\n",
      "The inverse time is 0.10976505279541016\n",
      "The inverse time is 0.2868490219116211\n",
      "The inverse time is 2.3199410438537598\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  8%|█████████                                                                                                  | 17/200 [03:35<33:59, 11.14s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.017891645431518555\n",
      "The inverse time is 0.12255716323852539\n",
      "The inverse time is 0.2715640068054199\n",
      "The inverse time is 2.4623050689697266\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  9%|█████████▋                                                                                                 | 18/200 [03:48<36:17, 11.96s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0018205642700195312\n",
      "The inverse time is 0.0014722347259521484\n",
      "The inverse time is 0.326671838760376\n",
      "The inverse time is 2.423311471939087\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|██████████▏                                                                                                | 19/200 [04:05<40:00, 13.26s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002148151397705078\n",
      "The inverse time is 0.09809231758117676\n",
      "The inverse time is 0.280287504196167\n",
      "The inverse time is 2.217639923095703\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|██████████▋                                                                                                | 20/200 [04:16<37:44, 12.58s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.002712726593017578\n",
      "The inverse time is 0.0951383113861084\n",
      "The inverse time is 0.4079594612121582\n",
      "The inverse time is 2.3629729747772217\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|███████████▏                                                                                               | 21/200 [04:26<35:20, 11.85s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.003805398941040039\n",
      "The inverse time is 0.06394147872924805\n",
      "The inverse time is 0.27323031425476074\n",
      "The inverse time is 2.377753734588623\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 11%|███████████▊                                                                                               | 22/200 [04:38<35:19, 11.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020885467529296875\n",
      "The inverse time is 0.029705047607421875\n",
      "The inverse time is 0.26555728912353516\n",
      "The inverse time is 2.348961114883423\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 12%|████████████▎                                                                                              | 23/200 [04:50<34:55, 11.84s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022459030151367188\n",
      "The inverse time is 0.09871864318847656\n",
      "The inverse time is 0.28361058235168457\n",
      "The inverse time is 2.467494249343872\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 12%|████████████▊                                                                                              | 24/200 [05:02<35:19, 12.04s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002052783966064453\n",
      "The inverse time is 0.03878617286682129\n",
      "The inverse time is 0.2989029884338379\n",
      "The inverse time is 2.3376967906951904\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 12%|█████████████▍                                                                                             | 25/200 [05:14<35:15, 12.09s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002002716064453125\n",
      "The inverse time is 0.11267781257629395\n",
      "The inverse time is 0.3320956230163574\n",
      "The inverse time is 2.195093870162964\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 13%|█████████████▉                                                                                             | 26/200 [05:28<36:40, 12.65s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001926422119140625\n",
      "The inverse time is 0.10566163063049316\n",
      "The inverse time is 0.2500028610229492\n",
      "The inverse time is 2.3615450859069824\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 14%|██████████████▍                                                                                            | 27/200 [05:43<37:54, 13.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020766258239746094\n",
      "The inverse time is 0.11620330810546875\n",
      "The inverse time is 0.33313488960266113\n",
      "The inverse time is 2.367056131362915\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 14%|██████████████▉                                                                                            | 28/200 [05:55<37:10, 12.97s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018596649169921875\n",
      "The inverse time is 0.11323404312133789\n",
      "The inverse time is 0.19103622436523438\n",
      "The inverse time is 2.4051122665405273\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 14%|███████████████▌                                                                                           | 29/200 [06:06<34:51, 12.23s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002415180206298828\n",
      "The inverse time is 0.0004642009735107422\n",
      "The inverse time is 0.32459449768066406\n",
      "The inverse time is 2.4179635047912598\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 15%|████████████████                                                                                           | 30/200 [06:18<35:04, 12.38s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0010802745819091797\n",
      "The inverse time is 0.02063918113708496\n",
      "The inverse time is 0.25201988220214844\n",
      "The inverse time is 2.315375566482544\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 16%|████████████████▌                                                                                          | 31/200 [06:33<36:30, 12.96s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001842975616455078\n",
      "The inverse time is 0.03522348403930664\n",
      "The inverse time is 0.2707386016845703\n",
      "The inverse time is 2.294188976287842\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 16%|█████████████████                                                                                          | 32/200 [06:45<35:44, 12.77s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n",
      "The inverse time is 0.005570411682128906\n",
      "The inverse time is 0.1074974536895752\n",
      "The inverse time is 0.29340457916259766\n",
      "The inverse time is 2.369572162628174\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 16%|█████████████████▋                                                                                         | 33/200 [06:56<34:26, 12.37s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002796649932861328\n",
      "The inverse time is 0.07686376571655273\n",
      "The inverse time is 0.32391810417175293\n",
      "The inverse time is 2.355823516845703\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 17%|██████████████████▏                                                                                        | 34/200 [07:07<32:56, 11.90s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002455711364746094\n",
      "The inverse time is 0.04393410682678223\n",
      "The inverse time is 0.30493617057800293\n",
      "The inverse time is 2.4875853061676025\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 18%|██████████████████▋                                                                                        | 35/200 [07:20<33:44, 12.27s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.007419109344482422\n",
      "The inverse time is 0.14706015586853027\n",
      "The inverse time is 0.26562047004699707\n",
      "The inverse time is 2.3674263954162598\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 18%|███████████████████▎                                                                                       | 36/200 [07:33<34:00, 12.44s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001900196075439453\n",
      "The inverse time is 0.0859532356262207\n",
      "The inverse time is 0.2892446517944336\n",
      "The inverse time is 2.4354140758514404\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 18%|███████████████████▊                                                                                       | 37/200 [07:44<32:21, 11.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00023126602172851562\n",
      "The inverse time is 0.025387287139892578\n",
      "The inverse time is 0.2871277332305908\n",
      "The inverse time is 2.303942918777466\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 19%|████████████████████▎                                                                                      | 38/200 [07:57<32:46, 12.14s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002048015594482422\n",
      "The inverse time is 0.01764082908630371\n",
      "The inverse time is 0.19019269943237305\n",
      "The inverse time is 2.2249743938446045\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|████████████████████▊                                                                                      | 39/200 [08:11<34:10, 12.74s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002410411834716797\n",
      "The inverse time is 0.02480340003967285\n",
      "The inverse time is 0.18054842948913574\n",
      "The inverse time is 2.300367593765259\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|█████████████████████▍                                                                                     | 40/200 [08:26<36:08, 13.55s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021338462829589844\n",
      "The inverse time is 0.15099191665649414\n",
      "The inverse time is 0.3396420478820801\n",
      "The inverse time is 2.28690767288208\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|█████████████████████▉                                                                                     | 41/200 [08:36<32:44, 12.36s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019168853759765625\n",
      "The inverse time is 0.1099393367767334\n",
      "The inverse time is 0.2636892795562744\n",
      "The inverse time is 2.3708274364471436\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 21%|██████████████████████▍                                                                                    | 42/200 [08:46<30:55, 11.74s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002040863037109375\n",
      "The inverse time is 0.08832192420959473\n",
      "The inverse time is 0.2839667797088623\n",
      "The inverse time is 2.280022621154785\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 22%|███████████████████████                                                                                    | 43/200 [08:55<28:39, 10.95s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018644332885742188\n",
      "The inverse time is 0.045668840408325195\n",
      "The inverse time is 0.2300095558166504\n",
      "The inverse time is 2.240713596343994\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 22%|███████████████████████▌                                                                                   | 44/200 [09:05<27:51, 10.71s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001995563507080078\n",
      "The inverse time is 0.011722087860107422\n",
      "The inverse time is 0.2607147693634033\n",
      "The inverse time is 2.4046850204467773\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 22%|████████████████████████                                                                                   | 45/200 [09:14<26:01, 10.07s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00023698806762695312\n",
      "The inverse time is 0.06329894065856934\n",
      "The inverse time is 0.31969499588012695\n",
      "The inverse time is 2.397941827774048\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 23%|████████████████████████▌                                                                                  | 46/200 [09:22<24:39,  9.61s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00014019012451171875\n",
      "The inverse time is 0.023894786834716797\n",
      "The inverse time is 0.18424177169799805\n",
      "The inverse time is 2.316850423812866\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 24%|█████████████████████████▏                                                                                 | 47/200 [09:32<24:18,  9.53s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019788742065429688\n",
      "The inverse time is 0.07353591918945312\n",
      "The inverse time is 0.2893824577331543\n",
      "The inverse time is 2.388991117477417\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 24%|█████████████████████████▋                                                                                 | 48/200 [09:41<23:36,  9.32s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021648406982421875\n",
      "The inverse time is 0.020178794860839844\n",
      "The inverse time is 0.28978705406188965\n",
      "The inverse time is 2.238588571548462\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 24%|██████████████████████████▏                                                                                | 49/200 [09:49<22:56,  9.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001811981201171875\n",
      "The inverse time is 0.02448558807373047\n",
      "The inverse time is 0.362823486328125\n",
      "The inverse time is 2.2340288162231445\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 25%|██████████████████████████▊                                                                                | 50/200 [09:58<22:34,  9.03s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002071857452392578\n",
      "The inverse time is 0.062073707580566406\n",
      "The inverse time is 0.3286566734313965\n",
      "The inverse time is 2.1178152561187744\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 26%|███████████████████████████▎                                                                               | 51/200 [10:08<22:54,  9.22s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021839141845703125\n",
      "The inverse time is 0.08647298812866211\n",
      "The inverse time is 0.245222806930542\n",
      "The inverse time is 2.192593574523926\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 26%|███████████████████████████▊                                                                               | 52/200 [10:17<22:37,  9.17s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00014162063598632812\n",
      "The inverse time is 0.019199848175048828\n",
      "The inverse time is 0.28481364250183105\n",
      "The inverse time is 2.0978500843048096\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 26%|████████████████████████████▎                                                                              | 53/200 [10:25<22:01,  8.99s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001900196075439453\n",
      "The inverse time is 0.024092435836791992\n",
      "The inverse time is 0.3083953857421875\n",
      "The inverse time is 2.362938642501831\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 27%|████████████████████████████▉                                                                              | 54/200 [10:36<23:07,  9.51s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021958351135253906\n",
      "The inverse time is 0.03347587585449219\n",
      "The inverse time is 0.3366692066192627\n",
      "The inverse time is 2.1778626441955566\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 28%|█████████████████████████████▍                                                                             | 55/200 [10:45<22:13,  9.20s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021147727966308594\n",
      "The inverse time is 0.09221005439758301\n",
      "The inverse time is 0.25466346740722656\n",
      "The inverse time is 2.2672276496887207\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 28%|█████████████████████████████▉                                                                             | 56/200 [10:53<21:37,  9.01s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00026869773864746094\n",
      "The inverse time is 0.11759686470031738\n",
      "The inverse time is 0.30420351028442383\n",
      "The inverse time is 2.329195261001587\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 28%|██████████████████████████████▍                                                                            | 57/200 [11:02<21:28,  9.01s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021910667419433594\n",
      "The inverse time is 0.0125732421875\n",
      "The inverse time is 0.20470786094665527\n",
      "The inverse time is 2.1833207607269287\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 29%|███████████████████████████████                                                                            | 58/200 [11:11<21:08,  8.93s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020432472229003906\n",
      "The inverse time is 0.020272254943847656\n",
      "The inverse time is 0.32003188133239746\n",
      "The inverse time is 2.31484317779541\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|███████████████████████████████▌                                                                           | 59/200 [11:20<20:59,  8.93s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019669532775878906\n",
      "The inverse time is 0.024874448776245117\n",
      "The inverse time is 0.2411031723022461\n",
      "The inverse time is 2.3143653869628906\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████████████████████████                                                                           | 60/200 [11:29<20:56,  8.98s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00017905235290527344\n",
      "The inverse time is 0.13940906524658203\n",
      "The inverse time is 0.3155395984649658\n",
      "The inverse time is 2.2654294967651367\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████████████████████████▋                                                                          | 61/200 [11:38<20:45,  8.96s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0014750957489013672\n",
      "The inverse time is 0.04921102523803711\n",
      "The inverse time is 0.26784229278564453\n",
      "The inverse time is 2.3080947399139404\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 31%|█████████████████████████████████▏                                                                         | 62/200 [11:47<20:59,  9.13s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001919269561767578\n",
      "The inverse time is 0.01639246940612793\n",
      "The inverse time is 0.37572717666625977\n",
      "The inverse time is 2.3130791187286377\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 32%|█████████████████████████████████▋                                                                         | 63/200 [11:57<21:13,  9.29s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002014636993408203\n",
      "The inverse time is 0.11998701095581055\n",
      "The inverse time is 0.2763841152191162\n",
      "The inverse time is 2.392119884490967\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 32%|██████████████████████████████████▏                                                                        | 64/200 [12:06<21:05,  9.30s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022554397583007812\n",
      "The inverse time is 0.12966489791870117\n",
      "The inverse time is 0.24822998046875\n",
      "The inverse time is 2.2127573490142822\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 32%|██████████████████████████████████▊                                                                        | 65/200 [12:15<20:34,  9.14s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002200603485107422\n",
      "The inverse time is 0.028211116790771484\n",
      "The inverse time is 0.29106640815734863\n",
      "The inverse time is 2.245323657989502\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 33%|███████████████████████████████████▎                                                                       | 66/200 [12:24<20:23,  9.13s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001971721649169922\n",
      "The inverse time is 0.01254892349243164\n",
      "The inverse time is 0.31763792037963867\n",
      "The inverse time is 2.3639729022979736\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 34%|███████████████████████████████████▊                                                                       | 67/200 [12:33<20:01,  9.04s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018024444580078125\n",
      "The inverse time is 0.024976491928100586\n",
      "The inverse time is 0.21212434768676758\n",
      "The inverse time is 2.607067823410034\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 34%|████████████████████████████████████▍                                                                      | 68/200 [12:43<20:17,  9.22s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019478797912597656\n",
      "The inverse time is 0.02086615562438965\n",
      "The inverse time is 0.2694413661956787\n",
      "The inverse time is 2.5133605003356934\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 34%|████████████████████████████████████▉                                                                      | 69/200 [12:53<20:38,  9.46s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002014636993408203\n",
      "The inverse time is 0.15375685691833496\n",
      "The inverse time is 0.23508548736572266\n",
      "The inverse time is 2.318582534790039\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 35%|█████████████████████████████████████▍                                                                     | 70/200 [13:02<20:16,  9.36s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001761913299560547\n",
      "The inverse time is 0.024478673934936523\n",
      "The inverse time is 0.2752823829650879\n",
      "The inverse time is 2.3099706172943115\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|█████████████████████████████████████▉                                                                     | 71/200 [13:11<20:19,  9.45s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n",
      "The inverse time is 0.00020623207092285156\n",
      "The inverse time is 0.02039504051208496\n",
      "The inverse time is 0.33316469192504883\n",
      "The inverse time is 2.282766580581665\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|██████████████████████████████████████▌                                                                    | 72/200 [13:20<19:52,  9.31s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018286705017089844\n",
      "The inverse time is 0.0005154609680175781\n",
      "The inverse time is 0.34938812255859375\n",
      "The inverse time is 2.355192184448242\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|███████████████████████████████████████                                                                    | 73/200 [13:29<19:26,  9.18s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018644332885742188\n",
      "The inverse time is 0.05823826789855957\n",
      "The inverse time is 0.29818105697631836\n",
      "The inverse time is 2.247356414794922\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 37%|███████████████████████████████████████▌                                                                   | 74/200 [13:39<19:48,  9.43s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018405914306640625\n",
      "The inverse time is 0.027598857879638672\n",
      "The inverse time is 0.21611499786376953\n",
      "The inverse time is 2.346043348312378\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 38%|████████████████████████████████████████▏                                                                  | 75/200 [13:48<19:17,  9.26s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001964569091796875\n",
      "The inverse time is 0.019742727279663086\n",
      "The inverse time is 0.27803540229797363\n",
      "The inverse time is 2.401747226715088\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 38%|████████████████████████████████████████▋                                                                  | 76/200 [13:58<19:18,  9.34s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020742416381835938\n",
      "The inverse time is 0.03253769874572754\n",
      "The inverse time is 0.30371975898742676\n",
      "The inverse time is 2.459747791290283\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 38%|█████████████████████████████████████████▏                                                                 | 77/200 [14:07<18:55,  9.23s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00017023086547851562\n",
      "The inverse time is 0.012522459030151367\n",
      "The inverse time is 0.3069911003112793\n",
      "The inverse time is 2.427736282348633\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 39%|█████████████████████████████████████████▋                                                                 | 78/200 [14:16<18:32,  9.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001583099365234375\n",
      "The inverse time is 0.0007877349853515625\n",
      "The inverse time is 0.22849225997924805\n",
      "The inverse time is 2.514561414718628\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|██████████████████████████████████████████▎                                                                | 79/200 [14:26<19:13,  9.53s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018477439880371094\n",
      "The inverse time is 0.0005843639373779297\n",
      "The inverse time is 0.21996092796325684\n",
      "The inverse time is 2.3942034244537354\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|██████████████████████████████████████████▊                                                                | 80/200 [14:35<18:52,  9.43s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019502639770507812\n",
      "The inverse time is 0.05125737190246582\n",
      "The inverse time is 0.2868537902832031\n",
      "The inverse time is 2.536309242248535\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|███████████████████████████████████████████▎                                                               | 81/200 [14:44<18:29,  9.32s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001990795135498047\n",
      "The inverse time is 0.024554967880249023\n",
      "The inverse time is 0.2871983051300049\n",
      "The inverse time is 2.1685690879821777\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 41%|███████████████████████████████████████████▊                                                               | 82/200 [14:53<18:01,  9.17s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.000225067138671875\n",
      "The inverse time is 0.03271985054016113\n",
      "The inverse time is 0.266998291015625\n",
      "The inverse time is 2.5207631587982178\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 42%|████████████████████████████████████████████▍                                                              | 83/200 [15:02<17:39,  9.05s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00024700164794921875\n",
      "The inverse time is 0.1150212287902832\n",
      "The inverse time is 0.2438805103302002\n",
      "The inverse time is 2.396484136581421\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 42%|████████████████████████████████████████████▉                                                              | 84/200 [15:11<17:19,  8.96s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00811767578125\n",
      "The inverse time is 0.00847625732421875\n",
      "The inverse time is 0.2238144874572754\n",
      "The inverse time is 2.2412378787994385\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 42%|█████████████████████████████████████████████▍                                                             | 85/200 [15:20<17:16,  9.01s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002162456512451172\n",
      "The inverse time is 0.07040190696716309\n",
      "The inverse time is 0.26218557357788086\n",
      "The inverse time is 2.095140218734741\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 43%|██████████████████████████████████████████████                                                             | 86/200 [15:29<17:16,  9.09s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018596649169921875\n",
      "The inverse time is 0.0006899833679199219\n",
      "The inverse time is 0.28171730041503906\n",
      "The inverse time is 2.1178197860717773\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 44%|██████████████████████████████████████████████▌                                                            | 87/200 [15:38<16:55,  8.99s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018525123596191406\n",
      "The inverse time is 0.04065084457397461\n",
      "The inverse time is 0.21720385551452637\n",
      "The inverse time is 2.40523624420166\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 44%|███████████████████████████████████████████████                                                            | 88/200 [15:48<17:16,  9.26s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002219676971435547\n",
      "The inverse time is 0.07449841499328613\n",
      "The inverse time is 0.2955484390258789\n",
      "The inverse time is 2.421722650527954\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 44%|███████████████████████████████████████████████▌                                                           | 89/200 [15:57<16:59,  9.18s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018405914306640625\n",
      "The inverse time is 0.04994678497314453\n",
      "The inverse time is 0.33267664909362793\n",
      "The inverse time is 2.2828495502471924\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 45%|████████████████████████████████████████████████▏                                                          | 90/200 [16:06<16:41,  9.10s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021839141845703125\n",
      "The inverse time is 0.11185312271118164\n",
      "The inverse time is 0.27840328216552734\n",
      "The inverse time is 2.3408725261688232\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 46%|████████████████████████████████████████████████▋                                                          | 91/200 [16:15<16:43,  9.21s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00817728042602539\n",
      "The inverse time is 0.0668184757232666\n",
      "The inverse time is 0.3011174201965332\n",
      "The inverse time is 2.3365132808685303\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 46%|█████████████████████████████████████████████████▏                                                         | 92/200 [16:24<16:28,  9.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0008816719055175781\n",
      "The inverse time is 0.12784790992736816\n",
      "The inverse time is 0.23062705993652344\n",
      "The inverse time is 2.105332374572754\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 46%|█████████████████████████████████████████████████▊                                                         | 93/200 [16:33<16:05,  9.02s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022983551025390625\n",
      "The inverse time is 0.03379654884338379\n",
      "The inverse time is 0.2617807388305664\n",
      "The inverse time is 2.1115400791168213\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 47%|██████████████████████████████████████████████████▎                                                        | 94/200 [16:41<15:32,  8.80s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021696090698242188\n",
      "The inverse time is 0.12468075752258301\n",
      "The inverse time is 0.3458704948425293\n",
      "The inverse time is 2.2667977809906006\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 48%|██████████████████████████████████████████████████▊                                                        | 95/200 [16:50<15:32,  8.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00023603439331054688\n",
      "The inverse time is 0.0628199577331543\n",
      "The inverse time is 0.32488489151000977\n",
      "The inverse time is 2.4421122074127197\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 48%|███████████████████████████████████████████████████▎                                                       | 96/200 [16:59<15:26,  8.90s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00027680397033691406\n",
      "The inverse time is 0.024636030197143555\n",
      "The inverse time is 0.31371617317199707\n",
      "The inverse time is 2.3016743659973145\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 48%|███████████████████████████████████████████████████▉                                                       | 97/200 [17:08<15:19,  8.92s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020051002502441406\n",
      "The inverse time is 0.03635811805725098\n",
      "The inverse time is 0.24534988403320312\n",
      "The inverse time is 2.6426146030426025\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 49%|████████████████████████████████████████████████████▍                                                      | 98/200 [17:18<15:32,  9.14s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022935867309570312\n",
      "The inverse time is 0.13709616661071777\n",
      "The inverse time is 0.2177278995513916\n",
      "The inverse time is 2.3487937450408936\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|████████████████████████████████████████████████████▉                                                      | 99/200 [17:27<15:19,  9.10s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.000186920166015625\n",
      "The inverse time is 0.046125173568725586\n",
      "The inverse time is 0.28377604484558105\n",
      "The inverse time is 2.3252806663513184\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|█████████████████████████████████████████████████████                                                     | 100/200 [17:38<16:18,  9.78s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002067089080810547\n",
      "The inverse time is 0.02621603012084961\n",
      "The inverse time is 0.26139044761657715\n",
      "The inverse time is 2.5257911682128906\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|█████████████████████████████████████████████████████▌                                                    | 101/200 [17:48<16:12,  9.82s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002040863037109375\n",
      "The inverse time is 0.03299355506896973\n",
      "The inverse time is 0.24750542640686035\n",
      "The inverse time is 2.3096909523010254\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 51%|██████████████████████████████████████████████████████                                                    | 102/200 [17:58<16:10,  9.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002353191375732422\n",
      "The inverse time is 0.059090375900268555\n",
      "The inverse time is 0.24350881576538086\n",
      "The inverse time is 2.2515218257904053\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 52%|██████████████████████████████████████████████████████▌                                                   | 103/200 [18:07<15:40,  9.69s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002033710479736328\n",
      "The inverse time is 0.03849053382873535\n",
      "The inverse time is 0.25231385231018066\n",
      "The inverse time is 2.3186521530151367\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 52%|███████████████████████████████████████████████████████                                                   | 104/200 [18:16<14:56,  9.34s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002009868621826172\n",
      "The inverse time is 0.08881640434265137\n",
      "The inverse time is 0.278594970703125\n",
      "The inverse time is 2.2227611541748047\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 52%|███████████████████████████████████████████████████████▋                                                  | 105/200 [18:25<14:34,  9.20s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018858909606933594\n",
      "The inverse time is 0.10101079940795898\n",
      "The inverse time is 0.3067777156829834\n",
      "The inverse time is 2.163095474243164\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 53%|████████████████████████████████████████████████████████▏                                                 | 106/200 [18:33<14:11,  9.06s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00017833709716796875\n",
      "The inverse time is 0.029272079467773438\n",
      "The inverse time is 0.2751145362854004\n",
      "The inverse time is 2.2468183040618896\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 54%|████████████████████████████████████████████████████████▋                                                 | 107/200 [18:42<13:54,  8.97s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002372264862060547\n",
      "The inverse time is 0.08533501625061035\n",
      "The inverse time is 0.3091752529144287\n",
      "The inverse time is 2.4016013145446777\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 54%|█████████████████████████████████████████████████████████▏                                                | 108/200 [18:51<13:51,  9.04s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018787384033203125\n",
      "The inverse time is 0.08099174499511719\n",
      "The inverse time is 0.24585413932800293\n",
      "The inverse time is 2.438215494155884\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████████████████████████████████████████▊                                                | 109/200 [19:00<13:30,  8.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018167495727539062\n",
      "The inverse time is 0.06274008750915527\n",
      "The inverse time is 0.25696873664855957\n",
      "The inverse time is 2.511920213699341\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|██████████████████████████████████████████████████████████▎                                               | 110/200 [19:10<13:44,  9.16s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.007231235504150391\n",
      "The inverse time is 0.03261709213256836\n",
      "The inverse time is 0.28321003913879395\n",
      "The inverse time is 2.3063361644744873\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 56%|██████████████████████████████████████████████████████████▊                                               | 111/200 [19:19<13:23,  9.03s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n",
      "The inverse time is 0.0002491474151611328\n",
      "The inverse time is 0.1312568187713623\n",
      "The inverse time is 0.28740930557250977\n",
      "The inverse time is 2.402043342590332\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 56%|███████████████████████████████████████████████████████████▎                                              | 112/200 [19:28<13:31,  9.22s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002446174621582031\n",
      "The inverse time is 0.0770418643951416\n",
      "The inverse time is 0.3237576484680176\n",
      "The inverse time is 2.3023886680603027\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 56%|███████████████████████████████████████████████████████████▉                                              | 113/200 [19:37<13:03,  9.01s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00017142295837402344\n",
      "The inverse time is 0.008325815200805664\n",
      "The inverse time is 0.345151424407959\n",
      "The inverse time is 2.4950618743896484\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 57%|████████████████████████████████████████████████████████████▍                                             | 114/200 [19:46<12:51,  8.97s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002129077911376953\n",
      "The inverse time is 0.05979514122009277\n",
      "The inverse time is 0.21952033042907715\n",
      "The inverse time is 2.0635874271392822\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 57%|████████████████████████████████████████████████████████████▉                                             | 115/200 [19:54<12:32,  8.86s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021529197692871094\n",
      "The inverse time is 0.024651288986206055\n",
      "The inverse time is 0.29889440536499023\n",
      "The inverse time is 2.3672523498535156\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 58%|█████████████████████████████████████████████████████████████▍                                            | 116/200 [20:03<12:31,  8.95s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022220611572265625\n",
      "The inverse time is 0.09627795219421387\n",
      "The inverse time is 0.3262901306152344\n",
      "The inverse time is 2.104763984680176\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 58%|██████████████████████████████████████████████████████████████                                            | 117/200 [20:12<12:09,  8.79s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.000152587890625\n",
      "The inverse time is 0.12312555313110352\n",
      "The inverse time is 0.26085710525512695\n",
      "The inverse time is 2.4049994945526123\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 59%|██████████████████████████████████████████████████████████████▌                                           | 118/200 [20:21<12:05,  8.85s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002014636993408203\n",
      "The inverse time is 0.07234668731689453\n",
      "The inverse time is 0.27318429946899414\n",
      "The inverse time is 2.6825356483459473\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████████████████████████████████████████████                                           | 119/200 [20:30<12:01,  8.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019168853759765625\n",
      "The inverse time is 0.07164907455444336\n",
      "The inverse time is 0.31514739990234375\n",
      "The inverse time is 2.3424103260040283\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|███████████████████████████████████████████████████████████████▌                                          | 120/200 [20:39<11:58,  8.98s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00016427040100097656\n",
      "The inverse time is 0.14142274856567383\n",
      "The inverse time is 0.3649425506591797\n",
      "The inverse time is 2.3000073432922363\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|████████████████████████████████████████████████████████████████▏                                         | 121/200 [20:48<11:53,  9.04s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001690387725830078\n",
      "The inverse time is 0.02437281608581543\n",
      "The inverse time is 0.23593688011169434\n",
      "The inverse time is 2.459495782852173\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 61%|████████████████████████████████████████████████████████████████▋                                         | 122/200 [20:57<11:41,  9.00s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019669532775878906\n",
      "The inverse time is 0.1385352611541748\n",
      "The inverse time is 0.24111247062683105\n",
      "The inverse time is 2.241269826889038\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 62%|█████████████████████████████████████████████████████████████████▏                                        | 123/200 [21:06<11:23,  8.87s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020122528076171875\n",
      "The inverse time is 0.07807493209838867\n",
      "The inverse time is 0.2521977424621582\n",
      "The inverse time is 2.385063886642456\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 62%|█████████████████████████████████████████████████████████████████▋                                        | 124/200 [21:15<11:26,  9.03s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001971721649169922\n",
      "The inverse time is 0.022052526473999023\n",
      "The inverse time is 0.254596471786499\n",
      "The inverse time is 2.381741762161255\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 62%|██████████████████████████████████████████████████████████████████▎                                       | 125/200 [21:24<11:18,  9.05s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020766258239746094\n",
      "The inverse time is 0.05264163017272949\n",
      "The inverse time is 0.27678990364074707\n",
      "The inverse time is 2.2854838371276855\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 63%|██████████████████████████████████████████████████████████████████▊                                       | 126/200 [21:33<10:59,  8.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021791458129882812\n",
      "The inverse time is 0.11083841323852539\n",
      "The inverse time is 0.33733487129211426\n",
      "The inverse time is 2.3136205673217773\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 64%|███████████████████████████████████████████████████████████████████▎                                      | 127/200 [21:42<10:59,  9.04s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022482872009277344\n",
      "The inverse time is 0.03337907791137695\n",
      "The inverse time is 0.20529985427856445\n",
      "The inverse time is 2.248234748840332\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 64%|███████████████████████████████████████████████████████████████████▊                                      | 128/200 [21:51<10:47,  8.99s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019407272338867188\n",
      "The inverse time is 0.016303300857543945\n",
      "The inverse time is 0.2962913513183594\n",
      "The inverse time is 2.3760507106781006\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 64%|████████████████████████████████████████████████████████████████████▎                                     | 129/200 [22:00<10:36,  8.97s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021028518676757812\n",
      "The inverse time is 0.12950682640075684\n",
      "The inverse time is 0.392911434173584\n",
      "The inverse time is 2.4121768474578857\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 65%|████████████████████████████████████████████████████████████████████▉                                     | 130/200 [22:09<10:39,  9.13s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019288063049316406\n",
      "The inverse time is 0.12749838829040527\n",
      "The inverse time is 0.30544400215148926\n",
      "The inverse time is 2.7574987411499023\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 66%|█████████████████████████████████████████████████████████████████████▍                                    | 131/200 [22:19<10:37,  9.24s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020194053649902344\n",
      "The inverse time is 0.03682732582092285\n",
      "The inverse time is 0.29203128814697266\n",
      "The inverse time is 2.446054697036743\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 66%|█████████████████████████████████████████████████████████████████████▉                                    | 132/200 [22:28<10:24,  9.19s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00015425682067871094\n",
      "The inverse time is 0.012604713439941406\n",
      "The inverse time is 0.26163649559020996\n",
      "The inverse time is 2.0658106803894043\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 66%|██████████████████████████████████████████████████████████████████████▍                                   | 133/200 [22:36<10:00,  8.97s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001804828643798828\n",
      "The inverse time is 0.09721064567565918\n",
      "The inverse time is 0.2521195411682129\n",
      "The inverse time is 2.3473541736602783\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 67%|███████████████████████████████████████████████████████████████████████                                   | 134/200 [22:45<09:51,  8.96s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018978118896484375\n",
      "The inverse time is 0.012651920318603516\n",
      "The inverse time is 0.2508280277252197\n",
      "The inverse time is 2.229696750640869\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 68%|███████████████████████████████████████████████████████████████████████▌                                  | 135/200 [22:55<09:52,  9.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002720355987548828\n",
      "The inverse time is 0.059911489486694336\n",
      "The inverse time is 0.26213669776916504\n",
      "The inverse time is 2.3480818271636963\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 68%|████████████████████████████████████████████████████████████████████████                                  | 136/200 [23:04<09:42,  9.10s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020623207092285156\n",
      "The inverse time is 0.1223912239074707\n",
      "The inverse time is 0.25255417823791504\n",
      "The inverse time is 2.260514974594116\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 68%|████████████████████████████████████████████████████████████████████████▌                                 | 137/200 [23:12<09:19,  8.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001952648162841797\n",
      "The inverse time is 0.024791240692138672\n",
      "The inverse time is 0.24293899536132812\n",
      "The inverse time is 2.3318638801574707\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 69%|█████████████████████████████████████████████████████████████████████████▏                                | 138/200 [23:21<09:08,  8.85s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00023865699768066406\n",
      "The inverse time is 0.025432109832763672\n",
      "The inverse time is 0.30950403213500977\n",
      "The inverse time is 2.383915662765503\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|█████████████████████████████████████████████████████████████████████████▋                                | 139/200 [23:30<08:56,  8.79s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021600723266601562\n",
      "The inverse time is 0.025945425033569336\n",
      "The inverse time is 0.2861337661743164\n",
      "The inverse time is 2.3010189533233643\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|██████████████████████████████████████████████████████████████████████████▏                               | 140/200 [23:38<08:47,  8.80s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002167224884033203\n",
      "The inverse time is 0.0991675853729248\n",
      "The inverse time is 0.1851792335510254\n",
      "The inverse time is 2.2702279090881348\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 70%|██████████████████████████████████████████████████████████████████████████▋                               | 141/200 [23:48<08:44,  8.89s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019049644470214844\n",
      "The inverse time is 0.0995626449584961\n",
      "The inverse time is 0.2588779926300049\n",
      "The inverse time is 2.104586362838745\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 71%|███████████████████████████████████████████████████████████████████████████▎                              | 142/200 [23:56<08:36,  8.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.004307746887207031\n",
      "The inverse time is 0.024438858032226562\n",
      "The inverse time is 0.3346707820892334\n",
      "The inverse time is 2.354870080947876\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 72%|███████████████████████████████████████████████████████████████████████████▊                              | 143/200 [24:05<08:26,  8.89s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019168853759765625\n",
      "The inverse time is 0.08745098114013672\n",
      "The inverse time is 0.19913053512573242\n",
      "The inverse time is 2.4683470726013184\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 72%|████████████████████████████████████████████████████████████████████████████▎                             | 144/200 [24:14<08:15,  8.84s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.008141279220581055\n",
      "The inverse time is 0.08061599731445312\n",
      "The inverse time is 0.3436715602874756\n",
      "The inverse time is 2.4683194160461426\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 72%|████████████████████████████████████████████████████████████████████████████▊                             | 145/200 [24:24<08:24,  9.18s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.008127450942993164\n",
      "The inverse time is 0.0779576301574707\n",
      "The inverse time is 0.30574727058410645\n",
      "The inverse time is 2.2645606994628906\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 73%|█████████████████████████████████████████████████████████████████████████████▍                            | 146/200 [24:33<08:11,  9.10s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.001172780990600586\n",
      "The inverse time is 0.06779026985168457\n",
      "The inverse time is 0.2556583881378174\n",
      "The inverse time is 2.4069459438323975\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 74%|█████████████████████████████████████████████████████████████████████████████▉                            | 147/200 [24:42<08:01,  9.09s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.006173133850097656\n",
      "The inverse time is 0.05092120170593262\n",
      "The inverse time is 0.21960783004760742\n",
      "The inverse time is 2.3577706813812256\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 74%|██████████████████████████████████████████████████████████████████████████████▍                           | 148/200 [24:51<07:54,  9.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0056760311126708984\n",
      "The inverse time is 0.09479522705078125\n",
      "The inverse time is 0.2848994731903076\n",
      "The inverse time is 2.561013698577881\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 74%|██████████████████████████████████████████████████████████████████████████████▉                           | 149/200 [25:01<07:53,  9.28s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002155303955078125\n",
      "The inverse time is 0.056095123291015625\n",
      "The inverse time is 0.20318007469177246\n",
      "The inverse time is 2.35774827003479\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 75%|███████████████████████████████████████████████████████████████████████████████▌                          | 150/200 [25:10<07:45,  9.30s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022101402282714844\n",
      "The inverse time is 0.012606382369995117\n",
      "The inverse time is 0.32939743995666504\n",
      "The inverse time is 2.2771387100219727\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 76%|████████████████████████████████████████████████████████████████████████████████                          | 151/200 [25:19<07:29,  9.17s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018095970153808594\n",
      "The inverse time is 0.016428470611572266\n",
      "The inverse time is 0.3140895366668701\n",
      "The inverse time is 2.3782753944396973\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 76%|████████████████████████████████████████████████████████████████████████████████▌                         | 152/200 [25:29<07:31,  9.41s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001468658447265625\n",
      "The inverse time is 0.1358792781829834\n",
      "The inverse time is 0.2909269332885742\n",
      "The inverse time is 2.3210434913635254\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 76%|█████████████████████████████████████████████████████████████████████████████████                         | 153/200 [25:39<07:24,  9.47s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00017499923706054688\n",
      "The inverse time is 0.16698861122131348\n",
      "The inverse time is 0.22504544258117676\n",
      "The inverse time is 2.2341463565826416\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 77%|█████████████████████████████████████████████████████████████████████████████████▌                        | 154/200 [25:48<07:09,  9.33s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.002027273178100586\n",
      "The inverse time is 0.11757755279541016\n",
      "The inverse time is 0.306795597076416\n",
      "The inverse time is 2.192885398864746\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 78%|██████████████████████████████████████████████████████████████████████████████████▏                       | 155/200 [25:57<06:56,  9.25s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002117156982421875\n",
      "The inverse time is 0.025521039962768555\n",
      "The inverse time is 0.26562023162841797\n",
      "The inverse time is 2.315321207046509\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 78%|██████████████████████████████████████████████████████████████████████████████████▋                       | 156/200 [26:05<06:38,  9.07s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0011234283447265625\n",
      "The inverse time is 0.0403292179107666\n",
      "The inverse time is 0.28580379486083984\n",
      "The inverse time is 2.2049436569213867\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 78%|███████████████████████████████████████████████████████████████████████████████████▏                      | 157/200 [26:14<06:27,  9.01s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022220611572265625\n",
      "The inverse time is 0.05654406547546387\n",
      "The inverse time is 0.2376718521118164\n",
      "The inverse time is 2.2715883255004883\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 79%|███████████████████████████████████████████████████████████████████████████████████▋                      | 158/200 [26:24<06:28,  9.26s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022339820861816406\n",
      "The inverse time is 0.020346641540527344\n",
      "The inverse time is 0.24242568016052246\n",
      "The inverse time is 2.338545083999634\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|████████████████████████████████████████████████████████████████████████████████████▎                     | 159/200 [26:33<06:10,  9.04s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.001392364501953125\n",
      "The inverse time is 0.08904004096984863\n",
      "The inverse time is 0.3548433780670166\n",
      "The inverse time is 2.2304270267486572\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|████████████████████████████████████████████████████████████████████████████████████▊                     | 160/200 [26:42<06:07,  9.18s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.008101940155029297\n",
      "The inverse time is 0.09113597869873047\n",
      "The inverse time is 0.30499792098999023\n",
      "The inverse time is 2.4202919006347656\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|█████████████████████████████████████████████████████████████████████████████████████▎                    | 161/200 [26:51<05:59,  9.22s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00017118453979492188\n",
      "The inverse time is 0.024602413177490234\n",
      "The inverse time is 0.2796669006347656\n",
      "The inverse time is 2.3263518810272217\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 81%|█████████████████████████████████████████████████████████████████████████████████████▊                    | 162/200 [27:00<05:47,  9.16s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00023984909057617188\n",
      "The inverse time is 0.03270578384399414\n",
      "The inverse time is 0.2827744483947754\n",
      "The inverse time is 2.329759120941162\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 82%|██████████████████████████████████████████████████████████████████████████████████████▍                   | 163/200 [27:10<05:45,  9.35s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022554397583007812\n",
      "The inverse time is 0.1606614589691162\n",
      "The inverse time is 0.2726449966430664\n",
      "The inverse time is 2.363133430480957\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 82%|██████████████████████████████████████████████████████████████████████████████████████▉                   | 164/200 [27:23<06:09, 10.27s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020503997802734375\n",
      "The inverse time is 0.05232429504394531\n",
      "The inverse time is 0.18925189971923828\n",
      "The inverse time is 2.2609074115753174\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 82%|███████████████████████████████████████████████████████████████████████████████████████▍                  | 165/200 [27:32<05:47,  9.93s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.000209808349609375\n",
      "The inverse time is 0.02653217315673828\n",
      "The inverse time is 0.2820165157318115\n",
      "The inverse time is 2.3838765621185303\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 83%|███████████████████████████████████████████████████████████████████████████████████████▉                  | 166/200 [27:42<05:37,  9.93s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0010254383087158203\n",
      "The inverse time is 0.08715486526489258\n",
      "The inverse time is 0.19904422760009766\n",
      "The inverse time is 2.2645821571350098\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 84%|████████████████████████████████████████████████████████████████████████████████████████▌                 | 167/200 [27:50<05:15,  9.56s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018930435180664062\n",
      "The inverse time is 0.025429725646972656\n",
      "The inverse time is 0.28849029541015625\n",
      "The inverse time is 2.3873841762542725\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 84%|█████████████████████████████████████████████████████████████████████████████████████████                 | 168/200 [28:01<05:10,  9.72s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00015115737915039062\n",
      "The inverse time is 0.015835046768188477\n",
      "The inverse time is 0.2751142978668213\n",
      "The inverse time is 2.468731641769409\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 84%|█████████████████████████████████████████████████████████████████████████████████████████▌                | 169/200 [28:10<05:01,  9.73s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00017642974853515625\n",
      "The inverse time is 0.0893561840057373\n",
      "The inverse time is 0.35822272300720215\n",
      "The inverse time is 2.256361246109009\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 85%|██████████████████████████████████████████████████████████████████████████████████████████                | 170/200 [28:19<04:43,  9.45s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021123886108398438\n",
      "The inverse time is 0.0009205341339111328\n",
      "The inverse time is 0.18481159210205078\n",
      "The inverse time is 2.207352638244629\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 86%|██████████████████████████████████████████████████████████████████████████████████████████▋               | 171/200 [28:28<04:25,  9.17s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020623207092285156\n",
      "The inverse time is 0.059339284896850586\n",
      "The inverse time is 0.3028590679168701\n",
      "The inverse time is 2.352163314819336\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 86%|███████████████████████████████████████████████████████████████████████████████████████████▏              | 172/200 [28:36<04:14,  9.09s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021195411682128906\n",
      "The inverse time is 0.024515628814697266\n",
      "The inverse time is 0.2270040512084961\n",
      "The inverse time is 2.3311243057250977\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 86%|███████████████████████████████████████████████████████████████████████████████████████████▋              | 173/200 [28:45<04:03,  9.03s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00022745132446289062\n",
      "The inverse time is 0.07385015487670898\n",
      "The inverse time is 0.2513236999511719\n",
      "The inverse time is 2.3382365703582764\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 87%|████████████████████████████████████████████████████████████████████████████████████████████▏             | 174/200 [28:55<03:57,  9.13s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.000156402587890625\n",
      "The inverse time is 0.07761859893798828\n",
      "The inverse time is 0.2519686222076416\n",
      "The inverse time is 2.280503988265991\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 88%|████████████████████████████████████████████████████████████████████████████████████████████▊             | 175/200 [29:04<03:48,  9.12s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019288063049316406\n",
      "The inverse time is 0.07647156715393066\n",
      "The inverse time is 0.2017958164215088\n",
      "The inverse time is 2.1810314655303955\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 88%|█████████████████████████████████████████████████████████████████████████████████████████████▎            | 176/200 [29:13<03:37,  9.05s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00014781951904296875\n",
      "The inverse time is 0.06927204132080078\n",
      "The inverse time is 0.2271125316619873\n",
      "The inverse time is 2.3420145511627197\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 88%|█████████████████████████████████████████████████████████████████████████████████████████████▊            | 177/200 [29:23<03:33,  9.29s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001919269561767578\n",
      "The inverse time is 0.1421966552734375\n",
      "The inverse time is 0.2379770278930664\n",
      "The inverse time is 2.380974531173706\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 89%|██████████████████████████████████████████████████████████████████████████████████████████████▎           | 178/200 [29:32<03:28,  9.47s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00016307830810546875\n",
      "The inverse time is 0.09400057792663574\n",
      "The inverse time is 0.27655887603759766\n",
      "The inverse time is 2.6782264709472656\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|██████████████████████████████████████████████████████████████████████████████████████████████▊           | 179/200 [29:41<03:16,  9.34s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018835067749023438\n",
      "The inverse time is 0.009619474411010742\n",
      "The inverse time is 0.2511305809020996\n",
      "The inverse time is 2.113725185394287\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████████████████████████████████████████████████████████████████▍          | 180/200 [29:51<03:04,  9.25s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019931793212890625\n",
      "The inverse time is 0.000926971435546875\n",
      "The inverse time is 0.2681899070739746\n",
      "The inverse time is 2.3865840435028076\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████████████████████████████████████████████████████████████████▉          | 181/200 [29:59<02:53,  9.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018334388732910156\n",
      "The inverse time is 0.07226896286010742\n",
      "The inverse time is 0.27258968353271484\n",
      "The inverse time is 2.4044315814971924\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 91%|████████████████████████████████████████████████████████████████████████████████████████████████▍         | 182/200 [30:09<02:44,  9.15s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.007476806640625\n",
      "The inverse time is 0.03923654556274414\n",
      "The inverse time is 0.2656385898590088\n",
      "The inverse time is 2.121950149536133\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 92%|████████████████████████████████████████████████████████████████████████████████████████████████▉         | 183/200 [30:17<02:31,  8.93s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020432472229003906\n",
      "The inverse time is 0.1169426441192627\n",
      "The inverse time is 0.3006551265716553\n",
      "The inverse time is 2.3046493530273438\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 92%|█████████████████████████████████████████████████████████████████████████████████████████████████▌        | 184/200 [30:26<02:23,  8.98s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021409988403320312\n",
      "The inverse time is 0.10782074928283691\n",
      "The inverse time is 0.3030240535736084\n",
      "The inverse time is 2.5005321502685547\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 92%|██████████████████████████████████████████████████████████████████████████████████████████████████        | 185/200 [30:35<02:16,  9.08s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00020051002502441406\n",
      "The inverse time is 0.028889179229736328\n",
      "The inverse time is 0.3013920783996582\n",
      "The inverse time is 2.547170877456665\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 93%|██████████████████████████████████████████████████████████████████████████████████████████████████▌       | 186/200 [30:45<02:08,  9.16s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001964569091796875\n",
      "The inverse time is 0.0722508430480957\n",
      "The inverse time is 0.3336052894592285\n",
      "The inverse time is 2.476804494857788\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 94%|███████████████████████████████████████████████████████████████████████████████████████████████████       | 187/200 [30:55<02:02,  9.43s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00017189979553222656\n",
      "The inverse time is 0.1461620330810547\n",
      "The inverse time is 0.21849679946899414\n",
      "The inverse time is 2.4583747386932373\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 94%|███████████████████████████████████████████████████████████████████████████████████████████████████▋      | 188/200 [31:04<01:50,  9.21s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00019788742065429688\n",
      "The inverse time is 0.10053658485412598\n",
      "The inverse time is 0.1860826015472412\n",
      "The inverse time is 2.1700360774993896\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 94%|████████████████████████████████████████████████████████████████████████████████████████████████████▏     | 189/200 [31:12<01:40,  9.11s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002181529998779297\n",
      "The inverse time is 0.017059326171875\n",
      "The inverse time is 0.3177046775817871\n",
      "The inverse time is 2.2025320529937744\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|████████████████████████████████████████████████████████████████████████████████████████████████████▋     | 190/200 [31:21<01:30,  9.05s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00018358230590820312\n",
      "The inverse time is 0.1076345443725586\n",
      "The inverse time is 0.29408740997314453\n",
      "The inverse time is 2.5124943256378174\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 96%|█████████████████████████████████████████████████████████████████████████████████████████████████████▏    | 191/200 [31:31<01:21,  9.11s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.002448558807373047\n",
      "The inverse time is 0.09458684921264648\n",
      "The inverse time is 0.31171393394470215\n",
      "The inverse time is 2.258408784866333\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 96%|█████████████████████████████████████████████████████████████████████████████████████████████████████▊    | 192/200 [31:39<01:12,  9.06s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00021004676818847656\n",
      "The inverse time is 0.02623438835144043\n",
      "The inverse time is 0.2677466869354248\n",
      "The inverse time is 2.1075599193573\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 96%|██████████████████████████████████████████████████████████████████████████████████████████████████████▎   | 193/200 [31:48<01:02,  8.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002276897430419922\n",
      "The inverse time is 0.1480240821838379\n",
      "The inverse time is 0.21369147300720215\n",
      "The inverse time is 2.0482609272003174\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 97%|██████████████████████████████████████████████████████████████████████████████████████████████████████▊   | 194/200 [31:57<00:53,  8.84s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00015616416931152344\n",
      "The inverse time is 0.09211015701293945\n",
      "The inverse time is 0.21538639068603516\n",
      "The inverse time is 2.0339486598968506\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 98%|███████████████████████████████████████████████████████████████████████████████████████████████████████▎  | 195/200 [32:06<00:44,  8.83s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001480579376220703\n",
      "The inverse time is 0.01974320411682129\n",
      "The inverse time is 0.16141176223754883\n",
      "The inverse time is 2.1485843658447266\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 98%|███████████████████████████████████████████████████████████████████████████████████████████████████████▉  | 196/200 [32:15<00:36,  9.00s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0002052783966064453\n",
      "The inverse time is 0.07236933708190918\n",
      "The inverse time is 0.2543354034423828\n",
      "The inverse time is 2.324188232421875\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 98%|████████████████████████████████████████████████████████████████████████████████████████████████████████▍ | 197/200 [32:24<00:27,  9.09s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.002669811248779297\n",
      "The inverse time is 0.01938629150390625\n",
      "The inverse time is 0.18296384811401367\n",
      "The inverse time is 2.2361257076263428\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 99%|████████████████████████████████████████████████████████████████████████████████████████████████████████▉ | 198/200 [32:33<00:18,  9.07s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.00024390220642089844\n",
      "The inverse time is 0.07517218589782715\n",
      "The inverse time is 0.28507542610168457\n",
      "The inverse time is 2.183077573776245\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████████████████▍| 199/200 [32:42<00:08,  8.95s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 0.0001697540283203125\n",
      "The inverse time is 0.008452892303466797\n",
      "The inverse time is 0.29216456413269043\n",
      "The inverse time is 2.332702875137329\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████| 200/200 [32:51<00:00,  9.86s/it]\n"
     ]
    }
   ],
   "source": [
    "# small scale\n",
    "n_list = [10, 50, 100, 500]\n",
    "nconstraints_list = [5, 10, 20, 100]\n",
    "stats = {}\n",
    "iters = 200\n",
    "for i in tqdm(range(iters)):\n",
    "    for ndim,neq in zip(n_list,nconstraints_list):\n",
    "        P,q,G,b,A,h,osA,l,u,Pm = QP_instances(ndim,neq,neq) # neq = nineq\n",
    "        x_value, y_value, time_spent_forward_osqp = osqp_interface(P,q,osA,l,u) # OSQP Forward\n",
    "        x_grad, y_grad, time_spent_backward_osqp = QP_OSQP_backward(y_value,P,G,A) # OSQP Backward\n",
    "        x_cp_value,y_cp_value, x_cp_grad,time_spent_forward,time_spent_backward = QP_cvxpy_backward(Pm,G,A,h,b,q) # cvxpy Forward and Backward\n",
    "\n",
    "        x_qpth_value, x_qpth_grad,time_spent_qpth_forward, time_qpth_backward= QP_qpth_evaluate(Pm,G,A,h,b,q)\n",
    "        x_alt, x_alt_grad, inverse_time,sol_time = alt_diff(Pm,q,A, b, G, h)\n",
    "        # JAXOpt\n",
    "        # t0 = time.time():\n",
    "        # if t0\n",
    "        # Q = Pm\n",
    "        # c = q\n",
    "        # qp = jaxopt.OSQP()\n",
    "        # t7 = time.time()\n",
    "        # sol = qp.run(params_obj=(Q, c), params_eq=(A, b), params_ineq=(G, h)).params\n",
    "        # t8 = time.time()\n",
    "        # jax_gradient = jax.grad(jax_qp_solver)(c)\n",
    "        # t9 = time.time()\n",
    "\n",
    "        exact_bb,t_spent = QP_cal_exact_backward(Pm,G,A,y_value,x_value,h) # Exact backward\n",
    "        acc_forward = cal_cos_accuracy(x_value,x_qpth_value)\n",
    "        acc_osqp_bb = cal_cos_accuracy(exact_bb,x_grad)\n",
    "        acc_cvxpy_bb = cal_cos_accuracy(exact_bb,x_cp_grad)\n",
    "        acc_qpth_bb = cal_cos_accuracy(exact_bb,x_qpth_grad)\n",
    "        acc_alt_bb = cal_cos_accuracy(exact_bb,x_alt_grad)\n",
    "        # acc_jax_bb = cal_cos_accuracy(exact_bb,jax_gradient)\n",
    "\n",
    "        dict_report(stats, 'Time OSQP Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_spent_forward_osqp)\n",
    "        dict_report(stats, 'Time OSQP Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_spent_backward_osqp)\n",
    "        dict_report(stats, 'Time CVXPY Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_spent_forward)\n",
    "        dict_report(stats, 'Time CVXPY Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_spent_backward)\n",
    "        dict_report(stats, 'Time Exact Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', t_spent)\n",
    "        dict_report(stats, 'Time Qpth Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_qpth_backward)\n",
    "        dict_report(stats, 'Time Qpth Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_spent_qpth_forward)\n",
    "        dict_report(stats, 'Time Alt-diff Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', inverse_time)\n",
    "        dict_report(stats, 'Time Alt-diff Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', sol_time)\n",
    "        # dict_report(stats, 'Time JAXOpt Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', t8-t7)\n",
    "        # dict_report(stats, 'Time JAXOpt Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', t9-t8)\n",
    "\n",
    "        dict_report(stats, 'Accuracy Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_forward)\n",
    "        # dict_report(stats, 'Accuracy JAXOpt Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_jax_bb)\n",
    "        dict_report(stats, 'Accuracy Alt-diff Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_alt_bb)\n",
    "        dict_report(stats, 'Accuracy OSQP Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_osqp_bb)\n",
    "        dict_report(stats, 'Accuracy CVXPY Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_cvxpy_bb)\n",
    "        dict_report(stats, 'Accuracy QPTH Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_qpth_bb)\n",
    "    try:\n",
    "        pd.DataFrame(stats).to_csv('./Small_scale_qp.csv')\n",
    "    except:\n",
    "        pd.Series(stats).to_csv('./Small_scale_qp.csv')"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|                                                                                                                     | 0/10 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 2.4442741870880127\n",
      "The inverse time is 5.583889722824097\n",
      "The inverse time is 8.419827461242676\n",
      "The inverse time is 11.875808715820312\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 10%|██████████▌                                                                                               | 1/10 [14:15<2:08:15, 855.06s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 2.7547011375427246\n",
      "The inverse time is 6.391331911087036\n",
      "The inverse time is 8.5011727809906\n",
      "The inverse time is 11.625163555145264\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|█████████████████████▏                                                                                    | 2/10 [28:30<1:54:03, 855.50s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 2.458101511001587\n",
      "The inverse time is 5.927773714065552\n",
      "The inverse time is 8.60107946395874\n",
      "The inverse time is 11.531760215759277\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|███████████████████████████████▊                                                                          | 3/10 [44:13<1:44:25, 895.13s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 2.5043604373931885\n",
      "The inverse time is 5.733512878417969\n",
      "The inverse time is 8.439656972885132\n",
      "The inverse time is 12.593015909194946\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|██████████████████████████████████████████▍                                                               | 4/10 [58:44<1:28:33, 885.60s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The inverse time is 2.579462766647339\n",
      "The inverse time is 6.085495471954346\n",
      "The inverse time is 8.772213459014893\n",
      "The inverse time is 11.94961142539978\n"
     ]
    }
   ],
   "source": [
    "# large scale\n",
    "n_list = [500, 1500, 3000,5000]\n",
    "nconstraints_list = [200, 500, 1000,2000]\n",
    "stats_large = {}\n",
    "iters = 10\n",
    "for i in tqdm(range(iters)):\n",
    "    for ndim,neq in zip(n_list,nconstraints_list):\n",
    "        P,q,G,b,A,h,osA,l,u,Pm = QP_instances(ndim,neq,neq) # neq = nineq\n",
    "        x_value, y_value, time_spent_forward_osqp = osqp_interface(P,q,osA,l,u) # OSQP Forward\n",
    "        x_grad, y_grad, time_spent_backward_osqp = QP_OSQP_backward(y_value,P,G,A) # OSQP Backward\n",
    "\n",
    "        # x_qpth_value, x_qpth_grad,time_spent_qpth_forward, time_qpth_backward= QP_qpth_evaluate(Pm,G,A,h,b,q)\n",
    "        x_alt, x_alt_grad, inverse_time,sol_time = alt_diff(Pm,q,A, b, G, h)\n",
    "\n",
    "        exact_bb,t_spent = QP_cal_exact_backward(Pm,G,A,y_value,x_value,h) # Exact backward\n",
    "        # acc_forward = cal_cos_accuracy(x_value,x_qpth_value)\n",
    "        acc_osqp_bb = cal_cos_accuracy(exact_bb,x_grad)\n",
    "\n",
    "        # acc_qpth_bb = cal_cos_accuracy(exact_bb,x_qpth_grad)\n",
    "        acc_alt_bb = cal_cos_accuracy(exact_bb,x_alt_grad)\n",
    "        # acc_jax_bb = cal_cos_accuracy(exact_bb,jax_gradient)\n",
    "\n",
    "        dict_report(stats_large, 'Time OSQP Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_spent_forward_osqp)\n",
    "        dict_report(stats_large, 'Time OSQP Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_spent_backward_osqp)\n",
    "        dict_report(stats_large, 'Time Exact Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', t_spent)\n",
    "        # dict_report(stats_large, 'Time Qpth Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_qpth_backward)\n",
    "        # dict_report(stats_large, 'Time Qpth Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', time_spent_qpth_forward)\n",
    "        dict_report(stats_large, 'Time Alt-diff Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', inverse_time)\n",
    "        dict_report(stats_large, 'Time Alt-diff Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', sol_time)\n",
    "\n",
    "        # dict_report(stats_large, 'Accuracy Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_forward)\n",
    "        dict_report(stats_large, 'Accuracy Alt-diff Forward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_alt_bb)\n",
    "        dict_report(stats_large, 'Accuracy OSQP Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_osqp_bb)\n",
    "        # dict_report(stats_large, 'Accuracy QPTH Backward'+f' ndim:{ndim}'+f' neq=nineq={neq}', acc_qpth_bb)\n",
    "    try:\n",
    "        pd.DataFrame(stats_large).to_csv('./Large_scale_qp.csv')\n",
    "    except:\n",
    "        pd.Series(stats_large).to_csv('./Large_scale_qp.csv')"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n",
     "is_executing": true
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "outputs": [],
   "source": [
    "def get_results_table(results_dict):\n",
    "    d = {}\n",
    "    missing_methods = []\n",
    "    for method in results_dict.keys():\n",
    "        if method in results_dict:\n",
    "            d[method] = ['{:.1e}({:.1e})'.format(np.nanmean(results_dict[method]),np.nanstd(results_dict[method]))]\n",
    "        else:\n",
    "            missing_methods.append(method)\n",
    "    df = pd.DataFrame.from_dict(d, orient='index')\n",
    "    df.index.names = ['avg']\n",
    "    return df"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "outputs": [
    {
     "data": {
      "text/plain": "                                                                 0\navg                                                               \nTime OSQP Forward ndim:10 neq=nineq=5             1.4e-04(7.9e-05)\nTime OSQP Backward ndim:10 neq=nineq=5            5.1e-05(6.2e-06)\nTime CVXPY Backward ndim:10 neq=nineq=5                   nan(nan)\nTime Exact Backward ndim:10 neq=nineq=5           7.0e-03(5.4e-03)\nTime Qpth Backward ndim:10 neq=nineq=5            2.6e-03(6.7e-04)\nTime Qpth Forward ndim:10 neq=nineq=5             1.1e-01(1.9e-02)\nTime Alt-diff Inverse ndim:10 neq=nineq=5         3.7e-04(9.8e-04)\nTime Alt-diff solution ndim:10 neq=nineq=5        3.8e-02(8.8e-02)\nAccuracy Forward ndim:10 neq=nineq=5              9.5e-01(1.9e-01)\nAccuracy Alt-diff Forward ndim:10 neq=nineq=5     9.3e-01(2.8e-01)\nAccuracy OSQP Backward ndim:10 neq=nineq=5        9.7e-01(2.0e-01)\nAccuracy CVXPY Backward ndim:10 neq=nineq=5               nan(nan)\nAccuracy QPTH Backward ndim:10 neq=nineq=5        9.6e-01(1.9e-01)\nTime OSQP Forward ndim:50 neq=nineq=10            6.8e-04(6.2e-05)\nTime OSQP Backward ndim:50 neq=nineq=10           3.9e-04(8.5e-04)\nTime CVXPY Backward ndim:50 neq=nineq=10                  nan(nan)\nTime Exact Backward ndim:50 neq=nineq=10          7.4e-02(2.9e-02)\nTime Qpth Backward ndim:50 neq=nineq=10           2.5e-03(4.4e-04)\nTime Qpth Forward ndim:50 neq=nineq=10            1.1e-01(1.3e-02)\nTime Alt-diff Inverse ndim:50 neq=nineq=10        2.6e-02(6.1e-03)\nTime Alt-diff solution ndim:50 neq=nineq=10       1.3e-02(8.9e-03)\nAccuracy Forward ndim:50 neq=nineq=10             1.0e+00(7.7e-07)\nAccuracy Alt-diff Forward ndim:50 neq=nineq=10    1.0e+00(1.1e-02)\nAccuracy OSQP Backward ndim:50 neq=nineq=10       1.0e+00(4.1e-07)\nAccuracy CVXPY Backward ndim:50 neq=nineq=10              nan(nan)\nAccuracy QPTH Backward ndim:50 neq=nineq=10       1.0e+00(1.1e-04)\nTime OSQP Forward ndim:100 neq=nineq=20           4.5e-03(4.5e-03)\nTime OSQP Backward ndim:100 neq=nineq=20          1.3e-03(2.4e-03)\nTime CVXPY Backward ndim:100 neq=nineq=20                 nan(nan)\nTime Exact Backward ndim:100 neq=nineq=20         4.7e-01(6.2e-02)\nTime Qpth Backward ndim:100 neq=nineq=20          2.7e-03(3.7e-04)\nTime Qpth Forward ndim:100 neq=nineq=20           1.3e-01(1.7e-02)\nTime Alt-diff Inverse ndim:100 neq=nineq=20       2.4e-01(4.0e-02)\nTime Alt-diff solution ndim:100 neq=nineq=20      9.8e-02(7.8e-02)\nAccuracy Forward ndim:100 neq=nineq=20            1.0e+00(4.5e-07)\nAccuracy Alt-diff Forward ndim:100 neq=nineq=20   1.0e+00(1.3e-02)\nAccuracy OSQP Backward ndim:100 neq=nineq=20      1.0e+00(2.1e-09)\nAccuracy CVXPY Backward ndim:100 neq=nineq=20             nan(nan)\nAccuracy QPTH Backward ndim:100 neq=nineq=20      1.0e+00(4.6e-03)\nTime OSQP Forward ndim:500 neq=nineq=100          5.3e-02(2.3e-02)\nTime OSQP Backward ndim:500 neq=nineq=100         1.1e-02(6.0e-04)\nTime CVXPY Backward ndim:500 neq=nineq=100                nan(nan)\nTime Exact Backward ndim:500 neq=nineq=100        3.3e+00(1.4e-01)\nTime Qpth Backward ndim:500 neq=nineq=100                 nan(nan)\nTime Qpth Forward ndim:500 neq=nineq=100                  nan(nan)\nTime Alt-diff Inverse ndim:500 neq=nineq=100      2.4e+00(1.3e-01)\nTime Alt-diff solution ndim:500 neq=nineq=100     6.5e-01(4.0e-01)\nAccuracy Forward ndim:500 neq=nineq=100                   nan(nan)\nAccuracy Alt-diff Forward ndim:500 neq=nineq=100  1.0e+00(2.2e-03)\nAccuracy OSQP Backward ndim:500 neq=nineq=100     1.0e+00(2.1e-07)\nAccuracy CVXPY Backward ndim:500 neq=nineq=100            nan(nan)\nAccuracy QPTH Backward ndim:500 neq=nineq=100             nan(nan)",
      "text/html": "<div>\n<style scoped>\n    .dataframe tbody tr th:only-of-type {\n        vertical-align: middle;\n    }\n\n    .dataframe tbody tr th {\n        vertical-align: top;\n    }\n\n    .dataframe thead th {\n        text-align: right;\n    }\n</style>\n<table border=\"1\" class=\"dataframe\">\n  <thead>\n    <tr style=\"text-align: right;\">\n      <th></th>\n      <th>0</th>\n    </tr>\n    <tr>\n      <th>avg</th>\n      <th></th>\n    </tr>\n  </thead>\n  <tbody>\n    <tr>\n      <th>Time OSQP Forward ndim:10 neq=nineq=5</th>\n      <td>1.4e-04(7.9e-05)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Backward ndim:10 neq=nineq=5</th>\n      <td>5.1e-05(6.2e-06)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Backward ndim:10 neq=nineq=5</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Time Exact Backward ndim:10 neq=nineq=5</th>\n      <td>7.0e-03(5.4e-03)</td>\n    </tr>\n    <tr>\n      <th>Time Qpth Backward ndim:10 neq=nineq=5</th>\n      <td>2.6e-03(6.7e-04)</td>\n    </tr>\n    <tr>\n      <th>Time Qpth Forward ndim:10 neq=nineq=5</th>\n      <td>1.1e-01(1.9e-02)</td>\n    </tr>\n    <tr>\n      <th>Time Alt-diff Inverse ndim:10 neq=nineq=5</th>\n      <td>3.7e-04(9.8e-04)</td>\n    </tr>\n    <tr>\n      <th>Time Alt-diff solution ndim:10 neq=nineq=5</th>\n      <td>3.8e-02(8.8e-02)</td>\n    </tr>\n    <tr>\n      <th>Accuracy Forward ndim:10 neq=nineq=5</th>\n      <td>9.5e-01(1.9e-01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy Alt-diff Forward ndim:10 neq=nineq=5</th>\n      <td>9.3e-01(2.8e-01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Backward ndim:10 neq=nineq=5</th>\n      <td>9.7e-01(2.0e-01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Backward ndim:10 neq=nineq=5</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Backward ndim:10 neq=nineq=5</th>\n      <td>9.6e-01(1.9e-01)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Forward ndim:50 neq=nineq=10</th>\n      <td>6.8e-04(6.2e-05)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Backward ndim:50 neq=nineq=10</th>\n      <td>3.9e-04(8.5e-04)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Backward ndim:50 neq=nineq=10</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Time Exact Backward ndim:50 neq=nineq=10</th>\n      <td>7.4e-02(2.9e-02)</td>\n    </tr>\n    <tr>\n      <th>Time Qpth Backward ndim:50 neq=nineq=10</th>\n      <td>2.5e-03(4.4e-04)</td>\n    </tr>\n    <tr>\n      <th>Time Qpth Forward ndim:50 neq=nineq=10</th>\n      <td>1.1e-01(1.3e-02)</td>\n    </tr>\n    <tr>\n      <th>Time Alt-diff Inverse ndim:50 neq=nineq=10</th>\n      <td>2.6e-02(6.1e-03)</td>\n    </tr>\n    <tr>\n      <th>Time Alt-diff solution ndim:50 neq=nineq=10</th>\n      <td>1.3e-02(8.9e-03)</td>\n    </tr>\n    <tr>\n      <th>Accuracy Forward ndim:50 neq=nineq=10</th>\n      <td>1.0e+00(7.7e-07)</td>\n    </tr>\n    <tr>\n      <th>Accuracy Alt-diff Forward ndim:50 neq=nineq=10</th>\n      <td>1.0e+00(1.1e-02)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Backward ndim:50 neq=nineq=10</th>\n      <td>1.0e+00(4.1e-07)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Backward ndim:50 neq=nineq=10</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Backward ndim:50 neq=nineq=10</th>\n      <td>1.0e+00(1.1e-04)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Forward ndim:100 neq=nineq=20</th>\n      <td>4.5e-03(4.5e-03)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Backward ndim:100 neq=nineq=20</th>\n      <td>1.3e-03(2.4e-03)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Backward ndim:100 neq=nineq=20</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Time Exact Backward ndim:100 neq=nineq=20</th>\n      <td>4.7e-01(6.2e-02)</td>\n    </tr>\n    <tr>\n      <th>Time Qpth Backward ndim:100 neq=nineq=20</th>\n      <td>2.7e-03(3.7e-04)</td>\n    </tr>\n    <tr>\n      <th>Time Qpth Forward ndim:100 neq=nineq=20</th>\n      <td>1.3e-01(1.7e-02)</td>\n    </tr>\n    <tr>\n      <th>Time Alt-diff Inverse ndim:100 neq=nineq=20</th>\n      <td>2.4e-01(4.0e-02)</td>\n    </tr>\n    <tr>\n      <th>Time Alt-diff solution ndim:100 neq=nineq=20</th>\n      <td>9.8e-02(7.8e-02)</td>\n    </tr>\n    <tr>\n      <th>Accuracy Forward ndim:100 neq=nineq=20</th>\n      <td>1.0e+00(4.5e-07)</td>\n    </tr>\n    <tr>\n      <th>Accuracy Alt-diff Forward ndim:100 neq=nineq=20</th>\n      <td>1.0e+00(1.3e-02)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Backward ndim:100 neq=nineq=20</th>\n      <td>1.0e+00(2.1e-09)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Backward ndim:100 neq=nineq=20</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Backward ndim:100 neq=nineq=20</th>\n      <td>1.0e+00(4.6e-03)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Forward ndim:500 neq=nineq=100</th>\n      <td>5.3e-02(2.3e-02)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Backward ndim:500 neq=nineq=100</th>\n      <td>1.1e-02(6.0e-04)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Backward ndim:500 neq=nineq=100</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Time Exact Backward ndim:500 neq=nineq=100</th>\n      <td>3.3e+00(1.4e-01)</td>\n    </tr>\n    <tr>\n      <th>Time Qpth Backward ndim:500 neq=nineq=100</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Time Qpth Forward ndim:500 neq=nineq=100</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Time Alt-diff Inverse ndim:500 neq=nineq=100</th>\n      <td>2.4e+00(1.3e-01)</td>\n    </tr>\n    <tr>\n      <th>Time Alt-diff solution ndim:500 neq=nineq=100</th>\n      <td>6.5e-01(4.0e-01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy Forward ndim:500 neq=nineq=100</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Accuracy Alt-diff Forward ndim:500 neq=nineq=100</th>\n      <td>1.0e+00(2.2e-03)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Backward ndim:500 neq=nineq=100</th>\n      <td>1.0e+00(2.1e-07)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Backward ndim:500 neq=nineq=100</th>\n      <td>nan(nan)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Backward ndim:500 neq=nineq=100</th>\n      <td>nan(nan)</td>\n    </tr>\n  </tbody>\n</table>\n</div>"
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df = get_results_table(stats)\n",
    "df"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "outputs": [],
   "source": [
    "pd.DataFrame(stats).to_csv('./analysis/BPQP_QP_results.csv')"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "outputs": [
    {
     "data": {
      "text/plain": "Time OSQP Forward ndim:500 neq=nineq=200                0.045332\nTime OSQP Backward ndim:500 neq=nineq=200               0.018485\nTime Exact Backward ndim:500 neq=nineq=200              4.180052\nTime Alt-diff Inverse ndim:500 neq=nineq=200            2.563820\nTime Alt-diff solution ndim:500 neq=nineq=200           5.055840\nAccuracy Alt-diff Forward ndim:500 neq=nineq=200        0.982949\nAccuracy OSQP Backward ndim:500 neq=nineq=200           1.000000\nTime OSQP Forward ndim:1500 neq=nineq=500               0.983984\nTime OSQP Backward ndim:1500 neq=nineq=500              0.157939\nTime Exact Backward ndim:1500 neq=nineq=500             6.957500\nTime Alt-diff Inverse ndim:1500 neq=nineq=500           6.225643\nTime Alt-diff solution ndim:1500 neq=nineq=500         15.299222\nAccuracy Alt-diff Forward ndim:1500 neq=nineq=500       0.998509\nAccuracy OSQP Backward ndim:1500 neq=nineq=500          1.000000\nTime OSQP Forward ndim:5000 neq=nineq=2000             24.900213\nTime OSQP Backward ndim:5000 neq=nineq=2000             2.299988\nTime Exact Backward ndim:5000 neq=nineq=2000           18.854699\nTime Alt-diff Inverse ndim:5000 neq=nineq=2000         12.589617\nTime Alt-diff solution ndim:5000 neq=nineq=2000       317.574720\nAccuracy Alt-diff Forward ndim:5000 neq=nineq=2000      0.999317\nAccuracy OSQP Backward ndim:5000 neq=nineq=2000         1.000000\ndtype: float64"
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dfl = get_results_table(stats_large)\n",
    "dfl#.to_csv('./analysis/Large_scale_qp.csv')\n",
    "pd.Series(stats_large)#.to_csv('./analysis/Large_scale_qp.csv')"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "markdown",
   "source": [
    "# 9. Large scale LPs Performance"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%% md\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "outputs": [],
   "source": [
    "def LP_instances(ndim,nineq,delta):\n",
    "    neq = nineq\n",
    "    s0 = np.random.randn(nineq)\n",
    "    lamb0 = np.maximum(-s0, 0)\n",
    "    s0 = np.maximum(s0, 0)\n",
    "    x0 = np.random.random(ndim) # one feasible solution\n",
    "\n",
    "    G = np.random.random((nineq, ndim))\n",
    "    A = np.random.random((neq, ndim))\n",
    "    b = A@x0\n",
    "    h = G @ x0 + s0\n",
    "    c = -G.T @ lamb0\n",
    "    delta = 1e-6\n",
    "    P = np.eye(ndim,ndim)*delta\n",
    "    return P,c,b,A,G,h\n",
    "\n",
    "def LP_cvxpy_backward(P,c,A,b,G,h):\n",
    "    nineq,ndim = A.shape\n",
    "    qq = cp.Parameter(ndim)\n",
    "    qq.value = c\n",
    "    x = cp.Variable(ndim)\n",
    "\n",
    "    prob = cp.Problem(cp.Minimize(qq.T@x),\n",
    "                 [A @ x == b,\n",
    "                  G @ x <= h\n",
    "                  ])\n",
    "    t3 = time.time()\n",
    "\n",
    "    prob.solve(requires_grad=True, solver='SCS')\n",
    "    time_spent_forward = time.time() - t3\n",
    "    t4 = time.time()\n",
    "    prob.backward()\n",
    "    time_spent_backward = time.time() - t4\n",
    "    return x.value,prob.value, qq.gradient,time_spent_forward,time_spent_backward\n",
    "\n",
    "def LP_OSQP_forward(P,G,A,b,h):\n",
    "    nineq,ndim = G.shape\n",
    "    neq = A.shape[0]\n",
    "    Pm = sparse.csc_matrix(P)\n",
    "    osA = np.vstack([G,A])\n",
    "    osA = sparse.csc_matrix(osA)\n",
    "    l = np.hstack([-np.inf*np.ones(nineq),b])\n",
    "    u = np.hstack([h,b])\n",
    "    x_value, y_value, time_spent = osqp_interface(Pm,c,osA,l,u)\n",
    "\n",
    "    return x_value,y_value, time_spent, np.dot(c,x_value)\n",
    "\n",
    "def LP_OSQP_backward(y_value,P,G,A):\n",
    "    nineq,ndim = G.shape\n",
    "    neq = A.shape[0]\n",
    "    lambs = y_value[:nineq] # active set\n",
    "    active_set = np.argwhere(lambs>1e-8)\n",
    "    bG = G[active_set,:].squeeze()\n",
    "    bb = np.zeros(neq)\n",
    "    bh = np.zeros(len(active_set))\n",
    "    bq = np.ones(ndim)\n",
    "    osnewA = np.vstack([bG,A])\n",
    "    osnewA = sparse.csc_matrix(osnewA)\n",
    "    l_new = np.hstack([bh,bb])\n",
    "    u_new = np.hstack([bh,bb])\n",
    "    x_grad, y_grad, time_spent_backward = osqp_interface(P,bq,osnewA,l_new,u_new)\n",
    "    return x_grad, y_grad, time_spent_backward\n",
    "\n",
    "def LP_qpth_evaluate(Pm, G, A, h, b, q):\n",
    "    nineq, ndim = G.shape\n",
    "    neq = A.shape[0]\n",
    "    qpf = QPFunction(verbose=0, maxIter=4000)\n",
    "    tP, tq, tG, th, tA, tb = [torch.Tensor(x).unsqueeze(0).cuda() for x in [Pm, q, G, h, A, b]]\n",
    "    tq.requires_grad = True\n",
    "\n",
    "    t6 = time.time()\n",
    "    qpth_x_value = qpf(tP, tq, tG, th, tA, tb)\n",
    "    t7 = time.time()\n",
    "    qpth_x_value.backward(torch.ones(1, ndim).cuda())\n",
    "    qpth_x_grad = tq.grad.squeeze().cpu().numpy()\n",
    "    qpth_x_value = qpth_x_value.squeeze().detach().cpu().numpy()\n",
    "    return qpth_x_value, qpth_x_grad, time.time() - t6, time.time() - t7\n",
    "\n",
    "def LP_cal_exact_backward(Pm,G,A,y_value,x_value,h):\n",
    "    nineq,ndim = G.shape\n",
    "    neq = A.shape[0]\n",
    "    lambs = y_value[:nineq] # active set\n",
    "    KKT_L1 = np.hstack([Pm,G.T,A.T])\n",
    "    KKT_L2 = np.hstack([np.diag(lambs)@G, np.diag(G@x_value-h),np.zeros((nineq,neq))])\n",
    "    KKT_L3 = np.hstack([A, np.zeros((neq,neq)),np.zeros((neq,nineq))])\n",
    "    KKT = np.vstack([KKT_L1,KKT_L2,KKT_L3])\n",
    "    t5 = time.time()\n",
    "    exact_bb =-(np.linalg.inv(KKT)@np.hstack([np.ones(ndim),np.zeros(nineq),np.zeros(neq)]))[:ndim]\n",
    "    return exact_bb,time.time()-t5"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 17%|██████████████████▏                                                                                        | 34/200 [02:20<11:52,  4.29s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 18%|███████████████████▊                                                                                       | 37/200 [02:34<12:03,  4.44s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 30%|████████████████████████████████                                                                           | 60/200 [04:17<10:12,  4.37s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 34%|███████████████████████████████████▊                                                                       | 67/200 [04:49<09:52,  4.45s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 38%|█████████████████████████████████████████▏                                                                 | 77/200 [05:35<09:39,  4.71s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|███████████████████████████████████████████▎                                                               | 81/200 [05:53<08:57,  4.52s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 43%|██████████████████████████████████████████████                                                             | 86/200 [06:16<08:38,  4.55s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|████████████████████████████████████████████████████▉                                                      | 99/200 [07:15<07:44,  4.59s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 50%|█████████████████████████████████████████████████████▌                                                    | 101/200 [07:24<07:25,  4.50s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 54%|████████████████████████████████████████████████████████▋                                                 | 107/200 [07:51<06:54,  4.46s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 54%|█████████████████████████████████████████████████████████▏                                                | 108/200 [07:55<06:54,  4.50s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 55%|█████████████████████████████████████████████████████████▊                                                | 109/200 [07:59<06:46,  4.47s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 56%|███████████████████████████████████████████████████████████▉                                              | 113/200 [08:18<06:34,  4.53s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 58%|██████████████████████████████████████████████████████████████                                            | 117/200 [08:36<06:16,  4.53s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 66%|██████████████████████████████████████████████████████████████████████▍                                   | 133/200 [09:48<05:02,  4.52s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 76%|████████████████████████████████████████████████████████████████████████████████▌                         | 152/200 [11:14<03:38,  4.54s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 82%|██████████████████████████████████████████████████████████████████████████████████████▉                   | 164/200 [12:08<02:42,  4.51s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 86%|███████████████████████████████████████████████████████████████████████████████████████████▋              | 173/200 [12:49<02:03,  4.57s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 87%|████████████████████████████████████████████████████████████████████████████████████████████▏             | 174/200 [12:53<01:57,  4.53s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 89%|██████████████████████████████████████████████████████████████████████████████████████████████▎           | 178/200 [13:11<01:37,  4.43s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 90%|███████████████████████████████████████████████████████████████████████████████████████████████▉          | 181/200 [13:24<01:24,  4.44s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 95%|████████████████████████████████████████████████████████████████████████████████████████████████████▋     | 190/200 [14:04<00:44,  4.42s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 96%|██████████████████████████████████████████████████████████████████████████████████████████████████████▎   | 193/200 [14:17<00:31,  4.50s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 98%|████████████████████████████████████████████████████████████████████████████████████████████████████████▍ | 197/200 [14:35<00:13,  4.46s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--------\n",
      "qpth warning: Returning an inaccurate and potentially incorrect solution.\n",
      "\n",
      "Some residual is large.\n",
      "Your problem may be infeasible or difficult.\n",
      "\n",
      "You can try using the CVXPY solver to see if your problem is feasible\n",
      "and you can use the verbose option to check the convergence status of\n",
      "our solver while increasing the number of iterations.\n",
      "\n",
      "Advanced users:\n",
      "You can also try to enable iterative refinement in the solver:\n",
      "https://github.com/locuslab/qpth/issues/6\n",
      "--------\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████████████████████████████████████████████████████████| 200/200 [14:48<00:00,  4.44s/it]\n"
     ]
    }
   ],
   "source": [
    "n_list = [10, 50, 100, 500]\n",
    "nconstraints_list = [5, 10, 20, 100]\n",
    "delta = 1e-6\n",
    "lp_stats = {}\n",
    "iters = 200\n",
    "for i in tqdm(range(iters)):\n",
    "    for ndim, neq in zip(n_list, nconstraints_list):\n",
    "        P,c,b,A,G,h = LP_instances(ndim, neq,delta)  # neq = nineq\n",
    "        Pm = sparse.csc_matrix(P)\n",
    "        x_value,y_value, time_spent, f_value = LP_OSQP_forward(Pm,G,A,b,h) # OSQP Forward\n",
    "\n",
    "        x_cp_value, y_cp_value, x_cp_grad, time_spent_forward, time_spent_backward = LP_cvxpy_backward(P,c,A,b,G,h)  # cvxpy Forward and Backward\n",
    "        x_grad, y_grad, time_spent_backward_osqp = LP_OSQP_backward(y_value,Pm,G,A)  # OSQP Backward\n",
    "        x_qpth_value, x_qpth_grad,time_spent_qpth_forward, time_qpth_backward=LP_qpth_evaluate(P, G, A, h, b, c)\n",
    "        exact_bb, t_spent = LP_cal_exact_backward(P,G,A,y_value,x_value,h)  # Exact backward\n",
    "\n",
    "        acc_osqp_bb = cal_cos_accuracy(exact_bb, x_grad)\n",
    "        acc_cvxpy_bb = cal_cos_accuracy(exact_bb, x_cp_grad)\n",
    "        acc_qpth_bb = cal_cos_accuracy(exact_bb, x_qpth_grad)\n",
    "\n",
    "        dict_report(lp_stats, 'Time OSQP Forward' + f' ndim:{ndim}' + f' neq=nineq={neq}', time_spent)\n",
    "        dict_report(lp_stats, 'Time OSQP Backward' + f' ndim:{ndim}' + f' neq=nineq={neq}', time_spent_backward_osqp)\n",
    "        dict_report(lp_stats, 'Time CVXPY Forward' + f' ndim:{ndim}' + f' neq=nineq={neq}', time_spent_forward)\n",
    "        dict_report(lp_stats, 'Time CVXPY Backward' + f' ndim:{ndim}' + f' neq=nineq={neq}', time_spent_backward)\n",
    "        dict_report(lp_stats, 'Time QPTH Forward' + f' ndim:{ndim}' + f' neq=nineq={neq}', time_spent_qpth_forward)\n",
    "        dict_report(lp_stats, 'Time QPTH Backward' + f' ndim:{ndim}' + f' neq=nineq={neq}', time_qpth_backward)\n",
    "        dict_report(lp_stats, 'Time Exact Backward' + f' ndim:{ndim}' + f' neq=nineq={neq}', t_spent)\n",
    "\n",
    "        dict_report(lp_stats, 'Accuracy OSQP Forward' + f' ndim:{ndim}' + f' neq=nineq={neq}', f_value)\n",
    "        dict_report(lp_stats, 'Accuracy CVXPY Forward' + f' ndim:{ndim}' + f' neq=nineq={neq}', np.dot(c,x_cp_value))\n",
    "        dict_report(lp_stats, 'Accuracy QPTH Forward' + f' ndim:{ndim}' + f' neq=nineq={neq}', np.dot(c,x_qpth_value))\n",
    "        dict_report(lp_stats, 'Accuracy OSQP Backward' + f' ndim:{ndim}' + f' neq=nineq={neq}', acc_osqp_bb)\n",
    "        dict_report(lp_stats, 'Accuracy CVXPY Backward' + f' ndim:{ndim}' + f' neq=nineq={neq}', acc_cvxpy_bb)\n",
    "        dict_report(lp_stats, 'Accuracy QPTH Backward' + f' ndim:{ndim}' + f' neq=nineq={neq}', acc_qpth_bb)\n",
    "    if i>1:\n",
    "        pd.DataFrame(lp_stats).to_csv('~/PONet/solver/BPQP_LP_raw_results(2).csv')\n"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "outputs": [
    {
     "data": {
      "text/plain": "                                                                0\navg                                                              \nTime OSQP Forward ndim:10 neq=nineq=5            1.1e-04(7.1e-05)\nTime OSQP Backward ndim:10 neq=nineq=5           1.3e-04(8.4e-04)\nTime CVXPY Forward ndim:10 neq=nineq=5           2.4e-02(7.7e-03)\nTime CVXPY Backward ndim:10 neq=nineq=5          4.3e-03(2.8e-03)\nTime QPTH Forward ndim:10 neq=nineq=5            1.1e-01(2.4e-02)\nTime QPTH Backward ndim:10 neq=nineq=5           3.9e-03(1.1e-03)\nTime Exact Backward ndim:10 neq=nineq=5          1.2e-03(2.0e-03)\nAccuracy OSQP Forward ndim:10 neq=nineq=5       -4.7e+00(3.6e+00)\nAccuracy CVXPY Forward ndim:10 neq=nineq=5      -4.7e+00(3.6e+00)\nAccuracy QPTH Forward ndim:10 neq=nineq=5       -4.7e+00(3.6e+00)\nAccuracy OSQP Backward ndim:10 neq=nineq=5       1.8e-02(2.4e-01)\nAccuracy CVXPY Backward ndim:10 neq=nineq=5      3.4e+05(1.8e+05)\nAccuracy QPTH Backward ndim:10 neq=nineq=5       2.3e+04(6.9e+04)\nTime OSQP Forward ndim:50 neq=nineq=10           3.9e-04(1.3e-04)\nTime OSQP Backward ndim:50 neq=nineq=10          1.3e-04(3.7e-05)\nTime CVXPY Forward ndim:50 neq=nineq=10          2.2e-02(5.6e-03)\nTime CVXPY Backward ndim:50 neq=nineq=10         3.7e-03(9.6e-04)\nTime QPTH Forward ndim:50 neq=nineq=10           1.0e-01(2.5e-02)\nTime QPTH Backward ndim:50 neq=nineq=10          3.7e-03(1.1e-03)\nTime Exact Backward ndim:50 neq=nineq=10         1.9e-02(1.2e-02)\nAccuracy OSQP Forward ndim:50 neq=nineq=10      -4.9e+01(2.4e+01)\nAccuracy CVXPY Forward ndim:50 neq=nineq=10     -4.9e+01(2.4e+01)\nAccuracy QPTH Forward ndim:50 neq=nineq=10      -4.9e+01(2.4e+01)\nAccuracy OSQP Backward ndim:50 neq=nineq=10      8.4e-04(3.9e-03)\nAccuracy CVXPY Backward ndim:50 neq=nineq=10     8.7e+05(1.2e+05)\nAccuracy QPTH Backward ndim:50 neq=nineq=10      1.2e+05(1.3e+05)\nTime OSQP Forward ndim:100 neq=nineq=20          2.2e-03(6.6e-04)\nTime OSQP Backward ndim:100 neq=nineq=20         5.8e-04(1.3e-03)\nTime CVXPY Forward ndim:100 neq=nineq=20         3.9e-02(1.2e-02)\nTime CVXPY Backward ndim:100 neq=nineq=20        6.1e-03(2.2e-03)\nTime QPTH Forward ndim:100 neq=nineq=20          1.1e-01(2.3e-02)\nTime QPTH Backward ndim:100 neq=nineq=20         4.0e-03(9.7e-04)\nTime Exact Backward ndim:100 neq=nineq=20        2.4e-01(3.6e-02)\nAccuracy OSQP Forward ndim:100 neq=nineq=20     -2.0e+02(6.2e+01)\nAccuracy CVXPY Forward ndim:100 neq=nineq=20    -2.0e+02(6.2e+01)\nAccuracy QPTH Forward ndim:100 neq=nineq=20     -2.0e+02(7.6e+01)\nAccuracy OSQP Backward ndim:100 neq=nineq=20     1.9e-03(4.4e-03)\nAccuracy CVXPY Backward ndim:100 neq=nineq=20    8.9e+05(8.4e+04)\nAccuracy QPTH Backward ndim:100 neq=nineq=20     2.1e+05(1.1e+05)\nTime OSQP Forward ndim:500 neq=nineq=100         7.0e-02(2.1e-02)\nTime OSQP Backward ndim:500 neq=nineq=100        4.8e-03(7.9e-04)\nTime CVXPY Forward ndim:500 neq=nineq=100        2.8e-01(2.0e-02)\nTime CVXPY Backward ndim:500 neq=nineq=100       2.6e-02(2.8e-03)\nTime QPTH Forward ndim:500 neq=nineq=100         2.4e-01(5.8e-02)\nTime QPTH Backward ndim:500 neq=nineq=100        5.9e-03(9.2e-04)\nTime Exact Backward ndim:500 neq=nineq=100       3.0e+00(1.3e-01)\nAccuracy OSQP Forward ndim:500 neq=nineq=100    -4.9e+03(6.9e+02)\nAccuracy CVXPY Forward ndim:500 neq=nineq=100   -4.9e+03(6.9e+02)\nAccuracy QPTH Forward ndim:500 neq=nineq=100    -4.6e+03(5.6e+03)\nAccuracy OSQP Backward ndim:500 neq=nineq=100    2.0e-04(1.2e-03)\nAccuracy CVXPY Backward ndim:500 neq=nineq=100   8.8e+05(4.0e+04)\nAccuracy QPTH Backward ndim:500 neq=nineq=100    3.0e+05(6.1e+04)",
      "text/html": "<div>\n<style scoped>\n    .dataframe tbody tr th:only-of-type {\n        vertical-align: middle;\n    }\n\n    .dataframe tbody tr th {\n        vertical-align: top;\n    }\n\n    .dataframe thead th {\n        text-align: right;\n    }\n</style>\n<table border=\"1\" class=\"dataframe\">\n  <thead>\n    <tr style=\"text-align: right;\">\n      <th></th>\n      <th>0</th>\n    </tr>\n    <tr>\n      <th>avg</th>\n      <th></th>\n    </tr>\n  </thead>\n  <tbody>\n    <tr>\n      <th>Time OSQP Forward ndim:10 neq=nineq=5</th>\n      <td>1.1e-04(7.1e-05)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Backward ndim:10 neq=nineq=5</th>\n      <td>1.3e-04(8.4e-04)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Forward ndim:10 neq=nineq=5</th>\n      <td>2.4e-02(7.7e-03)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Backward ndim:10 neq=nineq=5</th>\n      <td>4.3e-03(2.8e-03)</td>\n    </tr>\n    <tr>\n      <th>Time QPTH Forward ndim:10 neq=nineq=5</th>\n      <td>1.1e-01(2.4e-02)</td>\n    </tr>\n    <tr>\n      <th>Time QPTH Backward ndim:10 neq=nineq=5</th>\n      <td>3.9e-03(1.1e-03)</td>\n    </tr>\n    <tr>\n      <th>Time Exact Backward ndim:10 neq=nineq=5</th>\n      <td>1.2e-03(2.0e-03)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Forward ndim:10 neq=nineq=5</th>\n      <td>-4.7e+00(3.6e+00)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Forward ndim:10 neq=nineq=5</th>\n      <td>-4.7e+00(3.6e+00)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Forward ndim:10 neq=nineq=5</th>\n      <td>-4.7e+00(3.6e+00)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Backward ndim:10 neq=nineq=5</th>\n      <td>1.8e-02(2.4e-01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Backward ndim:10 neq=nineq=5</th>\n      <td>3.4e+05(1.8e+05)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Backward ndim:10 neq=nineq=5</th>\n      <td>2.3e+04(6.9e+04)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Forward ndim:50 neq=nineq=10</th>\n      <td>3.9e-04(1.3e-04)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Backward ndim:50 neq=nineq=10</th>\n      <td>1.3e-04(3.7e-05)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Forward ndim:50 neq=nineq=10</th>\n      <td>2.2e-02(5.6e-03)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Backward ndim:50 neq=nineq=10</th>\n      <td>3.7e-03(9.6e-04)</td>\n    </tr>\n    <tr>\n      <th>Time QPTH Forward ndim:50 neq=nineq=10</th>\n      <td>1.0e-01(2.5e-02)</td>\n    </tr>\n    <tr>\n      <th>Time QPTH Backward ndim:50 neq=nineq=10</th>\n      <td>3.7e-03(1.1e-03)</td>\n    </tr>\n    <tr>\n      <th>Time Exact Backward ndim:50 neq=nineq=10</th>\n      <td>1.9e-02(1.2e-02)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Forward ndim:50 neq=nineq=10</th>\n      <td>-4.9e+01(2.4e+01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Forward ndim:50 neq=nineq=10</th>\n      <td>-4.9e+01(2.4e+01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Forward ndim:50 neq=nineq=10</th>\n      <td>-4.9e+01(2.4e+01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Backward ndim:50 neq=nineq=10</th>\n      <td>8.4e-04(3.9e-03)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Backward ndim:50 neq=nineq=10</th>\n      <td>8.7e+05(1.2e+05)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Backward ndim:50 neq=nineq=10</th>\n      <td>1.2e+05(1.3e+05)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Forward ndim:100 neq=nineq=20</th>\n      <td>2.2e-03(6.6e-04)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Backward ndim:100 neq=nineq=20</th>\n      <td>5.8e-04(1.3e-03)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Forward ndim:100 neq=nineq=20</th>\n      <td>3.9e-02(1.2e-02)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Backward ndim:100 neq=nineq=20</th>\n      <td>6.1e-03(2.2e-03)</td>\n    </tr>\n    <tr>\n      <th>Time QPTH Forward ndim:100 neq=nineq=20</th>\n      <td>1.1e-01(2.3e-02)</td>\n    </tr>\n    <tr>\n      <th>Time QPTH Backward ndim:100 neq=nineq=20</th>\n      <td>4.0e-03(9.7e-04)</td>\n    </tr>\n    <tr>\n      <th>Time Exact Backward ndim:100 neq=nineq=20</th>\n      <td>2.4e-01(3.6e-02)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Forward ndim:100 neq=nineq=20</th>\n      <td>-2.0e+02(6.2e+01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Forward ndim:100 neq=nineq=20</th>\n      <td>-2.0e+02(6.2e+01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Forward ndim:100 neq=nineq=20</th>\n      <td>-2.0e+02(7.6e+01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Backward ndim:100 neq=nineq=20</th>\n      <td>1.9e-03(4.4e-03)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Backward ndim:100 neq=nineq=20</th>\n      <td>8.9e+05(8.4e+04)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Backward ndim:100 neq=nineq=20</th>\n      <td>2.1e+05(1.1e+05)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Forward ndim:500 neq=nineq=100</th>\n      <td>7.0e-02(2.1e-02)</td>\n    </tr>\n    <tr>\n      <th>Time OSQP Backward ndim:500 neq=nineq=100</th>\n      <td>4.8e-03(7.9e-04)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Forward ndim:500 neq=nineq=100</th>\n      <td>2.8e-01(2.0e-02)</td>\n    </tr>\n    <tr>\n      <th>Time CVXPY Backward ndim:500 neq=nineq=100</th>\n      <td>2.6e-02(2.8e-03)</td>\n    </tr>\n    <tr>\n      <th>Time QPTH Forward ndim:500 neq=nineq=100</th>\n      <td>2.4e-01(5.8e-02)</td>\n    </tr>\n    <tr>\n      <th>Time QPTH Backward ndim:500 neq=nineq=100</th>\n      <td>5.9e-03(9.2e-04)</td>\n    </tr>\n    <tr>\n      <th>Time Exact Backward ndim:500 neq=nineq=100</th>\n      <td>3.0e+00(1.3e-01)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Forward ndim:500 neq=nineq=100</th>\n      <td>-4.9e+03(6.9e+02)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Forward ndim:500 neq=nineq=100</th>\n      <td>-4.9e+03(6.9e+02)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Forward ndim:500 neq=nineq=100</th>\n      <td>-4.6e+03(5.6e+03)</td>\n    </tr>\n    <tr>\n      <th>Accuracy OSQP Backward ndim:500 neq=nineq=100</th>\n      <td>2.0e-04(1.2e-03)</td>\n    </tr>\n    <tr>\n      <th>Accuracy CVXPY Backward ndim:500 neq=nineq=100</th>\n      <td>8.8e+05(4.0e+04)</td>\n    </tr>\n    <tr>\n      <th>Accuracy QPTH Backward ndim:500 neq=nineq=100</th>\n      <td>3.0e+05(6.1e+04)</td>\n    </tr>\n  </tbody>\n</table>\n</div>"
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_lp = get_results_table(lp_stats)\n",
    "df_lp"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "outputs": [],
   "source": [
    "pd.DataFrame(lp_stats).to_csv('./analysis/BPQP_LP_results.csv')"
   ],
   "metadata": {
    "collapsed": false,
    "pycharm": {
     "name": "#%%\n"
    }
   }
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 2
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython2",
   "version": "2.7.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 0
}