# modified from HuggingFace diffusers (0.32.1) `pipelines/stable_diffusion/pipeline_stable_diffusion.py`

# Copyright 2024 The HuggingFace 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.
from util.logger import logger

from typing import Any, Tuple, Callable, Dict, List, Optional, Union

import gc

import torch

import numpy as np

from diffusers.callbacks import MultiPipelineCallbacks, PipelineCallback
from diffusers.image_processor import PipelineImageInput
from diffusers.utils import (
    deprecate,
    logging,
    replace_example_docstring
)
from diffusers.pipelines.stable_diffusion.pipeline_output import StableDiffusionPipelineOutput
from diffusers.pipelines.stable_diffusion.safety_checker import StableDiffusionSafetyChecker

from diffusers.pipelines.stable_diffusion.pipeline_stable_diffusion import (
    EXAMPLE_DOC_STRING, 
    rescale_noise_cfg, 
    retrieve_timesteps, 
    XLA_AVAILABLE
)

from types import MethodType

from .util.timestep_util import prepare_timestep

from util.torch_util import tsfm_to_1d_array
from util.pipeline_util import img_latent_to_pil


logger = logging.get_logger(__name__)  # pylint: disable=invalid-name


@torch.no_grad()
@replace_example_docstring(EXAMPLE_DOC_STRING)
def forward(
    self, 

    prompt: Union[str, List[str]] = None,
    height: Optional[int] = None,
    width: Optional[int] = None,
    num_inference_steps: int = 50,
    timesteps: List[int] = None,
    sigmas: List[float] = None,
    guidance_scale: float = 7.5,
    negative_prompt: Optional[Union[str, List[str]]] = None,
    num_images_per_prompt: Optional[int] = 1,
    # eta: float = 0.0,  # disabled
    generator: Optional[Union[torch.Generator, List[torch.Generator]]] = None,
    latents: Optional[torch.Tensor] = None,
    prompt_embeds: Optional[torch.Tensor] = None,
    negative_prompt_embeds: Optional[torch.Tensor] = None,
    ip_adapter_image: Optional[PipelineImageInput] = None,
    ip_adapter_image_embeds: Optional[List[torch.Tensor]] = None,
    # output_type: Optional[str] = "pil",
    return_dict: bool = True,
    cross_attention_kwargs: Optional[Dict[str, Any]] = None,
    guidance_rescale: float = 0.0,
    clip_skip: Optional[int] = None,
    callback_on_step_end: Optional[
            Union[Callable[[int, int, Dict], None], 
            PipelineCallback, 
            MultiPipelineCallbacks
        ]
    ] = None,
    callback_on_step_end_tensor_inputs: List[str] = ["latents"],

    inference_step_minus_one: Optional[bool] = False, 

    # ---------= [Eta & Eps] =---------
    eta_list: Union[torch.Tensor, List[torch.Tensor]] = None, 
    eps_list: Union[torch.Tensor, List[torch.Tensor]] = None, 

    # save inference process
    save_inference_process_dict: Optional[Dict] = None, 

    **kwargs, 
):
    r"""
    The call function to the pipeline for generation.

    Args:
        prompt (`str` or `List[str]`, *optional*):
            The prompt or prompts to guide image generation. If not defined, you need to pass `prompt_embeds`.
        height (`int`, *optional*, defaults to `self.unet.config.sample_size * self.vae_scale_factor`):
            The height in pixels of the generated image.
        width (`int`, *optional*, defaults to `self.unet.config.sample_size * self.vae_scale_factor`):
            The width in pixels of the generated image.
        num_inference_steps (`int`, *optional*, defaults to 50):
            The number of denoising steps. More denoising steps usually lead to a higher quality image at the
            expense of slower inference.
        timesteps (`List[int]`, *optional*):
            Custom timesteps to use for the denoising process with schedulers which support a `timesteps` argument
            in their `set_timesteps` method. If not defined, the default behavior when `num_inference_steps` is
            passed will be used. Must be in descending order.
        sigmas (`List[float]`, *optional*):
            Custom sigmas to use for the denoising process with schedulers which support a `sigmas` argument in
            their `set_timesteps` method. If not defined, the default behavior when `num_inference_steps` is passed
            will be used.
        guidance_scale (`float`, *optional*, defaults to 7.5):
            A higher guidance scale value encourages the model to generate images closely linked to the text
            `prompt` at the expense of lower image quality. Guidance scale is enabled when `guidance_scale > 1`.
        negative_prompt (`str` or `List[str]`, *optional*):
            The prompt or prompts to guide what to not include in image generation. If not defined, you need to
            pass `negative_prompt_embeds` instead. Ignored when not using guidance (`guidance_scale < 1`).
        num_images_per_prompt (`int`, *optional*, defaults to 1):
            The number of images to generate per prompt.
        eta (`float`, *optional*, defaults to 0.0):
            Corresponds to parameter eta (η) from the [DDIM](https://arxiv.org/abs/2010.02502) paper. Only applies
            to the [`~schedulers.DDIMScheduler`], and is ignored in other schedulers.
        generator (`torch.Generator` or `List[torch.Generator]`, *optional*):
            A [`torch.Generator`](https://pytorch.org/docs/stable/generated/torch.Generator.html) to make
            generation deterministic.
        latents (`torch.Tensor`, *optional*):
            Pre-generated noisy latents sampled from a Gaussian distribution, to be used as inputs for image
            generation. Can be used to tweak the same generation with different prompts. If not provided, a latents
            tensor is generated by sampling using the supplied random `generator`.
        prompt_embeds (`torch.Tensor`, *optional*):
            Pre-generated text embeddings. Can be used to easily tweak text inputs (prompt weighting). If not
            provided, text embeddings are generated from the `prompt` input argument.
        negative_prompt_embeds (`torch.Tensor`, *optional*):
            Pre-generated negative text embeddings. Can be used to easily tweak text inputs (prompt weighting). If
            not provided, `negative_prompt_embeds` are generated from the `negative_prompt` input argument.
        ip_adapter_image: (`PipelineImageInput`, *optional*): Optional image input to work with IP Adapters.
        ip_adapter_image_embeds (`List[torch.Tensor]`, *optional*):
            Pre-generated image embeddings for IP-Adapter. It should be a list of length same as number of
            IP-adapters. Each element should be a tensor of shape `(batch_size, num_images, emb_dim)`. It should
            contain the negative image embedding if `do_classifier_free_guidance` is set to `True`. If not
            provided, embeddings are computed from the `ip_adapter_image` input argument.
        output_type (`str`, *optional*, defaults to `"pil"`):
            The output format of the generated image. Choose between `PIL.Image` or `np.array`.
        return_dict (`bool`, *optional*, defaults to `True`):
            Whether or not to return a [`~pipelines.stable_diffusion.StableDiffusionPipelineOutput`] instead of a
            plain tuple.
        cross_attention_kwargs (`dict`, *optional*):
            A kwargs dictionary that if specified is passed along to the [`AttentionProcessor`] as defined in
            [`self.processor`](https://github.com/huggingface/diffusers/blob/main/src/diffusers/models/attention_processor.py).
        guidance_rescale (`float`, *optional*, defaults to 0.0):
            Guidance rescale factor from [Common Diffusion Noise Schedules and Sample Steps are
            Flawed](https://arxiv.org/pdf/2305.08891.pdf). Guidance rescale factor should fix overexposure when
            using zero terminal SNR.
        clip_skip (`int`, *optional*):
            Number of layers to be skipped from CLIP while computing the prompt embeddings. A value of 1 means that
            the output of the pre-final layer will be used for computing the prompt embeddings.
        callback_on_step_end (`Callable`, `PipelineCallback`, `MultiPipelineCallbacks`, *optional*):
            A function or a subclass of `PipelineCallback` or `MultiPipelineCallbacks` that is called at the end of
            each denoising step during the inference. with the following arguments: `callback_on_step_end(self:
            DiffusionPipeline, step: int, timestep: int, callback_kwargs: Dict)`. `callback_kwargs` will include a
            list of all tensors as specified by `callback_on_step_end_tensor_inputs`.
        callback_on_step_end_tensor_inputs (`List`, *optional*):
            The list of tensor inputs for the `callback_on_step_end` function. The tensors specified in the list
            will be passed as `callback_kwargs` argument. You will only be able to include variables listed in the
            `._callback_tensor_inputs` attribute of your pipeline class.

    Examples:

    Returns:
        [`~pipelines.stable_diffusion.StableDiffusionPipelineOutput`] or `tuple`:
            If `return_dict` is `True`, [`~pipelines.stable_diffusion.StableDiffusionPipelineOutput`] is returned,
            otherwise a `tuple` is returned where the first element is a list with the generated images and the
            second element is a list of `bool`s indicating whether the corresponding generated image contains
            "not-safe-for-work" (nsfw) content.
    """
    
    callback = kwargs.pop("callback", None)
    callback_steps = kwargs.pop("callback_steps", None)

    if callback is not None:
        deprecate(
            "callback",
            "1.0.0",
            "Passing `callback` as an input argument to `__call__` is deprecated, consider using `callback_on_step_end`",
        )
    if callback_steps is not None:
        deprecate(
            "callback_steps",
            "1.0.0",
            "Passing `callback_steps` as an input argument to `__call__` is deprecated, consider using `callback_on_step_end`",
        )

    if isinstance(callback_on_step_end, (PipelineCallback, MultiPipelineCallbacks)):
        callback_on_step_end_tensor_inputs = callback_on_step_end.tensor_inputs

    # 0. Default height and width to unet
    if not height or not width:
        height = (
            self.unet.config.sample_size
            if self._is_unet_config_sample_size_int
            else self.unet.config.sample_size[0]
        )
        width = (
            self.unet.config.sample_size
            if self._is_unet_config_sample_size_int
            else self.unet.config.sample_size[1]
        )
        height, width = height * self.vae_scale_factor, width * self.vae_scale_factor
    # to deal with lora scaling and other possible forward hooks
    
    # 1. Check inputs. Raise error if not correct
    self.check_inputs(
        prompt,
        height,
        width,
        callback_steps,
        negative_prompt,
        prompt_embeds,
        negative_prompt_embeds,
        ip_adapter_image,
        ip_adapter_image_embeds,
        callback_on_step_end_tensor_inputs,
    )

    self._guidance_scale = guidance_scale
    self._guidance_rescale = guidance_rescale
    self._clip_skip = clip_skip
    self._cross_attention_kwargs = cross_attention_kwargs
    self._interrupt = False

    # 2. Define call parameters
    if prompt is not None and isinstance(prompt, str):
        batch_size = 1
    elif prompt is not None and isinstance(prompt, list):
        batch_size = len(prompt)
    else:
        batch_size = prompt_embeds.shape[0]
    
    device = self._execution_device

    # 3. Encode input prompt
    lora_scale = (
        self.cross_attention_kwargs.get("scale", None) if self.cross_attention_kwargs is not None else None
    )

    prompt_embeds, negative_prompt_embeds = self.encode_prompt(
        prompt,
        device,
        num_images_per_prompt,
        self.do_classifier_free_guidance,
        negative_prompt,
        prompt_embeds=prompt_embeds,
        negative_prompt_embeds=negative_prompt_embeds,
        lora_scale=lora_scale,
        clip_skip=self.clip_skip,
    )

    # For classifier free guidance, we need to do two forward passes.
    # Here we concatenate the unconditional and text embeddings into a single batch
    # to avoid doing two forward passes
    if self.do_classifier_free_guidance:
        prompt_embeds = torch.cat([negative_prompt_embeds, prompt_embeds])

    if ip_adapter_image is not None or ip_adapter_image_embeds is not None:
        image_embeds = self.prepare_ip_adapter_image_embeds(
            ip_adapter_image,
            ip_adapter_image_embeds,
            device,
            batch_size * num_images_per_prompt,
            self.do_classifier_free_guidance,
        )

    # 4. Prepare timesteps
    if inference_step_minus_one:
        num_inference_steps -= 1
    timesteps, num_inference_steps = retrieve_timesteps(
        self.scheduler, num_inference_steps, device, timesteps, sigmas
    )

    # print(f"timesteps: {timesteps} (len = {len(timesteps)})")

    if save_inference_process_dict is not None:
        if save_inference_process_dict["noise_pred"]:
            # noise_pred_list.shape = (num_inference_step, batch_size, num_channel, noise_pred_h, noise_pred_w)
            noise_pred_list = []
        
        if save_inference_process_dict["latent"]:
            # latent_list.shape = (num_inference_step + 1, batch_size, num_channel, latent_h, latent_w)
            latent_list = []

        if save_inference_process_dict["pil"]:
            # pil_list.shape = (num_inference_step + 1, batch_size)
            pil_list = []

    # 5. Prepare latent variables
    num_channels_latents = self.unet.config.in_channels
    latents = self.prepare_latents(
        batch_size * num_images_per_prompt,
        num_channels_latents,
        height,
        width,
        prompt_embeds.dtype,
        device,
        generator,
        latents,
    )

    if save_inference_process_dict is not None:
        if save_inference_process_dict["latent"] or save_inference_process_dict["pil"]:
            tmp_latent = latents.detach() \
                .cpu() \
                .numpy()

        if save_inference_process_dict["latent"]:
            latent_list.append(tmp_latent)
        
        if save_inference_process_dict["pil"]:
            pil = img_latent_to_pil(
                img_latent = latents, 
                pipeline = self
            )
            pil_list.append(pil)

    # 6. Prepare extra step kwargs. TODO: Logic should ideally just be moved out of the pipeline
    # extra_step_kwargs = self.prepare_extra_step_kwargs(generator, eta)

    # 6.1 Add image embeds for IP-Adapter
    added_cond_kwargs = (
        {"image_embeds": image_embeds}
        if (ip_adapter_image is not None or ip_adapter_image_embeds is not None)
        else None
    )

    # 6.2 Optionally get Guidance Scale Embedding
    timestep_cond = None
    if self.unet.config.time_cond_proj_dim is not None:
        guidance_scale_tensor = torch.tensor(self.guidance_scale - 1).repeat(batch_size * num_images_per_prompt)
        timestep_cond = self.get_guidance_scale_embedding(
            guidance_scale_tensor, embedding_dim=self.unet.config.time_cond_proj_dim
        ).to(device=device, dtype=latents.dtype)

    # 7. Denoising loop
    num_warmup_steps = len(timesteps) - num_inference_steps * self.scheduler.order
    self._num_timesteps = len(timesteps)

    # ---------= [Prepare Eta & Eps] =---------
    if eta_list is None:
        logger(
            "`eta_list` is not provided, default `0.0` for all timesteps. ", 
            log_type = "warning"
        )

        eta_list = [0.0] * num_inference_steps
    else:
        eta_list = tsfm_to_1d_array(
            array = eta_list, 
            target_length = num_inference_steps, 

            dtype = latents.dtype, 
            device = latents.device
        )

    if eps_list is None:
        logger(
            "`eps_list` is not provided, noise will be generated with `generator`. ", 
            log_type = "warning"
        )

        eps_list = [None] * num_inference_steps
    else:
        eps_list = tsfm_to_1d_array(
            array = eps_list, 
            target_length = num_inference_steps, 

            dtype = latents.dtype, 
            device = latents.device
        )

    with self.progress_bar(total=num_inference_steps) as progress_bar:
        for i, t in enumerate(timesteps):
            if self.interrupt:
                continue

            # expand the latents if we are doing classifier free guidance
            latent_model_input = torch.cat([latents] * 2) if self.do_classifier_free_guidance else latents
            latent_model_input = self.scheduler.scale_model_input(latent_model_input, t)

            # predict the noise residual
            noise_pred = self.unet(
                latent_model_input,
                t,
                encoder_hidden_states = prompt_embeds,
                timestep_cond = timestep_cond,
                cross_attention_kwargs = self.cross_attention_kwargs,
                added_cond_kwargs = added_cond_kwargs,
                return_dict = False
            )[0]

            # perform guidance
            if self.do_classifier_free_guidance:
                noise_pred_uncond, noise_pred_text = noise_pred.chunk(2)
                noise_pred = noise_pred_uncond + self.guidance_scale * (noise_pred_text - noise_pred_uncond)

            if self.do_classifier_free_guidance and self.guidance_rescale > 0.0:
                # Based on 3.4. in https://arxiv.org/pdf/2305.08891.pdf
                noise_pred = rescale_noise_cfg(noise_pred, noise_pred_text, guidance_rescale=self.guidance_rescale)

            if save_inference_process_dict is not None:
                if save_inference_process_dict["noise_pred"]:
                    tmp_noise_pred = noise_pred.detach() \
                        .cpu() \
                        .numpy()
                    noise_pred_list.append(tmp_noise_pred)

            # compute the previous noisy sample x_t -> x_t-1
            # latents = self.scheduler.step(noise_pred, t, latents, **extra_step_kwargs, return_dict=False)[0]
            latents = self.scheduler.step(
                model_output = noise_pred, 
                timestep = t, 
                sample = latents, 

                eta = eta_list[i], 
                variance_noise = eps_list[i], 

                generator = generator, 

                return_dict = False
            )[0]

            if callback_on_step_end is not None:
                callback_kwargs = {}
                for k in callback_on_step_end_tensor_inputs:
                    callback_kwargs[k] = locals()[k]
                callback_outputs = callback_on_step_end(self, i, t, callback_kwargs)

                latents = callback_outputs.pop("latents", latents)
                prompt_embeds = callback_outputs.pop("prompt_embeds", prompt_embeds)
                negative_prompt_embeds = callback_outputs.pop("negative_prompt_embeds", negative_prompt_embeds)

            if save_inference_process_dict is not None:
                if save_inference_process_dict["latent"] or save_inference_process_dict["pil"]:
                    tmp_latent = latents.detach() \
                        .cpu() \
                        .numpy()

                if save_inference_process_dict["latent"]:
                    latent_list.append(tmp_latent)
                
                if save_inference_process_dict["pil"]:
                    pil = img_latent_to_pil(
                        img_latent = latents, 
                        pipeline = self
                    )
                    pil_list.append(pil)

            # call the callback, if provided
            if i == len(timesteps) - 1 or ((i + 1) > num_warmup_steps and (i + 1) % self.scheduler.order == 0):
                progress_bar.update()
                if callback is not None and i % callback_steps == 0:
                    step_idx = i // getattr(self.scheduler, "order", 1)
                    callback(step_idx, t, latents)

            if XLA_AVAILABLE:
                xm.mark_step()

    # if not output_type == "latent":
    #     image = self.vae.decode(latents / self.vae.config.scaling_factor, return_dict=False, generator=generator)[
    #         0
    #     ]
    #     image, has_nsfw_concept = self.run_safety_checker(image, device, prompt_embeds.dtype)
    # else:
    #     image = latents
    #     has_nsfw_concept = None

    # if has_nsfw_concept is None:
    #     do_denormalize = [True] * image.shape[0]
    # else:
    #     do_denormalize = [not has_nsfw for has_nsfw in has_nsfw_concept]

    # image = self.image_processor.postprocess(image, output_type=output_type, do_denormalize=do_denormalize)

    # Offload all models
    self.maybe_free_model_hooks()

    inference_process_dict = {}
    if save_inference_process_dict is not None:
        if save_inference_process_dict["noise_pred"]:
            inference_process_dict["noise_pred"] = noise_pred_list

        if save_inference_process_dict["latent"]:
            inference_process_dict["latent"] = latent_list
        
        if save_inference_process_dict["pil"]:
            inference_process_dict["pil"] = pil_list

    if not return_dict:
        # return (image, has_nsfw_concept, inference_process_dict)
        return inference_process_dict
    else:
        raise NotImplementedError(
            f"Only support `return_dict = False`. "
        )

    return StableDiffusionPipelineOutput(images=image, nsfw_content_detected=has_nsfw_concept)


