gpt2.py 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. #!/usr/bin/env python3
  2. from typing import Optional, Union
  3. import argparse
  4. import numpy as np
  5. import tiktoken
  6. from tinygrad import Tensor, TinyJit, Device, GlobalCounters, Variable
  7. from tinygrad.helpers import Timing, DEBUG, JIT, getenv, fetch, colored, trange
  8. from tinygrad.nn import Embedding, Linear, LayerNorm
  9. from tinygrad.nn.state import torch_load, load_state_dict, get_state_dict
  10. MAX_CONTEXT = getenv("MAX_CONTEXT", 128)
  11. HALF = getenv("HALF")
  12. class Attention:
  13. def __init__(self, dim, n_heads):
  14. self.c_attn = Linear(dim, 3*dim, bias=True)
  15. self.c_proj = Linear(dim, dim, bias=True)
  16. self.n_heads = n_heads
  17. self.dim = dim
  18. self.head_dim = dim // n_heads
  19. def __call__(self, x:Tensor, start_pos:Variable, mask:Optional[Tensor]) -> Tensor:
  20. if mask is not None or start_pos.val == 0:
  21. # no symbolic shape qkv when consuming prompts
  22. start_pos = start_pos.val
  23. if HALF: x = x.half()
  24. xqkv = self.c_attn(x)
  25. xq, xk, xv = [xqkv.shrink((None, None, (i*self.dim, (i+1)*self.dim))).reshape(None, None, self.n_heads, self.head_dim) for i in range(3)]
  26. bsz, seqlen, _, _ = xq.shape
  27. # create kv cache
  28. if not hasattr(self, "cache_kv"):
  29. self.cache_kv = Tensor.zeros(2, bsz, MAX_CONTEXT, self.n_heads, self.head_dim, dtype=x.dtype).contiguous().realize()
  30. # update the cache
  31. self.cache_kv.shrink((None, None,(start_pos,start_pos+seqlen),None,None)).assign(Tensor.stack(xk, xv)).realize()
  32. if start_pos > 0:
  33. keys = self.cache_kv[0].shrink((None, (0, start_pos+seqlen), None, None))
  34. values = self.cache_kv[1].shrink((None, (0, start_pos+seqlen), None, None))
  35. else:
  36. keys = xk
  37. values = xv
  38. xq, keys, values = xq.transpose(1, 2), keys.transpose(1, 2), values.transpose(1, 2)
  39. return self.c_proj(xq.scaled_dot_product_attention(keys, values, mask).transpose(1, 2).reshape(bsz, seqlen, self.dim))
  40. class FeedForward:
  41. def __init__(self, dim, hidden_dim):
  42. self.c_fc = Linear(dim, hidden_dim, bias=True)
  43. self.c_proj = Linear(hidden_dim, dim, bias=True)
  44. def __call__(self, x:Tensor) -> Tensor:
  45. return self.c_proj(self.c_fc(x).gelu())
  46. class TransformerBlock:
  47. def __init__(self, dim, n_heads, norm_eps):
  48. self.attn = Attention(dim, n_heads)
  49. self.mlp = FeedForward(dim, 4*dim)
  50. self.ln_1 = LayerNorm(dim, norm_eps)
  51. self.ln_2 = LayerNorm(dim, norm_eps)
  52. def __call__(self, x:Tensor, start_pos:Variable, mask:Optional[Tensor]):
  53. h = x + self.attn(self.ln_1(x), start_pos, mask).float()
  54. return (h + self.mlp(self.ln_2(h)))
  55. class Transformer:
  56. def __init__(self, dim, n_heads, n_layers, norm_eps, vocab_size, max_seq_len=1024):
  57. self.vocab_size = vocab_size
  58. self.wte = Embedding(vocab_size, dim)
  59. self.wpe = Embedding(max_seq_len, dim)
  60. self.h = [TransformerBlock(dim, n_heads, norm_eps) for _ in range(n_layers)]
  61. self.ln_f = LayerNorm(dim, norm_eps)
  62. self.lm_head = Linear(dim, vocab_size, bias=False)
  63. self.forward_jit = TinyJit(self.forward)
  64. def forward(self, tokens:Union[Tensor,Variable], start_pos:Variable, temperature:float=0.0):
  65. if not hasattr(self, 'allpos'): self.allpos = Tensor.arange(0, MAX_CONTEXT).reshape(1, -1).realize()
  66. if isinstance(tokens, Variable):
  67. seqlen = 1
  68. tok_emb = self.wte.weight.shrink(((tokens, tokens+1), None))
  69. else:
  70. seqlen = tokens.shape[1]
  71. tok_emb = self.wte(tokens)
  72. pos_emb = self.wpe(self.allpos.shrink((None, (start_pos, start_pos+seqlen))))
  73. h = tok_emb + pos_emb
  74. if HALF: h = h.half()
  75. mask = Tensor.full((1, 1, seqlen, start_pos.val+seqlen), float("-inf"), dtype=h.dtype).triu(start_pos.val+1) if seqlen > 1 else None
  76. for hi in self.h: h = hi(h, start_pos, mask)
  77. logits = self.lm_head(self.ln_f(h))
  78. if logits.shape[1] == 0:
  79. # special case for empty prompt
  80. logits = Tensor.ones((logits.shape[0], self.vocab_size), dtype=logits.dtype, device=logits.device)
  81. else:
  82. logits = logits[:, -1, :]
  83. if temperature < 1e-6:
  84. ret = logits.argmax(-1)
  85. else:
  86. ret = (logits / temperature).softmax().multinomial()
  87. return ret.flatten().realize()
  88. def __call__(self, tokens:Tensor, start_pos:Variable, temperature:float=0.0) -> Tensor:
  89. forward = (self.forward_jit if JIT and (isinstance(tokens, Variable) or tokens.shape[1] == 1) else self.forward)
  90. return forward(tokens, start_pos, temperature)
  91. VOCAB_SIZE = 50257
  92. MODEL_PARAMS = {
  93. 'gpt2': dict(n_layers=12, n_heads=12, dim=768, norm_eps=1e-5, vocab_size=VOCAB_SIZE), # 124M params
  94. 'gpt2-medium': dict(n_layers=24, n_heads=16, dim=1024, norm_eps=1e-5, vocab_size=VOCAB_SIZE), # 350M params
  95. 'gpt2-large': dict(n_layers=36, n_heads=20, dim=1280, norm_eps=1e-5, vocab_size=VOCAB_SIZE), # 774M params
  96. 'gpt2-xl': dict(n_layers=48, n_heads=25, dim=1600, norm_eps=1e-5, vocab_size=VOCAB_SIZE), # 1558M params
  97. }
  98. class GPT2:
  99. @staticmethod
  100. def build(model_size="gpt2"):
  101. tokenizer = tiktoken.get_encoding("gpt2")
  102. model = Transformer(**MODEL_PARAMS[model_size])
  103. weights = torch_load(fetch(f'https://huggingface.co/{model_size}/resolve/main/pytorch_model.bin'))
  104. # special treatment for the Conv1D weights we need to transpose
  105. transposed = ('attn.c_attn.weight', 'attn.c_proj.weight', 'mlp.c_fc.weight', 'mlp.c_proj.weight')
  106. for k in weights:
  107. if k.endswith(transposed):
  108. weights[k] = weights[k].T
  109. # lm head and wte are tied
  110. weights['lm_head.weight'] = weights['wte.weight']
  111. load_state_dict(model, weights)
  112. if HALF:
  113. for l in get_state_dict(model).values():
  114. l.replace(l.half().realize())
  115. return GPT2(model, tokenizer)
  116. def __init__(self, model, tokenizer):
  117. self.model = model
  118. self.tokenizer = tokenizer
  119. def generate(self, prompt:str, max_length:int, temperature:float, timing:bool=False, batch_size:int=1):
  120. prompt_tokens = self.tokenizer.encode(prompt, allowed_special={"<|endoftext|>"})
  121. toks = [prompt_tokens[:] for _ in range(batch_size)]
  122. start_pos = 0
  123. for _ in trange(max_length, disable=(timing==True)):
  124. GlobalCounters.reset()
  125. if timing: print("")
  126. st = GlobalCounters.time_sum_s
  127. with Timing("ran model in ", on_exit=(lambda et: (f", {(GlobalCounters.time_sum_s-st)*1e3:.2f} ms on GPU" if DEBUG>=2 else "")+
  128. f", {GlobalCounters.global_ops*1e-9:.2f} GOPS, {GlobalCounters.global_mem*1e-9:.2f} GB"+
  129. (f", {GlobalCounters.global_mem*1e-9/(GlobalCounters.time_sum_s-st):.2f} GB/s" if DEBUG>=2 else "")) if DEBUG else None, enabled=timing):
  130. if batch_size == 1 and len(toks[0][start_pos:]) == 1:
  131. tokens = Variable("tokens", 0, VOCAB_SIZE).bind(toks[0][start_pos])
  132. else:
  133. tokens = Tensor([x[start_pos:] for x in toks])
  134. tok = self.model(tokens, Variable("start_pos", 1 if start_pos else 0, MAX_CONTEXT).bind(start_pos), temperature).numpy().tolist()
  135. start_pos = len(toks[0])
  136. for i,t in enumerate(tok): toks[i].append(t)
  137. return [self.tokenizer.decode(x) for x in toks]
  138. # **** main code ****
  139. if __name__ == "__main__":
  140. Tensor.no_grad = True
  141. print(f"using {Device.DEFAULT} backend")
  142. default_prompt = "What is the answer to life, the universe, and everything?"
  143. parser = argparse.ArgumentParser(description='Run GPT2 in tinygrad', formatter_class=argparse.ArgumentDefaultsHelpFormatter)
  144. parser.add_argument('--prompt', type=str, default=default_prompt, help="Phrase to start with")
  145. parser.add_argument('--count', type=int, default=100, help="Max number of tokens to generate")
  146. parser.add_argument('--temperature', type=float, default=0.8, help="Temperature in the softmax")
  147. parser.add_argument('--model_size', type=str, default="gpt2-medium", help="Size of model to use [gpt2, gpt2-medium, gpt2-large, gpt2-xl]")
  148. parser.add_argument('--timing', action='store_true', help="Print timing per token")
  149. parser.add_argument('--seed', type=int, help="Set the random seed")
  150. parser.add_argument('--batch_size', type=int, default=1, help="Set the input batch size")
  151. parser.add_argument('--benchmark', type=int, default=-1, help="Benchmark GPT with the given number of tokens")
  152. parser.add_argument('--noshow', action='store_true', help="Don't show the output")
  153. args = parser.parse_args()
  154. if args.seed is not None:
  155. Tensor.manual_seed(args.seed)
  156. np.random.seed(args.seed)
  157. print(f"using {args.model_size}")
  158. gpt2 = GPT2.build(args.model_size)
  159. if args.benchmark != -1:
  160. gpt2.model(Tensor.rand(args.batch_size, args.benchmark), Variable("a", 0, MAX_CONTEXT).bind(0)).realize()
  161. else:
  162. texts = gpt2.generate(args.prompt, args.count, args.temperature, timing=args.timing, batch_size=args.batch_size)
  163. if not args.noshow:
  164. print('Generating text...')
  165. if len(texts) == 1: print(texts[0])
  166. else:
  167. for i,text in enumerate(texts): print(colored(f"Response {i}:", "green"), text)
  168. # validate output!
  169. if args.temperature == 0 and args.model_size == "gpt2-medium" and args.count == 10:
  170. expected = {
  171. default_prompt: "What is the answer to life, the universe, and everything?\n\nThe answer is that we are all one",
  172. "Hello.": "Hello. I'm a little late to the party, but",
  173. }
  174. try:
  175. assert texts[0] == expected[args.prompt]
  176. print(colored("output validated", "green"))
  177. except KeyError:
  178. pass