{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "01a251af",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "env: CUDA_VISIBLE_DEVICES=7\n"
     ]
    }
   ],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "%env CUDA_VISIBLE_DEVICES=7"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "49bf1eeb",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch as t\n",
    "from torch import nn\n",
    "import numpy as np\n",
    "import scipy\n",
    "import scipy.special\n",
    "import scipy.optimize\n",
    "import scipy.stats\n",
    "from matplotlib import pyplot as plt\n",
    "from tqdm.auto import trange, tqdm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "6005f8da",
   "metadata": {},
   "outputs": [],
   "source": [
    "def gelu(x):\n",
    "    return .5 * x * (1 + scipy.special.erf(x / np.sqrt(2)))\n",
    "\n",
    "\n",
    "def gelu_deriv(x):\n",
    "    return .5 * scipy.special.erf(x / np.sqrt(2)) + x * np.exp(-x**2 / 2) / np.sqrt(2 * np.pi) + .5"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b0b11198",
   "metadata": {},
   "source": [
    "# Approximation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "88aebc90",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0 0 0 1 2 2 2 2 3]\n",
      "[0 1 2 1 0 0 1]\n"
     ]
    }
   ],
   "source": [
    "a = [0, 0, 1, 1, 0, 0, 0, 1]\n",
    "b = np.pad(np.cumsum(a), (1, 0))\n",
    "print(b)\n",
    "print(b[2:] - b[:-2])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 63,
   "id": "ed352c43",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "ccc234059ac34c78a9181a3fc7755cc1",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/128 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.41818033389117926\n",
      "32662 65295\n",
      "0.5002220690711386\n"
     ]
    }
   ],
   "source": [
    "def one_trial(total, in_a_row, p):\n",
    "    a = np.random.random(total) < p\n",
    "    b = np.pad(np.cumsum(a), (1, 0))\n",
    "    sum_in_a_row = b[in_a_row:] - b[:-in_a_row]\n",
    "        \n",
    "    streaks = np.where(sum_in_a_row[:-1] == in_a_row)[0]\n",
    "#     if len(streaks) > 0:\n",
    "#         print(sum_in_a_row)\n",
    "#         print(streaks, a[streaks + in_a_row], a)\n",
    "#         print()\n",
    "        \n",
    "    return a[streaks + in_a_row]\n",
    "\n",
    "\n",
    "BS = 2**10\n",
    "n_trials = 2**7 * BS\n",
    "\n",
    "IN_A_ROW = 2\n",
    "TOTAL = IN_A_ROW + 2\n",
    "\n",
    "streaks = []\n",
    "for _ in trange(n_trials // BS):\n",
    "    for _ in range(BS):\n",
    "        streaks.append(one_trial(TOTAL, IN_A_ROW, .5))\n",
    "    \n",
    "    \n",
    "a = [s.sum() / len(s) for s in streaks if len(s) > 0]\n",
    "print(np.mean(a))\n",
    "    \n",
    "streaks = np.concatenate(streaks)\n",
    "print(streaks.sum(), len(streaks))\n",
    "print(streaks.sum() / len(streaks))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d4725411",
   "metadata": {},
   "source": [
    "### Dynamic Programming"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "86fd2e8e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "6ce572a07eb848e3b9f33d5a1f34f074",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/32 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def calc_y_opt(int_w, int_f_w):\n",
    "    num = int_f_w[None, :] - int_f_w[:, None]\n",
    "\n",
    "    den = int_w[None, :] - int_w[:, None]\n",
    "    np.fill_diagonal(den, 1.)\n",
    "    den[den == 0] = 1.\n",
    "\n",
    "    return num / den\n",
    "\n",
    "\n",
    "def calc_trainsition_matrix(int_w, int_f2_w, y_opt):\n",
    "    translation_matrix = int_f2_w[None, :] - int_f2_w[:, None] - y_opt**2 * (int_w[None, :] - int_w[:, None])\n",
    "    translation_matrix = np.where(np.tri(len(int_w), k=-1, dtype=np.bool).T, translation_matrix, np.inf)\n",
    "    return translation_matrix\n",
    "    \n",
    "    \n",
    "def restore_answer(frm):\n",
    "    current = len(frm[-1]) - 1\n",
    "    points = [current]\n",
    "    for frm_curr in frm[::-1]:\n",
    "        current = frm_curr[current]\n",
    "        points.append(current)\n",
    "    points = points[::-1]\n",
    "    \n",
    "    assert points[0] == 0\n",
    "    assert points[-1] == len(frm[-1]) - 1\n",
    "    \n",
    "    return points\n",
    "\n",
    "\n",
    "def dynamic_programming(int_w, int_f_w, int_f2_w, xs, k: int):\n",
    "    y_opt = calc_y_opt(int_w, int_f_w)\n",
    "    transition_matrix = calc_trainsition_matrix(int_w, int_f2_w, y_opt)\n",
    "\n",
    "    def one_step(dp):\n",
    "        dp_new = dp[:, None] + transition_matrix\n",
    "        frm = np.argmin(dp_new, axis=0)\n",
    "        return np.min(dp_new, axis=0), frm\n",
    "\n",
    "    dp = np.full(len(int_w), np.inf)\n",
    "    dp[0] = 0\n",
    "    frm = []\n",
    "\n",
    "    for _ in trange(k):\n",
    "        dp, from_curr = one_step(dp)\n",
    "        frm.append(from_curr)\n",
    "\n",
    "    points = restore_answer(frm)\n",
    "    \n",
    "    return xs[points], y_opt[points[:-1], points[1:]]\n",
    "\n",
    "\n",
    "def numerical_integral(f_values, delta_x):\n",
    "    return np.cumsum(f_values) * delta_x\n",
    "\n",
    "\n",
    "def uniform_weight(n_segments, integral_n_coeff, L: float, R: float, n_bits: int, act, act_deriv):\n",
    "    xs = np.linspace(L, R, n_segments + 1)\n",
    "\n",
    "    int_w = xs - xs[0]\n",
    "    int_f_w = act(xs) - act(xs[0])\n",
    "\n",
    "    xs_int = np.linspace(L, R, (n_segments + 1) * (integral_n_coeff + 1) - integral_n_coeff)\n",
    "    int_f2_w = numerical_integral(act_deriv(xs_int)**2, xs_int[1] - xs_int[0])[::integral_n_coeff + 1]\n",
    "    \n",
    "    return dynamic_programming(int_w, int_f_w, int_f2_w, xs, 2**n_bits)\n",
    "\n",
    "\n",
    "def all_numerical(n_segments, integral_n_coeff, L: float, R: float, n_bits: int, act_deriv, w):\n",
    "    xs = np.linspace(L, R, n_segments + 1)\n",
    "    xs_int = np.linspace(L, R, (n_segments + 1) * (integral_n_coeff + 1) - integral_n_coeff)\n",
    "    delta_x = xs_int[1] - xs_int[0]\n",
    "    \n",
    "    int_w = numerical_integral(w(xs_int), delta_x)[::integral_n_coeff + 1]\n",
    "    int_f_w = numerical_integral(act_deriv(xs_int) * w(xs_int), delta_x)[::integral_n_coeff + 1]\n",
    "    int_f2_w = numerical_integral(act_deriv(xs_int)**2 * w(xs_int), delta_x)[::integral_n_coeff + 1]\n",
    "    \n",
    "    return dynamic_programming(int_w, int_f_w, int_f2_w, xs, 2**n_bits)\n",
    "\n",
    "\n",
    "N_SEGMENTS = 2**12\n",
    "INTEGRAL_N_COEFF = 2**10\n",
    "L = -10\n",
    "R = 10\n",
    "N_BITS = 5\n",
    "\n",
    "xs, ys = uniform_weight(N_SEGMENTS, INTEGRAL_N_COEFF, L, R, N_BITS, gelu, gelu_deriv)\n",
    "# xs, ys = all_numerical(N_SEGMENTS, INTEGRAL_N_COEFF, L, R, N_BITS, gelu_deriv, lambda x: scipy.stats.norm.pdf(x * .5))\n",
    "\n",
    "plt.plot(np.linspace(L, R, 2**10), gelu_deriv(np.linspace(L, R, 2**10)))\n",
    "for x1, x2, y in zip(xs[:-1], xs[1:], ys):\n",
    "    plt.plot([x1, x2], [y, y])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d01f2a20",
   "metadata": {},
   "source": [
    "### Gradient Descent"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 113,
   "id": "737bd9dd",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/gnovikov/data/miniconda3/envs/ml/lib/python3.9/site-packages/scipy/optimize/_constraints.py:432: OptimizeWarning: Equality and inequality constraints are specified in the same element of the constraint list. For efficient use with this method, equality and inequality constraints should be specified in separate elements of the constraint list. \n",
      "  warn(\"Equality and inequality constraints are specified in the same \"\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "N_POINTS = 2**10\n",
    "\n",
    "def error(xs, ys):\n",
    "    err = 0\n",
    "    for x1, x2, y in zip(xs[:-1], xs[1:], ys):\n",
    "        pts = np.linspace(x1, x2, N_POINTS)\n",
    "        err += ((gelu_deriv(pts) - y)**2 * scipy.stats.norm.pdf(pts)).mean() * (x2 - x1)\n",
    "    return err\n",
    "\n",
    "    \n",
    "def generate_x_constraints(n_segments: int, l: float, r: float):\n",
    "    ls, rs, As = [], [], []\n",
    "    \n",
    "    eye = np.eye(n_segments * 2 + 1)\n",
    "    \n",
    "    ls.append(l)\n",
    "    As.append(eye[0])\n",
    "    rs.append(l)\n",
    "    \n",
    "    for i in range(n_segments):\n",
    "        ls.append(0)\n",
    "        As.append(eye[i + 1] - eye[i])\n",
    "        rs.append(np.inf)\n",
    "    \n",
    "    ls.append(r)\n",
    "    As.append(eye[n_segments])\n",
    "    rs.append(r)\n",
    "    \n",
    "    return np.array(ls), np.array(As), np.array(rs)\n",
    "\n",
    "\n",
    "def struct_to_vector(xs, ys):\n",
    "    return np.concatenate([xs, ys])\n",
    "\n",
    "\n",
    "def vector_to_struct(x):\n",
    "    l = len(x)\n",
    "    n_segments = (l - 1) // 2\n",
    "    \n",
    "    return x[:n_segments + 1], x[n_segments + 1:]\n",
    "\n",
    "\n",
    "def generate_init(n_segments, l, r):\n",
    "    return np.linspace(l, r, n_segments + 1), np.zeros(n_segments)\n",
    "\n",
    "\n",
    "N_SEGMENTS = 8\n",
    "L = -10\n",
    "R = 10\n",
    "\n",
    "\n",
    "pts = np.linspace(L, R, 2**10)\n",
    "plt.plot(pts, gelu(pts))\n",
    "plt.show()\n",
    "plt.plot(pts, gelu_deriv(pts))\n",
    "plt.show()\n",
    "\n",
    "\n",
    "x0 = struct_to_vector(*generate_init(N_SEGMENTS, L, R))\n",
    "lb, A, ub = generate_x_constraints(N_SEGMENTS, L, R)\n",
    "\n",
    "\n",
    "x = scipy.optimize.minimize(\n",
    "    fun=lambda x: error(*vector_to_struct(x)),\n",
    "    x0=x0,\n",
    "    constraints=scipy.optimize.LinearConstraint(A, lb, ub),\n",
    ")\n",
    "\n",
    "xs, ys = vector_to_struct(x.x)\n",
    "for x1, x2, y in zip(xs[:-1], xs[1:], ys):\n",
    "    pts = np.linspace(x1, x2, 2**10)\n",
    "    plt.plot(pts, np.ones_like(pts) * y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 211,
   "id": "9579c1ec",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAWoAAAD4CAYAAADFAawfAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/MnkTPAAAACXBIWXMAAAsTAAALEwEAmpwYAAAeYklEQVR4nO3deXhV5d3u8e+PDIRAmAMoU0AZBRmSAmod0dapdagDKhyrVhQnHN5ah+qpPW3x1bbq0VbxWOupIIggtdYJ61htBQmEUUDmmYQxQMi4f+8fiS3QQHYgO2vtnftzXfsiO3tYN5Dce+1nP+tZ5u6IiEh4NQo6gIiIHJ6KWkQk5FTUIiIhp6IWEQk5FbWISMglx+JJ27Zt61lZWbF4ahGRhJSbm7vV3TOruy0mRZ2VlcXs2bNj8dQiIgnJzNYc6jYNfYiIhJyKWkQk5FTUIiIhp6IWEQk5FbWISMipqEVEQk5FLSIScipqEZE68OXq7bzw95XEYuloFbWIyFHaUljMLRPnMHHmWopKK+r8+WNyZKKISENRWh7hlolz2FtSzoQbhtK0cd3XqopaROQo/OKtxeSu2cEzVw+iV4eMmGxDQx8iIkdoWu56/vTPNdx4ajcuPPHYmG1HRS0icgQWbtjFA9MXMKx7a35ybu+YbktFLSJSSzuLSrl5Qi6tm6byzNWDSU6KbZVqjFpEpBYqIs4dk/PILyzh1ZuG0bZZ45hvU0UtIlILT/5tGZ8uK+BXl/RnUJdW9bJNDX2IiERpxqLNPP3hcq7M6cxVQzrX23ZV1CIiUVhZsId7pszjxE4teOSiEzCzett2jUVtZr3MLG+/S6GZ3VkP2UREQmFvSTk3vZxLSnIjnh2ZTVpKUr1uv8YxandfCgwEMLMkYAMwPbaxRETCwd25d9p8VhTs4eUbhtKxZZN6z1DboY/hwAp3P+RJGEVEEskLf1/FW/M3ce+5vTnl+LaBZKhtUY8AJlV3g5mNNrPZZja7oKDg6JOJiATsHyu2Mu6drzivXwduOq17YDmiLmozSwW+D7xW3e3u/ry757h7TmZmZl3lExEJxMad+7j9lbl0z2zG45cPqNcPDw9Wmz3q84A57r4lVmFERMKgpLyCMRPnUFIe4bmR2TSLwYp4tVGbrV/FIYY9REQSyc/+sph563by3Mhsjm/XLOg40e1Rm1lT4Bzg9djGEREJ1qtfrmXSrLXccsZxnNuvQ9BxgCj3qN19L9AmxllERAI1b91OHnpjEaf2aMs93+kVdJx/0ZGJIiLAtj0ljJmQS2azxjw1YhBJjYL78PBgWpRJRBq88ooId0yey7a9pUwbczKtm6YGHekAKmoRafAen7GUz5dv4/HLTqRfxxZBx/kPGvoQkQbtnQWbGP/JSkYO68LlOfW3Il5tqKhFpMH6estu/uu1eQzq0pKHLzwh6DiHpKIWkQZpd3EZN72cS5PUJJ69JpvU5PDWocaoRaTBiUSce6bMY832Iib+aCgdWqQFHemwwvsSIiISI89+soIZi7fwwPl9GNY9/IeIqKhFpEH5dFkBv5mxlO8NOJbrT8kKOk5UVNQi0mCs217EHZPn0qNdBv/9g/6BrohXGypqEWkQissqGDMxl4qIM35UNump8fMRXfwkFRE5Qu7OT/+8kIUbCvnDtTlktW0adKRa0R61iCS8iTPXMjV3PWOH92B4n/ZBx6k1FbWIJLTcNTt45M1FnNkrk7HDewQd54ioqEUkYRXsLuGWibkc06IJT145iEYhWhGvNjRGLSIJqawiwq2vzGHXvjKm3zKEFukpQUc6YipqEUlI495ewqxV23nyyoH0OaZ50HGOSrSn4mppZlPNbImZfWVmJ8U6mIjIkXojbwMvfr6KH56cxcWDOgYd56hFu0f9FPCuu19mZqlAegwziYgcsSWbC7lv2gKGZLXmwQv6BB2nTtRY1GbWAjgN+CGAu5cCpbGNJSJSe7v2Va6Il5GWzDPXDCIlKTHmS0Tzt+gGFAB/NLO5ZvZC1VnJD2Bmo81stpnNLigoqPOgIiKHE4k4d7+ax8ad+3h25GDaZYR7RbzaiKaok4HBwLPuPgjYC9x38J3c/Xl3z3H3nMzMzDqOKSJyeE9/uJwPluTz8IV9ye7aOug4dSqaol4PrHf3mVXXp1JZ3CIiofDRknye/GAZlw7uyMhhXYOOU+dqLGp33wysM7NeVd8aDiyOaSoRkSit2baXsZPn0qdDc351SfysiFcb0c76uB2YWDXjYyVwXewiiYhEZ19pBTe9nIuZMX5UNmkpSUFHiomoitrd84Cc2EYREYmeu3Pf6/NZumU3L103hM6tE3fWcGLMXRGRBuelf6zmjbyN3HNOT07vmdgTGFTUIhJ3Zq3azi/f+opz+rbnljOODzpOzKmoRSSubCks5paJc+jSOp3fXDEgblfEqw0tyiQicaO0PMKYCbkUlZYz6cahNE+L3xXxakNFLSJx4xdvLWbO2p387urB9GifEXSceqOhDxGJC9Ny1/Onf65h9GndueDEY4KOU69U1CISegs37OKB6Qs4qXsb7v1ur5ofkGBU1CISajv2lnLzhFxaN03l6asHkZwgK+LVhsaoRSS0KiLO2FfzyC8sYcrNJ9G2WeOgIwVCRS0iofXE+8v4dFkB4y7tz8DOLYOOE5iG9x5CROLCjEWbeeaj5Yz4VmeuGtIl6DiBUlGLSOisKNjD3VPmMaBTC372/ROCjhM4FbWIhMreknJufjmX1ORGPDsycVfEqw0VtYiEhrtz79T5rCjYwzNXDeLYlk2CjhQKKmoRCY3/9/eVvLVgEz85tzcnH9826DihoaIWkVD4x/KtPPrOEs7v34HRp3UPOk6oqKhFJHAbd+7jtklz6Z7ZjMcuG5CQp9M6GlHNozaz1cBuoAIod3ed7UVE6kRxWQVjJuRSWh5h/KhsmjXW4R0Hq82/yJnuvjVmSUSkQXrkzUXMW7+L8aOyOS6zWdBxQklDHyISmMmz1jJp1jpuPfM4vntCh6DjhFa0Re3ADDPLNbPR1d3BzEab2Wwzm11QUFB3CUUkIc1bt5OH31jEqT3acvc5DW9FvNqItqi/7e6DgfOAW83stIPv4O7Pu3uOu+dkZib2iSZF5Ohs21PCmAm5tGvemP87YhBJDeB0WkcjqqJ29w1Vf+YD04EhsQwlIomrvCLC7ZPmsm1vKc+NzKZV09SgI4VejUVtZk3NLOObr4HvAAtjHUxEEtPj7y3lHyu28ctL+tOvY4ug48SFaGZ9tAemV81rTAZecfd3Y5pKRBLSW/M3Mf7TlYwa1pXLsjsFHSdu1FjU7r4SGFAPWUQkgX29ZTc/njqPwV1a8tCFfYOOE1c0PU9EYq6wuIybXs4lPTWZ31+TTWqyqqc2dAiQiMRUJOL815R5rNlexCs/GkqHFmlBR4o7elkTkZh69pMVzFi8hQfP78PQ7m2CjhOXVNQiEjOfLivg1zOWctHAY7nulKyg48QtFbWIxMS67UXcMXkuvdpnMO7S/loR7yioqEWkzhWXVXDzhFwiEWf8qGzSU/Vx2NHQv56I1Cl358HpC1m0sZAXf5hD1zZNg44U97RHLSJ1asLMtUybs547z+7BWb3bBx0nIaioRaTO5K7Zwc/fXMRZvdtxx1k9go6TMFTUIlIn8ncXc8vEXI5t2YQnrhhII62IV2dU1CJy1MoqItw2cS679pXx3MhsWqSnBB0poejDRBE5auPeXsKs1dt5asRA+hzTPOg4CUd71CJyVN7I28CLn6/i+lO6cdHAjkHHSUgqahE5Yl9tKuQn0+YzpFtr7j+/d9BxEpaKWkSOyK59Zdw8IZcWTVJ45upBpCSpTmJFY9QiUmuRiHP3q3ls3LmPyaNPol2GVsSLpahfAs0syczmmtlfYxlIRMLv6Q+X88GSfB6+sC/ZXVsFHSfh1ea9yljgq1gFEZH48NHSfJ78YBmXDu7IyGFdg47TIERV1GbWCbgAeCG2cUQkzNZuK2LspLn06dCcX12iFfHqS7R71E8C9wKRQ93BzEab2Wwzm11QUFAX2UQkRPaVVnDThFzMjPGjsklLSQo6UoNRY1Gb2YVAvrvnHu5+7v68u+e4e05mZmadBRSR4FWuiLeAJZsLeWrEQDq3Tg86UoMSzR71KcD3zWw1MBk4y8wmxDSViITKy1+s4fW5G7jr7J6c0atd0HEanBqL2t3vd/dO7p4FjAA+dPeRMU8mIqGQu2Y7P39zMWf3acdtZx4fdJwGSTPUReSQ8ncXM2bCHDq1asJvtCJeYGp1wIu7fwx8HJMkIhIq36yIt7u4nD/dMIQWTbQiXlB0ZKKIVGv/FfF6d9CKeEHS0IeI/IdvVsS77pQsrYgXAipqETnAks2F3DdtAUOyWvPA+X2CjiOoqEVkP7v2lXHzy7lkpCXzzDVaES8sNEYtIkDlinj3TMlj/Y59TB49TCvihYheLkUEgN99tJy/fZXPTy/oQ05W66DjyH5U1CLCx0vz+e3flnHxwGO59uSsoOPIQVTUIg3cuu1FjJ2cR6/2GYy79EStiBdCKmqRBqy4rIKbXs7F3Rk/KpsmqVoRL4z0YaJIA1W5It5CFm8q5MUf5tC1TdOgI8khaI9apIGaMHMt0+asZ+zwHpzVu33QceQwVNQiDVDumh38/M1FnNkrk7HDewQdR2qgohZpYAp2l3DLxFyOadGEJ68cpBXx4oDGqEUakPKKCLe9Modd+8p4fcwQWqRrRbx4oKIWaUAefWcJM1dt54krB9D3WK2IFy809CHSQLw5byMvfLaKa0/qyiWDOgUdR2pBRS3SACzbspufTJtPdtdWPHhB36DjSC1FcxbyNDObZWbzzGyRmT1SH8FEpG4UFleuiNe0cTK/v2YwqcnaP4s30YxRlwBnufseM0sBPjOzd9z9ixhnE5Gj5O7c+9p81mwvYtKNw2jfXCvixaNozkLu7r6n6mpK1cVjmkpE6sQLf1/Fu4s2c/95vRnSTSvixauo3gOZWZKZ5QH5wPvuPrOa+4w2s9lmNrugoKCOY4pIbc1cuY1H313Cef06cMO3uwUdR45CVEXt7hXuPhDoBAwxs37V3Od5d89x95zMzMw6jikitZFfWMxtk+bStXU6j12mFfHiXa0+VXD3ncBHwLkxSSMiR628IsJtk+ayp7icZ0dmk5Gmg1riXTSzPjLNrGXV102Ac4AlMc4lIkfo8feWMmvVdsZd2p9eHTKCjiN1IJpZH8cA/9/Mkqgs9inu/tfYxhKRI/Huwk2M/3QlI4d14eJBHYOOI3WkxqJ29/nAoHrIIiJHYdXWvfz4tfkM6NSChy7UQS2JRDPfRRLAvtIKxkzIJTnJ+P3IbBon60wtiUSLMonEucoztSxg6ZbdvHTdEDq2bBJ0JKlj2qMWiXOvzFrL63M3MHZ4D07vqamxiUhFLRLH5q/fySN/WczpPTO54yydqSVRqahF4tSOvaWMmTCHzIzGPHnlQJ2pJYFpjFokDkUizl1T8ijYXcJrN59Eq6apQUeSGNIetUgceu7TFXy8tICHvteXAZ1bBh1HYkxFLRJnZq3azm9mLOPCE49h5NAuQceReqCiFokj2/aUcMekuXRu1YRxl/bXYksNhMaoReJEJOLcPWUe24tKeX3MyVpsqQHRHrVInHju0xV8sqyAhy/sS7+OLYKOI/VIRS0SB/Yfl75G49INjopaJOQ0Li0aoxYJMY1LC2iPWiTUNC4toKIWCS2NS8s3VNQiIbSzqJSxkzUuLZWiOWdiZzP7yMwWm9kiMxtbH8FEGip354HpC9i6p4SnrxqscWmJ6sPEcuAed59jZhlArpm97+6LY5xNpEGamruetxds5r7zetO/k8alJYo9anff5O5zqr7eDXwF6KyZIjGweutefvaXRQzr3pobT+0edBwJiVqNUZtZFpUnup1ZzW2jzWy2mc0uKCioo3giDUdZRYQ7X80jqZHx2ysGkqT1paVK1EVtZs2AacCd7l548O3u/ry757h7TmamTgckUltPf7icvHU7GXfpiRyr8x7KfqIqajNLobKkJ7r767GNJNLwzF69nWc+/JofDO7EBSceE3QcCZloZn0Y8AfgK3f/bewjiTQshcVl3PlqHp1apfOz7/cNOo6EUDR71KcAo4CzzCyv6nJ+jHOJNBj/+41FbNpVzBNXDtRUPKlWjdPz3P0zQJ9qiMTAG3kbmD53A3ed3ZPsrq2CjiMhpSMTRQKyfkcRP/3zQrK7tuLWM48LOo6EmIpaJAAVEefuV+fhDk9cMZDkJP0qyqFpmVORADz3yQpmrd7Ob68YQJc26UHHkZDTy7hIPZu3bidPvF+5Kt4lg3SQr9RMRS1Sj/aWlHPnq3m0y2jMLy/WqngSHQ19iNSjX7y1mNXb9jLpxmG0SNdUPImO9qhF6sm7CzczadY6bj79OIZ1bxN0HIkjKmqRerClsJj7Xp9P/44tuOvsnkHHkTijohaJsUjEuWfKPErKIjw5YiCpyfq1k9rRT4xIjL34+So+W76Vh7/Xl+MymwUdR+KQilokhhZvLOSxd5dyTt/2jPhW56DjSJxSUYvESHFZBWMnz6VFegr//YMTNRVPjpim54nEyKPvLOHr/D386fohtG6aGnQciWPaoxaJgY+W5PPSP1Zz/SndOK2nzngkR0dFLVLHtu4p4cdT59G7Qwb3ntsr6DiSADT0IVKH3J2fTJ1PYXE5E380jLSUpKAjSQLQHrVIHZowcy0fLMnngfN606tDRtBxJEFEc87EF80s38wW1kcgkXi1PH83v/jrYk7vmcm1J2cFHUcSSDR71C8B58Y4h0hcKymv4I5JeTRtnMzjl2sqntStGova3T8FttdDFpG49fi7S1m8qZDHfnAi7TLSgo4jCabOxqjNbLSZzTaz2QUFBXX1tCKh9/HSfF74bBXXntSVs/u2DzqOJKA6K2p3f97dc9w9JzNT80alYSjYXcJ/vTaPXu0zuP/8PkHHkQSl6XkiR8jd+fHUeezWVDyJMU3PEzlCf/x8NR8vLeCnF/TRVDyJqWim500C/gn0MrP1ZnZD7GOJhNvijYU8+s4Szu7TjpHDugYdRxJcjUMf7n5VfQQRiRd7S8q5fdIcWqan8NhlAzQVT2JOY9QiteDuPDB9Aau27mXCDUO1Kp7UC41Ri9TCxJlreSNvI3ef05OTj28bdBxpIFTUIlFasH4XP39zMWf0yuSWM44POo40ICpqkSjsKipjzMRc2jZL5YkrBtKokcalpf5ojFqkBhUR5+4peWwpLObVm06ilcalpZ5pj1qkBr+ZsZQPluTz0IV9GdylVdBxpAFSUYscxht5G/j9xyu4emgXRmm+tARERS1yCPPW7eTeqfMZ0q01P/veCZovLYFRUYtUY8POfYx+eTaZGY159prBpCbrV0WCow8TRQ6yY28po/4wk6LSCl67eQhtmjUOOpI0cCpqkf0UlZZz3Utfsn7HPl6+fgi9OzQPOpKIhj5EvlFSXsGtE+cwf/1Onr5qEEO7twk6kgigPWoRoLKkb5kwh4+WFjDu0v5894QOQUcS+RcVtTR4xWUVjJmQy0dLC/jlJf24akiXoCOJHEBFLQ1aYXEZYybk8vnybYy7tL9KWkJJRS0N1qZd+7juj1+yPH8Pv758AJdldwo6kki1ovow0czONbOlZrbczO6LdSiRWMtds52Lf/c563fs46XrhqikJdSiORVXEvA74DygL3CVmfWNdTCRWHB3Xvj7Sq4c/wVpKUlMHXMS3+6hdaUl3KIZ+hgCLHf3lQBmNhm4CFhc12HmrdtJxP1f1/c/ZHf/g3cPPpLX9rv1cEf57n/b4R4Tzf0O3syBz3GY5z7gMdX//WqT9ZCPieK5DSM1uVHlJakRKUmW0IdJr91WxAPTF/DZ8q1894T2PH75AJqnpQQdS6RG0RR1R2DdftfXA0NjEWbE81+wr6wiFk8tUUpNbkTjpEY0Tqks77SUJDLSkslIS6F5k2QyGlf9mZZC66aptMtoTLvmabTLaExmRmNSksI3Nb+4rII/fr6apz5YRnKjRvyfi/sxcmiXhH5RksRSZx8mmtloYDRAly5H9sn5+FHZVHyzR/3vHWt8vyv77XD/x3U/4PsH3vHA2w51y+Ger/o8tclw6Mcc5vmO4O93qH+7g58v4lBaXkFpRYTS8spLSUWEkrLIv75XXFbB7uJyCovL2FxYTOG+MnYXlx/yBbV101Q6tmxClzbpdG2dTtc26XRp3ZSubdLp0DytXhfcLy6rYGruep7+8Gu2FJZwdp92/Pyifhzbskm9ZRCpC9EU9Qag837XO1V97wDu/jzwPEBOTs6hG+owTuuZeSQPkwCUlkfYUVRKfmEJ+buLyd9dQn5hCVt2F7N+xz4WbdjFews3Ux75949C4+RGZLVpSlbbdLq1bUa3qj+z2qaT2axxnezhujuLNhYyfe4GpuauZ9e+MrK7tuKpEYMYpiMNJU5FU9RfAj3MrBuVBT0CuDqmqST0UpMb0b55Gu2bpwEtqr1PeUWETbuKWbOtiDXb97J6615WbS1ief4ePlyST1nFv0u8WeNkstqmk9WmKd3bNiWr6tKtTVNapqdUW+KRiLOjqJR1O/axcMMu5q/fyafLtrK5sJiUJOO7J3TgmqFdGda9tYY5JK7VWNTuXm5mtwHvAUnAi+6+KObJJO4lJzWic+t0OrdO59scOLOivCLCxp3FrNr2TYFXXuav38XbCzax3444SY2MZo2TyUhLJiWpEWUVEcoqImzfW3pA2bdMT2FYtzYM79OOs3q306p3kjCiGqN297eBt2OcRRqQ5KRGdGmTTpc26Zx+0JBXaXmEtduLWL11L6u37WVHUSl7isvZXVJOWYWT0shITjJaNU2lQ/M0jm3ZhL7HNKdTqybac5aEpCMTJXRSkxtxfLtmHN+uWdBRREIhfHOpRETkACpqEZGQU1GLiIScilpEJORU1CIiIaeiFhEJORW1iEjIqahFRELODrfC2xE/qVkBsOagb7cFttb5xupO2PNB+DOGPR8oY10Iez4If8bq8nV192pXpotJUVe7IbPZ7p5TLxs7AmHPB+HPGPZ8oIx1Iez5IPwZa5tPQx8iIiGnohYRCbn6LOrn63FbRyLs+SD8GcOeD5SxLoQ9H4Q/Y63y1dsYtYiIHBkNfYiIhJyKWkQk5OqtqM1soJl9YWZ5ZjbbzIbU17Zrw8xuN7MlZrbIzB4LOs+hmNk9ZuZm1rbme9cfM3u86t9vvplNN7OWQWcCMLNzzWypmS03s/uCznMwM+tsZh+Z2eKqn72xQWeqjpklmdlcM/tr0FmqY2YtzWxq1c/gV2Z2UtCZDmZmd1X9Hy80s0lmllbTY+pzj/ox4BF3Hwg8XHU9VMzsTOAiYIC7nwD8OuBI1TKzzsB3gLVBZ6nG+0A/dz8RWAbcH3AezCwJ+B1wHtAXuMrM+gab6j+UA/e4e19gGHBrCDMCjAW+CjrEYTwFvOvuvYEBhCyrmXUE7gBy3L0fleehHVHT4+qzqB1oXvV1C2BjPW47WmOAR929BMDd8wPOcyhPAPdS+W8aKu4+w93Lq65+AXQKMk+VIcByd1/p7qXAZCpfkEPD3Te5+5yqr3dTWTAdg011IDPrBFwAvBB0luqYWQvgNOAPAO5e6u47Aw1VvWSgiZklA+lE0YX1WdR3Ao+b2Toq91QD39OqRk/gVDObaWafmNm3gg50MDO7CNjg7vOCzhKF64F3gg5BZeGt2+/6ekJWgvszsyxgEDAz4CgHe5LKHYRIwDkOpRtQAPyxanjmBTNrGnSo/bn7Bir7by2wCdjl7jNqelydntzWzP4GdKjmpgeB4cBd7j7NzK6g8lXv7LrcfjRqyJgMtKbyree3gClm1t3reQ5jDRkfoHLYIzCHy+fub1Td50Eq385PrM9s8c7MmgHTgDvdvTDoPN8wswuBfHfPNbMzAo5zKMnAYOB2d59pZk8B9wEPBRvr38ysFZXv5roBO4HXzGyku0847APdvV4uwC7+PW/bgML62nYtMr4LnLnf9RVAZtC59svTH8gHVlddyql8Ze4QdLaDcv4Q+CeQHnSWqjwnAe/td/1+4P6gc1WTMwV4D7g76CzVZBtH5TuR1cBmoAiYEHSugzJ2AFbvd/1U4K2gcx2U8XLgD/td/1/A72t6XH0OfWwETq/6+izg63rcdrT+DJwJYGY9gVRCtAKXuy9w93bunuXuWVT+4gx2980BR/sXMzuXyrfH33f3oqDzVPkS6GFm3cwslcoPb/4ScKYDmJlR+S7zK3f/bdB5Dubu97t7p6qfuxHAh+4+MuBYB6j6PVhnZr2qvjUcWBxgpOqsBYaZWXrV//lwovjAs06HPmpwI/BU1QB6MTC6HrcdrReBF81sIVAKXOtVL3sStWeAxsD7lT+HfOHuNwcZyN3Lzew2KvdWk4AX3X1RkJmqcQowClhgZnlV33vA3d8OLlJcuh2YWPWCvBK4LuA8B/DKIZmpwBwq3xHPJYrDyXUIuYhIyOnIRBGRkFNRi4iEnIpaRCTkVNQiIiGnohYRCTkVtYhIyKmoRURC7n8A6O0oe16rq64AAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "N_POINTS = 2**15\n",
    "\n",
    "def gelu(x):\n",
    "    return .5 * x * (1 + scipy.special.erf(x / np.sqrt(2)))\n",
    "\n",
    "\n",
    "def gelu_deriv(x):\n",
    "    return .5 * scipy.special.erf(x / np.sqrt(2)) + x * np.exp(-x**2 / 2) / np.sqrt(2 * np.pi) + .5\n",
    "\n",
    "\n",
    "def get_y_from_x(xs):\n",
    "    ys = []\n",
    "    for x1, x2 in zip(xs[:-1], xs[1:]):\n",
    "        pts = np.linspace(x1, x2, N_POINTS)\n",
    "        ys.append(gelu_deriv(pts).mean())\n",
    "    return np.array(ys)\n",
    "\n",
    "        \n",
    "def error(xs):\n",
    "    err = 0\n",
    "    ys = get_y_from_x(xs)\n",
    "    for x1, x2, y in zip(xs[:-1], xs[1:], ys):\n",
    "        pts = np.linspace(x1, x2, N_POINTS)\n",
    "        err += ((gelu_deriv(pts) - y)**2).mean() * (x2 - x1)\n",
    "    return err\n",
    "\n",
    "    \n",
    "def generate_x_constraints(n_segments: int, l: float, r: float):\n",
    "    ls, rs, As = [], [], []\n",
    "    \n",
    "    eye = np.eye(n_segments + 1)\n",
    "    \n",
    "    ls.append(l)\n",
    "    As.append(eye[0])\n",
    "    rs.append(l)\n",
    "    \n",
    "    for i in range(n_segments):\n",
    "        ls.append(0)\n",
    "        As.append(eye[i + 1] - eye[i])\n",
    "        rs.append(np.inf)\n",
    "    \n",
    "    ls.append(r)\n",
    "    As.append(eye[n_segments])\n",
    "    rs.append(r)\n",
    "    \n",
    "    return np.array(ls), np.array(As), np.array(rs)\n",
    "\n",
    "\n",
    "def struct_to_vector(xs):\n",
    "    return xs\n",
    "\n",
    "\n",
    "def vector_to_struct(x):\n",
    "    return x\n",
    "\n",
    "\n",
    "def generate_init(n_segments, l, r):\n",
    "    return np.linspace(l, r, n_segments + 1)\n",
    "\n",
    "\n",
    "N_SEGMENTS = 4\n",
    "V = 7.5\n",
    "L = -V\n",
    "R = V\n",
    "\n",
    "\n",
    "x0 = struct_to_vector(generate_init(N_SEGMENTS, L, R))\n",
    "lb, A, ub = generate_x_constraints(N_SEGMENTS, L, R)\n",
    "\n",
    "\n",
    "x = scipy.optimize.minimize(\n",
    "    fun=lambda x: error(vector_to_struct(x)),\n",
    "    x0=x0,\n",
    "    constraints=scipy.optimize.LinearConstraint(A, lb, ub),\n",
    "    \n",
    ")\n",
    "\n",
    "pts = np.linspace(L, R, 2**10)\n",
    "plt.plot(pts, gelu(pts))\n",
    "plt.show()\n",
    "\n",
    "plt.plot(pts, gelu_deriv(pts))\n",
    "xs = vector_to_struct(x.x)\n",
    "ys = get_y_from_x(xs)\n",
    "for x1, x2, y in zip(xs[:-1], xs[1:], ys):\n",
    "    pts = np.linspace(x1, x2, 2**10)\n",
    "    plt.plot(pts, np.ones_like(pts) * y)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "329888ce",
   "metadata": {},
   "source": [
    "### Direct Grad Evaluation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "4f5f0f4e",
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "58fab2c226df470ab44113f998d30bd6",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/32 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/gnovikov/data/miniconda3/envs/ml/lib/python3.9/site-packages/scipy/optimize/_constraints.py:432: OptimizeWarning: Equality and inequality constraints are specified in the same element of the constraint list. For efficient use with this method, equality and inequality constraints should be specified in separate elements of the constraint list. \n",
      "  warn(\"Equality and inequality constraints are specified in the same \"\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f9fd1f320cb644d0973da140cbed3a9a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/32 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/gnovikov/data/miniconda3/envs/ml/lib/python3.9/site-packages/scipy/optimize/_constraints.py:432: OptimizeWarning: Equality and inequality constraints are specified in the same element of the constraint list. For efficient use with this method, equality and inequality constraints should be specified in separate elements of the constraint list. \n",
      "  warn(\"Equality and inequality constraints are specified in the same \"\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import random\n",
    "\n",
    "N_POINTS = 2**10\n",
    "\n",
    "def gelu(x):\n",
    "    return .5 * x * (1 + scipy.special.erf(x / np.sqrt(2)))\n",
    "\n",
    "\n",
    "def gelu_deriv(x):\n",
    "    return .5 * scipy.special.erf(x / np.sqrt(2)) + x * np.exp(-x**2 / 2) / np.sqrt(2 * np.pi) + .5\n",
    "\n",
    "\n",
    "def get_y_from_x(xs):\n",
    "    return (gelu(xs[1:]) - gelu(xs[:-1])) / (xs[1:] - xs[:-1])\n",
    "\n",
    "    ys = []\n",
    "    for x1, x2 in zip(xs[:-1], xs[1:]):\n",
    "        pts = np.linspace(x1, x2, N_POINTS)\n",
    "        ys.append((gelu_deriv(pts) * w(pts)).mean() / w(pts).mean())\n",
    "    return np.array(ys)\n",
    "\n",
    "\n",
    "def w(x):\n",
    "    return np.ones_like(x)\n",
    "#     return scipy.stats.norm.pdf(x)\n",
    "\n",
    "        \n",
    "def error(xs, n_pts: int = N_POINTS):\n",
    "    err = 0\n",
    "    ys = get_y_from_x(xs)\n",
    "    for x1, x2, y in zip(xs[:-1], xs[1:], ys):\n",
    "        pts = np.linspace(x1, x2, n_pts)\n",
    "        err += ((gelu_deriv(pts) - y)**2 * w(pts)).mean() * (x2 - x1)\n",
    "    return err\n",
    "\n",
    "\n",
    "def error_grad(xs):\n",
    "    ys = get_y_from_x(xs)\n",
    "     \n",
    "    grad = [0] + [\n",
    "        ((gelu_deriv(x) - y1)**2 - (gelu_deriv(x) - y2)**2)\n",
    "        for x, y1, y2 in zip(xs[1:-1], ys[:-1], ys[1:])\n",
    "    ] + [0]\n",
    "    \n",
    "    return np.array(grad)\n",
    "    \n",
    "    \n",
    "def generate_x_constraints(n_segments: int, l: float, r: float):\n",
    "    ls, rs, As = [], [], []\n",
    "    \n",
    "    eye = np.eye(n_segments + 1)\n",
    "    \n",
    "    ls.append(l)\n",
    "    As.append(eye[0])\n",
    "    rs.append(l)\n",
    "    \n",
    "    for i in range(n_segments):\n",
    "        ls.append(0)\n",
    "        As.append(eye[i + 1] - eye[i])\n",
    "        rs.append(np.inf)\n",
    "    \n",
    "    ls.append(r)\n",
    "    As.append(eye[n_segments])\n",
    "    rs.append(r)\n",
    "    \n",
    "    return np.array(ls), np.array(As), np.array(rs)\n",
    "\n",
    "\n",
    "def generate_init(n_segments, l, r):\n",
    "    return np.linspace(-2, 2, n_segments + 1)\n",
    "    return np.array([l] + [random.random() * (r - l) + l for _ in range(n_segments - 1)] + [r])\n",
    "\n",
    "\n",
    "xs_for_bits = []\n",
    "ys_for_bits = []\n",
    "\n",
    "for n_bits in range(1, 3):\n",
    "    N_SEGMENTS = 2**n_bits\n",
    "    V = 20\n",
    "    L = -V\n",
    "    R = +V\n",
    "\n",
    "\n",
    "    best_sol = None\n",
    "    for _ in trange(2**5):\n",
    "        x0 = generate_init(N_SEGMENTS, L, R)\n",
    "        lb, A, ub = generate_x_constraints(N_SEGMENTS, L, R)\n",
    "\n",
    "        x = scipy.optimize.minimize(\n",
    "            fun=error, \n",
    "            jac=error_grad,\n",
    "            x0=x0,\n",
    "            constraints=scipy.optimize.LinearConstraint(A, lb, ub),\n",
    "        )\n",
    "\n",
    "\n",
    "        if best_sol is None or error(best_sol, 2**15) > error(x.x, 2**15):\n",
    "            best_sol = x.x\n",
    "\n",
    "    xs = best_sol\n",
    "    ys = get_y_from_x(xs)\n",
    "\n",
    "    pts = np.linspace(L, R, 2**10)\n",
    "    plt.plot(pts, gelu_deriv(pts))\n",
    "    for x1, x2, y in zip(xs[:-1], xs[1:], ys):\n",
    "        pts = np.linspace(x1, x2, 2**10)\n",
    "        plt.plot(pts, np.ones_like(pts) * y)\n",
    "    plt.show()\n",
    "    \n",
    "    xs_for_bits.append(xs)\n",
    "    ys_for_bits.append(ys)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "11028c73",
   "metadata": {},
   "source": [
    "# Dynamic Programming as good initial state for further optimization"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "432f68cd",
   "metadata": {},
   "source": [
    "# Conversion to Python code"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "097a4773",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "    xs_const = [\n",
      "        t.tensor([0.00000000000000000]),\n",
      "        t.tensor([-0.46132802110453103,\n",
      "            -0.00000000000000055,\n",
      "            0.46132802110452981]),\n",
      "        t.tensor([-2.40884797702306042,\n",
      "            -0.71192497573751246,\n",
      "            -0.32633073511790534,\n",
      "            -0.00000000000000026,\n",
      "            0.32633073511790522,\n",
      "            0.71192497573751290,\n",
      "            2.40884797702305997]),\n",
      "        t.tensor([-2.64952682104609716,\n",
      "            -2.02781335198672830,\n",
      "            -0.92408609910716744,\n",
      "            -0.66777523353780222,\n",
      "            -0.46992418705007644,\n",
      "            -0.30085904432656912,\n",
      "            -0.14501492885208012,\n",
      "            0.00000000000000026,\n",
      "            0.14501492885207939,\n",
      "            0.30085904432656857,\n",
      "            0.46992418705007816,\n",
      "            0.66777523353780444,\n",
      "            0.92408609910717177,\n",
      "            2.02781335198672696,\n",
      "            2.64952682104609893]),\n",
      "        t.tensor([-2.92710030191747705,\n",
      "            -2.37891574922338567,\n",
      "            -1.98382220976643753,\n",
      "            -1.52832260384709029,\n",
      "            -1.36664201183946266,\n",
      "            -1.19128153542139237,\n",
      "            -0.97707449316668482,\n",
      "            -0.81655879956253341,\n",
      "            -0.68992865286588045,\n",
      "            -0.56706086359017105,\n",
      "            -0.46063400362679208,\n",
      "            -0.36083071867741179,\n",
      "            -0.27213466889115817,\n",
      "            -0.18725333960087970,\n",
      "            -0.09545320590096010,\n",
      "            -0.00000000000016468,\n",
      "            0.09545320590116181,\n",
      "            0.18725333960099158,\n",
      "            0.27213466889079202,\n",
      "            0.36083071867779953,\n",
      "            0.46063400362653045,\n",
      "            0.56706086359029162,\n",
      "            0.68992865286583815,\n",
      "            0.81655879956254707,\n",
      "            0.97707449316669170,\n",
      "            1.19128153542138793,\n",
      "            1.36664201183946377,\n",
      "            1.52832260384708984,\n",
      "            1.98382220976642665,\n",
      "            2.37891574922339410,\n",
      "            2.92710030191748238]),\n",
      "        t.tensor([-2.82908059050189387,\n",
      "            -2.26284818482951255,\n",
      "            -1.82685489973241766,\n",
      "            -1.75049771653586728,\n",
      "            -1.68804605157983634,\n",
      "            -1.62557643759734938,\n",
      "            -1.56303423098593863,\n",
      "            -1.50039423041699127,\n",
      "            -1.43763056102797160,\n",
      "            -1.37471847611981346,\n",
      "            -1.31163666119048195,\n",
      "            -1.24836995326919165,\n",
      "            -1.18491233070203483,\n",
      "            -1.12126991197296477,\n",
      "            -1.05746356803751462,\n",
      "            -0.99353063436294531,\n",
      "            -0.92952516171149446,\n",
      "            -0.86551623302573599,\n",
      "            -0.80158413899831715,\n",
      "            -0.73781462415744559,\n",
      "            -0.67429187959286629,\n",
      "            -0.61109130208778561,\n",
      "            -0.54827312030519804,\n",
      "            -0.48587777060813292,\n",
      "            -0.42392347923054063,\n",
      "            -0.36240603621418210,\n",
      "            -0.30130037583300529,\n",
      "            -0.24056338119248749,\n",
      "            -0.18013730508720732,\n",
      "            -0.11995329902345060,\n",
      "            -0.05993471089287655,\n",
      "            0.00000000000000023,\n",
      "            0.05993471089287550,\n",
      "            0.11995329902344819,\n",
      "            0.18013730508720646,\n",
      "            0.24056338119248902,\n",
      "            0.30130037583300756,\n",
      "            0.36240603621418088,\n",
      "            0.42392347923053325,\n",
      "            0.48587777060813098,\n",
      "            0.54827312030520670,\n",
      "            0.61109130208778950,\n",
      "            0.67429187959286507,\n",
      "            0.73781462415744126,\n",
      "            0.80158413899831116,\n",
      "            0.86551623302573999,\n",
      "            0.92952516171149524,\n",
      "            0.99353063436294486,\n",
      "            1.05746356803751618,\n",
      "            1.12126991197296499,\n",
      "            1.18491233070203217,\n",
      "            1.24836995326919209,\n",
      "            1.31163666119048283,\n",
      "            1.37471847611981302,\n",
      "            1.43763056102797049,\n",
      "            1.50039423041699171,\n",
      "            1.56303423098594063,\n",
      "            1.62557643759735071,\n",
      "            1.68804605157983922,\n",
      "            1.75049771653586461,\n",
      "            1.82685489973241588,\n",
      "            2.26284818482951300,\n",
      "            2.82908059050189475]),\n",
      "        t.tensor([-2.88528048414695926,\n",
      "            -2.32849632999631462,\n",
      "            -1.91753552506321467,\n",
      "            -1.87504313068165085,\n",
      "            -1.84379099310485439,\n",
      "            -1.81254936311919557,\n",
      "            -1.78130730712385787,\n",
      "            -1.75006454499142006,\n",
      "            -1.71882077022800850,\n",
      "            -1.68757564630236745,\n",
      "            -1.65632880888022860,\n",
      "            -1.62507986892908796,\n",
      "            -1.59382841673355569,\n",
      "            -1.56257402684728408,\n",
      "            -1.53131626398879894,\n",
      "            -1.50005468986750046,\n",
      "            -1.46878887090272481,\n",
      "            -1.43751838677328969,\n",
      "            -1.40624283970796338,\n",
      "            -1.37496186439904089,\n",
      "            -1.34367513839225872,\n",
      "            -1.31238239277768609,\n",
      "            -1.28108342297808231,\n",
      "            -1.24977809940541729,\n",
      "            -1.21846637773280175,\n",
      "            -1.18714830851012998,\n",
      "            -1.15582404583772758,\n",
      "            -1.12449385480492792,\n",
      "            -1.09315811740081514,\n",
      "            -1.06181733661348310,\n",
      "            -1.03047213845306129,\n",
      "            -0.99912327166296688,\n",
      "            -0.96777160492377334,\n",
      "            -0.93641812140432301,\n",
      "            -0.90506391057489488,\n",
      "            -0.87371015726572165,\n",
      "            -0.84235812802918786,\n",
      "            -0.81100915494337356,\n",
      "            -0.77966461707514034,\n",
      "            -0.74832591989959718,\n",
      "            -0.71699447304614849,\n",
      "            -0.68567166680602054,\n",
      "            -0.65435884788909637,\n",
      "            -0.62305729495662410,\n",
      "            -0.59176819447843099,\n",
      "            -0.56049261746823575,\n",
      "            -0.52923149763717003,\n",
      "            -0.49798561147569753,\n",
      "            -0.46675556072766844,\n",
      "            -0.43554175766067044,\n",
      "            -0.40434441346673067,\n",
      "            -0.37316353004905911,\n",
      "            -0.34199889536879641,\n",
      "            -0.31085008244234680,\n",
      "            -0.27971645199901091,\n",
      "            -0.24859715873170521,\n",
      "            -0.21749116100424748,\n",
      "            -0.18639723381580167,\n",
      "            -0.15531398477054340,\n",
      "            -0.12423987275554012,\n",
      "            -0.09317322899443597,\n",
      "            -0.06211228011611894,\n",
      "            -0.03105517285702154,\n",
      "            0.00000000000000004,\n",
      "            0.03105517285702139,\n",
      "            0.06211228011611803,\n",
      "            0.09317322899443668,\n",
      "            0.12423987275553956,\n",
      "            0.15531398477054290,\n",
      "            0.18639723381580456,\n",
      "            0.21749116100424717,\n",
      "            0.24859715873170263,\n",
      "            0.27971645199901546,\n",
      "            0.31085008244235057,\n",
      "            0.34199889536879102,\n",
      "            0.37316353004906105,\n",
      "            0.40434441346673899,\n",
      "            0.43554175766066527,\n",
      "            0.46675556072766483,\n",
      "            0.49798561147570342,\n",
      "            0.52923149763716848,\n",
      "            0.56049261746823231,\n",
      "            0.59176819447842566,\n",
      "            0.62305729495662154,\n",
      "            0.65435884788910181,\n",
      "            0.68567166680601865,\n",
      "            0.71699447304614883,\n",
      "            0.74832591989959074,\n",
      "            0.77966461707513235,\n",
      "            0.81100915494337822,\n",
      "            0.84235812802918753,\n",
      "            0.87371015726572698,\n",
      "            0.90506391057490554,\n",
      "            0.93641812140432235,\n",
      "            0.96777160492377357,\n",
      "            0.99912327166296488,\n",
      "            1.03047213845305397,\n",
      "            1.06181733661348288,\n",
      "            1.09315811740082225,\n",
      "            1.12449385480492769,\n",
      "            1.15582404583772447,\n",
      "            1.18714830851012998,\n",
      "            1.21846637773280375,\n",
      "            1.24977809940542017,\n",
      "            1.28108342297807853,\n",
      "            1.31238239277768542,\n",
      "            1.34367513839226094,\n",
      "            1.37496186439903934,\n",
      "            1.40624283970796471,\n",
      "            1.43751838677328969,\n",
      "            1.46878887090272525,\n",
      "            1.50005468986750046,\n",
      "            1.53131626398879805,\n",
      "            1.56257402684728186,\n",
      "            1.59382841673355613,\n",
      "            1.62507986892909018,\n",
      "            1.65632880888023148,\n",
      "            1.68757564630236989,\n",
      "            1.71882077022800828,\n",
      "            1.75006454499141495,\n",
      "            1.78130730712385676,\n",
      "            1.81254936311919868,\n",
      "            1.84379099310485528,\n",
      "            1.87504313068164885,\n",
      "            1.91753552506321157,\n",
      "            2.32849632999631195,\n",
      "            2.88528048414696370]),\n",
      "        t.tensor([-2.90212209091241302,\n",
      "            -2.35928671715313376,\n",
      "            -1.96430075965753348,\n",
      "            -1.93750744254844132,\n",
      "            -1.92187759368646516,\n",
      "            -1.90625314738377316,\n",
      "            -1.89062870899535063,\n",
      "            -1.87500427452834773,\n",
      "            -1.85937984089812525,\n",
      "            -1.84375540475908295,\n",
      "            -1.82813096250145790,\n",
      "            -1.81250651024946530,\n",
      "            -1.79688204386070516,\n",
      "            -1.78125755892688842,\n",
      "            -1.76563305077596233,\n",
      "            -1.75000851447576466,\n",
      "            -1.73438394483921487,\n",
      "            -1.71875933643113976,\n",
      "            -1.70313468357683817,\n",
      "            -1.68750998037237299,\n",
      "            -1.67188522069669077,\n",
      "            -1.65626039822562410,\n",
      "            -1.64063550644774980,\n",
      "            -1.62501053868220846,\n",
      "            -1.60938548809844661,\n",
      "            -1.59376034773792741,\n",
      "            -1.57813511053775346,\n",
      "            -1.56250976935625396,\n",
      "            -1.54688431700046580,\n",
      "            -1.53125874625546832,\n",
      "            -1.51563304991553660,\n",
      "            -1.50000722081701632,\n",
      "            -1.48438125187289738,\n",
      "            -1.46875513610890573,\n",
      "            -1.45312886670109065,\n",
      "            -1.43750243701472002,\n",
      "            -1.42187584064437011,\n",
      "            -1.40624907145508038,\n",
      "            -1.39062212362433990,\n",
      "            -1.37499499168481631,\n",
      "            -1.35936767056754992,\n",
      "            -1.34374015564546734,\n",
      "            -1.32811244277697549,\n",
      "            -1.31248452834937046,\n",
      "            -1.29685640932190771,\n",
      "            -1.28122808326822080,\n",
      "            -1.26559954841781797,\n",
      "            -1.24997080369646585,\n",
      "            -1.23434184876513586,\n",
      "            -1.21871268405723976,\n",
      "            -1.20308331081392472,\n",
      "            -1.18745373111709429,\n",
      "            -1.17182394791994771,\n",
      "            -1.15619396507467820,\n",
      "            -1.14056378735716502,\n",
      "            -1.12493342048831191,\n",
      "            -1.10930287115182380,\n",
      "            -1.09367214700818405,\n",
      "            -1.07804125670456274,\n",
      "            -1.06241020988049395,\n",
      "            -1.04677901716910315,\n",
      "            -1.03114769019364827,\n",
      "            -1.01551624155930442,\n",
      "            -0.99988468484002524,\n",
      "            -0.98425303456034119,\n",
      "            -0.96862130617198550,\n",
      "            -0.95298951602533866,\n",
      "            -0.93735768133569708,\n",
      "            -0.92172582014420479,\n",
      "            -0.90609395127362524,\n",
      "            -0.89046209427892753,\n",
      "            -0.87483026939284547,\n",
      "            -0.85919849746646315,\n",
      "            -0.84356679990495920,\n",
      "            -0.82793519859883780,\n",
      "            -0.81230371585069472,\n",
      "            -0.79667237429787219,\n",
      "            -0.78104119683128870,\n",
      "            -0.76541020651068370,\n",
      "            -0.74977942647666040,\n",
      "            -0.73414887985993627,\n",
      "            -0.71851858968814930,\n",
      "            -0.70288857879062294,\n",
      "            -0.68725886970161210,\n",
      "            -0.67162948456237115,\n",
      "            -0.65600044502265797,\n",
      "            -0.64037177214205676,\n",
      "            -0.62474348629167886,\n",
      "            -0.60911560705676504,\n",
      "            -0.59348815314065406,\n",
      "            -0.57786114227070229,\n",
      "            -0.56223459110662533,\n",
      "            -0.54660851515181685,\n",
      "            -0.53098292866810115,\n",
      "            -0.51535784459445932,\n",
      "            -0.49973327447025506,\n",
      "            -0.48410922836328629,\n",
      "            -0.46848571480323647,\n",
      "            -0.45286274072091165,\n",
      "            -0.43724031139364783,\n",
      "            -0.42161843039722058,\n",
      "            -0.40599709956470043,\n",
      "            -0.39037631895238345,\n",
      "            -0.37475608681324257,\n",
      "            -0.35913639957801236,\n",
      "            -0.34351725184409176,\n",
      "            -0.32789863637249078,\n",
      "            -0.31228054409280864,\n",
      "            -0.29666296411643323,\n",
      "            -0.28104588375784090,\n",
      "            -0.26542928856407633,\n",
      "            -0.24981316235227818,\n",
      "            -0.23419748725514519,\n",
      "            -0.21858224377419913,\n",
      "            -0.20296741084060996,\n",
      "            -0.18735296588336781,\n",
      "            -0.17173888490450759,\n",
      "            -0.15612514256100932,\n",
      "            -0.14051171225312367,\n",
      "            -0.12489856621860219,\n",
      "            -0.10928567563248753,\n",
      "            -0.09367301071195598,\n",
      "            -0.07806054082574637,\n",
      "            -0.06244823460763993,\n",
      "            -0.04683606007349175,\n",
      "            -0.03122398474118920,\n",
      "            -0.01561197575305361,\n",
      "            -0.00000000000000001,\n",
      "            0.01561197575305308,\n",
      "            0.03122398474118990,\n",
      "            0.04683606007349152,\n",
      "            0.06244823460763812,\n",
      "            0.07806054082574569,\n",
      "            0.09367301071195608,\n",
      "            0.10928567563248735,\n",
      "            0.12489856621860289,\n",
      "            0.14051171225312384,\n",
      "            0.15612514256100921,\n",
      "            0.17173888490450637,\n",
      "            0.18735296588336911,\n",
      "            0.20296741084060804,\n",
      "            0.21858224377419594,\n",
      "            0.23419748725514794,\n",
      "            0.24981316235227999,\n",
      "            0.26542928856407960,\n",
      "            0.28104588375783307,\n",
      "            0.29666296411643089,\n",
      "            0.31228054409281081,\n",
      "            0.32789863637248962,\n",
      "            0.34351725184409171,\n",
      "            0.35913639957801463,\n",
      "            0.37475608681324590,\n",
      "            0.39037631895237779,\n",
      "            0.40599709956470220,\n",
      "            0.42161843039722463,\n",
      "            0.43724031139364150,\n",
      "            0.45286274072091520,\n",
      "            0.46848571480323825,\n",
      "            0.48410922836328263,\n",
      "            0.49973327447025739,\n",
      "            0.51535784459446143,\n",
      "            0.53098292866809427,\n",
      "            0.54660851515181996,\n",
      "            0.56223459110663243,\n",
      "            0.57786114227069840,\n",
      "            0.59348815314065229,\n",
      "            0.60911560705676715,\n",
      "            0.62474348629167831,\n",
      "            0.64037177214205820,\n",
      "            0.65600044502265864,\n",
      "            0.67162948456237759,\n",
      "            0.68725886970161687,\n",
      "            0.70288857879061017,\n",
      "            0.71851858968813298,\n",
      "            0.73414887985994182,\n",
      "            0.74977942647666851,\n",
      "            0.76541020651068470,\n",
      "            0.78104119683129458,\n",
      "            0.79667237429787030,\n",
      "            0.81230371585068961,\n",
      "            0.82793519859884246,\n",
      "            0.84356679990496408,\n",
      "            0.85919849746646237,\n",
      "            0.87483026939284836,\n",
      "            0.89046209427892520,\n",
      "            0.90609395127361869,\n",
      "            0.92172582014420412,\n",
      "            0.93735768133570441,\n",
      "            0.95298951602534021,\n",
      "            0.96862130617197750,\n",
      "            0.98425303456034507,\n",
      "            0.99988468484002824,\n",
      "            1.01551624155929576,\n",
      "            1.03114769019364538,\n",
      "            1.04677901716910893,\n",
      "            1.06241020988049328,\n",
      "            1.07804125670455875,\n",
      "            1.09367214700818294,\n",
      "            1.10930287115182891,\n",
      "            1.12493342048831324,\n",
      "            1.14056378735715991,\n",
      "            1.15619396507467664,\n",
      "            1.17182394791994615,\n",
      "            1.18745373111709496,\n",
      "            1.20308331081392517,\n",
      "            1.21871268405723976,\n",
      "            1.23434184876513742,\n",
      "            1.24997080369646785,\n",
      "            1.26559954841781486,\n",
      "            1.28122808326821902,\n",
      "            1.29685640932190838,\n",
      "            1.31248452834936979,\n",
      "            1.32811244277697527,\n",
      "            1.34374015564546911,\n",
      "            1.35936767056754926,\n",
      "            1.37499499168481543,\n",
      "            1.39062212362433968,\n",
      "            1.40624907145508038,\n",
      "            1.42187584064437145,\n",
      "            1.43750243701471936,\n",
      "            1.45312886670109176,\n",
      "            1.46875513610890573,\n",
      "            1.48438125187289649,\n",
      "            1.50000722081701632,\n",
      "            1.51563304991553371,\n",
      "            1.53125874625546743,\n",
      "            1.54688431700046691,\n",
      "            1.56250976935625530,\n",
      "            1.57813511053775146,\n",
      "            1.59376034773792763,\n",
      "            1.60938548809845039,\n",
      "            1.62501053868220890,\n",
      "            1.64063550644775069,\n",
      "            1.65626039822562610,\n",
      "            1.67188522069669032,\n",
      "            1.68750998037236810,\n",
      "            1.70313468357683373,\n",
      "            1.71875933643114442,\n",
      "            1.73438394483921554,\n",
      "            1.75000851447576222,\n",
      "            1.76563305077596300,\n",
      "            1.78125755892688842,\n",
      "            1.79688204386070294,\n",
      "            1.81250651024946907,\n",
      "            1.82813096250145990,\n",
      "            1.84375540475907918,\n",
      "            1.85937984089812192,\n",
      "            1.87500427452834906,\n",
      "            1.89062870899535418,\n",
      "            1.90625314738377516,\n",
      "            1.92187759368646116,\n",
      "            1.93750744254844198,\n",
      "            1.96430075965754036,\n",
      "            2.35928671715313998,\n",
      "            2.90212209091241968]),\n",
      "    ]\n",
      "    ys_const = [\n",
      "        t.tensor([0.00000000000000000,\n",
      "            1.00000000000000000]),\n",
      "        t.tensor([-0.00760939905895985,\n",
      "            0.32228164205929283,\n",
      "            0.67771835794070634,\n",
      "            1.00760939905895985]),\n",
      "        t.tensor([-0.00109568445401875,\n",
      "            -0.08859905421738420,\n",
      "            0.12499325245301279,\n",
      "            0.37208706831557098,\n",
      "            0.62791293168442885,\n",
      "            0.87500674754698693,\n",
      "            1.08859905421738445,\n",
      "            1.00109568445401886]),\n",
      "        t.tensor([-0.00061544137315238,\n",
      "            -0.05226400945224528,\n",
      "            -0.10968093075636222,\n",
      "            -0.01616241151827657,\n",
      "            0.09284718920680775,\n",
      "            0.20788265818235127,\n",
      "            0.32538257439567753,\n",
      "            0.44234954243304520,\n",
      "            0.55765045756695475,\n",
      "            0.67461742560432192,\n",
      "            0.79211734181764903,\n",
      "            0.90715281079319321,\n",
      "            1.01616241151827857,\n",
      "            1.10968093075636243,\n",
      "            1.05226400945224530,\n",
      "            1.00061544137315250]),\n",
      "        t.tensor([-0.00029329340169840,\n",
      "            -0.02854139952539195,\n",
      "            -0.06641448071143671,\n",
      "            -0.10915824625047750,\n",
      "            -0.12825952808128421,\n",
      "            -0.12406392804652215,\n",
      "            -0.09986820686193564,\n",
      "            -0.05358452610203352,\n",
      "            -0.00010720712329275,\n",
      "            0.05951007627620760,\n",
      "            0.12435347674215998,\n",
      "            0.19026991730183726,\n",
      "            0.25588057946045256,\n",
      "            0.32002152891506930,\n",
      "            0.38804219558887709,\n",
      "            0.46197742838139511,\n",
      "            0.53802257161855382,\n",
      "            0.61195780441124548,\n",
      "            0.67997847108483567,\n",
      "            0.74411942053955260,\n",
      "            0.80973008269820845,\n",
      "            0.87564652325779502,\n",
      "            0.94048992372381446,\n",
      "            1.00010720712328482,\n",
      "            1.05358452610203734,\n",
      "            1.09986820686193632,\n",
      "            1.12406392804652100,\n",
      "            1.12825952808128571,\n",
      "            1.10915824625047810,\n",
      "            1.06641448071143685,\n",
      "            1.02854139952539136,\n",
      "            1.00029329340169837]),\n",
      "        t.tensor([-0.00038456582627704,\n",
      "            -0.03558473441315251,\n",
      "            -0.08052000068658792,\n",
      "            -0.10725340496319002,\n",
      "            -0.11364620936774648,\n",
      "            -0.11872324596227261,\n",
      "            -0.12298118622336887,\n",
      "            -0.12622479141152740,\n",
      "            -0.12824477968402076,\n",
      "            -0.12882103148996180,\n",
      "            -0.12772510670012566,\n",
      "            -0.12472370596442466,\n",
      "            -0.11958333584539720,\n",
      "            -0.11207640028901288,\n",
      "            -0.10198881774590096,\n",
      "            -0.08912902765950530,\n",
      "            -0.07333790442004298,\n",
      "            -0.05449869461222381,\n",
      "            -0.03254575041946415,\n",
      "            -0.00747070445727027,\n",
      "            0.02067505125705777,\n",
      "            0.05178213983257415,\n",
      "            0.08568898513139335,\n",
      "            0.12219060292394657,\n",
      "            0.16104891655868597,\n",
      "            0.20200305165131152,\n",
      "            0.24477836538698960,\n",
      "            0.28909347104441679,\n",
      "            0.33466504258847318,\n",
      "            0.38121059236453880,\n",
      "            0.42844964772304611,\n",
      "            0.47610381714039773,\n",
      "            0.52389618285960204,\n",
      "            0.57155035227695261,\n",
      "            0.61878940763545975,\n",
      "            0.66533495741152693,\n",
      "            0.71090652895558493,\n",
      "            0.75522163461301073,\n",
      "            0.79799694834868540,\n",
      "            0.83895108344131109,\n",
      "            0.87780939707605554,\n",
      "            0.91431101486860944,\n",
      "            0.94821786016742693,\n",
      "            0.97932494874294029,\n",
      "            1.00747070445726994,\n",
      "            1.03254575041946217,\n",
      "            1.05449869461222523,\n",
      "            1.07333790442004129,\n",
      "            1.08912902765950781,\n",
      "            1.10198881774589919,\n",
      "            1.11207640028901622,\n",
      "            1.11958333584539438,\n",
      "            1.12472370596442706,\n",
      "            1.12772510670012571,\n",
      "            1.12882103148995672,\n",
      "            1.12824477968402248,\n",
      "            1.12622479141152954,\n",
      "            1.12298118622336940,\n",
      "            1.11872324596226957,\n",
      "            1.11364620936774883,\n",
      "            1.10725340496319080,\n",
      "            1.08052000068658849,\n",
      "            1.03558473441315235,\n",
      "            1.00038456582627711]),\n",
      "        t.tensor([-0.00032963801607456,\n",
      "            -0.03144899474976939,\n",
      "            -0.07237475700758451,\n",
      "            -0.09634133849055662,\n",
      "            -0.10019163865503100,\n",
      "            -0.10337801328801051,\n",
      "            -0.10647568979161709,\n",
      "            -0.10946593993161140,\n",
      "            -0.11232851660834062,\n",
      "            -0.11504224011654615,\n",
      "            -0.11758503839729113,\n",
      "            -0.11993399428201490,\n",
      "            -0.12206540015412194,\n",
      "            -0.12395482017400423,\n",
      "            -0.12557716019059412,\n",
      "            -0.12690674543644861,\n",
      "            -0.12791740607459506,\n",
      "            -0.12858257063040973,\n",
      "            -0.12887536730095286,\n",
      "            -0.12876873308402831,\n",
      "            -0.12823553060903956,\n",
      "            -0.12724867247958555,\n",
      "            -0.12578125285157568,\n",
      "            -0.12380668587069379,\n",
      "            -0.12129885047714439,\n",
      "            -0.11823224095572617,\n",
      "            -0.11458212246486700,\n",
      "            -0.11032469062304774,\n",
      "            -0.10543723406677229,\n",
      "            -0.09989829872705513,\n",
      "            -0.09368785240527382,\n",
      "            -0.08678744807239387,\n",
      "            -0.07918038417476764,\n",
      "            -0.07085186011302450,\n",
      "            -0.06178912497703734,\n",
      "            -0.05198161757636834,\n",
      "            -0.04142109580967498,\n",
      "            -0.03010175347350466,\n",
      "            -0.01802032272339011,\n",
      "            -0.00517616056997490,\n",
      "            0.00842868198333836,\n",
      "            0.02278940928127171,\n",
      "            0.03789844964187317,\n",
      "            0.05374544372714435,\n",
      "            0.07031725275834511,\n",
      "            0.08759798631036457,\n",
      "            0.10556904906649998,\n",
      "            0.12420920558818814,\n",
      "            0.14349466186675403,\n",
      "            0.16339916218689465,\n",
      "            0.18389409965260192,\n",
      "            0.20494863861034671,\n",
      "            0.22652984715340754,\n",
      "            0.24860283790303903,\n",
      "            0.27113091533248224,\n",
      "            0.29407572802116183,\n",
      "            0.31739742438954849,\n",
      "            0.34105481065999815,\n",
      "            0.36500551000425752,\n",
      "            0.38920612206322297,\n",
      "            0.41361238224866209,\n",
      "            0.43817932045027935,\n",
      "            0.46286141896622940,\n",
      "            0.48761276964403705,\n",
      "            0.51238723035596290,\n",
      "            0.53713858103377021,\n",
      "            0.56182067954972070,\n",
      "            0.58638761775133763,\n",
      "            0.61079387793677709,\n",
      "            0.63499448999574337,\n",
      "            0.65894518934000257,\n",
      "            0.68260257561045057,\n",
      "            0.70592427197883911,\n",
      "            0.72886908466752032,\n",
      "            0.75139716209696139,\n",
      "            0.77347015284659137,\n",
      "            0.79505136138965604,\n",
      "            0.81610590034740038,\n",
      "            0.83660083781310102,\n",
      "            0.85650533813324692,\n",
      "            0.87579079441181340,\n",
      "            0.89443095093349811,\n",
      "            0.91240201368963370,\n",
      "            0.92968274724165312,\n",
      "            0.94625455627285515,\n",
      "            0.96210155035812772,\n",
      "            0.97721059071873084,\n",
      "            0.99157131801665899,\n",
      "            1.00517616056997339,\n",
      "            1.01802032272338550,\n",
      "            1.03010175347350508,\n",
      "            1.04142109580967723,\n",
      "            1.05198161757637187,\n",
      "            1.06178912497703837,\n",
      "            1.07085186011302347,\n",
      "            1.07918038417476625,\n",
      "            1.08678744807239269,\n",
      "            1.09368785240527644,\n",
      "            1.09989829872705225,\n",
      "            1.10543723406677130,\n",
      "            1.11032469062305084,\n",
      "            1.11458212246487065,\n",
      "            1.11823224095572660,\n",
      "            1.12129885047714350,\n",
      "            1.12380668587068833,\n",
      "            1.12578125285157649,\n",
      "            1.12724867247959071,\n",
      "            1.12823553060903903,\n",
      "            1.12876873308403147,\n",
      "            1.12887536730094684,\n",
      "            1.12858257063040957,\n",
      "            1.12791740607459379,\n",
      "            1.12690674543644653,\n",
      "            1.12557716019059728,\n",
      "            1.12395482017400528,\n",
      "            1.12206540015411727,\n",
      "            1.11993399428201990,\n",
      "            1.11758503839728607,\n",
      "            1.11504224011655251,\n",
      "            1.11232851660833942,\n",
      "            1.10946593993160847,\n",
      "            1.10647568979161925,\n",
      "            1.10337801328800289,\n",
      "            1.10019163865503922,\n",
      "            1.09634133849055670,\n",
      "            1.07237475700758478,\n",
      "            1.03144899474976981,\n",
      "            1.00032963801607466]),\n",
      "        t.tensor([-0.00031455783889701,\n",
      "            -0.02988220810728750,\n",
      "            -0.06838815676661437,\n",
      "            -0.09052381626826272,\n",
      "            -0.09279728070701555,\n",
      "            -0.09446262302455798,\n",
      "            -0.09611762826211076,\n",
      "            -0.09776054918065887,\n",
      "            -0.09938928584286907,\n",
      "            -0.10100167418675354,\n",
      "            -0.10259548653090733,\n",
      "            -0.10416843212275302,\n",
      "            -0.10571815779420764,\n",
      "            -0.10724224872670400,\n",
      "            -0.10873822932783883,\n",
      "            -0.11020356422140801,\n",
      "            -0.11163565935265615,\n",
      "            -0.11303186321018566,\n",
      "            -0.11438946816616835,\n",
      "            -0.11570571193582646,\n",
      "            -0.11697777915745033,\n",
      "            -0.11820280309367875,\n",
      "            -0.11937786745480922,\n",
      "            -0.12050000834451111,\n",
      "            -0.12156621632827740,\n",
      "            -0.12257343862456249,\n",
      "            -0.12351858141854224,\n",
      "            -0.12439851229798785,\n",
      "            -0.12521006281063407,\n",
      "            -0.12595003114217504,\n",
      "            -0.12661518491378290,\n",
      "            -0.12720226409773372,\n",
      "            -0.12770798404951841,\n",
      "            -0.12812903865462680,\n",
      "            -0.12846210358773133,\n",
      "            -0.12870383968198731,\n",
      "            -0.12885089640574107,\n",
      "            -0.12889991544356544,\n",
      "            -0.12884753437866370,\n",
      "            -0.12869039047290171,\n",
      "            -0.12842512454069202,\n",
      "            -0.12804838491299517,\n",
      "            -0.12755683148657232,\n",
      "            -0.12694713985434555,\n",
      "            -0.12621600551170145,\n",
      "            -0.12536014813363192,\n",
      "            -0.12437631591725358,\n",
      "            -0.12326128998394514,\n",
      "            -0.12201188883519327,\n",
      "            -0.12062497285565056,\n",
      "            -0.11909744885720500,\n",
      "            -0.11742627465696613,\n",
      "            -0.11560846368223408,\n",
      "            -0.11364108959530056,\n",
      "            -0.11152129093037615,\n",
      "            -0.10924627573498059,\n",
      "            -0.10681332620801866,\n",
      "            -0.10421980332610868,\n",
      "            -0.10146315145018489,\n",
      "            -0.09854090290363757,\n",
      "            -0.09545068251360159,\n",
      "            -0.09219021210643127,\n",
      "            -0.08875731494874418,\n",
      "            -0.08514992012489672,\n",
      "            -0.08136606684205296,\n",
      "            -0.07740390865375155,\n",
      "            -0.07326171759284809,\n",
      "            -0.06893788820497800,\n",
      "            -0.06443094147339049,\n",
      "            -0.05973952862624372,\n",
      "            -0.05486243481754034,\n",
      "            -0.04979858267311532,\n",
      "            -0.04454703569275254,\n",
      "            -0.03910700150051654,\n",
      "            -0.03347783493467905,\n",
      "            -0.02765904096961883,\n",
      "            -0.02165027746184888,\n",
      "            -0.01545135771282986,\n",
      "            -0.00906225284155850,\n",
      "            -0.00248309396012123,\n",
      "            0.00428582585400533,\n",
      "            0.01124404979495591,\n",
      "            0.01839095578019531,\n",
      "            0.02572575512213179,\n",
      "            0.03324749142416124,\n",
      "            0.04095503970533251,\n",
      "            0.04884710575724134,\n",
      "            0.05692222573630709,\n",
      "            0.06517876599410004,\n",
      "            0.07361492314770822,\n",
      "            0.08222872439180305,\n",
      "            0.09101802805342750,\n",
      "            0.09998052438993345,\n",
      "            0.10911373663009812,\n",
      "            0.11841502225785103,\n",
      "            0.12788157453742816,\n",
      "            0.13751042427843285,\n",
      "            0.14729844183868110,\n",
      "            0.15724233936216325,\n",
      "            0.16733867324903981,\n",
      "            0.17758384685420067,\n",
      "            0.18797411341025075,\n",
      "            0.19850557917062545,\n",
      "            0.20917420676791010,\n",
      "            0.21997581878220465,\n",
      "            0.23090610151387087,\n",
      "            0.24196060895484517,\n",
      "            0.25313476695217318,\n",
      "            0.26442387755726598,\n",
      "            0.27582312355406297,\n",
      "            0.28732757315902224,\n",
      "            0.29893218488563772,\n",
      "            0.31063181256599859,\n",
      "            0.32242121052164247,\n",
      "            0.33429503887592693,\n",
      "            0.34624786899980142,\n",
      "            0.35827418908292513,\n",
      "            0.37036840982181118,\n",
      "            0.38252487021667442,\n",
      "            0.39473784346850216,\n",
      "            0.40700154296784891,\n",
      "            0.41931012836676890,\n",
      "            0.43165771172523826,\n",
      "            0.44403836372340311,\n",
      "            0.45644611993092293,\n",
      "            0.46887498712468101,\n",
      "            0.48131894964608551,\n",
      "            0.49377197578918691,\n",
      "            0.50622802421081292,\n",
      "            0.51868105035391421,\n",
      "            0.53112501287531932,\n",
      "            0.54355388006907612,\n",
      "            0.55596163627659578,\n",
      "            0.56834228827476230,\n",
      "            0.58068987163323071,\n",
      "            0.59299845703215071,\n",
      "            0.60526215653149917,\n",
      "            0.61747512978332419,\n",
      "            0.62963159017818915,\n",
      "            0.64172581091707503,\n",
      "            0.65375213100019824,\n",
      "            0.66570496112407151,\n",
      "            0.67757878947835659,\n",
      "            0.68936818743400397,\n",
      "            0.70106781511436289,\n",
      "            0.71267242684097609,\n",
      "            0.72417687644593454,\n",
      "            0.73557612244273374,\n",
      "            0.74686523304782793,\n",
      "            0.75803939104515428,\n",
      "            0.76909389848612986,\n",
      "            0.78002418121779771,\n",
      "            0.79082579323208679,\n",
      "            0.80149442082937317,\n",
      "            0.81202588658975350,\n",
      "            0.82241615314579497,\n",
      "            0.83266132675096127,\n",
      "            0.84275766063783841,\n",
      "            0.85270155816131732,\n",
      "            0.86248957572156837,\n",
      "            0.87211842546257534,\n",
      "            0.88158497774214473,\n",
      "            0.89088626336989707,\n",
      "            0.90001947561007434,\n",
      "            0.90898197194657415,\n",
      "            0.91777127560819560,\n",
      "            0.92638507685229232,\n",
      "            0.93482123400589601,\n",
      "            0.94307777426369310,\n",
      "            0.95115289424276328,\n",
      "            0.95904496029466846,\n",
      "            0.96675250857584227,\n",
      "            0.97427424487786418,\n",
      "            0.98160904421979878,\n",
      "            0.98875595020504026,\n",
      "            0.99571417414600072,\n",
      "            1.00248309396012769,\n",
      "            1.00906225284155204,\n",
      "            1.01545135771283390,\n",
      "            1.02165027746184989,\n",
      "            1.02765904096961225,\n",
      "            1.03347783493467849,\n",
      "            1.03910700150051838,\n",
      "            1.04454703569276153,\n",
      "            1.04979858267310533,\n",
      "            1.05486243481754638,\n",
      "            1.05973952862623721,\n",
      "            1.06443094147339479,\n",
      "            1.06893788820498226,\n",
      "            1.07326171759283784,\n",
      "            1.07740390865375124,\n",
      "            1.08136606684205927,\n",
      "            1.08514992012489975,\n",
      "            1.08875731494873618,\n",
      "            1.09219021210643352,\n",
      "            1.09545068251360012,\n",
      "            1.09854090290364081,\n",
      "            1.10146315145018203,\n",
      "            1.10421980332610281,\n",
      "            1.10681332620802841,\n",
      "            1.10924627573497592,\n",
      "            1.11152129093037022,\n",
      "            1.11364108959530239,\n",
      "            1.11560846368224431,\n",
      "            1.11742627465694833,\n",
      "            1.11909744885722429,\n",
      "            1.12062497285564033,\n",
      "            1.12201188883519509,\n",
      "            1.12326128998395158,\n",
      "            1.12437631591724840,\n",
      "            1.12536014813363017,\n",
      "            1.12621600551170431,\n",
      "            1.12694713985434380,\n",
      "            1.12755683148657737,\n",
      "            1.12804838491300630,\n",
      "            1.12842512454068311,\n",
      "            1.12869039047288999,\n",
      "            1.12884753437867258,\n",
      "            1.12889991544355173,\n",
      "            1.12885089640574687,\n",
      "            1.12870383968198951,\n",
      "            1.12846210358772892,\n",
      "            1.12812903865463676,\n",
      "            1.12770798404951300,\n",
      "            1.12720226409773505,\n",
      "            1.12661518491379242,\n",
      "            1.12595003114215975,\n",
      "            1.12521006281063585,\n",
      "            1.12439851229799315,\n",
      "            1.12351858141853889,\n",
      "            1.12257343862456560,\n",
      "            1.12156621632826603,\n",
      "            1.12050000834453600,\n",
      "            1.11937786745479806,\n",
      "            1.11820280309367148,\n",
      "            1.11697777915746044,\n",
      "            1.11570571193580714,\n",
      "            1.11438946816618079,\n",
      "            1.11303186321019676,\n",
      "            1.11163565935263753,\n",
      "            1.11020356422141631,\n",
      "            1.10873822932783717,\n",
      "            1.10724224872670596,\n",
      "            1.10571815779420612,\n",
      "            1.10416843212275695,\n",
      "            1.10259548653090800,\n",
      "            1.10100167418676365,\n",
      "            1.09938928584285445,\n",
      "            1.09776054918065791,\n",
      "            1.09611762826210790,\n",
      "            1.09446262302456487,\n",
      "            1.09279728070701454,\n",
      "            1.09052381626826556,\n",
      "            1.06838815676661425,\n",
      "            1.02988220810728670,\n",
      "            1.00031455783889700]),\n",
      "    ]\n",
      "\n"
     ]
    }
   ],
   "source": [
    "separator1 = ',\\n            '\n",
    "separator2 = ',\\n        '\n",
    "\n",
    "xs_per_bits = []\n",
    "ys_per_bits = []\n",
    "for xs, ys in zip(xs_for_bits, ys_for_bits):\n",
    "    xs_const = [f'{x.item():.17f}' for x in xs[1:-1]]\n",
    "    ys_const = [f'{y.item():.17f}' for y in ys]\n",
    "    \n",
    "    xs_per_bits.append(f't.tensor([{separator1.join(xs_const)}])')\n",
    "    ys_per_bits.append(f't.tensor([{separator1.join(ys_const)}])')\n",
    "    \n",
    "print(f'    xs_const = [{separator2[1:]}{separator2.join(xs_per_bits)},\\n    ]\\n'\n",
    "      f'    ys_const = [{separator2[1:]}{separator2.join(ys_per_bits)},\\n    ]\\n')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "eab26c66",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "xs = best_sol\n",
    "ys = get_y_from_x(xs)\n",
    "\n",
    "pts = np.linspace(L, R, 2**10)\n",
    "plt.plot(pts, gelu(pts))\n",
    "plt.show()\n",
    "\n",
    "plt.plot(pts, gelu_deriv(pts))\n",
    "for x1, x2, y in zip(xs[:-1], xs[1:], ys):\n",
    "    pts = np.linspace(x1, x2, 2**10)\n",
    "    plt.plot(pts, np.ones_like(pts) * y)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "3889d493",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.collections.PathCollection at 0x7f96861305e0>"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.scatter(np.arange(len(xs[1:-1])), xs[1:-1])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8c0a88f0",
   "metadata": {},
   "source": [
    "# Benchamrk"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ed433854",
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch import nn\n",
    "\n",
    "\n",
    "def ff(dim: int, act_fn):\n",
    "    return nn.Sequential(\n",
    "        nn.Linear(dim, 4 * dim),\n",
    "        act_fn(),\n",
    "        nn.Linear(4 * dim, dim),\n",
    "        act_fn(),\n",
    "    )\n",
    "\n",
    "\n",
    "def network(dim: int, act_fn):\n",
    "    print(act_fn)\n",
    "    return nn.Sequential(\n",
    "        ff(d_in, act_fn),\n",
    "        ff(d_in, act_fn),\n",
    "        ff(d_in, act_fn),\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "d0940060",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[autoreload of custom_gelu failed: Traceback (most recent call last):\n",
      "  File \"/home/gnovikov/data/miniconda3/envs/ml/lib/python3.9/site-packages/IPython/extensions/autoreload.py\", line 245, in check\n",
      "    superreload(m, reload, self.old_objects)\n",
      "  File \"/home/gnovikov/data/miniconda3/envs/ml/lib/python3.9/site-packages/IPython/extensions/autoreload.py\", line 410, in superreload\n",
      "    update_generic(old_obj, new_obj)\n",
      "  File \"/home/gnovikov/data/miniconda3/envs/ml/lib/python3.9/site-packages/IPython/extensions/autoreload.py\", line 347, in update_generic\n",
      "    update(a, b)\n",
      "  File \"/home/gnovikov/data/miniconda3/envs/ml/lib/python3.9/site-packages/IPython/extensions/autoreload.py\", line 292, in update_class\n",
      "    if (old_obj == new_obj) is True:\n",
      "RuntimeError: Boolean value of Tensor with more than one value is ambiguous\n",
      "]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "<class 'torch.nn.modules.linear.Identity'>\n",
      "input_memory=4194304\n",
      "forward_memory=512\n",
      "\n",
      "input_memory=4194304\n",
      "forward_memory=58720768\n",
      "\n",
      "\n",
      "\n",
      "<class 'torch.nn.modules.activation.ReLU'>\n",
      "input_memory=4194304\n",
      "forward_memory=512\n",
      "\n",
      "input_memory=4194304\n",
      "forward_memory=121635328\n",
      "\n",
      "\n",
      "\n",
      "<class 'torch.nn.modules.activation.GELU'>\n",
      "input_memory=4194304\n",
      "forward_memory=512\n",
      "\n",
      "input_memory=4194304\n",
      "forward_memory=121635328\n",
      "\n",
      "\n",
      "\n",
      "<function <lambda> at 0x7fb41608f3a0>\n",
      "input_memory=4194304\n",
      "forward_memory=1024\n",
      "\n",
      "(tensor([[0, 0, 0,  ..., 1, 1, 1],\n",
      "        [0, 0, 0,  ..., 1, 1, 1],\n",
      "        [0, 0, 0,  ..., 1, 1, 1],\n",
      "        ...,\n",
      "        [0, 0, 0,  ..., 1, 1, 1],\n",
      "        [0, 0, 0,  ..., 1, 1, 1],\n",
      "        [0, 0, 0,  ..., 1, 1, 1]], device='cuda:0', dtype=torch.uint8),)\n",
      "(tensor([[0, 0, 0,  ..., 0, 0, 0],\n",
      "        [0, 0, 0,  ..., 0, 1, 0],\n",
      "        [0, 0, 0,  ..., 0, 0, 0],\n",
      "        ...,\n",
      "        [0, 0, 0,  ..., 0, 0, 0],\n",
      "        [0, 0, 0,  ..., 0, 1, 0],\n",
      "        [0, 0, 0,  ..., 0, 1, 0]], device='cuda:0', dtype=torch.uint8),)\n",
      "(tensor([[1, 0, 0,  ..., 0, 1, 0],\n",
      "        [1, 0, 1,  ..., 0, 0, 0],\n",
      "        [1, 0, 1,  ..., 0, 0, 0],\n",
      "        ...,\n",
      "        [1, 1, 0,  ..., 1, 0, 0],\n",
      "        [0, 1, 1,  ..., 0, 1, 1],\n",
      "        [1, 0, 1,  ..., 0, 0, 0]], device='cuda:0', dtype=torch.uint8),)\n",
      "(tensor([[1, 0, 1,  ..., 1, 1, 1],\n",
      "        [0, 0, 0,  ..., 1, 0, 1],\n",
      "        [1, 0, 0,  ..., 1, 0, 1],\n",
      "        ...,\n",
      "        [1, 0, 1,  ..., 0, 0, 1],\n",
      "        [0, 1, 1,  ..., 0, 1, 1],\n",
      "        [0, 1, 1,  ..., 1, 1, 0]], device='cuda:0', dtype=torch.uint8),)\n",
      "(tensor([[0, 0, 1,  ..., 1, 1, 1],\n",
      "        [0, 1, 0,  ..., 0, 0, 1],\n",
      "        [0, 1, 0,  ..., 1, 1, 1],\n",
      "        ...,\n",
      "        [0, 0, 0,  ..., 0, 0, 1],\n",
      "        [1, 1, 0,  ..., 1, 0, 0],\n",
      "        [0, 1, 0,  ..., 0, 0, 1]], device='cuda:0', dtype=torch.uint8),)\n",
      "(tensor([[1, 1, 0,  ..., 1, 0, 1],\n",
      "        [1, 0, 1,  ..., 0, 0, 0],\n",
      "        [0, 1, 1,  ..., 0, 0, 0],\n",
      "        ...,\n",
      "        [0, 1, 1,  ..., 0, 0, 1],\n",
      "        [0, 0, 1,  ..., 0, 1, 1],\n",
      "        [0, 0, 1,  ..., 0, 0, 1]], device='cuda:0', dtype=torch.uint8),)\n",
      "input_memory=4194304\n",
      "forward_memory=74449408\n",
      "\n",
      "\n",
      "\n"
     ]
    }
   ],
   "source": [
    "import torch as t\n",
    "from custom_gelu import MyGelu\n",
    "\n",
    "\n",
    "\n",
    "batch_sz = 1024\n",
    "d_in = 1024\n",
    "\n",
    "def test_f(backward: bool = False):\n",
    "    start_memory = t.cuda.memory_allocated()\n",
    "    \n",
    "    t.manual_seed(19)\n",
    "    inp = t.randn(batch_sz, d_in).cuda()  \n",
    "    \n",
    "    if backward:\n",
    "        inp.requires_grad_(True)\n",
    "    \n",
    "    after_input_memory = t.cuda.memory_allocated()\n",
    "    input_memory = after_input_memory - start_memory\n",
    "    \n",
    "    out = net(inp).mean()\n",
    "    after_forward_memory = t.cuda.memory_allocated()\n",
    "    forward_memory = after_forward_memory - after_input_memory\n",
    "    \n",
    "#     print(out)\n",
    "    if backward:\n",
    "        out.backward()\n",
    "#         print(inp.grad.mean())\n",
    "    \n",
    "    print(f'{input_memory=}')\n",
    "    print(f'{forward_memory=}')\n",
    "\n",
    "    \n",
    "net = network(d_in, nn.Identity).cuda()\n",
    "with t.no_grad():\n",
    "    test_f()\n",
    "print()\n",
    "test_f(True)\n",
    "print()\n",
    "print()\n",
    "print()\n",
    "\n",
    "net = network(d_in, nn.ReLU).cuda()\n",
    "with t.no_grad():\n",
    "    test_f()\n",
    "print()\n",
    "test_f(True)\n",
    "print()\n",
    "print()\n",
    "print()\n",
    "\n",
    "net = network(d_in, nn.GELU).cuda()\n",
    "with t.no_grad():\n",
    "    test_f()\n",
    "print()\n",
    "test_f(True)\n",
    "print()\n",
    "print()\n",
    "print()\n",
    "\n",
    "\n",
    "net = network(d_in, lambda: MyGelu(1)).cuda()\n",
    "with t.no_grad():\n",
    "    test_f()\n",
    "print()\n",
    "test_f(True)\n",
    "print()\n",
    "print()\n",
    "print()\n",
    "\n",
    "\n",
    "# net = network(d_in, MyReLU).cuda()\n",
    "# with t.no_grad():\n",
    "#     test_f()\n",
    "# print()\n",
    "# test_f(True)\n",
    "# print()\n",
    "# print()\n",
    "# print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "4e26c937",
   "metadata": {},
   "outputs": [],
   "source": [
    "class MyReluFn(t.autograd.Function):\n",
    "    @staticmethod\n",
    "    def forward(ctx, x):\n",
    "        x = nnF.relu(x)\n",
    "        ctx.save_for_backward(x)\n",
    "        return x\n",
    "    \n",
    "    @staticmethod\n",
    "    def backward(ctx, dy):\n",
    "        x, = ctx.saved_tensors\n",
    "        return dy * (x > 0)\n",
    "    \n",
    "    \n",
    "class MyReLU(nn.Module):\n",
    "    def forward(self, x):\n",
    "        return MyReluFn.apply(x)        "
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
