2023-01-28 20:56:06 +00:00
|
|
|
from enum import IntEnum
|
2023-02-18 21:44:39 +00:00
|
|
|
from logging import getLogger
|
|
|
|
from typing import Any, Dict, List, Literal, Optional, Tuple, Union
|
2023-01-28 04:48:06 +00:00
|
|
|
|
2023-02-26 21:21:58 +00:00
|
|
|
from .torch_before_ort import GraphOptimizationLevel, SessionOptions
|
2023-02-18 21:44:39 +00:00
|
|
|
|
|
|
|
logger = getLogger(__name__)
|
2023-02-15 01:01:14 +00:00
|
|
|
|
2023-01-28 04:48:06 +00:00
|
|
|
|
2023-01-28 20:56:06 +00:00
|
|
|
class SizeChart(IntEnum):
|
2023-02-05 13:53:26 +00:00
|
|
|
mini = 128 # small tile for very expensive models
|
|
|
|
half = 256 # half tile for outpainting
|
|
|
|
auto = 512 # auto tile size
|
2023-01-28 20:56:06 +00:00
|
|
|
hd1k = 2**10
|
|
|
|
hd2k = 2**11
|
|
|
|
hd4k = 2**12
|
|
|
|
hd8k = 2**13
|
|
|
|
hd16k = 2**14
|
|
|
|
hd64k = 2**16
|
|
|
|
|
|
|
|
|
2023-02-12 00:00:18 +00:00
|
|
|
class TileOrder:
|
|
|
|
grid = "grid"
|
|
|
|
kernel = "kernel"
|
|
|
|
spiral = "spiral"
|
|
|
|
|
|
|
|
|
2023-01-28 04:48:06 +00:00
|
|
|
Param = Union[str, int, float]
|
|
|
|
Point = Tuple[int, int]
|
|
|
|
|
|
|
|
|
|
|
|
class Border:
|
|
|
|
def __init__(self, left: int, right: int, top: int, bottom: int) -> None:
|
|
|
|
self.left = left
|
|
|
|
self.right = right
|
|
|
|
self.top = top
|
|
|
|
self.bottom = bottom
|
|
|
|
|
2023-01-28 23:09:19 +00:00
|
|
|
def __str__(self) -> str:
|
2023-02-05 13:53:26 +00:00
|
|
|
return "%s %s %s %s" % (self.left, self.top, self.right, self.bottom)
|
2023-01-28 23:09:19 +00:00
|
|
|
|
2023-02-02 04:31:01 +00:00
|
|
|
def tojson(self):
|
|
|
|
return {
|
2023-02-05 13:53:26 +00:00
|
|
|
"left": self.left,
|
|
|
|
"right": self.right,
|
|
|
|
"top": self.top,
|
|
|
|
"bottom": self.bottom,
|
2023-02-02 04:31:01 +00:00
|
|
|
}
|
|
|
|
|
2023-02-18 22:27:48 +00:00
|
|
|
def with_args(self, **kwargs):
|
|
|
|
return Border(
|
|
|
|
kwargs.get("left", self.left),
|
|
|
|
kwargs.get("right", self.right),
|
|
|
|
kwargs.get("top", self.top),
|
|
|
|
kwargs.get("bottom", self.bottom),
|
|
|
|
)
|
|
|
|
|
2023-01-28 15:55:47 +00:00
|
|
|
@classmethod
|
|
|
|
def even(cls, all: int):
|
|
|
|
return Border(all, all, all, all)
|
|
|
|
|
2023-01-28 04:48:06 +00:00
|
|
|
|
|
|
|
class Size:
|
|
|
|
def __init__(self, width: int, height: int) -> None:
|
|
|
|
self.width = width
|
|
|
|
self.height = height
|
|
|
|
|
2023-02-18 11:44:43 +00:00
|
|
|
def __iter__(self):
|
|
|
|
return iter([self.width, self.height])
|
|
|
|
|
2023-01-28 23:09:19 +00:00
|
|
|
def __str__(self) -> str:
|
2023-02-05 13:53:26 +00:00
|
|
|
return "%sx%s" % (self.width, self.height)
|
2023-01-28 23:09:19 +00:00
|
|
|
|
2023-01-28 04:48:06 +00:00
|
|
|
def add_border(self, border: Border):
|
2023-02-05 13:53:26 +00:00
|
|
|
return Size(
|
|
|
|
border.left + self.width + border.right,
|
|
|
|
border.top + self.height + border.right,
|
|
|
|
)
|
2023-01-28 04:48:06 +00:00
|
|
|
|
|
|
|
def tojson(self) -> Dict[str, int]:
|
|
|
|
return {
|
2023-02-05 13:53:26 +00:00
|
|
|
"height": self.height,
|
|
|
|
"width": self.width,
|
2023-01-28 04:48:06 +00:00
|
|
|
}
|
|
|
|
|
2023-02-18 22:27:48 +00:00
|
|
|
def with_args(self, **kwargs):
|
|
|
|
return Size(
|
|
|
|
kwargs.get("height", self.height),
|
|
|
|
kwargs.get("width", self.width),
|
|
|
|
)
|
|
|
|
|
2023-01-28 04:48:06 +00:00
|
|
|
|
2023-02-04 19:49:34 +00:00
|
|
|
class DeviceParams:
|
2023-02-05 13:53:26 +00:00
|
|
|
def __init__(
|
2023-02-18 21:44:39 +00:00
|
|
|
self,
|
|
|
|
device: str,
|
|
|
|
provider: str,
|
|
|
|
options: Optional[dict] = None,
|
|
|
|
optimizations: Optional[List[str]] = None,
|
2023-02-05 13:53:26 +00:00
|
|
|
) -> None:
|
2023-02-04 19:49:34 +00:00
|
|
|
self.device = device
|
|
|
|
self.provider = provider
|
|
|
|
self.options = options
|
2023-02-18 21:44:39 +00:00
|
|
|
self.optimizations = optimizations
|
2023-02-04 19:49:34 +00:00
|
|
|
|
2023-02-04 19:56:17 +00:00
|
|
|
def __str__(self) -> str:
|
2023-02-05 13:53:26 +00:00
|
|
|
return "%s - %s (%s)" % (self.device, self.provider, self.options)
|
2023-02-04 19:56:17 +00:00
|
|
|
|
2023-02-15 00:57:50 +00:00
|
|
|
def ort_provider(self) -> Tuple[str, Any]:
|
|
|
|
if self.options is None:
|
|
|
|
return self.provider
|
|
|
|
else:
|
|
|
|
return (self.provider, self.options)
|
|
|
|
|
|
|
|
def sess_options(self) -> SessionOptions:
|
2023-02-18 21:44:39 +00:00
|
|
|
sess = SessionOptions()
|
|
|
|
|
|
|
|
if "onnx-low-memory" in self.optimizations:
|
|
|
|
logger.debug("enabling ONNX low-memory optimizations")
|
|
|
|
sess.enable_cpu_mem_arena = False
|
|
|
|
sess.enable_mem_pattern = False
|
|
|
|
sess.enable_mem_reuse = False
|
|
|
|
|
2023-02-18 22:06:05 +00:00
|
|
|
if "onnx-graph-disable" in self.optimizations:
|
2023-02-18 21:47:31 +00:00
|
|
|
logger.debug("disabling all ONNX graph optimizations")
|
2023-02-18 21:44:39 +00:00
|
|
|
sess.graph_optimization_level = GraphOptimizationLevel.ORT_DISABLE_ALL
|
2023-02-18 22:06:05 +00:00
|
|
|
elif "onnx-graph-basic" in self.optimizations:
|
2023-02-18 21:47:31 +00:00
|
|
|
logger.debug("enabling basic ONNX graph optimizations")
|
2023-02-18 21:44:39 +00:00
|
|
|
sess.graph_optimization_level = GraphOptimizationLevel.ORT_ENABLE_BASIC
|
2023-02-18 22:06:05 +00:00
|
|
|
elif "onnx-graph-all" in self.optimizations:
|
2023-02-18 21:47:31 +00:00
|
|
|
logger.debug("enabling all ONNX graph optimizations")
|
2023-02-18 21:44:39 +00:00
|
|
|
sess.graph_optimization_level = GraphOptimizationLevel.ORT_ENABLE_ALL
|
|
|
|
|
|
|
|
if "onnx-deterministic-compute" in self.optimizations:
|
2023-02-18 21:47:31 +00:00
|
|
|
logger.debug("enabling ONNX deterministic compute")
|
2023-02-18 21:44:39 +00:00
|
|
|
sess.use_deterministic_compute = True
|
2023-02-15 00:57:50 +00:00
|
|
|
|
2023-02-18 21:45:28 +00:00
|
|
|
return sess
|
|
|
|
|
2023-02-17 00:11:35 +00:00
|
|
|
def torch_str(self) -> str:
|
2023-02-05 13:53:26 +00:00
|
|
|
if self.device.startswith("cuda"):
|
2023-02-04 19:49:34 +00:00
|
|
|
return self.device
|
|
|
|
else:
|
2023-02-05 13:53:26 +00:00
|
|
|
return "cpu"
|
2023-02-04 19:49:34 +00:00
|
|
|
|
|
|
|
|
2023-01-28 04:48:06 +00:00
|
|
|
class ImageParams:
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
model: str,
|
2023-02-26 16:15:12 +00:00
|
|
|
scheduler: str,
|
2023-01-28 04:48:06 +00:00
|
|
|
prompt: str,
|
|
|
|
cfg: float,
|
|
|
|
steps: int,
|
2023-02-05 13:53:26 +00:00
|
|
|
seed: int,
|
2023-02-05 23:15:37 +00:00
|
|
|
negative_prompt: Optional[str] = None,
|
2023-02-20 05:29:26 +00:00
|
|
|
lpw: bool = False,
|
|
|
|
eta: float = 0.0,
|
2023-02-20 14:35:18 +00:00
|
|
|
batch: int = 1,
|
2023-02-25 18:54:51 +00:00
|
|
|
inversion: Optional[str] = None,
|
2023-01-28 04:48:06 +00:00
|
|
|
) -> None:
|
|
|
|
self.model = model
|
|
|
|
self.scheduler = scheduler
|
|
|
|
self.prompt = prompt
|
|
|
|
self.negative_prompt = negative_prompt
|
|
|
|
self.cfg = cfg
|
|
|
|
self.seed = seed
|
2023-02-05 16:50:15 +00:00
|
|
|
self.steps = steps
|
2023-02-05 23:15:37 +00:00
|
|
|
self.lpw = lpw or False
|
2023-02-20 05:29:26 +00:00
|
|
|
self.eta = eta
|
2023-02-20 14:35:18 +00:00
|
|
|
self.batch = batch
|
2023-02-22 05:08:13 +00:00
|
|
|
self.inversion = inversion
|
2023-01-28 04:48:06 +00:00
|
|
|
|
2023-02-05 03:17:39 +00:00
|
|
|
def tojson(self) -> Dict[str, Optional[Param]]:
|
2023-01-28 04:48:06 +00:00
|
|
|
return {
|
2023-02-05 13:53:26 +00:00
|
|
|
"model": self.model,
|
2023-02-26 16:15:12 +00:00
|
|
|
"scheduler": self.scheduler,
|
2023-02-05 13:53:26 +00:00
|
|
|
"prompt": self.prompt,
|
2023-02-18 22:27:48 +00:00
|
|
|
"negative_prompt": self.negative_prompt,
|
2023-02-05 16:50:15 +00:00
|
|
|
"cfg": self.cfg,
|
|
|
|
"seed": self.seed,
|
2023-02-05 13:53:26 +00:00
|
|
|
"steps": self.steps,
|
2023-02-05 23:15:37 +00:00
|
|
|
"lpw": self.lpw,
|
2023-02-20 05:29:26 +00:00
|
|
|
"eta": self.eta,
|
2023-02-20 14:35:18 +00:00
|
|
|
"batch": self.batch,
|
2023-02-22 05:08:13 +00:00
|
|
|
"inversion": self.inversion,
|
2023-01-28 04:48:06 +00:00
|
|
|
}
|
|
|
|
|
2023-02-18 22:27:48 +00:00
|
|
|
def with_args(self, **kwargs):
|
|
|
|
return ImageParams(
|
|
|
|
kwargs.get("model", self.model),
|
|
|
|
kwargs.get("scheduler", self.scheduler),
|
|
|
|
kwargs.get("prompt", self.prompt),
|
|
|
|
kwargs.get("cfg", self.cfg),
|
|
|
|
kwargs.get("steps", self.steps),
|
|
|
|
kwargs.get("seed", self.seed),
|
|
|
|
kwargs.get("negative_prompt", self.negative_prompt),
|
|
|
|
kwargs.get("lpw", self.lpw),
|
2023-02-20 05:29:26 +00:00
|
|
|
kwargs.get("eta", self.eta),
|
2023-02-20 14:35:18 +00:00
|
|
|
kwargs.get("batch", self.batch),
|
2023-02-22 05:08:13 +00:00
|
|
|
kwargs.get("inversion", self.inversion),
|
2023-02-18 22:27:48 +00:00
|
|
|
)
|
|
|
|
|
2023-01-28 04:48:06 +00:00
|
|
|
|
|
|
|
class StageParams:
|
2023-02-05 13:53:26 +00:00
|
|
|
"""
|
2023-01-28 04:48:06 +00:00
|
|
|
Parameters for a chained pipeline stage
|
2023-02-05 13:53:26 +00:00
|
|
|
"""
|
2023-01-28 04:48:06 +00:00
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
name: Optional[str] = None,
|
|
|
|
outscale: int = 1,
|
2023-02-12 00:00:18 +00:00
|
|
|
tile_order: str = TileOrder.grid,
|
|
|
|
tile_size: int = SizeChart.auto,
|
2023-01-28 04:48:06 +00:00
|
|
|
# batch_size: int = 1,
|
|
|
|
) -> None:
|
|
|
|
self.name = name
|
|
|
|
self.outscale = outscale
|
2023-02-12 00:00:18 +00:00
|
|
|
self.tile_order = tile_order
|
|
|
|
self.tile_size = tile_size
|
2023-01-28 04:48:06 +00:00
|
|
|
|
|
|
|
|
2023-02-05 13:53:26 +00:00
|
|
|
class UpscaleParams:
|
2023-01-28 04:48:06 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
upscale_model: str,
|
|
|
|
correction_model: Optional[str] = None,
|
|
|
|
denoise: float = 0.5,
|
|
|
|
faces=True,
|
2023-02-06 23:13:37 +00:00
|
|
|
face_outscale: int = 1,
|
2023-01-28 04:48:06 +00:00
|
|
|
face_strength: float = 0.5,
|
2023-02-05 13:53:26 +00:00
|
|
|
format: Literal["onnx", "pth"] = "onnx",
|
2023-01-28 04:48:06 +00:00
|
|
|
half=False,
|
|
|
|
outscale: int = 1,
|
|
|
|
scale: int = 4,
|
|
|
|
pre_pad: int = 0,
|
|
|
|
tile_pad: int = 10,
|
2023-02-18 17:05:38 +00:00
|
|
|
upscale_order: Literal[
|
|
|
|
"correction-first", "correction-last", "correction-both"
|
|
|
|
] = "correction-first",
|
2023-01-28 04:48:06 +00:00
|
|
|
) -> None:
|
|
|
|
self.upscale_model = upscale_model
|
|
|
|
self.correction_model = correction_model
|
|
|
|
self.denoise = denoise
|
|
|
|
self.faces = faces
|
2023-02-06 23:13:37 +00:00
|
|
|
self.face_outscale = face_outscale
|
2023-01-28 04:48:06 +00:00
|
|
|
self.face_strength = face_strength
|
|
|
|
self.format = format
|
|
|
|
self.half = half
|
|
|
|
self.outscale = outscale
|
|
|
|
self.pre_pad = pre_pad
|
|
|
|
self.scale = scale
|
|
|
|
self.tile_pad = tile_pad
|
2023-02-18 16:59:59 +00:00
|
|
|
self.upscale_order = upscale_order
|
2023-01-28 04:48:06 +00:00
|
|
|
|
2023-01-31 23:08:30 +00:00
|
|
|
def rescale(self, scale: int):
|
|
|
|
return UpscaleParams(
|
|
|
|
self.upscale_model,
|
|
|
|
correction_model=self.correction_model,
|
|
|
|
denoise=self.denoise,
|
|
|
|
faces=self.faces,
|
2023-02-06 23:13:37 +00:00
|
|
|
face_outscale=self.face_outscale,
|
2023-01-31 23:08:30 +00:00
|
|
|
face_strength=self.face_strength,
|
|
|
|
format=self.format,
|
|
|
|
half=self.half,
|
|
|
|
outscale=scale,
|
|
|
|
scale=scale,
|
|
|
|
pre_pad=self.pre_pad,
|
|
|
|
tile_pad=self.tile_pad,
|
2023-02-18 16:59:59 +00:00
|
|
|
upscale_order=self.upscale_order,
|
2023-01-31 23:08:30 +00:00
|
|
|
)
|
|
|
|
|
2023-01-28 04:48:06 +00:00
|
|
|
def resize(self, size: Size) -> Size:
|
2023-02-18 17:14:34 +00:00
|
|
|
face_outscale = self.face_outscale
|
|
|
|
if self.upscale_order == "correction-both":
|
|
|
|
face_outscale *= self.face_outscale
|
|
|
|
|
2023-02-06 23:59:34 +00:00
|
|
|
return Size(
|
2023-02-19 16:27:05 +00:00
|
|
|
size.width * self.outscale * face_outscale,
|
|
|
|
size.height * self.outscale * face_outscale,
|
2023-02-06 23:59:34 +00:00
|
|
|
)
|
2023-02-02 04:31:01 +00:00
|
|
|
|
|
|
|
def tojson(self):
|
|
|
|
return {
|
2023-02-05 16:50:15 +00:00
|
|
|
"upscale_model": self.upscale_model,
|
|
|
|
"correction_model": self.correction_model,
|
|
|
|
"denoise": self.denoise,
|
|
|
|
"faces": self.faces,
|
2023-02-06 23:13:37 +00:00
|
|
|
"face_outscale": self.face_outscale,
|
2023-02-05 16:50:15 +00:00
|
|
|
"face_strength": self.face_strength,
|
|
|
|
"format": self.format,
|
|
|
|
"half": self.half,
|
2023-02-05 13:53:26 +00:00
|
|
|
"outscale": self.outscale,
|
2023-02-05 16:50:15 +00:00
|
|
|
"pre_pad": self.pre_pad,
|
|
|
|
"scale": self.scale,
|
|
|
|
"tile_pad": self.tile_pad,
|
2023-02-18 16:59:59 +00:00
|
|
|
"upscale_order": self.upscale_order,
|
2023-02-02 04:31:01 +00:00
|
|
|
}
|
2023-02-18 22:27:48 +00:00
|
|
|
|
|
|
|
def with_args(self, **kwargs):
|
2023-02-19 04:36:26 +00:00
|
|
|
return UpscaleParams(
|
2023-02-18 22:27:48 +00:00
|
|
|
kwargs.get("upscale_model", self.upscale_model),
|
|
|
|
kwargs.get("correction_model", self.correction_model),
|
|
|
|
kwargs.get("denoise", self.denoise),
|
|
|
|
kwargs.get("faces", self.faces),
|
|
|
|
kwargs.get("face_outscale", self.face_outscale),
|
|
|
|
kwargs.get("face_strength", self.face_strength),
|
|
|
|
kwargs.get("format", self.format),
|
|
|
|
kwargs.get("half", self.half),
|
|
|
|
kwargs.get("outscale", self.outscale),
|
|
|
|
kwargs.get("pre_pad", self.pre_pad),
|
|
|
|
kwargs.get("scale", self.scale),
|
|
|
|
kwargs.get("tile_pad", self.tile_pad),
|
|
|
|
kwargs.get("upscale_order", self.upscale_order),
|
|
|
|
)
|