| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| | """Conversion script for the LoRA's safetensors checkpoints.""" |
| |
|
| | import argparse |
| |
|
| | import torch |
| | from safetensors.torch import load_file |
| |
|
| | from diffusers import StableDiffusionPipeline |
| |
|
| |
|
| | def convert(base_model_path, checkpoint_path, LORA_PREFIX_UNET, LORA_PREFIX_TEXT_ENCODER, alpha): |
| | |
| | pipeline = StableDiffusionPipeline.from_pretrained(base_model_path, torch_dtype=torch.float32) |
| |
|
| | |
| | state_dict = load_file(checkpoint_path) |
| |
|
| | visited = [] |
| |
|
| | |
| | for key in state_dict: |
| | |
| | |
| |
|
| | |
| | if ".alpha" in key or key in visited: |
| | continue |
| |
|
| | if "text" in key: |
| | layer_infos = key.split(".")[0].split(LORA_PREFIX_TEXT_ENCODER + "_")[-1].split("_") |
| | curr_layer = pipeline.text_encoder |
| | else: |
| | layer_infos = key.split(".")[0].split(LORA_PREFIX_UNET + "_")[-1].split("_") |
| | curr_layer = pipeline.unet |
| |
|
| | |
| | temp_name = layer_infos.pop(0) |
| | while len(layer_infos) > -1: |
| | try: |
| | curr_layer = curr_layer.__getattr__(temp_name) |
| | if len(layer_infos) > 0: |
| | temp_name = layer_infos.pop(0) |
| | elif len(layer_infos) == 0: |
| | break |
| | except Exception: |
| | if len(temp_name) > 0: |
| | temp_name += "_" + layer_infos.pop(0) |
| | else: |
| | temp_name = layer_infos.pop(0) |
| |
|
| | pair_keys = [] |
| | if "lora_down" in key: |
| | pair_keys.append(key.replace("lora_down", "lora_up")) |
| | pair_keys.append(key) |
| | else: |
| | pair_keys.append(key) |
| | pair_keys.append(key.replace("lora_up", "lora_down")) |
| |
|
| | |
| | if len(state_dict[pair_keys[0]].shape) == 4: |
| | weight_up = state_dict[pair_keys[0]].squeeze(3).squeeze(2).to(torch.float32) |
| | weight_down = state_dict[pair_keys[1]].squeeze(3).squeeze(2).to(torch.float32) |
| | curr_layer.weight.data += alpha * torch.mm(weight_up, weight_down).unsqueeze(2).unsqueeze(3) |
| | else: |
| | weight_up = state_dict[pair_keys[0]].to(torch.float32) |
| | weight_down = state_dict[pair_keys[1]].to(torch.float32) |
| | curr_layer.weight.data += alpha * torch.mm(weight_up, weight_down) |
| |
|
| | |
| | for item in pair_keys: |
| | visited.append(item) |
| |
|
| | return pipeline |
| |
|
| |
|
| | if __name__ == "__main__": |
| | parser = argparse.ArgumentParser() |
| |
|
| | parser.add_argument( |
| | "--base_model_path", default=None, type=str, required=True, help="Path to the base model in diffusers format." |
| | ) |
| | parser.add_argument( |
| | "--checkpoint_path", default=None, type=str, required=True, help="Path to the checkpoint to convert." |
| | ) |
| | parser.add_argument("--dump_path", default=None, type=str, required=True, help="Path to the output model.") |
| | parser.add_argument( |
| | "--lora_prefix_unet", default="lora_unet", type=str, help="The prefix of UNet weight in safetensors" |
| | ) |
| | parser.add_argument( |
| | "--lora_prefix_text_encoder", |
| | default="lora_te", |
| | type=str, |
| | help="The prefix of text encoder weight in safetensors", |
| | ) |
| | parser.add_argument("--alpha", default=0.75, type=float, help="The merging ratio in W = W0 + alpha * deltaW") |
| | parser.add_argument( |
| | "--to_safetensors", action="store_true", help="Whether to store pipeline in safetensors format or not." |
| | ) |
| | parser.add_argument("--device", type=str, help="Device to use (e.g. cpu, cuda:0, cuda:1, etc.)") |
| |
|
| | args = parser.parse_args() |
| |
|
| | base_model_path = args.base_model_path |
| | checkpoint_path = args.checkpoint_path |
| | dump_path = args.dump_path |
| | lora_prefix_unet = args.lora_prefix_unet |
| | lora_prefix_text_encoder = args.lora_prefix_text_encoder |
| | alpha = args.alpha |
| |
|
| | pipe = convert(base_model_path, checkpoint_path, lora_prefix_unet, lora_prefix_text_encoder, alpha) |
| |
|
| | pipe = pipe.to(args.device) |
| | pipe.save_pretrained(args.dump_path, safe_serialization=args.to_safetensors) |
| |
|