@torch.no_grad()
def prepare_everything(
    self,

    prompt: Union[str, List[str]] = None,
    height: Optional[int] = None,
    width: Optional[int] = None,
    num_inference_steps: int = 50,
    timesteps: List[int] = None,
    sigmas: List[float] = None,
    guidance_scale: float = 7.5,
    negative_prompt: Optional[Union[str, List[str]]] = None,
    num_images_per_prompt: Optional[int] = 1,
    # eta: float = 0.0,  # disabled
    generator: Optional[Union[torch.Generator, List[torch.Generator]]] = None,
    latents: Optional[torch.Tensor] = None,
    prompt_embeds: Optional[torch.Tensor] = None,
    negative_prompt_embeds: Optional[torch.Tensor] = None,
    ip_adapter_image: Optional[PipelineImageInput] = None,
    ip_adapter_image_embeds: Optional[List[torch.Tensor]] = None,
    # output_type: Optional[str] = "pil",
    # return_dict: bool = True,
    cross_attention_kwargs: Optional[Dict[str, Any]] = None,
    guidance_rescale: float = 0.0,
    clip_skip: Optional[int] = None,
    callback_on_step_end: Optional[
            Union[Callable[[int, int, Dict], None], 
            PipelineCallback, 
            MultiPipelineCallbacks
        ]
    ] = None,
    callback_on_step_end_tensor_inputs: List[str] = ["latents"],

    inference_step_minus_one: Optional[bool] = False, 

    **kwargs, 
) -> Dict:
    """
    Func:
        Prepare parameters used for denoising. 

    Ret:
        `prompt_embeds` (`torch.Tensor`): The list of prompt embeddings. 
        `param_dict` (`Dict`): The dictionary of parameters. 
        `latents` (`torch.Tensor`): The initial latent. 
    """
    
    callback = kwargs.pop("callback", None)
    callback_steps = kwargs.pop("callback_steps", None)

    if callback is not None:
        deprecate(
            "callback",
            "1.0.0",
            "Passing `callback` as an input argument to `__call__` is deprecated, consider using `callback_on_step_end`",
        )
    if callback_steps is not None:
        deprecate(
            "callback_steps",
            "1.0.0",
            "Passing `callback_steps` as an input argument to `__call__` is deprecated, consider using `callback_on_step_end`",
        )

    if isinstance(callback_on_step_end, (PipelineCallback, MultiPipelineCallbacks)):
        callback_on_step_end_tensor_inputs = callback_on_step_end.tensor_inputs

    # 0. Default height and width to unet
    if not height or not width:
        height = (
            self.unet.config.sample_size
            if self._is_unet_config_sample_size_int
            else self.unet.config.sample_size[0]
        )
        width = (
            self.unet.config.sample_size
            if self._is_unet_config_sample_size_int
            else self.unet.config.sample_size[1]
        )
        height, width = height * self.vae_scale_factor, width * self.vae_scale_factor
    # to deal with lora scaling and other possible forward hooks
    
    # 1. Check inputs. Raise error if not correct
    self.check_inputs(
        prompt,
        height,
        width,
        callback_steps,
        negative_prompt,
        prompt_embeds,
        negative_prompt_embeds,
        ip_adapter_image,
        ip_adapter_image_embeds,
        callback_on_step_end_tensor_inputs,
    )

    self._guidance_scale = guidance_scale
    self._guidance_rescale = guidance_rescale
    self._clip_skip = clip_skip
    self._cross_attention_kwargs = cross_attention_kwargs
    self._interrupt = False

    # 2. Define call parameters
    if prompt is not None and isinstance(prompt, str):
        batch_size = 1
    elif prompt is not None and isinstance(prompt, list):
        batch_size = len(prompt)
    else:
        batch_size = prompt_embeds.shape[0]
    
    device = self._execution_device

    # 3. Encode input prompt
    lora_scale = (
        self.cross_attention_kwargs.get("scale", None) if self.cross_attention_kwargs is not None else None
    )

    # prompt_embeds.shape = (num_images_per_prompt, 77, 1024)
    prompt_embeds, negative_prompt_embeds = self.encode_prompt(
        prompt,
        device,
        num_images_per_prompt,
        self.do_classifier_free_guidance,
        negative_prompt,
        prompt_embeds=prompt_embeds,
        negative_prompt_embeds=negative_prompt_embeds,
        lora_scale=lora_scale,
        clip_skip=self.clip_skip,
    )
    
    # For classifier free guidance, we need to do two forward passes.
    # Here we concatenate the unconditional and text embeddings into a single batch
    # to avoid doing two forward passes
    if self.do_classifier_free_guidance:
        prompt_embeds = torch.cat([negative_prompt_embeds, prompt_embeds])

    if ip_adapter_image is not None or ip_adapter_image_embeds is not None:
        image_embeds = self.prepare_ip_adapter_image_embeds(
            ip_adapter_image,
            ip_adapter_image_embeds,
            device,
            batch_size * num_images_per_prompt,
            self.do_classifier_free_guidance,
        )

    # 4. Prepare timesteps
    if inference_step_minus_one:
        num_inference_steps -= 1
    timesteps, num_inference_steps = retrieve_timesteps(
        self.scheduler, num_inference_steps, device, timesteps, sigmas
    )

    inv_scheduler = getattr(self, "inv_scheduler")
    if inv_scheduler is not None:
        inv_scheduler.set_timesteps(
            num_inference_steps, 
            device = device
        )

    # 5. Prepare latent variables
    num_channels_latents = self.unet.config.in_channels
    
    latents = self.prepare_latents(
        batch_size * num_images_per_prompt,
        num_channels_latents,
        height,
        width,
        prompt_embeds.dtype,
        device,
        generator,
        latents,
    )

    # 6. Prepare extra step kwargs. TODO: Logic should ideally just be moved out of the pipeline
    # extra_step_kwargs = self.prepare_extra_step_kwargs(generator, eta)

    # 6.1 Add image embeds for IP-Adapter
    added_cond_kwargs = (
        {"image_embeds": image_embeds}
        if (ip_adapter_image is not None or ip_adapter_image_embeds is not None)
        else None
    )

    # 6.2 Optionally get Guidance Scale Embedding
    timestep_cond = None
    if self.unet.config.time_cond_proj_dim is not None:
        guidance_scale_tensor = torch.tensor(self.guidance_scale - 1).repeat(batch_size * num_images_per_prompt)
        timestep_cond = self.get_guidance_scale_embedding(
            guidance_scale_tensor, embedding_dim=self.unet.config.time_cond_proj_dim
        ).to(device=device, dtype=latents.dtype)

    # 7. Denoising loop
    num_warmup_steps = len(timesteps) - num_inference_steps * self.scheduler.order
    self._num_timesteps = len(timesteps)

    param_dict = {
        # "prompt_embeds": prompt_embeds, 

        "timesteps": timesteps, 

        "added_cond_kwargs": added_cond_kwargs, 

        "timestep_cond": timestep_cond
    }
    
    # `prepare_everything()` done
    return (
        prompt_embeds, 
        param_dict, 
        latents
    )


