Skip to content

Latest commit

 

History

History
311 lines (253 loc) · 15.5 KB

README.md

File metadata and controls

311 lines (253 loc) · 15.5 KB

Stable Diffusion Implementation in PyTorch

This repository implements Stable Diffusion. As of today the repo provides code to do the following:

  • Training and Inference on Unconditional Latent Diffusion Models
  • Training a Class Conditional Latent Diffusion Model
  • Training a Text Conditioned Latent Diffusion Model
  • Training a Semantic Mask Conditioned Latent Diffusion Model
  • Any Combination of the above three conditioning

For autoencoder I provide code for vae as well as vqvae. But both the stages of training use VQVAE only. One can easily change that to vae if needed

For diffusion part, as of now it only implements DDPM with linear schedule.

Stable Diffusion Tutorial Videos

Stable Diffusion Tutorial Stable Diffusion Conditioning Tutorial ___

Sample Output for Autoencoder on CelebHQ

Image - Top, Reconstructions - Below

Sample Output for Unconditional LDM on CelebHQ (not fully converged)

Sample Output for Conditional LDM

Sample Output for Class Conditioned on MNIST

50 50 50 50 50

Sample Output for Text(using CLIP) and Mask Conditioned on CelebHQ (not converged)


Text - She is a woman with blond hair
Text - She is a woman with black hair

Setup

  • Create a new conda environment with python 3.8 then run below commands
  • conda activate <environment_name>
  • git clone https://github.com/explainingai-code/StableDiffusion-PyTorch.git
  • cd StableDiffusion-PyTorch
  • pip install -r requirements.txt
  • Download lpips weights by opening this link in browser(dont use cURL or wget) https://github.com/richzhang/PerceptualSimilarity/blob/master/lpips/weights/v0.1/vgg.pth and downloading the raw file. Place the downloaded weights file in models/weights/v0.1/vgg.pth

Data Preparation

Mnist

For setting up the mnist dataset follow - https://github.com/explainingai-code/Pytorch-VAE#data-preparation

Ensure directory structure is following

StableDiffusion-PyTorch
    -> data
        -> mnist
            -> train
                -> images
                    -> *.png
            -> test
                -> images
                    -> *.png

CelebHQ

Unconditional

For setting up on CelebHQ for unconditional, simply download the images from the official repo of CelebMASK HQ here.

Ensure directory structure is the following

StableDiffusion-PyTorch
    -> data
        -> CelebAMask-HQ
            -> CelebA-HQ-img
                -> *.jpg

Mask Conditional

For CelebHQ for mask conditional LDM additionally do the following:

Ensure directory structure is the following

StableDiffusion-PyTorch
    -> data
        -> CelebAMask-HQ
            -> CelebA-HQ-img
                -> *.jpg
            -> CelebAMask-HQ-mask-anno
                -> 0/1/2/3.../14
                    -> *.png
            
  • Run python -m utils.create_celeb_mask from repo root to create the mask images from mask annotations

Ensure directory structure is the following

StableDiffusion-PyTorch
    -> data
        -> CelebAMask-HQ
            -> CelebA-HQ-img
                -> *.jpg
            -> CelebAMask-HQ-mask-anno
                -> 0/1/2/3.../14
                    -> *.png
            -> CelebAMask-HQ-mask
                  -> *.png

Text Conditional

For CelebHQ for text conditional LDM additionally do the following:

Ensure directory structure is the following

StableDiffusion-PyTorch
    -> data
        -> CelebAMask-HQ
            -> CelebA-HQ-img
                -> *.jpg
            -> CelebAMask-HQ-mask-anno
                -> 0/1/2/3.../14
                    -> *.png
            -> CelebAMask-HQ-mask
                -> *.png
            -> celeba-caption
                -> *.txt

Configuration

Allows you to play with different components of ddpm and autoencoder training

  • config/mnist.yaml - Small autoencoder and ldm can even be trained on CPU
  • config/celebhq.yaml - Configuration used for celebhq dataset

Relevant configuration parameters

Most parameters are self explanatory but below I mention couple which are specific to this repo.

  • autoencoder_acc_steps : For accumulating gradients if image size is too large for larger batch sizes
  • save_latents : Enable this to save the latents , during inference of autoencoder. That way ddpm training will be faster

Training

The repo provides training and inference for Mnist(Unconditional and Class Conditional) and CelebHQ (Unconditional, Text and/or Mask Conditional).

For working on your own dataset:

  • Create your own config and have the path in config point to images (look at celebhq.yaml for guidance)
  • Create your own dataset class which will just collect all the filenames and return the image in its getitem method. Look at mnist_dataset.py or celeb_dataset.py for guidance

Once the config and dataset is setup:

Training AutoEncoder for LDM

  • For training autoencoder on mnist,ensure the right path is mentioned in mnist.yaml
  • For training autoencoder on celebhq,ensure the right path is mentioned in celebhq.yaml
  • For training autoencoder on your own dataset
    • Create your own config and have the path point to images (look at celebhq.yaml for guidance)
    • Create your own dataset class, similar to celeb_dataset.py without conditining parts
  • Map the dataset name to the right class in the training code here
  • For training autoencoder run python -m tools.train_vqvae --config config/mnist.yaml for training vqvae with the desire config file
  • For inference using trained autoencoder runpython -m tools.infer_vqvae --config config/mnist.yaml for generating reconstructions with right config file. Use save_latent in config to save the latent files

Training Unconditional LDM

Train the autoencoder first and setup dataset accordingly.

For training unconditional LDM map the dataset to the right class in train_ddpm_vqvae.py

  • python -m tools.train_ddpm_vqvae --config config/mnist.yaml for training unconditional ddpm using right config
  • python -m tools.sample_ddpm_vqvae --config config/mnist.yaml for generating images using trained ddpm

