{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os \n",
    "os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"0\"\n",
    "import json\n",
    "import math\n",
    "import pathlib\n",
    "from typing import Dict, Optional, Sequence\n",
    "\n",
    "from peft import get_peft_config, get_peft_model, PromptTuningInit, PromptTuningConfig, TaskType, PeftType, AutoPeftModel, AutoPeftModelForCausalLM\n",
    "import numpy as np\n",
    "import torch\n",
    "from torch import nn\n",
    "from torch.utils.data import Dataset\n",
    "import transformers\n",
    "from transformers import Trainer, BitsAndBytesConfig\n",
    "from transformers.trainer_pt_utils import LabelSmoother\n",
    "from transformers import LlamaForCausalLM\n",
    "from prompt.model.modeling_llama_custom import LlamaForCausalLM as CustomLlamaForCausalLM\n",
    "\n",
    "from fastchat.conversation import SeparatorStyle\n",
    "from fastchat.model.model_adapter import get_conversation_template\n",
    "from torch.nn import CrossEntropyLoss\n",
    "from torch.nn import functional as F\n",
    "\n",
    "from contextlib import contextmanager\n",
    "from prompt.utils import *\n",
    "from prompt.model.model import PromptDecoder, AutoPromptDecoder, PromptConfig\n",
    "from prompt.model.kv_cache import *\n",
    "\n",
    "from pprint import pprint\n",
    "\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "torch.cuda.empty_cache()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "_____Python, Pytorch, Cuda info____\n",
      "__Python VERSION: 3.10.12 (main, Nov 20 2023, 15:14:05) [GCC 11.4.0]\n",
      "__pyTorch VERSION: 2.2.2+cu121\n",
      "__CUDA RUNTIME API VERSION\n",
      "nvcc: NVIDIA (R) Cuda compiler driver\n",
      "Copyright (c) 2005-2022 NVIDIA Corporation\n",
      "Built on Wed_Sep_21_10:33:58_PDT_2022\n",
      "Cuda compilation tools, release 11.8, V11.8.89\n",
      "Build cuda_11.8.r11.8/compiler.31833905_0\n",
      "__CUDNN VERSION: 8902\n",
      "_____nvidia-smi GPU details____\n",
      "index, name, driver_version, memory.total [MiB], memory.used [MiB], memory.free [MiB]\n",
      "0, NVIDIA GeForce RTX 4090, 535.154.05, 24564 MiB, 14 MiB, 24203 MiB\n",
      "1, NVIDIA GeForce RTX 4090, 535.154.05, 24564 MiB, 24181 MiB, 35 MiB\n",
      "2, NVIDIA GeForce RTX 4090, 535.154.05, 24564 MiB, 24181 MiB, 35 MiB\n",
      "3, NVIDIA GeForce RTX 4090, 535.154.05, 24564 MiB, 24181 MiB, 35 MiB\n",
      "_____Device assignments____\n",
      "Number CUDA Devices: 1\n",
      "Current cuda device:  0  **May not correspond to nvidia-smi ID above, check visibility parameter\n",
      "Device name:  NVIDIA GeForce RTX 4090\n",
      "Compute Capability: (8, 9)\n",
      "Number of Multiprocessors (Compute Units): 128\n",
      "Total Memory (GB): 23.64971923828125\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import sys\n",
    "import os\n",
    "from subprocess import call\n",
    "print('_____Python, Pytorch, Cuda info____')\n",
    "print('__Python VERSION:', sys.version)\n",
    "print('__pyTorch VERSION:', torch.__version__)\n",
    "print('__CUDA RUNTIME API VERSION')\n",
    "os.system('nvcc --version')\n",
    "print('__CUDNN VERSION:', torch.backends.cudnn.version())\n",
    "print('_____nvidia-smi GPU details____')\n",
    "call([\"nvidia-smi\", \"--format=csv\", \"--query-gpu=index,name,driver_version,memory.total,memory.used,memory.free\"])\n",
    "print('_____Device assignments____')\n",
    "print('Number CUDA Devices:', torch.cuda.device_count())\n",
    "print ('Current cuda device: ', torch.cuda.current_device(), ' **May not correspond to nvidia-smi ID above, check visibility parameter')\n",
    "print(\"Device name: \", torch.cuda.get_device_name(torch.cuda.current_device()))\n",
    "print(f\"Compute Capability: {torch.cuda.get_device_capability(device)}\")\n",
    "print(f\"Number of Multiprocessors (Compute Units): {torch.cuda.get_device_properties(device).multi_processor_count}\")\n",
    "print(f\"Total Memory (GB): {torch.cuda.get_device_properties(device).total_memory / (1024 ** 3)}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  ['r ', 'A0', '01', '07', 'B0', 'A1', '11', 'A2', 'C0', 'A3']\n",
      "  ['0', '-3', '1', '7', '-2', '-3', '11', '-3', '-1', '-3']\n",
      "  ['0', '1', '7', '11']\n",
      "[[ 0  1  3]\n",
      " [ 0  2 -1]]\n",
      "['r ' '01' '07' '11']\n",
      "[['r ' '01' '11']\n",
      " ['r ' '07' '11']]\n"
     ]
    }
   ],
   "source": [
    "def pad_path(path, length, pad_value=-2):\n",
    "    \"\"\"\n",
    "    Pad the given path list with a specific value up to a specified length.\n",
    "    \n",
    "    Parameters:\n",
    "    - path (list): The original list that needs padding.\n",
    "    - length (int): The desired length of the padded list.\n",
    "    - pad_value (optional, default=-2): The value to use for padding.\n",
    "    \n",
    "    Returns:\n",
    "    - list: A new list based on the original path but padded to the desired length.\n",
    "    \n",
    "    Example:\n",
    "    >>> pad_path([1,2,3], 5)\n",
    "    [1, 2, 3, -2, -2]\n",
    "    \n",
    "    Note:\n",
    "    If the given path is already longer than the specified length, \n",
    "    then no padding occurs, and the original path is returned.\n",
    "    \"\"\"\n",
    "    \n",
    "    # Calculate the number of padding values needed by subtracting the length\n",
    "    # of the path from the desired length.\n",
    "    # Append the padding values to the original path and return the new list.\n",
    "    return path + [pad_value] * (length - len(path))\n",
    "\n",
    "def generate_buffers(candidate_lists, TOPK=10, **kwargs):\n",
    "    num_special_tokens = 3\n",
    "    # add special tokens to the original_paths\n",
    "    candidate_lists = sorted([(path.copy(), acc, n) for path, acc, n in candidate_lists], key=lambda x: (len(x[0]), x[0]))\n",
    "    original_paths = [path.copy() for (path, _, _) in candidate_lists]\n",
    "    paths = [(path.copy(), n) for (path, _, n) in candidate_lists]\n",
    "    # add special tokens to the paths\n",
    "    if num_special_tokens > 0:\n",
    "        paths += [(path + [-1], n) for (path, n) in paths] + [([-1], num_special_tokens)]\n",
    "    for i in range(2, num_special_tokens+1):\n",
    "        paths += [(path + [-i], n) for (path, n) in paths if path[-1]-1 == -i and i <= n]\n",
    "    paths = sorted([path for path, _ in paths], key=lambda x: (len(x), x))\n",
    "    # Sort the choices based on their lengths and then their values\n",
    "    paths = sorted(paths, key=lambda x: (len(x), x))\n",
    "    prompt_length = len(paths) + 1\n",
    "    \n",
    "    # Get indices for each special token\n",
    "    special_token_indices = {}\n",
    "    for i, path in enumerate([[]] + original_paths):\n",
    "        candidate_special_token_indices = []\n",
    "        for j in range(1, num_special_tokens+1):\n",
    "            extended_path = path+[-n for n in range(1, j+1)]\n",
    "            if extended_path in paths:\n",
    "                candidate_special_token_indices.append(paths.index(extended_path)+1)\n",
    "        special_token_indices[i] = candidate_special_token_indices\n",
    "    normal_token_indices = [0] + [i+1 for i in range(len(paths)) if paths[i][-1] >= 0]\n",
    "    normal_token_indices = torch.tensor(normal_token_indices, dtype=torch.long)\n",
    "\n",
    "    # Initialize depth_counts to keep track of how many choices have a particular depth\n",
    "    depth_counts = []\n",
    "    prev_depth = 0\n",
    "    for path in paths:\n",
    "        depth = len(path)\n",
    "        if depth != prev_depth:\n",
    "            depth_counts.append(0)\n",
    "        depth_counts[depth - 1] += 1\n",
    "        prev_depth = depth\n",
    "    \n",
    "    # Create the attention mask\n",
    "    attn_mask = torch.eye(prompt_length, prompt_length)\n",
    "    attn_mask[:, 0] = 1\n",
    "    start = 0\n",
    "    for i in range(len(depth_counts)):\n",
    "        for j in range(depth_counts[i]):\n",
    "            cur_path = paths[start + j]\n",
    "            # retrieve ancestor position\n",
    "            if len(cur_path) == 1:\n",
    "                continue\n",
    "            ancestor_idx = []\n",
    "            for c in range(len(cur_path) - 1):\n",
    "                ancestor_idx.append(paths.index(cur_path[:c+1]) + 1)\n",
    "            attn_mask[start+j+1, ancestor_idx] = 1\n",
    "        start += depth_counts[i]\n",
    "    # print the attention mask row by row\n",
    "    S = ['r ']\n",
    "    counter = {'A': 0, 'B': 0, 'C': 0}\n",
    "    for path in paths:\n",
    "        if path[-1] == -1: \n",
    "            S.append('A' + str(counter['A']))\n",
    "            counter['A'] += 1\n",
    "        elif path[-1] == -2: \n",
    "            S.append('B' + str(counter['B']))\n",
    "            counter['B'] += 1\n",
    "        elif path[-1] == -3:\n",
    "            S.append('C' + str(counter['C']))\n",
    "            counter['C'] += 1\n",
    "        else: \n",
    "            if str(path[-1]+1) not in counter:\n",
    "                counter[str(path[-1]+1)] = 0\n",
    "            S.append(str(counter[str(path[-1]+1)]) + str(path[-1]+1))\n",
    "            counter[str(path[-1]+1)] += 1\n",
    "    print(' ', S)\n",
    "    # for i in range(prompt_length):\n",
    "    #     s = []\n",
    "    #     for n in attn_mask[i].cpu().numpy():\n",
    "    #         s.append(str(int(n)))\n",
    "    #     print(S[i], s)\n",
    "\n",
    "    # Generate tree indices\n",
    "    tree_indices = torch.zeros(prompt_length, dtype=torch.long)\n",
    "    tree_indices[0] = 0\n",
    "    start = 0\n",
    "    for i in range(len(depth_counts)):\n",
    "        for j in range(depth_counts[i]):\n",
    "            cur_path = paths[start + j]\n",
    "            if cur_path[-1] < 0:\n",
    "                tree_indices[start + j + 1] = -(num_special_tokens+cur_path[-1]+1)\n",
    "            else:\n",
    "                tree_indices[start + j + 1] = cur_path[-1] + TOPK * i + 1\n",
    "        start += depth_counts[i]\n",
    "    print(' ', [str(x) for x in tree_indices.cpu().numpy()])\n",
    "    print(' ', [str(x) for x in tree_indices[normal_token_indices].cpu().numpy()])\n",
    "    \n",
    "    # Generate position ids \n",
    "    position_ids = torch.arange(prompt_length, dtype=torch.long)\n",
    "    start = 0 \n",
    "    for i in range(len(depth_counts)):\n",
    "        position_ids[start+1:start+depth_counts[i]+1] = i + 1 \n",
    "        start += depth_counts[i]\n",
    "    \n",
    "    retrieve_indices_nest = []\n",
    "    retrieve_paths = []\n",
    "    for i in range(len(original_paths)):\n",
    "        cur_choice = original_paths[-i-1]\n",
    "        retrieve_indices = []\n",
    "        if cur_choice in retrieve_paths:\n",
    "            continue\n",
    "        else:\n",
    "            for c in range(len(cur_choice)):\n",
    "                retrieve_indices.append(original_paths.index(cur_choice[:c+1]))\n",
    "                retrieve_paths.append(cur_choice[:c+1])\n",
    "        retrieve_indices_nest.append(retrieve_indices)\n",
    "    max_length = max([len(x) for x in retrieve_indices_nest])\n",
    "    retrieve_indices = [pad_path(path, max_length) for path in retrieve_indices_nest]\n",
    "    retrieve_indices = torch.tensor(retrieve_indices, dtype=torch.long)\n",
    "    retrieve_indices = retrieve_indices + 1\n",
    "    retrieve_indices = torch.cat([torch.zeros((retrieve_indices.shape[0], 1), dtype=torch.long), retrieve_indices], dim=1)\n",
    "    print(retrieve_indices.cpu().numpy())\n",
    "    Sn = np.array(S)\n",
    "    print(Sn[normal_token_indices])\n",
    "    print(Sn[normal_token_indices][retrieve_indices.cpu().numpy()])\n",
    "    # for indices in special_token_indices:\n",
    "    #     print(Sn[indices])\n",
    "    #     print(Sn[indices][retrieve_indices.cpu().numpy()])\n",
    "\n",
    "paths = [[[0], 0.17027000000000003, 1], \n",
    "        [[0, 0], 0.09115999999999999, 1], \n",
    "        [[6], 0.01658, 1]]\n",
    "generate_buffers(paths)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "\n",
    "@contextmanager\n",
    "def timed(wall_times, key):\n",
    "    torch.cuda.synchronize()\n",
    "    start = time.time()\n",
    "    yield\n",
    "    torch.cuda.synchronize()\n",
    "    end = time.time()\n",
    "    elapsed_time = end - start\n",
    "    wall_times[key].append(elapsed_time)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Prompt Decoding"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading checkpoint shards: 100%|██████████| 2/2 [00:00<00:00, 11.18it/s]\n",
      "You are using the default legacy behaviour of the <class 'transformers.models.llama.tokenization_llama.LlamaTokenizer'>. This is expected, and simply means that the `legacy` (previous) behavior will be used so nothing changes for you. If you want to use the new behaviour, set `legacy=False`. This should only be set if you understand what it means, and thoroughly read the reason why this was added as explained in https://github.com/huggingface/transformers/pull/24565\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "<class 'prompt.model.model.PromptDecoder'>\n",
      "trainable params: 12,288 || all params: 6,738,427,904 || trainable%: 0.00018235707460349495\n",
      "None\n"
     ]
    }
   ],
   "source": [
    "model = AutoPromptDecoder.from_pretrained(\n",
    "    \"../test/vicuna-7b-3-1\",\n",
    "    low_cpu_mem_usage=True,\n",
    "    # load_in_4bit=True,\n",
    "    torch_dtype=torch.float16,\n",
    ")\n",
    "model.to(device)\n",
    "print(type(model))\n",
    "print(model.print_trainable_parameters())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "candidate_lists = {1: [([0], 0.48335, 3),\n",
    "     ([1], 0.14342, 3),\n",
    "     ([2], 0.0736, 3),\n",
    "     ([3], 0.0454, 3),\n",
    "     ([4], 0.03087, 3),\n",
    "     ([5], 0.02155, 3),\n",
    "     ([6], 0.01658, 3),\n",
    "     ([7], 0.01323, 3),\n",
    "     ([8], 0.00932, 3),\n",
    "     ([9], 0.00845, 3)],\n",
    " 2: [([0], 0.48335, 3),\n",
    "     ([1], 0.14342, 2),\n",
    "     ([0, 0], 0.12691, 2),\n",
    "     ([2], 0.0736, 2),\n",
    "     ([0, 1], 0.07038, 2),\n",
    "     ([3], 0.0454, 2),\n",
    "     ([0, 2], 0.03922, 2),\n",
    "     ([1, 0], 0.03766, 2),\n",
    "     ([4], 0.03087, 2),\n",
    "     ([0, 3], 0.02784, 2),\n",
    "     ([5], 0.02155, 2),\n",
    "     ([1, 1], 0.02088, 2),\n",
    "     ([0, 4], 0.0207, 2),\n",
    "     ([2, 0], 0.01933, 2),\n",
    "     ([6], 0.01658, 1),\n",
    "     ([0, 5], 0.01561, 1),\n",
    "     ([7], 0.01323, 1),\n",
    "     ([0, 6], 0.01242, 1),\n",
    "     ([3, 0], 0.01192, 1),\n",
    "     ([1, 2], 0.01164, 1),\n",
    "     ([2, 1], 0.01072, 1),\n",
    "     ([0, 7], 0.0104, 1),\n",
    "     ([8], 0.00932, 1),\n",
    "     ([0, 8], 0.0092, 1)],\n",
    " 3: [([0], 0.48335, 3),\n",
    "     ([1], 0.14342, 2),\n",
    "     ([0, 0], 0.12691, 2),\n",
    "     ([2], 0.0736, 2),\n",
    "     ([0, 1], 0.07038, 2),\n",
    "     ([3], 0.0454, 2),\n",
    "     ([0, 2], 0.03922, 2),\n",
    "     ([1, 0], 0.03766, 2),\n",
    "     ([4], 0.03087, 2),\n",
    "     ([0, 3], 0.02784, 2),\n",
    "     ([5], 0.02155, 1),\n",
    "     ([1, 1], 0.02088, 1),\n",
    "     ([0, 0, 0], 0.02083, 1),\n",
    "     ([0, 4], 0.0207, 1),\n",
    "     ([2, 0], 0.01933, 1),\n",
    "     ([6], 0.01658, 1),\n",
    "     ([0, 5], 0.01561, 1),\n",
    "     ([0, 0, 1], 0.01492, 1),\n",
    "     ([7], 0.01323, 1),\n",
    "     ([0, 6], 0.01242, 1),\n",
    "     ([3, 0], 0.01192, 1),\n",
    "     ([1, 2], 0.01164, 1),\n",
    "     ([0, 1, 0], 0.01155, 1),\n",
    "     ([2, 1], 0.01072, 1),\n",
    "     ([0, 7], 0.0104, 1),\n",
    "     ([0, 0, 2], 0.01027, 1)]}\n",
    "\n",
    "from prompt.inference.dynamic_sparse_trees_3_7b import * \n",
    "candidate_lists = dynamic_sparse_trees_60\n",
    "\n",
    "# candidate_lists = {\n",
    "#     1: [[[0], 0., 3]],\n",
    "#     2: [[[0], 0., 3]],\n",
    "#     3: [[[0], 0., 3]]}\n",
    "\n",
    "# candidate_lists = {\n",
    "#     1: [[[0], 0., 1]], \n",
    "#     2: [[[0], 0.17027000000000003, 1]], \n",
    "#     3: [[[0], 0.17027000000000003, 1], \n",
    "#         [[0, 0], 0.09115999999999999, 1], \n",
    "#         [[6], 0.01658, 1],  \n",
    "#         ]}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Generating buffers\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['\\n']\n"
     ]
    }
   ],
   "source": [
    "wall_times = {'init': [], 'candidates': [], 'forward_pass': [], 'evaluation': [], 'update': []}\n",
    "# prompt = \"A chat between a curious user and an artificial intelligence assistant. The assistant gives helpful, detailed, and polite answers to the user's questions. USER: Hi, could you share a tale about a charming llama that grows Medusa-like hair and starts its own coffee shop? ASSISTANT:\"\n",
    "prompt = \"\\n\\ndef truncate_number(number: float) -> float:\\n    \\\"\\\"\\\" Given a positive floating point number, it can be decomposed into\\n    and integer part (largest integer smaller than given number) and decimals\\n    (leftover part always smaller than 1).\\n\\n    Return the decimal part of the number.\\n    >>> truncate_number(3.5)\\n    0.5\\n    \\\"\\\"\\\"\\n    if number == 0:\\n        return 0\\n    else:\\n        return number // (10 ** len(str(number)))\\n\\ndef main():\\n    # Test cases\\n    test_cases = [\\n        (0.0, 0.0),\\n        (1.0, 1.0),\\n        (3.5, 0.5),\\n        (3.0, 0.3),\\n        (2.5, 0.25),\\n        (1.5, 0.15),\\n        (1.0, 0.1),\\n        (0.5, 0.05),\\n        (0.2, 0.02),\\n        (0.1, 0.01),\\n        (0.01, 0.001),\\n        (-0.01, -0.001),\\n        (-0.1, -0.01),\\n        (-0.2, -0.02),\\n        (-0.5, -0.05),\\n        (-0.75, -0.075),\\n        (-1.0, -0.1),\\n        (-1.5, -0.15),\\n        (-2.0, -0.2),\\n        (-3.0, -0.3),\\n        (-5.0, -0.5),\\n        (-10.0, -1.0),\\n    ]\\n\\n    for num, exp in test_cases:\\n        result = truncate_number(num)\\n        print(f\\\"{num} -> {result} (expected {exp})\\\")\\n\\nif __name__ == \\\"__main__\\\":\\n    main()\\n```\\n\"\n",
    "input_ids = model.tokenizer(prompt, return_tensors=\"pt\").input_ids.to(device)\n",
    "# paths = [[0], [0, 0], [0, 1], [1], [1, 0], [1, 1]]\n",
    "# paths = [[0], [0, 0], [1], [0, 1], [2], [1, 0], [0, 2], [3], [0, 3], [4], [0, 4], [2, 0], [0, 5], [5], [0, 6], [6], [0, 7], [1, 1], [7], [0, 8], [3, 0], [0, 9], [8], [9], [1, 2], [4, 0], [2, 1], [5, 0], [1, 3], [6, 0], [1, 4], [7, 0], [3, 1], [2, 2], [8, 0], [1, 5], [9, 0], [1, 6]]\n",
    "# paths = [[0], [1], [0, 0], [2], [0, 1], [3], [0, 2], [1, 0], [4], [0, 3], [0, 4], [5], [1, 1], [2, 0], [6], [0, 5], [0, 6], [1, 2]]\n",
    "with torch.inference_mode():\n",
    "  if not hasattr(model, \"inference_buffers\"):\n",
    "    print(\"Generating buffers\")\n",
    "    model.generate_dynamic_buffers(candidate_lists)\n",
    "  (past_key_values,\n",
    "      past_key_values_data,\n",
    "      current_length_data,\n",
    "  ) = initialize_past_key_values(model.base_model)\n",
    "  with timed(wall_times, 'init'):\n",
    "    logits, prompt_logits = model.start_inference(input_ids, past_key_values, current_length_data)\n",
    "new_token = 0\n",
    "accept_lengths = []\n",
    "print(model.tokenizer.batch_decode(logits.argmax(-1)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "temperature = 0.0\n",
    "posterior_threshold = 0.09\n",
    "posterior_alpha = 0.3\n",
    "sampling = 'greedy'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "KV length 491\n",
      "accept_id 12 accept_length 1\n",
      "Wall_times {'init': 0.5260498523712158, 'candidates': 0.030628204345703125, 'forward_pass': 0.038069725036621094, 'evaluation': 0.008563518524169922, 'update': 0.0006656646728515625}\n",
      "KV length 493\n",
      "accept_id 10 accept_length 2\n",
      "Wall_times {'init': 0.5260498523712158, 'candidates': 0.0005121231079101562, 'forward_pass': 0.02307915687561035, 'evaluation': 0.00016427040100097656, 'update': 0.0003566741943359375}\n",
      "KV length 496\n",
      "accept_id 0 accept_length 0\n",
      "Wall_times {'init': 0.5260498523712158, 'candidates': 0.00035858154296875, 'forward_pass': 0.022328615188598633, 'evaluation': 0.0001704692840576172, 'update': 0.00032210350036621094}\n",
      "<s> \n",
      "\n",
      "def truncate_number(number: float) -> float:\n",
      "    \"\"\" Given a positive floating point number, it can be decomposed into\n",
      "    and integer part (largest integer smaller than given number) and decimals\n",
      "    (leftover part always smaller than 1).\n",
      "\n",
      "    Return the decimal part of the number.\n",
      "    >>> truncate_number(3.5)\n",
      "    0.5\n",
      "    \"\"\"\n",
      "    if number == 0:\n",
      "        return 0\n",
      "    else:\n",
      "        return number // (10 ** len(str(number)))\n",
      "\n",
      "def main():\n",
      "    # Test cases\n",
      "    test_cases = [\n",
      "        (0.0, 0.0),\n",
      "        (1.0, 1.0),\n",
      "        (3.5, 0.5),\n",
      "        (3.0, 0.3),\n",
      "        (2.5, 0.25),\n",
      "        (1.5, 0.15),\n",
      "        (1.0, 0.1),\n",
      "        (0.5, 0.05),\n",
      "        (0.2, 0.02),\n",
      "        (0.1, 0.01),\n",
      "        (0.01, 0.001),\n",
      "        (-0.01, -0.001),\n",
      "        (-0.1, -0.01),\n",
      "        (-0.2, -0.02),\n",
      "        (-0.5, -0.05),\n",
      "        (-0.75, -0.075),\n",
      "        (-1.0, -0.1),\n",
      "        (-1.5, -0.15),\n",
      "        (-2.0, -0.2),\n",
      "        (-3.0, -0.3),\n",
      "        (-5.0, -0.5),\n",
      "        (-10.0, -1.0),\n",
      "    ]\n",
      "\n",
      "    for num, exp in test_cases:\n",
      "        result = truncate_number(num)\n",
      "        print(f\"{num} -> {result} (expected {exp})\")\n",
      "\n",
      "if __name__ == \"__main__\":\n",
      "    main()\n",
      "```\n",
      "\n",
      "```\n",
      "\n",
      "```</s>\n"
     ]
    }
   ],
   "source": [
    "kv_cache_lengths = []\n",
    "latencies = []\n",
    "for _ in range(512):\n",
    "  with torch.inference_mode():\n",
    "    with timed(wall_times, 'candidates'):\n",
    "      candidates, tree_candidates_embeds = model.generate_candidates(\n",
    "        logits, \n",
    "        prompt_logits, \n",
    "        temperature, \n",
    "        posterior_threshold, \n",
    "        posterior_alpha, \n",
    "        sampling)\n",
    "    print('KV length', past_key_values[0][0].shape[2])\n",
    "    kv_cache_lengths.append(past_key_values[0][0].shape[2])\n",
    "    with timed(wall_times, 'forward_pass'):\n",
    "      logits, all_logits = model.tree_decoding(tree_candidates_embeds, past_key_values, input_ids)\n",
    "    # print('candidates', model.tokenizer.batch_decode(logits.argmax(-1)))\n",
    "    latencies.append(wall_times['forward_pass'][-1])\n",
    "    with timed(wall_times, 'evaluation'):\n",
    "      best_candidate, accept_length = model.evaluate_posterior(\n",
    "        logits, \n",
    "        candidates, \n",
    "        temperature, \n",
    "        posterior_threshold, \n",
    "        posterior_alpha,\n",
    "        sampling)\n",
    "    print('accept_id', best_candidate.cpu().item(), 'accept_length', accept_length.cpu().item())\n",
    "    accept_lengths.append(accept_length.cpu().item()+1)\n",
    "    with timed(wall_times, 'update'):\n",
    "            input_ids, logits, prompt_logits, new_token = model.update_inference_inputs(\n",
    "                    input_ids,\n",
    "                    candidates,\n",
    "                    best_candidate,\n",
    "                    accept_length,\n",
    "                    logits,\n",
    "                    all_logits,\n",
    "                    new_token,\n",
    "                    past_key_values_data,\n",
    "                    current_length_data,\n",
    "            )\n",
    "    # print(torch.cuda.utilization())\n",
    "    # print(\"New token\", new_token.cpu().item())\n",
    "    print('Wall_times', {k: v[-1] for k, v in wall_times.items()})\n",
    "    # print('Average accept length', sum(accept_lengths)/len(accept_lengths))\n",
    "    torch.cuda.empty_cache()\n",
    "    if model.tokenizer.eos_token_id in input_ids[0, :].tolist():\n",
    "      break\n",
    "print(model.tokenizer.decode(input_ids[0], spaces_between_special_tokens=False,))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average 1 forward pass time 0.0405427614847819 s\n",
      "Token generation speed 46.890888391601095 tokens/s\n",
      "Average accept length 1.3333333333333333\n",
      "Percentage of time spent on candidates generation 0.027695741854220975 average time 0.0011812845865885417\n",
      "Percentage of time spent on forward pass 0.9505430519351058 average time 0.0405427614847819\n",
      "Percentage of time spent on evaluation 0.006454389786263553 average time 0.0002752939860026042\n",
      "Percentage of time spent on update 0.015306816424409668 average time 0.0006528695424397787\n"
     ]
    },
    {
     "data": {
      "image/png": "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aP1Vc/o4b948XVtHR0dh3LhxQnJyst7xjPkcCoIgpKenCwCEH374od6xDO0c3r7lu+5yu08zZswQRowYUW8fX19fQaFQCN26dRM+++yzese90/fsfmpu/0aMGHHH9oJwa1oDZ2dnQaFQCJ07dxZCQkKE8+fP39+O/UVz+7hy5Uqhe/fugkqlEmxtbYWRI0cKP/30U73jGus5FIRbt7R36NBB2LhxY4PHNKRz2FDfAOj9Xhnie6Hkj+KJiIiI6A8cg0RERERUBwMSERERUR0MSERERER1MCARERER1cGARERERFQHAxIRERFRHQxIRERERHUwIBERERHVwYBERO3Cs88+C4lEAolEAjMzM3Tt2hWvvvoqKisrxS6NiAyQXOwCiIjulzFjxuCzzz5DTU0NkpKSMGPGDEgkEqxcuVLs0ojIwPAKEhG1G0qlEk5OTnB1dUVwcDACAwOxb98+AICHhwdWr16t197X1xdLly7VfS2RSPB///d/mDRpEszNzdGzZ098/fXX97EHRHS/MCARUbuUlpaGgwcPQqFQNGu/ZcuWYcqUKTh+/DjGjRuHp556CgUFBW1UJRGJhQGJiNqNPXv2oGPHjlCpVPDy8kJubi4WLFjQrGM8++yzCA0NRY8ePfDOO++gtLQUR44caaOKiUgsHINERO3GqFGj8PHHH6OsrAwffvgh5HI5Jk+e3KxjeHt76/5uYWEBKysr5ObmtnapRCQyXkEionbDwsICPXr0gI+PDz799FMcPnwYmzZtAgBIpVIIgqDXvqampt4xzMzM9L6WSCTQarVtVzQRiYIBiYjaJalUitdffx2LFi1CRUUFOnXqhKysLN324uJiXLp0ScQKiUhMDEhE1G49+eSTkMlkWL9+Pf72t7/hv//9Lw4cOIATJ05gxowZkMlkYpdIRCLhGCQiarfkcjnCwsLw7rvv4ty5c7h06RIeffRRqNVqLF++nFeQiNoxiVD3Q3ciIiKido4fsRERERHVwYBEREREVAcDEhEREVEdDEhEREREdTAgEREREdXBgERERERUBwMSERERUR0MSERERER1MCARERER1cGARERERFQHAxIRERFRHQxIRERERHX8P0RlrGYzqgbQAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "candidate_time = sum(wall_times['candidates'])\n",
    "forward_time = sum(wall_times['forward_pass'])\n",
    "evaluation_time = sum(wall_times['evaluation'])\n",
    "update_time = sum(wall_times['update'])\n",
    "total_time = candidate_time + forward_time + evaluation_time + update_time\n",
    "\n",
    "print('Average 1 forward pass time', {k: sum(v)/len(v) for k, v in wall_times.items()}['forward_pass'], 's')\n",
    "print('Token generation speed', new_token.cpu().item()/total_time, 'tokens/s')\n",
    "print('Average accept length', sum(accept_lengths)/len(accept_lengths))\n",
    "\n",
    "# print percentage of time spent on each step\n",
    "print('Percentage of time spent on candidates generation', candidate_time/total_time, 'average time', candidate_time/len(latencies))\n",
    "print('Percentage of time spent on forward pass', forward_time/total_time, 'average time', forward_time/len(latencies))\n",
    "print('Percentage of time spent on evaluation', evaluation_time/total_time, 'average time', evaluation_time/len(latencies))\n",
    "print('Percentage of time spent on update', update_time/total_time, 'average time', update_time/len(latencies))\n",
    "\n",
    "# plot latencies against runs\n",
    "import matplotlib.pyplot as plt\n",
    "plt.plot(latencies)\n",
    "plt.xlabel('Run')\n",
    "plt.ylabel('Latency (s)')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Vicuna (with the same kernel as prompt decoding)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading checkpoint shards: 100%|██████████| 2/2 [00:00<00:00,  7.82it/s]\n",
      "You are using the default legacy behaviour of the <class 'transformers.models.llama.tokenization_llama.LlamaTokenizer'>. This is expected, and simply means that the `legacy` (previous) behavior will be used so nothing changes for you. If you want to use the new behaviour, set `legacy=False`. This should only be set if you understand what it means, and thoroughly read the reason why this was added as explained in https://github.com/huggingface/transformers/pull/24565\n"
     ]
    }
   ],
   "source": [
    "base_model = CustomLlamaForCausalLM.from_pretrained(\n",
    "            \"lmsys/vicuna-7b-v1.3\",\n",
    "            low_cpu_mem_usage=True,\n",
    "            torch_dtype=torch.float16,\n",
    "            # load_in_4bit=True\n",
    "        )\n",
    "base_model.to(device)\n",
    "tokenizer = transformers.AutoTokenizer.from_pretrained(\"lmsys/vicuna-7b-v1.3\")\n",
    "# prompt = \"A chat between a curious user and an artificial intelligence assistant. The assistant gives helpful, detailed, and polite answers to the user's questions. USER: Hi, could you share a tale about a charming llama that grows Medusa-like hair and starts its own coffee shop? ASSISTANT:\"\n",
    "prompt = \"\\n\\ndef truncate_number(number: float) -> float:\\n    \\\"\\\"\\\" Given a positive floating point number, it can be decomposed into\\n    and integer part (largest integer smaller than given number) and decimals\\n    (leftover part always smaller than 1).\\n\\n    Return the decimal part of the number.\\n    >>> truncate_number(3.5)\\n    0.5\\n    \\\"\\\"\\\"\\n    if number == 0:\\n        return 0\\n    else:\\n        return number // (10 ** len(str(number)))\\n\\ndef main():\\n    # Test cases\\n    test_cases = [\\n        (0.0, 0.0),\\n        (1.0, 1.0),\\n        (3.5, 0.5),\\n        (3.0, 0.3),\\n        (2.5, 0.25),\\n        (1.5, 0.15),\\n        (1.0, 0.1),\\n        (0.5, 0.05),\\n        (0.2, 0.02),\\n        (0.1, 0.01),\\n        (0.01, 0.001),\\n        (-0.01, -0.001),\\n        (-0.1, -0.01),\\n        (-0.2, -0.02),\\n        (-0.5, -0.05),\\n        (-0.75, -0.075),\\n        (-1.0, -0.1),\\n        (-1.5, -0.15),\\n        (-2.0, -0.2),\\n        (-3.0, -0.3),\\n        (-5.0, -0.5),\\n        (-10.0, -1.0),\\n    ]\\n\\n    for num, exp in test_cases:\\n        result = truncate_number(num)\\n        print(f\\\"{num} -> {result} (expected {exp})\\\")\\n\\nif __name__ == \\\"__main__\\\":\\n    main()\\n```\\n\"\n",
    "input_ids = tokenizer(prompt, return_tensors=\"pt\").input_ids.to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "wall_times_baseline = {'init': [], 'forward_pass': []}\n",
    "with torch.inference_mode():\n",
    "  with timed(wall_times_baseline, 'init'):\n",
    "    (past_key_values,\n",
    "        past_key_values_data,\n",
    "        current_length_data,\n",
    "    ) = initialize_past_key_values(base_model)\n",
    "    input_len = input_ids.shape[1]\n",
    "    outputs = base_model(input_ids, past_key_values=past_key_values)\n",
    "    new_token = 0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "<s> \n",
      "\n",
      "def truncate_number(number: float) -> float:\n",
      "    \"\"\" Given a positive floating point number, it can be decomposed into\n",
      "    and integer part (largest integer smaller than given number) and decimals\n",
      "    (leftover part always smaller than 1).\n",
      "\n",
      "    Return the decimal part of the number.\n",
      "    >>> truncate_number(3.5)\n",
      "    0.5\n",
      "    \"\"\"\n",
      "    if number == 0:\n",
      "        return 0\n",
      "    else:\n",
      "        return number // (10 ** len(str(number)))\n",
      "\n",
      "def main():\n",
      "    # Test cases\n",
      "    test_cases = [\n",
      "        (0.0, 0.0),\n",
      "        (1.0, 1.0),\n",
      "        (3.5, 0.5),\n",
      "        (3.0, 0.3),\n",
      "        (2.5, 0.25),\n",
      "        (1.5, 0.15),\n",
      "        (1.0, 0.1),\n",
      "        (0.5, 0.05),\n",
      "        (0.2, 0.02),\n",
      "        (0.1, 0.01),\n",
      "        (0.01, 0.001),\n",
      "        (-0.01, -0.001),\n",
      "        (-0.1, -0.01),\n",
      "        (-0.2, -0.02),\n",
      "        (-0.5, -0.05),\n",
      "        (-0.75, -0.075),\n",
      "        (-1.0, -0.1),\n",
      "        (-1.5, -0.15),\n",
      "        (-2.0, -0.2),\n",
      "        (-3.0, -0.3),\n",
      "        (-5.0, -0.5),\n",
      "        (-10.0, -1.0),\n",
      "    ]\n",
      "\n",
      "    for num, exp in test_cases:\n",
      "        result = truncate_number(num)\n",
      "        print(f\"{num} -> {result} (expected {exp})\")\n",
      "\n",
      "if __name__ == \"__main__\":\n",
      "    main()\n",
      "```\n",
      "\n",
      "```\n",
      "\n",
      "```</s>\n"
     ]
    }
   ],
   "source": [
    "with torch.inference_mode():\n",
    "  while new_token < 512:\n",
    "    with timed(wall_times_baseline, 'forward_pass'):\n",
    "      input_id = outputs.logits[:, -1:].argmax(dim=-1)\n",
    "      outputs = base_model(input_id, use_cache=True, past_key_values=past_key_values)\n",
    "      input_ids = torch.cat([input_ids, input_id], dim=-1)\n",
    "      new_token += 1\n",
    "      torch.cuda.empty_cache()\n",
    "      if tokenizer.eos_token_id in input_ids[0, :].tolist():\n",
    "        break\n",
    "print(tokenizer.decode(input_ids[0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average 1 forward pass time 0.03011168718338013 s\n",
      "Token generation speed 33.20969675030169 tokens/s\n"
     ]
    }
   ],
   "source": [
    "print('Average 1 forward pass time', {k: sum(v)/len(v) for k, v in wall_times_baseline.items()}['forward_pass'], 's')\n",
    "print(\"Token generation speed\", new_token/({k: sum(v) for k, v in wall_times_baseline.items()}['forward_pass']), 'tokens/s')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Vanilla Vicuna (with the latest kernel)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading checkpoint shards: 100%|██████████| 2/2 [00:00<00:00,  9.61it/s]\n",
      "You are using the default legacy behaviour of the <class 'transformers.models.llama.tokenization_llama.LlamaTokenizer'>. This is expected, and simply means that the `legacy` (previous) behavior will be used so nothing changes for you. If you want to use the new behaviour, set `legacy=False`. This should only be set if you understand what it means, and thoroughly read the reason why this was added as explained in https://github.com/huggingface/transformers/pull/24565\n"
     ]
    }
   ],
   "source": [
    "base_model = LlamaForCausalLM.from_pretrained(\n",
    "            \"lmsys/vicuna-7b-v1.3\",\n",
    "            low_cpu_mem_usage=True,\n",
    "            torch_dtype=torch.float16,\n",
    "            # load_in_4bit=True\n",
    "        )\n",
    "base_model.to(device)\n",
    "tokenizer = transformers.AutoTokenizer.from_pretrained(\"lmsys/vicuna-7b-v1.3\")\n",
    "# prompt = \"A chat between a curious user and an artificial intelligence assistant. The assistant gives helpful, detailed, and polite answers to the user's questions. USER: Hi, could you share a tale about a charming llama that grows Medusa-like hair and starts its own coffee shop? ASSISTANT:\"\n",
    "prompt = \"\\n\\ndef truncate_number(number: float) -> float:\\n    \\\"\\\"\\\" Given a positive floating point number, it can be decomposed into\\n    and integer part (largest integer smaller than given number) and decimals\\n    (leftover part always smaller than 1).\\n\\n    Return the decimal part of the number.\\n    >>> truncate_number(3.5)\\n    0.5\\n    \\\"\\\"\\\"\\n    if number == 0:\\n        return 0\\n    else:\\n        return number // (10 ** len(str(number)))\\n\\ndef main():\\n    # Test cases\\n    test_cases = [\\n        (0.0, 0.0),\\n        (1.0, 1.0),\\n        (3.5, 0.5),\\n        (3.0, 0.3),\\n        (2.5, 0.25),\\n        (1.5, 0.15),\\n        (1.0, 0.1),\\n        (0.5, 0.05),\\n        (0.2, 0.02),\\n        (0.1, 0.01),\\n        (0.01, 0.001),\\n        (-0.01, -0.001),\\n        (-0.1, -0.01),\\n        (-0.2, -0.02),\\n        (-0.5, -0.05),\\n        (-0.75, -0.075),\\n        (-1.0, -0.1),\\n        (-1.5, -0.15),\\n        (-2.0, -0.2),\\n        (-3.0, -0.3),\\n        (-5.0, -0.5),\\n        (-10.0, -1.0),\\n    ]\\n\\n    for num, exp in test_cases:\\n        result = truncate_number(num)\\n        print(f\\\"{num} -> {result} (expected {exp})\\\")\\n\\nif __name__ == \\\"__main__\\\":\\n    main()\\n```\\n\"\n",
    "input_ids = tokenizer(prompt, return_tensors=\"pt\").input_ids.to(device)\n",
    "wall_times_baseline = {'init': [], 'forward_pass': []}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "wall_times_baseline = {'init': [], 'forward_pass': []}\n",
    "with torch.inference_mode():    \n",
    "  with timed(wall_times_baseline, 'init'):\n",
    "    outputs = base_model(input_ids, use_cache=True, past_key_values=None)\n",
    "    new_token = 0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "<s> \n",
      "\n",
      "def truncate_number(number: float) -> float:\n",
      "    \"\"\" Given a positive floating point number, it can be decomposed into\n",
      "    and integer part (largest integer smaller than given number) and decimals\n",
      "    (leftover part always smaller than 1).\n",
      "\n",
      "    Return the decimal part of the number.\n",
      "    >>> truncate_number(3.5)\n",
      "    0.5\n",
      "    \"\"\"\n",
      "    if number == 0:\n",
      "        return 0\n",
      "    else:\n",
      "        return number // (10 ** len(str(number)))\n",
      "\n",
      "def main():\n",
      "    # Test cases\n",
      "    test_cases = [\n",
      "        (0.0, 0.0),\n",
      "        (1.0, 1.0),\n",
      "        (3.5, 0.5),\n",
      "        (3.0, 0.3),\n",
      "        (2.5, 0.25),\n",
      "        (1.5, 0.15),\n",
      "        (1.0, 0.1),\n",
      "        (0.5, 0.05),\n",
      "        (0.2, 0.02),\n",
      "        (0.1, 0.01),\n",
      "        (0.01, 0.001),\n",
      "        (-0.01, -0.001),\n",
      "        (-0.1, -0.01),\n",
      "        (-0.2, -0.02),\n",
      "        (-0.5, -0.05),\n",
      "        (-0.75, -0.075),\n",
      "        (-1.0, -0.1),\n",
      "        (-1.5, -0.15),\n",
      "        (-2.0, -0.2),\n",
      "        (-3.0, -0.3),\n",
      "        (-5.0, -0.5),\n",
      "        (-10.0, -1.0),\n",
      "    ]\n",
      "\n",
      "    for num, exp in test_cases:\n",
      "        result = truncate_number(num)\n",
      "        print(f\"{num} -> {result} (expected {exp})\")\n",
      "\n",
      "if __name__ == \"__main__\":\n",
      "    main()\n",
      "```\n",
      "\n",
      "```\n",
      "\n",
      "```</s><s> package com.example.coolweather.gson;\n",
      "\n",
      "import com.google.gson.annotations.SerializedName;\n",
      "\n",
      "import java.util.List;\n",
      "\n",
      "/**\n",
      " * Created by Administrator on 2017/5/20.\n",
      " */\n",
      "public class Weather {\n",
      "    public String status;\n",
      "    public Basic basic;\n",
      "    public Now now;\n",
      "    public List<AQ\n"
     ]
    }
   ],
   "source": [
    "with torch.inference_mode():\n",
    "  while new_token < 100:\n",
    "    with timed(wall_times_baseline, 'forward_pass'):\n",
    "      input_id = outputs.logits[:, -1:].argmax(dim=-1)\n",
    "      outputs = base_model(input_id, use_cache=True, past_key_values=outputs.past_key_values)\n",
    "      input_ids = torch.cat([input_ids, input_id], dim=-1)\n",
    "      new_token += 1\n",
    "      if tokenizer.eos_token_id in input_ids[0, :].tolist():\n",
    "        break\n",
    "print(tokenizer.decode(input_ids[0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average 1 forward pass time 0.02508216381072998 s\n",
      "Token generation speed 39.86896854458014 tokens/s\n"
     ]
    }
   ],
   "source": [
    "print('Average 1 forward pass time', {k: sum(v)/len(v) for k, v in wall_times_baseline.items()}['forward_pass'], 's')\n",
    "print(\"Token generation speed\", new_token/({k: sum(v) for k, v in wall_times_baseline.items()}['forward_pass']), 'tokens/s')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Misc"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "612\n"
     ]
    }
   ],
   "source": [
    "import json\n",
    "from fastchat.model.model_adapter import get_conversation_template\n",
    "data = json.load(open(\"data/alpaca_eval.json\"))\n",
    "tokenizer = transformers.AutoTokenizer.from_pretrained(\"lmsys/vicuna-7b-v1.3\")\n",
    "max_length = 0\n",
    "for sample in data:\n",
    "  conv = get_conversation_template(\"vicuna\")\n",
    "  conv.messages = []\n",
    "  conv.append_message(conv.roles[0], sample[\"instruction\"])\n",
    "  conv.append_message(conv.roles[1], \"\")\n",
    "  prompt = conv.get_prompt()\n",
    "  ids = tokenizer(prompt, return_tensors=\"pt\").input_ids.to(device)\n",
    "  if ids.shape[1] > max_length:\n",
    "    max_length = ids.shape[1]\n",
    "print(max_length)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Max length 1021\n",
      "Min length 39\n",
      "Average length 689.6422045086924\n"
     ]
    }
   ],
   "source": [
    "dataset_1024 = torch.load(\"data/1024/ShareGPT_training_dataset_2.pt\")\n",
    "lengths = []\n",
    "for sample in dataset_1024:\n",
    "  input_ids = sample[\"input_ids\"]\n",
    "  non_padded_ids = input_ids[input_ids != tokenizer.pad_token_id]\n",
    "  lengths.append(non_padded_ids.shape[0])\n",
    "print(\"Max length\", max(lengths))\n",
    "print(\"Min length\", min(lengths))\n",
    "print(\"Average length\", sum(lengths)/len(lengths))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Max length 2045\n",
      "Min length 39\n",
      "Average length 1174.1331186336943\n"
     ]
    }
   ],
   "source": [
    "dataset_2048 = torch.load(\"data/ShareGPT_training_dataset_2_finetune.pt\")\n",
    "lengths = []\n",
    "for sample in dataset_2048:\n",
    "  input_ids = sample[\"input_ids\"]\n",
    "  non_padded_ids = input_ids[input_ids != tokenizer.pad_token_id]\n",
    "  lengths.append(non_padded_ids.shape[0])\n",
    "print(\"Max length\", max(lengths))\n",
    "print(\"Min length\", min(lengths))\n",
    "print(\"Average length\", sum(lengths)/len(lengths))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Token indices sequence length is longer than the specified maximum sequence length for this model (3574 > 2048). Running this sequence through the model will result in indexing errors\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Max length 2048\n",
      "Min length 42\n",
      "Average length 1356.0956530609271\n"
     ]
    }
   ],
   "source": [
    "from utils import preprocess\n",
    "raw_data = json.load(open(\"data/ShareGPT_V4.3_unfiltered_cleaned_split.json\"))\n",
    "sources = [example[\"conversations\"] for example in raw_data]\n",
    "tokenizer = transformers.AutoTokenizer.from_pretrained(\"lmsys/vicuna-7b-v1.3\")\n",
    "data_dict = preprocess(sources, tokenizer)\n",
    "lengths = []\n",
    "for sample in data_dict[\"input_ids\"]:\n",
    "  non_padded_ids = sample[sample != tokenizer.pad_token_id]\n",
    "  lengths.append(non_padded_ids.shape[0])\n",
    "print(\"Max length\", max(lengths))\n",
    "print(\"Min length\", min(lengths))\n",
    "print(\"Average length\", sum(lengths)/len(lengths))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "venv",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
