| | from __future__ import annotations
|
| | import uuid
|
| | import math
|
| | import collections
|
| | import comfy.model_management
|
| | import comfy.conds
|
| | import comfy.utils
|
| | import comfy.hooks
|
| | import comfy.patcher_extension
|
| | from typing import TYPE_CHECKING
|
| | if TYPE_CHECKING:
|
| | from comfy.model_patcher import ModelPatcher
|
| | from comfy.model_base import BaseModel
|
| | from comfy.controlnet import ControlBase
|
| |
|
| | def prepare_mask(noise_mask, shape, device):
|
| | return comfy.utils.reshape_mask(noise_mask, shape).to(device)
|
| |
|
| | def get_models_from_cond(cond, model_type):
|
| | models = []
|
| | for c in cond:
|
| | if model_type in c:
|
| | if isinstance(c[model_type], list):
|
| | models += c[model_type]
|
| | else:
|
| | models += [c[model_type]]
|
| | return models
|
| |
|
| | def get_hooks_from_cond(cond, full_hooks: comfy.hooks.HookGroup):
|
| |
|
| | cnets: list[ControlBase] = []
|
| | for c in cond:
|
| | if 'hooks' in c:
|
| | for hook in c['hooks'].hooks:
|
| | full_hooks.add(hook)
|
| | if 'control' in c:
|
| | cnets.append(c['control'])
|
| |
|
| | def get_extra_hooks_from_cnet(cnet: ControlBase, _list: list):
|
| | if cnet.extra_hooks is not None:
|
| | _list.append(cnet.extra_hooks)
|
| | if cnet.previous_controlnet is None:
|
| | return _list
|
| | return get_extra_hooks_from_cnet(cnet.previous_controlnet, _list)
|
| |
|
| | hooks_list = []
|
| | cnets = set(cnets)
|
| | for base_cnet in cnets:
|
| | get_extra_hooks_from_cnet(base_cnet, hooks_list)
|
| | extra_hooks = comfy.hooks.HookGroup.combine_all_hooks(hooks_list)
|
| | if extra_hooks is not None:
|
| | for hook in extra_hooks.hooks:
|
| | full_hooks.add(hook)
|
| |
|
| | return full_hooks
|
| |
|
| | def convert_cond(cond):
|
| | out = []
|
| | for c in cond:
|
| | temp = c[1].copy()
|
| | model_conds = temp.get("model_conds", {})
|
| | if c[0] is not None:
|
| | temp["cross_attn"] = c[0]
|
| | temp["model_conds"] = model_conds
|
| | temp["uuid"] = uuid.uuid4()
|
| | out.append(temp)
|
| | return out
|
| |
|
| | def get_additional_models(conds, dtype):
|
| | """loads additional models in conditioning"""
|
| | cnets: list[ControlBase] = []
|
| | gligen = []
|
| | add_models = []
|
| |
|
| | for k in conds:
|
| | cnets += get_models_from_cond(conds[k], "control")
|
| | gligen += get_models_from_cond(conds[k], "gligen")
|
| | add_models += get_models_from_cond(conds[k], "additional_models")
|
| |
|
| | control_nets = set(cnets)
|
| |
|
| | inference_memory = 0
|
| | control_models = []
|
| | for m in control_nets:
|
| | control_models += m.get_models()
|
| | inference_memory += m.inference_memory_requirements(dtype)
|
| |
|
| | gligen = [x[1] for x in gligen]
|
| | models = control_models + gligen + add_models
|
| |
|
| | return models, inference_memory
|
| |
|
| | def get_additional_models_from_model_options(model_options: dict[str]=None):
|
| | """loads additional models from registered AddModels hooks"""
|
| | models = []
|
| | if model_options is not None and "registered_hooks" in model_options:
|
| | registered: comfy.hooks.HookGroup = model_options["registered_hooks"]
|
| | for hook in registered.get_type(comfy.hooks.EnumHookType.AdditionalModels):
|
| | hook: comfy.hooks.AdditionalModelsHook
|
| | models.extend(hook.models)
|
| | return models
|
| |
|
| | def cleanup_additional_models(models):
|
| | """cleanup additional models that were loaded"""
|
| | for m in models:
|
| | if hasattr(m, 'cleanup'):
|
| | m.cleanup()
|
| |
|
| | def estimate_memory(model, noise_shape, conds):
|
| | cond_shapes = collections.defaultdict(list)
|
| | cond_shapes_min = {}
|
| | for _, cs in conds.items():
|
| | for cond in cs:
|
| | for k, v in model.model.extra_conds_shapes(**cond).items():
|
| | cond_shapes[k].append(v)
|
| | if cond_shapes_min.get(k, None) is None:
|
| | cond_shapes_min[k] = [v]
|
| | elif math.prod(v) > math.prod(cond_shapes_min[k][0]):
|
| | cond_shapes_min[k] = [v]
|
| |
|
| | memory_required = model.model.memory_required([noise_shape[0] * 2] + list(noise_shape[1:]), cond_shapes=cond_shapes)
|
| | minimum_memory_required = model.model.memory_required([noise_shape[0]] + list(noise_shape[1:]), cond_shapes=cond_shapes_min)
|
| | return memory_required, minimum_memory_required
|
| |
|
| | def prepare_sampling(model: ModelPatcher, noise_shape, conds, model_options=None):
|
| | executor = comfy.patcher_extension.WrapperExecutor.new_executor(
|
| | _prepare_sampling,
|
| | comfy.patcher_extension.get_all_wrappers(comfy.patcher_extension.WrappersMP.PREPARE_SAMPLING, model_options, is_model_options=True)
|
| | )
|
| | return executor.execute(model, noise_shape, conds, model_options=model_options)
|
| |
|
| | def _prepare_sampling(model: ModelPatcher, noise_shape, conds, model_options=None):
|
| | real_model: BaseModel = None
|
| | models, inference_memory = get_additional_models(conds, model.model_dtype())
|
| | models += get_additional_models_from_model_options(model_options)
|
| | models += model.get_nested_additional_models()
|
| | memory_required, minimum_memory_required = estimate_memory(model, noise_shape, conds)
|
| | comfy.model_management.load_models_gpu([model] + models, memory_required=memory_required + inference_memory, minimum_memory_required=minimum_memory_required + inference_memory)
|
| | real_model = model.model
|
| |
|
| | return real_model, conds, models
|
| |
|
| | def cleanup_models(conds, models):
|
| | cleanup_additional_models(models)
|
| |
|
| | control_cleanup = []
|
| | for k in conds:
|
| | control_cleanup += get_models_from_cond(conds[k], "control")
|
| |
|
| | cleanup_additional_models(set(control_cleanup))
|
| |
|
| | def prepare_model_patcher(model: ModelPatcher, conds, model_options: dict):
|
| | '''
|
| | Registers hooks from conds.
|
| | '''
|
| |
|
| | hooks = comfy.hooks.HookGroup()
|
| | for k in conds:
|
| | get_hooks_from_cond(conds[k], hooks)
|
| |
|
| | comfy.patcher_extension.merge_nested_dicts(model_options["transformer_options"].setdefault("wrappers", {}), model.wrappers, copy_dict1=False)
|
| | comfy.patcher_extension.merge_nested_dicts(model_options["transformer_options"].setdefault("callbacks", {}), model.callbacks, copy_dict1=False)
|
| |
|
| | registered = comfy.hooks.HookGroup()
|
| | target_dict = comfy.hooks.create_target_dict(comfy.hooks.EnumWeightTarget.Model)
|
| |
|
| | for hook in hooks.get_type(comfy.hooks.EnumHookType.TransformerOptions):
|
| | hook: comfy.hooks.TransformerOptionsHook
|
| | hook.add_hook_patches(model, model_options, target_dict, registered)
|
| |
|
| | for hook in hooks.get_type(comfy.hooks.EnumHookType.AdditionalModels):
|
| | hook: comfy.hooks.AdditionalModelsHook
|
| | hook.add_hook_patches(model, model_options, target_dict, registered)
|
| |
|
| | model.register_all_hook_patches(hooks, target_dict, model_options, registered)
|
| |
|
| | if len(registered) > 0:
|
| | model_options["registered_hooks"] = registered
|
| |
|
| | to_load_options: dict[str] = model_options.setdefault("to_load_options", {})
|
| | for wc_name in ["wrappers", "callbacks"]:
|
| | comfy.patcher_extension.merge_nested_dicts(to_load_options.setdefault(wc_name, {}), model_options["transformer_options"][wc_name],
|
| | copy_dict1=False)
|
| | return to_load_options
|
| |
|