Training Conditional LDM

For training conditional models we need two changes:

  • Dataset classes must provide the additional conditional inputs(see below)
  • Config must be changed with additional conditioning config added

Specifically the dataset getitem will return the following:

  • image_tensor for unconditional training
  • tuple of (image_tensor, cond_input ) for conditional training where cond_input is a dictionary consisting of keys {class/text/image}

Training Class Conditional LDM

The repo provides class conditional latent diffusion model training code for mnist dataset, so one can use that to follow the same for their own dataset

  • Use mnist_class_cond.yaml config file as a guide to create your class conditional config file. Specifically following new keys need to be modified according to your dataset within ldm_params.
  • condition_config:
      condition_types: ['class']
      class_condition_config :
        num_classes : <number of classes: 10 for mnist>
        cond_drop_prob : <probability of dropping class labels>
    
  • Create a dataset class similar to mnist where the getitem method now returns a tuple of image_tensor and dictionary of conditional_inputs.
  • For class, conditional input will ONLY be the integer class
  •   (image_tensor, {
                      'class' : {0/1/.../num_classes}
                      })
    
    

For training class conditional LDM map the dataset to the right class in train_ddpm_cond and run the below commands using desired config

  • python -m tools.train_ddpm_cond --config config/mnist_class_cond.yaml for training class conditional on mnist
  • python -m tools.sample_ddpm_class_cond --config config/mnist.yaml for generating images using class conditional trained ddpm

Training Text Conditional LDM

The repo provides text conditional latent diffusion model training code for celebhq dataset, so one can use that to follow the same for their own dataset

  • Use celebhq_text_cond.yaml config file as a guide to create your config file. Specifically following new keys need to be modified according to your dataset within ldm_params.
  •   condition_config:
          condition_types: [ 'text' ]
          text_condition_config:
              text_embed_model: 'clip' or 'bert'
              text_embed_dim: 512 or 768
              cond_drop_prob: 0.1
    
  • Create a dataset class similar to celebhq where the getitem method now returns a tuple of image_tensor and dictionary of conditional_inputs.
  • For text, conditional input will ONLY be the caption
  •   (image_tensor, {
                      'text' : 'a sample caption for image_tensor'
                      })
    
    

For training text conditional LDM map the dataset to the right class in train_ddpm_cond and run the below commands using desired config

  • python -m tools.train_ddpm_cond --config config/celebhq_text_cond.yaml for training text conditioned ldm on celebhq
  • python -m tools.sample_ddpm_text_cond --config config/celebhq_text_cond.yaml for generating images using text conditional trained ddpm

Training Text and Mask Conditional LDM

The repo provides text and mask conditional latent diffusion model training code for celebhq dataset, so one can use that to follow the same for their own dataset and can even use that train a mask only conditional ldm

  • Use celebhq_text_image_cond.yaml config file as a guide to create your config file. Specifically following new keys need to be modified according to your dataset within ldm_params.
  •   condition_config:
          condition_types: [ 'text', 'image' ]
          text_condition_config:
              text_embed_model: 'clip' or 'bert
              text_embed_dim: 512 or 768
              cond_drop_prob: 0.1
          image_condition_config:
             image_condition_input_channels: 18
             image_condition_output_channels: 3
             image_condition_h : 512 
             image_condition_w : 512
             cond_drop_prob: 0.1
    
  • Create a dataset class similar to celebhq where the getitem method now returns a tuple of image_tensor and dictionary of conditional_inputs.
  • For text and mask, conditional input will be caption and mask image
  •   (image_tensor, {
                      'text' : 'a sample caption for image_tensor',
                      'image' : NUM_CLASSES x MASK_H x MASK_W
                      })
    
    

For training text unconditional LDM map the dataset to the right class in train_ddpm_cond and run the below commands using desired config

  • python -m tools.train_ddpm_cond --config config/celebhq_text_image_cond.yaml for training text and mask conditioned ldm on celebhq
  • python -m tools.sample_ddpm_text_image_cond --config config/celebhq_text_image_cond.yaml for generating images using text and mask conditional trained ddpm

Output

Outputs will be saved according to the configuration present in yaml files.

For every run a folder of task_name key in config will be created

During training of autoencoder the following output will be saved

  • Latest Autoencoder and discriminator checkpoint in task_name directory
  • Sample reconstructions in task_name/vqvae_autoencoder_samples

During inference of autoencoder the following output will be saved

  • Reconstructions for random images in task_name
  • Latents will be save in task_name/vqvae_latent_dir_name if mentioned in config

During training and inference of ddpm following output will be saved

  • During training of unconditional or conditional DDPM we will save the latest checkpoint in task_name directory
  • During sampling, unconditional sampled image grid for all timesteps in task_name/samples/*.png . The final decoded generated image will be x0_0.png. Images from x0_999.png to x0_1.png will be latent image predictions of denoising process from T=999 to T=1. Generated Image is at T=0
  • During sampling, class conditionally sampled image grid for all timesteps in task_name/cond_class_samples/*.png . The final decoded generated image will be x0_0.png. Images from x0_999.png to x0_1.png will be latent image predictions of denoising process from T=999 to T=1. Generated Image is at T=0
  • During sampling, text only conditionally sampled image grid for all timesteps in task_name/cond_text_samples/*.png . The final decoded generated image will be x0_0.png . Images from x0_999.png to x0_1.png will be latent image predictions of denoising process from T=999 to T=1. Generated Image is at T=0
  • During sampling, image only conditionally sampled image grid for all timesteps in task_name/cond_text_image_samples/*.png . The final decoded generated image will be x0_0.png. Images from x0_999.png to x0_1.png will be latent image predictions of denoising process from T=999 to T=1. Generated Image is at T=0