def get_noise_pred(
    self, 

    param_dict: Dict, 
    prompt_emb_list: List[torch.Tensor], 

    # latent_list.shape = (batch_size, 4, latent_height, latent_width)
    latent_list: torch.Tensor = None, 
    
    timestep_list: Optional[Union[torch.Tensor, List[torch.Tensor]]] = None, 
    timestep_idx_list: Optional[Union[int, List[int]]] = None, 
) -> Tuple[
    torch.Tensor,  # noise_pred.shape = (batch_size, 4, latent_height, latent_width)
    torch.Tensor  # timestep.shape = (batch_size, )
]:
    """
    Func:
        Denoising for a single step at timestep `t`. 

    Ret:
        `noise_pred` (`torch.Tensor`): The predicted noise residual. 
        `timestep` (`torch.Tensor`): The timestep derived from `timestep_list` or `timestep_idx_list`. 
    """

    if isinstance(prompt_emb_list, list):
        prompt_emb_list = torch.stack(prompt_emb_list)

    if isinstance(latent_list, list):
        latent_list = torch.stack(latent_list)

    # prompt_embeds.shape = (1, 77, 1024)
    # prompt_embeds = param_dict["prompt_embeds"]

    timesteps = param_dict["timesteps"]
    added_cond_kwargs = param_dict["added_cond_kwargs"]
    timestep_cond = param_dict["timestep_cond"]

    batch_size = latent_list.shape[0]
    num_prompt_emb = len(prompt_emb_list)

    if self.do_classifier_free_guidance:
        if num_prompt_emb != batch_size * 2:
            raise ValueError(
                f"Enable CFG, the shape `prompt_emb_list` does not match the shape of `latent_list`, "
                f"got `{prompt_emb_list.shape}` and `{latent_list.shape}`. "
            )
    else:
        if num_prompt_emb != batch_size:
            raise ValueError(
                f"Disable CFG, the shape `prompt_emb_list` does not match the shape of `latent_list`, "
                f"got `{prompt_emb_list.shape}` and `{latent_list.shape}`. "
            )
    
    prompt_embeds = prompt_emb_list
    
    timestep = prepare_timestep(
        pipeline = self, 

        timesteps = timesteps, 

        target_length = batch_size, 

        timestep_list = timestep_list, 
        timestep_idx_list = timestep_idx_list
    )
    
    batch_size_prompt_emb = prompt_embeds.shape[0]
    
    # expand the latents if we are doing classifier free guidance
    if self.do_classifier_free_guidance:
        latent_model_input = torch.cat([latent_list] * 2)
        
        timestep_model_input = timestep.repeat(2)

        if batch_size_prompt_emb != batch_size * 2:
            prompt_emb_model_input = torch.vstack(
                [
                    prompt_embeds[0].unsqueeze(0) \
                        .repeat(batch_size, 1, 1), 
                    prompt_embeds[1].unsqueeze(0) \
                        .repeat(batch_size, 1, 1)
                ]
            )
        else:
            prompt_emb_model_input = prompt_embeds
    else:
        latent_model_input = latent_list

        timestep_model_input = timestep

        if batch_size_prompt_emb != batch_size:
            prompt_emb_model_input = prompt_embeds.repeat(batch_size, 1, 1)
        else:
            prompt_emb_model_input = prompt_embeds
    
    latent_model_input = self.scheduler.scale_model_input(latent_model_input, timestep) 
    
    # predict the noise residual
    noise_pred = self.unet(
        latent_model_input,
        timestep = timestep_model_input,
        encoder_hidden_states = prompt_emb_model_input,
        timestep_cond = timestep_cond,
        cross_attention_kwargs = self.cross_attention_kwargs,
        added_cond_kwargs = added_cond_kwargs,
        return_dict = False
    )[0]

    # perform guidance
    if self.do_classifier_free_guidance:
        noise_pred_uncond, noise_pred_text = noise_pred.chunk(2)
        noise_pred = noise_pred_uncond + self.guidance_scale * (noise_pred_text - noise_pred_uncond)

    if self.do_classifier_free_guidance and self.guidance_rescale > 0.0:
        # Based on 3.4. in https://arxiv.org/pdf/2305.08891.pdf
        noise_pred = rescale_noise_cfg(noise_pred, noise_pred_text, guidance_rescale=self.guidance_rescale)

    # ---------= [Clean Up] =---------
    del prompt_embeds
    del latent_model_input, timestep_model_input, prompt_emb_model_input
    gc.collect()
    torch.cuda.empty_cache()

    # `get_noise_pred()` done
    return (
        # noise_pred.shape = (batch_size, 4, latent_height, latent_width)
        noise_pred, 

        # timestep.shape = (batch_size, )
        timestep
    )


