| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| """ 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) |
|
|