# Copyright 2023 DAMO Academy and The HuggingFace Inc. team. All rights reserved. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. import argparse import gc import json import math import os import shutil import warnings import torch from transformers import LlamaTokenizer from .configuration_mplug_docowl import MPLUGDocOwlConfig from icecream import ic try: from transformers import LlamaTokenizerFast except ImportError as e: warnings.warn(e) warnings.warn( "The converted tokenizer will be the `slow` tokenizer. To use the fast, update your `tokenizers` library and re-run the tokenizer conversion" ) LlamaTokenizerFast = None """ Sample usage: ``` python3 /pure-mlo-scratch/sfan/model-parallel-trainer/llama2megatron/convert_llama2hf.py \ --input_dir /pure-mlo-scratch/llama/ --model_size 7 --output_dir /pure-mlo-scratch/llama/converted_HF_7B ``` Thereafter, models can be loaded via: ```py from transformers import LlamaForCausalLM, LlamaTokenizer model = LlamaForCausalLM.from_pretrained("/output/path") tokenizer = LlamaTokenizer.from_pretrained("/output/path") ``` Important note: you need to be able to host the whole model in RAM to execute this script (even if the biggest versions come in several checkpoints they each contain a part of each weight of the model, so we need to load them all in RAM). """ llama_s2layer = {7: 32, 13: 40, 30: 60, 65: 80, 70: 80} llama_s2heads = {7: 32, 13: 40, 30: 52, 65: 64, 70: 64} llama_s2dense = {7: 11008, 13: 13824, 30: 17920, 65: 22016, 70: 28672} # should be (2/3)*4*d, but it isn't exaclty that llama_s2hidden = {7: 4096, 13: 5120, 32: 6656, 65: 8192, 70: 8192} def compute_intermediate_size(n): return int(math.ceil(n * 8 / 3) + 255) // 256 * 256 def read_json(path): with open(path, "r") as f: return json.load(f) def write_json(text, path): with open(path, "w") as f: json.dump(text, f) def write_model(model_path, input_base_path, model_size, num_input_shards=1, num_output_shards=2, skip_permute=True, norm_eps=1e-05): # if os.path.exists(model_path): # shutil.rmtree(model_path) os.makedirs(model_path, exist_ok=True) # tmp_model_path = os.path.join(model_path, "tmp") tmp_model_path = model_path os.makedirs(tmp_model_path, exist_ok=True) num_shards = num_input_shards n_layers = llama_s2layer[model_size] n_heads = llama_s2heads[model_size] n_heads_per_shard = n_heads // num_shards n_dense = llama_s2dense[model_size] n_hidden = llama_s2hidden[model_size] hidden_per_head = n_hidden // n_heads base = 10000.0 inv_freq = 1.0 / (base ** (torch.arange(0, hidden_per_head, 2).float() / hidden_per_head)) # permute for sliced rotary def permute(w, skip_permute=skip_permute): if skip_permute: return w return w.view(n_heads, n_hidden // n_heads // 2, 2, n_hidden).transpose(1, 2).reshape(n_hidden, n_hidden) print(f"Fetching all parameters from the checkpoint at {input_base_path}.") # Load weights if num_shards==1: # Not sharded # (The sharded implementation would also work, but this is simpler.) # /pure-mlo-scratch/alhernan/megatron-data/checkpoints/llama2-7b-tp4-pp1-optim/release/mp_rank_00/model_optim_rng.pt if os.path.exists(os.path.join(input_base_path, 'release')): filename = os.path.join(input_base_path, 'release', 'mp_rank_00', 'model_optim_rng.pt') elif input_base_path.split('/')[-1].startswith('iter_'): iteration = eval(input_base_path.split('/')[-1].replace('iter_', '').lstrip('0')) load_dir = '/'.join(input_base_path.split('/')[:-1]) filename = os.path.join(input_base_path, 'mp_rank_00', 'model_optim_rng.pt') if not os.path.exists(filename): filename = filename.replace('model_optim_rng.pt', 'model_rng.pt') else: tracker_filename = os.path.join(input_base_path, 'latest_checkpointed_iteration.txt') with open(tracker_filename, 'r') as f: metastring = f.read().strip() iteration = 'iter_{:07d}'.format(int(metastring)) filename = os.path.join(input_base_path, iteration, 'mp_rank_00', 'model_optim_rng.pt') if not os.path.exists(filename): filename = filename.replace('model_optim_rng.pt', 'model_rng.pt') original_filename = filename loaded = torch.load(filename, map_location="cpu")['model']['language_model'] else: # Sharded filenames = [] for i in range(num_shards): if os.path.exists(os.path.join(input_base_path, 'release')): filename = os.path.join(input_base_path, 'release', f'mp_rank_{i:02d}', 'model_optim_rng.pt') else: tracker_filename = os.path.join(input_base_path, 'latest_checkpointed_iteration.txt') with open(tracker_filename, 'r') as f: metastring = f.read().strip() iteration = 'iter_{:07d}'.format(int(metastring)) filename = os.path.join(input_base_path, iteration, f'mp_rank_{i:02d}', 'model_optim_rng.pt') if not os.path.exists(filename): filename = filename.replace('model_optim_rng.pt', 'model_rng.pt') filenames.append(filename) loaded = [ torch.load(filenames[i], map_location="cpu")['model']['language_model'] for i in range(num_shards) ] print('Llama-Megatron Loaded!') param_count = 0 index_dict = {"weight_map": {}} print(f'Weighted Converting for {n_layers} layers...') for layer_i in range(n_layers): print(layer_i) filename = f"pytorch_model-{layer_i + 1}-of-{n_layers + 1}.bin" if num_shards == 1: # Unsharded state_dict = { f"model.layers.{layer_i}.self_attn.q_proj.weight": loaded['encoder'][f"layers.{layer_i}.self_attention.q_proj.weight"], f"model.layers.{layer_i}.self_attn.k_proj.multiway.0.weight": loaded['encoder'][f"layers.{layer_i}.self_attention.k_proj.multiway.0.weight"], f"model.layers.{layer_i}.self_attn.v_proj.multiway.0.weight": loaded['encoder'][f"layers.{layer_i}.self_attention.v_proj.multiway.0.weight"], f"model.layers.{layer_i}.self_attn.k_proj.multiway.1.weight": loaded['encoder'][f"layers.{layer_i}.self_attention.k_proj.multiway.1.weight"], f"model.layers.{layer_i}.self_attn.v_proj.multiway.1.weight": loaded['encoder'][f"layers.{layer_i}.self_attention.v_proj.multiway.1.weight"], f"model.layers.{layer_i}.self_attn.o_proj.weight": loaded['encoder'][f"layers.{layer_i}.self_attention.o_proj.weight"], f"model.layers.{layer_i}.mlp.gate_proj.weight": loaded['encoder'][f"layers.{layer_i}.mlp.gate_proj.weight"], f"model.layers.{layer_i}.mlp.down_proj.weight": loaded['encoder'][f"layers.{layer_i}.mlp.down_proj.weight"], f"model.layers.{layer_i}.mlp.up_proj.weight": loaded['encoder'][f"layers.{layer_i}.mlp.up_proj.weight"], f"model.layers.{layer_i}.input_layernorm.multiway.0.weight": loaded['encoder'][f"layers.{layer_i}.input_layernorm.multiway.0.weight"], f"model.layers.{layer_i}.post_attention_layernorm.multiway.0.weight": loaded['encoder'][f"layers.{layer_i}.post_attention_layernorm.multiway.0.weight"], f"model.layers.{layer_i}.input_layernorm.multiway.1.weight": loaded['encoder'][f"layers.{layer_i}.input_layernorm.multiway.1.weight"], f"model.layers.{layer_i}.post_attention_layernorm.multiway.1.weight": loaded['encoder'][f"layers.{layer_i}.post_attention_layernorm.multiway.1.weight"], } else: raise NotImplemented state_dict[f"model.layers.{layer_i}.self_attn.rotary_emb.inv_freq"] = inv_freq for k, v in state_dict.items(): index_dict["weight_map"][k] = filename param_count += v.numel() torch.save(state_dict, os.path.join(tmp_model_path, filename)) print(f'Sharded file saved to {filename}') filename = f"pytorch_model-{n_layers + 1}-of-{n_layers + 1}.bin" if num_shards==1: # Unsharded state_dict = { "model.embed_tokens.weight": loaded['embedding']['word_embeddings']['weight'], "model.norm.weight": loaded['encoder']['norm.weight'], "lm_head.weight": loaded['encoder']['lm_head.weight'], } else: state_dict = { "model.embed_tokens.weight": loaded[0]['embedding']['word_embeddings']['weight'], "model.norm.weight": loaded[0]['encoder']['norm.weight'], "lm_head.weight": loaded[0]['encoder']['lm_head.weight'], } loaded_all = torch.load(original_filename, map_location="cpu")['model'] # Vision Part state_dict.update({ "model.vision_model.embeddings.cls_token": loaded_all['vision_model']['cls_token'], "model.vision_model.embeddings.patch_embed.weight": loaded_all['vision_model']['patch_embed']['weight'], "model.vision_model.embeddings.position_embedding": loaded_all['vision_model']['position_embeddings'], "model.vision_model.embeddings.pre_layernorm.bias": loaded_all['vision_model']['pre_layernorm']['bias'], "model.vision_model.embeddings.pre_layernorm.weight": loaded_all['vision_model']['pre_layernorm']['weight'], "model.vision_model.post_layernorm.bias": loaded_all['vision_model']['transformer']['final_layernorm.bias'], "model.vision_model.post_layernorm.weight": loaded_all['vision_model']['transformer']['final_layernorm.weight'], }) for v_layer_idx in range(24): state_dict.update({ f"model.vision_model.encoder.layers.{v_layer_idx}.input_layernorm.bias": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.input_layernorm.bias'], f"model.vision_model.encoder.layers.{v_layer_idx}.input_layernorm.weight": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.input_layernorm.weight'], f"model.vision_model.encoder.layers.{v_layer_idx}.mlp.fc1.bias": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.mlp.dense_h_to_4h.bias'], f"model.vision_model.encoder.layers.{v_layer_idx}.mlp.fc1.weight": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.mlp.dense_h_to_4h.weight'], f"model.vision_model.encoder.layers.{v_layer_idx}.mlp.fc2.bias": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.mlp.dense_4h_to_h.bias'], f"model.vision_model.encoder.layers.{v_layer_idx}.mlp.fc2.weight": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.mlp.dense_4h_to_h.weight'], f"model.vision_model.encoder.layers.{v_layer_idx}.post_attention_layernorm.bias": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.post_attention_layernorm.bias'], f"model.vision_model.encoder.layers.{v_layer_idx}.post_attention_layernorm.weight": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.post_attention_layernorm.weight'], f"model.vision_model.encoder.layers.{v_layer_idx}.self_attn.dense.bias": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.self_attention.dense.bias'], f"model.vision_model.encoder.layers.{v_layer_idx}.self_attn.dense.weight": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.self_attention.dense.weight'], f"model.vision_model.encoder.layers.{v_layer_idx}.self_attn.query_key_value.bias": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.self_attention.query_key_value.bias'], f"model.vision_model.encoder.layers.{v_layer_idx}.self_attn.query_key_value.weight": loaded_all['vision_model']['transformer'][f'layers.{v_layer_idx}.self_attention.query_key_value.weight'], }) # Vision2Text Part: HReducer state_dict.update({ "model.vision2text.ln_q.weight": loaded_all['hreducer3']['ln_q']['weight'], "model.vision2text.ln_q.bias": loaded_all['hreducer3']['ln_q']['bias'], "model.vision2text.visual_fc.bias": loaded_all['hreducer3']['visual_fc']['bias'], "model.vision2text.visual_fc.weight": loaded_all['hreducer3']['visual_fc']['weight'], "model.vision2text.vit_eos": loaded_all['hreducer3']['vit_eos'], }) # reducer_before conv (layer 0) + gleu (layer 1) state_dict.update({ f"model.vision2text.reducer_before.0.weight": loaded_all['hreducer3']['reducer_before']["0.weight"], f"model.vision2text.reducer_before.0.bias": loaded_all['hreducer3']['reducer_before']["0.bias"], }) # reducer conv state_dict.update({ f"model.vision2text.reducer.weight": loaded_all['hreducer3']['reducer']["weight"], f"model.vision2text.reducer.bias": loaded_all['hreducer3']['reducer']["bias"], }) for k, v in state_dict.items(): # ic(k, v) index_dict["weight_map"][k] = filename param_count += v.numel() torch.save(state_dict, os.path.join(tmp_model_path, filename)) # Write configs index_dict["metadata"] = {"total_size": param_count * 2} write_json(index_dict, os.path.join(tmp_model_path, "pytorch_model.bin.index.json")) config = MPLUGDocOwlConfig() config.save_pretrained(tmp_model_path) # Make space so we can load the model properly now. del state_dict del loaded del loaded_all gc.collect() def write_tokenizer(tokenizer_path, input_tokenizer_path): # Initialize the tokenizer based on the `spm` model tokenizer_class = LlamaTokenizer if LlamaTokenizerFast is None else LlamaTokenizerFast print(f"Saving a {tokenizer_class.__name__} to {tokenizer_path}.") tokenizer = tokenizer_class(input_tokenizer_path) tokenizer.save_pretrained(tokenizer_path) def main(): parser = argparse.ArgumentParser() parser.add_argument( "--input_dir", help="Location of LLaMA_Megatron weights", ) parser.add_argument( "--model_size", type=int, default=7, choices=[7, 13, 30, 65, 70], ) parser.add_argument( "--num_input_shards", type=int, default=1, ) parser.add_argument( "--num_output_shards", type=int, default=1, ) parser.add_argument('--skip_permute', action='store_true') parser.add_argument( "--output_dir", help="Location to write HF model and tokenizer", ) args = parser.parse_args() write_model( model_path=args.output_dir, input_base_path=args.input_dir, model_size=args.model_size, num_input_shards=args.num_input_shards, num_output_shards=args.num_output_shards, skip_permute=args.skip_permute ) if __name__ == "__main__": main()