mirror of
https://github.com/hpcaitech/Open-Sora.git
synced 2026-04-14 18:25:35 +02:00
343 lines
12 KiB
Python
343 lines
12 KiB
Python
# Adapted from PixArt
|
|
#
|
|
# Copyright (C) 2023 PixArt-alpha/PixArt-alpha
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Affero General Public License as published
|
|
# by the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU Affero General Public License for more details.
|
|
#
|
|
#
|
|
# This source code is licensed under the license found in the
|
|
# LICENSE file in the root directory of this source tree.
|
|
# --------------------------------------------------------
|
|
# References:
|
|
# PixArt: https://github.com/PixArt-alpha/PixArt-alpha
|
|
# DiT: https://github.com/facebookresearch/DiT/tree/main
|
|
# --------------------------------------------------------
|
|
|
|
import numpy as np
|
|
import torch
|
|
import torch.nn as nn
|
|
from einops import rearrange
|
|
from timm.models.layers import DropPath
|
|
from timm.models.vision_transformer import Mlp
|
|
|
|
# from .builder import MODELS
|
|
from opensora.acceleration.checkpoint import auto_grad_checkpoint
|
|
from opensora.models.layers.blocks import (
|
|
CaptionEmbedder,
|
|
KVCompressAttention,
|
|
MultiHeadCrossAttention,
|
|
PatchEmbed3D,
|
|
T2IFinalLayer,
|
|
TimestepEmbedder,
|
|
approx_gelu,
|
|
get_1d_sincos_pos_embed,
|
|
get_2d_sincos_pos_embed,
|
|
get_layernorm,
|
|
t2i_modulate,
|
|
)
|
|
from opensora.registry import MODELS
|
|
from opensora.utils.ckpt_utils import load_checkpoint
|
|
|
|
|
|
class PixArtBlock(nn.Module):
|
|
"""
|
|
A PixArt block with adaptive layer norm (adaLN-single) conditioning.
|
|
"""
|
|
|
|
def __init__(
|
|
self,
|
|
hidden_size,
|
|
num_heads,
|
|
mlp_ratio=4.0,
|
|
drop_path=0.0,
|
|
enable_flash_attn=False,
|
|
enable_layernorm_kernel=False,
|
|
enable_sequence_parallelism=False,
|
|
qk_norm=False,
|
|
sampling="conv",
|
|
sr_ratio=1,
|
|
):
|
|
super().__init__()
|
|
self.hidden_size = hidden_size
|
|
self.enable_flash_attn = enable_flash_attn
|
|
self._enable_sequence_parallelism = enable_sequence_parallelism
|
|
assert not enable_sequence_parallelism, "Sequence parallelism is not supported in this version."
|
|
|
|
self.attn_cls = KVCompressAttention
|
|
self.mha_cls = MultiHeadCrossAttention
|
|
|
|
self.norm1 = get_layernorm(hidden_size, eps=1e-6, affine=False, use_kernel=enable_layernorm_kernel)
|
|
self.attn = self.attn_cls(
|
|
hidden_size,
|
|
num_heads=num_heads,
|
|
qkv_bias=True,
|
|
enable_flash_attn=enable_flash_attn,
|
|
qk_norm=qk_norm,
|
|
sr_ratio=sr_ratio,
|
|
sampling=sampling,
|
|
attn_half=True,
|
|
)
|
|
self.cross_attn = self.mha_cls(hidden_size, num_heads)
|
|
self.norm2 = get_layernorm(hidden_size, eps=1e-6, affine=False, use_kernel=enable_layernorm_kernel)
|
|
self.mlp = Mlp(
|
|
in_features=hidden_size, hidden_features=int(hidden_size * mlp_ratio), act_layer=approx_gelu, drop=0
|
|
)
|
|
self.drop_path = DropPath(drop_path) if drop_path > 0.0 else nn.Identity()
|
|
self.scale_shift_table = nn.Parameter(torch.randn(6, hidden_size) / hidden_size**0.5)
|
|
self.sampling = sampling
|
|
self.sr_ratio = sr_ratio
|
|
|
|
def forward(self, x, y, t, hw, mask=None):
|
|
B, N, C = x.shape
|
|
|
|
shift_msa, scale_msa, gate_msa, shift_mlp, scale_mlp, gate_mlp = (
|
|
self.scale_shift_table[None] + t.reshape(B, 6, -1)
|
|
).chunk(6, dim=1)
|
|
x = x + self.drop_path(
|
|
gate_msa * self.attn(t2i_modulate(self.norm1(x), shift_msa, scale_msa), HW=hw).reshape(B, N, C)
|
|
)
|
|
x = x + self.cross_attn(x, y, mask)
|
|
x = x + self.drop_path(gate_mlp * self.mlp(t2i_modulate(self.norm2(x), shift_mlp, scale_mlp)))
|
|
|
|
return x
|
|
|
|
|
|
@MODELS.register_module()
|
|
class PixArt_Sigma(nn.Module):
|
|
"""
|
|
Diffusion model with a Transformer backbone.
|
|
"""
|
|
|
|
def __init__(
|
|
self,
|
|
input_size=(1, 32, 32),
|
|
in_channels=4,
|
|
patch_size=(1, 2, 2),
|
|
hidden_size=1152,
|
|
depth=28,
|
|
num_heads=16,
|
|
mlp_ratio=4.0,
|
|
class_dropout_prob=0.1,
|
|
pred_sigma=True,
|
|
drop_path: float = 0.0,
|
|
no_temporal_pos_emb=False,
|
|
caption_channels=4096,
|
|
model_max_length=120,
|
|
dtype=torch.float32,
|
|
freeze=None,
|
|
qk_norm=False,
|
|
space_scale=1.0,
|
|
time_scale=1.0,
|
|
enable_flash_attn=False,
|
|
enable_layernorm_kernel=False,
|
|
enable_sequence_parallelism=False,
|
|
kv_compress_config=None,
|
|
):
|
|
super().__init__()
|
|
assert enable_sequence_parallelism is False, "Sequence parallelism is not supported in this version."
|
|
self.pred_sigma = pred_sigma
|
|
self.in_channels = in_channels
|
|
self.out_channels = in_channels * 2 if pred_sigma else in_channels
|
|
self.hidden_size = hidden_size
|
|
self.patch_size = patch_size
|
|
self.input_size = input_size
|
|
num_patches = np.prod([input_size[i] // patch_size[i] for i in range(3)])
|
|
self.num_patches = num_patches
|
|
self.num_temporal = input_size[0] // patch_size[0]
|
|
self.num_spatial = num_patches // self.num_temporal
|
|
self.base_size = int(np.sqrt(self.num_spatial))
|
|
self.num_heads = num_heads
|
|
self.dtype = dtype
|
|
self.no_temporal_pos_emb = no_temporal_pos_emb
|
|
self.depth = depth
|
|
self.mlp_ratio = mlp_ratio
|
|
self.enable_flash_attn = enable_flash_attn
|
|
self.enable_layernorm_kernel = enable_layernorm_kernel
|
|
self.space_scale = space_scale
|
|
self.time_scale = time_scale
|
|
|
|
self.x_embedder = PatchEmbed3D(patch_size, in_channels, hidden_size)
|
|
self.t_embedder = TimestepEmbedder(hidden_size)
|
|
self.t_block = nn.Sequential(nn.SiLU(), nn.Linear(hidden_size, 6 * hidden_size, bias=True))
|
|
self.y_embedder = CaptionEmbedder(
|
|
in_channels=caption_channels,
|
|
hidden_size=hidden_size,
|
|
uncond_prob=class_dropout_prob,
|
|
act_layer=approx_gelu,
|
|
token_num=model_max_length,
|
|
)
|
|
|
|
self.register_buffer("pos_embed", self.get_spatial_pos_embed())
|
|
self.register_buffer("pos_embed_temporal", self.get_temporal_pos_embed())
|
|
|
|
drop_path = [x.item() for x in torch.linspace(0, drop_path, depth)] # stochastic depth decay rule
|
|
|
|
self.kv_compress_config = kv_compress_config
|
|
if kv_compress_config is None:
|
|
self.kv_compress_config = {
|
|
"sampling": None,
|
|
"scale_factor": 1,
|
|
"kv_compress_layer": [],
|
|
}
|
|
|
|
self.blocks = nn.ModuleList(
|
|
[
|
|
PixArtBlock(
|
|
hidden_size,
|
|
num_heads,
|
|
mlp_ratio=mlp_ratio,
|
|
drop_path=drop_path[i],
|
|
enable_flash_attn=enable_flash_attn,
|
|
enable_layernorm_kernel=enable_layernorm_kernel,
|
|
qk_norm=qk_norm,
|
|
sr_ratio=(
|
|
int(self.kv_compress_config["scale_factor"])
|
|
if i in self.kv_compress_config["kv_compress_layer"]
|
|
else 1
|
|
),
|
|
sampling=self.kv_compress_config["sampling"],
|
|
)
|
|
for i in range(depth)
|
|
]
|
|
)
|
|
self.final_layer = T2IFinalLayer(hidden_size, np.prod(self.patch_size), self.out_channels)
|
|
|
|
self.initialize_weights()
|
|
if freeze is not None:
|
|
assert freeze in ["text"]
|
|
if freeze == "text":
|
|
self.freeze_text()
|
|
|
|
def forward(self, x, timestep, y, mask=None):
|
|
"""
|
|
Forward pass of PixArt.
|
|
x: (N, C, H, W) tensor of spatial inputs (images or latent representations of images)
|
|
t: (N,) tensor of diffusion timesteps
|
|
y: (N, 1, 120, C) tensor of class labels
|
|
"""
|
|
x = x.to(self.dtype)
|
|
timestep = timestep.to(self.dtype)
|
|
y = y.to(self.dtype)
|
|
pos_embed = self.get_spatial_pos_embed((x.shape[-2], x.shape[-1])).to(x.dtype)
|
|
hw = (x.shape[-2] // self.patch_size[-2], x.shape[-1] // self.patch_size[-1])
|
|
|
|
# embedding
|
|
x = self.x_embedder(x) # (B, N, D)
|
|
x = rearrange(x, "b (t s) d -> b t s d", t=self.num_temporal, s=self.num_spatial)
|
|
x = x + pos_embed.to(x.device)
|
|
if not self.no_temporal_pos_emb:
|
|
x = rearrange(x, "b t s d -> b s t d")
|
|
x = x + self.pos_embed_temporal
|
|
x = rearrange(x, "b s t d -> b (t s) d")
|
|
else:
|
|
x = rearrange(x, "b t s d -> b (t s) d")
|
|
|
|
t = self.t_embedder(timestep, dtype=x.dtype) # (N, D)
|
|
t0 = self.t_block(t)
|
|
y = self.y_embedder(y, self.training) # (N, 1, L, D)
|
|
if mask is not None:
|
|
if mask.shape[0] != y.shape[0]:
|
|
mask = mask.repeat(y.shape[0] // mask.shape[0], 1)
|
|
mask = mask.squeeze(1).squeeze(1)
|
|
y = y.squeeze(1).masked_select(mask.unsqueeze(-1) != 0).view(1, -1, x.shape[-1])
|
|
y_lens = mask.sum(dim=1).tolist()
|
|
else:
|
|
y_lens = [y.shape[2]] * y.shape[0]
|
|
y = y.squeeze(1).view(1, -1, x.shape[-1])
|
|
|
|
# blocks
|
|
for block in self.blocks:
|
|
x = auto_grad_checkpoint(block, x, y, t0, hw, y_lens)
|
|
|
|
# final process
|
|
x = self.final_layer(x, t) # (N, T, patch_size ** 2 * out_channels)
|
|
x = self.unpatchify(x) # (N, out_channels, H, W)
|
|
|
|
# cast to float32 for better accuracy
|
|
x = x.to(torch.float32)
|
|
return x
|
|
|
|
def unpatchify(self, x):
|
|
c = self.out_channels
|
|
t, h, w = [self.input_size[i] // self.patch_size[i] for i in range(3)]
|
|
pt, ph, pw = self.patch_size
|
|
|
|
x = x.reshape(shape=(x.shape[0], t, h, w, pt, ph, pw, c))
|
|
x = rearrange(x, "n t h w r p q c -> n c t r h p w q")
|
|
imgs = x.reshape(shape=(x.shape[0], c, t * pt, h * ph, w * pw))
|
|
return imgs
|
|
|
|
def get_spatial_pos_embed(self, grid_size=None):
|
|
if grid_size is None:
|
|
grid_size = self.input_size[1:]
|
|
pos_embed = get_2d_sincos_pos_embed(
|
|
self.hidden_size,
|
|
(grid_size[0] // self.patch_size[1], grid_size[1] // self.patch_size[2]),
|
|
scale=self.space_scale,
|
|
base_size=self.base_size,
|
|
)
|
|
pos_embed = torch.from_numpy(pos_embed).float().unsqueeze(0).requires_grad_(False)
|
|
return pos_embed
|
|
|
|
def get_temporal_pos_embed(self):
|
|
pos_embed = get_1d_sincos_pos_embed(
|
|
self.hidden_size,
|
|
self.input_size[0] // self.patch_size[0],
|
|
scale=self.time_scale,
|
|
)
|
|
pos_embed = torch.from_numpy(pos_embed).float().unsqueeze(0).requires_grad_(False)
|
|
return pos_embed
|
|
|
|
def freeze_text(self):
|
|
for n, p in self.named_parameters():
|
|
if "cross_attn" in n:
|
|
p.requires_grad = False
|
|
|
|
def initialize_weights(self):
|
|
# Initialize transformer layers:
|
|
def _basic_init(module):
|
|
if isinstance(module, nn.Linear):
|
|
torch.nn.init.xavier_uniform_(module.weight)
|
|
if module.bias is not None:
|
|
nn.init.constant_(module.bias, 0)
|
|
|
|
self.apply(_basic_init)
|
|
|
|
# Initialize patch_embed like nn.Linear (instead of nn.Conv2d):
|
|
w = self.x_embedder.proj.weight.data
|
|
nn.init.xavier_uniform_(w.view([w.shape[0], -1]))
|
|
|
|
# Initialize timestep embedding MLP:
|
|
nn.init.normal_(self.t_embedder.mlp[0].weight, std=0.02)
|
|
nn.init.normal_(self.t_embedder.mlp[2].weight, std=0.02)
|
|
nn.init.normal_(self.t_block[1].weight, std=0.02)
|
|
|
|
# Initialize caption embedding MLP:
|
|
nn.init.normal_(self.y_embedder.y_proj.fc1.weight, std=0.02)
|
|
nn.init.normal_(self.y_embedder.y_proj.fc2.weight, std=0.02)
|
|
|
|
# Zero-out adaLN modulation layers in PixArt blocks:
|
|
for block in self.blocks:
|
|
nn.init.constant_(block.cross_attn.proj.weight, 0)
|
|
nn.init.constant_(block.cross_attn.proj.bias, 0)
|
|
|
|
# Zero-out output layers:
|
|
nn.init.constant_(self.final_layer.linear.weight, 0)
|
|
nn.init.constant_(self.final_layer.linear.bias, 0)
|
|
|
|
|
|
@MODELS.register_module("PixArt-Sigma-XL/2")
|
|
def PixArt_Sigma_XL_2(from_pretrained=None, **kwargs):
|
|
model = PixArt_Sigma(depth=28, hidden_size=1152, patch_size=(1, 2, 2), num_heads=16, **kwargs)
|
|
if from_pretrained is not None:
|
|
load_checkpoint(model, from_pretrained)
|
|
return model
|