mirror of
https://github.com/AUTOMATIC1111/stable-diffusion-webui.git
synced 2025-08-04 11:12:35 +00:00
Deduplicate tiled inference code from SwinIR/ScuNET
This commit is contained in:
@@ -6,7 +6,7 @@ import torch
|
||||
import tqdm
|
||||
from PIL import Image
|
||||
|
||||
from modules import images
|
||||
from modules import images, shared
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
@@ -68,3 +68,73 @@ def upscale_with_model(
|
||||
overlap=grid.overlap * scale_factor,
|
||||
)
|
||||
return images.combine_grid(newgrid)
|
||||
|
||||
|
||||
def tiled_upscale_2(
|
||||
img,
|
||||
model,
|
||||
*,
|
||||
tile_size: int,
|
||||
tile_overlap: int,
|
||||
scale: int,
|
||||
device,
|
||||
desc="Tiled upscale",
|
||||
):
|
||||
# Alternative implementation of `upscale_with_model` originally used by
|
||||
# SwinIR and ScuNET. It differs from `upscale_with_model` in that tiling and
|
||||
# weighting is done in PyTorch space, as opposed to `images.Grid` doing it in
|
||||
# Pillow space without weighting.
|
||||
b, c, h, w = img.size()
|
||||
tile_size = min(tile_size, h, w)
|
||||
|
||||
if tile_size <= 0:
|
||||
logger.debug("Upscaling %s without tiling", img.shape)
|
||||
return model(img)
|
||||
|
||||
stride = tile_size - tile_overlap
|
||||
h_idx_list = list(range(0, h - tile_size, stride)) + [h - tile_size]
|
||||
w_idx_list = list(range(0, w - tile_size, stride)) + [w - tile_size]
|
||||
result = torch.zeros(
|
||||
b,
|
||||
c,
|
||||
h * scale,
|
||||
w * scale,
|
||||
device=device,
|
||||
).type_as(img)
|
||||
weights = torch.zeros_like(result)
|
||||
logger.debug("Upscaling %s to %s with tiles", img.shape, result.shape)
|
||||
with tqdm.tqdm(total=len(h_idx_list) * len(w_idx_list), desc=desc) as pbar:
|
||||
for h_idx in h_idx_list:
|
||||
if shared.state.interrupted or shared.state.skipped:
|
||||
break
|
||||
|
||||
for w_idx in w_idx_list:
|
||||
if shared.state.interrupted or shared.state.skipped:
|
||||
break
|
||||
|
||||
in_patch = img[
|
||||
...,
|
||||
h_idx : h_idx + tile_size,
|
||||
w_idx : w_idx + tile_size,
|
||||
]
|
||||
out_patch = model(in_patch)
|
||||
|
||||
result[
|
||||
...,
|
||||
h_idx * scale : (h_idx + tile_size) * scale,
|
||||
w_idx * scale : (w_idx + tile_size) * scale,
|
||||
].add_(out_patch)
|
||||
|
||||
out_patch_mask = torch.ones_like(out_patch)
|
||||
|
||||
weights[
|
||||
...,
|
||||
h_idx * scale : (h_idx + tile_size) * scale,
|
||||
w_idx * scale : (w_idx + tile_size) * scale,
|
||||
].add_(out_patch_mask)
|
||||
|
||||
pbar.update(1)
|
||||
|
||||
output = result.div_(weights)
|
||||
|
||||
return output
|
||||
|
Reference in New Issue
Block a user