hypernetwork.py 13.9 KB
Newer Older
A
AUTOMATIC 已提交
1 2 3 4 5 6
import datetime
import glob
import html
import os
import sys
import traceback
D
update  
discus0434 已提交
7 8
import tqdm
import csv
A
AUTOMATIC 已提交
9 10

import torch
D
update  
discus0434 已提交
11

12
from ldm.util import default
D
update  
discus0434 已提交
13 14 15 16 17
from modules import devices, shared, processing, sd_models
import torch
from torch import einsum
from einops import rearrange, repeat
import modules.textual_inversion.dataset
18
from modules.textual_inversion import textual_inversion
19
from modules.textual_inversion.learn_schedule import LearnRateScheduler
20 21


A
AUTOMATIC 已提交
22
class HypernetworkModule(torch.nn.Module):
A
AUTOMATIC 已提交
23 24
    multiplier = 1.0

25
    def __init__(self, dim, state_dict=None, layer_structure=None, add_layer_norm=False):
A
AUTOMATIC 已提交
26
        super().__init__()
27 28 29 30

        assert layer_structure is not None, "layer_structure mut not be None"
        assert layer_structure[0] == 1, "Multiplier Sequence should start with size 1!"
        assert layer_structure[-1] == 1, "Multiplier Sequence should end with size 1!"
A
AUTOMATIC 已提交
31

32 33 34
        linears = []
        for i in range(len(layer_structure) - 1):
            linears.append(torch.nn.Linear(int(dim * layer_structure[i]), int(dim * layer_structure[i+1])))
35
            if add_layer_norm:
36 37 38
                linears.append(torch.nn.LayerNorm(int(dim * layer_structure[i+1])))

        self.linear = torch.nn.Sequential(*linears)
A
AUTOMATIC 已提交
39 40

        if state_dict is not None:
41 42
            self.fix_old_state_dict(state_dict)
            self.load_state_dict(state_dict)
A
AUTOMATIC 已提交
43
        else:
44
            for layer in self.linear:
45
                layer.weight.data.normal_(mean=0.0, std=0.01)
46
                layer.bias.data.zero_()
A
AUTOMATIC 已提交
47 48 49

        self.to(devices.device)

50 51 52 53 54 55 56 57 58 59 60 61 62 63 64
    def fix_old_state_dict(self, state_dict):
        changes = {
            'linear1.bias': 'linear.0.bias',
            'linear1.weight': 'linear.0.weight',
            'linear2.bias': 'linear.1.bias',
            'linear2.weight': 'linear.1.weight',
        }

        for fr, to in changes.items():
            x = state_dict.get(fr, None)
            if x is None:
                continue

            del state_dict[fr]
            state_dict[to] = x
65

A
AUTOMATIC 已提交
66
    def forward(self, x):
67 68 69
        return x + self.linear(x) * self.multiplier

    def trainables(self):
70
        layer_structure = []
71
        for layer in self.linear:
72 73
            layer_structure += [layer.weight, layer.bias]
        return layer_structure
A
AUTOMATIC 已提交
74 75 76 77


def apply_strength(value=None):
    HypernetworkModule.multiplier = value if value is not None else shared.opts.sd_hypernetwork_strength
A
AUTOMATIC 已提交
78 79 80 81 82 83


class Hypernetwork:
    filename = None
    name = None

84
    def __init__(self, name=None, enable_sizes=None, layer_structure=None, add_layer_norm=False):
A
AUTOMATIC 已提交
85 86 87 88 89 90
        self.filename = None
        self.name = name
        self.layers = {}
        self.step = 0
        self.sd_checkpoint = None
        self.sd_checkpoint_name = None
91 92
        self.layer_structure = layer_structure
        self.add_layer_norm = add_layer_norm
A
AUTOMATIC 已提交
93

94
        for size in enable_sizes or []:
95 96 97 98
            self.layers[size] = (
                HypernetworkModule(size, None, self.layer_structure, self.add_layer_norm),
                HypernetworkModule(size, None, self.layer_structure, self.add_layer_norm),
            )
A
AUTOMATIC 已提交
99 100 101 102 103 104 105

    def weights(self):
        res = []

        for k, layers in self.layers.items():
            for layer in layers:
                layer.train()
106
                res += layer.trainables()
A
AUTOMATIC 已提交
107 108 109 110 111 112 113 114 115 116 117

        return res

    def save(self, filename):
        state_dict = {}

        for k, v in self.layers.items():
            state_dict[k] = (v[0].state_dict(), v[1].state_dict())

        state_dict['step'] = self.step
        state_dict['name'] = self.name