def step(
    self, 

    # latent_list.shape = (batch_size, 4, latent_height, latent_width)
    latent_list: torch.Tensor, 

    # noise_pred.shape = (batch_size, 4, latent_height, latent_width)
    noise_pred: torch.Tensor, 

    # timestep.shape = (batch_size, )
    timestep: Optional[torch.Tensor] = None, 
    prev_timestep: Optional[torch.Tensor] = None, 

    eta_list: Union[float, torch.Tensor] = None, 
    eps: torch.Tensor = None
) -> torch.Tensor:
    """
    NB:
        The input `latent_list` will be modified. 
    
    Func:
        Denoising `latent_list` from `timestep` to `prev_timestep`. 

    Ret:
        `latent_list` (`Dict`): The list of previous latents. 
    """
    
    if not isinstance(eta_list, torch.Tensor):
        eta_list = torch.tensor(
            [eta_list], 

            dtype = latent_list.dtype, 
            device = latent_list.device
        )

    num_eta = len(eta_list)
    num_latent = len(latent_list)

    if num_eta == 1:
        eta_list = eta_list.repeat(
            (num_latent, *eta_list.shape[1: ])
        )
    elif num_eta != num_latent:
        raise ValueError(
            f"The length of `eta_list` does not match the length of `latent_list`, "
            f"got `{num_eta}` and `{num_latent}`. "
        )

    # compute the previous noisy sample x_t -> x_t-1
    latent_list = self.scheduler.step(
        model_output = noise_pred, 

        timestep = timestep, 
        prev_timestep = prev_timestep, 

        sample = latent_list, 

        eta = eta_list, 
        variance_noise = eps, 

        return_dict = False
    )[0]

    # `step()` done
    return latent_list


def register_pipeline_stable_diffusion(
    pipeline, 
    **kwargs
):
    """
    Func:
        Register custom methods: `forward()`, `prepare_everything()`, `step()`. 
    """

    pipeline.forward = MethodType(
        forward, 
        pipeline
    )

    pipeline.prepare_everything = MethodType(
        prepare_everything, 
        pipeline
    )

    pipeline.get_noise_pred = MethodType(
        get_noise_pred, 
        pipeline
    )

    pipeline.step = MethodType(
        step, 
        pipeline
    )

    # `register_pipeline_stable_diffusion()` done
    pass
