| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| from contextlib import contextmanager |
|
|
| from ..utils.logging import get_logger |
|
|
|
|
| logger = get_logger(__name__) |
|
|
|
|
| class CacheMixin: |
| r""" |
| A class for enable/disabling caching techniques on diffusion models. |
| |
| Supported caching techniques: |
| - [Pyramid Attention Broadcast](https://huggingface.co/papers/2408.12588) |
| - [FasterCache](https://huggingface.co/papers/2410.19355) |
| - [FirstBlockCache](https://github.com/chengzeyi/ParaAttention/blob/7a266123671b55e7e5a2fe9af3121f07a36afc78/README.md#first-block-cache-our-dynamic-caching) |
| """ |
|
|
| _cache_config = None |
|
|
| @property |
| def is_cache_enabled(self) -> bool: |
| return self._cache_config is not None |
|
|
| def enable_cache(self, config) -> None: |
| r""" |
| Enable caching techniques on the model. |
| |
| Args: |
| config (`PyramidAttentionBroadcastConfig | FasterCacheConfig | FirstBlockCacheConfig`): |
| The configuration for applying the caching technique. Currently supported caching techniques are: |
| - [`~hooks.PyramidAttentionBroadcastConfig`] |
| - [`~hooks.FasterCacheConfig`] |
| - [`~hooks.FirstBlockCacheConfig`] |
| |
| Example: |
| |
| ```python |
| >>> import torch |
| >>> from diffusers import CogVideoXPipeline, PyramidAttentionBroadcastConfig |
| |
| >>> pipe = CogVideoXPipeline.from_pretrained("THUDM/CogVideoX-5b", torch_dtype=torch.bfloat16) |
| >>> pipe.to("cuda") |
| |
| >>> config = PyramidAttentionBroadcastConfig( |
| ... spatial_attention_block_skip_range=2, |
| ... spatial_attention_timestep_skip_range=(100, 800), |
| ... current_timestep_callback=lambda: pipe.current_timestep, |
| ... ) |
| >>> pipe.transformer.enable_cache(config) |
| ``` |
| """ |
|
|
| from ..hooks import ( |
| FasterCacheConfig, |
| FirstBlockCacheConfig, |
| MagCacheConfig, |
| PyramidAttentionBroadcastConfig, |
| TaylorSeerCacheConfig, |
| apply_faster_cache, |
| apply_first_block_cache, |
| apply_mag_cache, |
| apply_pyramid_attention_broadcast, |
| apply_taylorseer_cache, |
| ) |
|
|
| if self.is_cache_enabled: |
| raise ValueError( |
| f"Caching has already been enabled with {type(self._cache_config)}. To apply a new caching technique, please disable the existing one first." |
| ) |
|
|
| if isinstance(config, FasterCacheConfig): |
| apply_faster_cache(self, config) |
| elif isinstance(config, FirstBlockCacheConfig): |
| apply_first_block_cache(self, config) |
| elif isinstance(config, MagCacheConfig): |
| apply_mag_cache(self, config) |
| elif isinstance(config, PyramidAttentionBroadcastConfig): |
| apply_pyramid_attention_broadcast(self, config) |
| elif isinstance(config, TaylorSeerCacheConfig): |
| apply_taylorseer_cache(self, config) |
| else: |
| raise ValueError(f"Cache config {type(config)} is not supported.") |
|
|
| self._cache_config = config |
|
|
| def disable_cache(self) -> None: |
| from ..hooks import ( |
| FasterCacheConfig, |
| FirstBlockCacheConfig, |
| HookRegistry, |
| MagCacheConfig, |
| PyramidAttentionBroadcastConfig, |
| TaylorSeerCacheConfig, |
| ) |
| from ..hooks.faster_cache import _FASTER_CACHE_BLOCK_HOOK, _FASTER_CACHE_DENOISER_HOOK |
| from ..hooks.first_block_cache import _FBC_BLOCK_HOOK, _FBC_LEADER_BLOCK_HOOK |
| from ..hooks.mag_cache import _MAG_CACHE_BLOCK_HOOK, _MAG_CACHE_LEADER_BLOCK_HOOK |
| from ..hooks.pyramid_attention_broadcast import _PYRAMID_ATTENTION_BROADCAST_HOOK |
| from ..hooks.taylorseer_cache import _TAYLORSEER_CACHE_HOOK |
|
|
| if self._cache_config is None: |
| logger.warning("Caching techniques have not been enabled, so there's nothing to disable.") |
| return |
|
|
| registry = HookRegistry.check_if_exists_or_initialize(self) |
| if isinstance(self._cache_config, FasterCacheConfig): |
| registry.remove_hook(_FASTER_CACHE_DENOISER_HOOK, recurse=True) |
| registry.remove_hook(_FASTER_CACHE_BLOCK_HOOK, recurse=True) |
| elif isinstance(self._cache_config, FirstBlockCacheConfig): |
| registry.remove_hook(_FBC_LEADER_BLOCK_HOOK, recurse=True) |
| registry.remove_hook(_FBC_BLOCK_HOOK, recurse=True) |
| elif isinstance(self._cache_config, MagCacheConfig): |
| registry.remove_hook(_MAG_CACHE_LEADER_BLOCK_HOOK, recurse=True) |
| registry.remove_hook(_MAG_CACHE_BLOCK_HOOK, recurse=True) |
| elif isinstance(self._cache_config, PyramidAttentionBroadcastConfig): |
| registry.remove_hook(_PYRAMID_ATTENTION_BROADCAST_HOOK, recurse=True) |
| elif isinstance(self._cache_config, TaylorSeerCacheConfig): |
| registry.remove_hook(_TAYLORSEER_CACHE_HOOK, recurse=True) |
| else: |
| raise ValueError(f"Cache config {type(self._cache_config)} is not supported.") |
|
|
| self._cache_config = None |
|
|
| def _reset_stateful_cache(self, recurse: bool = True) -> None: |
| from ..hooks import HookRegistry |
|
|
| HookRegistry.check_if_exists_or_initialize(self).reset_stateful_hooks(recurse=recurse) |
|
|
| @contextmanager |
| def cache_context(self, name: str): |
| r"""Context manager that provides additional methods for cache management.""" |
| from ..hooks import HookRegistry |
|
|
| registry = HookRegistry.check_if_exists_or_initialize(self) |
| registry._set_context(name) |
|
|
| yield |
|
|
| registry._set_context(None) |
|
|