118 119
        state_dict['layer_structure'] = self.layer_structure
        state_dict['is_layer_norm'] = self.add_layer_norm
A
AUTOMATIC 已提交
120 121 122 123 124 125 126 127 128 129 130 131
        state_dict['sd_checkpoint'] = self.sd_checkpoint
        state_dict['sd_checkpoint_name'] = self.sd_checkpoint_name

        torch.save(state_dict, filename)

    def load(self, filename):
        self.filename = filename
        if self.name is None:
            self.name = os.path.splitext(os.path.basename(filename))[0]

        state_dict = torch.load(filename, map_location='cpu')

132 133 134
        self.layer_structure = state_dict.get('layer_structure', [1, 2, 1])
        self.add_layer_norm = state_dict.get('is_layer_norm', False)

A
AUTOMATIC 已提交
135 136
        for size, sd in state_dict.items():
            if type(size) == int:
137
                self.layers[size] = (
138 139
                    HypernetworkModule(size, sd[0], self.layer_structure, self.add_layer_norm),
                    HypernetworkModule(size, sd[1], self.layer_structure, self.add_layer_norm),
140
                )
A
AUTOMATIC 已提交
141 142 143 144 145 146 147

        self.name = state_dict.get('name', self.name)
        self.step = state_dict.get('step', 0)
        self.sd_checkpoint = state_dict.get('sd_checkpoint', None)
        self.sd_checkpoint_name = state_dict.get('sd_checkpoint_name', None)


A
AUTOMATIC 已提交
148
def list_hypernetworks(path):
A
AUTOMATIC 已提交
149
    res = {}
A
AUTOMATIC 已提交
150 151 152 153
    for filename in glob.iglob(os.path.join(path, '**/*.pt'), recursive=True):
        name = os.path.splitext(os.path.basename(filename))[0]
        res[name] = filename
    return res
A
AUTOMATIC 已提交
154

A
AUTOMATIC 已提交
155 156 157 158 159

def load_hypernetwork(filename):
    path = shared.hypernetworks.get(filename, None)
    if path is not None:
        print(f"Loading hypernetwork {filename}")
A
AUTOMATIC 已提交
160
        try:
A
AUTOMATIC 已提交
161 162 163
            shared.loaded_hypernetwork = Hypernetwork()
            shared.loaded_hypernetwork.load(path)

A
AUTOMATIC 已提交
164
        except Exception:
A
AUTOMATIC 已提交
165
            print(f"Error loading hypernetwork {path}", file=sys.stderr)
A
AUTOMATIC 已提交
166
            print(traceback.format_exc(), file=sys.stderr)
A
AUTOMATIC 已提交
167 168 169
    else:
        if shared.loaded_hypernetwork is not None:
            print(f"Unloading hypernetwork")
A
AUTOMATIC 已提交
170

A
AUTOMATIC 已提交
171
        shared.loaded_hypernetwork = None
A
AUTOMATIC 已提交
172 173


M
Milly 已提交
174 175 176 177 178 179 180 181 182 183 184
def find_closest_hypernetwork_name(search: str):
    if not search:
        return None
    search = search.lower()
    applicable = [name for name in shared.hypernetworks if search in name.lower()]
    if not applicable:
        return None
    applicable = sorted(applicable, key=lambda name: len(name))
    return applicable[0]


A
AUTOMATIC 已提交
185 186
def apply_hypernetwork(hypernetwork, context, layer=None):
    hypernetwork_layers = (hypernetwork.layers if hypernetwork is not None else {}).get(context.shape[2], None)
A
AUTOMATIC 已提交
187

A
AUTOMATIC 已提交
188 189
    if hypernetwork_layers is None:
        return context, context
A
AUTOMATIC 已提交
190

A
AUTOMATIC 已提交
191 192 193
    if layer is not None:
        layer.hyper_k = hypernetwork_layers[0]
        layer.hyper_v = hypernetwork_layers[1]
A
AUTOMATIC 已提交
194

A
AUTOMATIC 已提交
195 196 197
    context_k = hypernetwork_layers[0](context)
    context_v = hypernetwork_layers[1](context)
    return context_k, context_v
A
AUTOMATIC 已提交
198 199


A
AUTOMATIC 已提交
200 201 202 203 204
def attention_CrossAttention_forward(self, x, context=None, mask=None):
    h = self.heads

    q = self.to_q(x)
    context = default(context, x)
A
AUTOMATIC 已提交
205

A
AUTOMATIC 已提交
206
    context_k, context_v = apply_hypernetwork(shared.loaded_hypernetwork, context, self)
