| from .k_diffusion import sampling as k_diffusion_sampling
|
| from .extra_samplers import uni_pc
|
| import torch
|
| import collections
|
| from comfy import model_management
|
| import math
|
| import logging
|
| import comfy.sampler_helpers
|
| import scipy
|
| import numpy
|
|
|
| def get_area_and_mult(conds, x_in, timestep_in):
|
| dims = tuple(x_in.shape[2:])
|
| area = None
|
| strength = 1.0
|
|
|
| if 'timestep_start' in conds:
|
| timestep_start = conds['timestep_start']
|
| if timestep_in[0] > timestep_start:
|
| return None
|
| if 'timestep_end' in conds:
|
| timestep_end = conds['timestep_end']
|
| if timestep_in[0] < timestep_end:
|
| return None
|
| if 'area' in conds:
|
| area = list(conds['area'])
|
| if 'strength' in conds:
|
| strength = conds['strength']
|
|
|
| input_x = x_in
|
| if area is not None:
|
| for i in range(len(dims)):
|
| area[i] = min(input_x.shape[i + 2] - area[len(dims) + i], area[i])
|
| input_x = input_x.narrow(i + 2, area[len(dims) + i], area[i])
|
|
|
| if 'mask' in conds:
|
|
|
|
|
| mask_strength = 1.0
|
| if "mask_strength" in conds:
|
| mask_strength = conds["mask_strength"]
|
| mask = conds['mask']
|
| assert(mask.shape[1:] == x_in.shape[2:])
|
|
|
| mask = mask[:input_x.shape[0]]
|
| if area is not None:
|
| for i in range(len(dims)):
|
| mask = mask.narrow(i + 1, area[len(dims) + i], area[i])
|
|
|
| mask = mask * mask_strength
|
| mask = mask.unsqueeze(1).repeat(input_x.shape[0] // mask.shape[0], input_x.shape[1], 1, 1)
|
| else:
|
| mask = torch.ones_like(input_x)
|
| mult = mask * strength
|
|
|
| if 'mask' not in conds and area is not None:
|
| rr = 8
|
| for i in range(len(dims)):
|
| if area[len(dims) + i] != 0:
|
| for t in range(rr):
|
| m = mult.narrow(i + 2, t, 1)
|
| m *= ((1.0/rr) * (t + 1))
|
| if (area[i] + area[len(dims) + i]) < x_in.shape[i + 2]:
|
| for t in range(rr):
|
| m = mult.narrow(i + 2, area[i] - 1 - t, 1)
|
| m *= ((1.0/rr) * (t + 1))
|
|
|
| conditioning = {}
|
| model_conds = conds["model_conds"]
|
| for c in model_conds:
|
| conditioning[c] = model_conds[c].process_cond(batch_size=x_in.shape[0], device=x_in.device, area=area)
|
|
|
| control = conds.get('control', None)
|
|
|
| patches = None
|
| if 'gligen' in conds:
|
| gligen = conds['gligen']
|
| patches = {}
|
| gligen_type = gligen[0]
|
| gligen_model = gligen[1]
|
| if gligen_type == "position":
|
| gligen_patch = gligen_model.model.set_position(input_x.shape, gligen[2], input_x.device)
|
| else:
|
| gligen_patch = gligen_model.model.set_empty(input_x.shape, input_x.device)
|
|
|
| patches['middle_patch'] = [gligen_patch]
|
|
|
| cond_obj = collections.namedtuple('cond_obj', ['input_x', 'mult', 'conditioning', 'area', 'control', 'patches'])
|
| return cond_obj(input_x, mult, conditioning, area, control, patches)
|
|
|
| def cond_equal_size(c1, c2):
|
| if c1 is c2:
|
| return True
|
| if c1.keys() != c2.keys():
|
| return False
|
| for k in c1:
|
| if not c1[k].can_concat(c2[k]):
|
| return False
|
| return True
|
|
|
| def can_concat_cond(c1, c2):
|
| if c1.input_x.shape != c2.input_x.shape:
|
| return False
|
|
|
| def objects_concatable(obj1, obj2):
|
| if (obj1 is None) != (obj2 is None):
|
| return False
|
| if obj1 is not None:
|
| if obj1 is not obj2:
|
| return False
|
| return True
|
|
|
| if not objects_concatable(c1.control, c2.control):
|
| return False
|
|
|
| if not objects_concatable(c1.patches, c2.patches):
|
| return False
|
|
|
| return cond_equal_size(c1.conditioning, c2.conditioning)
|
|
|
| def cond_cat(c_list):
|
| c_crossattn = []
|
| c_concat = []
|
| c_adm = []
|
| crossattn_max_len = 0
|
|
|
| temp = {}
|
| for x in c_list:
|
| for k in x:
|
| cur = temp.get(k, [])
|
| cur.append(x[k])
|
| temp[k] = cur
|
|
|
| out = {}
|
| for k in temp:
|
| conds = temp[k]
|
| out[k] = conds[0].concat(conds[1:])
|
|
|
| return out
|
|
|
| def calc_cond_batch(model, conds, x_in, timestep, model_options):
|
| out_conds = []
|
| out_counts = []
|
| to_run = []
|
|
|
| for i in range(len(conds)):
|
| out_conds.append(torch.zeros_like(x_in))
|
| out_counts.append(torch.ones_like(x_in) * 1e-37)
|
|
|
| cond = conds[i]
|
| if cond is not None:
|
| for x in cond:
|
| p = get_area_and_mult(x, x_in, timestep)
|
| if p is None:
|
| continue
|
|
|
| to_run += [(p, i)]
|
|
|
| while len(to_run) > 0:
|
| first = to_run[0]
|
| first_shape = first[0][0].shape
|
| to_batch_temp = []
|
| for x in range(len(to_run)):
|
| if can_concat_cond(to_run[x][0], first[0]):
|
| to_batch_temp += [x]
|
|
|
| to_batch_temp.reverse()
|
| to_batch = to_batch_temp[:1]
|
|
|
| free_memory = model_management.get_free_memory(x_in.device)
|
| for i in range(1, len(to_batch_temp) + 1):
|
| batch_amount = to_batch_temp[:len(to_batch_temp)//i]
|
| input_shape = [len(batch_amount) * first_shape[0]] + list(first_shape)[1:]
|
| if model.memory_required(input_shape) < free_memory:
|
| to_batch = batch_amount
|
| break
|
|
|
| input_x = []
|
| mult = []
|
| c = []
|
| cond_or_uncond = []
|
| area = []
|
| control = None
|
| patches = None
|
| for x in to_batch:
|
| o = to_run.pop(x)
|
| p = o[0]
|
| input_x.append(p.input_x)
|
| mult.append(p.mult)
|
| c.append(p.conditioning)
|
| area.append(p.area)
|
| cond_or_uncond.append(o[1])
|
| control = p.control
|
| patches = p.patches
|
|
|
| batch_chunks = len(cond_or_uncond)
|
| input_x = torch.cat(input_x)
|
| c = cond_cat(c)
|
| timestep_ = torch.cat([timestep] * batch_chunks)
|
|
|
| if control is not None:
|
| c['control'] = control.get_control(input_x, timestep_, c, len(cond_or_uncond))
|
|
|
| transformer_options = {}
|
| if 'transformer_options' in model_options:
|
| transformer_options = model_options['transformer_options'].copy()
|
|
|
| if patches is not None:
|
| if "patches" in transformer_options:
|
| cur_patches = transformer_options["patches"].copy()
|
| for p in patches:
|
| if p in cur_patches:
|
| cur_patches[p] = cur_patches[p] + patches[p]
|
| else:
|
| cur_patches[p] = patches[p]
|
| transformer_options["patches"] = cur_patches
|
| else:
|
| transformer_options["patches"] = patches
|
|
|
| transformer_options["cond_or_uncond"] = cond_or_uncond[:]
|
| transformer_options["sigmas"] = timestep
|
|
|
| c['transformer_options'] = transformer_options
|
|
|
| if 'model_function_wrapper' in model_options:
|
| output = model_options['model_function_wrapper'](model.apply_model, {"input": input_x, "timestep": timestep_, "c": c, "cond_or_uncond": cond_or_uncond}).chunk(batch_chunks)
|
| else:
|
| output = model.apply_model(input_x, timestep_, **c).chunk(batch_chunks)
|
|
|
| for o in range(batch_chunks):
|
| cond_index = cond_or_uncond[o]
|
| a = area[o]
|
| if a is None:
|
| out_conds[cond_index] += output[o] * mult[o]
|
| out_counts[cond_index] += mult[o]
|
| else:
|
| out_c = out_conds[cond_index]
|
| out_cts = out_counts[cond_index]
|
| dims = len(a) // 2
|
| for i in range(dims):
|
| out_c = out_c.narrow(i + 2, a[i + dims], a[i])
|
| out_cts = out_cts.narrow(i + 2, a[i + dims], a[i])
|
| out_c += output[o] * mult[o]
|
| out_cts += mult[o]
|
|
|
| for i in range(len(out_conds)):
|
| out_conds[i] /= out_counts[i]
|
|
|
| return out_conds
|
|
|
| def calc_cond_uncond_batch(model, cond, uncond, x_in, timestep, model_options):
|
| logging.warning("WARNING: The comfy.samplers.calc_cond_uncond_batch function is deprecated please use the calc_cond_batch one instead.")
|
| return tuple(calc_cond_batch(model, [cond, uncond], x_in, timestep, model_options))
|
|
|
| def cfg_function(model, cond_pred, uncond_pred, cond_scale, x, timestep, model_options={}, cond=None, uncond=None):
|
| if "sampler_cfg_function" in model_options:
|
| args = {"cond": x - cond_pred, "uncond": x - uncond_pred, "cond_scale": cond_scale, "timestep": timestep, "input": x, "sigma": timestep,
|
| "cond_denoised": cond_pred, "uncond_denoised": uncond_pred, "model": model, "model_options": model_options}
|
| cfg_result = x - model_options["sampler_cfg_function"](args)
|
| else:
|
| cfg_result = uncond_pred + (cond_pred - uncond_pred) * cond_scale
|
|
|
| for fn in model_options.get("sampler_post_cfg_function", []):
|
| args = {"denoised": cfg_result, "cond": cond, "uncond": uncond, "model": model, "uncond_denoised": uncond_pred, "cond_denoised": cond_pred,
|
| "sigma": timestep, "model_options": model_options, "input": x}
|
| cfg_result = fn(args)
|
|
|
| return cfg_result
|
|
|
|
|
|
|
| def sampling_function(model, x, timestep, uncond, cond, cond_scale, model_options={}, seed=None):
|
| if math.isclose(cond_scale, 1.0) and model_options.get("disable_cfg1_optimization", False) == False:
|
| uncond_ = None
|
| else:
|
| uncond_ = uncond
|
|
|
| conds = [cond, uncond_]
|
| out = calc_cond_batch(model, conds, x, timestep, model_options)
|
|
|
| for fn in model_options.get("sampler_pre_cfg_function", []):
|
| args = {"conds":conds, "conds_out": out, "cond_scale": cond_scale, "timestep": timestep,
|
| "input": x, "sigma": timestep, "model": model, "model_options": model_options}
|
| out = fn(args)
|
|
|
| return cfg_function(model, out[0], out[1], cond_scale, x, timestep, model_options=model_options, cond=cond, uncond=uncond_)
|
|
|
|
|
| class KSamplerX0Inpaint:
|
| def __init__(self, model, sigmas):
|
| self.inner_model = model
|
| self.sigmas = sigmas
|
| def __call__(self, x, sigma, denoise_mask, model_options={}, seed=None):
|
| if denoise_mask is not None:
|
| if "denoise_mask_function" in model_options:
|
| denoise_mask = model_options["denoise_mask_function"](sigma, denoise_mask, extra_options={"model": self.inner_model, "sigmas": self.sigmas})
|
| latent_mask = 1. - denoise_mask
|
| x = x * denoise_mask + self.inner_model.inner_model.model_sampling.noise_scaling(sigma.reshape([sigma.shape[0]] + [1] * (len(self.noise.shape) - 1)), self.noise, self.latent_image) * latent_mask
|
| out = self.inner_model(x, sigma, model_options=model_options, seed=seed)
|
| if denoise_mask is not None:
|
| out = out * denoise_mask + self.latent_image * latent_mask
|
| return out
|
|
|
| def simple_scheduler(model_sampling, steps):
|
| s = model_sampling
|
| sigs = []
|
| ss = len(s.sigmas) / steps
|
| for x in range(steps):
|
| sigs += [float(s.sigmas[-(1 + int(x * ss))])]
|
| sigs += [0.0]
|
| return torch.FloatTensor(sigs)
|
|
|
| def ddim_scheduler(model_sampling, steps):
|
| s = model_sampling
|
| sigs = []
|
| x = 1
|
| if math.isclose(float(s.sigmas[x]), 0, abs_tol=0.00001):
|
| steps += 1
|
| sigs = []
|
| else:
|
| sigs = [0.0]
|
|
|
| ss = max(len(s.sigmas) // steps, 1)
|
| while x < len(s.sigmas):
|
| sigs += [float(s.sigmas[x])]
|
| x += ss
|
| sigs = sigs[::-1]
|
| return torch.FloatTensor(sigs)
|
|
|
| def normal_scheduler(model_sampling, steps, sgm=False, floor=False):
|
| s = model_sampling
|
| start = s.timestep(s.sigma_max)
|
| end = s.timestep(s.sigma_min)
|
|
|
| append_zero = True
|
| if sgm:
|
| timesteps = torch.linspace(start, end, steps + 1)[:-1]
|
| else:
|
| if math.isclose(float(s.sigma(end)), 0, abs_tol=0.00001):
|
| steps += 1
|
| append_zero = False
|
| timesteps = torch.linspace(start, end, steps)
|
|
|
| sigs = []
|
| for x in range(len(timesteps)):
|
| ts = timesteps[x]
|
| sigs.append(float(s.sigma(ts)))
|
|
|
| if append_zero:
|
| sigs += [0.0]
|
|
|
| return torch.FloatTensor(sigs)
|
|
|
|
|
| def beta_scheduler(model_sampling, steps, alpha=0.6, beta=0.6):
|
| total_timesteps = (len(model_sampling.sigmas) - 1)
|
| ts = 1 - numpy.linspace(0, 1, steps, endpoint=False)
|
| ts = numpy.rint(scipy.stats.beta.ppf(ts, alpha, beta) * total_timesteps)
|
|
|
| sigs = []
|
| for t in ts:
|
| sigs += [float(model_sampling.sigmas[int(t)])]
|
| sigs += [0.0]
|
| return torch.FloatTensor(sigs)
|
|
|
| def get_mask_aabb(masks):
|
| if masks.numel() == 0:
|
| return torch.zeros((0, 4), device=masks.device, dtype=torch.int)
|
|
|
| b = masks.shape[0]
|
|
|
| bounding_boxes = torch.zeros((b, 4), device=masks.device, dtype=torch.int)
|
| is_empty = torch.zeros((b), device=masks.device, dtype=torch.bool)
|
| for i in range(b):
|
| mask = masks[i]
|
| if mask.numel() == 0:
|
| continue
|
| if torch.max(mask != 0) == False:
|
| is_empty[i] = True
|
| continue
|
| y, x = torch.where(mask)
|
| bounding_boxes[i, 0] = torch.min(x)
|
| bounding_boxes[i, 1] = torch.min(y)
|
| bounding_boxes[i, 2] = torch.max(x)
|
| bounding_boxes[i, 3] = torch.max(y)
|
|
|
| return bounding_boxes, is_empty
|
|
|
| def resolve_areas_and_cond_masks_multidim(conditions, dims, device):
|
|
|
|
|
| for i in range(len(conditions)):
|
| c = conditions[i]
|
| if 'area' in c:
|
| area = c['area']
|
| if area[0] == "percentage":
|
| modified = c.copy()
|
| a = area[1:]
|
| a_len = len(a) // 2
|
| area = ()
|
| for d in range(len(dims)):
|
| area += (max(1, round(a[d] * dims[d])),)
|
| for d in range(len(dims)):
|
| area += (round(a[d + a_len] * dims[d]),)
|
|
|
| modified['area'] = area
|
| c = modified
|
| conditions[i] = c
|
|
|
| if 'mask' in c:
|
| mask = c['mask']
|
| mask = mask.to(device=device)
|
| modified = c.copy()
|
| if len(mask.shape) == len(dims):
|
| mask = mask.unsqueeze(0)
|
| if mask.shape[1:] != dims:
|
| mask = torch.nn.functional.interpolate(mask.unsqueeze(1), size=dims, mode='bilinear', align_corners=False).squeeze(1)
|
|
|
| if modified.get("set_area_to_bounds", False):
|
| bounds = torch.max(torch.abs(mask),dim=0).values.unsqueeze(0)
|
| boxes, is_empty = get_mask_aabb(bounds)
|
| if is_empty[0]:
|
|
|
| modified['area'] = (8, 8, 0, 0)
|
| else:
|
| box = boxes[0]
|
| H, W, Y, X = (box[3] - box[1] + 1, box[2] - box[0] + 1, box[1], box[0])
|
| H = max(8, H)
|
| W = max(8, W)
|
| area = (int(H), int(W), int(Y), int(X))
|
| modified['area'] = area
|
|
|
| modified['mask'] = mask
|
| conditions[i] = modified
|
|
|
| def resolve_areas_and_cond_masks(conditions, h, w, device):
|
| logging.warning("WARNING: The comfy.samplers.resolve_areas_and_cond_masks function is deprecated please use the resolve_areas_and_cond_masks_multidim one instead.")
|
| return resolve_areas_and_cond_masks_multidim(conditions, [h, w], device)
|
|
|
| def create_cond_with_same_area_if_none(conds, c):
|
| if 'area' not in c:
|
| return
|
|
|
| c_area = c['area']
|
| smallest = None
|
| for x in conds:
|
| if 'area' in x:
|
| a = x['area']
|
| if c_area[2] >= a[2] and c_area[3] >= a[3]:
|
| if a[0] + a[2] >= c_area[0] + c_area[2]:
|
| if a[1] + a[3] >= c_area[1] + c_area[3]:
|
| if smallest is None:
|
| smallest = x
|
| elif 'area' not in smallest:
|
| smallest = x
|
| else:
|
| if smallest['area'][0] * smallest['area'][1] > a[0] * a[1]:
|
| smallest = x
|
| else:
|
| if smallest is None:
|
| smallest = x
|
| if smallest is None:
|
| return
|
| if 'area' in smallest:
|
| if smallest['area'] == c_area:
|
| return
|
|
|
| out = c.copy()
|
| out['model_conds'] = smallest['model_conds'].copy()
|
| conds += [out]
|
|
|
| def calculate_start_end_timesteps(model, conds):
|
| s = model.model_sampling
|
| for t in range(len(conds)):
|
| x = conds[t]
|
|
|
| timestep_start = None
|
| timestep_end = None
|
| if 'start_percent' in x:
|
| timestep_start = s.percent_to_sigma(x['start_percent'])
|
| if 'end_percent' in x:
|
| timestep_end = s.percent_to_sigma(x['end_percent'])
|
|
|
| if (timestep_start is not None) or (timestep_end is not None):
|
| n = x.copy()
|
| if (timestep_start is not None):
|
| n['timestep_start'] = timestep_start
|
| if (timestep_end is not None):
|
| n['timestep_end'] = timestep_end
|
| conds[t] = n
|
|
|
| def pre_run_control(model, conds):
|
| s = model.model_sampling
|
| for t in range(len(conds)):
|
| x = conds[t]
|
|
|
| timestep_start = None
|
| timestep_end = None
|
| percent_to_timestep_function = lambda a: s.percent_to_sigma(a)
|
| if 'control' in x:
|
| x['control'].pre_run(model, percent_to_timestep_function)
|
|
|
| def apply_empty_x_to_equal_area(conds, uncond, name, uncond_fill_func):
|
| cond_cnets = []
|
| cond_other = []
|
| uncond_cnets = []
|
| uncond_other = []
|
| for t in range(len(conds)):
|
| x = conds[t]
|
| if 'area' not in x:
|
| if name in x and x[name] is not None:
|
| cond_cnets.append(x[name])
|
| else:
|
| cond_other.append((x, t))
|
| for t in range(len(uncond)):
|
| x = uncond[t]
|
| if 'area' not in x:
|
| if name in x and x[name] is not None:
|
| uncond_cnets.append(x[name])
|
| else:
|
| uncond_other.append((x, t))
|
|
|
| if len(uncond_cnets) > 0:
|
| return
|
|
|
| for x in range(len(cond_cnets)):
|
| temp = uncond_other[x % len(uncond_other)]
|
| o = temp[0]
|
| if name in o and o[name] is not None:
|
| n = o.copy()
|
| n[name] = uncond_fill_func(cond_cnets, x)
|
| uncond += [n]
|
| else:
|
| n = o.copy()
|
| n[name] = uncond_fill_func(cond_cnets, x)
|
| uncond[temp[1]] = n
|
|
|
| def encode_model_conds(model_function, conds, noise, device, prompt_type, **kwargs):
|
| for t in range(len(conds)):
|
| x = conds[t]
|
| params = x.copy()
|
| params["device"] = device
|
| params["noise"] = noise
|
| default_width = None
|
| if len(noise.shape) >= 4:
|
| default_width = noise.shape[3] * 8
|
| params["width"] = params.get("width", default_width)
|
| params["height"] = params.get("height", noise.shape[2] * 8)
|
| params["prompt_type"] = params.get("prompt_type", prompt_type)
|
| for k in kwargs:
|
| if k not in params:
|
| params[k] = kwargs[k]
|
|
|
| out = model_function(**params)
|
| x = x.copy()
|
| model_conds = x['model_conds'].copy()
|
| for k in out:
|
| model_conds[k] = out[k]
|
| x['model_conds'] = model_conds
|
| conds[t] = x
|
| return conds
|
|
|
| class Sampler:
|
| def sample(self):
|
| pass
|
|
|
| def max_denoise(self, model_wrap, sigmas):
|
| max_sigma = float(model_wrap.inner_model.model_sampling.sigma_max)
|
| sigma = float(sigmas[0])
|
| return math.isclose(max_sigma, sigma, rel_tol=1e-05) or sigma > max_sigma
|
|
|
| KSAMPLER_NAMES = ["euler", "euler_cfg_pp", "euler_ancestral", "euler_ancestral_cfg_pp", "heun", "heunpp2","dpm_2", "dpm_2_ancestral",
|
| "lms", "dpm_fast", "dpm_adaptive", "dpmpp_2s_ancestral", "dpmpp_sde", "dpmpp_sde_gpu",
|
| "dpmpp_2m", "dpmpp_2m_sde", "dpmpp_2m_sde_gpu", "dpmpp_3m_sde", "dpmpp_3m_sde_gpu", "ddpm", "lcm",
|
| "ipndm", "ipndm_v", "deis"]
|
|
|
| class KSAMPLER(Sampler):
|
| def __init__(self, sampler_function, extra_options={}, inpaint_options={}):
|
| self.sampler_function = sampler_function
|
| self.extra_options = extra_options
|
| self.inpaint_options = inpaint_options
|
|
|
| def sample(self, model_wrap, sigmas, extra_args, callback, noise, latent_image=None, denoise_mask=None, disable_pbar=False):
|
| extra_args["denoise_mask"] = denoise_mask
|
| model_k = KSamplerX0Inpaint(model_wrap, sigmas)
|
| model_k.latent_image = latent_image
|
| if self.inpaint_options.get("random", False):
|
| generator = torch.manual_seed(extra_args.get("seed", 41) + 1)
|
| model_k.noise = torch.randn(noise.shape, generator=generator, device="cpu").to(noise.dtype).to(noise.device)
|
| else:
|
| model_k.noise = noise
|
|
|
| noise = model_wrap.inner_model.model_sampling.noise_scaling(sigmas[0], noise, latent_image, self.max_denoise(model_wrap, sigmas))
|
|
|
| k_callback = None
|
| total_steps = len(sigmas) - 1
|
| if callback is not None:
|
| k_callback = lambda x: callback(x["i"], x["denoised"], x["x"], total_steps)
|
|
|
| samples = self.sampler_function(model_k, noise, sigmas, extra_args=extra_args, callback=k_callback, disable=disable_pbar, **self.extra_options)
|
| samples = model_wrap.inner_model.model_sampling.inverse_noise_scaling(sigmas[-1], samples)
|
| return samples
|
|
|
|
|
| def ksampler(sampler_name, extra_options={}, inpaint_options={}):
|
| if sampler_name == "dpm_fast":
|
| def dpm_fast_function(model, noise, sigmas, extra_args, callback, disable):
|
| if len(sigmas) <= 1:
|
| return noise
|
|
|
| sigma_min = sigmas[-1]
|
| if sigma_min == 0:
|
| sigma_min = sigmas[-2]
|
| total_steps = len(sigmas) - 1
|
| return k_diffusion_sampling.sample_dpm_fast(model, noise, sigma_min, sigmas[0], total_steps, extra_args=extra_args, callback=callback, disable=disable)
|
| sampler_function = dpm_fast_function
|
| elif sampler_name == "dpm_adaptive":
|
| def dpm_adaptive_function(model, noise, sigmas, extra_args, callback, disable, **extra_options):
|
| if len(sigmas) <= 1:
|
| return noise
|
|
|
| sigma_min = sigmas[-1]
|
| if sigma_min == 0:
|
| sigma_min = sigmas[-2]
|
| return k_diffusion_sampling.sample_dpm_adaptive(model, noise, sigma_min, sigmas[0], extra_args=extra_args, callback=callback, disable=disable, **extra_options)
|
| sampler_function = dpm_adaptive_function
|
| else:
|
| sampler_function = getattr(k_diffusion_sampling, "sample_{}".format(sampler_name))
|
|
|
| return KSAMPLER(sampler_function, extra_options, inpaint_options)
|
|
|
|
|
| def process_conds(model, noise, conds, device, latent_image=None, denoise_mask=None, seed=None):
|
| for k in conds:
|
| conds[k] = conds[k][:]
|
| resolve_areas_and_cond_masks_multidim(conds[k], noise.shape[2:], device)
|
|
|
| for k in conds:
|
| calculate_start_end_timesteps(model, conds[k])
|
|
|
| if hasattr(model, 'extra_conds'):
|
| for k in conds:
|
| conds[k] = encode_model_conds(model.extra_conds, conds[k], noise, device, k, latent_image=latent_image, denoise_mask=denoise_mask, seed=seed)
|
|
|
|
|
| for k in conds:
|
| for c in conds[k]:
|
| for kk in conds:
|
| if k != kk:
|
| create_cond_with_same_area_if_none(conds[kk], c)
|
|
|
| for k in conds:
|
| pre_run_control(model, conds[k])
|
|
|
| if "positive" in conds:
|
| positive = conds["positive"]
|
| for k in conds:
|
| if k != "positive":
|
| apply_empty_x_to_equal_area(list(filter(lambda c: c.get('control_apply_to_uncond', False) == True, positive)), conds[k], 'control', lambda cond_cnets, x: cond_cnets[x])
|
| apply_empty_x_to_equal_area(positive, conds[k], 'gligen', lambda cond_cnets, x: cond_cnets[x])
|
|
|
| return conds
|
|
|
| class CFGGuider:
|
| def __init__(self, model_patcher):
|
| self.model_patcher = model_patcher
|
| self.model_options = model_patcher.model_options
|
| self.original_conds = {}
|
| self.cfg = 1.0
|
|
|
| def set_conds(self, positive, negative):
|
| self.inner_set_conds({"positive": positive, "negative": negative})
|
|
|
| def set_cfg(self, cfg):
|
| self.cfg = cfg
|
|
|
| def inner_set_conds(self, conds):
|
| for k in conds:
|
| self.original_conds[k] = comfy.sampler_helpers.convert_cond(conds[k])
|
|
|
| def __call__(self, *args, **kwargs):
|
| return self.predict_noise(*args, **kwargs)
|
|
|
| def predict_noise(self, x, timestep, model_options={}, seed=None):
|
| return sampling_function(self.inner_model, x, timestep, self.conds.get("negative", None), self.conds.get("positive", None), self.cfg, model_options=model_options, seed=seed)
|
|
|
| def inner_sample(self, noise, latent_image, device, sampler, sigmas, denoise_mask, callback, disable_pbar, seed):
|
| if latent_image is not None and torch.count_nonzero(latent_image) > 0:
|
| latent_image = self.inner_model.process_latent_in(latent_image)
|
|
|
| self.conds = process_conds(self.inner_model, noise, self.conds, device, latent_image, denoise_mask, seed)
|
|
|
| extra_args = {"model_options": self.model_options, "seed":seed}
|
|
|
| samples = sampler.sample(self, sigmas, extra_args, callback, noise, latent_image, denoise_mask, disable_pbar)
|
| return self.inner_model.process_latent_out(samples.to(torch.float32))
|
|
|
| def sample(self, noise, latent_image, sampler, sigmas, denoise_mask=None, callback=None, disable_pbar=False, seed=None):
|
| if sigmas.shape[-1] == 0:
|
| return latent_image
|
|
|
| self.conds = {}
|
| for k in self.original_conds:
|
| self.conds[k] = list(map(lambda a: a.copy(), self.original_conds[k]))
|
|
|
| self.inner_model, self.conds, self.loaded_models = comfy.sampler_helpers.prepare_sampling(self.model_patcher, noise.shape, self.conds)
|
| device = self.model_patcher.load_device
|
|
|
| if denoise_mask is not None:
|
| denoise_mask = comfy.sampler_helpers.prepare_mask(denoise_mask, noise.shape, device)
|
|
|
| noise = noise.to(device)
|
| latent_image = latent_image.to(device)
|
| sigmas = sigmas.to(device)
|
|
|
| output = self.inner_sample(noise, latent_image, device, sampler, sigmas, denoise_mask, callback, disable_pbar, seed)
|
|
|
| comfy.sampler_helpers.cleanup_models(self.conds, self.loaded_models)
|
| del self.inner_model
|
| del self.conds
|
| del self.loaded_models
|
| return output
|
|
|
|
|
| def sample(model, noise, positive, negative, cfg, device, sampler, sigmas, model_options={}, latent_image=None, denoise_mask=None, callback=None, disable_pbar=False, seed=None):
|
| cfg_guider = CFGGuider(model)
|
| cfg_guider.set_conds(positive, negative)
|
| cfg_guider.set_cfg(cfg)
|
| return cfg_guider.sample(noise, latent_image, sampler, sigmas, denoise_mask, callback, disable_pbar, seed)
|
|
|
|
|
| SCHEDULER_NAMES = ["normal", "karras", "exponential", "sgm_uniform", "simple", "ddim_uniform", "beta"]
|
| SAMPLER_NAMES = KSAMPLER_NAMES + ["ddim", "uni_pc", "uni_pc_bh2"]
|
|
|
| def calculate_sigmas(model_sampling, scheduler_name, steps):
|
| if scheduler_name == "karras":
|
| sigmas = k_diffusion_sampling.get_sigmas_karras(n=steps, sigma_min=float(model_sampling.sigma_min), sigma_max=float(model_sampling.sigma_max))
|
| elif scheduler_name == "exponential":
|
| sigmas = k_diffusion_sampling.get_sigmas_exponential(n=steps, sigma_min=float(model_sampling.sigma_min), sigma_max=float(model_sampling.sigma_max))
|
| elif scheduler_name == "normal":
|
| sigmas = normal_scheduler(model_sampling, steps)
|
| elif scheduler_name == "simple":
|
| sigmas = simple_scheduler(model_sampling, steps)
|
| elif scheduler_name == "ddim_uniform":
|
| sigmas = ddim_scheduler(model_sampling, steps)
|
| elif scheduler_name == "sgm_uniform":
|
| sigmas = normal_scheduler(model_sampling, steps, sgm=True)
|
| elif scheduler_name == "beta":
|
| sigmas = beta_scheduler(model_sampling, steps)
|
| else:
|
| logging.error("error invalid scheduler {}".format(scheduler_name))
|
| return sigmas
|
|
|
| def sampler_object(name):
|
| if name == "uni_pc":
|
| sampler = KSAMPLER(uni_pc.sample_unipc)
|
| elif name == "uni_pc_bh2":
|
| sampler = KSAMPLER(uni_pc.sample_unipc_bh2)
|
| elif name == "ddim":
|
| sampler = ksampler("euler", inpaint_options={"random": True})
|
| else:
|
| sampler = ksampler(name)
|
| return sampler
|
|
|
| class KSampler:
|
| SCHEDULERS = SCHEDULER_NAMES
|
| SAMPLERS = SAMPLER_NAMES
|
| DISCARD_PENULTIMATE_SIGMA_SAMPLERS = set(('dpm_2', 'dpm_2_ancestral', 'uni_pc', 'uni_pc_bh2'))
|
|
|
| def __init__(self, model, steps, device, sampler=None, scheduler=None, denoise=None, model_options={}):
|
| self.model = model
|
| self.device = device
|
| if scheduler not in self.SCHEDULERS:
|
| scheduler = self.SCHEDULERS[0]
|
| if sampler not in self.SAMPLERS:
|
| sampler = self.SAMPLERS[0]
|
| self.scheduler = scheduler
|
| self.sampler = sampler
|
| self.set_steps(steps, denoise)
|
| self.denoise = denoise
|
| self.model_options = model_options
|
|
|
| def calculate_sigmas(self, steps):
|
| sigmas = None
|
|
|
| discard_penultimate_sigma = False
|
| if self.sampler in self.DISCARD_PENULTIMATE_SIGMA_SAMPLERS:
|
| steps += 1
|
| discard_penultimate_sigma = True
|
|
|
| sigmas = calculate_sigmas(self.model.get_model_object("model_sampling"), self.scheduler, steps)
|
|
|
| if discard_penultimate_sigma:
|
| sigmas = torch.cat([sigmas[:-2], sigmas[-1:]])
|
| return sigmas
|
|
|
| def set_steps(self, steps, denoise=None):
|
| self.steps = steps
|
| if denoise is None or denoise > 0.9999:
|
| self.sigmas = self.calculate_sigmas(steps).to(self.device)
|
| else:
|
| if denoise <= 0.0:
|
| self.sigmas = torch.FloatTensor([])
|
| else:
|
| new_steps = int(steps/denoise)
|
| sigmas = self.calculate_sigmas(new_steps).to(self.device)
|
| self.sigmas = sigmas[-(steps + 1):]
|
|
|
| def sample(self, noise, positive, negative, cfg, latent_image=None, start_step=None, last_step=None, force_full_denoise=False, denoise_mask=None, sigmas=None, callback=None, disable_pbar=False, seed=None):
|
| if sigmas is None:
|
| sigmas = self.sigmas
|
|
|
| if last_step is not None and last_step < (len(sigmas) - 1):
|
| sigmas = sigmas[:last_step + 1]
|
| if force_full_denoise:
|
| sigmas[-1] = 0
|
|
|
| if start_step is not None:
|
| if start_step < (len(sigmas) - 1):
|
| sigmas = sigmas[start_step:]
|
| else:
|
| if latent_image is not None:
|
| return latent_image
|
| else:
|
| return torch.zeros_like(noise)
|
|
|
| sampler = sampler_object(self.sampler)
|
|
|
| return sample(self.model, noise, positive, negative, cfg, self.device, sampler, sigmas, self.model_options, latent_image=latent_image, denoise_mask=denoise_mask, callback=callback, disable_pbar=disable_pbar, seed=seed)
|
|
|