{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": [],
      "machine_shape": "hm",
      "collapsed_sections": [
        "TX7J3qgw7HmS"
      ],
      "toc_visible": true,
      "gpuType": "L4"
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    },
    "accelerator": "GPU",
    "widgets": {
      "application/vnd.jupyter.widget-state+json": {
        "e4e553c8ab254ee4b5d2549f090702e9": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HBoxModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HBoxModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HBoxView",
            "box_style": "",
            "children": [
              "IPY_MODEL_d7ed656e47a742adbef67a45d7004952",
              "IPY_MODEL_1de16e4b8c7649358f6ff317a2700052",
              "IPY_MODEL_e2d85edc602048edb740e578eb8df681"
            ],
            "layout": "IPY_MODEL_d911c936404b43e0812a81f7b0398260"
          }
        },
        "d7ed656e47a742adbef67a45d7004952": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HTMLModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_7e35db8b08224c4eaefaf5e60019168a",
            "placeholder": "​",
            "style": "IPY_MODEL_d5f3e61b1b0c4ca3bc15b96780336f53",
            "value": "Generating: 100%"
          }
        },
        "1de16e4b8c7649358f6ff317a2700052": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "FloatProgressModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "FloatProgressModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "ProgressView",
            "bar_style": "success",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_cd1bfd535c594826b1e3cfe5b6ed556e",
            "max": 200000,
            "min": 0,
            "orientation": "horizontal",
            "style": "IPY_MODEL_47a4b971ac3344048da931f00225f651",
            "value": 200000
          }
        },
        "e2d85edc602048edb740e578eb8df681": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HTMLModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_90f9d3644f9c45b3ad49ab94bb6c5319",
            "placeholder": "​",
            "style": "IPY_MODEL_29f54132454c40b689f7b5932dceeb53",
            "value": " 200000/200000 [00:11&lt;00:00, 31222.49seq/s, workers_done=12]"
          }
        },
        "d911c936404b43e0812a81f7b0398260": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "7e35db8b08224c4eaefaf5e60019168a": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "d5f3e61b1b0c4ca3bc15b96780336f53": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "DescriptionStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "cd1bfd535c594826b1e3cfe5b6ed556e": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "47a4b971ac3344048da931f00225f651": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "ProgressStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "ProgressStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "bar_color": null,
            "description_width": ""
          }
        },
        "90f9d3644f9c45b3ad49ab94bb6c5319": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "29f54132454c40b689f7b5932dceeb53": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "DescriptionStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "34d608cd88ea49d9a403c3ac5c07868d": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HBoxModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HBoxModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HBoxView",
            "box_style": "",
            "children": [
              "IPY_MODEL_43c2cd0f674f47b791d94b8195b81bdb",
              "IPY_MODEL_bfe33eeaf99b4c4ea4ccd972b7118751",
              "IPY_MODEL_0a45990982c04d9a8c0bf171eb632bd6"
            ],
            "layout": "IPY_MODEL_3de5bdc977354efb8f68b35437e2d2b5"
          }
        },
        "43c2cd0f674f47b791d94b8195b81bdb": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HTMLModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_542ee0fc026c431e84b4c7fad0e8ccff",
            "placeholder": "​",
            "style": "IPY_MODEL_646c0e4a2e8f4de6a70f01b33c77deea",
            "value": "Generating splits...: 100%"
          }
        },
        "bfe33eeaf99b4c4ea4ccd972b7118751": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "FloatProgressModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "FloatProgressModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "ProgressView",
            "bar_style": "",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_2b7fc0a5378b472e816b91c79d03fbcb",
            "max": 1,
            "min": 0,
            "orientation": "horizontal",
            "style": "IPY_MODEL_2485519b99ac4293b7ca22199b760b00",
            "value": 1
          }
        },
        "0a45990982c04d9a8c0bf171eb632bd6": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HTMLModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_7381f6500f904c44b906c5a06c0c7562",
            "placeholder": "​",
            "style": "IPY_MODEL_bc74d578734946b1a4df1e92169f3f2a",
            "value": " 1/1 [00:10&lt;00:00, 10.97s/ splits]"
          }
        },
        "3de5bdc977354efb8f68b35437e2d2b5": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": "hidden",
            "width": null
          }
        },
        "542ee0fc026c431e84b4c7fad0e8ccff": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "646c0e4a2e8f4de6a70f01b33c77deea": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "DescriptionStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "2b7fc0a5378b472e816b91c79d03fbcb": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "2485519b99ac4293b7ca22199b760b00": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "ProgressStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "ProgressStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "bar_color": null,
            "description_width": ""
          }
        },
        "7381f6500f904c44b906c5a06c0c7562": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "bc74d578734946b1a4df1e92169f3f2a": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "DescriptionStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "414493525d8e44db83eba6be77239889": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HBoxModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HBoxModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HBoxView",
            "box_style": "",
            "children": [
              "IPY_MODEL_f60c451dfb8b468b9590ac52df0e85eb",
              "IPY_MODEL_8da6911a3c5649979c18ce2172d83eee",
              "IPY_MODEL_b8d7fc8bbae44bbc8b471f6b6aacf5e2"
            ],
            "layout": "IPY_MODEL_3010a10fe5064e6cb48c380d1dcd7909"
          }
        },
        "f60c451dfb8b468b9590ac52df0e85eb": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HTMLModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_7b2ae67bc966499f845a49b71c0e9788",
            "placeholder": "​",
            "style": "IPY_MODEL_f0a6fd867f604ab1a7d5cce66449d0ac",
            "value": "Generating train examples...: "
          }
        },
        "8da6911a3c5649979c18ce2172d83eee": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "FloatProgressModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "FloatProgressModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "ProgressView",
            "bar_style": "info",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_955c082d66354e3fa52385c87bf8ef43",
            "max": 1,
            "min": 0,
            "orientation": "horizontal",
            "style": "IPY_MODEL_49c1a85dad0f42068928328faeccb051",
            "value": 1
          }
        },
        "b8d7fc8bbae44bbc8b471f6b6aacf5e2": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HTMLModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_d21a5e2a32e04c238fe9fe3d7b5c2c4c",
            "placeholder": "​",
            "style": "IPY_MODEL_3e8365bf757d4a768b4b2cda3fcae7d3",
            "value": " 175579/? [00:07&lt;00:00, 26203.95 examples/s]"
          }
        },
        "3010a10fe5064e6cb48c380d1dcd7909": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": "hidden",
            "width": null
          }
        },
        "7b2ae67bc966499f845a49b71c0e9788": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "f0a6fd867f604ab1a7d5cce66449d0ac": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "DescriptionStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "955c082d66354e3fa52385c87bf8ef43": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": "20px"
          }
        },
        "49c1a85dad0f42068928328faeccb051": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "ProgressStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "ProgressStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "bar_color": null,
            "description_width": ""
          }
        },
        "d21a5e2a32e04c238fe9fe3d7b5c2c4c": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "3e8365bf757d4a768b4b2cda3fcae7d3": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "DescriptionStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "cbb62b1310114fc38af946d0ff2afcfb": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HBoxModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HBoxModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HBoxView",
            "box_style": "",
            "children": [
              "IPY_MODEL_0be8ac04651e4bc5b5a8d7c4e12c4d70",
              "IPY_MODEL_3d801cd090d04e55b04b68f912c3525f",
              "IPY_MODEL_7804334613044b948af9e2dfad6ec3e4"
            ],
            "layout": "IPY_MODEL_649fdba4fbc64a75811f3459abbd0717"
          }
        },
        "0be8ac04651e4bc5b5a8d7c4e12c4d70": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HTMLModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_e419bc4aa1034c1e90f5e20a40e0c40c",
            "placeholder": "​",
            "style": "IPY_MODEL_2956579dabd74c62bde5cb644d4ef31f",
            "value": "Shuffling /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/empirical_binary/incomplete.80OE0H_1.0.0/empirical_binary-train.array_record*...: 100%"
          }
        },
        "3d801cd090d04e55b04b68f912c3525f": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "FloatProgressModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "FloatProgressModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "ProgressView",
            "bar_style": "",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_63693aed932f400fa437b1e1c20f1f1c",
            "max": 200000,
            "min": 0,
            "orientation": "horizontal",
            "style": "IPY_MODEL_e55c41332a98448f98284fbef59bae14",
            "value": 200000
          }
        },
        "7804334613044b948af9e2dfad6ec3e4": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "HTMLModel",
          "model_module_version": "1.5.0",
          "state": {
            "_dom_classes": [],
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "HTMLModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/controls",
            "_view_module_version": "1.5.0",
            "_view_name": "HTMLView",
            "description": "",
            "description_tooltip": null,
            "layout": "IPY_MODEL_87c62c2fe5f34c04b1ebfa5910c7e26b",
            "placeholder": "​",
            "style": "IPY_MODEL_b2564e5d26f14e559fcf8bccf5b26d56",
            "value": " 200000/200000 [00:03&lt;00:00, 68082.15 examples/s]"
          }
        },
        "649fdba4fbc64a75811f3459abbd0717": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": "hidden",
            "width": null
          }
        },
        "e419bc4aa1034c1e90f5e20a40e0c40c": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "2956579dabd74c62bde5cb644d4ef31f": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "DescriptionStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        },
        "63693aed932f400fa437b1e1c20f1f1c": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "e55c41332a98448f98284fbef59bae14": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "ProgressStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "ProgressStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "bar_color": null,
            "description_width": ""
          }
        },
        "87c62c2fe5f34c04b1ebfa5910c7e26b": {
          "model_module": "@jupyter-widgets/base",
          "model_name": "LayoutModel",
          "model_module_version": "1.2.0",
          "state": {
            "_model_module": "@jupyter-widgets/base",
            "_model_module_version": "1.2.0",
            "_model_name": "LayoutModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "LayoutView",
            "align_content": null,
            "align_items": null,
            "align_self": null,
            "border": null,
            "bottom": null,
            "display": null,
            "flex": null,
            "flex_flow": null,
            "grid_area": null,
            "grid_auto_columns": null,
            "grid_auto_flow": null,
            "grid_auto_rows": null,
            "grid_column": null,
            "grid_gap": null,
            "grid_row": null,
            "grid_template_areas": null,
            "grid_template_columns": null,
            "grid_template_rows": null,
            "height": null,
            "justify_content": null,
            "justify_items": null,
            "left": null,
            "margin": null,
            "max_height": null,
            "max_width": null,
            "min_height": null,
            "min_width": null,
            "object_fit": null,
            "object_position": null,
            "order": null,
            "overflow": null,
            "overflow_x": null,
            "overflow_y": null,
            "padding": null,
            "right": null,
            "top": null,
            "visibility": null,
            "width": null
          }
        },
        "b2564e5d26f14e559fcf8bccf5b26d56": {
          "model_module": "@jupyter-widgets/controls",
          "model_name": "DescriptionStyleModel",
          "model_module_version": "1.5.0",
          "state": {
            "_model_module": "@jupyter-widgets/controls",
            "_model_module_version": "1.5.0",
            "_model_name": "DescriptionStyleModel",
            "_view_count": null,
            "_view_module": "@jupyter-widgets/base",
            "_view_module_version": "1.2.0",
            "_view_name": "StyleView",
            "description_width": ""
          }
        }
      }
    }
  },
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "nEUBka2LJGMz",
        "outputId": "56810b94-ff09-4ee4-873b-21192d999d44",
        "colab": {
          "base_uri": "https://localhost:8080/"
        }
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "fatal: destination path 'nanodo' already exists and is not an empty directory.\n"
          ]
        }
      ],
      "source": [
        "!git clone https://github.com/google-deepmind/nanodo.git"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "from google.colab import drive\n",
        "drive.mount('/content/drive')"
      ],
      "metadata": {
        "id": "ju3u7x_oMOvq",
        "collapsed": true,
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "7cdaeb56-ea11-470e-8ada-26739834693e"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Mounted at /content/drive\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "!pip install -U \"jax[cuda12_pip]==0.4.35\" \\\n",
        "  -f https://storage.googleapis.com/jax-releases/jax_cuda_releases.html \\\n",
        "  orbax-checkpoint==0.6.0 \\\n",
        "  flax==0.8.5 \\\n",
        "  optax==0.2.2 \\\n",
        "  clu==0.0.12 \\\n",
        "  grain==0.2.1 \\\n",
        "  tensorflow-datasets\n"
      ],
      "metadata": {
        "id": "_XC9cAiEmmGY",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "f102c11f-22d8-4134-af40-668b8dca565e",
        "collapsed": true
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Looking in links: https://storage.googleapis.com/jax-releases/jax_cuda_releases.html\n",
            "Requirement already satisfied: jax==0.4.35 in /usr/local/lib/python3.12/dist-packages (from jax[cuda12_pip]==0.4.35) (0.4.35)\n",
            "Requirement already satisfied: orbax-checkpoint==0.6.0 in /usr/local/lib/python3.12/dist-packages (0.6.0)\n",
            "Requirement already satisfied: flax==0.8.5 in /usr/local/lib/python3.12/dist-packages (0.8.5)\n",
            "Requirement already satisfied: optax==0.2.2 in /usr/local/lib/python3.12/dist-packages (0.2.2)\n",
            "Requirement already satisfied: clu==0.0.12 in /usr/local/lib/python3.12/dist-packages (0.0.12)\n",
            "Requirement already satisfied: grain==0.2.1 in /usr/local/lib/python3.12/dist-packages (0.2.1)\n",
            "Requirement already satisfied: tensorflow-datasets in /usr/local/lib/python3.12/dist-packages (4.9.9)\n",
            "Requirement already satisfied: jaxlib<=0.4.35,>=0.4.34 in /usr/local/lib/python3.12/dist-packages (from jax==0.4.35->jax[cuda12_pip]==0.4.35) (0.4.34)\n",
            "Requirement already satisfied: ml-dtypes>=0.4.0 in /usr/local/lib/python3.12/dist-packages (from jax==0.4.35->jax[cuda12_pip]==0.4.35) (0.5.3)\n",
            "Requirement already satisfied: numpy>=1.24 in /usr/local/lib/python3.12/dist-packages (from jax==0.4.35->jax[cuda12_pip]==0.4.35) (2.0.2)\n",
            "Requirement already satisfied: opt-einsum in /usr/local/lib/python3.12/dist-packages (from jax==0.4.35->jax[cuda12_pip]==0.4.35) (3.4.0)\n",
            "Requirement already satisfied: scipy>=1.10 in /usr/local/lib/python3.12/dist-packages (from jax==0.4.35->jax[cuda12_pip]==0.4.35) (1.16.2)\n",
            "Requirement already satisfied: absl-py in /usr/local/lib/python3.12/dist-packages (from orbax-checkpoint==0.6.0) (1.4.0)\n",
            "Requirement already satisfied: etils[epath,epy] in /usr/local/lib/python3.12/dist-packages (from orbax-checkpoint==0.6.0) (1.13.0)\n",
            "Requirement already satisfied: typing_extensions in /usr/local/lib/python3.12/dist-packages (from orbax-checkpoint==0.6.0) (4.15.0)\n",
            "Requirement already satisfied: msgpack in /usr/local/lib/python3.12/dist-packages (from orbax-checkpoint==0.6.0) (1.1.1)\n",
            "Requirement already satisfied: pyyaml in /usr/local/lib/python3.12/dist-packages (from orbax-checkpoint==0.6.0) (6.0.2)\n",
            "Requirement already satisfied: tensorstore>=0.1.60 in /usr/local/lib/python3.12/dist-packages (from orbax-checkpoint==0.6.0) (0.1.76)\n",
            "Requirement already satisfied: nest_asyncio in /usr/local/lib/python3.12/dist-packages (from orbax-checkpoint==0.6.0) (1.6.0)\n",
            "Requirement already satisfied: protobuf in /usr/local/lib/python3.12/dist-packages (from orbax-checkpoint==0.6.0) (5.29.5)\n",
            "Requirement already satisfied: humanize in /usr/local/lib/python3.12/dist-packages (from orbax-checkpoint==0.6.0) (4.13.0)\n",
            "Requirement already satisfied: rich>=11.1 in /usr/local/lib/python3.12/dist-packages (from flax==0.8.5) (13.9.4)\n",
            "Requirement already satisfied: chex>=0.1.86 in /usr/local/lib/python3.12/dist-packages (from optax==0.2.2) (0.1.90)\n",
            "Requirement already satisfied: ml-collections in /usr/local/lib/python3.12/dist-packages (from clu==0.0.12) (1.1.0)\n",
            "Requirement already satisfied: packaging in /usr/local/lib/python3.12/dist-packages (from clu==0.0.12) (25.0)\n",
            "Requirement already satisfied: wrapt in /usr/local/lib/python3.12/dist-packages (from clu==0.0.12) (1.17.3)\n",
            "Requirement already satisfied: array-record in /usr/local/lib/python3.12/dist-packages (from grain==0.2.1) (0.8.1)\n",
            "Requirement already satisfied: cloudpickle in /usr/local/lib/python3.12/dist-packages (from grain==0.2.1) (3.1.1)\n",
            "Requirement already satisfied: dm-tree in /usr/local/lib/python3.12/dist-packages (from grain==0.2.1) (0.1.9)\n",
            "Requirement already satisfied: jaxtyping in /usr/local/lib/python3.12/dist-packages (from grain==0.2.1) (0.3.2)\n",
            "Requirement already satisfied: more-itertools>=9.1.0 in /usr/local/lib/python3.12/dist-packages (from grain==0.2.1) (10.8.0)\n",
            "Requirement already satisfied: jax-cuda12-plugin<=0.4.35,>=0.4.34 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (0.4.35)\n",
            "Requirement already satisfied: immutabledict in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (4.2.1)\n",
            "Requirement already satisfied: promise in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (2.3)\n",
            "Requirement already satisfied: psutil in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (5.9.5)\n",
            "Requirement already satisfied: pyarrow in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (18.1.0)\n",
            "Requirement already satisfied: requests>=2.19.0 in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (2.32.4)\n",
            "Requirement already satisfied: simple_parsing in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (0.1.7)\n",
            "Requirement already satisfied: tensorflow-metadata in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (1.17.2)\n",
            "Requirement already satisfied: termcolor in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (3.1.0)\n",
            "Requirement already satisfied: toml in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (0.10.2)\n",
            "Requirement already satisfied: tqdm in /usr/local/lib/python3.12/dist-packages (from tensorflow-datasets) (4.67.1)\n",
            "Requirement already satisfied: setuptools in /usr/local/lib/python3.12/dist-packages (from chex>=0.1.86->optax==0.2.2) (75.2.0)\n",
            "Requirement already satisfied: toolz>=0.9.0 in /usr/local/lib/python3.12/dist-packages (from chex>=0.1.86->optax==0.2.2) (0.12.1)\n",
            "Requirement already satisfied: einops in /usr/local/lib/python3.12/dist-packages (from etils[edc,enp,epath,epy,etree]>=1.9.1; python_version >= \"3.11\"->tensorflow-datasets) (0.8.1)\n",
            "Requirement already satisfied: fsspec in /usr/local/lib/python3.12/dist-packages (from etils[edc,enp,epath,epy,etree]>=1.9.1; python_version >= \"3.11\"->tensorflow-datasets) (2025.3.0)\n",
            "Requirement already satisfied: importlib_resources in /usr/local/lib/python3.12/dist-packages (from etils[edc,enp,epath,epy,etree]>=1.9.1; python_version >= \"3.11\"->tensorflow-datasets) (6.5.2)\n",
            "Requirement already satisfied: zipp in /usr/local/lib/python3.12/dist-packages (from etils[edc,enp,epath,epy,etree]>=1.9.1; python_version >= \"3.11\"->tensorflow-datasets) (3.23.0)\n",
            "Requirement already satisfied: jax-cuda12-pjrt==0.4.35 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin<=0.4.35,>=0.4.34->jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (0.4.35)\n",
            "Requirement already satisfied: nvidia-cublas-cu12>=12.1.3.1 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (12.6.4.1)\n",
            "Requirement already satisfied: nvidia-cuda-cupti-cu12>=12.1.105 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (12.6.80)\n",
            "Requirement already satisfied: nvidia-cuda-nvcc-cu12>=12.1.105 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (12.5.82)\n",
            "Requirement already satisfied: nvidia-cuda-runtime-cu12>=12.1.105 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (12.6.77)\n",
            "Requirement already satisfied: nvidia-cudnn-cu12<10.0,>=9.1 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (9.10.2.21)\n",
            "Requirement already satisfied: nvidia-cufft-cu12>=11.0.2.54 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (11.3.0.4)\n",
            "Requirement already satisfied: nvidia-cusolver-cu12>=11.4.5.107 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (11.7.1.2)\n",
            "Requirement already satisfied: nvidia-cusparse-cu12>=12.1.0.106 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (12.5.4.2)\n",
            "Requirement already satisfied: nvidia-nccl-cu12>=2.18.1 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (2.27.3)\n",
            "Requirement already satisfied: nvidia-nvjitlink-cu12>=12.1.105 in /usr/local/lib/python3.12/dist-packages (from jax-cuda12-plugin[with_cuda]<=0.4.35,>=0.4.34; extra == \"cuda12-pip\"->jax[cuda12_pip]==0.4.35) (12.6.85)\n",
            "Requirement already satisfied: charset_normalizer<4,>=2 in /usr/local/lib/python3.12/dist-packages (from requests>=2.19.0->tensorflow-datasets) (3.4.3)\n",
            "Requirement already satisfied: idna<4,>=2.5 in /usr/local/lib/python3.12/dist-packages (from requests>=2.19.0->tensorflow-datasets) (3.10)\n",
            "Requirement already satisfied: urllib3<3,>=1.21.1 in /usr/local/lib/python3.12/dist-packages (from requests>=2.19.0->tensorflow-datasets) (2.5.0)\n",
            "Requirement already satisfied: certifi>=2017.4.17 in /usr/local/lib/python3.12/dist-packages (from requests>=2.19.0->tensorflow-datasets) (2025.8.3)\n",
            "Requirement already satisfied: markdown-it-py>=2.2.0 in /usr/local/lib/python3.12/dist-packages (from rich>=11.1->flax==0.8.5) (4.0.0)\n",
            "Requirement already satisfied: pygments<3.0.0,>=2.13.0 in /usr/local/lib/python3.12/dist-packages (from rich>=11.1->flax==0.8.5) (2.19.2)\n",
            "Requirement already satisfied: attrs>=18.2.0 in /usr/local/lib/python3.12/dist-packages (from dm-tree->grain==0.2.1) (25.3.0)\n",
            "Requirement already satisfied: wadler-lindig>=0.1.3 in /usr/local/lib/python3.12/dist-packages (from jaxtyping->grain==0.2.1) (0.1.7)\n",
            "Requirement already satisfied: six in /usr/local/lib/python3.12/dist-packages (from promise->tensorflow-datasets) (1.17.0)\n",
            "Requirement already satisfied: docstring-parser<1.0,>=0.15 in /usr/local/lib/python3.12/dist-packages (from simple_parsing->tensorflow-datasets) (0.17.0)\n",
            "Requirement already satisfied: googleapis-common-protos<2,>=1.56.4 in /usr/local/lib/python3.12/dist-packages (from tensorflow-metadata->tensorflow-datasets) (1.70.0)\n",
            "Requirement already satisfied: mdurl~=0.1 in /usr/local/lib/python3.12/dist-packages (from markdown-it-py>=2.2.0->rich>=11.1->flax==0.8.5) (0.1.2)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "from google.colab import drive\n",
        "import sys\n",
        "import os\n",
        "sys.path.append(\"/content/nanodo\")"
      ],
      "metadata": {
        "id": "Ba3slj9aMUwB"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "print(sys.path)\n",
        "print(os.getcwd())     # where you are\n"
      ],
      "metadata": {
        "id": "DKDJyFLv1sY5",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "8da6f402-b0ed-46b3-ff90-aaf823a572e6"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "['/content', '/env/python', '/usr/lib/python312.zip', '/usr/lib/python3.12', '/usr/lib/python3.12/lib-dynload', '', '/usr/local/lib/python3.12/dist-packages', '/usr/lib/python3/dist-packages', '/usr/local/lib/python3.12/dist-packages/IPython/extensions', '/root/.ipython', '/content/nanodo']\n",
            "/content\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Utils"
      ],
      "metadata": {
        "id": "TX7J3qgw7HmS"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "eps = 1e-6\n",
        "ALPHA = 1.5\n",
        "\n",
        "PAD_ID = 0         # keep!\n",
        "BOS_ID = 1\n",
        "# EOS_ID = 2\n",
        "ZERO_ID = 2\n",
        "\n",
        "\n",
        "ONE_ID  = 3\n",
        "\n",
        "VOCAB_SIZE = 4      # convenience constant"
      ],
      "metadata": {
        "id": "n95aDz061uKf"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Copyright 2024 DeepMind Technologies Limited.\n",
        "#\n",
        "# Licensed under the Apache License, Version 2.0 (the \"License\");\n",
        "# you may not use this file except in compliance with the License.\n",
        "# You may obtain a copy of the License at\n",
        "#\n",
        "# https://www.apache.org/licenses/LICENSE-2.0\n",
        "#\n",
        "# Unless required by applicable law or agreed to in writing, software\n",
        "# distributed under the License is distributed on an \"AS IS\" BASIS,\n",
        "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n",
        "# See the License for the specific language governing permissions and\n",
        "# limitations under the License.\n",
        "\"\"\"Transformer Decoder-only model.\"\"\"\n",
        "\n",
        "# pylint: disable=g-importing-member\n",
        "# pylint: disable=invalid-name\n",
        "\n",
        "import dataclasses\n",
        "from functools import partial\n",
        "\n",
        "from flax import linen as nn\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "from nanodo import fsdp\n",
        "\n",
        "\n",
        "@dataclasses.dataclass\n",
        "class DoConfig:\n",
        "  \"\"\"Hyper-parameters for Transformer decoder-only.\"\"\"\n",
        "  D: int  # model/embed dim  = qkv dim\n",
        "  H: int  # num attention heads\n",
        "  L: int  # max context/sequence length (move out of config?)\n",
        "  N: int  # number of transformer block layers\n",
        "  V: int  # vocab size\n",
        "  F: int  # FF inner dimension\n",
        "  kernel_init: nn.initializers.Initializer = nn.initializers.xavier_uniform()\n",
        "  embed_init: nn.initializers.Initializer = nn.initializers.variance_scaling(\n",
        "      1.0, 'fan_in', 'normal', out_axis=0)\n",
        "  dtype: jnp.dtype = jnp.float32\n",
        "  fsdp_enabled: bool = True\n",
        "\n",
        "  # Transformer block rematerialization / gradient checkpointing to save memory.\n",
        "  remat: bool = False\n",
        "\n",
        "\n",
        "class TransformerDo(nn.Module):\n",
        "  \"\"\"Transformer decoder-only.\"\"\"\n",
        "  docfg: DoConfig\n",
        "\n",
        "  def setup(self):\n",
        "    cfg = self.docfg\n",
        "    self.embed = nn.Embed(\n",
        "        num_embeddings=cfg.V,\n",
        "        features=cfg.D,\n",
        "        embedding_init=fsdp.init('embedding', cfg),\n",
        "        name = \"tok_emb\",\n",
        "    )\n",
        "    self.pos_embed = nn.Embed(\n",
        "        num_embeddings=cfg.L,\n",
        "        features=cfg.D,\n",
        "        embedding_init=fsdp.init('embedding', cfg),\n",
        "        name = \"pos_embed\",\n",
        "        #embedding_init=nn.initializers.variance_scaling(1.0, 'fan_in', 'normal', out_axis=0)\n",
        "    )\n",
        "    block = nn.remat(TBlock) if cfg.remat else TBlock\n",
        "    self.blocks = [block(cfg) for _ in range(cfg.N)]\n",
        "    self.out_ln = nn.LayerNorm(dtype=cfg.dtype, use_bias=False)\n",
        "    self.dense_output = nn.Dense(features=cfg.V, dtype=jnp.float32, name = \"lm_head\")\n",
        "\n",
        "  def __call__(self, y_BxL: jax.Array):\n",
        "    # For training on concatenated examples.\n",
        "    y_BxL = y_BxL.astype(jnp.int32)\n",
        "    y_BxLxD = self.embed(y_BxL)\n",
        "    y_BxLxD += self.pos_embed(jnp.arange(0, y_BxL.shape[1])[None, ...])\n",
        "    for block in self.blocks:\n",
        "      y_BxLxD = block(y_BxLxD)\n",
        "    y_BxLxD = self.out_ln(y_BxLxD)\n",
        "    # Without embedding\n",
        "    # logits_BxLxV = self.dense_output(y_BxLxD)  # Use self.dense_output here\n",
        "\n",
        "    # With embedding\n",
        "    logits_BxLxV = self.embed.attend(y_BxLxD.astype(jnp.float32))\n",
        "\n",
        "    ## require output to be a distribution\n",
        "    probs_BxLxV = jax.nn.softmax(logits_BxLxV, axis=-1)\n",
        "    return probs_BxLxV\n",
        "    #logits_BxLx2 = nn.Dense(features=2, dtype=jnp.float32)(y_BxLxD)  # Binary output\n",
        "    #predicted_tokens_BxL = jnp.argmax(logits_BxLxV, axis=-1)  # <-- Change to predicted_tokens_BxL\n",
        "    # return logits_BxLxV #predicted_tokens_BxL\n",
        "\n",
        "\n",
        "\n",
        "class Mlp(nn.Module):\n",
        "  \"\"\"Multilayer perceptron.\"\"\"\n",
        "  cfg: DoConfig\n",
        "\n",
        "  @nn.compact\n",
        "  def __call__(self, x_BxLxD: jax.Array):\n",
        "    cfg = self.cfg\n",
        "    linear = partial(\n",
        "        nn.Dense, kernel_init=fsdp.init('mlp_kernel', cfg), use_bias=False,\n",
        "        dtype=cfg.dtype\n",
        "    )\n",
        "    x_BxLxF = linear(cfg.F)(x_BxLxD)\n",
        "    x_BxLxF = jax.nn.gelu(x_BxLxF)\n",
        "    x_BxLxD = linear(cfg.D)(x_BxLxF)\n",
        "    return x_BxLxD\n",
        "\n",
        "\n",
        "class TBlock(nn.Module):\n",
        "  \"\"\"Transformer Block.\"\"\"\n",
        "  docfg: DoConfig\n",
        "\n",
        "  @nn.compact\n",
        "  def __call__(self, in_BxLxD: jax.Array):\n",
        "    cfg = self.docfg\n",
        "\n",
        "    # \"pre-layernorm\"\n",
        "    x_BxLxD = nn.LayerNorm(dtype=cfg.dtype, use_bias=False)(in_BxLxD)\n",
        "    x_BxLxD = CausalAttn(cfg)(x_BxLxD)\n",
        "    x_BxLxD += in_BxLxD\n",
        "\n",
        "    z_BxLxD = nn.LayerNorm(dtype=cfg.dtype, use_bias=False)(x_BxLxD)\n",
        "    z_BxLxD = Mlp(cfg)(z_BxLxD)\n",
        "\n",
        "    return x_BxLxD + z_BxLxD\n",
        "\n",
        "\n",
        "class CausalAttn(nn.Module):\n",
        "  \"\"\"Causal attention layer.\"\"\"\n",
        "  cfg: DoConfig\n",
        "\n",
        "  @nn.compact\n",
        "  def __call__(self, x_BxLxD: jax.Array):\n",
        "    cfg = self.cfg\n",
        "\n",
        "    assert cfg.D % cfg.H == 0, f'D {cfg.D} not divisible by H {cfg.H}'\n",
        "    Dh = cfg.D // cfg.H\n",
        "\n",
        "    # Maps D -> (H, Dh)\n",
        "    multilinear = partial(\n",
        "        nn.DenseGeneral,\n",
        "        axis=-1,\n",
        "        features=(cfg.H, Dh),\n",
        "        kernel_init=fsdp.init('attn_in_proj', cfg),\n",
        "        use_bias=False,\n",
        "        dtype=cfg.dtype,\n",
        "    )\n",
        "\n",
        "    q_BxLxHxDh, k_BxLxHxDh, v_BxLxHxDh = (\n",
        "        multilinear(name='query')(x_BxLxD),\n",
        "        multilinear(name='key')(x_BxLxD),\n",
        "        multilinear(name='value')(x_BxLxD),\n",
        "    )\n",
        "    q_BxLxHxDh /= Dh**0.5\n",
        "    att_BxHxLxL = jnp.einsum('...qhd,...khd->...hqk', q_BxLxHxDh, k_BxLxHxDh)\n",
        "    # cast to fp32 for softmax\n",
        "    att_BxHxLxL = att_BxHxLxL.astype(jnp.float32)\n",
        "\n",
        "    # causal attention mask\n",
        "    L = x_BxLxD.shape[1]\n",
        "    mask_1x1xLxL = jnp.tril(jnp.ones((1, 1, L, L), dtype=jnp.bool_))\n",
        "\n",
        "    _NEG_INF = jnp.finfo(cfg.dtype).min\n",
        "    att_BxHxLxL = jnp.where(mask_1x1xLxL, att_BxHxLxL, _NEG_INF)\n",
        "    att_BxHxLxL = jax.nn.softmax(att_BxHxLxL, axis=-1)\n",
        "    att_BxHxLxL = att_BxHxLxL.astype(cfg.dtype)\n",
        "    out_BxLxHxDh = jnp.einsum('...hqk,...khd->...qhd', att_BxHxLxL, v_BxLxHxDh)\n",
        "    # Output projection followed by contraction back to original dims\n",
        "    out_BxLxD = nn.DenseGeneral(\n",
        "        features=cfg.D,\n",
        "        name='attn_out_proj',\n",
        "        axis=(-2, -1),\n",
        "        kernel_init=fsdp.init('attn_out_proj', cfg),\n",
        "        use_bias=False,\n",
        "        dtype=cfg.dtype,\n",
        "    )(out_BxLxHxDh)\n",
        "    return out_BxLxD\n"
      ],
      "metadata": {
        "id": "C8tjmAPbYrH8"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "def preprocess_binary_string(binary_string):\n",
        "    \"\"\"Converts a binary string to a numerical array.\"\"\"\n",
        "    return jnp.array([int(bit) for bit in binary_string])"
      ],
      "metadata": {
        "id": "Yju467BqKx2Q"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Configuration\n",
        "docfg = DoConfig(D = 32, H=4, V = 5, L=10, N=2, F=64, dtype=jnp.float32)\n",
        "\n",
        "# Initialize the model\n",
        "key = jax.random.PRNGKey(45)  # Random key for initialization\n",
        "model = TransformerDo(docfg)\n",
        "\n",
        "# Create dummy input with correct shape\n",
        "dummy_input = jnp.ones((1, 7, 1), dtype=jnp.int32)\n",
        "\n",
        "params = model.init(key, dummy_input)['params']  # Initialize with dummy input\n",
        "\n",
        "# Input binary string\n",
        "binary_string = \"01234111\"\n",
        "input_array = preprocess_binary_string(binary_string)[None, ...]  # Add batch dimension\n",
        "\n",
        "\n",
        "# Apply the model\n",
        "output = model.apply({'params': params}, input_array)\n",
        "\n",
        "print(\"output array:\")\n",
        "# Print the output\n",
        "print(output)"
      ],
      "metadata": {
        "id": "qvbcbEtZbEK1",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "883afd9c-3d86-44f7-fb88-2e49a9fadc80"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "output array:\n",
            "[[[0.14667867 0.45827392 0.13339897 0.22950353 0.03214497]\n",
            "  [0.10793231 0.44728756 0.06730265 0.30312526 0.07435232]\n",
            "  [0.05950737 0.09133135 0.6206208  0.10320667 0.1253339 ]\n",
            "  [0.04484144 0.07518799 0.71766144 0.12630464 0.03600451]\n",
            "  [0.14590712 0.04328534 0.10069599 0.12755546 0.58255607]\n",
            "  [0.10134835 0.24420965 0.35561073 0.17971648 0.11911482]\n",
            "  [0.02935442 0.09212374 0.7719167  0.06706459 0.03954056]\n",
            "  [0.04451034 0.2828079  0.4341601  0.12624532 0.11227641]]]\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "import jax\n",
        "import jax.numpy as jnp\n",
        "\n",
        "# IDs for your vocabulary\n",
        "\n",
        "def get_in_out_with_regret_mask(\n",
        "    in_BxL: jax.Array,\n",
        "    pad_id: int = PAD_ID,\n",
        "):\n",
        "  \"\"\"Returns input, output, and weights with extra masking of the prefix 1..t*.\n",
        "\n",
        "  Rule: find the first t* (counted over bit tokens only) such that\n",
        "        regret_t >= alpha / sqrt(t) against the piecewise-constant reference\n",
        "        p_ref(t) = 1/3 for t <= floor(T/2), 2/3 for t > floor(T/2),\n",
        "        where T is the total number of bit tokens in the example.\n",
        "  Then set weight=0 for those positions within the offending prefix (inclusive).\n",
        "  \"\"\"\n",
        "  # --- your original shifting ---\n",
        "  x_BxL = in_BxL\n",
        "  y_BxL = jnp.pad(in_BxL[:, 1:], ((0, 0), (0, 1)), mode=\"constant\", constant_values=pad_id)\n",
        "  base_weights_BxL = (y_BxL != pad_id).astype(jnp.float32)\n",
        "\n",
        "  B, L = y_BxL.shape\n",
        "\n",
        "  # Identify bit tokens (only 0/1 should count toward t); ignore BOS/EOS/etc.\n",
        "  is_zero = (y_BxL == ZERO_ID)\n",
        "  is_one  = (y_BxL == ONE_ID)\n",
        "  is_bit  = is_zero | is_one\n",
        "\n",
        "  # t index counts ONLY bit tokens: t=1,2,... per example\n",
        "  t_idx = jnp.cumsum(is_bit.astype(jnp.int32), axis=1)  # (B, L), 0 where not a bit\n",
        "  T_total = jnp.max(t_idx, axis=1)                      # (B,) number of bit tokens\n",
        "  # half point per example\n",
        "  half = (T_total // 2)                                  # (B,)\n",
        "\n",
        "  # Reference p_ref(t): 1/3 for t <= half, else 2/3 (broadcast over positions)\n",
        "  # For non-bit positions (t_idx==0) we don't care; set p_ref arbitrarily there.\n",
        "  t_idx_f = t_idx.astype(jnp.float32)\n",
        "  half_b  = half[:, None].astype(jnp.float32)\n",
        "  p_ref   = jnp.where(t_idx_f > 0, jnp.where(t_idx_f <= half_b, 1.0/3.0, 2.0/3.0), 0.0)  # (B, L)\n",
        "\n",
        "  # Observed bits b in {0,1} (0 for non-bit positions too; they’ll be masked out)\n",
        "  b = is_one.astype(jnp.float32)  # (B, L)\n",
        "\n",
        "  # Per-step mismatch against p_ref\n",
        "  mismatch = jnp.where(b == 1.0, 1.0 - p_ref, p_ref) * is_bit.astype(jnp.float32)  # (B, L)\n",
        "\n",
        "  # Cumulative quantities over positions; non-bits add 0\n",
        "  ones_cum      = jnp.cumsum(b, axis=1)                  # (B, L)\n",
        "  cum_mismatch  = jnp.cumsum(mismatch, axis=1)           # (B, L)\n",
        "  # Best constant action up to t is min(#1s, t-#1s), but only defined on bit positions\n",
        "  best_const = jnp.minimum(ones_cum, t_idx_f - ones_cum) # (B, L)\n",
        "\n",
        "  regret = cum_mismatch - best_const                     # (B, L)\n",
        "\n",
        "  # Threshold α / sqrt(t); set to +inf where t==0 so it never triggers on non-bit steps\n",
        "  inv_sqrt_t = jnp.where(t_idx_f > 0, 1.0 / jnp.sqrt(t_idx_f), jnp.inf)\n",
        "  thresh = ALPHA * inv_sqrt_t\n",
        "\n",
        "  # Trigger where regret >= threshold on bit positions\n",
        "  triggered = (regret >= thresh) & (t_idx_f > 0)\n",
        "\n",
        "  # First violating position (argwhere True → first index); if none, use -1 sentinel\n",
        "  first_idx = jnp.argmax(triggered, axis=1)              # (B,)\n",
        "  has_any   = jnp.any(triggered, axis=1)                 # (B,)\n",
        "  # If no violation -> set index to -1 so we zero out nothing below\n",
        "  first_idx = jnp.where(has_any, first_idx, -jnp.ones_like(first_idx))\n",
        "\n",
        "  # Build \"zero this prefix (inclusive) on bit positions\" mask\n",
        "  pos = jnp.arange(L)[None, :]                           # (1, L)\n",
        "  zero_prefix = (pos <= first_idx[:, None]) & has_any[:, None]  # (B, L)\n",
        "  # Only apply to bit positions; leave non-bit weights unchanged by this rule\n",
        "  extra_mask = jnp.where(zero_prefix & is_bit, 0.0, 1.0).astype(jnp.float32)\n",
        "\n",
        "  # Final weights: original non-pad mask, further zeroed by regret rule on offending prefix\n",
        "  weights_BxL = base_weights_BxL * extra_mask\n",
        "  print(weights_BxL)\n",
        "\n",
        "  return x_BxL, y_BxL, weights_BxL\n",
        "\n",
        "def get_in_out(\n",
        "  in_BxL: jax.Array,\n",
        "  pad_id: int = PAD_ID,\n",
        ") -> tuple[jax.Array, jax.Array, jax.Array]:\n",
        "    \"\"\"Returns input, output, and weights for a batch of examples.\"\"\"\n",
        "    # Assumes input of the form <BOS> <IDs> <EOS> for eval.\n",
        "    x_BxL = in_BxL\n",
        "    y_BxL = jnp.pad(\n",
        "        in_BxL[:, 1:],\n",
        "        ((0, 0), (0, 1)),\n",
        "        mode='constant',\n",
        "        constant_values=pad_id,\n",
        "    )\n",
        "    weights_BxL = jnp.where(y_BxL != pad_id, 1, 0).astype(jnp.float32)\n",
        "\n",
        "    return x_BxL, y_BxL, weights_BxL\n",
        "\n"
      ],
      "metadata": {
        "id": "7t68d-6a69gJ"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Copyright 2024 DeepMind Technologies Limited.\n",
        "#\n",
        "# Licensed under the Apache License, Version 2.0 (the \"License\");\n",
        "# you may not use this file except in compliance with the License.\n",
        "# You may obtain a copy of the License at\n",
        "#\n",
        "# https://www.apache.org/licenses/LICENSE-2.0\n",
        "#\n",
        "# Unless required by applicable law or agreed to in writing, software\n",
        "# distributed under the License is distributed on an \"AS IS\" BASIS,\n",
        "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n",
        "# See the License for the specific language governing permissions and\n",
        "# limitations under the License.\n",
        "\"\"\"Loss functions.\"\"\"\n",
        "\n",
        "# pylint: disable=invalid-name,g-import-not-at-top,g-bare-generic\n",
        "\n",
        "from typing import Any, Callable, TYPE_CHECKING\n",
        "\n",
        "from flax.struct import dataclass\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "from optax import losses\n",
        "# from nanodo import data as data_lib\n",
        "\n",
        "if TYPE_CHECKING:\n",
        "  import ml_collections\n",
        "\n",
        "\n",
        "PyTree = Any\n",
        "\n",
        "\n",
        "@dataclass\n",
        "class LossAuxData:\n",
        "  ntokens: jax.Array\n",
        "  state: PyTree\n",
        "  log_perplexity: jax.Array\n",
        "\n",
        "# loss(params) function to be used in `jax.value_and_grad`.\n",
        "LossFn = Callable[[PyTree], tuple[jax.Array, LossAuxData]]\n",
        "\n",
        "LossFnFactory = Callable[\n",
        "    [jax.Array, Callable, \"ml_collections.ConfigDict\"],\n",
        "    LossFn,\n",
        "]\n",
        "\n",
        "\n",
        "def get_default_loss_fn(\n",
        "    in_BxL: jax.Array,\n",
        "    apply_fn: Callable,\n",
        "    c: \"ml_collections.ConfigDict\",\n",
        ") -> LossFn:\n",
        "  \"\"\"Standard next-token-prediction language modeling loss.\"\"\"\n",
        "  def loss_fn(params: PyTree) -> tuple[jax.Array, LossAuxData]:\n",
        "    # x_BxL, y_BxL, weights_BxL =  get_in_out_with_regret_mask(in_BxL)\n",
        "    x_BxL, y_BxL, weights_BxL = get_in_out(in_BxL)\n",
        "\n",
        "    mutable = (\n",
        "        \"intermediate_acts\",) if c.get(\"log_internal_metrics\", False) else ()\n",
        "    logits_BxLxV, state = apply_fn(\n",
        "        {\"params\": params},\n",
        "        x_BxL,\n",
        "        mutable=mutable,\n",
        "    )\n",
        "\n",
        "    losses_BxL = -jnp.log(\n",
        "        jnp.take_along_axis(logits_BxLxV, y_BxL[..., None], axis=-1).squeeze(-1)\n",
        "    )\n",
        "\n",
        "\n",
        "    # losses_BxL = losses.softmax_cross_entropy_with_integer_labels(\n",
        "    #     logits_BxLxV, y_BxL\n",
        "    # )\n",
        "    ntokens = weights_BxL.sum()\n",
        "    mean_loss = jnp.sum(losses_BxL * weights_BxL) / ntokens\n",
        "    #mean_loss = jnp.sum(losses_BxL * weights_BxL)\n",
        "    return mean_loss, LossAuxData(\n",
        "        ntokens=ntokens, state=state, log_perplexity=mean_loss)\n",
        "\n",
        "  return loss_fn\n",
        "\n",
        "\n"
      ],
      "metadata": {
        "id": "MtDUU2g0iN-B"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "\n",
        "from typing import TYPE_CHECKING\n",
        "#from nanodo import loss as loss_lib\n",
        "\n",
        "\n",
        "if TYPE_CHECKING:\n",
        "  import ml_collections\n",
        "\n",
        "\n",
        "def get_model_and_loss(\n",
        "    c: \"ml_collections.ConfigDict\",\n",
        "    vocab_size: int,\n",
        ") -> tuple[nn.Module, LossFnFactory]:\n",
        "  \"\"\"Returns an instantiated (potentially experimental) model.\"\"\"\n",
        "  # default model and configs\n",
        "#  m = model\n",
        "  get_loss_fn = get_default_loss_fn\n",
        "  #get_loss_fn = get_default_loss_fn\n",
        "\n",
        "  cfg = DoConfig(**c.model)  # pytype:disable=attribute-error\n",
        "  module = TransformerDo(cfg)  # pytype:disable=attribute-error\n",
        "  return module, get_loss_fn"
      ],
      "metadata": {
        "id": "SWU4ZLVqHtiR"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Copyright 2024 DeepMind Technologies Limited.\n",
        "#\n",
        "# Licensed under the Apache License, Version 2.0 (the \"License\");\n",
        "# you may not use this file except in compliance with the License.\n",
        "# You may obtain a copy of the License at\n",
        "#\n",
        "# https://www.apache.org/licenses/LICENSE-2.0\n",
        "#\n",
        "# Unless required by applicable law or agreed to in writing, software\n",
        "# distributed under the License is distributed on an \"AS IS\" BASIS,\n",
        "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n",
        "# See the License for the specific language governing permissions and\n",
        "# limitations under the License.\n",
        "\"\"\"Data pipeline.\"\"\"\n",
        "\n",
        "from collections.abc import Mapping, Sequence\n",
        "import dataclasses\n",
        "import enum\n",
        "import functools\n",
        "from typing import Iterator\n",
        "\n",
        "import grain.python as grain\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "import numpy as np\n",
        "import tensorflow_datasets as tfds\n",
        "\n",
        "\n",
        "\n",
        "\n",
        "### pure python helpers for use with grain ###\n",
        "\n",
        "\n",
        "class Preprocess(enum.Enum):\n",
        "  NOAM_PACKED = 1\n",
        "  PADDED = 2\n",
        "\n",
        "\n",
        "def py_batched_tfds(\n",
        "    *,\n",
        "    tfds_name: str,\n",
        "    split: str,\n",
        "    context_size: int,\n",
        "    worker_count: int,\n",
        "#    vocab_path: str,\n",
        "    batch_size: int,\n",
        "    seed: int | None = 1234,\n",
        "    num_epochs: int | None = None,\n",
        "    num_records: int | None = None,\n",
        "    preprocessing: Preprocess = Preprocess.NOAM_PACKED,\n",
        "    worker_buffer_size: int = 2,\n",
        "    shuffle: bool = True,\n",
        ") -> grain.DataLoader:\n",
        "  \"\"\"Returns iterator for regularly batched text examples.\"\"\"\n",
        "  datasource = tfds.data_source(tfds_name, split=split, data_dir = work_path)\n",
        "  index_sampler = grain.IndexSampler(\n",
        "      num_records=num_records if num_records is not None else len(datasource),\n",
        "      num_epochs=num_epochs,\n",
        "      shard_options=grain.NoSharding(),\n",
        "      shuffle=shuffle,\n",
        "      seed=seed,\n",
        "  )\n",
        "  #spt = _SPTokenizer(vocab_path)\n",
        "\n",
        "  pad_len = None if preprocessing == Preprocess.NOAM_PACKED else context_size\n",
        "  pygrain_ops = [\n",
        "      grain.MapOperation(\n",
        "          map_function=functools.partial(\n",
        "              _py_tokenize,\n",
        "              #spt=spt, # Removed spt from here\n",
        "              pad_len=pad_len,\n",
        "          )\n",
        "      )\n",
        "  ]\n",
        "  if preprocessing == Preprocess.NOAM_PACKED:\n",
        "    pygrain_ops.append(_NoamPack(context_size=context_size))\n",
        "  elif preprocessing == Preprocess.PADDED:\n",
        "    pygrain_ops.append(grain.MapOperation(map_function=np.array))\n",
        "  else:\n",
        "    raise ValueError(f'Unknown preprocessing: {preprocessing}')\n",
        "  pygrain_ops.append(grain.Batch(batch_size=batch_size, drop_remainder=True))\n",
        "  batched_dataloader = grain.DataLoader(\n",
        "      data_source=datasource,\n",
        "      operations=pygrain_ops,\n",
        "      sampler=index_sampler,\n",
        "      worker_count=worker_count,\n",
        "      worker_buffer_size=worker_buffer_size,\n",
        "  )\n",
        "  return batched_dataloader\n",
        "\n",
        "\n",
        "def _py_tokenize(\n",
        "    features: Mapping[str, str],\n",
        "    pad_len: int | None = None,\n",
        "    pad_id: int = PAD_ID,\n",
        ") -> Sequence[int]:\n",
        "  \"\"\"Map '0101...' -> [BOS_ID, 3,4,3,4, EOS_ID, ...].\"\"\"\n",
        "\n",
        "  '''\n",
        "  text = features[\"text\"].strip()           # ensure no stray whitespace\n",
        "  '''\n",
        "  text = features[\"text\"]\n",
        "  if isinstance(text, bytes):\n",
        "      text = text.decode(\"utf-8\")\n",
        "  # Strict check: raise if an unexpected symbol appears\n",
        "  if set(text) - {\"0\", \"1\"}:\n",
        "      print(\"Invalid example:\", repr(text))\n",
        "      raise ValueError(f\"Unexpected character in {text!r}\")\n",
        "\n",
        "  #ids = [BOS_ID] + [ZERO_ID if c == \"0\" else ONE_ID for c in text] + [EOS_ID]\n",
        "  ids = [ZERO_ID if c == \"0\" else ONE_ID for c in text]\n",
        "  ids = [BOS_ID] + ids # Add BOS_ID here\n",
        "\n",
        "  if pad_len is not None:\n",
        "    ids = (ids + [pad_id] * (pad_len - len(ids)))[:pad_len]\n",
        "\n",
        "  return ids\n",
        "\n",
        "\n",
        "\n",
        "@dataclasses.dataclass\n",
        "class _NoamPack:\n",
        "  \"\"\"Pygrain operation for tokenizing and Noam packing text.\"\"\"\n",
        "\n",
        "  context_size: int\n",
        "\n",
        "  def __call__(\n",
        "      self, idseq_iterator: Iterator[grain.Record]\n",
        "  ) -> Iterator[grain.Record]:\n",
        "    packed_ids = []\n",
        "    for input_record in idseq_iterator:\n",
        "      start = 0\n",
        "      while start < len(input_record.data):\n",
        "        rem_data = input_record.data[start:]\n",
        "        if len(packed_ids) + len(rem_data) < self.context_size:\n",
        "          packed_ids.extend(rem_data)  # use rest of example, move-on\n",
        "          break\n",
        "        else:\n",
        "          take = self.context_size - len(packed_ids)\n",
        "          packed_ids.extend(rem_data[:take])\n",
        "          last_record_key = input_record.metadata.remove_record_key()\n",
        "          yield grain.Record(\n",
        "              last_record_key, np.array(packed_ids, dtype=np.int32)\n",
        "          )\n",
        "          start += take\n",
        "          packed_ids = []\n",
        "          # Drop remainder for simplicity.\n",
        "          # We lose the rest of the example on restore.\n",
        "\n",
        "\n",
        "# pylint: disable=invalid-name"
      ],
      "metadata": {
        "id": "_gvYWfT2S0CG"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# Copyright 2024 DeepMind Technologies Limited.\n",
        "#\n",
        "# Licensed under the Apache License, Version 2.0 (the \"License\");\n",
        "# you may not use this file except in compliance with the License.\n",
        "# You may obtain a copy of the License at\n",
        "#\n",
        "# https://www.apache.org/licenses/LICENSE-2.0\n",
        "#\n",
        "# Unless required by applicable law or agreed to in writing, software\n",
        "# distributed under the License is distributed on an \"AS IS\" BASIS,\n",
        "# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n",
        "# See the License for the specific language governing permissions and\n",
        "# limitations under the License.\n",
        "\"\"\"Training loop.\"\"\"\n",
        "\n",
        "# pylint: disable=invalid-name,g-importing-member,g-import-not-at-top,unused-import\n",
        "\n",
        "import functools\n",
        "import time\n",
        "from typing import Any, Iterator, TYPE_CHECKING, Optional\n",
        "\n",
        "from absl import logging\n",
        "from clu import metric_writers\n",
        "from clu import periodic_actions\n",
        "from flax import linen as nn\n",
        "from flax.training.train_state import TrainState\n",
        "import grain.python as grain\n",
        "import jax\n",
        "from jax.experimental import mesh_utils\n",
        "import jax.numpy as jnp\n",
        "from jax.sharding import Mesh\n",
        "from jax.sharding import NamedSharding\n",
        "from jax.sharding import PartitionSpec as P\n",
        "from nanodo import evaluate\n",
        "# from nanodo import loss as loss_lib\n",
        "from nanodo import metrics as metrics_lib\n",
        "from nanodo import optimizer\n",
        "import numpy as np\n",
        "import optax\n",
        "import orbax.checkpoint as ocp\n",
        "\n",
        "import os\n",
        "\n",
        "\n",
        "if TYPE_CHECKING:\n",
        "  import ml_collections  # pylint: disable=g-bad-import-order\n",
        "\n",
        "\n",
        "PyTree = Any\n",
        "\n",
        "\n",
        "def train_and_evaluate(c: \"ml_collections.ConfigDict\"):\n",
        "  \"\"\"Train loop.\"\"\"\n",
        "\n",
        "  mesh = Mesh(mesh_utils.create_device_mesh((jax.device_count(),)), (\"data\",))\n",
        "  # For multistep gradient accumulator to simulate large batch sizes.\n",
        "  grad_accumulation_steps = c.get(\"grad_accumulation_steps\", 1)\n",
        "  micro_batch_size, r = divmod(c.batch_size, grad_accumulation_steps)\n",
        "  if grad_accumulation_steps > 1:\n",
        "    logging.info(\"Gradient accumulation steps: %d\", grad_accumulation_steps)\n",
        "    logging.info(\n",
        "        \"Using total batch size = %d, micro batch size = %d\",\n",
        "        c.batch_size, micro_batch_size\n",
        "    )\n",
        "  if r:\n",
        "    raise ValueError(\n",
        "        \"Batch size must be divisible by the gradient accumulation steps.\"\n",
        "    )\n",
        "  if micro_batch_size % jax.device_count() != 0:\n",
        "    raise ValueError(\"Batch size must be divisible by the number of devices.\")\n",
        "  os.makedirs(c.workdir, exist_ok=True)\n",
        "  rng = jax.random.PRNGKey(c.seed)\n",
        "\n",
        "  #tokenizer = data.get_py_tokenizer(c.vocab_path)\n",
        "  vocab_size = VOCAB_SIZE #tokenizer.GetPieceSize()\n",
        "\n",
        "  model, get_loss_fn = get_model_and_loss(c, vocab_size)#model_factory.get_model_and_loss(c, vocab_size)\n",
        "\n",
        "  tic = time.time()\n",
        "  shardings, state = _init_train_state(c, model, rng=rng, mesh=mesh)\n",
        "  init_time = time.time() - tic\n",
        "  logging.info(\"[TIMING]: get_new_state (jit init) time: %.2fs\", init_time)\n",
        "\n",
        "  train_ds = py_batched_tfds(\n",
        "      tfds_name=c.ds_name,\n",
        "      split=\"train\",\n",
        "      context_size=c.model.L,\n",
        "      worker_count=c.pygrain_worker_count,\n",
        " #     vocab_path=c.vocab_path,\n",
        "      batch_size=micro_batch_size,\n",
        "      num_epochs= None, #c.train_epochs,\n",
        "      preprocessing=Preprocess.NOAM_PACKED,\n",
        "      worker_buffer_size=c.pygrain_worker_buffer_size,\n",
        "      shuffle=True,\n",
        "  )\n",
        "  train_iter = iter(train_ds)\n",
        "\n",
        "\n",
        "  '''\n",
        "  if c.checkpoint:\n",
        "    ckpt_mngr = _get_ckpt_manager(c.workdir, c)\n",
        "    if c.checkpoint_restore_dir is not None:\n",
        "      logging.info(\"Restoring checkpoint from %s\", c.checkpoint_restore_dir)\n",
        "      ex_ckpt_mngr = _get_ckpt_manager(c.checkpoint_restore_dir, c)\n",
        "      state, train_iter = _restore_ckpt(ex_ckpt_mngr, state, train_iter)\n",
        "\n",
        "    elif ckpt_mngr.latest_step() is not None:\n",
        "      latest_step = ckpt_mngr.latest_step()\n",
        "      logging.info(\"Restoring checkpoint %d from %s\", latest_step, c.workdir)\n",
        "      state, train_iter = _restore_ckpt(ckpt_mngr, state, train_iter)\n",
        "  '''\n",
        "\n",
        "  # --- Replace your block with this ---\n",
        "  ckpt_mngr = None  # ensure it's defined for _checkpoint() closure\n",
        "\n",
        "  if c.checkpoint:\n",
        "    # Always create a manager for THIS run (used for saving/resume)\n",
        "    ckpt_mngr = _get_ckpt_manager(c.workdir, c)\n",
        "\n",
        "    # Detect finetune source (either cfg.init_from.checkpoint_dir or checkpoint_restore_dir)\n",
        "    finetune_src = None\n",
        "    finetune_step = \"latest\"\n",
        "    if hasattr(c, \"init_from\") and getattr(c.init_from, \"checkpoint_dir\", None):\n",
        "      finetune_src = c.init_from.checkpoint_dir\n",
        "      finetune_step = getattr(c.init_from, \"step\", \"latest\")\n",
        "    elif c.checkpoint_restore_dir is not None and c.checkpoint_restore_dir != c.workdir:\n",
        "      finetune_src = c.checkpoint_restore_dir\n",
        "      finetune_step = \"latest\"\n",
        "\n",
        "    is_finetune = (getattr(c, \"run_mode\", \"\") == \"finetune\") or (finetune_src is not None)\n",
        "\n",
        "    if is_finetune and finetune_src:\n",
        "      # -------- FINETUNE: restore PARAMS ONLY from external dir; keep train_iter as built --------\n",
        "      logging.info(\"Finetune: loading params from %s (weights only)\", finetune_src)\n",
        "      ex_ckpt_mngr = _get_ckpt_manager(finetune_src, c)\n",
        "\n",
        "      if finetune_step in (None, \"latest\"):\n",
        "        target_step = ex_ckpt_mngr.latest_step()\n",
        "      else:\n",
        "        target_step = int(finetune_step)\n",
        "\n",
        "      if target_step is None:\n",
        "        logging.warning(\"No checkpoint found in %s; continuing without restore.\", finetune_src)\n",
        "      else:\n",
        "        restore_args = ocp.checkpoint_utils.construct_restore_args(state)\n",
        "        restored = ex_ckpt_mngr.restore(\n",
        "            target_step,\n",
        "            items={\"state\": state},\n",
        "            restore_kwargs={\"state\": {\"restore_args\": restore_args}},\n",
        "        )\n",
        "        state = restored[\"state\"]\n",
        "        # start the new run at step 0\n",
        "        if hasattr(state, \"step\"):\n",
        "          state = state.replace(step=jnp.array(0, dtype=state.step.dtype))\n",
        "\n",
        "    elif ckpt_mngr.latest_step() is not None:\n",
        "      # -------- RESUME same run: try full restore (state + dataloader); fallback to state-only --------\n",
        "      latest_step = ckpt_mngr.latest_step()\n",
        "      logging.info(\"Resuming checkpoint %d from %s\", latest_step, c.workdir)\n",
        "      try:\n",
        "        state, train_iter = _restore_ckpt(ckpt_mngr, state, train_iter, step=latest_step)\n",
        "      except Exception as e:\n",
        "        logging.warning(\"Full restore failed (%s). Falling back to state-only.\", e)\n",
        "        restore_args = ocp.checkpoint_utils.construct_restore_args(state)\n",
        "        restored = ckpt_mngr.restore(\n",
        "            latest_step,\n",
        "            items={\"state\": state},\n",
        "            restore_kwargs={\"state\": {\"restore_args\": restore_args}},\n",
        "        )\n",
        "        state = restored[\"state\"]\n",
        "  # --- end replace ---\n",
        "\n",
        "  trainer = Trainer(\n",
        "      c=c,\n",
        "      state=state,\n",
        "      mesh=mesh,\n",
        "      shardings=shardings,\n",
        "      get_loss_fn=get_loss_fn,\n",
        "  )\n",
        "\n",
        "\n",
        "\n",
        "  # We may evaluate on larger context length than training to measure length\n",
        "  # generalization.\n",
        "  if c.model.L < c.eval_max_target_length:\n",
        "    logging.warning(\n",
        "        \"L (context length) %d is smaller than eval_max_target_length %d\",\n",
        "        c.model.L,\n",
        "        c.eval_max_target_length,\n",
        "    )\n",
        "  eval_batch_size = c.get(\"eval_batch_size\", micro_batch_size)\n",
        "  if eval_batch_size % jax.device_count() != 0:\n",
        "    raise ValueError(\n",
        "        \"Eval Batch size must be divisible by the number of devices.\")\n",
        "\n",
        "  eval_ds = py_batched_tfds(\n",
        "      tfds_name=c.ds_name,\n",
        "      split=c.eval_split,\n",
        "      context_size=c.model.L,\n",
        "      worker_count=c.pygrain_worker_count,\n",
        "#      vocab_path=c.vocab_path,\n",
        "      batch_size=eval_batch_size,\n",
        "      num_epochs=1,\n",
        "      num_records=None,\n",
        "      preprocessing=Preprocess.PADDED,\n",
        "      shuffle=False,\n",
        "  )\n",
        "  evaluator = evaluate.Evaluator(c, model, eval_ds, mesh, shardings)\n",
        "\n",
        "  writer = metric_writers.create_default_writer(\n",
        "      c.workdir,\n",
        "      just_logging=jax.process_index() > 0,\n",
        "  )\n",
        "  if trainer.step == 0:\n",
        "    writer.write_hparams(dict(c))\n",
        "    writer.write_scalars(trainer.step, {\"jit_compilation_time\": init_time})\n",
        "\n",
        "  report_progress = periodic_actions.ReportProgress(\n",
        "      num_train_steps=c.opt.num_train_steps,\n",
        "      writer=writer,\n",
        "      every_steps=c.write_perf_metrics_every_steps,\n",
        "      every_secs=None,\n",
        "  )\n",
        "\n",
        "  if jax.process_index() == 0:\n",
        "    hooks = [\n",
        "        report_progress,\n",
        "        periodic_actions.Profile(logdir=c.workdir, num_profile_steps=5),\n",
        "    ]\n",
        "  else:\n",
        "    hooks = []\n",
        "\n",
        "  with metric_writers.ensure_flushes(writer):\n",
        "    def _eval():\n",
        "      with report_progress.timed(\"eval\"):\n",
        "        step = trainer.step\n",
        "        eval_metrics = evaluator.eval(trainer.state.params)\n",
        "        writer.write_scalars(step, eval_metrics)\n",
        "\n",
        "    def _checkpoint():\n",
        "      if c.checkpoint:\n",
        "        step = trainer.step\n",
        "        logging.info(\"Saving last checkpoint step %d\", step)\n",
        "        ckpt_mngr.save(step, {\"state\": trainer.state, \"data\": train_iter})  # pylint: disable=undefined-variable\n",
        "\n",
        "    def _process_metrics(step, microbatch_metrics):\n",
        "      if microbatch_metrics and step % c.write_train_metrics_every_steps == 0:\n",
        "        microbatch_metrics = [trainer.get_metrics(step, m)\n",
        "                              for m in microbatch_metrics]\n",
        "        metrics = metrics_lib.aggregate_microbatch_metrics(microbatch_metrics)\n",
        "\n",
        "\n",
        "        # Print batch loss\n",
        "        print(f\"Step {step}: train_loss = {metrics['train_loss']:.4f}\")\n",
        "\n",
        "\n",
        "        writer.write_scalars(step, metrics)\n",
        "        # Simple check for NaN/Inf for early termination.\n",
        "        loss = metrics[\"train_loss\"]\n",
        "        if np.isnan(loss) or np.isinf(loss):\n",
        "          # Terminate training. The next step has already been dispatched.\n",
        "          logging.error(\n",
        "              \"[TRAINING ERROR] Nan/Inf encountered in training loop.\\n \"\n",
        "              \"Terminating training loop at step: %d\", step + 1\n",
        "          )\n",
        "          _eval()\n",
        "          raise FloatingPointError(step + 1, loss)\n",
        "\n",
        "    pending_microbatch_metrics = []\n",
        "    for step in range(trainer.step, c.opt.num_train_steps + 1):\n",
        "      is_final_step = step == c.opt.num_train_steps\n",
        "      if step % c.eval_every_steps == 0 or is_final_step:\n",
        "        _eval()\n",
        "      if step % c.checkpoint_every_steps == 0 or is_final_step:\n",
        "        print(\"checkpointing\")\n",
        "        _checkpoint()\n",
        "\n",
        "      for h in hooks:\n",
        "        h(step)\n",
        "\n",
        "      # Schedule this step's tasks.\n",
        "      # Initialize metrics for microbatch accumulation.\n",
        "      new_microbatch_metrics = []\n",
        "      for _ in range(grad_accumulation_steps):\n",
        "        try:\n",
        "          in_BxL = next(train_iter)\n",
        "        except StopIteration:\n",
        "          logging.warning(\"Ran out of data at step %d. Stopping.\", step)\n",
        "          break\n",
        "        # Async dispatch next step.\n",
        "        new_microbatch_metrics.append(trainer.do_step(step, in_BxL))\n",
        "\n",
        "      # Download to host and process the previous step's metrics after having\n",
        "      # asynchronously dispatched the new step.\n",
        "      _process_metrics(step - 1, pending_microbatch_metrics)\n",
        "      pending_microbatch_metrics = new_microbatch_metrics\n",
        "      logging.log_first_n(\n",
        "          logging.INFO, \"Finished training step %d.\", 5, step - 1)\n",
        "    # Download to host and process the final step's metrics.\n",
        "    _process_metrics(c.opt.num_train_steps, pending_microbatch_metrics)\n",
        "\n",
        "  if c.checkpoint:\n",
        "    ckpt_mngr.close()  # pylint: disable=undefined-variable\n",
        "\n",
        "\n",
        "class Trainer:\n",
        "  \"\"\"Executes training step.\"\"\"\n",
        "\n",
        "  def __init__(\n",
        "      self,\n",
        "      c: \"ml_collections.ConfigDict\",\n",
        "      state: TrainState,\n",
        "      mesh: Mesh,\n",
        "      shardings: PyTree,\n",
        "      get_loss_fn: LossFnFactory = get_default_loss_fn,\n",
        "  ):\n",
        "    self.state = state\n",
        "    self.init_metrics = None\n",
        "\n",
        "    # In the jit call below, in_shardings and out_shardings specify the\n",
        "    # shardings of the input and output of the jitted function.\n",
        "    # There is just as many in_shardings as input arguments, and likewise for\n",
        "    # outputs. \"shardings\" is the shardings of the state, P(\"data\") denotes\n",
        "    # that the argument is split along the data axis (in this case the\n",
        "    # input data), and P() denotes that the result is replicated on each\n",
        "    # device (in this case the train metrics).\n",
        "    self.step_fn = jax.jit(\n",
        "        functools.partial(\n",
        "            _train_step,\n",
        "            c=c,\n",
        "            get_loss_fn=get_loss_fn,\n",
        "            mesh=mesh,\n",
        "        ),\n",
        "        in_shardings=(\n",
        "            shardings,\n",
        "            NamedSharding(mesh, P()),\n",
        "        ),\n",
        "        out_shardings=(shardings, NamedSharding(mesh, P())),\n",
        "        donate_argnames=(\"state\", \"in_BxL\"),\n",
        "    )\n",
        "\n",
        "  @property\n",
        "  def step(self) -> int:\n",
        "    return int(self.state.step)\n",
        "\n",
        "  def get_metrics(\n",
        "      self, step: int, metrics: dict[str, float]\n",
        "  ) -> dict[str, float]:\n",
        "    # Grab the (possibly previous step's) metrics from device.\n",
        "    metrics = jax.device_get(metrics)\n",
        "    if step == 0:\n",
        "      metrics |= self.init_metrics\n",
        "    return metrics\n",
        "\n",
        "  def do_step(self, step: int, in_BxL: jax.Array) -> dict[str, float]:\n",
        "    \"\"\"Async dispatch one training step and return metrics.\"\"\"\n",
        "    # Note that the device may be busy with the previous step.\n",
        "    # Avoid calling self.step as that would block until the device is ready.\n",
        "    if step == 0 or self.init_metrics is None:\n",
        "      self.init_metrics = metrics_lib.get_init_metrics(self.state)\n",
        "\n",
        "    self.state, metrics = self.step_fn(self.state, in_BxL)\n",
        "    return metrics\n",
        "\n",
        "\n",
        "def _train_step(\n",
        "    state: TrainState,\n",
        "    in_BxL: jax.Array,\n",
        "    c: \"ml_collections.ConfigDict\",\n",
        "    get_loss_fn: LossFnFactory = get_default_loss_fn,\n",
        "    mesh: Mesh | None = None,\n",
        ") -> tuple[TrainState, dict[str, float | jax.Array]]:\n",
        "  \"\"\"One forward/backward pass.\"\"\"\n",
        "  if mesh is not None:\n",
        "    in_BxL = jax.lax.with_sharding_constraint(\n",
        "        in_BxL, NamedSharding(mesh, P(\"data\"))\n",
        "    )\n",
        "  grad_fn = jax.value_and_grad(\n",
        "      get_loss_fn(in_BxL, state.apply_fn, c), has_aux=True\n",
        "  )\n",
        "  (loss, aux_data), grads = grad_fn(state.params)\n",
        "\n",
        "  # Access to optax updates.\n",
        "  updates, new_opt_state = state.tx.update(grads, state.opt_state, state.params)\n",
        "  new_params = optax.apply_updates(state.params, updates)\n",
        "  new_state = state.replace(\n",
        "      # Keep gradient_step as Trainer's step.\n",
        "      step=state.opt_state.gradient_step + 1,  # pytype: disable=attribute-error\n",
        "      params=new_params,\n",
        "      opt_state=new_opt_state,\n",
        "  )\n",
        "\n",
        "  metrics = metrics_lib.get_metrics(aux_data, c, loss, state, grads, updates)\n",
        "  return new_state, metrics\n",
        "\n",
        "\n",
        "def _init_train_state(\n",
        "    c: \"ml_collections.ConfigDict\",\n",
        "    module: nn.Module,\n",
        "    rng: jax.Array,\n",
        "    mesh: Mesh,\n",
        ") -> tuple[PyTree, TrainState]:\n",
        "  \"\"\"Creates a sharding and model state.\"\"\"\n",
        "  inputs = jax.ShapeDtypeStruct(shape=(1, c.model.L), dtype=jnp.int32)\n",
        "\n",
        "  def init(rng, inputs):\n",
        "    params = module.init(rng, inputs)\n",
        "    return TrainState.create(\n",
        "        apply_fn=module.apply,\n",
        "        params=params[\"params\"],\n",
        "        tx=optimizer.get_optimizer(c.opt),\n",
        "    )\n",
        "\n",
        "  params = jax.eval_shape(init, rng, inputs)\n",
        "  shardings = nn.get_sharding(params, mesh)\n",
        "  state = jax.jit(init, out_shardings=shardings)(rng, inputs)\n",
        "  return shardings, state\n",
        "\n",
        "\n",
        "def _get_ckpt_manager(\n",
        "    ckpt_dir: str,\n",
        "    c: \"ml_collections.ConfigDict\",\n",
        ") -> ocp.CheckpointManager:\n",
        "  options = ocp.CheckpointManagerOptions(max_to_keep=c.max_to_keep)\n",
        "  checkpointers = dict(\n",
        "      state=ocp.AsyncCheckpointer(ocp.PyTreeCheckpointHandler()),\n",
        "      data=ocp.Checkpointer(grain.PyGrainCheckpointHandler()),  # pytype:disable=wrong-arg-types\n",
        "  )\n",
        "  return ocp.CheckpointManager(ckpt_dir, checkpointers, options)\n",
        "\n",
        "'''\n",
        "def _restore_ckpt(\n",
        "    ckpt_mngr: ocp.CheckpointManager,\n",
        "    state: TrainState,\n",
        "    train_iter: Iterator[jax.Array],\n",
        "    step: int | None = None,\n",
        ") -> tuple[TrainState, Iterator[jax.Array]]:\n",
        "  \"\"\"Restore a checkpoint.\"\"\"\n",
        "  restore_args = ocp.checkpoint_utils.construct_restore_args(state)\n",
        "  restore_kwargs = {\"state\": {\"restore_args\": restore_args}}\n",
        "  restored = ckpt_mngr.restore(\n",
        "      ckpt_mngr.latest_step() if step is None else step,\n",
        "      items={\"state\": state, \"data\": train_iter},\n",
        "      restore_kwargs=restore_kwargs,\n",
        "  )\n",
        "  return restored[\"state\"], restored[\"data\"]\n",
        "'''\n",
        "def _restore_ckpt(\n",
        "    ckpt_mngr: ocp.CheckpointManager,\n",
        "    state: TrainState,\n",
        "    train_iter: Iterator[jax.Array],\n",
        "    step: Optional[int] = None,\n",
        "    c: Optional[\"ml_collections.ConfigDict\"] = None,  # <-- use Optional[...] not `\"...\" | None`\n",
        ") -> tuple[TrainState, Iterator[jax.Array]]:\n",
        "  \"\"\"Restore a checkpoint.\n",
        "\n",
        "  - If c.run_mode == \"finetune\": load params only from c.init_from.checkpoint_dir,\n",
        "    leave train_iter untouched, and start step from 0 in your training loop.\n",
        "  - Else (resume): restore params + dataloader. If the dataloader sampler mismatches,\n",
        "    fall back to restoring params only.\n",
        "  \"\"\"\n",
        "  # Helper: restore STATE only from a given manager/step.\n",
        "  def _restore_state_only(mngr: ocp.CheckpointManager, which_step: int | None):\n",
        "    tgt_step = mngr.latest_step() if which_step is None else which_step\n",
        "    if tgt_step is None:\n",
        "      return None\n",
        "    restore_args = ocp.checkpoint_utils.construct_restore_args(state)\n",
        "    restore_kwargs = {\"state\": {\"restore_args\": restore_args}}\n",
        "    restored = mngr.restore(tgt_step, items={\"state\": state}, restore_kwargs=restore_kwargs)\n",
        "    return restored[\"state\"]\n",
        "\n",
        "  # === Finetune path: read weights from PRETRAINED dir, ignore dataloader ===\n",
        "  if c is not None and getattr(c, \"run_mode\", \"\") == \"finetune\":\n",
        "    init_from = getattr(c, \"init_from\", None)\n",
        "    src_dir = getattr(init_from, \"checkpoint_dir\", None) if init_from is not None else None\n",
        "    if not src_dir:\n",
        "      # nothing to load; return original state/iter\n",
        "      return state, train_iter\n",
        "\n",
        "    # Use a temporary manager pointed at the PRETRAINED workdir (read-only).\n",
        "    src_mgr = _get_ckpt_manager(src_dir, c)\n",
        "    # Choose step\n",
        "    src_step = None\n",
        "    if init_from is not None and hasattr(init_from, \"step\") and init_from.step != \"latest\":\n",
        "      try:\n",
        "        src_step = int(init_from.step)\n",
        "      except Exception:\n",
        "        src_step = None\n",
        "\n",
        "    restored_state = _restore_state_only(src_mgr, src_step)\n",
        "    if restored_state is None:\n",
        "      # no checkpoint found at source; proceed as scratch\n",
        "      return state, train_iter\n",
        "    return restored_state, train_iter\n",
        "\n",
        "  # === Resume path: restore state + data from THIS run's manager ===\n",
        "  tgt_step = ckpt_mngr.latest_step() if step is None else step\n",
        "  if tgt_step is None:\n",
        "    # nothing to resume\n",
        "    return state, train_iter\n",
        "\n",
        "  restore_args = ocp.checkpoint_utils.construct_restore_args(state)\n",
        "  restore_kwargs = {\"state\": {\"restore_args\": restore_args}}\n",
        "\n",
        "  # Try full restore (state + data). If sampler mismatches, fall back to state only.\n",
        "  try:\n",
        "    restored = ckpt_mngr.restore(\n",
        "        tgt_step,\n",
        "        items={\"state\": state, \"data\": train_iter},\n",
        "        restore_kwargs=restore_kwargs,\n",
        "    )\n",
        "    return restored[\"state\"], restored[\"data\"]\n",
        "  except Exception as e:\n",
        "    # Common case: Grain complains \"Sampler in checkpoint does not match dataloader sampler.\"\n",
        "    # Retry with state-only restore so training can continue.\n",
        "    restored_state = _restore_state_only(ckpt_mngr, tgt_step)\n",
        "    if restored_state is None:\n",
        "      raise\n",
        "    return restored_state, train_iter\n",
        "\n"
      ],
      "metadata": {
        "id": "yJFGIaqFqZ4R"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "def get_ckpt_path(ckpt_manager, step):\n",
        "    # Orbax checkpoints are typically stored under: {workdir}/checkpoint_{step}\n",
        "    return os.path.join(ckpt_manager.directory, f\"checkpoint_{step}\")\n",
        "\n",
        "def create_mesh():\n",
        "    \"\"\"Create a single-axis mesh over all available devices.\"\"\"\n",
        "    return Mesh(mesh_utils.create_device_mesh((jax.device_count(),)), (\"data\",))\n",
        "\n",
        "\n",
        "def restore_model(workdir, config, mesh):\n",
        "    \"\"\"Restore TrainState from checkpoint using mesh sharding.\"\"\"\n",
        "    model, _ = get_model_and_loss(config, VOCAB_SIZE)\n",
        "\n",
        "    dummy_input = jax.ShapeDtypeStruct((1, config.model.L), dtype=jnp.int32)\n",
        "\n",
        "    def init(rng, inputs):\n",
        "        variables = model.init(rng, inputs)\n",
        "        return TrainState.create(\n",
        "            apply_fn=model.apply,\n",
        "            params=variables[\"params\"],\n",
        "            tx=optimizer.get_optimizer(config.opt),  # ✅ proper tx\n",
        "        )\n",
        "\n",
        "    rng = jax.random.PRNGKey(config.seed)\n",
        "    params = jax.eval_shape(init, rng, dummy_input)\n",
        "    shardings = nn.get_sharding(params, mesh)\n",
        "    state = jax.jit(init, out_shardings=shardings)(rng, dummy_input)\n",
        "\n",
        "\n",
        "    ckpt_manager = ocp.CheckpointManager(\n",
        "        workdir,\n",
        "        {\"state\": ocp.AsyncCheckpointer(ocp.PyTreeCheckpointHandler())},\n",
        "        ocp.CheckpointManagerOptions(max_to_keep=config.max_to_keep),\n",
        "    )\n",
        "\n",
        "    latest_step = ckpt_manager.latest_step()\n",
        "    print(f\"Restoring checkpoint from step {latest_step}\")\n",
        "\n",
        "    ckpt_path = get_ckpt_path(ckpt_manager, latest_step)\n",
        "    print(f\"Restored checkpoint path: {ckpt_path}\")\n",
        "\n",
        "    restored = ckpt_manager.restore(latest_step, items={\"state\": state})\n",
        "    state = restored[\"state\"]\n",
        "\n",
        "    # Print the full checkpoint path (optional)\n",
        "    #ckpt_path = ckpt_manager._checkpoint_path(latest_step)  # private method, OK for debugging\n",
        "    # print(f\"Restored checkpoint path: {ckpt_path}\")\n",
        "    #state = ckpt_manager.restore(ckpt_manager.latest_step(), items={\"state\": state})[\"state\"]\n",
        "    return model, state\n",
        "\n",
        "\n",
        "def string_to_token_ids(s: str, pad_len: int) -> jnp.ndarray:\n",
        "    \"\"\"Convert binary string to token IDs.\"\"\"\n",
        "    token_ids = _py_tokenize({\"text\": s}, pad_len=pad_len)\n",
        "    return jnp.array(token_ids)[None, :]  # shape (1, L)\n",
        "\n",
        "def token_ids_to_string(token_ids):\n",
        "    \"\"\"Convert a sequence of token IDs to a binary string.\"\"\"\n",
        "    id_to_bit = {\n",
        "        ZERO_ID: '0',\n",
        "        ONE_ID: '1',\n",
        "    }\n",
        "    # Convert to flat list of Python ints\n",
        "    token_ids = list(map(int, jax.device_get(token_ids).flatten()))\n",
        "    return ''.join(\n",
        "        id_to_bit[token] for token in token_ids if token in id_to_bit\n",
        "    )\n",
        "\n"
      ],
      "metadata": {
        "id": "CnC0q6fv_Lvq"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "\n",
        "\n",
        "# Start training"
      ],
      "metadata": {
        "id": "M8Q_Z03jusw6"
      }
    },
    {
      "cell_type": "markdown",
      "source": [
        "### Build dataset\n"
      ],
      "metadata": {
        "id": "CYxPrcKDuyh5"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "# ========= CONFIG =========\n",
        "CONTEXT_LEN    = 1024        # length of each binary string\n",
        "NUM_SEQUENCES  = 200000     # how many strings to generate\n",
        "SEED           = 42         # RNG seed for reproducibility\n",
        "ROBUST_ALPHA = ALPHA\n",
        "# ==========================\n",
        "\n",
        "import random, json, pathlib, time\n",
        "import multiprocessing\n",
        "import os\n",
        "import math\n",
        "\n",
        "rng = random.Random(SEED)\n",
        "# print(\"threshold for robust bernoulli distribution: \", math.sqrt((1.0 + ROBUST_ALPHA) * 8.0 * math.log(CONTEXT_LEN) / CONTEXT_LEN) + 1.0 / math.sqrt(CONTEXT_LEN) )"
      ],
      "metadata": {
        "id": "Wm2eC1ee-arq"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# specify model\n",
        "\n",
        "# TRAIN_DATA = \"POLYAURN\"\n",
        "# TRAIN_DATA = \"ROBUST_BERNOULLI\"\n",
        "# TRAIN_DATA = \"BERNOULLI\"\n",
        "TRAIN_DATA = \"BERNOULLI_1\"\n",
        "\n",
        "# WORK_PATH_DICT = {\n",
        "#     \"BERNOULLI\": '/content/drive/MyDrive/next-token-no-regret/bernoulli/',\n",
        "#     \"POLYAURN\": '/content/drive/MyDrive/next-token-no-regret/polyaurn-small/',\n",
        "#     \"ROBUST_BERNOULLI_TEST1\": '/content/drive/MyDrive/next-token-no-regret/polyaurn-robust-test1-1024/',\n",
        "#     \"ROBUST_BERNOULLI\": '/content/drive/MyDrive/next-token-no-regret/polyaurn-robust-test0-1024/',\n",
        "# }\n",
        "\n",
        "# DATA_PATH_DICT = {\n",
        "#     \"BERNOULLI\":'/content/drive/MyDrive/next-token-no-regret/bernoulli/',\n",
        "#     \"POLYAURN\": '/content/drive/MyDrive/next-token-no-regret/polyaurn-small/',\n",
        "#     \"ROBUST_BERNOULLI_TEST1\": '/content/drive/MyDrive/next-token-no-regret/polyaurn-robust-test1/', # Not implemented yet\n",
        "#     \"ROBUST_BERNOULLI\": '/content/drive/MyDrive/next-token-no-regret/polyaurn-robust-test0/',\n",
        "# }\n",
        "\n",
        "# data_path = DATA_PATH_DICT[TRAIN_DATA]\n",
        "work_path = f'/content/drive/MyDrive/next-token-no-regret-0924/{TRAIN_DATA}-{CONTEXT_LEN}/'\n",
        "data_path = f'/content/drive/MyDrive/next-token-no-regret-0924/{TRAIN_DATA}-{CONTEXT_LEN}/'\n",
        "print(\"Set work path to: \", work_path)\n",
        "print(\"Set data path to: \", data_path)"
      ],
      "metadata": {
        "id": "hY23wtLAvH0M",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "4012f35e-44fa-494d-d1a0-eaddc5d39476"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Set work path to:  /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/\n",
            "Set data path to:  /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "from tqdm.auto import tqdm\n",
        "OUT_PATH       = data_path + \"empirical_binary.jsonl\"#\"/content/binary_dataset/train.jsonl\"  # where to save\n",
        "\n",
        "out_path = pathlib.Path(OUT_PATH)\n",
        "\n",
        "def polya_seeded_fast(L: int, rng, np_rng: \"np.random.Generator|None\" = None):\n",
        "    \"\"\"\n",
        "    Exact-in-law to your seeded Pólya-urn:\n",
        "      - pick prefix as '01' or '10' with 50/50;\n",
        "      - sample p ~ Beta(1,1) (i.e., Uniform[0,1]);\n",
        "      - draw remaining L-2 bits i.i.d. Bernoulli(p).\n",
        "    Returns: list of '0'/'1' chars, length L.\n",
        "    \"\"\"\n",
        "    assert L >= 2\n",
        "    prefix = rng.choice([\"01\", \"10\"])\n",
        "    p = rng.random()  # Beta(1,1)\n",
        "\n",
        "    if np_rng is None:\n",
        "        # pure-Python fast path (still O(L), but no O(t) counts each step)\n",
        "        tail = ['1' if rng.random() < p else '0' for _ in range(L - 2)]\n",
        "    else:\n",
        "        # NumPy vectorized tail (faster for big batches)\n",
        "        tail_mask = np_rng.random(L - 2) < p\n",
        "        tail = np.where(tail_mask, '1', '0').tolist()\n",
        "\n",
        "    return list(prefix) + tail\n",
        "\n",
        "def _precomp_ref(L: int):\n",
        "    split = L // 2\n",
        "    p_ref = np.concatenate([\n",
        "        np.full(split, 1/3, np.float32),\n",
        "        np.full(L - split, 2/3, np.float32),\n",
        "    ])\n",
        "    inv_sqrt_t = 1.0 / np.sqrt(np.arange(1, L+1, dtype=np.float32))\n",
        "    t = np.arange(1, L+1, dtype=np.int32)\n",
        "    return p_ref, inv_sqrt_t, t\n",
        "\n",
        "def passes_regret(bits, L: int, alpha: float) -> bool:\n",
        "    \"\"\"Vectorized check: ∃t: regret_t >= alpha / sqrt(t) against 1/3→2/3 reference.\"\"\"\n",
        "    p_ref, inv_sqrt_t, t = _precomp_ref(L)\n",
        "    # bits may be list of '0'/'1' or a string\n",
        "    if isinstance(bits, list):\n",
        "        s = ''.join(bits)\n",
        "    else:\n",
        "        s = bits\n",
        "    b = (np.frombuffer(s.encode('ascii'), dtype=np.uint8) - 48).astype(np.uint8)  # (L,)\n",
        "    ones_cum = np.add.accumulate(b, dtype=np.int32)                               # (L,)\n",
        "    mismatch = np.where(b == 1, 1.0 - p_ref, p_ref)                               # (L,)\n",
        "    cum_mismatch = np.add.accumulate(mismatch, dtype=np.float32)                  # (L,)\n",
        "    best_const = np.minimum(ones_cum, t - ones_cum).astype(np.float32)            # (L,)\n",
        "    regret = cum_mismatch - best_const                                            # (L,)\n",
        "    return bool((regret >= (alpha * inv_sqrt_t)).any())\n",
        "\n",
        "\n",
        "# Check if the file already exists\n",
        "if out_path.exists():\n",
        "    print(f\"File already exists: {out_path}. Skipping generation.\")\n",
        "else:\n",
        "    # Ensure the parent directory exists before starting any processes\n",
        "    os.makedirs(out_path.parent, exist_ok=True)\n",
        "\n",
        "\n",
        "    def generate_sequence(L: int, rng: random.Random) -> Optional[str]:\n",
        "        \"\"\"Return a length‑L binary string or None if it is all 0s / all 1s.\"\"\"\n",
        "        if TRAIN_DATA == \"POLYAURN\":\n",
        "            # Generate Polya-urn sequence (seeded to avoid trivial all-0/all-1)\n",
        "            bits = polya_seeded_fast(L, rng)\n",
        "            return \"\".join(bits)\n",
        "        elif TRAIN_DATA in (\"BERNOULLI\", \"BERNOULLI_1\"):\n",
        "            split_point = int(L / 2)\n",
        "            bits = []\n",
        "            for i in range(L):\n",
        "                if i < split_point:\n",
        "                    # First 2/3\n",
        "                    bits.append(\"1\" if rng.random() < (1/3) else \"0\")\n",
        "                else:\n",
        "                    # Last 1/3\n",
        "                    bits.append(\"1\" if rng.random() < (2/3) else \"0\")\n",
        "            # bits = [\"1\" if rng.random() < (1/3) else \"0\" for _ in range(L)]\n",
        "            return \"\".join(bits)\n",
        "        elif TRAIN_DATA == \"ROBUST_BERNOULLI_TEST1\":\n",
        "            # First 1/2 from Ber(1/3), last 1/2 from Ber(2/3)\n",
        "            split_point = int(L / 2)\n",
        "            bits = []\n",
        "            for i in range(L):\n",
        "                if i < split_point:\n",
        "                    # First 2/3\n",
        "                    bits.append(\"1\" if rng.random() < (1/3) else \"0\")\n",
        "                else:\n",
        "                    # Last 1/3\n",
        "                    bits.append(\"1\" if rng.random() < (2/3) else \"0\")\n",
        "            return \"\".join(bits)\n",
        "\n",
        "        elif TRAIN_DATA in (\"ROBUST_BERNOULLI\", \"ROBUST_BERNOULLI_PREFIX\"):\n",
        "            alpha = ROBUST_ALPHA\n",
        "\n",
        "            # 50/50: Bernoulli(1/3) or Polya-urn (subject to the robustness filter)\n",
        "            if rng.random() < 0.5:\n",
        "                split_point = int(L / 2)\n",
        "                bits = []\n",
        "                for i in range(L):\n",
        "                    if i < split_point:\n",
        "                        # First 2/3\n",
        "                        bits.append(\"1\" if rng.random() < (1/3) else \"0\")\n",
        "                    else:\n",
        "                        # Last 1/3\n",
        "                        bits.append(\"1\" if rng.random() < (2/3) else \"0\")\n",
        "                return \"\".join(bits)\n",
        "            else:\n",
        "                alpha = ROBUST_ALPHA\n",
        "                # Generate Polya-urn sequence (seeded to avoid trivial all-0/all-1)\n",
        "                bits = polya_seeded_fast(L, rng)\n",
        "                # Keep if the regret threshold is hit at some prefix\n",
        "                if passes_regret(bits, L, alpha):\n",
        "                    return \"\".join(bits)\n",
        "                else:\n",
        "                    return None\n",
        "\n",
        "\n",
        "        else:\n",
        "            raise ValueError(f\"Unknown TRAIN_DATA: {TRAIN_DATA}\")\n",
        "            return None\n",
        "\n",
        "    def generate_sequences_worker(num_to_generate, L, seed, queue):\n",
        "        local_rng = random.Random(seed)\n",
        "        count = 0\n",
        "        while count < num_to_generate:\n",
        "            seq = generate_sequence(L, local_rng)\n",
        "            if seq is not None:\n",
        "                queue.put({\"text\": seq})\n",
        "                count += 1\n",
        "        queue.put(None) # Signal completion\n",
        "\n",
        "    start = time.time()\n",
        "\n",
        "    # Determine number of processes to use\n",
        "    num_processes = multiprocessing.cpu_count()\n",
        "    sequences_per_process = NUM_SEQUENCES // num_processes\n",
        "    remainder = NUM_SEQUENCES % num_processes\n",
        "\n",
        "    # Create a queue to hold generated sequences\n",
        "    queue = multiprocessing.Queue(maxsize=10_000)\n",
        "\n",
        "    # Create and start worker processes\n",
        "    processes = []\n",
        "    for i in range(num_processes):\n",
        "        seed = SEED + i # Use different seeds for each process\n",
        "        num_to_gen = sequences_per_process + (1 if i < remainder else 0)\n",
        "        p = multiprocessing.Process(target=generate_sequences_worker, args=(num_to_gen, CONTEXT_LEN - 1, seed, queue))\n",
        "        processes.append(p)\n",
        "        p.start()\n",
        "\n",
        "    # Write sequences from the queue to the output file\n",
        "    written = 0\n",
        "    with out_path.open(\"w\") as fh, tqdm(total=NUM_SEQUENCES, desc=\"Generating\", unit=\"seq\") as pbar:\n",
        "        completed_processes = 0\n",
        "        while completed_processes < num_processes:\n",
        "            item = queue.get()  # blocks until something arrives\n",
        "            if item is None:\n",
        "                completed_processes += 1\n",
        "                # optional: show how many workers are done\n",
        "                pbar.set_postfix(workers_done=completed_processes)\n",
        "            else:\n",
        "                fh.write(json.dumps(item) + \"\\n\")\n",
        "                written += 1\n",
        "                pbar.update(1)\n",
        "\n",
        "    # Wait for all processes to finish\n",
        "    for p in processes:\n",
        "        p.join()\n",
        "\n",
        "    print(f\"✅  Wrote {NUM_SEQUENCES:,} sequences to {out_path} in \"\n",
        "          f\"{time.time() - start:.1f}s\")\n",
        "\n",
        "    # --- quick preview ---\n",
        "    print(\"\\nFirst 3 examples:\")\n",
        "    with out_path.open() as fh:\n",
        "        for _, line in zip(range(3), fh):\n",
        "            print(json.loads(line)[\"text\"])"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 243,
          "referenced_widgets": [
            "e4e553c8ab254ee4b5d2549f090702e9",
            "d7ed656e47a742adbef67a45d7004952",
            "1de16e4b8c7649358f6ff317a2700052",
            "e2d85edc602048edb740e578eb8df681",
            "d911c936404b43e0812a81f7b0398260",
            "7e35db8b08224c4eaefaf5e60019168a",
            "d5f3e61b1b0c4ca3bc15b96780336f53",
            "cd1bfd535c594826b1e3cfe5b6ed556e",
            "47a4b971ac3344048da931f00225f651",
            "90f9d3644f9c45b3ad49ab94bb6c5319",
            "29f54132454c40b689f7b5932dceeb53"
          ]
        },
        "id": "gVRzNLqii3gw",
        "outputId": "a6127c98-78a6-4f5e-cf50-6b8275d155e7",
        "collapsed": true
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/lib/python3.12/multiprocessing/popen_fork.py:66: RuntimeWarning: os.fork() was called. os.fork() is incompatible with multithreaded code, and JAX is multithreaded, so this will likely lead to a deadlock.\n",
            "  self.pid = os.fork()\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "Generating:   0%|          | 0/200000 [00:00<?, ?seq/s]"
            ],
            "application/vnd.jupyter.widget-view+json": {
              "version_major": 2,
              "version_minor": 0,
              "model_id": "e4e553c8ab254ee4b5d2549f090702e9"
            }
          },
          "metadata": {}
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/lib/python3.12/multiprocessing/popen_fork.py:66: RuntimeWarning: os.fork() was called. os.fork() is incompatible with multithreaded code, and JAX is multithreaded, so this will likely lead to a deadlock.\n",
            "  self.pid = os.fork()\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "✅  Wrote 200,000 sequences to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/empirical_binary.jsonl in 11.5s\n",
            "\n",
            "First 3 examples:\n",
            "011100010110110010010001001100000000000001111111000110001010000000111110010000110100010011100010000010001010000101001001000011100011001010010010010011000011001110100110110000011000000100011001100001000001010100100000001100001000101100110100000010100110100011100001000001010011101001010001011001000110000011100001110000100000101000010100100101111001000001010110010000101000000101100110011000001011000001000110000001111001001001010010100110000000000000000110000111110110010001000000010000000011010001010100101010110100111101101001111011001101001110010101100110011111001010010010111010010111110110100110101001101101111101011111101110011111111101010101111101011111110101011110011100111111111110010001101110011101110001111111100100011011011111110010100111101101111101011010101111110011010110011101001010101110111111111011010001100111100111011110101000110001110001110100101111101101101111100011111110111111010111001110101111001010111101101110100101001101000000101110010011101011001110111011000101111101110010101111011111011110110\n",
            "110101111011011110000001001000001000001101100100001000001010001000101011000001100000000010110000000000100110101000101010000001110010001010100010011111110100000101100000011100001010001011010000001110001000000000001101110101010000000000010111000011000101000001101000000100000010000000000001100111010001001010001000101000100000001000001100010010101001001101001000011001001110000111100010010110010010100000000110000001100100000100111100001010001100000000000010001101000010001001000001001100000000000001100001100100101011001111000010111101110111100100101101001101011111010001111111101101010111110111110100111111110000111111111111011110111110101111011101000111111101001010100111111111111100110101010010001100010111101010111101000011111111011001011111011101100110100001101111110011110111101011010011011011100010100110000111010110011111110010100011011111100101101111100110111110111111110011011111111111111011111001001111001011000110100101111111011110110111001010011111010111011100110111111100011100001111010100010111110010110101001\n",
            "001010010010100000010010000000111000110000010000101000001100000010100000001010101000011110010000010000010011100000101100011000101000001110001010000100000001100000001000110001111011100010101010100001011010001110011101001000011011001000100010000000001000000100010000000000001000010000001100000000001100101101000011010001011100101001010001001101101000000000001011111000101000000110101100000000110110100001100001000100100000001000001111101010110001000001010000001000010100011000000101011000000011001001010101010100111011011110100011110110100101111101110011011011111110001111011001111100010110000111110111101111111011100100011111100100101001101001011011111111110111011110010011101111010001110111111111101001111100011001111100100101011101110111000111101111110111110001011111010111011011001011111011011001111101101111011111111110110000111011111001010001110010100100110001001110011111111011000111000001011111000110000111101000100110011010101010101111111100011111010110110111010111110101111000111110111111111110111111101111111001110\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "import pathlib, shutil, tensorflow_datasets as tfds, tensorflow as tf, json\n",
        "\n",
        "# --- Your minimal builder class from before (unchanged) ---\n",
        "class EmpiricalBinary(tfds.core.GeneratorBasedBuilder):\n",
        "    VERSION = tfds.core.Version(\"1.0.0\")\n",
        "    MANUAL_DOWNLOAD_INSTRUCTIONS = \"Place empirical_binary.jsonl in downloads/manual.\"\n",
        "\n",
        "    def _info(self):\n",
        "        return tfds.core.DatasetInfo(\n",
        "            builder=self,\n",
        "            description=\"Binary strings generated by an empirical-urn process.\",\n",
        "            features=tfds.features.FeaturesDict({\"text\": tfds.features.Text()}),\n",
        "        )\n",
        "\n",
        "    def _split_generators(self, dl_manager):\n",
        "        import pathlib, json\n",
        "        j = pathlib.Path(dl_manager.manual_dir) / \"empirical_binary.jsonl\"\n",
        "        if not j.exists():\n",
        "            raise FileNotFoundError(f\"Missing {j}\")\n",
        "        return {\"train\": self._generate_examples(j)}\n",
        "\n",
        "    def _generate_examples(self, jsonl_path):\n",
        "        with open(jsonl_path) as fh:\n",
        "            for i, line in enumerate(fh):\n",
        "                yield i, {\"text\": json.loads(line)[\"text\"]}\n",
        "\n",
        "import pathlib, shutil, tensorflow_datasets as tfds, tensorflow as tf, json\n",
        "\n",
        "# --- Wipe TFRecord build so TFDS doesn't reuse it ---\n",
        "src_data_dir = pathlib.Path(data_path)\n",
        "\n",
        "data_dir = pathlib.Path(work_path) / \"downloads\" / \"manual/\"\n",
        "print (data_dir)\n",
        "\n",
        "src = pathlib.Path(src_data_dir) / \"empirical_binary.jsonl\"\n",
        "dst = data_dir / \"empirical_binary.jsonl\"\n",
        "\n",
        "if not dst.exists():\n",
        "    data_dir.mkdir(parents=True, exist_ok=True)\n",
        "    shutil.copy2(src, dst)\n",
        "    print(\"Copied JSONL from\", src, \"to\", dst)\n",
        "else:\n",
        "    print(\"File already exists at\", dst)\n",
        "    shutil.copy2(src, dst)\n",
        "\n",
        "# 2) Rebuild in default TFRecord format (note: no file_format arg)\n",
        "bd = EmpiricalBinary(data_dir=str(work_path), file_format=\"array_record\")\n",
        "bd.download_and_prepare()  # should say prepared to .../empirical_binary/1.0.0\n",
        "\n",
        "# 3) Now you can use tf.data APIs directly\n",
        "# ds = bd.as_dataset(split=\"train\")  # ✅ works for TFRecord\n",
        "ds = bd.as_data_source(split=\"train\")\n",
        "\n",
        "\n",
        "# Show first 5 examples (raw strings)\n",
        "for i, ex in enumerate(ds):\n",
        "    print(ex[\"text\"])\n",
        "    if i == 4:\n",
        "        break"
      ],
      "metadata": {
        "id": "mSlwNrKzuCzl",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 193,
          "referenced_widgets": [
            "34d608cd88ea49d9a403c3ac5c07868d",
            "43c2cd0f674f47b791d94b8195b81bdb",
            "bfe33eeaf99b4c4ea4ccd972b7118751",
            "0a45990982c04d9a8c0bf171eb632bd6",
            "3de5bdc977354efb8f68b35437e2d2b5",
            "542ee0fc026c431e84b4c7fad0e8ccff",
            "646c0e4a2e8f4de6a70f01b33c77deea",
            "2b7fc0a5378b472e816b91c79d03fbcb",
            "2485519b99ac4293b7ca22199b760b00",
            "7381f6500f904c44b906c5a06c0c7562",
            "bc74d578734946b1a4df1e92169f3f2a",
            "414493525d8e44db83eba6be77239889",
            "f60c451dfb8b468b9590ac52df0e85eb",
            "8da6911a3c5649979c18ce2172d83eee",
            "b8d7fc8bbae44bbc8b471f6b6aacf5e2",
            "3010a10fe5064e6cb48c380d1dcd7909",
            "7b2ae67bc966499f845a49b71c0e9788",
            "f0a6fd867f604ab1a7d5cce66449d0ac",
            "955c082d66354e3fa52385c87bf8ef43",
            "49c1a85dad0f42068928328faeccb051",
            "d21a5e2a32e04c238fe9fe3d7b5c2c4c",
            "3e8365bf757d4a768b4b2cda3fcae7d3",
            "cbb62b1310114fc38af946d0ff2afcfb",
            "0be8ac04651e4bc5b5a8d7c4e12c4d70",
            "3d801cd090d04e55b04b68f912c3525f",
            "7804334613044b948af9e2dfad6ec3e4",
            "649fdba4fbc64a75811f3459abbd0717",
            "e419bc4aa1034c1e90f5e20a40e0c40c",
            "2956579dabd74c62bde5cb644d4ef31f",
            "63693aed932f400fa437b1e1c20f1f1c",
            "e55c41332a98448f98284fbef59bae14",
            "87c62c2fe5f34c04b1ebfa5910c7e26b",
            "b2564e5d26f14e559fcf8bccf5b26d56"
          ]
        },
        "outputId": "c955d790-06f8-4571-ca7a-da258f141180"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "/content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/downloads/manual\n",
            "Copied JSONL from /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/empirical_binary.jsonl to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/downloads/manual/empirical_binary.jsonl\n",
            "Downloading and preparing dataset Unknown size (download: Unknown size, generated: Unknown size, total: Unknown size) to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/empirical_binary/1.0.0...\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "Generating splits...:   0%|          | 0/1 [00:00<?, ? splits/s]"
            ],
            "application/vnd.jupyter.widget-view+json": {
              "version_major": 2,
              "version_minor": 0,
              "model_id": "34d608cd88ea49d9a403c3ac5c07868d"
            }
          },
          "metadata": {}
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "Generating train examples...: 0 examples [00:00, ? examples/s]"
            ],
            "application/vnd.jupyter.widget-view+json": {
              "version_major": 2,
              "version_minor": 0,
              "model_id": "414493525d8e44db83eba6be77239889"
            }
          },
          "metadata": {}
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "Shuffling /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/empirical_binary/incomplete.80OE0H…"
            ],
            "application/vnd.jupyter.widget-view+json": {
              "version_major": 2,
              "version_minor": 0,
              "model_id": "cbb62b1310114fc38af946d0ff2afcfb"
            }
          },
          "metadata": {}
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Dataset empirical_binary downloaded and prepared to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/empirical_binary/1.0.0. Subsequent calls will reuse this data.\n",
            "b'000011011100101000100010100000010001110011011000000110100000100101000101110111000000100010010011100000000001000100010000011000110000000000111010100100000000100000100110010100110110000000000000000000111001010000001000010111000100010100010110001111111000110000011011100000101000101101010110011000010100000001001000000000001001010001000100110000000111010000101011101101000101010010010100010001100000000010000111000111100010101111101100001010001100000000111001011000001010001011001110100100100100001110110100101101101110001010111111111000011001100010111111111111110011111110111100110111001000110001101100011011111000110010101111111001110111101101111110110100100011111100111111111011110101111100110111101111010100111101111111010111111110011111100011111101011111111101011110111100100110011101100111101110100110101101111100011100011101111101101111011100111111111110110110111111110110111001011010111011111101100111011100111111101100101110001111111100101011011111101110011011110011100111011000100101101001110000111100100010001001011'\n",
            "b'100100001000010111000010000100001100000010000010011000000000000010000000001000001000111101010010001000001100000110110010011101000000100000011111000000001010100000011001010100100000010000000011100001011101111010100110011001001000101100010110001101101000100001000010011001000011101100010010000000010000000100100100000000000000000101100011010001100000101000010000000101101010001011011100011000000100000110001010000100011101010100111110011000011001001001101010010101010000000101010100000010101000010000000010100110010011110000101111111110100011111100111001001011000101110111101111100111101010101111111010111111100010001011001101111100011001101011110000111000001111110111110010001100111001111100110101110011001111110100111001111010111111111000011001111101111100001011101010010101111001011100001011111111101011111011110101110110001101111111011010001010101110111101011111111110011111111011110111111111100000001110111011111011111010011110111111001111101111111011111010111101101110100111101011100110111011111111001001011111101101111'\n",
            "b'100001100000000000000001000001000111010000111000000000000011000000000010000000000000110010000100101101010000011000010101001110110000110100010100001100110010110000111000000010010010111010001100000110001010000100111000100000010111101000000010001000101000001000101000000000010100000100010100111100000101011010100100000001100001010100110110011100100100000111101001000011000100011001000001010011000101011010000000000000111011010110101100001000010001100011000010000100001000100101011001100000000000110000111110010111011111111000011111100111011110111111010011110100111110001111111111010111011100011110011111111011111011111011110011011101110111101110100000111000001110111100101111101100000111010111111111001110011111101001110100010011101010101110011111111101010011000110111100000101001111111100111111010111101111111111111011010011100111011111111001011101100111111101011111101111111100101010011111011110110011001010110011101111010100011100101110111110101111001110110101110111110011111111111111110110111100111101110111111111111111011'\n",
            "b'010010100010111110110010110000001100001110100001000111010000010000010010000110110000001101000101000001100010110010001000000110101011000000011100001011100000011010001000000000001001100011000001110101110101010100111000000000000001100001101110010101000111011001000100000100001110111010101101000100000010001000010001100000010010001000101000011000100000110001110010000001010101000000110010101100010010101011100001000010000010001101101010011000110110110000000001000011100000100010010001100000011000000010000110000000011101111111111101111101011001011001010111101100011110011110110101010111000101111100011111100010111111110111110010111110001111111111011011111100001001101000111010001010111111010110010111101110001111011110001110110011111101111001011111100011011110100111110110011111111110100101111101001111111110111111110001001011110111101101111110011111011011010111101101110111011000111100100110010010111111111101110111110111111101101111001010110111101110110000111101100111111110011111111011111010010011111010101001110011101011011'\n",
            "b'111101100001000001000001101011000000001000100001000100001000101011000000001010000001010110111001011000000011110000011101101010001000100011011010010000101000000000001011101111001000101010001100011000000001001111011000100110000000001010101000001000001010100100011100011101100010111100111110010010000001101010111000001000011011001001010000101000101010110100011011001010000010000100110010001010101010010000100000001100000100100000000110011100000000000000010000000001000100000000000111010000000000000011001010000000011001110111111111011000111101011111111110110000110001010101110111010101101101110100010011100011011001111111010011110111000000110001011110111110110100100100101111101010101010100001111010110110000011011100110010111101011010011001111011101111110111011001101010101010111110101111000011001111001111011101111111110111111111110111011111011110101010111100011111111111111101011111111111111010100011011111000110111100110011001101111111111100010110111111110011111011111010010111011011011110101010010101111110000111111110110'\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "### Train"
      ],
      "metadata": {
        "id": "5NXJL5ZBu21N"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "from nanodo.configs.default import get_config\n",
        "from ml_collections import ConfigDict\n",
        "\n",
        "cfg = get_config()                    # cfg is a ml_collections.ConfigDict\n",
        "\n",
        "# If your cfg might be locked:\n",
        "was_locked = getattr(cfg, \"is_locked\", False) and cfg.is_locked\n",
        "if was_locked:\n",
        "    cfg.unlock()\n",
        "\n",
        "def set_model_fp32(cfg):\n",
        "    def as_fp32_like(x):\n",
        "        # match whatever your config currently uses (str or dtype object)\n",
        "        return \"float32\" if isinstance(x, str) else jnp.float32\n",
        "\n",
        "    for key in (\"dtype\", \"param_dtype\", \"compute_dtype\",\n",
        "                \"attn_dtype\", \"qkv_dtype\", \"kv_dtype\", \"ff_dtype\"):\n",
        "        if hasattr(cfg.model, key):\n",
        "            cur = getattr(cfg.model, key)\n",
        "            setattr(cfg.model, key, as_fp32_like(cur) if cur is not None else jnp.float32)\n",
        "    return cfg\n",
        "\n",
        "# — dataset & training —\n",
        "cfg.ds_name           = \"empirical_binary\"\n",
        "cfg.eval_split        = \"train\"\n",
        "cfg.eval_every_steps  = 200\n",
        "cfg.checkpoint        = True\n",
        "cfg.checkpoint_every_steps = 100\n",
        "cfg.max_to_keep = 10\n",
        "\n",
        "# — data loader —\n",
        "cfg.pygrain_worker_count        = 2\n",
        "cfg.pygrain_worker_buffer_size  = 2\n",
        "\n",
        "# — model —\n",
        "#cfg.model = ConfigDict()\n",
        "\n",
        "cfg.model.V = VOCAB_SIZE\n",
        "\n",
        "\n",
        "# tiny: CPU model\n",
        "cfg.model.L   = CONTEXT_LEN          # context length\n",
        "if CONTEXT_LEN == 128:\n",
        "    cfg.model.D = 32    # model/embed dim  = qkv dim\n",
        "    cfg.model.H = 2    # num attention heads L=10, N=2, F=64,\n",
        "    cfg.model.N = 2    # number of transformer block layers\n",
        "    cfg.model.F = cfg.model.D * 4    # FF inner dimension\n",
        "\n",
        "\n",
        "elif CONTEXT_LEN == 1024:\n",
        "    cfg.model.D = 256    # model/embed dim  = qkv dim\n",
        "    cfg.model.H = 4    # num attention heads L=10, N=2, F=64,\n",
        "    cfg.model.N = 3    # number of transformer block layers\n",
        "    cfg.model.F = cfg.model.D * 4    # FF inner dimension\n",
        "\n",
        "else:\n",
        "    raise ValueError(f\"Unknown CONTEXT_LEN: {CONTEXT_LEN}\")\n",
        "\n",
        "cfg.train_epochs      = 1\n",
        "cfg.batch_size        = 128\n",
        "\n",
        "cfg = set_model_fp32(cfg)\n",
        "\n",
        "\n",
        "\n",
        "# — optimisation —\n",
        "\n",
        "cfg.opt.init_learning_rate   = 1e-3\n",
        "cfg.opt.optimizer       = \"adamw\"   # whatever your optimizer.get_optimizer expects\n",
        "\n",
        "# Add/init the optional finetune source\n",
        "if TRAIN_DATA == \"ROBUST_BERNOULLI_TEST1\":\n",
        "\n",
        "    # Optional: how your code branches\n",
        "    cfg.run_mode = \"finetune\"  # {\"scratch\",\"resume\",\"finetune\"}\n",
        "    cfg.init_from = ConfigDict()\n",
        "    cfg.init_from.checkpoint_dir    = work_path  # <- where old weights live\n",
        "    cfg.init_from.step              = \"latest\"    # or an int\n",
        "    cfg.init_from.restore_optimizer = False       # finetune = weights only\n",
        "    cfg.init_from.strict_tree       = True        # fail if shapes don’t match\n",
        "\n",
        "\n",
        "if was_locked:\n",
        "    cfg.lock()\n",
        "# — misc —\n",
        "cfg.seed        = 10\n",
        "cfg.workdir     = work_path\n",
        "#cfg.max_to_keep = 2\n"
      ],
      "metadata": {
        "id": "sOdMJmOTMBkJ"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "train_and_evaluate(cfg)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 1000
        },
        "id": "OWTP_IeiaQWY",
        "outputId": "ebf16658-4035-46a0-c75f-ba97dfe1ea47",
        "collapsed": true
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "ERROR:absl:Orbax `CheckpointManager` is transitioning toward using asynchronous saving logic under the hood in all cases. Users that configure `CheckpointManager` with some `Checkpointer`s and some `AsyncCheckpointer`s will now see asynchronous logic used to save all items. This may result in breakages if the code is assuming that certain objects will be available immediately after saving. Ensure that if you depend on the result of `save` being fully written at a particular moment, use `wait_until_finished()`.\n",
            "/usr/local/lib/python3.12/dist-packages/jax/_src/interpreters/mlir.py:1114: UserWarning: Some donated buffers were not usable: ShapedArray(float32[4,64,256]), ShapedArray(float32[256,4,64]), ShapedArray(float32[256,4,64]), ShapedArray(float32[256,4,64]), ShapedArray(float32[256]), ShapedArray(float32[256]), ShapedArray(float32[256,1024]), ShapedArray(float32[1024,256]), ShapedArray(float32[4,64,256]), ShapedArray(float32[256,4,64]), ShapedArray(float32[256,4,64]), ShapedArray(float32[256,4,64]), ShapedArray(float32[256]), ShapedArray(float32[256]), ShapedArray(float32[256,1024]), ShapedArray(float32[1024,256]), ShapedArray(float32[4,64,256]), ShapedArray(float32[256,4,64]), ShapedArray(float32[256,4,64]), ShapedArray(float32[256,4,64]), ShapedArray(float32[256]), ShapedArray(float32[256]), ShapedArray(float32[256,1024]), ShapedArray(float32[1024,256]), ShapedArray(float32[256]), ShapedArray(float32[1024,256]), ShapedArray(float32[4,256]), ShapedArray(int32[128,1024]).\n",
            "See an explanation at https://jax.readthedocs.io/en/latest/faq.html#buffer-donation.\n",
            "  warnings.warn(\"Some donated buffers were not usable:\"\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "checkpointing\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/local/lib/python3.12/dist-packages/jax/_src/interpreters/mlir.py:1114: UserWarning: Some donated buffers were not usable: ShapedArray(int32[128,1024]).\n",
            "See an explanation at https://jax.readthedocs.io/en/latest/faq.html#buffer-donation.\n",
            "  warnings.warn(\"Some donated buffers were not usable:\"\n",
            "/usr/local/lib/python3.12/dist-packages/jax/_src/interpreters/mlir.py:1114: UserWarning: Some donated buffers were not usable: ShapedArray(int32[128,1024]).\n",
            "See an explanation at https://jax.readthedocs.io/en/latest/faq.html#buffer-donation.\n",
            "  warnings.warn(\"Some donated buffers were not usable:\"\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Step 0: train_loss = 1.3263\n",
            "Step 1: train_loss = 0.7693\n",
            "Step 2: train_loss = 3.9275\n",
            "Step 3: train_loss = 1.8250\n",
            "Step 4: train_loss = 0.9348\n",
            "Step 5: train_loss = 1.4681\n",
            "Step 6: train_loss = 1.1781\n",
            "Step 7: train_loss = 0.7463\n",
            "Step 8: train_loss = 0.7506\n",
            "Step 9: train_loss = 0.9595\n",
            "Step 10: train_loss = 1.0043\n",
            "Step 11: train_loss = 0.8836\n",
            "Step 12: train_loss = 0.7292\n",
            "Step 13: train_loss = 0.6877\n",
            "Step 14: train_loss = 0.7569\n",
            "Step 15: train_loss = 0.8274\n",
            "Step 16: train_loss = 0.8314\n",
            "Step 17: train_loss = 0.7744\n",
            "Step 18: train_loss = 0.7084\n",
            "Step 19: train_loss = 0.6833\n",
            "Step 20: train_loss = 0.7073\n",
            "Step 21: train_loss = 0.7444\n",
            "Step 22: train_loss = 0.7579\n",
            "Step 23: train_loss = 0.7399\n",
            "Step 24: train_loss = 0.7073\n",
            "Step 25: train_loss = 0.6840\n",
            "Step 26: train_loss = 0.6862\n",
            "Step 27: train_loss = 0.7043\n",
            "Step 28: train_loss = 0.7182\n",
            "Step 29: train_loss = 0.7167\n",
            "Step 30: train_loss = 0.7001\n",
            "Step 31: train_loss = 0.6854\n",
            "Step 32: train_loss = 0.6806\n",
            "Step 33: train_loss = 0.6875\n",
            "Step 34: train_loss = 0.6968\n",
            "Step 35: train_loss = 0.6999\n",
            "Step 36: train_loss = 0.6938\n",
            "Step 37: train_loss = 0.6835\n",
            "Step 38: train_loss = 0.6786\n",
            "Step 39: train_loss = 0.6817\n",
            "Step 40: train_loss = 0.6866\n",
            "Step 41: train_loss = 0.6874\n",
            "Step 42: train_loss = 0.6843\n",
            "Step 43: train_loss = 0.6780\n",
            "Step 44: train_loss = 0.6752\n",
            "Step 45: train_loss = 0.6760\n",
            "Step 46: train_loss = 0.6781\n",
            "Step 47: train_loss = 0.6792\n",
            "Step 48: train_loss = 0.6756\n",
            "Step 49: train_loss = 0.6711\n",
            "Step 50: train_loss = 0.6701\n",
            "Step 51: train_loss = 0.6692\n",
            "Step 52: train_loss = 0.6705\n",
            "Step 53: train_loss = 0.6684\n",
            "Step 54: train_loss = 0.6645\n",
            "Step 55: train_loss = 0.6628\n",
            "Step 56: train_loss = 0.6605\n",
            "Step 57: train_loss = 0.6606\n",
            "Step 58: train_loss = 0.6607\n",
            "Step 59: train_loss = 0.6578\n",
            "Step 60: train_loss = 0.6553\n",
            "Step 61: train_loss = 0.6527\n",
            "Step 62: train_loss = 0.6534\n",
            "Step 63: train_loss = 0.6499\n",
            "Step 64: train_loss = 0.6507\n",
            "Step 65: train_loss = 0.6481\n",
            "Step 66: train_loss = 0.6482\n",
            "Step 67: train_loss = 0.6467\n",
            "Step 68: train_loss = 0.6458\n",
            "Step 69: train_loss = 0.6424\n",
            "Step 70: train_loss = 0.6414\n",
            "Step 71: train_loss = 0.6421\n",
            "Step 72: train_loss = 0.6428\n",
            "Step 73: train_loss = 0.6403\n",
            "Step 74: train_loss = 0.6410\n",
            "Step 75: train_loss = 0.6434\n",
            "Step 76: train_loss = 0.6424\n",
            "Step 77: train_loss = 0.6421\n",
            "Step 78: train_loss = 0.6411\n",
            "Step 79: train_loss = 0.6412\n",
            "Step 80: train_loss = 0.6404\n",
            "Step 81: train_loss = 0.6409\n",
            "Step 82: train_loss = 0.6410\n",
            "Step 83: train_loss = 0.6386\n",
            "Step 84: train_loss = 0.6375\n",
            "Step 85: train_loss = 0.6403\n",
            "Step 86: train_loss = 0.6408\n",
            "Step 87: train_loss = 0.6378\n",
            "Step 88: train_loss = 0.6385\n",
            "Step 89: train_loss = 0.6371\n",
            "Step 90: train_loss = 0.6368\n",
            "Step 91: train_loss = 0.6378\n",
            "Step 92: train_loss = 0.6384\n",
            "Step 93: train_loss = 0.6396\n",
            "Step 94: train_loss = 0.6380\n",
            "Step 95: train_loss = 0.6358\n",
            "Step 96: train_loss = 0.6380\n",
            "Step 97: train_loss = 0.6385\n",
            "Step 98: train_loss = 0.6375\n",
            "checkpointing\n",
            "Step 99: train_loss = 0.6375\n",
            "Step 100: train_loss = 0.6381\n",
            "Step 101: train_loss = 0.6376\n",
            "Step 102: train_loss = 0.6380\n",
            "Step 103: train_loss = 0.6380\n",
            "Step 104: train_loss = 0.6370\n",
            "Step 105: train_loss = 0.6385\n",
            "Step 106: train_loss = 0.6387\n",
            "Step 107: train_loss = 0.6359\n",
            "Step 108: train_loss = 0.6386\n",
            "Step 109: train_loss = 0.6386\n",
            "Step 110: train_loss = 0.6382\n",
            "Step 111: train_loss = 0.6404\n",
            "Step 112: train_loss = 0.6376\n",
            "Step 113: train_loss = 0.6384\n",
            "Step 114: train_loss = 0.6373\n",
            "Step 115: train_loss = 0.6381\n",
            "Step 116: train_loss = 0.6394\n",
            "Step 117: train_loss = 0.6367\n",
            "Step 118: train_loss = 0.6364\n",
            "Step 119: train_loss = 0.6367\n",
            "Step 120: train_loss = 0.6383\n",
            "Step 121: train_loss = 0.6382\n",
            "Step 122: train_loss = 0.6379\n",
            "Step 123: train_loss = 0.6366\n",
            "Step 124: train_loss = 0.6375\n",
            "Step 125: train_loss = 0.6371\n",
            "Step 126: train_loss = 0.6371\n",
            "Step 127: train_loss = 0.6373\n",
            "Step 128: train_loss = 0.6395\n",
            "Step 129: train_loss = 0.6376\n",
            "Step 130: train_loss = 0.6357\n",
            "Step 131: train_loss = 0.6359\n",
            "Step 132: train_loss = 0.6373\n",
            "Step 133: train_loss = 0.6371\n",
            "Step 134: train_loss = 0.6359\n",
            "Step 135: train_loss = 0.6371\n",
            "Step 136: train_loss = 0.6375\n",
            "Step 137: train_loss = 0.6371\n",
            "Step 138: train_loss = 0.6376\n",
            "Step 139: train_loss = 0.6371\n",
            "Step 140: train_loss = 0.6383\n",
            "Step 141: train_loss = 0.6363\n",
            "Step 142: train_loss = 0.6380\n",
            "Step 143: train_loss = 0.6388\n",
            "Step 144: train_loss = 0.6378\n",
            "Step 145: train_loss = 0.6370\n",
            "Step 146: train_loss = 0.6384\n",
            "Step 147: train_loss = 0.6373\n",
            "Step 148: train_loss = 0.6359\n",
            "Step 149: train_loss = 0.6360\n",
            "Step 150: train_loss = 0.6366\n",
            "Step 151: train_loss = 0.6360\n",
            "Step 152: train_loss = 0.6366\n",
            "Step 153: train_loss = 0.6374\n",
            "Step 154: train_loss = 0.6370\n",
            "Step 155: train_loss = 0.6383\n",
            "Step 156: train_loss = 0.6361\n",
            "Step 157: train_loss = 0.6358\n",
            "Step 158: train_loss = 0.6377\n",
            "Step 159: train_loss = 0.6364\n",
            "Step 160: train_loss = 0.6371\n",
            "Step 161: train_loss = 0.6372\n",
            "Step 162: train_loss = 0.6360\n",
            "Step 163: train_loss = 0.6389\n",
            "Step 164: train_loss = 0.6383\n",
            "Step 165: train_loss = 0.6377\n",
            "Step 166: train_loss = 0.6369\n",
            "Step 167: train_loss = 0.6361\n",
            "Step 168: train_loss = 0.6375\n",
            "Step 169: train_loss = 0.6375\n",
            "Step 170: train_loss = 0.6371\n",
            "Step 171: train_loss = 0.6373\n",
            "Step 172: train_loss = 0.6372\n",
            "Step 173: train_loss = 0.6378\n",
            "Step 174: train_loss = 0.6373\n",
            "Step 175: train_loss = 0.6364\n",
            "Step 176: train_loss = 0.6366\n",
            "Step 177: train_loss = 0.6377\n",
            "Step 178: train_loss = 0.6369\n",
            "Step 179: train_loss = 0.6353\n",
            "Step 180: train_loss = 0.6359\n",
            "Step 181: train_loss = 0.6375\n",
            "Step 182: train_loss = 0.6359\n",
            "Step 183: train_loss = 0.6359\n",
            "Step 184: train_loss = 0.6355\n",
            "Step 185: train_loss = 0.6370\n",
            "Step 186: train_loss = 0.6366\n",
            "Step 187: train_loss = 0.6366\n",
            "Step 188: train_loss = 0.6368\n",
            "Step 189: train_loss = 0.6371\n",
            "Step 190: train_loss = 0.6370\n",
            "Step 191: train_loss = 0.6371\n",
            "Step 192: train_loss = 0.6379\n",
            "Step 193: train_loss = 0.6365\n",
            "Step 194: train_loss = 0.6342\n",
            "Step 195: train_loss = 0.6372\n",
            "Step 196: train_loss = 0.6372\n",
            "Step 197: train_loss = 0.6369\n",
            "Step 198: train_loss = 0.6357\n",
            "checkpointing\n",
            "Step 199: train_loss = 0.6372\n",
            "Step 200: train_loss = 0.6350\n",
            "Step 201: train_loss = 0.6368\n",
            "Step 202: train_loss = 0.6382\n",
            "Step 203: train_loss = 0.6379\n",
            "Step 204: train_loss = 0.6363\n",
            "Step 205: train_loss = 0.6370\n",
            "Step 206: train_loss = 0.6359\n",
            "Step 207: train_loss = 0.6371\n",
            "Step 208: train_loss = 0.6370\n",
            "Step 209: train_loss = 0.6354\n",
            "Step 210: train_loss = 0.6375\n",
            "Step 211: train_loss = 0.6375\n",
            "Step 212: train_loss = 0.6362\n",
            "Step 213: train_loss = 0.6363\n",
            "Step 214: train_loss = 0.6369\n",
            "Step 215: train_loss = 0.6368\n",
            "Step 216: train_loss = 0.6375\n",
            "Step 217: train_loss = 0.6364\n",
            "Step 218: train_loss = 0.6364\n",
            "Step 219: train_loss = 0.6361\n",
            "Step 220: train_loss = 0.6361\n",
            "Step 221: train_loss = 0.6364\n",
            "Step 222: train_loss = 0.6351\n",
            "Step 223: train_loss = 0.6380\n",
            "Step 224: train_loss = 0.6368\n",
            "Step 225: train_loss = 0.6357\n",
            "Step 226: train_loss = 0.6388\n",
            "Step 227: train_loss = 0.6366\n",
            "Step 228: train_loss = 0.6373\n",
            "Step 229: train_loss = 0.6373\n",
            "Step 230: train_loss = 0.6357\n",
            "Step 231: train_loss = 0.6370\n",
            "Step 232: train_loss = 0.6367\n",
            "Step 233: train_loss = 0.6354\n",
            "Step 234: train_loss = 0.6373\n",
            "Step 235: train_loss = 0.6347\n",
            "Step 236: train_loss = 0.6363\n",
            "Step 237: train_loss = 0.6365\n",
            "Step 238: train_loss = 0.6369\n",
            "Step 239: train_loss = 0.6371\n",
            "Step 240: train_loss = 0.6379\n",
            "Step 241: train_loss = 0.6364\n",
            "Step 242: train_loss = 0.6358\n",
            "Step 243: train_loss = 0.6368\n",
            "Step 244: train_loss = 0.6358\n",
            "Step 245: train_loss = 0.6370\n",
            "Step 246: train_loss = 0.6372\n",
            "Step 247: train_loss = 0.6380\n",
            "Step 248: train_loss = 0.6351\n",
            "Step 249: train_loss = 0.6372\n",
            "Step 250: train_loss = 0.6368\n",
            "Step 251: train_loss = 0.6359\n",
            "Step 252: train_loss = 0.6348\n",
            "Step 253: train_loss = 0.6363\n",
            "Step 254: train_loss = 0.6377\n",
            "Step 255: train_loss = 0.6358\n",
            "Step 256: train_loss = 0.6359\n",
            "Step 257: train_loss = 0.6370\n",
            "Step 258: train_loss = 0.6374\n",
            "Step 259: train_loss = 0.6362\n",
            "Step 260: train_loss = 0.6381\n",
            "Step 261: train_loss = 0.6365\n",
            "Step 262: train_loss = 0.6375\n",
            "Step 263: train_loss = 0.6366\n",
            "Step 264: train_loss = 0.6369\n",
            "Step 265: train_loss = 0.6371\n",
            "Step 266: train_loss = 0.6401\n",
            "Step 267: train_loss = 0.6375\n",
            "Step 268: train_loss = 0.6342\n",
            "Step 269: train_loss = 0.6374\n",
            "Step 270: train_loss = 0.6364\n",
            "Step 271: train_loss = 0.6365\n",
            "Step 272: train_loss = 0.6362\n",
            "Step 273: train_loss = 0.6375\n",
            "Step 274: train_loss = 0.6379\n",
            "Step 275: train_loss = 0.6362\n",
            "Step 276: train_loss = 0.6376\n",
            "Step 277: train_loss = 0.6371\n",
            "Step 278: train_loss = 0.6377\n",
            "Step 279: train_loss = 0.6385\n",
            "Step 280: train_loss = 0.6368\n",
            "Step 281: train_loss = 0.6361\n",
            "Step 282: train_loss = 0.6367\n",
            "Step 283: train_loss = 0.6368\n",
            "Step 284: train_loss = 0.6355\n",
            "Step 285: train_loss = 0.6362\n",
            "Step 286: train_loss = 0.6372\n",
            "Step 287: train_loss = 0.6361\n",
            "Step 288: train_loss = 0.6372\n",
            "Step 289: train_loss = 0.6345\n",
            "Step 290: train_loss = 0.6360\n",
            "Step 291: train_loss = 0.6372\n",
            "Step 292: train_loss = 0.6360\n",
            "Step 293: train_loss = 0.6368\n",
            "Step 294: train_loss = 0.6354\n",
            "Step 295: train_loss = 0.6359\n",
            "Step 296: train_loss = 0.6363\n",
            "Step 297: train_loss = 0.6360\n",
            "Step 298: train_loss = 0.6370\n",
            "checkpointing\n",
            "Step 299: train_loss = 0.6384\n",
            "Step 300: train_loss = 0.6423\n",
            "Step 301: train_loss = 0.6425\n",
            "Step 302: train_loss = 0.6381\n",
            "Step 303: train_loss = 0.6362\n",
            "Step 304: train_loss = 0.6401\n",
            "Step 305: train_loss = 0.6408\n",
            "Step 306: train_loss = 0.6389\n",
            "Step 307: train_loss = 0.6364\n",
            "Step 308: train_loss = 0.6372\n",
            "Step 309: train_loss = 0.6397\n",
            "Step 310: train_loss = 0.6393\n",
            "Step 311: train_loss = 0.6359\n",
            "Step 312: train_loss = 0.6373\n",
            "Step 313: train_loss = 0.6379\n",
            "Step 314: train_loss = 0.6375\n",
            "Step 315: train_loss = 0.6373\n",
            "Step 316: train_loss = 0.6382\n",
            "Step 317: train_loss = 0.6364\n",
            "Step 318: train_loss = 0.6371\n",
            "Step 319: train_loss = 0.6386\n",
            "Step 320: train_loss = 0.6377\n",
            "Step 321: train_loss = 0.6361\n",
            "Step 322: train_loss = 0.6359\n",
            "Step 323: train_loss = 0.6377\n",
            "Step 324: train_loss = 0.6397\n",
            "Step 325: train_loss = 0.6372\n",
            "Step 326: train_loss = 0.6364\n",
            "Step 327: train_loss = 0.6383\n",
            "Step 328: train_loss = 0.6397\n",
            "Step 329: train_loss = 0.6359\n",
            "Step 330: train_loss = 0.6377\n",
            "Step 331: train_loss = 0.6384\n",
            "Step 332: train_loss = 0.6373\n",
            "Step 333: train_loss = 0.6372\n",
            "Step 334: train_loss = 0.6388\n",
            "Step 335: train_loss = 0.6363\n",
            "Step 336: train_loss = 0.6366\n",
            "Step 337: train_loss = 0.6370\n",
            "Step 338: train_loss = 0.6386\n",
            "Step 339: train_loss = 0.6357\n",
            "Step 340: train_loss = 0.6372\n",
            "Step 341: train_loss = 0.6365\n",
            "Step 342: train_loss = 0.6377\n",
            "Step 343: train_loss = 0.6362\n",
            "Step 344: train_loss = 0.6358\n",
            "Step 345: train_loss = 0.6384\n",
            "Step 346: train_loss = 0.6370\n",
            "Step 347: train_loss = 0.6380\n",
            "Step 348: train_loss = 0.6369\n",
            "Step 349: train_loss = 0.6360\n",
            "Step 350: train_loss = 0.6380\n",
            "Step 351: train_loss = 0.6375\n",
            "Step 352: train_loss = 0.6370\n",
            "Step 353: train_loss = 0.6359\n",
            "Step 354: train_loss = 0.6385\n",
            "Step 355: train_loss = 0.6365\n",
            "Step 356: train_loss = 0.6363\n",
            "Step 357: train_loss = 0.6351\n",
            "Step 358: train_loss = 0.6393\n",
            "Step 359: train_loss = 0.6377\n",
            "Step 360: train_loss = 0.6359\n",
            "Step 361: train_loss = 0.6372\n",
            "Step 362: train_loss = 0.6378\n",
            "Step 363: train_loss = 0.6398\n",
            "Step 364: train_loss = 0.6401\n",
            "Step 365: train_loss = 0.6382\n",
            "Step 366: train_loss = 0.6371\n",
            "Step 367: train_loss = 0.6395\n",
            "Step 368: train_loss = 0.6391\n",
            "Step 369: train_loss = 0.6386\n",
            "Step 370: train_loss = 0.6359\n",
            "Step 371: train_loss = 0.6389\n",
            "Step 372: train_loss = 0.6395\n",
            "Step 373: train_loss = 0.6378\n",
            "Step 374: train_loss = 0.6379\n",
            "Step 375: train_loss = 0.6383\n",
            "Step 376: train_loss = 0.6373\n",
            "Step 377: train_loss = 0.6362\n",
            "Step 378: train_loss = 0.6372\n",
            "Step 379: train_loss = 0.6373\n",
            "Step 380: train_loss = 0.6365\n",
            "Step 381: train_loss = 0.6357\n",
            "Step 382: train_loss = 0.6383\n",
            "Step 383: train_loss = 0.6401\n",
            "Step 384: train_loss = 0.6382\n",
            "Step 385: train_loss = 0.6375\n",
            "Step 386: train_loss = 0.6379\n",
            "Step 387: train_loss = 0.6395\n",
            "Step 388: train_loss = 0.6378\n",
            "Step 389: train_loss = 0.6358\n",
            "Step 390: train_loss = 0.6392\n",
            "Step 391: train_loss = 0.6373\n",
            "Step 392: train_loss = 0.6366\n",
            "Step 393: train_loss = 0.6385\n",
            "Step 394: train_loss = 0.6376\n",
            "Step 395: train_loss = 0.6367\n",
            "Step 396: train_loss = 0.6373\n",
            "Step 397: train_loss = 0.6398\n",
            "Step 398: train_loss = 0.6364\n",
            "checkpointing\n",
            "Step 399: train_loss = 0.6371\n",
            "Step 400: train_loss = 0.6400\n",
            "Step 401: train_loss = 0.6370\n",
            "Step 402: train_loss = 0.6365\n",
            "Step 403: train_loss = 0.6391\n",
            "Step 404: train_loss = 0.6378\n",
            "Step 405: train_loss = 0.6350\n",
            "Step 406: train_loss = 0.6379\n",
            "Step 407: train_loss = 0.6396\n",
            "Step 408: train_loss = 0.6366\n",
            "Step 409: train_loss = 0.6368\n",
            "Step 410: train_loss = 0.6393\n",
            "Step 411: train_loss = 0.6349\n",
            "Step 412: train_loss = 0.6369\n",
            "Step 413: train_loss = 0.6381\n",
            "Step 414: train_loss = 0.6374\n",
            "Step 415: train_loss = 0.6365\n",
            "Step 416: train_loss = 0.6367\n",
            "Step 417: train_loss = 0.6378\n",
            "Step 418: train_loss = 0.6371\n",
            "Step 419: train_loss = 0.6370\n",
            "Step 420: train_loss = 0.6369\n",
            "Step 421: train_loss = 0.6365\n",
            "Step 422: train_loss = 0.6357\n",
            "Step 423: train_loss = 0.6367\n",
            "Step 424: train_loss = 0.6363\n",
            "Step 425: train_loss = 0.6356\n",
            "Step 426: train_loss = 0.6359\n",
            "Step 427: train_loss = 0.6366\n",
            "Step 428: train_loss = 0.6366\n",
            "Step 429: train_loss = 0.6364\n",
            "Step 430: train_loss = 0.6376\n",
            "Step 431: train_loss = 0.6360\n",
            "Step 432: train_loss = 0.6366\n",
            "Step 433: train_loss = 0.6384\n",
            "Step 434: train_loss = 0.6379\n",
            "Step 435: train_loss = 0.6372\n",
            "Step 436: train_loss = 0.6376\n",
            "Step 437: train_loss = 0.6357\n",
            "Step 438: train_loss = 0.6353\n",
            "Step 439: train_loss = 0.6374\n",
            "Step 440: train_loss = 0.6377\n",
            "Step 441: train_loss = 0.6376\n",
            "Step 442: train_loss = 0.6381\n",
            "Step 443: train_loss = 0.6358\n",
            "Step 444: train_loss = 0.6376\n",
            "Step 445: train_loss = 0.6380\n",
            "Step 446: train_loss = 0.6366\n",
            "Step 447: train_loss = 0.6348\n",
            "Step 448: train_loss = 0.6360\n",
            "Step 449: train_loss = 0.6371\n",
            "Step 450: train_loss = 0.6389\n",
            "Step 451: train_loss = 0.6374\n",
            "Step 452: train_loss = 0.6376\n",
            "Step 453: train_loss = 0.6370\n",
            "Step 454: train_loss = 0.6363\n",
            "Step 455: train_loss = 0.6381\n",
            "Step 456: train_loss = 0.6362\n",
            "Step 457: train_loss = 0.6364\n",
            "Step 458: train_loss = 0.6373\n",
            "Step 459: train_loss = 0.6372\n",
            "Step 460: train_loss = 0.6366\n",
            "Step 461: train_loss = 0.6361\n",
            "Step 462: train_loss = 0.6361\n",
            "Step 463: train_loss = 0.6363\n",
            "Step 464: train_loss = 0.6387\n",
            "Step 465: train_loss = 0.6380\n",
            "Step 466: train_loss = 0.6367\n",
            "Step 467: train_loss = 0.6366\n",
            "Step 468: train_loss = 0.6373\n",
            "Step 469: train_loss = 0.6367\n",
            "Step 470: train_loss = 0.6364\n",
            "Step 471: train_loss = 0.6368\n",
            "Step 472: train_loss = 0.6370\n",
            "Step 473: train_loss = 0.6373\n",
            "Step 474: train_loss = 0.6376\n",
            "Step 475: train_loss = 0.6376\n",
            "Step 476: train_loss = 0.6379\n",
            "Step 477: train_loss = 0.6394\n",
            "Step 478: train_loss = 0.6360\n",
            "Step 479: train_loss = 0.6370\n",
            "Step 480: train_loss = 0.6365\n",
            "Step 481: train_loss = 0.6356\n",
            "Step 482: train_loss = 0.6356\n",
            "Step 483: train_loss = 0.6382\n",
            "Step 484: train_loss = 0.6361\n",
            "Step 485: train_loss = 0.6370\n",
            "Step 486: train_loss = 0.6364\n",
            "Step 487: train_loss = 0.6373\n",
            "Step 488: train_loss = 0.6372\n",
            "Step 489: train_loss = 0.6374\n",
            "Step 490: train_loss = 0.6365\n",
            "Step 491: train_loss = 0.6358\n",
            "Step 492: train_loss = 0.6359\n",
            "Step 493: train_loss = 0.6353\n",
            "Step 494: train_loss = 0.6371\n",
            "Step 495: train_loss = 0.6365\n",
            "Step 496: train_loss = 0.6382\n",
            "Step 497: train_loss = 0.6370\n",
            "Step 498: train_loss = 0.6375\n",
            "checkpointing\n",
            "Step 499: train_loss = 0.6366\n",
            "Step 500: train_loss = 0.6388\n",
            "Step 501: train_loss = 0.6391\n",
            "Step 502: train_loss = 0.6362\n",
            "Step 503: train_loss = 0.6376\n",
            "Step 504: train_loss = 0.6373\n",
            "Step 505: train_loss = 0.6360\n",
            "Step 506: train_loss = 0.6387\n",
            "Step 507: train_loss = 0.6358\n",
            "Step 508: train_loss = 0.6384\n",
            "Step 509: train_loss = 0.6387\n",
            "Step 510: train_loss = 0.6405\n",
            "Step 511: train_loss = 0.6415\n",
            "Step 512: train_loss = 0.6380\n",
            "Step 513: train_loss = 0.6376\n",
            "Step 514: train_loss = 0.6395\n",
            "Step 515: train_loss = 0.6388\n",
            "Step 516: train_loss = 0.6371\n",
            "Step 517: train_loss = 0.6383\n",
            "Step 518: train_loss = 0.6371\n",
            "Step 519: train_loss = 0.6386\n",
            "Step 520: train_loss = 0.6373\n",
            "Step 521: train_loss = 0.6367\n",
            "Step 522: train_loss = 0.6384\n",
            "Step 523: train_loss = 0.6371\n",
            "Step 524: train_loss = 0.6374\n",
            "Step 525: train_loss = 0.6359\n",
            "Step 526: train_loss = 0.6365\n",
            "Step 527: train_loss = 0.6368\n",
            "Step 528: train_loss = 0.6358\n",
            "Step 529: train_loss = 0.6368\n",
            "Step 530: train_loss = 0.6362\n",
            "Step 531: train_loss = 0.6379\n",
            "Step 532: train_loss = 0.6363\n",
            "Step 533: train_loss = 0.6370\n"
          ]
        },
        {
          "output_type": "error",
          "ename": "KeyboardInterrupt",
          "evalue": "",
          "traceback": [
            "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
            "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/clu/metric_writers/async_writer.py\u001b[0m in \u001b[0;36mensure_flushes\u001b[0;34m(*writers)\u001b[0m\n\u001b[1;32m    146\u001b[0m     \u001b[0;31m# the first writer to stay backwards compatible for a single writer.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 147\u001b[0;31m     \u001b[0;32myield\u001b[0m \u001b[0mwriters\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    148\u001b[0m   \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/tmp/ipython-input-3666103795.py\u001b[0m in \u001b[0;36mtrain_and_evaluate\u001b[0;34m(c)\u001b[0m\n\u001b[1;32m    294\u001b[0m       \u001b[0;31m# asynchronously dispatched the new step.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 295\u001b[0;31m       \u001b[0m_process_metrics\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mstep\u001b[0m \u001b[0;34m-\u001b[0m \u001b[0;36m1\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mpending_microbatch_metrics\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    296\u001b[0m       \u001b[0mpending_microbatch_metrics\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnew_microbatch_metrics\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/tmp/ipython-input-3666103795.py\u001b[0m in \u001b[0;36m_process_metrics\u001b[0;34m(step, microbatch_metrics)\u001b[0m\n\u001b[1;32m    247\u001b[0m       \u001b[0;32mif\u001b[0m \u001b[0mmicrobatch_metrics\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0mstep\u001b[0m \u001b[0;34m%\u001b[0m \u001b[0mc\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mwrite_train_metrics_every_steps\u001b[0m \u001b[0;34m==\u001b[0m \u001b[0;36m0\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 248\u001b[0;31m         microbatch_metrics = [trainer.get_metrics(step, m)\n\u001b[0m\u001b[1;32m    249\u001b[0m                               for m in microbatch_metrics]\n",
            "\u001b[0;32m/tmp/ipython-input-3666103795.py\u001b[0m in \u001b[0;36mget_metrics\u001b[0;34m(self, step, metrics)\u001b[0m\n\u001b[1;32m    349\u001b[0m     \u001b[0;31m# Grab the (possibly previous step's) metrics from device.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 350\u001b[0;31m     \u001b[0mmetrics\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mjax\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdevice_get\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mmetrics\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    351\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0mstep\u001b[0m \u001b[0;34m==\u001b[0m \u001b[0;36m0\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/jax/_src/api.py\u001b[0m in \u001b[0;36mdevice_get\u001b[0;34m(x)\u001b[0m\n\u001b[1;32m   2483\u001b[0m         \u001b[0;32mpass\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2484\u001b[0;31m     \u001b[0;32mreturn\u001b[0m \u001b[0mtree_map\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0m_device_get\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mx\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2485\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/jax/_src/tree_util.py\u001b[0m in \u001b[0;36mtree_map\u001b[0;34m(f, tree, is_leaf, *rest)\u001b[0m\n\u001b[1;32m    343\u001b[0m   \u001b[0mall_leaves\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0mleaves\u001b[0m\u001b[0;34m]\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0mtreedef\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mflatten_up_to\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mr\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mfor\u001b[0m \u001b[0mr\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mrest\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 344\u001b[0;31m   \u001b[0;32mreturn\u001b[0m \u001b[0mtreedef\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0munflatten\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mf\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0mxs\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mfor\u001b[0m \u001b[0mxs\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mzip\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0mall_leaves\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    345\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/jax/_src/tree_util.py\u001b[0m in \u001b[0;36m<genexpr>\u001b[0;34m(.0)\u001b[0m\n\u001b[1;32m    343\u001b[0m   \u001b[0mall_leaves\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0mleaves\u001b[0m\u001b[0;34m]\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0mtreedef\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mflatten_up_to\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mr\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mfor\u001b[0m \u001b[0mr\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mrest\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 344\u001b[0;31m   \u001b[0;32mreturn\u001b[0m \u001b[0mtreedef\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0munflatten\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mf\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0mxs\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mfor\u001b[0m \u001b[0mxs\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mzip\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0mall_leaves\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    345\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/jax/_src/api.py\u001b[0m in \u001b[0;36m_device_get\u001b[0;34m(x)\u001b[0m\n\u001b[1;32m   2444\u001b[0m   \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2445\u001b[0;31m     \u001b[0;32mreturn\u001b[0m \u001b[0mtoarray\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2446\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/jax/_src/array.py\u001b[0m in \u001b[0;36m__array__\u001b[0;34m(self, dtype, context, copy)\u001b[0m\n\u001b[1;32m    428\u001b[0m     \u001b[0mkwds\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m{\u001b[0m\u001b[0;34m}\u001b[0m \u001b[0;32mif\u001b[0m \u001b[0mcopy\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mNone\u001b[0m \u001b[0;32melse\u001b[0m \u001b[0;34m{\u001b[0m\u001b[0;34m'copy'\u001b[0m\u001b[0;34m:\u001b[0m \u001b[0mcopy\u001b[0m\u001b[0;34m}\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 429\u001b[0;31m     \u001b[0;32mreturn\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0masarray\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_value\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdtype\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mdtype\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwds\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    430\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/jax/_src/profiler.py\u001b[0m in \u001b[0;36mwrapper\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m    332\u001b[0m     \u001b[0;32mwith\u001b[0m \u001b[0mTraceAnnotation\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mname\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mdecorator_kwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 333\u001b[0;31m       \u001b[0;32mreturn\u001b[0m \u001b[0mfunc\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    334\u001b[0m     \u001b[0;32mreturn\u001b[0m \u001b[0mwrapper\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/jax/_src/array.py\u001b[0m in \u001b[0;36m_value\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    627\u001b[0m       \u001b[0;32mif\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mis_fully_replicated\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 628\u001b[0;31m         \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_npy_value\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_single_device_array_to_np_array\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    629\u001b[0m         \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_npy_value\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mflags\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mwriteable\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mFalse\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;31mKeyboardInterrupt\u001b[0m: ",
            "\nDuring handling of the above exception, another exception occurred:\n",
            "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
            "\u001b[0;32m/tmp/ipython-input-1577672595.py\u001b[0m in \u001b[0;36m<cell line: 0>\u001b[0;34m()\u001b[0m\n\u001b[0;32m----> 1\u001b[0;31m \u001b[0mtrain_and_evaluate\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mcfg\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
            "\u001b[0;32m/tmp/ipython-input-3666103795.py\u001b[0m in \u001b[0;36mtrain_and_evaluate\u001b[0;34m(c)\u001b[0m\n\u001b[1;32m    231\u001b[0m     \u001b[0mhooks\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m[\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    232\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 233\u001b[0;31m   \u001b[0;32mwith\u001b[0m \u001b[0mmetric_writers\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mensure_flushes\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mwriter\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    234\u001b[0m     \u001b[0;32mdef\u001b[0m \u001b[0m_eval\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    235\u001b[0m       \u001b[0;32mwith\u001b[0m \u001b[0mreport_progress\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtimed\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m\"eval\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/lib/python3.12/contextlib.py\u001b[0m in \u001b[0;36m__exit__\u001b[0;34m(self, typ, value, traceback)\u001b[0m\n\u001b[1;32m    156\u001b[0m                 \u001b[0mvalue\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtyp\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    157\u001b[0m             \u001b[0;32mtry\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 158\u001b[0;31m                 \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mgen\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mthrow\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mvalue\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    159\u001b[0m             \u001b[0;32mexcept\u001b[0m \u001b[0mStopIteration\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0mexc\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    160\u001b[0m                 \u001b[0;31m# Suppress StopIteration *unless* it's the same exception that\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/clu/metric_writers/async_writer.py\u001b[0m in \u001b[0;36mensure_flushes\u001b[0;34m(*writers)\u001b[0m\n\u001b[1;32m    148\u001b[0m   \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    149\u001b[0m     \u001b[0;32mfor\u001b[0m \u001b[0mwriter\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mwriters\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 150\u001b[0;31m       \u001b[0mwriter\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mflush\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/clu/metric_writers/multi_writer.py\u001b[0m in \u001b[0;36mflush\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m     70\u001b[0m   \u001b[0;32mdef\u001b[0m \u001b[0mflush\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     71\u001b[0m     \u001b[0;32mfor\u001b[0m \u001b[0mw\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_writers\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 72\u001b[0;31m       \u001b[0mw\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mflush\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     73\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     74\u001b[0m   \u001b[0;32mdef\u001b[0m \u001b[0mclose\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/clu/metric_writers/async_writer.py\u001b[0m in \u001b[0;36mflush\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    118\u001b[0m   \u001b[0;32mdef\u001b[0m \u001b[0mflush\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    119\u001b[0m     \u001b[0;32mtry\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 120\u001b[0;31m       \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_pool\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mjoin\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    121\u001b[0m     \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    122\u001b[0m       \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_writer\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mflush\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/local/lib/python3.12/dist-packages/clu/asynclib.py\u001b[0m in \u001b[0;36mjoin\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m     89\u001b[0m       \u001b[0mexceptions\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0mreraised\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     90\u001b[0m     \"\"\"\n\u001b[0;32m---> 91\u001b[0;31m     \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_pool\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mshutdown\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mwait\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mTrue\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     92\u001b[0m     self._pool = concurrent.futures.ThreadPoolExecutor(\n\u001b[1;32m     93\u001b[0m         \u001b[0mmax_workers\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_max_workers\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/lib/python3.12/concurrent/futures/thread.py\u001b[0m in \u001b[0;36mshutdown\u001b[0;34m(self, wait, cancel_futures)\u001b[0m\n\u001b[1;32m    237\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mwait\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    238\u001b[0m             \u001b[0;32mfor\u001b[0m \u001b[0mt\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_threads\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 239\u001b[0;31m                 \u001b[0mt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mjoin\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    240\u001b[0m     \u001b[0mshutdown\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m__doc__\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0m_base\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mExecutor\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mshutdown\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m__doc__\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/lib/python3.12/threading.py\u001b[0m in \u001b[0;36mjoin\u001b[0;34m(self, timeout)\u001b[0m\n\u001b[1;32m   1147\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1148\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mtimeout\u001b[0m \u001b[0;32mis\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1149\u001b[0;31m             \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_wait_for_tstate_lock\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1150\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1151\u001b[0m             \u001b[0;31m# the behavior of a negative timeout isn't documented, but\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/usr/lib/python3.12/threading.py\u001b[0m in \u001b[0;36m_wait_for_tstate_lock\u001b[0;34m(self, block, timeout)\u001b[0m\n\u001b[1;32m   1167\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1168\u001b[0m         \u001b[0;32mtry\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1169\u001b[0;31m             \u001b[0;32mif\u001b[0m \u001b[0mlock\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0macquire\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mblock\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mtimeout\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1170\u001b[0m                 \u001b[0mlock\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrelease\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1171\u001b[0m                 \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_stop\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Evaluate Regret"
      ],
      "metadata": {
        "id": "J5YS6eyShy4c"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "print(work_path)"
      ],
      "metadata": {
        "id": "DRfpyEOrazE9"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "def count_prompt_len(token_ids, pad_id):\n",
        "    # token_ids: jnp.ndarray shape (1, L)\n",
        "    ids = jax.device_get(token_ids)[0].tolist()\n",
        "    n = 0\n",
        "    for t in ids:\n",
        "        if t == pad_id:\n",
        "            break\n",
        "        n += 1\n",
        "    return n\n",
        "\n",
        "import jax.numpy as jnp\n",
        "import jax\n",
        "\n",
        "def predict_next_token(model, state, token_ids, pad_id, temperature=1.0):\n",
        "    # token_ids: (1, L). We only look at the boundary position.\n",
        "    logits = model.apply({\"params\": state.params}, token_ids)  # (1, L, V)\n",
        "    n = count_prompt_len(token_ids, pad_id)\n",
        "    step_logits = logits[0, n-1]  # next-token distribution for position n\n",
        "\n",
        "    if temperature == 0:\n",
        "        next_id = int(jnp.argmax(step_logits))\n",
        "    else:\n",
        "        step_logits = step_logits / max(1e-6, float(temperature))\n",
        "        probs = jax.nn.softmax(step_logits)\n",
        "        key = jax.random.PRNGKey(0)  # replace with your key mgmt\n",
        "        next_id = int(jax.random.categorical(key, jnp.log(probs)))\n",
        "    return next_id\n",
        "\n",
        "def generate(model, state, prompt_str, config, max_new_tokens=128, temperature=0.7):\n",
        "    ids = string_to_token_ids(prompt_str, pad_len=config.model.L)  # (1, L)\n",
        "\n",
        "    n = count_prompt_len(ids, PAD_ID)\n",
        "    for _ in range(max_new_tokens):\n",
        "        logits = model.apply({\"params\": state.params}, ids)  # (1, L, V)\n",
        "        step_logits = logits[0, n-1]\n",
        "        if temperature == 0:\n",
        "            next_id = int(jnp.argmax(step_logits))\n",
        "        else:\n",
        "            step_logits = step_logits / max(1e-6, float(temperature))\n",
        "            probs = jax.nn.softmax(step_logits)\n",
        "            key = jax.random.PRNGKey(_)  # replace with proper RNG stream\n",
        "            next_id = int(jax.random.categorical(key, jnp.log(probs)))\n",
        "\n",
        "        # # stop if EOS (if you have one)\n",
        "        # if next_id == EOS_ID:\n",
        "        #     break\n",
        "\n",
        "        # write next_id at position n (the first pad slot)\n",
        "        ids = ids.at[0, n].set(next_id)\n",
        "        n += 1\n",
        "        if n >= config.model.L:\n",
        "            break\n",
        "\n",
        "    # Convert only the **prompt+new tokens** (not the whole padded length)\n",
        "    generated = token_ids_to_string(ids[0, :n])\n",
        "    return generated\n",
        "\n"
      ],
      "metadata": {
        "id": "OoF8dkLe3nI4"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "import numpy as np\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "from scipy.stats import norm\n",
        "import pathlib, time, json, math\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "# ------------------------- utilities -------------------------\n",
        "\n",
        "EPS_CLIP = 1e-9\n",
        "PHASE = np.pi / 6\n",
        "\n",
        "def _resolve_phi(phi_choice, L: int) -> float:\n",
        "    \"\"\"Accepts 'L/20' style or numeric and returns a float phi.\"\"\"\n",
        "    if isinstance(phi_choice, (int, float)):\n",
        "        return float(phi_choice)\n",
        "    if isinstance(phi_choice, str) and phi_choice.startswith(\"L/\"):\n",
        "        denom = float(phi_choice.split(\"/\", 1)[1])\n",
        "        return L / denom\n",
        "    raise ValueError(f\"Unrecognized phi choice: {phi_choice!r}\")\n",
        "\n",
        "def _rng_to_np(rng):\n",
        "    # Accepts either np.random.Generator or python's random.Random or None\n",
        "    if isinstance(rng, np.random.Generator):\n",
        "        return rng\n",
        "    import random as _py_random\n",
        "    if isinstance(rng, _py_random.Random):\n",
        "        return np.random.default_rng(rng.randrange(2**63 - 1))\n",
        "    # fallback: fresh generator\n",
        "    return np.random.default_rng(None)\n",
        "\n",
        "# --------------------- data generation ----------------------\n",
        "\n",
        "def generate_nonstationary_string(\n",
        "    L: int,\n",
        "    *,\n",
        "    dist: str,\n",
        "    phi_choice: str | float | None = None,\n",
        "    phase: float = PHASE,\n",
        "    rng=None,\n",
        ") -> str:\n",
        "    \"\"\"Sample binary string of length L with the requested non-stationary distribution.\"\"\"\n",
        "    np_rng = _rng_to_np(rng)\n",
        "    half = L // 2\n",
        "\n",
        "    if dist == \"2:3-1:3\":\n",
        "        first_half  = np_rng.binomial(1, 2.0/3.0, size=half)\n",
        "        second_half = np_rng.binomial(1, 1.0/3.0, size=L - half)\n",
        "        full = np.concatenate([first_half, second_half])\n",
        "\n",
        "    elif dist == \"1:3-2:3\":\n",
        "        first_half  = np_rng.binomial(1, 1.0/3.0, size=half)\n",
        "        second_half = np_rng.binomial(1, 2.0/3.0, size=L - half)\n",
        "        full = np.concatenate([first_half, second_half])\n",
        "\n",
        "    elif dist == \"1:3\":\n",
        "        full = np_rng.binomial(1, 1.0/3.0, size=L)\n",
        "\n",
        "    elif dist == \"2:3\":\n",
        "        full = np_rng.binomial(1, 2.0/3.0, size=L)\n",
        "\n",
        "    elif dist == \"SINUSOID\":\n",
        "        if phi_choice is None:\n",
        "            raise ValueError(\"SINUSOID requires phi_choice.\")\n",
        "        phi = _resolve_phi(phi_choice, L)\n",
        "        t = np.arange(1, L + 1, dtype=float)                # 1-indexed\n",
        "        p = np.abs(np.sin(phase + (t * np.pi) / phi))\n",
        "        p = np.clip(p, EPS_CLIP, 1.0 - EPS_CLIP)\n",
        "        full = np_rng.binomial(1, p)\n",
        "\n",
        "    else:\n",
        "        raise ValueError(f\"Unknown dist: {dist}\")\n",
        "\n",
        "    return ''.join(map(str, full))\n",
        "\n",
        "def sample_from_logits(logits, temperature):\n",
        "    \"\"\"Sample 0 or 1 from logits[ZERO_ID], logits[ONE_ID] with temperature.\"\"\"\n",
        "    logits = jnp.array([logits[ZERO_ID], logits[ONE_ID]]) / max(1e-6, float(temperature))\n",
        "    probs = jax.nn.softmax(logits)\n",
        "    return int(np.random.choice([0, 1], p=np.array(probs)))\n",
        "\n",
        "# ------------------- generate & save trials ------------------\n",
        "\n",
        "def generate_trials(\n",
        "    N: int,\n",
        "    L: int,\n",
        "    temperature: float,\n",
        "    model,\n",
        "    state,\n",
        "    config,\n",
        "    work_path: str,\n",
        "    *,\n",
        "    dist: str,\n",
        "    phi_choice: str | float | None = None,\n",
        "    rng=None,\n",
        "    filename: str | None = None,\n",
        "    append: bool = False,\n",
        "):\n",
        "    \"\"\"\n",
        "    Generate N trials of length L under `dist` (and optional `phi_choice` for SINUSOID),\n",
        "    run one forward pass per trial, sample next-token predictions, and save JSONL.\n",
        "    \"\"\"\n",
        "    wp = pathlib.Path(work_path)\n",
        "    wp.mkdir(parents=True, exist_ok=True)\n",
        "\n",
        "    # Default filename\n",
        "    if filename is None:\n",
        "        stamp = time.strftime(\"%Y%m%d-%H%M%S\")\n",
        "        tag = dist if dist != \"SINUSOID\" else f\"{dist}-phi-{str(phi_choice).replace('/', 'div')}\"\n",
        "        filename = f\"inference_{tag}_L{L}_{stamp}.jsonl\"\n",
        "\n",
        "    out_path = wp / filename\n",
        "    mode = \"a\" if append and out_path.exists() else \"w\"\n",
        "\n",
        "    start = time.time()\n",
        "    with out_path.open(mode) as fh:\n",
        "        for i in range(N):\n",
        "            s = generate_nonstationary_string(L, dist=dist, phi_choice=phi_choice, rng=rng)\n",
        "\n",
        "            token_ids_full = string_to_token_ids(s, pad_len=config.model.L)   # (1, L)\n",
        "            logits_full = model.apply({\"params\": state.params}, token_ids_full)  # (1, L, V)\n",
        "\n",
        "            T = len(s)\n",
        "            next_logits_all = np.array(logits_full[0, :T-1, :])  # (T-1, V)\n",
        "            tau = float(max(1e-6, temperature))\n",
        "            preds_bits = [sample_from_logits(next_logits_all[t], tau) for t in range(T - 1)]\n",
        "\n",
        "            record = {\n",
        "                \"trial\": i,\n",
        "                \"L\": int(L),\n",
        "                \"temperature\": float(temperature),\n",
        "                \"dist\": dist,\n",
        "                \"phi_choice\": phi_choice,\n",
        "                \"env_string\": s,\n",
        "                \"predictions\": \"\".join(str(b) for b in preds_bits),\n",
        "            }\n",
        "            fh.write(json.dumps(record) + \"\\n\")\n",
        "\n",
        "    print(f\"✅ Generated {N} trials to {out_path} in {time.time()-start:.1f}s\")\n",
        "    return str(out_path)\n",
        "\n",
        "# ------------------- evaluate (unchanged API) ----------------\n",
        "\n",
        "def evaluate_from_file(\n",
        "    path: str | pathlib.Path,\n",
        "    *,\n",
        "    require_constant_L: bool = True,\n",
        "    normalize_by_t: bool = False\n",
        "):\n",
        "    \"\"\"\n",
        "    Read JSONL created by `generate_trials` and compute regret statistics.\n",
        "\n",
        "    Regret definition matches your original:\n",
        "      - util_t = #correct model predictions up to time t\n",
        "      - fixed_util_t = max{ (#zeros up to t), (#ones up to t) }\n",
        "      - regret_t = fixed_util_t - util_t\n",
        "\n",
        "    Args:\n",
        "      path               : JSONL file written by `generate_trials`.\n",
        "      require_constant_L : If True, assert all records share same L.\n",
        "      normalize_by_t     : If True, divide regret_t by t for each t (t=1..L-1).\n",
        "\n",
        "    Returns:\n",
        "      dict with mean_path, std_path, ci bands, regrets_by_t, and final summary.\n",
        "    \"\"\"\n",
        "    path = pathlib.Path(path)\n",
        "    if not path.exists():\n",
        "        raise FileNotFoundError(f\"No such file: {path}\")\n",
        "\n",
        "    # Stream records; compute regrets per record\n",
        "    regrets_list = []\n",
        "    L_ref = None\n",
        "    rec_count = 0\n",
        "\n",
        "    with path.open() as fh:\n",
        "        for line in fh:\n",
        "            rec = json.loads(line)\n",
        "            s = rec[\"env_string\"]\n",
        "            preds = rec[\"predictions\"]\n",
        "            L = rec[\"L\"]\n",
        "\n",
        "            if require_constant_L:\n",
        "                if L_ref is None:\n",
        "                    L_ref = L\n",
        "                elif L != L_ref:\n",
        "                    raise ValueError(f\"Inconsistent L: saw {L_ref} then {L}\")\n",
        "\n",
        "            # Basic sanity: predictions should be length L-1\n",
        "            if len(preds) != L - 1:\n",
        "                raise ValueError(\n",
        "                    f\"Record {rec.get('trial')} predictions length != L-1 \"\n",
        "                    f\"({len(preds)} vs {L-1})\"\n",
        "                )\n",
        "\n",
        "            util = 0\n",
        "            fixed_util = [0, 0]\n",
        "            regrets = np.zeros(L - 1, dtype=np.int32)\n",
        "\n",
        "            for t in range(L - 1):\n",
        "                target = int(s[t+1])\n",
        "                pred_bit = int(preds[t])\n",
        "\n",
        "                util += int(pred_bit == target)\n",
        "                fixed_util[0] += int(target == 0)\n",
        "                fixed_util[1] += int(target == 1)\n",
        "                regrets[t] = max(fixed_util) - util\n",
        "\n",
        "            regrets_list.append(regrets)\n",
        "            rec_count += 1\n",
        "\n",
        "    if rec_count == 0:\n",
        "        raise ValueError(f\"No records found in {path}\")\n",
        "\n",
        "    # Stack & optionally normalize by t\n",
        "    regrets_by_t = np.vstack(regrets_list)  # shape: (N, L-1)\n",
        "    if normalize_by_t:\n",
        "        t_axis = np.arange(1, regrets_by_t.shape[1] + 1, dtype=np.float64)\n",
        "        regrets_by_t = regrets_by_t / t_axis  # broadcast over rows\n",
        "\n",
        "    # Aggregate stats\n",
        "    N, Tm1 = regrets_by_t.shape\n",
        "    mean_path = regrets_by_t.mean(axis=0)\n",
        "    std_path = regrets_by_t.std(axis=0, ddof=1) if N > 1 else np.zeros(Tm1)\n",
        "    stderr_path = std_path / np.sqrt(N) if N > 1 else np.zeros(Tm1)\n",
        "    z = norm.ppf(0.975)\n",
        "    ci_halfwidth = z * stderr_path\n",
        "    ci_lower_path = mean_path - ci_halfwidth\n",
        "    ci_upper_path = mean_path + ci_halfwidth\n",
        "\n",
        "    # \"Final\" summary at horizon t = L-1\n",
        "    final_regrets = regrets_by_t[:, -1]\n",
        "    final_mean = final_regrets.mean()\n",
        "    final_std = final_regrets.std(ddof=1) if N > 1 else 0.0\n",
        "    final_stderr = final_std / np.sqrt(N) if N > 1 else 0.0\n",
        "    final_ci_half = z * final_stderr\n",
        "    final_ci = (final_mean - final_ci_half, final_mean + final_ci_half)\n",
        "\n",
        "    return {\n",
        "        \"mean_path\": mean_path,                 # shape (L-1,)\n",
        "        \"std_path\": std_path,                   # shape (L-1,)\n",
        "        \"ci_lower_path\": ci_lower_path,         # shape (L-1,)\n",
        "        \"ci_upper_path\": ci_upper_path,         # shape (L-1,)\n",
        "        \"regrets_by_t\": regrets_by_t,           # shape (N, L-1)\n",
        "        \"final_mean\": final_mean,\n",
        "        \"final_std\": final_std,\n",
        "        \"final_95%_CI\": final_ci,\n",
        "        \"num_trials\": N,\n",
        "        \"L\": L_ref if L_ref is not None else (Tm1 + 1),\n",
        "        \"normalized\": normalize_by_t,\n",
        "        \"source_file\": str(path),\n",
        "    }\n",
        "\n",
        "# -------------------- plotting helper -----------------------\n",
        "\n",
        "def _pretty_title(dist: str, phi_choice):\n",
        "    if dist == \"2:3-1:3\":\n",
        "        return \"2/3 → 1/3\"\n",
        "    if dist == \"1:3-2:3\":\n",
        "        return \"1/3 → 2/3\"\n",
        "    if dist == \"1:3\":\n",
        "        return \"1/3\"\n",
        "    if dist == \"2:3\":\n",
        "        return \"2/3\"\n",
        "    if dist == \"SINUSOID\":\n",
        "        phi_label = phi_choice if isinstance(phi_choice, str) else f\"{phi_choice:g}\"\n",
        "        return f\"Sinusoid (φ={phi_label})\"\n",
        "    return dist\n",
        "\n",
        "def plot_regret(results, dist: str, phi_choice, TRAIN_DATA: str, save_dir: str, *, y_min=-0.2, y_max=0.8):\n",
        "    Y_MIN, Y_MAX = y_min, y_max\n",
        "    L = results[\"mean_path\"].shape[0] + 1\n",
        "    t_axis = np.arange(1, L)\n",
        "\n",
        "    mean_rate = results[\"mean_path\"] / t_axis\n",
        "    ci_lo_rate = results[\"ci_lower_path\"] / t_axis\n",
        "    ci_hi_rate = results[\"ci_upper_path\"] / t_axis\n",
        "\n",
        "    ci_lo_clip = np.clip(ci_lo_rate, Y_MIN, Y_MAX)\n",
        "    ci_hi_clip = np.clip(ci_hi_rate, Y_MIN, Y_MAX)\n",
        "\n",
        "    fig, ax = plt.subplots(figsize=(7, 4.5))\n",
        "    ax.plot(t_axis, mean_rate, marker=\"o\", markersize=1, linewidth=0.5)\n",
        "    ax.fill_between(t_axis, ci_lo_clip, ci_hi_clip, alpha=0.25)\n",
        "\n",
        "    ax.set_xlabel(\"Time horizon (t)\")\n",
        "    ax.set_ylabel(\"Avg regret up to t (regret/t)\")\n",
        "    ax.set_title(f\"Regret of {TRAIN_DATA} on {_pretty_title(dist, phi_choice)} (95% CI)\")\n",
        "    ax.grid(True, alpha=0.3)\n",
        "\n",
        "    ax.set_xlim(1, L - 1)\n",
        "    ax.set_ylim(Y_MIN, Y_MAX)\n",
        "\n",
        "    ticks = np.linspace(Y_MIN, Y_MAX, 7)\n",
        "    if Y_MIN < 0 < Y_MAX and not np.isclose(ticks, 0).any():\n",
        "        ticks = np.sort(np.append(ticks, 0.0))\n",
        "    ax.set_yticks(ticks)\n",
        "    ax.axhline(0, linestyle=\"--\", linewidth=1, alpha=0.9, zorder=3)\n",
        "\n",
        "    fig.tight_layout()\n",
        "\n",
        "    save_dir = pathlib.Path(save_dir)\n",
        "    save_dir.mkdir(parents=True, exist_ok=True)\n",
        "    phi_tag = (str(phi_choice).replace(\"/\", \"div\") if dist == \"SINUSOID\" else None)\n",
        "    base = f\"inference-{dist}\" if dist != \"SINUSOID\" else f\"inference-{dist}-phi-{phi_tag}\"\n",
        "    out_path = save_dir / (base + \".png\")\n",
        "    fig.savefig(out_path, dpi=300, bbox_inches=\"tight\")\n",
        "    print(f\"📈 Saved figure to {out_path}\")\n",
        "    plt.close(fig)\n",
        "\n",
        "# --------------- run all conditions in one go ----------------\n",
        "\n",
        "def run_all_conditions(\n",
        "    *,\n",
        "    work_path: str,\n",
        "    context_length: int,\n",
        "    model,\n",
        "    state,\n",
        "    cfg,\n",
        "    TRAIN_DATA: str,\n",
        "    temperature: float | None = None,\n",
        "    dists: list[str] | None = None,\n",
        "    sinusoid_phis: list[str | float] | None = None,\n",
        "    rng=None,\n",
        "):\n",
        "    \"\"\"\n",
        "    Runs generation+eval+plot for all requested distributions.\n",
        "    Returns a dict: {(dist, phi_tag): {\"jsonl\": path, \"results\": results, \"figure\": path}}\n",
        "    \"\"\"\n",
        "    if temperature is None:\n",
        "        temperature = 1 / math.sqrt(context_length)\n",
        "\n",
        "    if dists is None:\n",
        "        dists = [\"2:3-1:3\", \"1:3-2:3\", \"1:3\", \"2:3\", \"SINUSOID\"]\n",
        "\n",
        "    if sinusoid_phis is None:\n",
        "        sinusoid_phis = [\"L/20\", \"L/10\", \"L/5\", \"L/2\"]\n",
        "\n",
        "    out = {}\n",
        "    wp = pathlib.Path(work_path)\n",
        "    fig_dir = wp / \"fig\"\n",
        "\n",
        "    for dist in dists:\n",
        "        phi_list = sinusoid_phis if dist == \"SINUSOID\" else [None]\n",
        "        for phi_choice in phi_list:\n",
        "            # stable tag for filenames\n",
        "            phi_tag = (str(phi_choice).replace(\"/\", \"div\") if phi_choice is not None else None)\n",
        "            base_tag = dist if dist != \"SINUSOID\" else f\"{dist}-phi-{phi_tag}\"\n",
        "\n",
        "            # 1) Generate\n",
        "            jsonl_name = f\"inference_{base_tag}_L{context_length}.jsonl\"\n",
        "            jsonl_path = generate_trials(\n",
        "                N=100,\n",
        "                L=context_length,\n",
        "                temperature=temperature,\n",
        "                model=model,\n",
        "                state=state,\n",
        "                config=cfg,\n",
        "                work_path=work_path,\n",
        "                dist=dist,\n",
        "                phi_choice=phi_choice,\n",
        "                rng=rng,\n",
        "                filename=jsonl_name,\n",
        "            )\n",
        "\n",
        "            # 2) Evaluate (path must be FILE, not a directory)\n",
        "            results = evaluate_from_file(jsonl_path, require_constant_L=True, normalize_by_t=False)\n",
        "\n",
        "            # 3) Plot\n",
        "            plot_regret(results, dist, phi_choice, TRAIN_DATA, save_dir=fig_dir)\n",
        "\n",
        "            out[(dist, phi_tag or \"\")] = {\n",
        "                \"jsonl\": jsonl_path,\n",
        "                \"results\": results,\n",
        "                \"figure\": str((fig_dir / (f\"inference-{dist}-phi-{phi_tag}.png\" if dist == \"SINUSOID\" else f\"inference-{dist}.png\")).resolve()),\n",
        "            }\n",
        "\n",
        "            # Optional: print final stats normalized by (L-1)\n",
        "            L = results[\"mean_path\"].shape[0] + 1\n",
        "            final_mean = results[\"final_mean\"]/(L-1)\n",
        "            final_std  = results[\"final_std\"]/(L-1)\n",
        "            ci_lo, ci_hi = results[\"final_95%_CI\"]\n",
        "            print(f\"[{base_tag}] Final mean regret/t: {final_mean:.4f} | std: {final_std:.4f} | 95% CI: ({ci_lo/(L-1):.4f}, {ci_hi/(L-1):.4f})\")\n",
        "\n",
        "    return out\n",
        "\n"
      ],
      "metadata": {
        "id": "Wy2APDTgWEyF"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "context_length = CONTEXT_LEN\n",
        "print(cfg.workdir)\n",
        "\n",
        "mesh = create_mesh()\n",
        "model, state = restore_model(workdir=cfg.workdir, config=cfg, mesh=mesh)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "cLMi1WCyWJ6M",
        "outputId": "3b4eca9f-bcf5-4e25-d258-4f69f643cc2d"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "/content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "/usr/local/lib/python3.12/dist-packages/orbax/checkpoint/type_handlers.py:1544: UserWarning: Couldn't find sharding info under RestoreArgs. Populating sharding info from sharding file. Please note restoration time will be slightly increased due to reading from file instead of directly from RestoreArgs. Note also that this option is unsafe when restoring on a different topology than the checkpoint was saved with.\n",
            "  warnings.warn(\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Restoring checkpoint from step 500\n",
            "Restored checkpoint path: /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/checkpoint_500\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "# ----------------------- example call -----------------------\n",
        "\n",
        "out_summary = run_all_conditions(\n",
        "    work_path=work_path,\n",
        "    context_length=context_length,\n",
        "    model=model,\n",
        "    state=state,\n",
        "    cfg=cfg,\n",
        "    TRAIN_DATA=TRAIN_DATA,\n",
        "    temperature=1/np.sqrt(context_length),   # or None to auto\n",
        "    # dists=[\"2:3-1:3\",\"1:3-2:3\",\"1:3\",\"2:3\",\"SINUSOID\"],\n",
        "    dists=[\"SINUSOID\"],\n",
        "    sinusoid_phis=[\"L/10\",\"L/5\",\"L/2\"],\n",
        "    rng=None,  # or pass your seeded RNG\n",
        ")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "qANGRuP_WKxq",
        "outputId": "3a7c7f11-33c6-474d-867a-0272e593712b"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "✅ Generated 100 trials to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/inference_SINUSOID-phi-Ldiv10_L1024.jsonl in 128.3s\n",
            "📈 Saved figure to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/fig/inference-SINUSOID-phi-Ldiv10.png\n",
            "[SINUSOID-phi-Ldiv10] Final mean regret/t: 0.1352 | std: 0.0167 | 95% CI: (0.1319, 0.1384)\n",
            "✅ Generated 100 trials to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/inference_SINUSOID-phi-Ldiv5_L1024.jsonl in 123.1s\n",
            "📈 Saved figure to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/fig/inference-SINUSOID-phi-Ldiv5.png\n",
            "[SINUSOID-phi-Ldiv5] Final mean regret/t: 0.1832 | std: 0.0154 | 95% CI: (0.1801, 0.1862)\n",
            "✅ Generated 100 trials to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/inference_SINUSOID-phi-Ldiv2_L1024.jsonl in 125.0s\n",
            "📈 Saved figure to /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/fig/inference-SINUSOID-phi-Ldiv2.png\n",
            "[SINUSOID-phi-Ldiv2] Final mean regret/t: 0.1388 | std: 0.0199 | 95% CI: (0.1349, 0.1427)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# ======= Multi-condition plotting (one figure per condition; 3 curves each) =======\n",
        "from __future__ import annotations\n",
        "import pathlib\n",
        "import numpy as np\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "# You already have these:\n",
        "# evaluate_from_file(path: Path) -> dict with keys:\n",
        "#   \"mean_path\", \"ci_lower_path\", \"ci_upper_path\", \"final_mean\", \"final_std\", \"final_95%_CI\"\n",
        "# And a defined CONTEXT_LEN (int)\n",
        "\n",
        "# ====== USER CONFIG ======\n",
        "TRAIN_DATA_LIST = [\"BERNOULLI\", \"POLYAURN\", \"ROBUST_BERNOULLI\"]\n",
        "TRUE_DIST_FIXED = [\"2:3-1:3\", \"1:3-2:3\", \"1:3\", \"2:3\"]  # non-sinusoid\n",
        "PHI_CHOICE_LIST = [\"L/2\", \"L/5\", \"L/10\", \"L/20\"]        # for TRUE_DIST==\"SINUSOID\"\n",
        "\n",
        "Y_MIN, Y_MAX = -0.2, 0.8             # lock y-range across all figures\n",
        "MARKER_BY_TRAIN = {                  # markers; colors = matplotlib defaults\n",
        "    \"BERNOULLI\": \"o\",\n",
        "    \"POLYAURN\": \"s\",\n",
        "    \"ROBUST_BERNOULLI\": \"D\",\n",
        "}\n",
        "LINESTYLE_BY_TRAIN = {\n",
        "    \"BERNOULLI\": \"-\",\n",
        "    \"POLYAURN\": \"--\",\n",
        "    \"ROBUST_BERNOULLI\": \"-.\",\n",
        "}\n",
        "MARKERSIZE = 1.5\n",
        "LINEWIDTH = 0.9\n",
        "ALPHA_CI = 0.18\n",
        "SKIP_MISSING = True\n",
        "\n",
        "# --- Path builders ---\n",
        "def _root_for_train(train: str) -> pathlib.Path:\n",
        "    \"\"\"Per-train work dir as requested.\"\"\"\n",
        "    return pathlib.Path(f\"/content/drive/MyDrive/next-token-no-regret-0924/{train}-{CONTEXT_LEN}/\")\n",
        "\n",
        "def _save_root() -> pathlib.Path:\n",
        "    \"\"\"Shared figure output dir (not tied to a single train).\"\"\"\n",
        "    return pathlib.Path(f\"/content/drive/MyDrive/next-token-no-regret-0924/Figures-{CONTEXT_LEN}/\")\n",
        "\n",
        "def _out_dir_for(TRUE_DIST: str, PHI_CHOICE=None) -> str:\n",
        "    \"\"\"Subdir name relative to a train root, matching your naming scheme.\"\"\"\n",
        "    if TRUE_DIST == \"SINUSOID\":\n",
        "        phi_tag = str(PHI_CHOICE).replace(\"/\", \"div\")  # \"L/10\" -> \"Ldiv10\"\n",
        "        return f\"inference_{TRUE_DIST}-phi-{phi_tag}_L1024.jsonl\"\n",
        "    else:\n",
        "        return f\"inference_{TRUE_DIST}_L1024.jsonl\"\n",
        "\n",
        "def _title_for(TRUE_DIST: str, PHI_CHOICE=None) -> str:\n",
        "    if TRUE_DIST == \"2:3-1:3\":\n",
        "        base = \"2/3 → 1/3 Sequence\"\n",
        "    elif TRUE_DIST == \"1:3-2:3\":\n",
        "        base = \"1/3 → 2/3 Sequence\"\n",
        "    elif TRUE_DIST == \"1:3\":\n",
        "        base = \"1/3 Sequence\"\n",
        "    elif TRUE_DIST == \"2:3\":\n",
        "        base = \"2/3 Sequence\"\n",
        "    elif TRUE_DIST == \"SINUSOID\":\n",
        "        phi_label = PHI_CHOICE if isinstance(PHI_CHOICE, str) else f\"{PHI_CHOICE:g}\"\n",
        "        base = f\"Sinusoid (φ = {phi_label})\"\n",
        "    else:\n",
        "        base = TRUE_DIST\n",
        "    return f\"Regret on {base} (95% CI)\"\n",
        "\n",
        "def _compute_rates(results, y_min=Y_MIN, y_max=Y_MAX):\n",
        "    L = results[\"mean_path\"].shape[0] + 1\n",
        "    t_axis = np.arange(1, L)\n",
        "    mean_rate = results[\"mean_path\"] / t_axis\n",
        "    ci_lo_rate = results[\"ci_lower_path\"] / t_axis\n",
        "    ci_hi_rate = results[\"ci_upper_path\"] / t_axis\n",
        "    ci_lo_clip = np.clip(ci_lo_rate, y_min, y_max)\n",
        "    ci_hi_clip = np.clip(ci_hi_rate, y_min, y_max)\n",
        "    return t_axis, mean_rate, ci_lo_clip, ci_hi_clip\n",
        "\n",
        "def _truncate_to_common_prefix(curves):\n",
        "    if not curves:\n",
        "        return curves\n",
        "    min_len = min(len(c[\"t\"]) for c in curves)\n",
        "    synced = []\n",
        "    for c in curves:\n",
        "        synced.append({\n",
        "            \"label\": c[\"label\"],\n",
        "            \"t\": c[\"t\"][:min_len],\n",
        "            \"mean\": c[\"mean\"][:min_len],\n",
        "            \"lo\": c[\"lo\"][:min_len],\n",
        "            \"hi\": c[\"hi\"][:min_len],\n",
        "            \"marker\": c[\"marker\"],\n",
        "            \"linestyle\": c[\"linestyle\"],\n",
        "        })\n",
        "    return synced\n",
        "\n",
        "def _plot_condition(TRUE_DIST: str, phi_choice=None):\n",
        "    \"\"\"\n",
        "    Make a single figure for one condition (TRUE_DIST or SINUSOID+phi).\n",
        "    Overlays the 3 TRAIN_DATA curves, reading from each train's own work dir.\n",
        "    Saves to /content/drive/MyDrive/next-token-no-regret/Figures-{CONTEXT_LEN}/...\n",
        "    \"\"\"\n",
        "    curves = []\n",
        "\n",
        "    for train in TRAIN_DATA_LIST:\n",
        "        train_root = _root_for_train(train)\n",
        "        subdir = _out_dir_for(TRUE_DIST, phi_choice)\n",
        "        out_dir = train_root / subdir\n",
        "\n",
        "        # Allow either .../<train>/<inference-...>/<files> or .../<train>/<inference-...>/<train>/<files>\n",
        "        try:\n",
        "            results = evaluate_from_file(path=out_dir / train)\n",
        "        except Exception:\n",
        "            try:\n",
        "                results = evaluate_from_file(path=out_dir)\n",
        "            except Exception:\n",
        "                if SKIP_MISSING:\n",
        "                    print(f\"[skip] Missing results for {train} @ {out_dir}\")\n",
        "                    continue\n",
        "                else:\n",
        "                    raise\n",
        "\n",
        "        t, mean, lo, hi = _compute_rates(results, Y_MIN, Y_MAX)\n",
        "        curves.append({\n",
        "            \"label\": train,\n",
        "            \"t\": t,\n",
        "            \"mean\": mean,\n",
        "            \"lo\": lo,\n",
        "            \"hi\": hi,\n",
        "            \"marker\": MARKER_BY_TRAIN.get(train, \"o\"),\n",
        "            \"linestyle\": LINESTYLE_BY_TRAIN.get(train, \"-\"),\n",
        "        })\n",
        "\n",
        "    if not curves:\n",
        "        print(f\"[warn] No curves found for condition {TRUE_DIST}\" +\n",
        "              (f\" (φ={phi_choice})\" if TRUE_DIST == \"SINUSOID\" else \"\"))\n",
        "        return\n",
        "\n",
        "    curves = _truncate_to_common_prefix(curves)\n",
        "    L = curves[0][\"t\"][-1] + 1\n",
        "\n",
        "    fig, ax = plt.subplots(figsize=(7, 4.5))\n",
        "    for c in curves:\n",
        "        ax.plot(\n",
        "            c[\"t\"], c[\"mean\"],\n",
        "            marker=c[\"marker\"], markersize=MARKERSIZE,\n",
        "            linestyle=c[\"linestyle\"], linewidth=LINEWIDTH,\n",
        "            label=c[\"label\"],\n",
        "        )\n",
        "        ax.fill_between(c[\"t\"], c[\"lo\"], c[\"hi\"], alpha=ALPHA_CI)\n",
        "\n",
        "    ax.set_xlabel(\"Time horizon (t)\")\n",
        "    ax.set_ylabel(\"Avg regret up to t (regret/t)\")\n",
        "    ax.set_title(_title_for(TRUE_DIST, phi_choice))\n",
        "    ax.grid(True, alpha=0.3)\n",
        "\n",
        "    ax.set_xlim(1, L - 1)\n",
        "    ax.set_ylim(Y_MIN, Y_MAX)\n",
        "\n",
        "    ticks = np.linspace(Y_MIN, Y_MAX, 7)\n",
        "    if Y_MIN < 0 < Y_MAX and not np.isclose(ticks, 0).any():\n",
        "        ticks = np.sort(np.append(ticks, 0.0))\n",
        "    ax.set_yticks(ticks)\n",
        "\n",
        "    ax.axhline(0, linestyle=\"--\", linewidth=1, alpha=0.9, zorder=3)\n",
        "    ax.legend(title=\"TRAIN_DATA\", frameon=True)\n",
        "    fig.tight_layout()\n",
        "\n",
        "    # Save to shared figures dir\n",
        "    phi_tag = str(phi_choice).replace(\"/\", \"div\") if phi_choice is not None else None\n",
        "    out_file_name = (\n",
        "        f\"inference_{TRUE_DIST}-phi-{phi_tag}_L1024\"\n",
        "        if TRUE_DIST == \"SINUSOID\"\n",
        "        else f\"inference_{TRUE_DIST}_L1024\"\n",
        "    )\n",
        "    save_dir = _save_root()\n",
        "    save_dir.mkdir(parents=True, exist_ok=True)\n",
        "    save_path = save_dir / f\"{out_file_name}.png\"\n",
        "    fig.show()\n",
        "\n",
        "    # fig.savefig(save_path, dpi=300, bbox_inches=\"tight\")\n",
        "    # print(f\"Saved figure to {save_path}\")\n",
        "    # plt.close(fig)\n",
        "\n",
        "# ===== Run all conditions =====\n",
        "# 1) Fixed non-sinusoid regimes\n",
        "# for td in TRUE_DIST_FIXED:\n",
        "#     _plot_condition(td)\n",
        "\n",
        "# # 2) Sinusoid regimes (one figure per φ)\n",
        "# for phi in PHI_CHOICE_LIST:\n",
        "#     _plot_condition(\"SINUSOID\", phi_choice=phi)\n",
        "_plot_condition(TRUE_DIST_FIXED[0])\n"
      ],
      "metadata": {
        "id": "WNQ4-sK63nja",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 457
        },
        "outputId": "75af7a93-7d22-4308-e084-3f5d86f9cf55"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 700x450 with 1 Axes>"
            ],
            "image/png": "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34+fnx/Lly93egIc7E/yA44n+a8WEhJCSEgIAH369OG1116jY8eOHDlyxOmr3GtFRESQkJBQqPgmTJjAPffcQ5MmTQp7SfmKjY1l586dTJs2Ldc2vV5Phw4dAOjatSvt27fnrrvuIjg4+LpjFv38/HjjjTd44403OHnyJOvXr+ett95izpw5+Pn5MX36dJfv0bzug5o1axbhqu1cvWevd2/lJLN5zWSR48KFCyQlJTF//nzmz59fqPPmRVGUXGXTpk3j4sWLvPHGG8ycOROw36PDhg1j3rx5eHt753s8lUrFM888w9q1a9mwYQMPP/ywY1u5cuVo3ry54/WMGTMIDg5m6NChLFiwgHnz5rF48WJOnDhBv3792L9/f77/duSMh05NTb3uNV5PzntQ2LHfQuRFklghSkHTpk0d4zF79uxJq1ateOihhzh06BDe3t6OB1PGjx+fq9c0R1HHaeY17s5ms9GxY0eee+65PPe5euqem+nahDNHXn/085KcnMx9991HUlISmzZtynMqqNWrV3PXXXfh5+d33eP16dOHF198kRUrVjB8+PB8682ZM4esrKwCj7Vjxw7mzJlDq1atqFOnzvUvphB++uknjEYj7dq1u27dli1bEhYWxuLFi6+bxF6tUqVKPPLII/Tq1YsqVaqwePFipk+fflPv0bxYrVan167eszd6b+WcE+Dhhx/O1cOfo6Ax4gEBAahUqjw/lOn1ej755BNeffVVDh8+TEhICDVq1OChhx5CrVZf972MjIwE7L29+Tlx4gRvv/02P//8M2q1mq+++orhw4dzzz33APDpp5/y9ddf89JLL+W5f61atQD7g5E3Kuc9CAwMvOFjiduXJLFClDKNRsOMGTNo164dc+bMYcKECVSpUgUAnU7n6E3LT6VKlfJ8gr8wT/XnqFq1Kmlpadc9l6u9JpUqVeLQoUO5yg8ePOjYXlyysrLo1q0bhw8f5pdffskzUVQUhTVr1jB+/PhCHTPnK9zk5OQC6107af21du/ezbhx42jRogWrV68ucJ5WV/z444+0a9eu0A9IZWVlXfda8lOuXDmqVq3Kv//+C+DyPXrkyJFc5XndG+XKlcs1g4LJZMo1X25h79nCyrmenOvLS1BQED4+Plit1iKdV6vVUrVqVY4fP55vnau/DbBarWzYsIFmzZoV2BMLV3qS8+uZBvsHju7d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\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# ======= 2x4 grid: 8 conditions in one figure, 3 curves each, single legend, high-visibility styles =======\n",
        "from __future__ import annotations\n",
        "import pathlib\n",
        "import numpy as np\n",
        "import matplotlib.pyplot as plt\n",
        "import matplotlib.patheffects as pe\n",
        "\n",
        "# You already have: evaluate_from_file(path) and CONTEXT_LEN (int)\n",
        "\n",
        "TRAIN_DATA_LIST = [\"BERNOULLI\", \"POLYAURN\", \"ROBUST_BERNOULLI\"]\n",
        "TRUE_DIST_FIXED = [\"2:3-1:3\", \"1:3-2:3\", \"1:3\", \"2:3\"]\n",
        "PHI_CHOICE_LIST = [\"L/2\", \"L/5\", \"L/10\", \"L/20\"]  # for SINUSOID\n",
        "\n",
        "# ---- Visibility & layout config ----\n",
        "Y_MIN, Y_MAX = -0.2, 0.8\n",
        "SKIP_MISSING = True\n",
        "\n",
        "# Color-blind-friendly palette\n",
        "COLORS = {\n",
        "    \"BERNOULLI\":        \"#0072B2\",\n",
        "    \"POLYAURN\":         \"#009E73\",\n",
        "    \"ROBUST_BERNOULLI\": \"#E69F00\",\n",
        "}\n",
        "\n",
        "# Per-curve style (add ms = markersize, lw = linewidth)\n",
        "STYLE = {\n",
        "    \"BERNOULLI\":        dict(ls=\"None\",  lw=1.4, ms=4, alpha=1.0, z=4, marker=\"o\"),\n",
        "    \"POLYAURN\":         dict(ls=\"None\", lw=2.0, ms=4, alpha=1.0, z=3, marker=\"s\"),\n",
        "    \"ROBUST_BERNOULLI\": dict(ls=\"-\", lw=3.0, ms=2, alpha=1.00, z=1, marker=\"D\"),\n",
        "}\n",
        "\n",
        "# Optional global scalers to quickly make all lines/markers thicker/larger\n",
        "LINEWIDTH_SCALE = 1.0\n",
        "MARKERSIZE_SCALE = 1.0\n",
        "\n",
        "# CI appearance: robust gets lighter treatment to avoid occluding others\n",
        "CI_ALPHA_DEFAULT = 0.14\n",
        "CI_ALPHA_ROBUST  = 0.10\n",
        "USE_ROBUST_CI_LINES = False  # set True to use thin CI boundary lines for ROBUST_BERNOULLI\n",
        "\n",
        "\n",
        "def _root_for_train(train: str) -> pathlib.Path:\n",
        "    return pathlib.Path(f\"/content/drive/MyDrive/next-token-no-regret-0924/{train}-{CONTEXT_LEN}/\")\n",
        "\n",
        "def _save_root() -> pathlib.Path:\n",
        "    return pathlib.Path(f\"/content/drive/MyDrive/next-token-no-regret-0924/Figures-{CONTEXT_LEN}/\")\n",
        "\n",
        "def _subdir(TRUE_DIST: str, phi=None) -> str:\n",
        "    if TRUE_DIST == \"SINUSOID\":\n",
        "        phi_tag = str(phi).replace(\"/\", \"div\")\n",
        "        return f\"inference_{TRUE_DIST}-phi-{phi_tag}_L1024.jsonl\"\n",
        "    else:\n",
        "        return f\"inference_{TRUE_DIST}_L1024.jsonl\"\n",
        "\n",
        "def _format_phi_for_title(phi):\n",
        "    \"\"\"Show 'L/10' as 'T/10' in titles; handle numeric phi too.\"\"\"\n",
        "    if isinstance(phi, str):\n",
        "        # If you ever pass 'Ldiv10', show as 'T/10' nicely.\n",
        "        s = phi.replace(\"div\", \"/\")\n",
        "        return s.replace(\"L\", \"T\")  # only uppercase L → T\n",
        "    try:\n",
        "        return f\"{phi:g}\"\n",
        "    except Exception:\n",
        "        return str(phi)\n",
        "\n",
        "def _title(TRUE_DIST: str, phi=None) -> str:\n",
        "    if TRUE_DIST == \"2:3-1:3\":\n",
        "        base = \"2/3 → 1/3\"\n",
        "    elif TRUE_DIST == \"1:3-2:3\":\n",
        "        base = \"1/3 → 2/3\"\n",
        "    elif TRUE_DIST == \"1:3\":\n",
        "        base = \"1/3\"\n",
        "    elif TRUE_DIST == \"2:3\":\n",
        "        base = \"2/3\"\n",
        "    elif TRUE_DIST == \"SINUSOID\":\n",
        "        base = f\"Sinusoid (φ={_format_phi_for_title(phi)})\"\n",
        "    else:\n",
        "        base = TRUE_DIST\n",
        "    return base\n",
        "\n",
        "\n",
        "def _compute_rates(results, y_min=Y_MIN, y_max=Y_MAX):\n",
        "    L = results[\"mean_path\"].shape[0] + 1\n",
        "    t = np.arange(1, L)\n",
        "    mean = results[\"mean_path\"] / t\n",
        "    lo = np.clip(results[\"ci_lower_path\"] / t, y_min, y_max)\n",
        "    hi = np.clip(results[\"ci_upper_path\"] / t, y_min, y_max)\n",
        "    return t, mean, lo, hi\n",
        "\n",
        "def _sync_lengths(curves):\n",
        "    if not curves:\n",
        "        return curves\n",
        "    m = min(len(c[\"t\"]) for c in curves)\n",
        "    for c in curves:\n",
        "        c[\"t\"] = c[\"t\"][:m]\n",
        "        c[\"mean\"] = c[\"mean\"][:m]\n",
        "        c[\"lo\"] = c[\"lo\"][:m]\n",
        "        c[\"hi\"] = c[\"hi\"][:m]\n",
        "    return curves\n",
        "\n",
        "def _markevery(t):\n",
        "    \"\"\"Sparse markers so overlap stays readable (~≤30 markers per line).\"\"\"\n",
        "    step = max(1, len(t) // 30)\n",
        "    return slice(0, None, step)\n",
        "\n",
        "def _gather_curves(TRUE_DIST: str, phi=None):\n",
        "    curves = []\n",
        "    for train in TRAIN_DATA_LIST:\n",
        "        train_root = _root_for_train(train)\n",
        "        sub = _subdir(TRUE_DIST, phi)\n",
        "        out_dir = train_root / sub\n",
        "        try:\n",
        "            results = evaluate_from_file(path=out_dir / train)  # layout A: .../inference-.../<TRAIN_DATA>/*\n",
        "        except Exception:\n",
        "            try:\n",
        "                results = evaluate_from_file(path=out_dir)       # layout B: .../inference-.../*\n",
        "            except Exception:\n",
        "                if SKIP_MISSING:\n",
        "                    print(f\"[skip] Missing results for {train} @ {out_dir}\")\n",
        "                    continue\n",
        "                else:\n",
        "                    raise\n",
        "        t, mean, lo, hi = _compute_rates(results)\n",
        "        if train == \"ROBUST_BERNOULLI_PREFIX\":\n",
        "            train = \"ROBUST_BERNOULLI\"\n",
        "        curves.append({\n",
        "            \"label\": train,\n",
        "            \"t\": t, \"mean\": mean, \"lo\": lo, \"hi\": hi,\n",
        "        })\n",
        "    return _sync_lengths(curves)\n",
        "\n",
        "# ---- Build condition list (fixed first, then sinusoid φ's) ----\n",
        "conditions = [(td, None) for td in TRUE_DIST_FIXED] + [(\"SINUSOID\", phi) for phi in PHI_CHOICE_LIST]\n",
        "\n",
        "# ---- Create figure ----\n",
        "fig, axes = plt.subplots(2, 4, figsize=(16, 7), squeeze=False)\n",
        "legend_handles, legend_labels = [], []\n",
        "legend_captured = False\n",
        "\n",
        "# Draw order: robust (under), poly, bernoulli (top)\n",
        "draw_order = [\"ROBUST_BERNOULLI\", \"POLYAURN\", \"BERNOULLI\"]\n",
        "\n",
        "for idx, (TRUE_DIST, phi) in enumerate(conditions):\n",
        "    r, c = divmod(idx, 4)\n",
        "    ax = axes[r][c]\n",
        "    curves = _gather_curves(TRUE_DIST, phi)\n",
        "\n",
        "    if not curves:\n",
        "        ax.set_title(_title(TRUE_DIST, phi) + \" (no data)\")\n",
        "        ax.set_xlim(1, 2)\n",
        "        ax.set_ylim(Y_MIN, Y_MAX)\n",
        "        ax.axhline(0, linestyle=\"--\", linewidth=1, alpha=0.9, zorder=3)\n",
        "        ax.grid(True, alpha=0.3)\n",
        "        continue\n",
        "\n",
        "    # reorder for visibility\n",
        "    curves.sort(key=lambda cd: draw_order.index(cd[\"label\"]))\n",
        "\n",
        "    # X-limit based on shortest curve for this subplot\n",
        "    L_common = curves[0][\"t\"][-1] + 1\n",
        "\n",
        "    for cd in curves:\n",
        "        label = cd[\"label\"]\n",
        "        color = COLORS[label]\n",
        "        st = STYLE[label]\n",
        "\n",
        "        # (A) CI: robust lighter (or switch to boundary lines)\n",
        "        if label == \"ROBUST_BERNOULLI\" and USE_ROBUST_CI_LINES:\n",
        "            ax.plot(cd[\"t\"], cd[\"lo\"], color=color, alpha=0.5, linewidth=0.6, zorder=max(0, st[\"z\"]-1))\n",
        "            ax.plot(cd[\"t\"], cd[\"hi\"], color=color, alpha=0.5, linewidth=0.6, zorder=max(0, st[\"z\"]-1))\n",
        "        else:\n",
        "            ax.fill_between(\n",
        "                cd[\"t\"], cd[\"lo\"], cd[\"hi\"],\n",
        "                alpha=CI_ALPHA_ROBUST if label == \"ROBUST_BERNOULLI\" else CI_ALPHA_DEFAULT,\n",
        "                edgecolor=\"none\", facecolor=color, zorder=0\n",
        "            )\n",
        "\n",
        "        # (B) Mean line with white outline for legibility\n",
        "        st = STYLE[label]\n",
        "        (line,) = ax.plot(\n",
        "            cd[\"t\"], cd[\"mean\"],\n",
        "            color=color,\n",
        "            linestyle=st[\"ls\"],\n",
        "            linewidth=st[\"lw\"] * LINEWIDTH_SCALE,   # ← per-curve + global scale\n",
        "            alpha=st[\"alpha\"],\n",
        "            marker=st[\"marker\"],\n",
        "            markersize=st[\"ms\"] * MARKERSIZE_SCALE, # ← per-curve + global scale\n",
        "            markevery=_markevery(cd[\"t\"]),\n",
        "            zorder=st[\"z\"],\n",
        "            # path_effects=[pe.Stroke(linewidth=(st[\"lw\"] * LINEWIDTH_SCALE) + 1.2,\n",
        "            #                         foreground=\"white\", alpha=0.9),\n",
        "            #               pe.Normal()],\n",
        "            label=label,\n",
        "            # solid_capstyle=\"round\",      # <- smoother joins with markers\n",
        "            # solid_joinstyle=\"round\",\n",
        "        )\n",
        "\n",
        "\n",
        "        # Capture legend handles once (first subplot only)\n",
        "        if not legend_captured:\n",
        "            legend_handles.append(line)\n",
        "            legend_labels.append(label)\n",
        "\n",
        "    ax.set_title(_title(TRUE_DIST, phi))\n",
        "    ax.set_xlim(1, L_common - 1)\n",
        "    ax.set_ylim(Y_MIN, Y_MAX)\n",
        "    ax.axhline(0, linestyle=\"--\", linewidth=1, alpha=0.9, zorder=3)\n",
        "    ax.grid(True, alpha=0.3)\n",
        "\n",
        "    # y=0 must be a tick\n",
        "    ticks = np.linspace(Y_MIN, Y_MAX, 7)\n",
        "    if Y_MIN < 0 < Y_MAX and not np.isclose(ticks, 0).any():\n",
        "        ticks = np.sort(np.append(ticks, 0.0))\n",
        "    ax.set_yticks(ticks)\n",
        "\n",
        "    # Label only left column (y) and bottom row (x)\n",
        "    if c == 0:\n",
        "        ax.set_ylabel(\"Avg regret (regret/t)\")\n",
        "    else:\n",
        "        ax.set_yticklabels([])\n",
        "    if r == 1:\n",
        "        ax.set_xlabel(\"t\")\n",
        "    else:\n",
        "        ax.set_xticklabels([])\n",
        "\n",
        "    legend_captured = True\n",
        "\n",
        "# Single shared legend\n",
        "fig.legend(\n",
        "    legend_handles, legend_labels, title=\"TRAIN_DATA\",\n",
        "    loc=\"lower center\", ncol=3, frameon=True, bbox_to_anchor=(0.5, -0.02)\n",
        ")\n",
        "fig.tight_layout(rect=(0, 0.05, 1, 1))  # leave space for legend\n",
        "\n",
        "# Save\n",
        "save_dir = _save_root()\n",
        "save_dir.mkdir(parents=True, exist_ok=True)\n",
        "save_path = save_dir / \"all_conditions_grid.png\"\n",
        "\n",
        "fig.show()\n",
        "\n",
        "fig.savefig(save_path, dpi=300, bbox_inches=\"tight\")\n",
        "print(f\"Saved grid figure to {save_path}\")\n",
        "\n",
        "# plt.close(fig)\n"
      ],
      "metadata": {
        "id": "mn2KnEnjOkt4",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 424
        },
        "outputId": "f5cf6b12-14c1-47f7-980a-ac3e6f11736e"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Saved grid figure to /content/drive/MyDrive/next-token-no-regret-0924/Figures-1024/all_conditions_grid.png\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 1600x700 with 8 Axes>"
            ],
            "image/png": "iVBORw0KGgoAAAANSUhEUgAABjQAAALICAYAAADYAhnjAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjAsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvlHJYcgAAAAlwSFlzAAAPYQAAD2EBqD+naQABAABJREFUeJzs3Xl4FFXWBvC3qnpNZycLARLCGkAEFARRxpVFURRxQ1EQRxwdUT9RRxmXiKK4oIM7DiOOM+q4oo6KIIODgKCyyDIIAQRkyx6ydae3qvr+6KSToruTNCS95f09D4929e3K6ZvOreScuvcKqqqqICIiIiIiIiIiIiIiimBiuAMgIiIiIiIiIiIiIiJqCQsaREREREREREREREQU8VjQICIiIiIiIiIiIiKiiMeCBhERERERERERERERRTwWNIiIiIiIiIiIiIiIKOKxoEFERERERERERERERBGPBQ0iIiIiIiIiIiIiIop4LGgQEREREREREREREVHEY0GDiIiIiIiIiIiIiIgiHgsaREREREREREREREQU8VjQIAqzDRs2YObMmTjllFNgsViQk5ODa665Brt37w74mi+++AKiKKKoqAh1dXX4/e9/j4EDByIpKQnx8fEYPHgwXnzxRbhcrhC+E4/XX38dV199NXJyciAIAm666aZm2x87dgw6nQ4ffvghAOCpp57CmWeeifT0dJhMJvTp0wf/93//h9LS0hBET0TU9mpra5Gfn4+LLroIqampEAQBf//735t9TaSO84cOHcKcOXMwfPhwpKSkIC0tDeeddx7+85//BHwNx3kiorYR7PUkUq8lRESkFWxeiOM7dXS6cAdA1NE988wz+P7773H11Vdj0KBBKCoqwiuvvILTTz8dP/zwAwYOHOjzmq+++gpDhw5F586dUVFRgR07dmD8+PHIzc2FKIpYt24d7rnnHvz444947733Qv5+ampqMHz4cBQWFrbYfvny5RAEAWPHjgUAbNq0CUOGDMHkyZORkJCAnTt3YtGiRfjqq6+wZcsWWCyW9n4LRERtqqysDI8//jhycnIwePBgrFq1qsXXROo4//nnn+OZZ57BxIkTMW3aNLjdbvzjH//AmDFjsHjxYkyfPt3nNRzniYjaRrDXk0i9lhARkVaweSGO79ThqUQUVt9//73qcDg0x3bv3q0ajUZ1ypQpfl+TnZ2t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dDpRWVmpmaXR3Nc1Go0wGo0+x5OTk3mBhOdD7nQ6kZyc3OF/6NkXWuwPLfaHVnP90fA4VNPEOc4Hxs+tFvujEftCi/2h1VJ/hHKc5xgfGD+3WuwPLfaHFvujEcf46MHPrRb7oxH7Qov9oRXqcT5qetxgMGDo0KGaDb4VRcHKlSsxcuRIv6+x2Ww+nShJEgDP2l2tOefQoUOh1+s1bQoKCnDw4MGAX5eIiIiIiIiIiIiIiNpW1MzQAIBZs2Zh2rRpGDZsGIYPH44FCxbAarVi+vTpAICpU6eia9eumDdvHgBgwoQJeOGFF3Daaad5l5x65JFHMGHCBG9ho6VzJiUl4fe//z1mzZqF1NRUJCYm4s4778TIkSP9bghORERERERERERERERtL6oKGtdeey1KS0vx6KOPoqioCEOGDMGyZcu8m3ofPHhQMyPj4YcfhiAIePjhh3HkyBGkp6djwoQJePLJJ1t9TgD4y1/+AlEUceWVV8LhcGDcuHF47bXXQvfGiYiIiIiIiIiIiIg6uKgqaADAzJkzMXPmTL/PrVq1SvNYp9MhPz8f+fn5J3xOADCZTHj11Vfx6quvBh0vERERERERERERERGdvKjZQ4OIiIiIiIiIiIiIiDouFjSIiIiIiIiIiIiIiCjisaBBREREREREREREREQRjwUNIiIiIiIiIiIiIiKKeCxoEBERERERERERERFRxGNBg4iIiIiIiIiIiIiIIh4LGiEkW49AcVnDHQYRERERERERERERUdRhQSOESj8bAMehz8MdBhERERERERERERFR1GFBI8Tsh74MdwhERERERERERERERFGHBY0Qs//2cbhDICIiIiIiIiIiIiKKOixohJgpZ1K4QyAiIiIiIiIiIiIiijosaISYsetF4Q6BiIiIiIiIiIiIiCjqsKARckq4AyAiIiIiIiIiIiIiijosaISayoIGEREREREREREREVGwWNAIOTXcARARERERERERERERRR0WNEJM5QwNIiIiIiIiIiIiIqKgsaARaixoEBEREREREREREREFjQWNkOOSU0REREREREREREREwWJBI9Q4Q4OIiIiIiIiIiIiIKGgsaIQaCxpEREREREREREREREFjQSPkuOQUEREREREREREREVGwWNAIMZUzNIiIiIiIiIiIiIiIgsaCRqixoEFEREREREREREREFDQWNEKOS04REREREREREREREQWLBY1Q4wwNIiIiIiIiIiIiIqKgsaARcpyhQUREREREREREREQULBY0Qo0zNIiIiIiIiIiIiIiIgsaCRoipLGgQEREREREREUWGuz7GWQs3Ysm2wnBHQkRErcCCRshxySkiIiIiIiIiooigM+CXYiuufJtFDSKiaMCCRqhxhgYRERERERERUcRQAQgC8PiK3eEOhYiIWsCCRqixoEFEREREREREFFFUFSgoqQ13GERE1AIWNEKOS04REREREREREUUSQQDyMuLDHQYREbUg6goar776KnJzc2EymTBixAj89NNPzbavrKzEHXfcgaysLBiNRvTt2xdLly71Pv/6669j0KBBSExMRGJiIkaOHImvv/5ac44//OEP6NWrF8xmM9LT03H55Zdj165dJ/YGOEODiIiIiIiIiChiCPDM0Mgf2zfcoRARUQuiqqDxwQcfYNasWcjPz8fmzZsxePBgjBs3DiUlJX7bO51OjBkzBgcOHMDHH3+MgoICLFq0CF27dvW26datG55++mls2rQJGzduxAUXXIDLL78cO3bs8LYZOnQo3nrrLezcuRPLly+HqqoYO3YsZFkO+j2oYEGDiIiIiIiIiCgiuJ04JdOCJTcNwxWnZoU7GiIiaoEu3AEE44UXXsCMGTMwffp0AMDChQvx1VdfYfHixXjwwQd92i9evBgVFRVYt24d9Ho9ACA3N1fTZsKECZrHTz75JF5//XX88MMPOOWUUwAAt956q/f53NxczJ07F4MHD8aBAwfQq1ev4N6EyiWniIiIiIiIiIgiwktX4fuqKiQmJoY7EiIiaoWomaHhdDqxadMmjB492ntMFEWMHj0a69ev9/uaf//73xg5ciTuuOMOZGZmYuDAgXjqqacCzqyQZRnvv/8+rFYrRo4c6beN1WrFW2+9hR49eiA7Ozv4N8Ilp4iIiIiIiIiIiIiIghY1MzTKysogyzIyMzM1xzMzMwPuZ7Fv3z58++23mDJlCpYuXYq9e/fij3/8I1wuF/Lz873ttm/fjpEjR8JutyM+Ph6ffvopBgwYoDnXa6+9hj/96U+wWq3Iy8vDihUrYDAY/H5dh8MBh8PhfVxdXQ0AUCFAURUoSscuaiiKAlVVO3w/AOyL47E/tNgfWs31R6j7KNA4rygc4/m51WJ/NGJfaLE/tFrqj1D2E8f4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HOnp6ZAkqd1fP9bZbDbk5OTg4MGDSEpKinZzooqx0GI8tBgPrabiIYRAdXV1yCUI2wPzfHh832oxHvUYCy3GQ6u5eEQyzzPHh8f3rRbjocV4aDEe9ZjjOw6+b7UYj3qMhRbjoRXpPB/VAY37778fN954Y5NlevTogU6dOqGkpERz3Ov1ory8HJ06dWry+dXV1bjkkkuQmJiI//znPzAYDIHHOnXqhJ9++klTvri4OPCY/1//sYZlkpKSQs7OAACTyQSTyaQ5lpKS0mQ7T0ZJSUn8o6/DWGgxHlqMh1a4eERiZoYf83zz+L7VYjzqMRZajIdWU/GIVJ5njm8e37dajIcW46HFeNRjju84+L7VYjzqMRZajIdWpPJ8VAc0MjMzkZmZ2Wy5ESNGoLKyEuvXr8eQIUMAACtXroSqqhg+fHjY59lsNowZMwYmkwmffvopzGZzUL1PPfUUSkpKAktaLV++HElJSejfv3+gzJIlSzTPW758OUaMGNGqcyUiIiIiIiIiIiIiomPXIXYt6devHy655BLcdttt+Omnn/D9999j6tSpuPrqqwNTVQ4fPoy+ffsGZlzYbDaMHj0aNTU1eOONN2Cz2VBUVISioiIoigIAGD16NPr374/rr78eGzduxLJly/DII4/grrvuCozY33HHHdizZw8efPBBbN++HfPnz8e///1v3HfffdEJBhERERERERERERHRSahDbAoOAO+99x6mTp2Kiy66CLIsY9KkSfjLX/4SeNzj8aCgoAC1tbUAgJ9//hk//vgjAKBXr16auvbu3Yu8vDzodDr873//w5133okRI0YgPj4eU6ZMweOPPx4o2717d3z22We477778NJLL6Fr1654/fXXMWbMmAic9YnJZDJh5syZQdM8T0aMhRbjocV4aDEeHQN/T1qMRz3GQovx0GI8Ogb+nrQYDy3GQ4vxqMdYdBz8XWkxHvUYCy3GQyvS8ZCEECIir0RERERERERERERERHSMOsSSU0REREREREREREREdHLjgAYREREREREREREREcU8DmgQEREREREREREREVHM44AGERERERERERERERHFPA5oEBERERERERERERFRzOOABhERERERERERERERxTwOaBARERERERERERERUczjgAYREREREREREREREcU8DmgQEREREREREREREVHM44AGERERERERERERERHFPA5oEBERERERERERERFRzOOABhERERERERERERERxTwOaBARERERERERERERUczjgAYREREREREREREREcU8DmgQEREREREREREREVHM44AGERERERERERERERHFPA5oEBERERERERERERFRzOOABtExyMvLw4033hjtZrTY22+/DUmSsG/fvmbLtubcDh48CLPZjO+///74GhijFixYgNzcXLhcrmg3hYgiiDne50TP8Q8//DCGDx8e7WYQUTtiPvc50fN5S3k8HuTk5GD+/PnRbgoRtQHmeB/meB/m+JMLBzSIGti8eTOuvPJKdOvWDWazGV26dMHFF1+Ml19+OdpNi0mPP/44hg8fjrPPPjsirzdr1ixIktTsf+eff77meZs3b4YkSfjpp59w9OhRPP/88zj33HORmZmJlJQUnHnmmfjggw+CXu/GG2+E2+3GX//614icHxG1L+b41umIOR7wfQEM9bw77rhD87x7770XGzduxKeffhqR8yOitsN83jodNZ9/8MEHuO6669C7d++Q5RtyuVx46KGH0LlzZ1gsFgwfPhzLly/XlDEYDJg+fTqeeuopOJ3Otj5tImojzPGt0xFzfGv6ZQDmeAomCSFEtBtBFAtWr16NCy64ALm5uZgyZQo6deqEgwcP4ocffsDu3buxa9euQFmXywVZlmEwGKLY4pZTFAUejwcmkwmSJDVZNi8vD+effz7efvvtJsuVlpaiS5cuWLhwIa655po2bG14mzZtwqZNmwI/2+123HnnnZgwYQImTpwYOJ6VlYWLL7448PMzzzyDF198EUVFRfjss88wceJEjB07FhdccAH0ej0++ugjfPXVV3jssccwe/ZszWs+9NBD+OCDD7B3795mY0dEsYs53udEz/GSJCEvLw+pqam4//77NfX36dMHZ5xxhubYVVddhcLCQnzzzTftdFZE1NaYz31Ohnx+/vnnY/369Rg2bBg2bNiAgQMHYtWqVSFf85prrsGiRYtw7733onfv3nj77bexdu1afPXVVxg5cmSgXGVlJbKysvDaa6/h5ptvbvuTJ6Ljwhzvc6Ln+Nb2yzDHUxBBREIIIcaOHSsyMzNFRUVF0GPFxcWRb1CUdOvWTUyZMqXZcnPnzhUWi0VUV1e3f6PCKC0tFQDEzJkzmyx3zjnnBM5pz549Yt++fZrHVVUVF154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ta9P3/UdS2P1XVv7IK2X7Ci8mm0dE9eFR043es4y46tcb6/jy2E95Bz1KPfW1PnLSYDnZFRD9119+7hQWthsxn01+mbGNx/MNTlPqkdTv9avNNOaetxXq+/rcV++traOM4w0Vs43myM08vLysGnTJgDAww8/DIVCge7du+O7776TLup9/vx5WcB0794dt912G2bMmIFnn30Wfn5+WLZsmTSZkZubi+3bt8NiseDuu+8GALRs2RIbN26UJjN27dqF3NxcHDt2DL1794ZSqUTPnj3x8ccfN+bqExEREVEz0hyPQLrafdfW3mazyY7Cq9nWqDUC57Oc7kdEnifGEIjWytpPe1azvSdkFPu+en0z44n+vObwWmff12/f1c8w4qp9Q+d8s5luHTNmDI4dO4affvoJO3bsQJs2bdC+fXvZaZ9+/PFHLF++XPp92rRpOHHiBLZs2YK9e/ciMTERH374oXS7UqnE8OHD8cMPP+DcuXPYunUrfHx88Oabb0ptfHx8MG3aNOzfvx9paWlYsWIFjh49WuejM4iIiIiIiIiIiIiI6M9rFkdonDhxAt999x1+/fVX9OzZEwDw3nvv4c4778Sbb76JqKgop/sUFhZiyZIlWLVqFW677TYAwLJly5CQkIBffvkFN910EwIDA6VrZQBAq1at8OSTT+KNN96QlnXr1g3dunWTfm/dujU2bNiAnTt34rHHHrtaq0xERERERERERERERNU0iwmN5ORkBAQESJMZADB48GAolUqkpKRgxIgRTvfZt28fzGYzBg8eLC2Lj49HTEwMkpOTcdNNNznd59KlS9iwYQP69+/vdiwHDhzA7t278eqrr7ptYzKZYDKZpN+NRiMA++E3rg7Bud7YbDYIIVgLsBY1sR5yrIdcbfVo7Box593jdivHelRhLeRYD7nL1aMx68SMd4/brRzrIcd6yLEeVZjxzQe3WznWowprIcd6yDV2zjeLCY2MjAyEhYXJlqnVagQFBSEjI8PtfbRaLQICAmTLw8PDne4zevRobNy4EWVlZbjrrrvw6aefOvXXsmVLZGdnw2KxYM6cOXjkkUfcjnfBggWYO3eu0/Ls7GyUl5e7vd/1wmazobCwEEKIJr+oUlNjLeRYDznWQ662ehQVFTXqWJjz7nG7lWM9qrAWcqyH3OXq0Zg5z4x3j9utHOshx3rIsR5VmPHNB7dbOdajCmshx3rINXbON+mExsyZM/H666/X2ubEiRNXfRyLFi3Cyy+/jFOnTmHWrFl45plnZNfaAICdO3eiuLgYv/zyC2bOnIl27dph9OjRLvtz9OFgNBoRHR2N0NDQBrmSe3Nns9mgUCgQGhp63b/oWQs51kOO9ZCrrR56vb5Rx8Kcd4/brRzrUYW1kGM95C5Xj8bMeWa8e9xu5VgPOdZDjvWowoxvPrjdyrEeVVgLOdZDrrFzvkknNKZPn47x48fX2qZNmzaIiIhAVpb8SugWiwV5eXmIiIhweb+IiAhUVFSgoKBAdpRGZmam030iIiIQERGB+Ph4BAUF4dZbb8VLL72EyMhIqU1sbCwAoHPnzsjMzMScOXPcTmjodDrodDqn5Uqlkht5JYVCwXpUYi3kWA851kPOXT0auz7M+dpxu5VjPaqwFnKsh1xt9WjMGjHja8ftVo71kGM95FiPKsz45oPbrRzrUYW1kGM95Boz55t0QiM0NBShoaGXbdenTx8UFBRg37596NGjBwBg+/btsNls6N27t8v79OjRAxqNBtu2bcPIkSMBAKmpqTh//jz69Onj9rEc5/Sqfk5FV21qu52IiIiIiIiIiIiIiBpWs7iGRkJCAoYMGYJHH30UH330EcxmMyZPnoxRo0YhKioKAHDx4kUMGjQIn332GW688Ub4+/tjwoQJeOaZZxAUFAQ/Pz889dRT6NOnj3RB8G+//RaZmZno1asXDAYDjh07hueeew59+/ZF69atAQAffPABYmJiEB8fDwDYsWMH3nzzTTz99NNNUgsiIiIiIiIiIiIioutRs5jQAICVK1di8uTJGDRoEJRKJUaOHIl3331Xut1sNiM1NRWlpaXSskWLFkltTSYTkpKSZNfG8PLywieffIJp06bBZDIhOjoa99xzD2bOnCm1sdlsmDVrFtLS0qBWq9G2bVu8/vrrePzxxxtnxYmIiIiIiIiIiIiIqPlMaAQFBWHVqlVub2/dujWEELJler0eH3zwAT744AOX9xk4cCB2795d6+M+9dRTeOqpp+o/YCIiIiIiIiIiIiIiajC8agkREREREREREREREXk8TmgQEREREREREREREZHH44QGERERERERERERERF5PE5oEBERERERERERERGRx+OEBhEREREREREREREReTxOaBARERERERERERERkcfjhAYREREREREREREREXk8TmgQEREREREREREREZHH44QGERERERERERERERF5PE5oEBERERERERERERGRx+OEBhEREREREREREREReTxOaBARERERERERERERkcfjhAYREREREREREREREXk8TmgQEREREREREREREZHH44QGERERERERERERERF5PE5oEBERERERERERERGRx+OEBhEREREREREREREReTxOaBARERERERERERERkcfjhAYREREREREREREREXk8TmgQEREREREREREREZHH44QGERERERERERERERF5PE5oEBERERERERERERGRx+OEBhEREREREREREREReTxOaBARERERERERERERkcfjhAYREREREREREREREXk8TmgQEREREREREREREZHH44QGERERERERERERERF5PE5oEBERERERERERERGRx+OEBhEREREREREREREReTxOaBARERERERERERERkcfjhAYREREREREREREREXk8TmgQEREREREREREREZHHazYTGnl5eRgzZgz8/PwQEBCACRMmoLi4uNb2Tz31FOLi4uDl5YWYmBg8/fTTKCwslLU7f/48hg0bBm9vb4SFheG5556DxWKRtfnxxx/RvXt36HQ6tGvXDsuXL78aq0hERERERERERERERG40mwmNMWPG4NixY9iyZQs2b96MHTt24LHHHnPb/tKlS7h06RLefPNNHD16FMuXL8d3332HCRMmSG2sViuGDRuGiooK7N69GytWrMDy5csxe/ZsqU1aWhqGDRuGgQMH4uDBg5g6dSoeeeQRfP/991d1fYmIiIiIiIiIiIiIqIq6qQdQFydOnMB3332HX3/9FT179gQAvPfee7jzzjvx5ptvIioqyuk+nTp1wpdffin93rZtW8yfPx9jx46FxWKBWq3G//73Pxw/fhxbt25FeHg4unbtildeeQXPP/885syZA61Wi48++gixsbF46623AAAJCQnYtWsXFi1ahKSkpMYpABERERERERERERHRda5ZHKGRnJyMgIAAaTIDAAYPHgylUomUlJQ691NYWAg/Pz+o1Wqp386dOyM8PFxqk5SUBKPRiGPHjkltBg8eLOsnKSkJycnJf2aViIiIiIiIiIiIiIioHprFERoZGRkICwuTLVOr1QgKCkJGRkad+sjJycErr7wiO01VRkaGbDID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IiIiIiIiIiIiIiKPxwkNIiIiIiIiIiIiIiLyeJzQICIiIiIiIiIiIiIij8cJDSIiIiIiIiIiIiIi8nic0CAiIiIiIiIiIiIiIo/HCQ0iIiIiIiIiIiIiIvJ4nNAgIiIiIiIiIiIiIiKPxwkNIiIiIiIiIiIiIiLyeJzQICInAwYMwNSpU5t6GEREdBUw44mIrm3MeSKiaxcznogTGkRERERERERERERE1AwohBCiqQdBRJ5j/PjxWLFihWxZWloaWrdu3TQDIiKiBsOMJyK6tjHniYiuXcx4IjtOaBCRTGFhIYYOHYpOnTph3rx5AIDQ0FCoVKomHhkREf1ZzHgiomsbc56I6NrFjCeyUzf1AIjIs/j7+0Or1cLb2xsRERFNPRwiImpAzHgiomsbc56I6NrFjCey4zU0iIiIiIiIiIiIiIjI43FCg4iIiIiIiIiIiIiIPB4nNIjIiVarhdVqbephEBHRVcCMJyK6tjHniYiuXcx4Ik5oEJELrVu3RkpKCs6dO4ecnBzYbLamHhIRETUQZjwR0bWNOU9EdO1ixhNxQoOIXHj22WehUqmQmJiI0NBQnD9/vqmHREREDYQZT0R0bWPOExFdu5jxRIBCCCGaehBERERERERERERERES14REaRERERERERERERETk8TihQUREREREREREREREHo8TGkRERERERERERERE5PE4oUFERERERERERERERB6PExpEREREREREREREROTxOKFBREREREREREREREQejxMaRERERERERERERETk8TihQUREREREREREREREHo8TGkRERERERERERERE5PE4oUFERERERERERERERB6PExpEREREREREREREROTxOKFBREREREREREREREQeT93UAyAiIiKqCyEELBYLrFZrUw+FiIjomqRSqaBWq6FQKJp6KEREREQucUKDiIiIPF5FRQXS09NRWlra1EMhIiK6pnl7eyMyMhJarbaph0JERETkRCGEEE09CCIiIiJ3bDYbTp8+DZVKhdDQUGi1Wn5zlIiIqIEJIVBRUYHs7GxYrVa0b98eSiXPUk1ERESehUdoEBERkUerqKiAzWZDdHQ0vL29m3o4RERE1ywvLy9oNBr8/vvvqKiogF6vb+ohEREREcnw6xZERETULPBbokRERFcf32+JiIjIk3FPhYiIiIiIiIiIiIiIPB4nNIiIiIiIiIiIiIiIyONxQoOIiIiIqJlRKBS1/syZMwfnzp2TLQsKCkL//v2xc+dOl30+/vjjUKlUWL9+vdNtc+bMQdeuXWW/KxQKTJw4Udbu4MGDUCgUOHfu3GXXoeb4fH190bFjR0yaNAmnT592eZ/k5GSoVCoMGzZMWjZ+/Phaa9G6dWup7YIFC6BSqfDGG29cdnxEREREROR5OKFBRERERNTMpKenSz+LFy+Gn5+fbNmzzz4rtd26dSvS09OxY8cOREVF4S9/+QsyMzNl/ZWWlmLNmjWYMWMGli5dWqcx6PV6LFmyxO3kQ105xnfo0CG89tprOHHiBLp06YJt27Y5tV2yZAmeeuop7NixA5cuXQIAvPPOO7J1B4Bly5ZJv//666/S/ZcuXVqvdSQiIiIiIs/CCQ0iIiIiomYmIiJC+vH394dCoZAtMxgMUtvg4GBERESgU6dOeOGFF2A0GpGSkiLrb/369UhMTMTMmTOxY8cO/PHHH5cdQ1xcHAYOHIgXX3zxT62LY3xt2rTB3Xffja1bt6J3796YMGECrFar1K64uBhr167FE088gWHDhmH58uUAAH9/f9m6A0BAQID0e2hoKADgp59+QllZGebNmwej0Yjdu3f/qXETEREREVHj44QGEREREdF1oKysDJ999hkAQKvVym5bsmQJxo4dC39/fwwdOlSaLLichQsX4ssvv8TevXsbbJxKpRJTpkzB77//jn379knL161bh/j4eMTFxWHs2LFYunQphBB17nfJkiUYPXo0NBoNRo8ejSVLljTYmImIiIiIqHFwQoOIiIiI6Bp28803w2AwwMfHB2+++SZ69OiBQYMGSbefPn0av/zyCx544AEAwNixY7Fs2bI6TRZ0794d999/P55//vkGHXN8fDwAyK7F4Zh0AYAhQ4agsLAQP/30U536MxqN+OKLL6T7jx07FuvWrUNxcXGDjpuIiIiIiK4uTmgQEREREV3D1q5diwMHDuDLL79Eu3btsHz5cmg0Gun2pUuXIikpCSEhIQCAO++8E4WFhdi+fXud+n/11Vexc+dO/O9//2uwMTsmUxQKBQAgNTUVe/bswejRowEAarUaDzzwQJ2Psli9ejXatm2LLl26AAC6du2KVq1aYe3atQ02ZiIiIiIiuvrUTT0AIiIiIiK6eqKjo9G+fXu0b98eFosFI0aMwNGjR6HT6WC1WrFixQpkZGRAra7608BqtWLp0qWyIzncadu2LR599FHMnDmzwU7jdOLECQBAbGwsAPvRGRaLBVFRUVIbIQR0Oh3ef/99+Pv719rfkiVLcOzYMdk62mw2LF26FBMmTGiQMRMRERER0dXHCQ0iIiIiouvEvffei9mzZ+PDDz/EtGnT8O2336KoqAgHDhyASqWS2h09ehQPP/wwCgoKEBAQcNl+Z8+ejbZt22LNmjV/eow2mw3vvvsuYmNj0a1bN1gsFnz22Wd46623cMcdd8jaDh8+HKtXr8bEiRPd9nfkyBHs3bsXP/74I4KCgqTleXl5GDBgAE6ePCmd4oqIiIiIiDwbJzSIiIiIiK4TCoUCTz/9NObMmYPHH38cS5YswbBhw6RTMTkkJiZi2rRpWLlyJSZNmnTZfsPDw/HMM8/gjTfeqPeYcnNzkZGRgdLSUhw9ehSLFy/Gnj178M0330ClUuHrr79Gfn4+JkyY4HQkxsiRI7FkyZJaJzSWLFmCG2+8Ef369XO6rVevXliyZMkVjZuIiIiIiBofr6FBRERERHQdGTduHMxmM9577z188803GDlypFMbpVKJESNG1OsUUs8++ywMBkO9xzN48GBERkaic+fOmDlzJhISEnD48GEMHDgQgH1CYvDgwS5PKzVy5Ejs3bsXhw8fdtl3RUUFPv/8c5fr6Lj/Z599BrPZXO9xExERERFR41MIxxX3iIiIiDxQeXk50tLSEBsbC71e39TDISIiuqbxfZeIiIg8GY/QICIiIiIiIiIiIiIij8cJDSIiIiIianATJ06EwWBw+VPbNS+IiIiIiIjc4SmniIiIyKPx1BdEzVNWVhaMRqPL2/z8/BAWFtbIIyKiuuD7LhEREXkydVMPgIiIiIiIrj1hYWGctCAiIiIiogbFU04REREREREREREREZHH44QGERERERERERERERF5PE5oEBERERERERERERGRx+OEBhEREREREREREREReTxOaBARERERERERERERkcfjhAYREREREREREREREXk8TmgQERERXSXjx4+HQqGAQqGARqNBbGwsZsyYgfLyclm7zZs3o3///vD19YW3tzd69eqF5cuXy9qcO3dO6kuhUECr1aJdu3Z49dVXIYSQPebw4cOdxvLjjz9CoVCgoKAAAGC1WrFw4ULEx8fDy8sLQUFB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rO8jR47A19dX+l2hUKBLly4AgLKyMqdt3dfXF23btq2tfERERESeTxARERF5sLKyMnH8+HFRVlbW1EOpt3Hjxom7777bafmCBQtEaGioKC4uFkIIsXHjRnHrrbcKHx8fodfrRY8ePcTSpUtl90lLSxOwX1tXABAqlUq0bNlSPProoyIrK0vWNjU1VYwYMUJERUUJHx8f0aVLF/HJJ58Im80mtfn444/FwIEDRWhoqNBqtSImJkaMHz9enDt3zmm8y5YtE/7+/vVa97qO9+WXX5a1c/zodDqpTf/+/cWUKVOcHuPxxx8XHTt2FFarVXq8AwcOyNo89thjAoB4+eWXZcsPHz4sRo4cKYKCgoRGoxFt27YV//jHP0RJSYnT46xcuVL06NFD+Pr6ivDwcHHnnXeKQ4cOydr079/f5Xq88sorQgjh8jYAYufOneKHH34QAER+fn69atwUfi/KE/uy/3D6+b0o76o9prvXkYPZbBZvvPGG6Nixo9BqtcLPz08kJSWJXbt2ydotW7bM5XPQtm1b6fbq2/kXX3whlEqlyMjIcPm4CQkJYtq0adLv33//vejbt68IDAwUwcHBYsCAAeKnn36q07rUzAQA4quvvpK1ee211wQAMW7cOLe1uN6NGzdO9tz6+vqKXr16iS+++EJq4+61uGDBAiGEcPt6PH/+vFCr1WLt2rXSY9V8LtPS0oRWqxU1/8yu6zYqhBAXL14U48aNEzExMcLLy0skJiaKhQsXioqKCqmNY4w1f1QqlRDCfa4OGjRICOE+Ux2a8/suERERXfsUQvAKZUREROS5ysvLkZaWhtjYWOj1+qYeDhER0TWN77tERETkyXgNDSIiIiIiIiIiIiIi8nic0CAiIiKieps4cSIMBoPLn4kTJzb18IiIiIiIiOgaxFNOERERkUfjqS88U1ZWFoxGo8vb/Pz8EBYW1sgjIiKihsD3XSIiIvJk6qYeABERERE1P2FhYZy0ICIiIiIiokbFU04REREREREREREREZHH44QGERERNQs8SyYREdHVx/dbIiIi8mSc0CAiIiKPptFoAAClpaVNPBIiIqJrn+P91vH+S0RERORJeA0NIiIi8mgqlQoBAQHIysoCAHh7e0OhUDTxqIiIiK4tQgiUlpYiKysLAQEBUKlUTT0kIiIiIicKweNJiYiIyMMJIZCRkYGCgoKmHgoREdE1LSAgABEREfzyABEREXkkTmgQERFRs2G1WmE2m5t6GERERNckjUbDIzOIiIjIo3FCg4iIiIiIiIiIiIiIPB4vCk5ERERERERERERERB6PExpEREREREREREREROTxOKFBREREREREREREREQejxMaRERERERERERERETk8TihQUREREREREREREREHo8TGkRERERERERERERE5PE4oUFERERERERERERERB7v/wFdgVaDEGDPcgAAAABJRU5ErkJggg==\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# Suppose you already restored model/state and have cfg set up\n",
        "prompt = \"011111111\"\n",
        "\n",
        "# Generate 16 new bits with greedy decoding\n",
        "# greedy_out = generate(model, state, prompt, cfg, max_new_tokens=cfg.model.L, temperature=0)\n",
        "# print(\"Greedy:\", greedy_out)\n",
        "\n",
        "# Generate 16 new bits with sampling (temperature = 0.7)\n",
        "sampled_out = generate(model, state, prompt, cfg, max_new_tokens=cfg.model.L, temperature=1.0/np.sqrt(cfg.model.L))\n",
        "print(\"Sampled:\", sampled_out)\n"
      ],
      "metadata": {
        "id": "tAtxhsg035V6"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "print(norm.ppf(0.975))"
      ],
      "metadata": {
        "id": "3Y1lxJluUn6M"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "## Get code from the repo\n",
        "\n",
        "'''\n",
        "from IPython.core.getipython import get_ipython\n",
        "\n",
        "filename = \"train.py\"\n",
        "\n",
        "with open(filename, \"r\") as f:\n",
        "    code = f.read()\n",
        "\n",
        "# # Insert a new code cell below with the file's content\n",
        "get_ipython().set_next_input(code, replace=False)\n",
        "'''"
      ],
      "metadata": {
        "id": "YSOQA6S4Yle9"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Calculate TV distance"
      ],
      "metadata": {
        "id": "Dvn0TgIQtRGv"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "# === Fast Monte-Carlo TV (binary vocab, after BOS) ============================\n",
        "# Assumptions / globals you already have:\n",
        "#   PAD_ID, BOS_ID, ZERO_ID, ONE_ID\n",
        "#   string_to_token_ids(s, pad_len) -> jnp.ndarray  # wraps your _py_tokenize\n",
        "#   Your model supports batched inputs: (B, L) -> (B, L-1, V) or (B, L, V)\n",
        "#\n",
        "# Set head type once:\n",
        "#   \"logits\" -> head emits raw scores over vocab\n",
        "#   \"probs2\" -> head emits two nonnegative scores; LINEAR renorm to a prob vec\n",
        "#   \"prob1\"  -> head emits scalar p(1) in [0,1] (set PROB1_IS_LOGIT=True if it's a logit)\n",
        "OUTPUT_MODE = \"probs2\"     # or \"probs2\" or \"prob1\"\n",
        "PROB1_IS_LOGIT = False     # set True if your \"prob1\" head outputs a logit\n",
        "\n",
        "import math\n",
        "import numpy as np\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "from jax.scipy.special import logsumexp\n",
        "\n",
        "# --------------------- helpers: BOS-only inputs -------------------------------\n",
        "def make_bos_only_ids_batch(B: int, L: int) -> jnp.ndarray:\n",
        "    \"\"\"(B, L) with only BOS realized (rest PAD). Uses your tokenizer on empty string.\"\"\"\n",
        "    one = string_to_token_ids(\"\", pad_len=L)    # (1, L) -> [BOS, PAD, ...]\n",
        "    return jnp.repeat(one, repeats=B, axis=0)   # (B, L)\n",
        "\n",
        "def count_prompt_len_batch(ids_BxL: jnp.ndarray, pad_id: int) -> jnp.ndarray:\n",
        "    \"\"\"Vectorized 'first PAD index' per row. Returns (B,) ints in [1..L].\"\"\"\n",
        "    is_pad = (ids_BxL == pad_id)                # (B, L)\n",
        "    # first PAD per row = argmax over a cumulative mask that flips at first pad\n",
        "    # Easier: use where + argmin trick by appending a PAD sentinel at the end.\n",
        "    B, L = ids_BxL.shape\n",
        "    pad_pos = jnp.argmax(jnp.concatenate([is_pad, jnp.ones((B,1), dtype=is_pad.dtype)], axis=1), axis=1)\n",
        "    # If no PAD before end, argmax hits the appended 1 at L, so returns L (as desired).\n",
        "    return pad_pos  # (B,)\n",
        "\n",
        "# ----------------- normalize head outputs to probabilities --------------------\n",
        "def _two_class_logprobs_from_logits(logits_BxTxV):\n",
        "    # Take only {ZERO_ID, ONE_ID}, log-softmax across last dim, return (B,T)\n",
        "    two = logits_BxTxV[..., jnp.array([ZERO_ID, ONE_ID])]   # (B,T,2)\n",
        "    logp2 = jax.nn.log_softmax(two, axis=-1)               # (B,T,2)\n",
        "    return logp2  # log probs for classes [0,1]\n",
        "\n",
        "def _two_class_logprobs_from_probs2(scores_BxTxV, eps=1e-9):\n",
        "    # LINEAR renorm (clamp negatives to zero), then log\n",
        "    two = scores_BxTxV[..., jnp.array([ZERO_ID, ONE_ID])]  # (B,T,2)\n",
        "    two = jnp.maximum(two, 0.0)\n",
        "    Z = jnp.clip(two.sum(axis=-1, keepdims=True), a_min=eps)  # (B,T,1)\n",
        "    p2 = two / Z\n",
        "    p2 = jnp.clip(p2, eps, 1.0 - eps)\n",
        "    p2 = p2 / p2.sum(axis=-1, keepdims=True)\n",
        "    return jnp.log(p2)  # (B,T,2)\n",
        "\n",
        "def _logprobs_for_labels_prob1(out_BxT1, labels_BxT, eps=1e-9):\n",
        "    # labels_BxT are token IDs; map to {0,1}\n",
        "    y = (labels_BxT == ONE_ID).astype(jnp.float32)         # (B,T)\n",
        "    p1 = out_BxT1.squeeze(-1)                              # (B,T)\n",
        "    if PROB1_IS_LOGIT:\n",
        "        p1 = jax.nn.sigmoid(p1)\n",
        "    p1 = jnp.clip(p1, eps, 1.0 - eps)\n",
        "    return y * jnp.log(p1) + (1.0 - y) * jnp.log(1.0 - p1) # (B,T)\n",
        "\n",
        "# -------------------- score log P for a batch in ONE pass ---------------------\n",
        "def batch_sequence_logprob_after_bos(model, state, ids_BxL: jnp.ndarray,\n",
        "                                     pad_id: int) -> jnp.ndarray:\n",
        "    \"\"\"\n",
        "    Returns log P for each sequence in the batch, summing t=1..n-1 (tokens after BOS).\n",
        "    Uses ONE model.apply on the whole (B, L).\n",
        "    Handles both output shapes: (B, L-1, V) or (B, L, V) (we drop the last step).\n",
        "    \"\"\"\n",
        "    B, L = ids_BxL.shape\n",
        "    outs = model.apply({\"params\": state.params}, ids_BxL)  # (B, L-1, V) or (B, L, Vor1)\n",
        "    # Align outputs to predict tokens 1..L-1:\n",
        "    if outs.shape[1] == L:\n",
        "        outs = outs[:, :-1]   # (B, L-1, V or 1)\n",
        "\n",
        "    # Build labels matrix (B, L-1): the realized tokens after BOS, up to pad\n",
        "    n_B = count_prompt_len_batch(ids_BxL, pad_id)          # (B,)\n",
        "    labels = ids_BxL[:, 1:]                                # (B, L-1)\n",
        "    # Mask to keep only realized positions per row\n",
        "    ar = jnp.arange(L-1)[None, :]                          # (1, L-1)\n",
        "    mask = (ar < (n_B - 1)[:, None])                       # (B, L-1), True where token exists\n",
        "\n",
        "    # Compute per-position log-prob according to head type\n",
        "    if OUTPUT_MODE in (\"logits\", \"probs2\"):\n",
        "        if OUTPUT_MODE == \"logits\":\n",
        "            logp2 = _two_class_logprobs_from_logits(outs)  # (B, L-1, 2)\n",
        "        else:\n",
        "            logp2 = _two_class_logprobs_from_probs2(outs)  # (B, L-1, 2)\n",
        "        # Map token IDs {ZERO_ID, ONE_ID} -> class indices {0,1}\n",
        "        cls_idx = (labels == ONE_ID).astype(jnp.int32)[..., None]  # (B, L-1, 1)\n",
        "        # Gather log p(y_t): take_along_axis along last dim\n",
        "        logpy = jnp.take_along_axis(logp2, cls_idx, axis=-1).squeeze(-1)  # (B, L-1)\n",
        "\n",
        "    elif OUTPUT_MODE == \"prob1\":\n",
        "        # outs: (B, L-1, 1) or (B, L-1) — ensure shape (B, L-1, 1)\n",
        "        if outs.ndim == 2:\n",
        "            outs = outs[..., None]\n",
        "        logpy = _logprobs_for_labels_prob1(outs, labels)   # (B, L-1)\n",
        "\n",
        "    else:\n",
        "        raise ValueError(f\"Unknown OUTPUT_MODE={OUTPUT_MODE!r}\")\n",
        "\n",
        "    # Sum only over realized positions\n",
        "    logpy = jnp.where(mask, logpy, 0.0)                    # zero out padded parts\n",
        "    return logpy.sum(axis=1)                                # (B,)\n",
        "\n",
        "# -------------------- batched autoregressive sampling -------------------------\n",
        "def sample_batch_from_model(model, state, B: int, L: int, temperature: float = 1.0,\n",
        "                            key: jax.Array | None = None) -> tuple[jnp.ndarray, jax.Array]:\n",
        "    \"\"\"\n",
        "    Sample B sequences in parallel from a model, starting at BOS.\n",
        "    Single model.apply per step (shared boundary t for all rows).\n",
        "    \"\"\"\n",
        "    if key is None:\n",
        "        key = jax.random.PRNGKey(np.random.randint(0, 2**31 - 1))\n",
        "\n",
        "    ids = make_bos_only_ids_batch(B, L)        # (B, L)\n",
        "    # Emit exactly L-1 tokens after BOS (no EOS handling here)\n",
        "    for t in range(1, L):\n",
        "        outs = model.apply({\"params\": state.params}, ids)   # (B, L-1, V) or (B, L, V/1)\n",
        "        step_out = outs[:, t-1] if outs.shape[1] >= t else outs[:, -1]  # (B, V or 1)\n",
        "\n",
        "        # Convert to probs P(1|h); we only need class-1 prob to sample Bernoulli\n",
        "        if OUTPUT_MODE == \"logits\":\n",
        "            two = step_out[..., jnp.array([ZERO_ID, ONE_ID])]    # (B, 2)\n",
        "            logp2 = jax.nn.log_softmax(two, axis=-1)\n",
        "            p1 = jnp.exp(logp2[..., 1])                         # (B,)\n",
        "        elif OUTPUT_MODE == \"probs2\":\n",
        "            two = step_out[..., jnp.array([ZERO_ID, ONE_ID])]    # (B, 2)\n",
        "            two = jnp.maximum(two, 0.0)\n",
        "            denom = jnp.clip(two.sum(axis=-1), 1e-12)            # (B,)\n",
        "            p1 = jnp.clip(two[..., 1] / denom, 1e-9, 1-1e-9)\n",
        "        else:  # \"prob1\"\n",
        "            p1 = step_out.squeeze(-1)\n",
        "            if PROB1_IS_LOGIT:\n",
        "                p1 = jax.nn.sigmoid(p1)\n",
        "            p1 = jnp.clip(p1, 1e-9, 1-1e-9)\n",
        "\n",
        "        if temperature is not None and temperature != 1.0:\n",
        "            invT = 1.0 / float(temperature)\n",
        "            p0 = 1.0 - p1\n",
        "            a0 = p0 ** invT\n",
        "            a1 = p1 ** invT\n",
        "            Z = jnp.clip(a0 + a1, 1e-12)\n",
        "            p1 = a1 / Z\n",
        "\n",
        "        key, sk = jax.random.split(key)\n",
        "        bits = jax.random.bernoulli(sk, p1).astype(jnp.int32)    # (B,)\n",
        "        next_ids = jnp.where(bits == 1, ONE_ID, ZERO_ID)\n",
        "        ids = ids.at[:, t].set(next_ids)\n",
        "\n",
        "    return ids, key  # (B, L)\n",
        "\n",
        "\n"
      ],
      "metadata": {
        "id": "DL2xbicNtVtB"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "## DTV"
      ],
      "metadata": {
        "id": "eCwPKoRGNefc"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "from statistics import NormalDist\n",
        "import math\n",
        "import numpy as np\n",
        "import jax.numpy as jnp\n",
        "from jax.scipy.special import logsumexp\n",
        "import jax\n",
        "\n",
        "def estimate_tv_mixture_mc_fast(\n",
        "    modelP, stateP, modelQ, stateQ, L: int,\n",
        "    N: int = 2048, batch_size: int = 128,\n",
        "    temperature: float = 1.0, seed: int | None = None,\n",
        "    ci: float = 0.95,\n",
        "):\n",
        "    \"\"\"\n",
        "    Fast, batched mixture Monte Carlo estimator of TV(P,Q) over binary sequences\n",
        "    (after BOS), returning a normal-approximation confidence interval.\n",
        "\n",
        "    EXTERNAL DEPENDENCIES (must be defined elsewhere):\n",
        "      - sample_batch_from_model(model, state, B, L, temperature, key) -> (ids_BxL, next_key)\n",
        "      - batch_sequence_logprob_after_bos(model, state, ids_BxL, pad_id) -> jnp.ndarray shape (B,)\n",
        "      - PAD_ID (int)\n",
        "      - (Your model I/O and any OUTPUT_MODE config are consumed inside the helpers.)\n",
        "\n",
        "    Returns:\n",
        "      {\n",
        "        'tv_hat': float,        # Monte Carlo estimate of TV\n",
        "        'stderr': float,        # standard error of the estimate\n",
        "        'N': int,               # total samples\n",
        "        'ci_level': float,      # e.g., 0.95\n",
        "        'ci_lower': float,      # lower CI bound (clipped to [0,1])\n",
        "        'ci_upper': float       # upper CI bound (clipped to [0,1])\n",
        "      }\n",
        "    \"\"\"\n",
        "    assert 0 < ci < 1, \"ci must be in (0,1)\"\n",
        "    assert N > 0 and batch_size > 0\n",
        "\n",
        "    rng = np.random.default_rng(seed)\n",
        "    key = jax.random.PRNGKey(rng.integers(0, 2**31 - 1))\n",
        "\n",
        "    num_batches = (N + batch_size - 1) // batch_size\n",
        "    diffs_chunks = []\n",
        "\n",
        "    for _ in range(num_batches):\n",
        "        B = min(batch_size, N - sum(len(c) for c in diffs_chunks))\n",
        "        if B <= 0:\n",
        "            break\n",
        "\n",
        "        # Choose mixture component per row (roughly half from P, half from Q)\n",
        "        chooseP = rng.random(B) < 0.5\n",
        "        Bp = int(chooseP.sum())\n",
        "        Bq = B - Bp\n",
        "\n",
        "        # Sample from each model in batches, then stitch back preserving order\n",
        "        ids_BxL = jnp.zeros((B, L), dtype=jnp.int32)\n",
        "        if Bp > 0:\n",
        "            idsP, key = sample_batch_from_model(modelP, stateP, Bp, L, temperature, key)\n",
        "            ids_BxL = ids_BxL.at[chooseP].set(idsP)\n",
        "        if Bq > 0:\n",
        "            idsQ, key = sample_batch_from_model(modelQ, stateQ, Bq, L, temperature, key)\n",
        "            ids_BxL = ids_BxL.at[~chooseP].set(idsQ)\n",
        "\n",
        "        # Score log-likelihoods in one pass per model\n",
        "        logP = batch_sequence_logprob_after_bos(modelP, stateP, ids_BxL, PAD_ID)  # (B,)\n",
        "        logQ = batch_sequence_logprob_after_bos(modelQ, stateQ, ids_BxL, PAD_ID)  # (B,)\n",
        "\n",
        "        # Mixture weights under M = 0.5*(P+Q)\n",
        "        m_log = logsumexp(jnp.stack([logP, logQ], axis=0), axis=0) - math.log(2.0)  # (B,)\n",
        "        wP = jnp.exp(logP - m_log)\n",
        "        wQ = jnp.exp(logQ - m_log)\n",
        "\n",
        "        diffs_chunks.append(jnp.abs(wP - wQ))  # (B,)\n",
        "\n",
        "    diffs = jnp.concatenate(diffs_chunks, axis=0)[:N]  # (N,)\n",
        "    Y = 0.5 * diffs                                    # per-sample contributions in [0,1]\n",
        "    tv_hat = float(Y.mean())\n",
        "    stderr = float(Y.std(ddof=1) / math.sqrt(max(1, N)))\n",
        "\n",
        "    z = NormalDist().inv_cdf(0.5 + ci / 2.0)\n",
        "    half = z * stderr\n",
        "    ci_lower = max(0.0, tv_hat - half)\n",
        "    ci_upper = min(1.0, tv_hat + half)\n",
        "\n",
        "    return {\n",
        "        \"tv_hat\": tv_hat,\n",
        "        \"stderr\": stderr,\n",
        "        \"N\": int(N),\n",
        "        \"ci_level\": float(ci),\n",
        "        \"ci_lower\": float(ci_lower),\n",
        "        \"ci_upper\": float(ci_upper),\n",
        "    }\n"
      ],
      "metadata": {
        "id": "4l6DWmaIzkqB"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "import itertools\n",
        "import numpy as np\n",
        "import pandas as pd\n",
        "\n",
        "# ---- Config knobs ----\n",
        "N_TOTAL      = 128        # total mixture prefixes to sample per pair\n",
        "BATCH_SIZE   = 128         # compute batch size\n",
        "HEAD_TYPE    = \"probs2\"    # your head: \"probs2\" (linear renorm), or \"logits\"/\"prob1\"\n",
        "PROB1_IS_LOGIT = False     # only used if HEAD_TYPE==\"prob1\"\n",
        "TEMPERATURE  = 1.0         # sampling temperature for mixture prefixes\n",
        "SEED         = 0\n",
        "CI_LEVEL     = 0.95\n",
        "CONTEXT_LENGTH = CONTEXT_LEN\n",
        "\n",
        "\n",
        "# ---- Checkpoint set to compare ----\n",
        "TRAIN_SETS = [\"BERNOULLI\", \"BERNOULLI_1\", \"ROBUST_BERNOULLI\", \"POLYAURN\"]\n",
        "\n",
        "def load_model_for(train_name: str):\n",
        "    work_path = f'/content/drive/MyDrive/next-token-no-regret-0924/{train_name}-{CONTEXT_LENGTH}/'\n",
        "    cfg.workdir = work_path\n",
        "    mesh = create_mesh()\n",
        "    model, state = restore_model(workdir=work_path, config=cfg, mesh=mesh)\n",
        "    return model, state, mesh  # keep mesh alive to hold params on device"
      ],
      "metadata": {
        "id": "hY0k-oZZw9L2"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "\n",
        "# Load all models once\n",
        "models = {}\n",
        "for name in TRAIN_SETS:\n",
        "    models[name] = load_model_for(name)\n",
        "\n",
        "TARGETS = {\"BERNOULLI\", \"BERNOULLI_1\"}\n",
        "# Pairwise DTV with mixture prefixes\n",
        "pair_results = []\n",
        "for a, b in itertools.combinations(TRAIN_SETS, 2):\n",
        "    if not (a in TARGETS or b in TARGETS):\n",
        "        continue  # skip pairs with neither side in TARGETS\n",
        "    modelA, stateA, _ = models[a]\n",
        "    modelB, stateB, _ = models[b]\n",
        "    res = estimate_tv_mixture_mc_fast(\n",
        "        modelA, stateA, modelB, stateB, cfg.model.L, N=N_TOTAL, batch_size = BATCH_SIZE, temperature=1.0)\n",
        "    '''\n",
        "    {\n",
        "            \"tv_hat\": tv_hat,\n",
        "            \"stderr\": stderr,\n",
        "            \"N\": int(N),\n",
        "            \"ci_level\": float(ci),\n",
        "            \"ci_lower\": float(ci_lower),\n",
        "            \"ci_upper\": float(ci_upper),\n",
        "        }\n",
        "    '''\n",
        "    pair_results.append({\n",
        "        \"A\": a, \"B\": b,\n",
        "        \"mean_TV\": res[\"tv_hat\"],\n",
        "        \"ci_lower\": res[\"ci_lower\"],\n",
        "        \"ci_upper\": res[\"ci_upper\"],\n",
        "        \"stderr\": res[\"stderr\"],\n",
        "        \"N\": res[\"N\"],\n",
        "    })\n",
        "    print(\"pair \", a, \", \" , b, \" finished.\")\n",
        "\n",
        "# Build a symmetric matrix of mean next-token TV\n",
        "index = TRAIN_SETS\n",
        "mat = pd.DataFrame(0.0, index=index, columns=index, dtype=float)\n",
        "for r in pair_results:\n",
        "    a, b = r[\"A\"], r[\"B\"]\n",
        "    v = r[\"mean_TV\"]\n",
        "    mat.loc[a, b] = v\n",
        "    mat.loc[b, a] = v  # symmetric\n",
        "# Diagonal stays 0\n",
        "\n",
        "print(\"Pairwise mean TV:\")\n",
        "display(mat.style.format(\"{:.4f}\"))\n",
        "\n",
        "# Also show a tidy table with CIs\n",
        "df = pd.DataFrame(pair_results).sort_values([\"A\", \"B\"]).reset_index(drop=True)\n",
        "df[\"mean ± stderr\"] = df.apply(lambda r: f'{r[\"mean_TV\"]:.4f} ± {r[\"stderr\"]:.4f}', axis=1)\n",
        "df[\"{:.0f}% CI\".format(CI_LEVEL*100)] = df.apply(lambda r: f'[{r[\"ci_lower\"]:.4f}, {r[\"ci_upper\"]:.4f}]', axis=1)\n",
        "print(\"\\nPairwise details:\")\n",
        "display(df[[\"A\",\"B\",\"mean_TV\",\"mean ± stderr\",\"{:.0f}% CI\".format(CI_LEVEL*100),\"N\"]])\n"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 783
        },
        "id": "H7TviRlUZ6q0",
        "outputId": "82fcf802-fe36-4fff-cfb5-29cee1450f02"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "/usr/local/lib/python3.12/dist-packages/orbax/checkpoint/type_handlers.py:1544: UserWarning: Couldn't find sharding info under RestoreArgs. Populating sharding info from sharding file. Please note restoration time will be slightly increased due to reading from file instead of directly from RestoreArgs. Note also that this option is unsafe when restoring on a different topology than the checkpoint was saved with.\n",
            "  warnings.warn(\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Restoring checkpoint from step 500\n",
            "Restored checkpoint path: /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/checkpoint_500\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "/usr/local/lib/python3.12/dist-packages/orbax/checkpoint/type_handlers.py:1544: UserWarning: Couldn't find sharding info under RestoreArgs. Populating sharding info from sharding file. Please note restoration time will be slightly increased due to reading from file instead of directly from RestoreArgs. Note also that this option is unsafe when restoring on a different topology than the checkpoint was saved with.\n",
            "  warnings.warn(\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Restoring checkpoint from step 500\n",
            "Restored checkpoint path: /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/checkpoint_500\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "/usr/local/lib/python3.12/dist-packages/orbax/checkpoint/type_handlers.py:1544: UserWarning: Couldn't find sharding info under RestoreArgs. Populating sharding info from sharding file. Please note restoration time will be slightly increased due to reading from file instead of directly from RestoreArgs. Note also that this option is unsafe when restoring on a different topology than the checkpoint was saved with.\n",
            "  warnings.warn(\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Restoring checkpoint from step 500\n",
            "Restored checkpoint path: /content/drive/MyDrive/next-token-no-regret-0924/ROBUST_BERNOULLI-1024/checkpoint_500\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "/usr/local/lib/python3.12/dist-packages/orbax/checkpoint/type_handlers.py:1544: UserWarning: Couldn't find sharding info under RestoreArgs. Populating sharding info from sharding file. Please note restoration time will be slightly increased due to reading from file instead of directly from RestoreArgs. Note also that this option is unsafe when restoring on a different topology than the checkpoint was saved with.\n",
            "  warnings.warn(\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Restoring checkpoint from step 500\n",
            "Restored checkpoint path: /content/drive/MyDrive/next-token-no-regret-0924/POLYAURN-1024/checkpoint_500\n",
            "Pairwise mean TV:\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<pandas.io.formats.style.Styler at 0x7d419c67fe30>"
            ],
            "text/html": [
              "<style type=\"text/css\">\n",
              "</style>\n",
              "<table id=\"T_27023\" class=\"dataframe\">\n",
              "  <thead>\n",
              "    <tr>\n",
              "      <th class=\"blank level0\" >&nbsp;</th>\n",
              "      <th id=\"T_27023_level0_col0\" class=\"col_heading level0 col0\" >BERNOULLI</th>\n",
              "      <th id=\"T_27023_level0_col1\" class=\"col_heading level0 col1\" >BERNOULLI_1</th>\n",
              "      <th id=\"T_27023_level0_col2\" class=\"col_heading level0 col2\" >ROBUST_BERNOULLI</th>\n",
              "      <th id=\"T_27023_level0_col3\" class=\"col_heading level0 col3\" >POLYAURN</th>\n",
              "    </tr>\n",
              "  </thead>\n",
              "  <tbody>\n",
              "    <tr>\n",
              "      <th id=\"T_27023_level0_row0\" class=\"row_heading level0 row0\" >BERNOULLI</th>\n",
              "      <td id=\"T_27023_row0_col0\" class=\"data row0 col0\" >0.0000</td>\n",
              "      <td id=\"T_27023_row0_col1\" class=\"data row0 col1\" >0.4193</td>\n",
              "      <td id=\"T_27023_row0_col2\" class=\"data row0 col2\" >0.6869</td>\n",
              "      <td id=\"T_27023_row0_col3\" class=\"data row0 col3\" >1.0000</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th id=\"T_27023_level0_row1\" class=\"row_heading level0 row1\" >BERNOULLI_1</th>\n",
              "      <td id=\"T_27023_row1_col0\" class=\"data row1 col0\" >0.4193</td>\n",
              "      <td id=\"T_27023_row1_col1\" class=\"data row1 col1\" >0.0000</td>\n",
              "      <td id=\"T_27023_row1_col2\" class=\"data row1 col2\" >0.7602</td>\n",
              "      <td id=\"T_27023_row1_col3\" class=\"data row1 col3\" >1.0000</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th id=\"T_27023_level0_row2\" class=\"row_heading level0 row2\" >ROBUST_BERNOULLI</th>\n",
              "      <td id=\"T_27023_row2_col0\" class=\"data row2 col0\" >0.6869</td>\n",
              "      <td id=\"T_27023_row2_col1\" class=\"data row2 col1\" >0.7602</td>\n",
              "      <td id=\"T_27023_row2_col2\" class=\"data row2 col2\" >0.0000</td>\n",
              "      <td id=\"T_27023_row2_col3\" class=\"data row2 col3\" >0.0000</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th id=\"T_27023_level0_row3\" class=\"row_heading level0 row3\" >POLYAURN</th>\n",
              "      <td id=\"T_27023_row3_col0\" class=\"data row3 col0\" >1.0000</td>\n",
              "      <td id=\"T_27023_row3_col1\" class=\"data row3 col1\" >1.0000</td>\n",
              "      <td id=\"T_27023_row3_col2\" class=\"data row3 col2\" >0.0000</td>\n",
              "      <td id=\"T_27023_row3_col3\" class=\"data row3 col3\" >0.0000</td>\n",
              "    </tr>\n",
              "  </tbody>\n",
              "</table>\n"
            ]
          },
          "metadata": {}
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "Pairwise details:\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "             A                 B   mean_TV    mean ± stderr            95% CI  \\\n",
              "0    BERNOULLI       BERNOULLI_1  0.419342  0.4193 ± 0.0232  [0.3738, 0.4648]   \n",
              "1    BERNOULLI          POLYAURN  1.000022  1.0000 ± 0.0000  [1.0000, 1.0000]   \n",
              "2    BERNOULLI  ROBUST_BERNOULLI  0.686922  0.6869 ± 0.0295  [0.6291, 0.7448]   \n",
              "3  BERNOULLI_1          POLYAURN  1.000021  1.0000 ± 0.0000  [1.0000, 1.0000]   \n",
              "4  BERNOULLI_1  ROBUST_BERNOULLI  0.760152  0.7602 ± 0.0267  [0.7078, 0.8125]   \n",
              "\n",
              "     N  \n",
              "0  128  \n",
              "1  128  \n",
              "2  128  \n",
              "3  128  \n",
              "4  128  "
            ],
            "text/html": [
              "\n",
              "  <div id=\"df-8c3f7610-4b2b-4514-8f5b-9216b25bdd03\" class=\"colab-df-container\">\n",
              "    <div>\n",
              "<style scoped>\n",
              "    .dataframe tbody tr th:only-of-type {\n",
              "        vertical-align: middle;\n",
              "    }\n",
              "\n",
              "    .dataframe tbody tr th {\n",
              "        vertical-align: top;\n",
              "    }\n",
              "\n",
              "    .dataframe thead th {\n",
              "        text-align: right;\n",
              "    }\n",
              "</style>\n",
              "<table border=\"1\" class=\"dataframe\">\n",
              "  <thead>\n",
              "    <tr style=\"text-align: right;\">\n",
              "      <th></th>\n",
              "      <th>A</th>\n",
              "      <th>B</th>\n",
              "      <th>mean_TV</th>\n",
              "      <th>mean ± stderr</th>\n",
              "      <th>95% CI</th>\n",
              "      <th>N</th>\n",
              "    </tr>\n",
              "  </thead>\n",
              "  <tbody>\n",
              "    <tr>\n",
              "      <th>0</th>\n",
              "      <td>BERNOULLI</td>\n",
              "      <td>BERNOULLI_1</td>\n",
              "      <td>0.419342</td>\n",
              "      <td>0.4193 ± 0.0232</td>\n",
              "      <td>[0.3738, 0.4648]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>1</th>\n",
              "      <td>BERNOULLI</td>\n",
              "      <td>POLYAURN</td>\n",
              "      <td>1.000022</td>\n",
              "      <td>1.0000 ± 0.0000</td>\n",
              "      <td>[1.0000, 1.0000]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>2</th>\n",
              "      <td>BERNOULLI</td>\n",
              "      <td>ROBUST_BERNOULLI</td>\n",
              "      <td>0.686922</td>\n",
              "      <td>0.6869 ± 0.0295</td>\n",
              "      <td>[0.6291, 0.7448]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>3</th>\n",
              "      <td>BERNOULLI_1</td>\n",
              "      <td>POLYAURN</td>\n",
              "      <td>1.000021</td>\n",
              "      <td>1.0000 ± 0.0000</td>\n",
              "      <td>[1.0000, 1.0000]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>4</th>\n",
              "      <td>BERNOULLI_1</td>\n",
              "      <td>ROBUST_BERNOULLI</td>\n",
              "      <td>0.760152</td>\n",
              "      <td>0.7602 ± 0.0267</td>\n",
              "      <td>[0.7078, 0.8125]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "  </tbody>\n",
              "</table>\n",
              "</div>\n",
              "    <div class=\"colab-df-buttons\">\n",
              "\n",
              "  <div class=\"colab-df-container\">\n",
              "    <button class=\"colab-df-convert\" onclick=\"convertToInteractive('df-8c3f7610-4b2b-4514-8f5b-9216b25bdd03')\"\n",
              "            title=\"Convert this dataframe to an interactive table.\"\n",
              "            style=\"display:none;\">\n",
              "\n",
              "  <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\">\n",
              "    <path d=\"M120-120v-720h720v720H120Zm60-500h600v-160H180v160Zm220 220h160v-160H400v160Zm0 220h160v-160H400v160ZM180-400h160v-160H180v160Zm440 0h160v-160H620v160ZM180-180h160v-160H180v160Zm440 0h160v-160H620v160Z\"/>\n",
              "  </svg>\n",
              "    </button>\n",
              "\n",
              "  <style>\n",
              "    .colab-df-container {\n",
              "      display:flex;\n",
              "      gap: 12px;\n",
              "    }\n",
              "\n",
              "    .colab-df-convert {\n",
              "      background-color: #E8F0FE;\n",
              "      border: none;\n",
              "      border-radius: 50%;\n",
              "      cursor: pointer;\n",
              "      display: none;\n",
              "      fill: #1967D2;\n",
              "      height: 32px;\n",
              "      padding: 0 0 0 0;\n",
              "      width: 32px;\n",
              "    }\n",
              "\n",
              "    .colab-df-convert:hover {\n",
              "      background-color: #E2EBFA;\n",
              "      box-shadow: 0px 1px 2px rgba(60, 64, 67, 0.3), 0px 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "      fill: #174EA6;\n",
              "    }\n",
              "\n",
              "    .colab-df-buttons div {\n",
              "      margin-bottom: 4px;\n",
              "    }\n",
              "\n",
              "    [theme=dark] .colab-df-convert {\n",
              "      background-color: #3B4455;\n",
              "      fill: #D2E3FC;\n",
              "    }\n",
              "\n",
              "    [theme=dark] .colab-df-convert:hover {\n",
              "      background-color: #434B5C;\n",
              "      box-shadow: 0px 1px 3px 1px rgba(0, 0, 0, 0.15);\n",
              "      filter: drop-shadow(0px 1px 2px rgba(0, 0, 0, 0.3));\n",
              "      fill: #FFFFFF;\n",
              "    }\n",
              "  </style>\n",
              "\n",
              "    <script>\n",
              "      const buttonEl =\n",
              "        document.querySelector('#df-8c3f7610-4b2b-4514-8f5b-9216b25bdd03 button.colab-df-convert');\n",
              "      buttonEl.style.display =\n",
              "        google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "\n",
              "      async function convertToInteractive(key) {\n",
              "        const element = document.querySelector('#df-8c3f7610-4b2b-4514-8f5b-9216b25bdd03');\n",
              "        const dataTable =\n",
              "          await google.colab.kernel.invokeFunction('convertToInteractive',\n",
              "                                                    [key], {});\n",
              "        if (!dataTable) return;\n",
              "\n",
              "        const docLinkHtml = 'Like what you see? Visit the ' +\n",
              "          '<a target=\"_blank\" href=https://colab.research.google.com/notebooks/data_table.ipynb>data table notebook</a>'\n",
              "          + ' to learn more about interactive tables.';\n",
              "        element.innerHTML = '';\n",
              "        dataTable['output_type'] = 'display_data';\n",
              "        await google.colab.output.renderOutput(dataTable, element);\n",
              "        const docLink = document.createElement('div');\n",
              "        docLink.innerHTML = docLinkHtml;\n",
              "        element.appendChild(docLink);\n",
              "      }\n",
              "    </script>\n",
              "  </div>\n",
              "\n",
              "\n",
              "    <div id=\"df-3883fc51-5c9f-44fb-b1fb-1775a1b566e3\">\n",
              "      <button class=\"colab-df-quickchart\" onclick=\"quickchart('df-3883fc51-5c9f-44fb-b1fb-1775a1b566e3')\"\n",
              "                title=\"Suggest charts\"\n",
              "                style=\"display:none;\">\n",
              "\n",
              "<svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\"viewBox=\"0 0 24 24\"\n",
              "     width=\"24px\">\n",
              "    <g>\n",
              "        <path d=\"M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM9 17H7v-7h2v7zm4 0h-2V7h2v10zm4 0h-2v-4h2v4z\"/>\n",
              "    </g>\n",
              "</svg>\n",
              "      </button>\n",
              "\n",
              "<style>\n",
              "  .colab-df-quickchart {\n",
              "      --bg-color: #E8F0FE;\n",
              "      --fill-color: #1967D2;\n",
              "      --hover-bg-color: #E2EBFA;\n",
              "      --hover-fill-color: #174EA6;\n",
              "      --disabled-fill-color: #AAA;\n",
              "      --disabled-bg-color: #DDD;\n",
              "  }\n",
              "\n",
              "  [theme=dark] .colab-df-quickchart {\n",
              "      --bg-color: #3B4455;\n",
              "      --fill-color: #D2E3FC;\n",
              "      --hover-bg-color: #434B5C;\n",
              "      --hover-fill-color: #FFFFFF;\n",
              "      --disabled-bg-color: #3B4455;\n",
              "      --disabled-fill-color: #666;\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart {\n",
              "    background-color: var(--bg-color);\n",
              "    border: none;\n",
              "    border-radius: 50%;\n",
              "    cursor: pointer;\n",
              "    display: none;\n",
              "    fill: var(--fill-color);\n",
              "    height: 32px;\n",
              "    padding: 0;\n",
              "    width: 32px;\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart:hover {\n",
              "    background-color: var(--hover-bg-color);\n",
              "    box-shadow: 0 1px 2px rgba(60, 64, 67, 0.3), 0 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "    fill: var(--button-hover-fill-color);\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart-complete:disabled,\n",
              "  .colab-df-quickchart-complete:disabled:hover {\n",
              "    background-color: var(--disabled-bg-color);\n",
              "    fill: var(--disabled-fill-color);\n",
              "    box-shadow: none;\n",
              "  }\n",
              "\n",
              "  .colab-df-spinner {\n",
              "    border: 2px solid var(--fill-color);\n",
              "    border-color: transparent;\n",
              "    border-bottom-color: var(--fill-color);\n",
              "    animation:\n",
              "      spin 1s steps(1) infinite;\n",
              "  }\n",
              "\n",
              "  @keyframes spin {\n",
              "    0% {\n",
              "      border-color: transparent;\n",
              "      border-bottom-color: var(--fill-color);\n",
              "      border-left-color: var(--fill-color);\n",
              "    }\n",
              "    20% {\n",
              "      border-color: transparent;\n",
              "      border-left-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "    }\n",
              "    30% {\n",
              "      border-color: transparent;\n",
              "      border-left-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "      border-right-color: var(--fill-color);\n",
              "    }\n",
              "    40% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "    }\n",
              "    60% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "    }\n",
              "    80% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "      border-bottom-color: var(--fill-color);\n",
              "    }\n",
              "    90% {\n",
              "      border-color: transparent;\n",
              "      border-bottom-color: var(--fill-color);\n",
              "    }\n",
              "  }\n",
              "</style>\n",
              "\n",
              "      <script>\n",
              "        async function quickchart(key) {\n",
              "          const quickchartButtonEl =\n",
              "            document.querySelector('#' + key + ' button');\n",
              "          quickchartButtonEl.disabled = true;  // To prevent multiple clicks.\n",
              "          quickchartButtonEl.classList.add('colab-df-spinner');\n",
              "          try {\n",
              "            const charts = await google.colab.kernel.invokeFunction(\n",
              "                'suggestCharts', [key], {});\n",
              "          } catch (error) {\n",
              "            console.error('Error during call to suggestCharts:', error);\n",
              "          }\n",
              "          quickchartButtonEl.classList.remove('colab-df-spinner');\n",
              "          quickchartButtonEl.classList.add('colab-df-quickchart-complete');\n",
              "        }\n",
              "        (() => {\n",
              "          let quickchartButtonEl =\n",
              "            document.querySelector('#df-3883fc51-5c9f-44fb-b1fb-1775a1b566e3 button');\n",
              "          quickchartButtonEl.style.display =\n",
              "            google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "        })();\n",
              "      </script>\n",
              "    </div>\n",
              "\n",
              "    </div>\n",
              "  </div>\n"
            ],
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "dataframe",
              "summary": "{\n  \"name\": \"display(df[[\\\"A\\\",\\\"B\\\",\\\"mean_TV\\\",\\\"mean \\u00b1 stderr\\\",\\\"{:\",\n  \"rows\": 5,\n  \"fields\": [\n    {\n      \"column\": \"A\",\n      \"properties\": {\n        \"dtype\": \"category\",\n        \"num_unique_values\": 2,\n        \"samples\": [\n          \"BERNOULLI_1\",\n          \"BERNOULLI\"\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"B\",\n      \"properties\": {\n        \"dtype\": \"string\",\n        \"num_unique_values\": 3,\n        \"samples\": [\n          \"BERNOULLI_1\",\n          \"POLYAURN\"\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"mean_TV\",\n      \"properties\": {\n        \"dtype\": \"number\",\n        \"std\": 0.24275804300037396,\n        \"min\": 0.419342041015625,\n        \"max\": 1.0000224113464355,\n        \"num_unique_values\": 5,\n        \"samples\": [\n          1.0000224113464355,\n          0.7601515054702759\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"mean \\u00b1 stderr\",\n      \"properties\": {\n        \"dtype\": \"string\",\n        \"num_unique_values\": 4,\n        \"samples\": [\n          \"1.0000 \\u00b1 0.0000\",\n          \"0.7602 \\u00b1 0.0267\"\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"95% CI\",\n      \"properties\": {\n        \"dtype\": \"string\",\n        \"num_unique_values\": 4,\n        \"samples\": [\n          \"[1.0000, 1.0000]\",\n          \"[0.7078, 0.8125]\"\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"N\",\n      \"properties\": {\n        \"dtype\": \"number\",\n        \"std\": 0,\n        \"min\": 128,\n        \"max\": 128,\n        \"num_unique_values\": 1,\n        \"samples\": [\n          128\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    }\n  ]\n}"
            }
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "## Next token DTV"
      ],
      "metadata": {
        "id": "PKV7jhQ3NhZK"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "import math\n",
        "import numpy as np\n",
        "import jax\n",
        "import jax.numpy as jnp\n",
        "\n",
        "def estimate_next_token_tv_nonstationary_13_to_23(\n",
        "    modelP, stateP, modelQ, stateQ, L: int,\n",
        "    N: int = 4096, batch_size: int = 256,\n",
        "    head_type: str = \"probs2\", prob1_is_logit: bool = False,\n",
        "    seed: int | None = None, ci: float = 0.95,\n",
        "):\n",
        "    \"\"\"\n",
        "    Next-token TV with prefixes drawn from a *fixed exogenous* distribution:\n",
        "      first half of tokens after BOS are Ber(1/3), second half are Ber(2/3).\n",
        "\n",
        "    head_type: \"logits\" (log-softmax), \"probs2\" (linear renorm), \"prob1\" (scalar p(1) or logit)\n",
        "    Depends on: PAD_ID, BOS_ID, ZERO_ID, ONE_ID\n",
        "    \"\"\"\n",
        "\n",
        "    assert 0 < ci < 1\n",
        "    assert N > 0 and batch_size > 0\n",
        "    rng = np.random.default_rng(seed)\n",
        "\n",
        "    T = L - 1\n",
        "    sum_tv_t = jnp.zeros((T,), dtype=jnp.float32)\n",
        "    count_t = jnp.zeros((T,), dtype=jnp.float32)\n",
        "    flat_vals = []  # collect per-position TV values for CI over overall mean\n",
        "\n",
        "    # ----- head normalization: p1 from model outputs -----\n",
        "    def p1_from_out(outs):\n",
        "        \"\"\"outs: (B, T, V) or (B, T, 1) -> return p1 (B, T)\"\"\"\n",
        "        if head_type == \"logits\":\n",
        "            two = outs[..., jnp.array([ZERO_ID, ONE_ID])]         # (B, T, 2)\n",
        "            logp2 = jax.nn.log_softmax(two, axis=-1)\n",
        "            return jnp.exp(logp2[..., 1])\n",
        "        elif head_type == \"probs2\":\n",
        "            two = outs[..., jnp.array([ZERO_ID, ONE_ID])]         # (B, T, 2)\n",
        "            two = jnp.maximum(two, 0.0)\n",
        "            denom = jnp.clip(two.sum(axis=-1), 1e-12)             # (B, T)\n",
        "            return jnp.clip(two[..., 1] / denom, 1e-9, 1-1e-9)\n",
        "        else:  # \"prob1\"\n",
        "            p1 = outs.squeeze(-1)                                 # (B, T)\n",
        "            if prob1_is_logit:\n",
        "                p1 = jax.nn.sigmoid(p1)\n",
        "            return jnp.clip(p1, 1e-9, 1-1e-9)\n",
        "\n",
        "    num_batches = (N + batch_size - 1) // batch_size\n",
        "    total_seen = 0\n",
        "\n",
        "    # Precompute the per-position Bernoulli probs (length T)\n",
        "    T1 = T // 2\n",
        "    probs_vec = np.concatenate([np.full(T1, 1.0/3.0), np.full(T - T1, 2.0/3.0)])  # (T,)\n",
        "\n",
        "    for _ in range(num_batches):\n",
        "        B = min(batch_size, N - total_seen)\n",
        "        if B <= 0:\n",
        "            break\n",
        "        total_seen += B\n",
        "\n",
        "        # ---------- sample B sequences from the fixed source ----------\n",
        "        # bits_BxT ~ Ber(1/3) for t<=T1, Ber(2/3) for t>T1\n",
        "        U = rng.random((B, T))\n",
        "        bits_BxT = (U < probs_vec).astype(np.int32)               # (B, T) in {0,1}\n",
        "\n",
        "        # Build IDs (B, L): BOS at 0, then ZERO_ID/ONE_ID\n",
        "        ids_BxL = np.full((B, L), PAD_ID, dtype=np.int32)\n",
        "        ids_BxL[:, 0] = BOS_ID\n",
        "        ids_BxL[:, 1:] = np.where(bits_BxT == 1, ONE_ID, ZERO_ID)\n",
        "        ids_BxL = jnp.array(ids_BxL)\n",
        "\n",
        "        # ---------- one forward per model; align to predict tokens 1..L-1 ----------\n",
        "        outP = modelP.apply({\"params\": stateP.params}, ids_BxL)   # (B, L-1, V) or (B, L, V/1)\n",
        "        outQ = modelQ.apply({\"params\": stateQ.params}, ids_BxL)\n",
        "        if outP.shape[1] == L:   # if model returns (B, L, ·), drop last step\n",
        "            outP = outP[:, :-1]\n",
        "            outQ = outQ[:, :-1]\n",
        "\n",
        "        # Compute per-step p(1|h) for both models\n",
        "        p1P = p1_from_out(outP)                                   # (B, T)\n",
        "        p1Q = p1_from_out(outQ)                                   # (B, T)\n",
        "\n",
        "        # All positions are realized (no EOS); mask is all True\n",
        "        tv_bt = jnp.abs(p1P - p1Q)                                # (B, T)\n",
        "\n",
        "        sum_tv_t = sum_tv_t + tv_bt.sum(axis=0)                   # (T,)\n",
        "        count_t = count_t + jnp.full((T,), B, dtype=jnp.float32)  # (T,)\n",
        "\n",
        "        flat_vals.append(tv_bt.reshape(-1))                       # collect values for CI\n",
        "\n",
        "    # Per-step averages\n",
        "    denom = jnp.maximum(count_t, 1.0)\n",
        "    tv_per_t = (sum_tv_t / denom).astype(jnp.float32)             # (T,)\n",
        "\n",
        "    # Scalar summaries\n",
        "    weighted_mean = float(sum_tv_t.sum() / float(count_t.sum()))  # average over all (t, h)\n",
        "    uniform_over_t = float(tv_per_t.mean())                       # mean over steps (t)\n",
        "\n",
        "    # CI for the overall mean (normal approx)\n",
        "    if flat_vals:\n",
        "        vals = jnp.concatenate(flat_vals, axis=0)                 # shape (N*T,)\n",
        "        m = int(vals.shape[0])\n",
        "        mean = float(vals.mean())\n",
        "        sd = float(vals.std(ddof=1))\n",
        "        se = sd / math.sqrt(max(1, m))\n",
        "        from statistics import NormalDist\n",
        "        z = NormalDist().inv_cdf(0.5 + ci/2.0)\n",
        "        half = z * se\n",
        "        ci_lo = max(0.0, mean - half)\n",
        "        ci_hi = min(1.0, mean + half)\n",
        "    else:\n",
        "        mean, se, ci_lo, ci_hi = 0.0, 0.0, 0.0, 0.0\n",
        "\n",
        "    return {\n",
        "        \"per_step_tv\": tv_per_t,            # jnp.ndarray (T,)\n",
        "        \"mean_weighted\": weighted_mean,     # E_{t,h~source} |p1P - p1Q|\n",
        "        \"mean_uniform_over_t\": uniform_over_t,\n",
        "        \"ci_level\": float(ci),\n",
        "        \"mean_ci_lower\": ci_lo,\n",
        "        \"mean_ci_upper\": ci_hi,\n",
        "        \"stderr\": se,\n",
        "        \"N_prefixes\": int(N),\n",
        "    }\n"
      ],
      "metadata": {
        "id": "7MxlXL9qrWoS"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "import itertools\n",
        "import numpy as np\n",
        "import pandas as pd\n",
        "\n",
        "# ---- Config knobs ----\n",
        "N_TOTAL        = 128          # total prefixes to sample per pair\n",
        "BATCH_SIZE     = 128          # compute batch size\n",
        "HEAD_TYPE      = \"probs2\"     # \"probs2\" (linear renorm) or \"logits\"/\"prob1\"\n",
        "PROB1_IS_LOGIT = False        # only used if HEAD_TYPE==\"prob1\"\n",
        "SEED           = 0\n",
        "CI_LEVEL       = 0.95\n",
        "CONTEXT_LENGTH = CONTEXT_LEN\n",
        "\n",
        "# ---- Checkpoint set to compare ----\n",
        "TRAIN_SETS = [\"BERNOULLI\", \"BERNOULLI_1\", \"ROBUST_BERNOULLI\", \"POLYAURN\"]\n",
        "\n",
        "def load_model_for(train_name: str):\n",
        "    work_path = f'/content/drive/MyDrive/next-token-no-regret-0924/{train_name}-{CONTEXT_LENGTH}/'\n",
        "    cfg.workdir = work_path\n",
        "    mesh = create_mesh()\n",
        "    model, state = restore_model(workdir=work_path, config=cfg, mesh=mesh)\n",
        "    return model, state, mesh  # keep mesh alive to hold params on device\n",
        "\n",
        "# Load all models once\n",
        "models = {name: load_model_for(name) for name in TRAIN_SETS}\n",
        "\n",
        "TARGETS = {\"BERNOULLI\", \"BERNOULLI_1\"}  # only compute pairs touching these\n",
        "\n",
        "# Pairwise next-token DTV with non-stationary 1/3→2/3 prefixes\n",
        "pair_results = []\n",
        "for a, b in itertools.combinations(TRAIN_SETS, 2):\n",
        "    if not (a in TARGETS or b in TARGETS):\n",
        "        continue  # skip pairs with neither side in TARGETS\n",
        "\n",
        "    modelA, stateA, _ = models[a]\n",
        "    modelB, stateB, _ = models[b]\n",
        "    res = estimate_next_token_tv_nonstationary_13_to_23(\n",
        "        modelA, stateA, modelB, stateB, CONTEXT_LENGTH,\n",
        "        N=N_TOTAL, batch_size=BATCH_SIZE,\n",
        "        head_type=HEAD_TYPE, prob1_is_logit=PROB1_IS_LOGIT,\n",
        "        seed=SEED, ci=CI_LEVEL,\n",
        "    )\n",
        "    pair_results.append({\n",
        "        \"A\": a, \"B\": b,\n",
        "        \"mean_next_token_TV\": res[\"mean_weighted\"],\n",
        "        \"ci_lower\": res[\"mean_ci_lower\"],\n",
        "        \"ci_upper\": res[\"mean_ci_upper\"],\n",
        "        \"stderr\": res[\"stderr\"],\n",
        "        \"N_prefixes\": res[\"N_prefixes\"],\n",
        "    })\n",
        "\n",
        "# Build a symmetric matrix (NaN where not evaluated)\n",
        "index = TRAIN_SETS\n",
        "mat = pd.DataFrame(np.nan, index=index, columns=index, dtype=float)\n",
        "np.fill_diagonal(mat.values, 0.0)\n",
        "for r in pair_results:\n",
        "    a, b, v = r[\"A\"], r[\"B\"], r[\"mean_next_token_TV\"]\n",
        "    mat.loc[a, b] = v\n",
        "    mat.loc[b, a] = v\n",
        "\n",
        "print(\"Pairwise mean next-token TV (prefixes: first half Ber(1/3), second half Ber(2/3)):\")\n",
        "display(mat.style.format(\"{:.4f}\"))\n",
        "\n",
        "# Tidy table with CIs for computed pairs\n",
        "df = pd.DataFrame(pair_results).sort_values([\"A\", \"B\"]).reset_index(drop=True)\n",
        "df[\"mean ± stderr\"] = df.apply(lambda r: f'{r[\"mean_next_token_TV\"]:.4f} ± {r[\"stderr\"]:.4f}', axis=1)\n",
        "df[f\"{int(CI_LEVEL*100)}% CI\"] = df.apply(lambda r: f'[{r[\"ci_lower\"]:.4f}, {r[\"ci_upper\"]:.4f}]', axis=1)\n",
        "print(\"\\nEvaluated pairs:\")\n",
        "display(df[[\"A\",\"B\",\"mean_next_token_TV\",\"mean ± stderr\",f\"{int(CI_LEVEL*100)}% CI\",\"N_prefixes\"]])\n"
      ],
      "metadata": {
        "id": "Zz5QS2UormJz",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 887
        },
        "outputId": "ce9440f9-25ac-48c5-c3f8-924ed5fc7999"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "/usr/local/lib/python3.12/dist-packages/orbax/checkpoint/type_handlers.py:1544: UserWarning: Couldn't find sharding info under RestoreArgs. Populating sharding info from sharding file. Please note restoration time will be slightly increased due to reading from file instead of directly from RestoreArgs. Note also that this option is unsafe when restoring on a different topology than the checkpoint was saved with.\n",
            "  warnings.warn(\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Restoring checkpoint from step 500\n",
            "Restored checkpoint path: /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI-1024/checkpoint_500\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "/usr/local/lib/python3.12/dist-packages/orbax/checkpoint/type_handlers.py:1544: UserWarning: Couldn't find sharding info under RestoreArgs. Populating sharding info from sharding file. Please note restoration time will be slightly increased due to reading from file instead of directly from RestoreArgs. Note also that this option is unsafe when restoring on a different topology than the checkpoint was saved with.\n",
            "  warnings.warn(\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Restoring checkpoint from step 500\n",
            "Restored checkpoint path: /content/drive/MyDrive/next-token-no-regret-0924/BERNOULLI_1-1024/checkpoint_500\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "/usr/local/lib/python3.12/dist-packages/orbax/checkpoint/type_handlers.py:1544: UserWarning: Couldn't find sharding info under RestoreArgs. Populating sharding info from sharding file. Please note restoration time will be slightly increased due to reading from file instead of directly from RestoreArgs. Note also that this option is unsafe when restoring on a different topology than the checkpoint was saved with.\n",
            "  warnings.warn(\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Restoring checkpoint from step 500\n",
            "Restored checkpoint path: /content/drive/MyDrive/next-token-no-regret-0924/ROBUST_BERNOULLI-1024/checkpoint_500\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "WARNING:absl:Configured `CheckpointManager` using deprecated legacy API. Please follow the instructions at https://orbax.readthedocs.io/en/latest/api_refactor.html to migrate.\n",
            "/usr/local/lib/python3.12/dist-packages/orbax/checkpoint/type_handlers.py:1544: UserWarning: Couldn't find sharding info under RestoreArgs. Populating sharding info from sharding file. Please note restoration time will be slightly increased due to reading from file instead of directly from RestoreArgs. Note also that this option is unsafe when restoring on a different topology than the checkpoint was saved with.\n",
            "  warnings.warn(\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Restoring checkpoint from step 500\n",
            "Restored checkpoint path: /content/drive/MyDrive/next-token-no-regret-0924/POLYAURN-1024/checkpoint_500\n",
            "Pairwise mean next-token TV (prefixes: first half Ber(1/3), second half Ber(2/3)):\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<pandas.io.formats.style.Styler at 0x7d3b34502fc0>"
            ],
            "text/html": [
              "<style type=\"text/css\">\n",
              "</style>\n",
              "<table id=\"T_11d85\" class=\"dataframe\">\n",
              "  <thead>\n",
              "    <tr>\n",
              "      <th class=\"blank level0\" >&nbsp;</th>\n",
              "      <th id=\"T_11d85_level0_col0\" class=\"col_heading level0 col0\" >BERNOULLI</th>\n",
              "      <th id=\"T_11d85_level0_col1\" class=\"col_heading level0 col1\" >BERNOULLI_1</th>\n",
              "      <th id=\"T_11d85_level0_col2\" class=\"col_heading level0 col2\" >ROBUST_BERNOULLI</th>\n",
              "      <th id=\"T_11d85_level0_col3\" class=\"col_heading level0 col3\" >POLYAURN</th>\n",
              "    </tr>\n",
              "  </thead>\n",
              "  <tbody>\n",
              "    <tr>\n",
              "      <th id=\"T_11d85_level0_row0\" class=\"row_heading level0 row0\" >BERNOULLI</th>\n",
              "      <td id=\"T_11d85_row0_col0\" class=\"data row0 col0\" >0.0000</td>\n",
              "      <td id=\"T_11d85_row0_col1\" class=\"data row0 col1\" >0.0156</td>\n",
              "      <td id=\"T_11d85_row0_col2\" class=\"data row0 col2\" >0.0199</td>\n",
              "      <td id=\"T_11d85_row0_col3\" class=\"data row0 col3\" >0.1529</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th id=\"T_11d85_level0_row1\" class=\"row_heading level0 row1\" >BERNOULLI_1</th>\n",
              "      <td id=\"T_11d85_row1_col0\" class=\"data row1 col0\" >0.0156</td>\n",
              "      <td id=\"T_11d85_row1_col1\" class=\"data row1 col1\" >0.0000</td>\n",
              "      <td id=\"T_11d85_row1_col2\" class=\"data row1 col2\" >0.0299</td>\n",
              "      <td id=\"T_11d85_row1_col3\" class=\"data row1 col3\" >0.1655</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th id=\"T_11d85_level0_row2\" class=\"row_heading level0 row2\" >ROBUST_BERNOULLI</th>\n",
              "      <td id=\"T_11d85_row2_col0\" class=\"data row2 col0\" >0.0199</td>\n",
              "      <td id=\"T_11d85_row2_col1\" class=\"data row2 col1\" >0.0299</td>\n",
              "      <td id=\"T_11d85_row2_col2\" class=\"data row2 col2\" >0.0000</td>\n",
              "      <td id=\"T_11d85_row2_col3\" class=\"data row2 col3\" >nan</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th id=\"T_11d85_level0_row3\" class=\"row_heading level0 row3\" >POLYAURN</th>\n",
              "      <td id=\"T_11d85_row3_col0\" class=\"data row3 col0\" >0.1529</td>\n",
              "      <td id=\"T_11d85_row3_col1\" class=\"data row3 col1\" >0.1655</td>\n",
              "      <td id=\"T_11d85_row3_col2\" class=\"data row3 col2\" >nan</td>\n",
              "      <td id=\"T_11d85_row3_col3\" class=\"data row3 col3\" >0.0000</td>\n",
              "    </tr>\n",
              "  </tbody>\n",
              "</table>\n"
            ]
          },
          "metadata": {}
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "Evaluated pairs:\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "             A                 B  mean_next_token_TV    mean ± stderr  \\\n",
              "0    BERNOULLI       BERNOULLI_1            0.015590  0.0156 ± 0.0000   \n",
              "1    BERNOULLI          POLYAURN            0.152893  0.1529 ± 0.0003   \n",
              "2    BERNOULLI  ROBUST_BERNOULLI            0.019881  0.0199 ± 0.0001   \n",
              "3  BERNOULLI_1          POLYAURN            0.165492  0.1655 ± 0.0004   \n",
              "4  BERNOULLI_1  ROBUST_BERNOULLI            0.029917  0.0299 ± 0.0001   \n",
              "\n",
              "             95% CI  N_prefixes  \n",
              "0  [0.0156, 0.0156]         128  \n",
              "1  [0.1522, 0.1536]         128  \n",
              "2  [0.0198, 0.0200]         128  \n",
              "3  [0.1648, 0.1662]         128  \n",
              "4  [0.0298, 0.0300]         128  "
            ],
            "text/html": [
              "\n",
              "  <div id=\"df-f04b740d-299d-41c9-b809-5b44632210a3\" class=\"colab-df-container\">\n",
              "    <div>\n",
              "<style scoped>\n",
              "    .dataframe tbody tr th:only-of-type {\n",
              "        vertical-align: middle;\n",
              "    }\n",
              "\n",
              "    .dataframe tbody tr th {\n",
              "        vertical-align: top;\n",
              "    }\n",
              "\n",
              "    .dataframe thead th {\n",
              "        text-align: right;\n",
              "    }\n",
              "</style>\n",
              "<table border=\"1\" class=\"dataframe\">\n",
              "  <thead>\n",
              "    <tr style=\"text-align: right;\">\n",
              "      <th></th>\n",
              "      <th>A</th>\n",
              "      <th>B</th>\n",
              "      <th>mean_next_token_TV</th>\n",
              "      <th>mean ± stderr</th>\n",
              "      <th>95% CI</th>\n",
              "      <th>N_prefixes</th>\n",
              "    </tr>\n",
              "  </thead>\n",
              "  <tbody>\n",
              "    <tr>\n",
              "      <th>0</th>\n",
              "      <td>BERNOULLI</td>\n",
              "      <td>BERNOULLI_1</td>\n",
              "      <td>0.015590</td>\n",
              "      <td>0.0156 ± 0.0000</td>\n",
              "      <td>[0.0156, 0.0156]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>1</th>\n",
              "      <td>BERNOULLI</td>\n",
              "      <td>POLYAURN</td>\n",
              "      <td>0.152893</td>\n",
              "      <td>0.1529 ± 0.0003</td>\n",
              "      <td>[0.1522, 0.1536]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>2</th>\n",
              "      <td>BERNOULLI</td>\n",
              "      <td>ROBUST_BERNOULLI</td>\n",
              "      <td>0.019881</td>\n",
              "      <td>0.0199 ± 0.0001</td>\n",
              "      <td>[0.0198, 0.0200]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>3</th>\n",
              "      <td>BERNOULLI_1</td>\n",
              "      <td>POLYAURN</td>\n",
              "      <td>0.165492</td>\n",
              "      <td>0.1655 ± 0.0004</td>\n",
              "      <td>[0.1648, 0.1662]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>4</th>\n",
              "      <td>BERNOULLI_1</td>\n",
              "      <td>ROBUST_BERNOULLI</td>\n",
              "      <td>0.029917</td>\n",
              "      <td>0.0299 ± 0.0001</td>\n",
              "      <td>[0.0298, 0.0300]</td>\n",
              "      <td>128</td>\n",
              "    </tr>\n",
              "  </tbody>\n",
              "</table>\n",
              "</div>\n",
              "    <div class=\"colab-df-buttons\">\n",
              "\n",
              "  <div class=\"colab-df-container\">\n",
              "    <button class=\"colab-df-convert\" onclick=\"convertToInteractive('df-f04b740d-299d-41c9-b809-5b44632210a3')\"\n",
              "            title=\"Convert this dataframe to an interactive table.\"\n",
              "            style=\"display:none;\">\n",
              "\n",
              "  <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\">\n",
              "    <path d=\"M120-120v-720h720v720H120Zm60-500h600v-160H180v160Zm220 220h160v-160H400v160Zm0 220h160v-160H400v160ZM180-400h160v-160H180v160Zm440 0h160v-160H620v160ZM180-180h160v-160H180v160Zm440 0h160v-160H620v160Z\"/>\n",
              "  </svg>\n",
              "    </button>\n",
              "\n",
              "  <style>\n",
              "    .colab-df-container {\n",
              "      display:flex;\n",
              "      gap: 12px;\n",
              "    }\n",
              "\n",
              "    .colab-df-convert {\n",
              "      background-color: #E8F0FE;\n",
              "      border: none;\n",
              "      border-radius: 50%;\n",
              "      cursor: pointer;\n",
              "      display: none;\n",
              "      fill: #1967D2;\n",
              "      height: 32px;\n",
              "      padding: 0 0 0 0;\n",
              "      width: 32px;\n",
              "    }\n",
              "\n",
              "    .colab-df-convert:hover {\n",
              "      background-color: #E2EBFA;\n",
              "      box-shadow: 0px 1px 2px rgba(60, 64, 67, 0.3), 0px 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "      fill: #174EA6;\n",
              "    }\n",
              "\n",
              "    .colab-df-buttons div {\n",
              "      margin-bottom: 4px;\n",
              "    }\n",
              "\n",
              "    [theme=dark] .colab-df-convert {\n",
              "      background-color: #3B4455;\n",
              "      fill: #D2E3FC;\n",
              "    }\n",
              "\n",
              "    [theme=dark] .colab-df-convert:hover {\n",
              "      background-color: #434B5C;\n",
              "      box-shadow: 0px 1px 3px 1px rgba(0, 0, 0, 0.15);\n",
              "      filter: drop-shadow(0px 1px 2px rgba(0, 0, 0, 0.3));\n",
              "      fill: #FFFFFF;\n",
              "    }\n",
              "  </style>\n",
              "\n",
              "    <script>\n",
              "      const buttonEl =\n",
              "        document.querySelector('#df-f04b740d-299d-41c9-b809-5b44632210a3 button.colab-df-convert');\n",
              "      buttonEl.style.display =\n",
              "        google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "\n",
              "      async function convertToInteractive(key) {\n",
              "        const element = document.querySelector('#df-f04b740d-299d-41c9-b809-5b44632210a3');\n",
              "        const dataTable =\n",
              "          await google.colab.kernel.invokeFunction('convertToInteractive',\n",
              "                                                    [key], {});\n",
              "        if (!dataTable) return;\n",
              "\n",
              "        const docLinkHtml = 'Like what you see? Visit the ' +\n",
              "          '<a target=\"_blank\" href=https://colab.research.google.com/notebooks/data_table.ipynb>data table notebook</a>'\n",
              "          + ' to learn more about interactive tables.';\n",
              "        element.innerHTML = '';\n",
              "        dataTable['output_type'] = 'display_data';\n",
              "        await google.colab.output.renderOutput(dataTable, element);\n",
              "        const docLink = document.createElement('div');\n",
              "        docLink.innerHTML = docLinkHtml;\n",
              "        element.appendChild(docLink);\n",
              "      }\n",
              "    </script>\n",
              "  </div>\n",
              "\n",
              "\n",
              "    <div id=\"df-12c7b602-ee16-4110-9565-d4ca1bc66ca0\">\n",
              "      <button class=\"colab-df-quickchart\" onclick=\"quickchart('df-12c7b602-ee16-4110-9565-d4ca1bc66ca0')\"\n",
              "                title=\"Suggest charts\"\n",
              "                style=\"display:none;\">\n",
              "\n",
              "<svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\"viewBox=\"0 0 24 24\"\n",
              "     width=\"24px\">\n",
              "    <g>\n",
              "        <path d=\"M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM9 17H7v-7h2v7zm4 0h-2V7h2v10zm4 0h-2v-4h2v4z\"/>\n",
              "    </g>\n",
              "</svg>\n",
              "      </button>\n",
              "\n",
              "<style>\n",
              "  .colab-df-quickchart {\n",
              "      --bg-color: #E8F0FE;\n",
              "      --fill-color: #1967D2;\n",
              "      --hover-bg-color: #E2EBFA;\n",
              "      --hover-fill-color: #174EA6;\n",
              "      --disabled-fill-color: #AAA;\n",
              "      --disabled-bg-color: #DDD;\n",
              "  }\n",
              "\n",
              "  [theme=dark] .colab-df-quickchart {\n",
              "      --bg-color: #3B4455;\n",
              "      --fill-color: #D2E3FC;\n",
              "      --hover-bg-color: #434B5C;\n",
              "      --hover-fill-color: #FFFFFF;\n",
              "      --disabled-bg-color: #3B4455;\n",
              "      --disabled-fill-color: #666;\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart {\n",
              "    background-color: var(--bg-color);\n",
              "    border: none;\n",
              "    border-radius: 50%;\n",
              "    cursor: pointer;\n",
              "    display: none;\n",
              "    fill: var(--fill-color);\n",
              "    height: 32px;\n",
              "    padding: 0;\n",
              "    width: 32px;\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart:hover {\n",
              "    background-color: var(--hover-bg-color);\n",
              "    box-shadow: 0 1px 2px rgba(60, 64, 67, 0.3), 0 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "    fill: var(--button-hover-fill-color);\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart-complete:disabled,\n",
              "  .colab-df-quickchart-complete:disabled:hover {\n",
              "    background-color: var(--disabled-bg-color);\n",
              "    fill: var(--disabled-fill-color);\n",
              "    box-shadow: none;\n",
              "  }\n",
              "\n",
              "  .colab-df-spinner {\n",
              "    border: 2px solid var(--fill-color);\n",
              "    border-color: transparent;\n",
              "    border-bottom-color: var(--fill-color);\n",
              "    animation:\n",
              "      spin 1s steps(1) infinite;\n",
              "  }\n",
              "\n",
              "  @keyframes spin {\n",
              "    0% {\n",
              "      border-color: transparent;\n",
              "      border-bottom-color: var(--fill-color);\n",
              "      border-left-color: var(--fill-color);\n",
              "    }\n",
              "    20% {\n",
              "      border-color: transparent;\n",
              "      border-left-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "    }\n",
              "    30% {\n",
              "      border-color: transparent;\n",
              "      border-left-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "      border-right-color: var(--fill-color);\n",
              "    }\n",
              "    40% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "    }\n",
              "    60% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "    }\n",
              "    80% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "      border-bottom-color: var(--fill-color);\n",
              "    }\n",
              "    90% {\n",
              "      border-color: transparent;\n",
              "      border-bottom-color: var(--fill-color);\n",
              "    }\n",
              "  }\n",
              "</style>\n",
              "\n",
              "      <script>\n",
              "        async function quickchart(key) {\n",
              "          const quickchartButtonEl =\n",
              "            document.querySelector('#' + key + ' button');\n",
              "          quickchartButtonEl.disabled = true;  // To prevent multiple clicks.\n",
              "          quickchartButtonEl.classList.add('colab-df-spinner');\n",
              "          try {\n",
              "            const charts = await google.colab.kernel.invokeFunction(\n",
              "                'suggestCharts', [key], {});\n",
              "          } catch (error) {\n",
              "            console.error('Error during call to suggestCharts:', error);\n",
              "          }\n",
              "          quickchartButtonEl.classList.remove('colab-df-spinner');\n",
              "          quickchartButtonEl.classList.add('colab-df-quickchart-complete');\n",
              "        }\n",
              "        (() => {\n",
              "          let quickchartButtonEl =\n",
              "            document.querySelector('#df-12c7b602-ee16-4110-9565-d4ca1bc66ca0 button');\n",
              "          quickchartButtonEl.style.display =\n",
              "            google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "        })();\n",
              "      </script>\n",
              "    </div>\n",
              "\n",
              "    </div>\n",
              "  </div>\n"
            ],
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "dataframe",
              "summary": "{\n  \"name\": \"display(df[[\\\"A\\\",\\\"B\\\",\\\"mean_next_token_TV\\\",\\\"mean \\u00b1 stderr\\\",f\\\"{int(CI_LEVEL*100)}% CI\\\",\\\"N_prefixes\\\"]])\",\n  \"rows\": 5,\n  \"fields\": [\n    {\n      \"column\": \"A\",\n      \"properties\": {\n        \"dtype\": \"category\",\n        \"num_unique_values\": 2,\n        \"samples\": [\n          \"BERNOULLI_1\",\n          \"BERNOULLI\"\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"B\",\n      \"properties\": {\n        \"dtype\": \"string\",\n        \"num_unique_values\": 3,\n        \"samples\": [\n          \"BERNOULLI_1\",\n          \"POLYAURN\"\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"mean_next_token_TV\",\n      \"properties\": {\n        \"dtype\": \"number\",\n        \"std\": 0.0755658448151357,\n        \"min\": 0.01559029147028923,\n        \"max\": 0.16549187898635864,\n        \"num_unique_values\": 5,\n        \"samples\": [\n          0.15289264917373657,\n          0.02991674654185772\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"mean \\u00b1 stderr\",\n      \"properties\": {\n        \"dtype\": \"string\",\n        \"num_unique_values\": 5,\n        \"samples\": [\n          \"0.1529 \\u00b1 0.0003\",\n          \"0.0299 \\u00b1 0.0001\"\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"95% CI\",\n      \"properties\": {\n        \"dtype\": \"string\",\n        \"num_unique_values\": 5,\n        \"samples\": [\n          \"[0.1522, 0.1536]\",\n          \"[0.0298, 0.0300]\"\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"N_prefixes\",\n      \"properties\": {\n        \"dtype\": \"number\",\n        \"std\": 0,\n        \"min\": 128,\n        \"max\": 128,\n        \"num_unique_values\": 1,\n        \"samples\": [\n          128\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    }\n  ]\n}"
            }
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Plot mean p(1|h_t) curves and per-half NT-TV under Ber(1/3)->Ber(2/3) source ===\n",
        "# Assumes:\n",
        "#   - You already have modelA/stateA and modelB/stateB loaded (e.g., \"POLYAURN\" vs \"BERNOULLI\")\n",
        "#   - PAD_ID, BOS_ID, ZERO_ID, ONE_ID are defined\n",
        "#   - Your model.apply({\"params\": state.params}, ids_BxL) returns (B, L-1, V) or (B, L, V/1)\n",
        "#   - Set HEAD_TYPE = \"probs2\" (linear renorm), or \"logits\" (two-logit logaddexp), or \"prob1\"\n",
        "#   - If \"prob1\" emits a logit, set PROB1_IS_LOGIT = True\n",
        "\n",
        "import math, numpy as np, jax, jax.numpy as jnp\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "HEAD_TYPE = \"probs2\"     # \"probs2\" | \"logits\" | \"prob1\"\n",
        "PROB1_IS_LOGIT = False\n",
        "SEED = 0\n",
        "\n",
        "def _p1_from_out(outs, head_type, prob1_is_logit):\n",
        "    \"\"\"\n",
        "    outs: (B, T, V) or (B, T, 1) -> returns p1 of shape (B, T).\n",
        "    - logits: use two-class logaddexp (no full softmax)\n",
        "    - probs2: linear renorm over {ZERO_ID, ONE_ID}\n",
        "    - prob1 : scalar p(1) (or sigmoid(logit) if prob1_is_logit)\n",
        "    \"\"\"\n",
        "    if head_type == \"logits\":\n",
        "        # two-logit logaddexp (memory-safe)\n",
        "        z0 = outs[..., ZERO_ID]\n",
        "        z1 = outs[..., ONE_ID]\n",
        "        lse = jnp.logaddexp(z0, z1)\n",
        "        return jnp.exp(jnp.clip(z1 - lse, -60.0, 60.0))\n",
        "    elif head_type == \"probs2\":\n",
        "        two = outs[..., jnp.array([ZERO_ID, ONE_ID])]    # (B, T, 2)\n",
        "        two = jnp.maximum(two, 0.0)\n",
        "        denom = jnp.clip(two.sum(axis=-1), 1e-12)        # (B, T)\n",
        "        return jnp.clip(two[..., 1] / denom, 1e-9, 1-1e-9)\n",
        "    else:  # \"prob1\"\n",
        "        p1 = outs.squeeze(-1)\n",
        "        if prob1_is_logit:\n",
        "            p1 = jax.nn.sigmoid(p1)\n",
        "        return jnp.clip(p1, 1e-9, 1-1e-9)\n",
        "\n",
        "def _align_outputs(outs, L):\n",
        "    # Align to predict tokens 1..L-1 (T = L-1)\n",
        "    return outs[:, :-1] if outs.shape[1] == L else outs\n",
        "\n",
        "def estimate_curves_and_nttv_halves_nonstationary(\n",
        "    modelA, stateA, modelB, stateB, L: int,\n",
        "    N: int = 8192, batch_size: int = 256,\n",
        "    head_type: str = HEAD_TYPE, prob1_is_logit: bool = PROB1_IS_LOGIT,\n",
        "    seed: int = SEED,\n",
        "):\n",
        "    \"\"\"\n",
        "    Returns:\n",
        "      t (T,), mean_p1_A (T,), mean_p1_B (T,),\n",
        "      nttv_first_half, nttv_second_half, nttv_overall\n",
        "    \"\"\"\n",
        "    rng = np.random.default_rng(seed)\n",
        "    T = L - 1\n",
        "    T1 = T // 2\n",
        "    probs_vec = np.concatenate([np.full(T1, 1/3), np.full(T - T1, 2/3)])  # (T,)\n",
        "\n",
        "    sum_p1_A = jnp.zeros((T,), dtype=jnp.float32)\n",
        "    sum_p1_B = jnp.zeros((T,), dtype=jnp.float32)\n",
        "    sum_abs_diff = jnp.zeros((T,), dtype=jnp.float32)\n",
        "    count = 0\n",
        "\n",
        "    num_batches = (N + batch_size - 1) // batch_size\n",
        "    for _ in range(num_batches):\n",
        "        B = min(batch_size, N - count)\n",
        "        if B <= 0:\n",
        "            break\n",
        "        count += B\n",
        "\n",
        "        # Sample bits under the 1/3 -> 2/3 process\n",
        "        U = rng.random((B, T))\n",
        "        bits = (U < probs_vec).astype(np.int32)  # (B, T) in {0,1}\n",
        "\n",
        "        # Build ids (B, L): BOS + realized bits\n",
        "        ids = np.full((B, L), PAD_ID, dtype=np.int32)\n",
        "        ids[:, 0] = BOS_ID\n",
        "        ids[:, 1:] = np.where(bits == 1, ONE_ID, ZERO_ID)\n",
        "        ids = jnp.array(ids)\n",
        "\n",
        "        # One forward per model\n",
        "        outA = _align_outputs(modelA.apply({\"params\": stateA.params}, ids), L)\n",
        "        outB = _align_outputs(modelB.apply({\"params\": stateB.params}, ids), L)\n",
        "\n",
        "        p1A = _p1_from_out(outA, head_type, prob1_is_logit)  # (B, T)\n",
        "        p1B = _p1_from_out(outB, head_type, prob1_is_logit)  # (B, T)\n",
        "\n",
        "        sum_p1_A += p1A.sum(axis=0)\n",
        "        sum_p1_B += p1B.sum(axis=0)\n",
        "        sum_abs_diff += jnp.abs(p1A - p1B).sum(axis=0)\n",
        "\n",
        "    mean_p1_A = sum_p1_A / max(1, count)\n",
        "    mean_p1_B = sum_p1_B / max(1, count)\n",
        "    mean_abs_diff = sum_abs_diff / max(1, count)  # per-step NT-TV (binary)\n",
        "\n",
        "    # Per-half and overall means\n",
        "    nttv_first = float(mean_abs_diff[:T1].mean() if T1 > 0 else 0.0)\n",
        "    nttv_second = float(mean_abs_diff[T1:].mean() if T - T1 > 0 else 0.0)\n",
        "    nttv_overall = float(mean_abs_diff.mean())\n",
        "\n",
        "    t = np.arange(1, L)  # positions after BOS\n",
        "    return t, np.array(mean_p1_A), np.array(mean_p1_B), nttv_first, nttv_second, nttv_overall\n",
        "\n",
        "def plot_mean_p1_curves(\n",
        "    modelA, stateA, labelA,\n",
        "    modelB, stateB, labelB,\n",
        "    L: int, N: int = 8192, batch_size: int = 256,\n",
        "    head_type: str = HEAD_TYPE, prob1_is_logit: bool = PROB1_IS_LOGIT,\n",
        "    seed: int = SEED,\n",
        "):\n",
        "    t, mA, mB, nttv_first, nttv_second, nttv_overall = estimate_curves_and_nttv_halves_nonstationary(\n",
        "        modelA, stateA, modelB, stateB, L, N, batch_size, head_type, prob1_is_logit, seed\n",
        "    )\n",
        "    T = L - 1; T1 = T // 2\n",
        "\n",
        "    # Plot curves + ground-truth schedule\n",
        "    plt.figure(figsize=(8, 4.2))\n",
        "    plt.plot(t, mA, label=labelA)\n",
        "    plt.plot(t, mB, label=labelB)\n",
        "    # ground-truth p*(t)\n",
        "    true = np.concatenate([np.full(T1, 1/3), np.full(T - T1, 2/3)])\n",
        "    plt.plot(t, true, linestyle='--', linewidth=1.2, label='target (1/3 → 2/3)')\n",
        "    plt.axvline(x=T1, linestyle=':', linewidth=1.0)\n",
        "    plt.xlabel('position t (after BOS)')\n",
        "    plt.ylabel('mean p(1 | h_t)')\n",
        "    plt.title('Mean next-token probability under Ber(1/3) → Ber(2/3) source')\n",
        "    plt.legend()\n",
        "    plt.tight_layout()\n",
        "    plt.show()\n",
        "\n",
        "    print(f\"Per-half NT-TV: first half = {nttv_first:.4f}, second half = {nttv_second:.4f}\")\n",
        "    print(f\"Overall mean NT-TV = {nttv_overall:.4f}\")\n"
      ],
      "metadata": {
        "id": "yMjDw-iNwDIX"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# ---- Checkpoint set to compare ----\n",
        "TRAIN_SETS = [\"BERNOULLI\", \"BERNOULLI_1\", \"ROBUST_BERNOULLI\", \"POLYAURN\"]\n",
        "# Load all models once\n",
        "models = {name: load_model_for(name) for name in TRAIN_SETS}"
      ],
      "metadata": {
        "id": "Gq6FFsF-4GbN"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "\n",
        "\n",
        "print(HEAD_TYPE)\n",
        "# ==== Example usage ====\n",
        "modelA, stateA, _ = models[\"POLYAURN\"]\n",
        "modelB, stateB, _ = models[\"BERNOULLI\"]\n",
        "plot_mean_p1_curves(modelA, stateA, \"POLYAURN\", modelB, stateB, \"BERNOULLI\", L=cfg.model.L,\n",
        "                    N=128, batch_size=128, head_type=HEAD_TYPE, prob1_is_logit=PROB1_IS_LOGIT, seed=0)\n"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 478
        },
        "id": "KG4tqldCwpj-",
        "outputId": "86de8490-c9e1-4427-a83d-5537667ca8a3"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "probs2\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 800x420 with 1 Axes>"
            ],
            "image/png": "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1YqXl5fi4uKitGnTRvn9998VRVGUZcuWKYBy5swZ4+/mP88995yiKIoSGRmpvPPOO8ZzHj16VLn22msVV1dXxd3dXbnpppuUzMxM4/7nnntO6dChg/L1118rkZGRioeHhzJq1CglPz/f5n3Wde2aTp48qWg0GmXnzp0W29966y0lISFBcXFxUcLCwpQHHnhAKSgoqPX8ffv2tdq2bNkyRVEUJS0tTbnpppsUT09PxdvbW7n22muV1NRU47XGjBmjjBgxQnn55ZeV4OBgJSoqymZev/zyS8XT09Nim16vVxwcHJQffvjBeC0HBwdl8uTJVse/9957CqCsW7dOURRFmThxotKvXz+b1zK3YsUKRafTKcXFxWdNK4QQF4u0WAghRCP76quvsLe3Z8OGDbz77ru8/fbbfPbZZ8b9Y8eOZdOmTSxcuJC1a9eiKArDhg0zPsWeMGECZWVlrFy5kh07dvD666/j5uZmdZ1evXoxY8YMiyfs5k/LqxkMBkaMGMHp06dZsWIFS5cu5fDhw4waNcoi3aFDh1iwYAGLFi1i0aJFrFixgtdee83mPdb32gCrV6/GxcWF1q1bW2zXarW899577Nq1i6+++op//vmHKVOmGM9f3ULw888/k5GRwcKFC7n55psZMmSI8Zq9evWioqKCwYMH4+7uzqpVq1izZg1ubm4MGTLEomUiOTmZffv2sXTpUhYtWmQzrzXp9Xq++uorADp37gzATz/9REVFhc37vf/++3Fzc+O7774DICgoiF27drFz5846r5OYmEhlZSXr16+vV76EEOJisG/sDAghRHMXHh7OO++8g0ajIS4ujh07dvDOO+9w7733cuDAARYuXMiaNWvo1asXAN988w3h4eEsWLCAm266ibS0NEaOHEm7du0AiImJsXkdnU6Hp6cnGo2GoKCgWvOTnJzMjh07SE1NJTw8HICvv/6atm3bsnHjRrp27QqoAcjs2bONXY7uuOMOkpOTeeWVV8752gBHjx4lMDDQqhuU+TiDqKgoXn75ZcaPH8+HH36ITqcjICAAAB8fH+M1nJ2dKSsrs7jm3LlzMRgMfPbZZ8YuYl9++SVeXl4sX76cQYMGAeDq6spnn32GTqerM795eXnGQK6kpAQHBwc++eQTWrRoAcD+/fvx9PQkODjY5usSExPD/v37AXj44YdZtWoV7dq1IzIykh49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\n"
          },
          "metadata": {}
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Per-half NT-TV: first half = 0.0370, second half = 0.2685\n",
            "Overall mean NT-TV = 0.1529\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "from google.colab import drive\n",
        "drive.flush_and_unmount()"
      ],
      "metadata": {
        "id": "kzgcX59uMYB1"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "import os\n",
        "os.kill(os.getpid(), 9)"
      ],
      "metadata": {
        "id": "4WG5STRCexqI"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "# Calculate theoretical optimal"
      ],
      "metadata": {
        "id": "uOzuei0dZMfB"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "import numpy as np\n",
        "import matplotlib.pyplot as plt\n",
        "from scipy.stats import gaussian_kde\n",
        "\n",
        "rng = np.random.default_rng(0)  # reproducible\n",
        "\n",
        "# ---- Polya-urn sequence ----\n",
        "def simulate_one(L: int):\n",
        "    \"\"\"Simulate one sequence and accumulate best-possible log-loss per step.\"\"\"\n",
        "    bits = [rng.choice([\"0\",\"1\"]), rng.choice([\"0\",\"1\"])]\n",
        "    loss_sum = 0.0     # sum of per-step cross-entropy (nats)\n",
        "    H_sum = 0.0        # sum of per-step Bernoulli entropy H(p_t) (nats)\n",
        "    tokens = 0\n",
        "\n",
        "    for _ in range(2, L):\n",
        "        p = bits.count(\"1\") / len(bits)          # true best predictor at this step\n",
        "        y = 1 if rng.random() < p else 0         # next bit\n",
        "\n",
        "        if 0.0 < p < 1.0:\n",
        "            loss_sum += -(y*np.log(p) + (1-y)*np.log(1-p))          # nats\n",
        "            H_sum   += -(p*np.log(p) + (1-p)*np.log(1-p))           # nats\n",
        "        # if p is 0 or 1, the process is deterministic => both terms contribute 0\n",
        "        tokens += 1\n",
        "        bits.append(\"1\" if y == 1 else \"0\")\n",
        "\n",
        "    degenerate = (bits.count(\"1\") in (0, L))  # drop all-0/all-1 to match your original filter\n",
        "    return degenerate, bits, loss_sum, H_sum, tokens\n",
        "\n",
        "def expected_logloss_best_predictor(L: int, trials: int = 10_000):\n",
        "    final_ps = []\n",
        "    total_loss = 0.0\n",
        "    total_H = 0.0\n",
        "    total_tokens = 0\n",
        "\n",
        "    for _ in range(trials):\n",
        "        degenerate, bits, loss_sum, H_sum, tokens = simulate_one(L)\n",
        "        if degenerate:\n",
        "            continue\n",
        "        final_ps.append(bits.count(\"1\") / L)\n",
        "        total_loss += loss_sum\n",
        "        total_H += H_sum\n",
        "        total_tokens += tokens\n",
        "\n",
        "    ce_per_token_nats = total_loss / total_tokens\n",
        "    H_per_token_nats  = total_H / total_tokens\n",
        "    return np.array(final_ps), ce_per_token_nats, H_per_token_nats\n",
        "\n",
        "# ---- Parameters ----\n",
        "L = 128\n",
        "trials = 10_000\n",
        "\n",
        "# ---- Run simulation ----\n",
        "freqs, ce_nats, H_nats = expected_logloss_best_predictor(L, trials)\n",
        "ce_bits = ce_nats / np.log(2.0)\n",
        "H_bits  = H_nats  / np.log(2.0)\n",
        "\n",
        "# ---- Constant-1/2 baseline ----\n",
        "ce05_nats = np.log(2.0)          # = 0.6931... nats/token\n",
        "ce05_bits = ce05_nats / np.log(2.0)  # = 1.0 bits/token\n",
        "\n",
        "# ---- KDE + histogram of final empirical p ----\n",
        "kde = gaussian_kde(freqs)\n",
        "x_eval = np.linspace(0, 1, 200)\n",
        "density = kde(x_eval)\n",
        "\n",
        "plt.figure(figsize=(7,4.5))\n",
        "plt.hist(freqs, bins=50, density=True, alpha=0.3, label=\"Histogram of final p\")\n",
        "plt.plot(x_eval, density, label=\"KDE of final p\")\n",
        "plt.xlabel('Empirical Frequency of 1s (final p)')\n",
        "plt.ylabel('Density')\n",
        "plt.title(f'Final p under Polya-urn (L={L}, trials={trials})')\n",
        "\n",
        "# Annotate expected per-token loss/entropy (best predictor vs. 1/2 baseline)\n",
        "plt.text(0.02, max(density)*0.92,\n",
        "         f'Best possible expected loss (nats/token): {ce_nats:.4f}\\n'\n",
        "         f'Best possible expected loss (bits/token): {ce_bits:.4f}\\n'\n",
        "         f'Avg conditional entropy H(p_t) (bits/token): {H_bits:.4f}\\n'\n",
        "         f'Baseline CE (q=1/2) (nats/token): {ce05_nats:.4f}\\n'\n",
        "         f'Baseline CE (q=1/2) (bits/token): {ce05_bits:.4f}\\n'\n",
        "         f'Improvement over q=1/2 (nats/token): {ce05_nats - ce_nats:.4f}\\n'\n",
        "         f'|CE - H| (nats/token): {abs(ce_nats - H_nats):.2e}')\n",
        "\n",
        "plt.legend()\n",
        "plt.tight_layout()\n",
        "plt.show()\n",
        "\n",
        "print(f\"Best possible expected log-loss (nats/token): {ce_nats:.6f}\")\n",
        "print(f\"Average conditional entropy H(p_t) (nats/token): {H_nats:.6f}\")\n",
        "print(f\"Difference CE - H (nats/token): {ce_nats - H_nats:.2e}\")\n",
        "print(f\"(in bits) CE: {ce_bits:.6f}, H: {H_bits:.6f}\")\n",
        "print(f\"Baseline CE q=1/2 (nats/token): {ce05_nats:.6f}  |  (bits/token): {ce05_bits:.6f}\")\n",
        "print(f\"Improvement over q=1/2 (nats/token): {ce05_nats - ce_nats:.6f}  \"\n",
        "      f\"|  (bits/token): {1.0 - ce_bits:.6f}\")\n"
      ],
      "metadata": {
        "id": "ojr7N0YaQJ3W"
      },
      "execution_count": null,
      "outputs": []
    }
  ]
}