A
AUTOMATIC 已提交
207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227
    k = self.to_k(context_k)
    v = self.to_v(context_v)

    q, k, v = map(lambda t: rearrange(t, 'b n (h d) -> (b h) n d', h=h), (q, k, v))

    sim = einsum('b i d, b j d -> b i j', q, k) * self.scale

    if mask is not None:
        mask = rearrange(mask, 'b ... -> b (...)')
        max_neg_value = -torch.finfo(sim.dtype).max
        mask = repeat(mask, 'b j -> (b h) () j', h=h)
        sim.masked_fill_(~mask, max_neg_value)

    # attention, what we cannot get enough of
    attn = sim.softmax(dim=-1)

    out = einsum('b i j, b j d -> b i d', attn, v)
    out = rearrange(out, '(b h) n d -> b n (h d)', h=h)
    return self.to_out(out)


228 229 230 231 232 233 234 235 236 237 238 239 240 241
def stack_conds(conds):
    if len(conds) == 1:
        return torch.stack(conds)

    # same as in reconstruct_multicond_batch
    token_count = max([x.shape[0] for x in conds])
    for i in range(len(conds)):
        if conds[i].shape[0] != token_count:
            last_vector = conds[i][-1:]
            last_vector_repeated = last_vector.repeat([token_count - conds[i].shape[0], 1])
            conds[i] = torch.vstack([conds[i], last_vector_repeated])

    return torch.stack(conds)

242

243
def train_hypernetwork(hypernetwork_name, learn_rate, batch_size, data_root, log_directory, training_width, training_height, steps, create_image_every, save_hypernetwork_every, template_file, preview_from_txt2img, preview_prompt, preview_negative_prompt, preview_steps, preview_sampler_index, preview_cfg_scale, preview_seed, preview_width, preview_height):
A
AUTOMATIC 已提交
244
    assert hypernetwork_name, 'hypernetwork not selected'
A
AUTOMATIC 已提交
245

A
AUTOMATIC 已提交
246 247 248
    path = shared.hypernetworks.get(hypernetwork_name, None)
    shared.loaded_hypernetwork = Hypernetwork()
    shared.loaded_hypernetwork.load(path)
A
AUTOMATIC 已提交
249 250 251 252 253 254 255

    shared.state.textinfo = "Initializing hypernetwork training..."
    shared.state.job_count = steps

    filename = os.path.join(shared.cmd_opts.hypernetwork_dir, f'{hypernetwork_name}.pt')

    log_directory = os.path.join(log_directory, datetime.datetime.now().strftime("%Y-%m-%d"), hypernetwork_name)
256
    unload = shared.opts.unload_models_when_training
A
AUTOMATIC 已提交
257 258 259 260 261 262 263 264 265 266 267 268 269 270 271

    if save_hypernetwork_every > 0:
        hypernetwork_dir = os.path.join(log_directory, "hypernetworks")
        os.makedirs(hypernetwork_dir, exist_ok=True)
    else:
        hypernetwork_dir = None

    if create_image_every > 0:
        images_dir = os.path.join(log_directory, "images")
        os.makedirs(images_dir, exist_ok=True)
    else:
        images_dir = None

    shared.state.textinfo = f"Preparing dataset from {html.escape(data_root)}..."
    with torch.autocast("cuda"):
272
        ds = modules.textual_inversion.dataset.PersonalizedBase(data_root=data_root, width=training_width, height=training_height, repeats=shared.opts.training_image_repeats_per_epoch, placeholder_token=hypernetwork_name, model=shared.sd_model, device=devices.device, template_file=template_file, include_cond=True, batch_size=batch_size)
273 274 275
    if unload:
        shared.sd_model.cond_stage_model.to(devices.cpu)
        shared.sd_model.first_stage_model.to(devices.cpu)
A
AUTOMATIC 已提交
276

A
AUTOMATIC 已提交
277
    hypernetwork = shared.loaded_hypernetwork
A
AUTOMATIC 已提交
278 279 280 281 282 283 284 285 286 287 288 289 290
    weights = hypernetwork.weights()
    for weight in weights:
        weight.requires_grad = True

    losses = torch.zeros((32,))

    last_saved_file = "<none>"
    last_saved_image = "<none>"

    ititial_step = hypernetwork.step or 0
    if ititial_step > steps:
        return hypernetwork, filename

291 292
    scheduler = LearnRateScheduler(learn_rate, steps, ititial_step)
    optimizer = torch.optim.AdamW(weights, lr=scheduler.learn_rate)
A
AUTOMATIC 已提交
293

