{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%env CUDA_VISIBLE_DEVICES=7\n",
    "from jaxutil.auto import *\n",
    "from jax import random as jr\n",
    "from flax.traverse_util import flatten_dict, unflatten_dict\n",
    "from muP import muP\n",
    "import optax"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Quad Features"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "d = 32\n",
    "rng = RNG(0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_all_n(A, ns, depth):\n",
    "    f_star = lambda x: nn.relu(x.T @ A @ x)\n",
    "\n",
    "    n_test = 100000\n",
    "    test_x = jr.normal(rng.next(), (n_test, d))\n",
    "    test_y = vmap(f_star)(test_x)\n",
    "\n",
    "    if depth == 2:\n",
    "        width = 2048\n",
    "    else:\n",
    "        width = 256\n",
    "    model = muP(d, width, depth)\n",
    "\n",
    "    p0 = model.init(rng.next(), test_x)\n",
    "    p0 = flatten_dict(p0[\"params\"], sep=\".\")\n",
    "\n",
    "    @partial(jit, static_argnames=\"mutable\")\n",
    "    def f(p, *args, **kwargs):\n",
    "        p = dict(params=unflatten_dict(p, sep=\".\"))\n",
    "        return model.apply(p, *args, **kwargs)\n",
    "\n",
    "    opt = optax.sgd(.3, momentum=0.9)\n",
    "    # opt = optax.adam(1e-2)\n",
    "\n",
    "    criterion = lambda f, y: (f - y)**2\n",
    "    test_vals = []\n",
    "\n",
    "    @jit\n",
    "    def loss_fn(p, x, y):\n",
    "        y_hat = f(p, x)\n",
    "        return vmap(criterion)(y_hat, y).mean()\n",
    "\n",
    "    @jit\n",
    "    def step(p, opt_state, train_x, train_y, key):\n",
    "        key, subkey = jr.split(key)\n",
    "        loss, g = jax.value_and_grad(loss_fn)(p, train_x, train_y)\n",
    "        updates, opt_state = opt.update(g, opt_state)\n",
    "        p = optax.apply_updates(p, updates)\n",
    "        return (p, opt_state, key), loss \n",
    "\n",
    "    def train(n):\n",
    "        # n = 100000\n",
    "        train_x = jr.normal(rng.next(), (n, d))\n",
    "        train_y = vmap(f_star)(train_x)\n",
    "\n",
    "        p = p0\n",
    "        key = rng.next()\n",
    "        opt_state = opt.init(p0)\n",
    "        save_list = []\n",
    "        for _ in trange(10000):\n",
    "            (p, opt_state, key), loss = step(p, opt_state,train_x, train_y, key)\n",
    "            save_list.append(loss)\n",
    "            if loss < 1e-2:\n",
    "                break\n",
    "        save_list = tree_stack(save_list)\n",
    "\n",
    "        test_loss = loss_fn(p, test_x, test_y)\n",
    "        return save_list, test_loss\n",
    "\n",
    "    for n in ns:\n",
    "        out, test = train(n)\n",
    "        print(n, out[-1], test)\n",
    "        test_vals.append(test)\n",
    "        # plt.plot(out)\n",
    "\n",
    "    return test_vals"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Test a single A"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "ns = [2**8, 2**10, 2**12, 2**14, 2**16]\n",
    "\n",
    "A = jr.normal(rng.next(), (d, d))\n",
    "A = A + A.T\n",
    "A = A - jnp.trace(A)*jnp.eye(d)/d\n",
    "A = A/jnp.linalg.norm(A)\n",
    "\n",
    "test_vals_3 = run_all_n(A, ns, 3)\n",
    "test_vals_2 = run_all_n(A, ns, 2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.plot(ns, test_vals_2)\n",
    "plt.plot(ns, test_vals_3)\n",
    "plt.xscale(\"log\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Average over multiple A's"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "trials = 5\n",
    "ns = [2**8, 2**10, 2**12, 2**14, 2**16, 2**20]\n",
    "test_vals_3_all = []\n",
    "test_vals_2_all = []\n",
    "for _ in range(trials):\n",
    "    A = jr.normal(rng.next(), (d, d))\n",
    "    A = A + A.T\n",
    "    A = A - jnp.trace(A)*jnp.eye(d)/d\n",
    "    A = A/jnp.linalg.norm(A)\n",
    "    test_vals_3_all.append(run_all_n(A, ns, 3))\n",
    "    test_vals_2_all.append(run_all_n(A, ns, 2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "## DUMP TO PICKLE\n",
    "\n",
    "import pickle\n",
    "with open('muP_2_3_data_run2.pickle', 'wb') as handle:\n",
    "    pickle.dump([test_vals_2_all, test_vals_3_all], handle)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "## LOAD FROM PICKLE\n",
    "import pickle\n",
    "with open('muP_2_3_data.pickle', \"rb\") as input_file:\n",
    "    data = pickle.load(input_file)\n",
    "\n",
    "test_vals_2_all, test_vals_3_all = data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "ns = [2**8, 2**10, 2**12, 2**14, 2**16, 2**20]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "plt.style.use('style.mplstyle')\n",
    "\n",
    "mean_2 = jnp.mean(jnp.array(test_vals_2_all), axis=0)\n",
    "std_2 = jnp.std(jnp.array(test_vals_2_all), axis=0)\n",
    "plt.plot(ns, mean_2, marker = 'o', color='r', label=\"2 layer network\")\n",
    "plt.fill_between(ns, mean_2 - 2*std_2, mean_2 + 2*std_2, color = 'r', alpha = 0.15)\n",
    "\n",
    "mean_3 = jnp.mean(jnp.array(test_vals_3_all[:5]), axis=0)\n",
    "std_3 = jnp.std(jnp.array(test_vals_3_all[:5]), axis=0)\n",
    "plt.plot(ns, mean_3, marker = 'o', color='b', label=\"3 layer network\")\n",
    "plt.fill_between(ns, mean_3 - 2*std_3, mean_3 + 2*std_3, color = 'b', alpha = 0.15)\n",
    "\n",
    "plt.plot(ns, [0. for n in ns], linestyle ='dashed', color = 'black')\n",
    "\n",
    "plt.xscale(\"log\")\n",
    "plt.xlabel(\"Samples\")\n",
    "plt.ylabel(\"Test Loss\")\n",
    "plt.title(\"$f^*(x) = q(x^TAx), d = 32$\")\n",
    "plt.legend()\n",
    "plt.tight_layout()\n",
    "plt.savefig(\"muP_2_vs_3_final.pdf\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "jax",
   "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.8.13"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