A
AUTOMATIC 已提交
294
    pbar = tqdm.tqdm(enumerate(ds), total=steps - ititial_step)
295
    for i, entries in pbar:
A
AUTOMATIC 已提交
296 297
        hypernetwork.step = i + ititial_step

298 299 300
        scheduler.apply(optimizer, hypernetwork.step)
        if scheduler.finished:
            break
A
AUTOMATIC 已提交
301 302 303 304 305

        if shared.state.interrupted:
            break

        with torch.autocast("cuda"):
306
            c = stack_conds([entry.cond for entry in entries]).to(devices.device)
D
update  
discus0434 已提交
307
            # c = torch.vstack([entry.cond for entry in entries]).to(devices.device)
308 309
            x = torch.stack([entry.latent for entry in entries]).to(devices.device)
            loss = shared.sd_model(x, c)[0]
A
AUTOMATIC 已提交
310
            del x
311
            del c
A
AUTOMATIC 已提交
312 313 314 315 316 317

            losses[hypernetwork.step % losses.shape[0]] = loss.item()

            optimizer.zero_grad()
            loss.backward()
            optimizer.step()
318 319 320 321
        mean_loss = losses.mean()
        if torch.isnan(mean_loss):
            raise RuntimeError("Loss diverged.")
        pbar.set_description(f"loss: {mean_loss:.7f}")
A
AUTOMATIC 已提交
322 323 324 325 326

        if hypernetwork.step > 0 and hypernetwork_dir is not None and hypernetwork.step % save_hypernetwork_every == 0:
            last_saved_file = os.path.join(hypernetwork_dir, f'{hypernetwork_name}-{hypernetwork.step}.pt')
            hypernetwork.save(last_saved_file)

327
        textual_inversion.write_loss(log_directory, "hypernetwork_loss.csv", hypernetwork.step, len(ds), {
328
            "loss": f"{mean_loss:.7f}",
D
update  
discus0434 已提交
329
            "learn_rate": scheduler.learn_rate
330
        })
331

A
AUTOMATIC 已提交
332 333 334
        if hypernetwork.step > 0 and images_dir is not None and hypernetwork.step % create_image_every == 0:
            last_saved_image = os.path.join(images_dir, f'{hypernetwork_name}-{hypernetwork.step}.png')

335 336 337 338
            optimizer.zero_grad()
            shared.sd_model.cond_stage_model.to(devices.device)
            shared.sd_model.first_stage_model.to(devices.device)

A
AUTOMATIC 已提交
339 340 341 342 343 344
            p = processing.StableDiffusionProcessingTxt2Img(
                sd_model=shared.sd_model,
                do_not_save_grid=True,
                do_not_save_samples=True,
            )

345 346 347 348 349 350 351 352 353 354
            if preview_from_txt2img:
                p.prompt = preview_prompt
                p.negative_prompt = preview_negative_prompt
                p.steps = preview_steps
                p.sampler_index = preview_sampler_index
                p.cfg_scale = preview_cfg_scale
                p.seed = preview_seed
                p.width = preview_width
                p.height = preview_height
            else:
355
                p.prompt = entries[0].cond_text
356 357 358 359
                p.steps = 20

            preview_text = p.prompt

A
AUTOMATIC 已提交
360
            processed = processing.process_images(p)
361
            image = processed.images[0] if len(processed.images)>0 else None
A
AUTOMATIC 已提交
362

363 364 365 366
            if unload:
                shared.sd_model.cond_stage_model.to(devices.cpu)
                shared.sd_model.first_stage_model.to(devices.cpu)

367 368 369 370
            if image is not None:
                shared.state.current_image = image
                image.save(last_saved_image)
                last_saved_image += f", prompt: {preview_text}"
A
AUTOMATIC 已提交
371 372 373 374 375

        shared.state.job_no = hypernetwork.step

        shared.state.textinfo = f"""
<p>
376
Loss: {mean_loss:.7f}<br/>
A
AUTOMATIC 已提交
377
Step: {hypernetwork.step}<br/>
378
Last prompt: {html.escape(entries[0].cond_text)}<br/>
A
AUTOMATIC 已提交
379 380 381 382 383 384 385 386 387 388 389 390 391 392
Last saved embedding: {html.escape(last_saved_file)}<br/>
Last saved image: {html.escape(last_saved_image)}<br/>
</p>
"""

    checkpoint = sd_models.select_checkpoint()

    hypernetwork.sd_checkpoint = checkpoint.hash
    hypernetwork.sd_checkpoint_name = checkpoint.model_name
    hypernetwork.save(filename)

    return hypernetwork, filename