import copy
import numpy as np
import warnings
from collections import OrderedDict, deque

import torch
from torch.nn import Parameter

from .bound_op_map import bound_op_map
from .bound_ops import *
from .bounded_tensor import BoundedTensor, BoundedParameter
from .parse_graph import parse_module
from .perturbations import *
from .utils import *
from .patches import Patches



warnings.simplefilter('once')

class BoundedModule(nn.Module):
    """Bounded module with support for automatically computing bounds.

    Args:
        model (nn.Module): The original model to be wrapped by BoundedModule.

        global_input (tuple): A dummy input to the original model. The shape of
        the dummy input should be consistent with the actual input to the model
        except for the batch dimension.

        bound_opts (dict): Options for bounds. See
        `Bound Options <bound_opts.html>`_.

        device (str or torch.device): Device of the bounded module.
        If 'auto', the device will be automatically inferred from the device of
        parameters in the original model or the dummy input.

        custom_ops (dict): A dictionary of custom operators.
        The dictionary maps operator names to their corresponding bound classes
        (subclasses of `Bound`).

    """
    def __init__(self, model, global_input, bound_opts=None,
                device='auto', verbose=False, custom_ops=None):
        super().__init__()
        if isinstance(model, BoundedModule):
            for key in model.__dict__.keys():
                setattr(self, key, getattr(model, key))
            return

        self.global_input = global_input
        self.ori_training = model.training
        self.check_incompatible_nodes(model)

        if bound_opts is None:
            bound_opts = {}
        # Default options.
        default_bound_opts = {
            'conv_mode': 'patches',
            'sparse_intermediate_bounds': False,
            'sparse_conv_intermediate_bounds': True,
            'sparse_intermediate_bounds_with_ibp': False,
            'sparse_features_alpha': True,
            'sparse_spec_alpha': False,
            'minimum_sparsity': 1.0,
            'enable_opt_interm_bounds': True,
            'crown_batch_size': np.inf,
            'forward_refinement': False,
            'dynamic_forward': False,
            'forward_max_dim': int(1e9),
            # Do not share alpha for conv layers.
            'use_full_conv_alpha': False,
            # Threshold for number of unstable neurons for each layer to disable
            #  use_full_conv_alpha.
            'use_full_conv_alpha_thresh': 512,
            'verbosity': 1 if verbose else 0,
        }
        default_bound_opts.update(bound_opts)
        self.bound_opts = default_bound_opts
        # print('all alpha {}'.format(self.bound_opts['use_full_conv_alpha']))
        self.verbose = verbose
        self.custom_ops = custom_ops if custom_ops is not None else {}
        if device == 'auto':
            try:
                self.device = next(model.parameters()).device
            except StopIteration:
                # Model has no parameters. We use the device of input tensor.
                self.device = global_input.device
        else:
            self.device = device
        self.conv_mode = self.bound_opts.get('conv_mode', 'patches')
        # Cached IBP results which may be reused
        self.ibp_lower, self.ibp_upper = None, None

        self.optimizable_activations = []
        self.optimize_layers_bounds = []
        self.track_bounds = False
        self.relus = []  # save relu layers for convenience

        state_dict_copy = copy.deepcopy(model.state_dict())
        object.__setattr__(self, 'ori_state_dict', state_dict_copy)
        model.to(self.device)
        self.final_shape = model(
            *unpack_inputs(global_input, device=self.device)).shape
        self.bound_opts.update({'final_shape': self.final_shape})
        self._convert(model, global_input)
        self._mark_perturbed_nodes()

        # set the default values here
        optimize_bound_args = {
            'enable_alpha_crown': True,  # Enable optimization of alpha.
            'enable_beta_crown': False,  # Enable beta split constraint.
            'iteration': 10,  # Number of alpha/beta optimization iterations.
            # Share some alpha variables to save memory at the cost of slightly
            # looser bounds.
            'use_shared_alpha': False,
            # Optimize coeffs during intermediate_refinement.
            'opt_coeffs': False,
            # Optimize constraint bias during intermediate_refinement.
            'opt_bias': False,
            # Optimizer used for alpha and beta optimization.
            'optimizer': 'adam',
            # Save best results of alpha/beta/bounds during optimization.
            'keep_best': True,
            # Only optimize bounds of last layer during alpha/beta CROWN.
            'fix_intermediate_layer_bounds': False,
            # Learning rate for the optimizable parameter alpha in alpha-CROWN.
            # 'lr_alpha': 0.5,
            'lr_alpha': 0.2,
            # Learning rate for the optimizable parameter beta in beta-CROWN.
            'lr_beta': 0.05,
            'lr_cut_beta': 5e-3,  # Learning rate for optimizing cut betas.
            # Initial alpha variables by calling CROWN once.
            'init_alpha': True,
            # Only split single nodes in branch and bound.
            'single_node_split': True,
            # Learning rate for intermediate layer beta for refinement.
            'lr_intermediate_beta': 0.1,
            'lr_coeffs': 0.01,  # Learning rate for coeffs for refinement
            # Optimize constraint bias in compute bounds.
            'intermediate_beta': False,
            # Layers to be refined, separated by commas.
            # -1 means preactivation before last relu.
            'intermediate_refinement_layers': [-1],
            # When batch size is not 1, this reduction function is applied to
            # reduce the bounds into a scalar.
            'loss_reduction_func': reduction_min,
            # Criteria function of early stop.
            'stop_criterion_func': lambda x: False,
            # Learning rate decay factor during bounds optimization.
            'lr_decay': 0.98,
            # Number of iterations that we will start considering early stop
            # if tracking no improvement.
            'early_stop_patience': 10,
            # Start to save optimized best bounds
            # when current_iteration > int(iteration*start_save_best)
            'start_save_best': 0.5,
            # Use double fp (float64) at the last iteration in alpha/beta CROWN.
            'use_float64_in_last_iteration': False,
            # Prune verified domain within iteration.
            'pruning_in_iteration': False,
            # Percentage of the minimum domains that can apply pruning.
            'pruning_in_iteration_threshold': 0.2,
            # For specification that will output multiple bounds for one
            # property, we use this function to prune them.
            'multi_spec_keep_func': lambda x: True
        }

        # change by bound_opts
        optimize_bound_args.update(
            self.bound_opts.get('optimize_bound_args', {}))
        self.bound_opts.update({'optimize_bound_args': optimize_bound_args})

        self.next_split_hint = []  # Split hints, used in beta optimization.
        # Beta values for all intermediate bounds.
        # Set to None (not used) by default.
        self.best_intermediate_betas = None
        # Initialization value for intermediate betas.
        self.init_intermediate_betas = None
        # whether using cut
        self.cut_used = False
        # a placeholder for cut timestamp, which would be a non-positive int
        self.cut_timestamp = -1

        # input node name (Used for adptive learning)
        self.input_node_name = None        
        # List of operators. When we are computing intermediate bounds for these
        # ops, we simply use IBP to propagate bounds from its input nodes,
        # instead of CROWN.
        # NOTE Currently only operators with a single input can be supported.
        self.ibp_intermediate = [
                BoundRelu,
                BoundNeg,
                BoundTranspose,
                BoundSin,
                BoundCos,
                BoundTan,
                BoundAtan]

        # a placeholder to save the latest samplewise mask for
        # pruning-in-iteration optimization
        self.last_update_preserve_mask = None
    
    def set_input_node_name(self, input_node_name):
        self.input_node_name = input_node_name

    def set_optimize_layers_bounds(self, optimize_layers_bounds):
        self.optimize_layers_bounds = optimize_layers_bounds

    def get_perturbed_optimizable_activations(self):
        return [n for n in self.optimizable_activations if n.perturbed]

    def get_splittable_activations(self):
        """
        Activation functions that can be split during branch and bound.
        """
        # TODO: Add other activation functions in a sysmatic manner.
        # Do not hard code relus or other functions.
        # A node must be perturbed, otherwise it is a constant and no need
        # to perturb.
        return [n for n in self.relus if n.perturbed]

    def get_layers_requiring_bounds(self):
        """
        Layer names whose intermediate layer bounds are required.
        """
        intermediate_layers = []
        for node in self.modules():
            for i in getattr(node, 'requires_input_bounds', []):
                input_node = node.inputs[i]
                if input_node.used and input_node.perturbed:
                    intermediate_layers.append(input_node)

        return intermediate_layers

    def check_incompatible_nodes(self, model):
        """Check whether the model has incompatible nodes that the conversion
        may be inaccurate"""
        node_types = [type(m) for m in list(model.modules())]

        if (torch.nn.Dropout in node_types
                and torch.nn.BatchNorm1d in node_types
                and self.global_input.shape[0] == 1):
            print('We cannot support torch.nn.Dropout and torch.nn.BatchNorm1d '
                  'at the same time!')
            print('Suggest to use another dummy input which has batch size '
                  'larger than 1 and set model to train() mode.')
            return

        if not self.ori_training and torch.nn.Dropout in node_types:
            print('Dropout operation CANNOT be parsed during conversion when '
                  'the model is in eval() mode!')
            print('Set model to train() mode!')
            self.ori_training = True

        if self.ori_training and torch.nn.BatchNorm1d in node_types:
            print('BatchNorm1d may raise error during conversion when the model'
                  ' is in train() mode!')
            print('Set model to eval() mode!')
            self.ori_training = False

    def non_deter_wrapper(self, op, *args, **kwargs):
        """Some operations are non-deterministic and deterministic mode will
        fail. So we temporary disable it."""
        if self.bound_opts.get('deterministic', False):
            torch.use_deterministic_algorithms(False)
        ret = op(*args, **kwargs)
        if self.bound_opts.get('deterministic', False):
            torch.use_deterministic_algorithms(True)
        return ret

    def non_deter_scatter_add(self, *args, **kwargs):
        return self.non_deter_wrapper(torch.scatter_add, *args, **kwargs)

    def non_deter_index_select(self, *args, **kwargs):
        return self.non_deter_wrapper(torch.index_select, *args, **kwargs)

    def set_bound_opts(self, new_opts):
        for k, v in new_opts.items():
            # assert v is not dict, 'only support change optimize_bound_args'
            if type(v) == dict:
                self.bound_opts[k].update(v)
            else:
                self.bound_opts[k] = v

    @staticmethod
    def _get_A_norm(A):
        if not isinstance(A, (list, tuple)):
            A = (A, )
        norms = []
        for aa in A:
            if aa is not None:
                if isinstance(aa, Patches):
                    aa = aa.patches
                norms.append(aa.abs().sum().item())
            else:
                norms.append(None)
        return norms

    def __call__(self, *input, **kwargs):
        if 'method_opt' in kwargs:
            opt = kwargs['method_opt']
            kwargs.pop('method_opt')
        else:
            opt = 'forward'
        for kwarg in [
            'disable_multi_gpu', 'no_replicas', 'get_property',
            'node_class', 'att_name']:
            if kwarg in kwargs:
                kwargs.pop(kwarg)
        if opt == 'compute_bounds':
            return self.compute_bounds(**kwargs)
        else:
            return self.forward(*input, **kwargs)

    def register_parameter(self, name, param):
        r"""Adds a parameter to the module.

        The parameter can be accessed as an attribute using given name.

        Args:
            name (string): name of the parameter. The parameter can be accessed
                from this module using the given name
            param (Parameter): parameter to be added to the module.
        """
        if '_parameters' not in self.__dict__:
            raise AttributeError(
                'cannot assign parameter before Module.__init__() call')

        elif not isinstance(name, torch._six.string_classes):
            raise TypeError('parameter name should be a string. '
                            f'Got {torch.typename(name)}')
        elif name == '':
            raise KeyError('parameter name can\'t be empty string')
        elif hasattr(self, name) and name not in self._parameters:
            raise KeyError(f'attribute "{name}" already exists')

        if param is None:
            self._parameters[name] = None
        elif not isinstance(param, Parameter):
            raise TypeError(
                f'cannot assign "{torch.typename(param)}" object to '
                f'parameter "{name}" '
                '(torch.nn.Parameter or None required)')
        elif param.grad_fn:
            raise ValueError(
                f'Cannot assign non-leaf Tensor to parameter "{name}". Model '
                'parameters must be created explicitly. To express "{name}" '
                'as a function of another Tensor, compute the value in '
                'the forward() method.')
        else:
            self._parameters[name] = param

    def load_state_dict(self, state_dict, strict=False):
        new_dict = OrderedDict()
        # translate name to ori_name
        for k, v in state_dict.items():
            if k in self.node_name_map:
                new_dict[self.node_name_map[k]] = v
        return super().load_state_dict(new_dict, strict=strict)

    def _named_members(self, get_members_fn, prefix='', recurse=True):
        r"""Helper method for yielding various names + members of modules."""
        memo = set()
        modules = self.named_modules(prefix=prefix) if recurse else [
                                     (prefix, self)]
        for module_prefix, module in modules:
            members = get_members_fn(module)
            for k, v in members:
                if v is None or v in memo:
                    continue
                memo.add(v)
                name = module_prefix + ('.' if module_prefix else '') + k
                # translate name to ori_name
                if name in self.node_name_map:
                    name = self.node_name_map[name]
                yield name, v

    def train(self, mode=True):
        super().train(mode)
        for node in self._modules.values():
            node.train(mode=mode)

    def eval(self):
        super().eval()
        for node in self._modules.values():
            node.eval()

    def to(self, *args, **kwargs):
        # Moves and/or casts some attributes except pytorch will do by default.
        for node in self._modules.values():
            for attr in ['lower', 'upper', 'forward_value', 'd', 'lA',]:
                if hasattr(node, attr):
                    this_attr = getattr(node, attr)
                    if isinstance(this_attr, torch.Tensor):
                        # print(node, attr)
                        this_attr = this_attr.to(*args, **kwargs)
                        setattr(node, attr, this_attr)

            if hasattr(node, 'interval'):
                # construct new interval
                this_attr = getattr(node, 'interval')
                setattr(node, 'interval', (this_attr[0].to(
                    *args, **kwargs), this_attr[1].to(*args, **kwargs)))

        return super().to(*args, **kwargs)

    def __getitem__(self, name):
        return self._modules[name]

    def final_node(self):
        return self[self.final_name]

    def get_forward_value(self, node):
        """ Recursively get `forward_value` for `node` and its parent nodes"""
        if getattr(node, 'forward_value', None) is not None:
            return node.forward_value
        inputs = [self.get_forward_value(inp) for inp in node.inputs]
        for inp in node.inputs:
            node.from_input = node.from_input or inp.from_input
        node.input_shape = inputs[0].shape if len(inputs) > 0 else None
        fv = node.forward(*inputs)
        if isinstance(fv, (torch.Size, tuple)):
            fv = torch.tensor(fv, device=self.device)
        node.forward_value = fv
        node.output_shape = fv.shape
        # In most cases, the batch dimension is just the first dimension
        # if the node depends on input. Otherwise if the node doesn't
        # depend on input, there is no batch dimension.
        node.batch_dim = 0 if node.from_input else -1
        # Unperturbed node but it is not a root node.
        # Save forward_value to value. (Can be used in forward bounds.)
        if not node.from_input and len(node.inputs) > 0:
            node.value = node.forward_value
        return fv

    def forward(self, *x, final_node_name=None, clear_forward_only=False,
            reset_perturbed_nodes=True):
        r"""Standard forward computation for the network.

        Args:
            x (tuple or None): Input to the model.

            final_node_name (str, optional): The name of the final node in the
            model. The value on the corresponding node will be returned.

            clear_forward_only (bool, default `False`): Whether only standard
            forward values stored on the nodes should be cleared. If `True`,
            only standard forward values stored on the nodes will be cleared.
            Otherwise, bound information on the nodes will also be cleared.

            reset_perturbed_nodes (bool, default `True`): Mark all perturbed
            nodes with input perturbations. When set to `True`, it may
            accidentally clear all .perturbed properties for intermediate
            nodes.

        Returns:
            output: The output of the model, or if `final_node_name` is not
            `None`, return the value on the corresponding node instead.
        """
        self._set_input(*x, clear_forward_only=clear_forward_only,
                reset_perturbed_nodes=reset_perturbed_nodes)
        if final_node_name:
            return self.get_forward_value(self[final_node_name])
        else:
            out = deque([self.get_forward_value(self[n])
                        for n in self.output_name])

            def _fill_template(template):
                if template is None:
                    return out.popleft()
                elif isinstance(template, (list, tuple)):
                    res = []
                    for t in template:
                        res.append(_fill_template(t))
                    return tuple(res) if isinstance(template, tuple) else res
                elif isinstance(template, dict):
                    res = {}
                    for key in template:
                        res[key] = _fill_template(template[key])
                    return res
                else:
                    raise NotImplementedError

            return _fill_template(self.output_template)

    def _mark_perturbed_nodes(self):
        """Mark the graph nodes and determine which nodes need perturbation."""
        degree_in = {}
        queue = deque()
        # Initially the queue contains all "root" nodes.
        for key in self._modules.keys():
            l = self[key]
            degree_in[l.name] = len(l.inputs)
            if degree_in[l.name] == 0:
                queue.append(l)  # in_degree ==0 -> root node

        while len(queue) > 0:
            node = queue.popleft()
            # Obtain all output node, and add the output nodes to the queue if
            # all its input nodes have been visited.
            # The initial "perturbed" property is set in BoundInput or
            # BoundParams object, depending on ptb.
            for name_next in node.output_name:
                node_next = self[name_next]
                if isinstance(node, BoundShape):
                    # Some nodes like Shape, even connected,
                    # do not really propagate bounds.
                    # TODO: make this a property of node?
                    pass
                else:
                    # The next node is perturbed if it is already perturbed,
                    # or this node is perturbed.
                    node_next.perturbed = node_next.perturbed or node.perturbed
                degree_in[name_next] -= 1
                # all inputs of this node have been visited,
                # now put it in queue.
                if degree_in[name_next] == 0:
                    queue.append(node_next)
            # Check whether weights are perturbed
            if isinstance(node, (BoundLinear, BoundConv, BoundBatchNormalization)):
                weight_perturbation = False
                for n in node.inputs[1:]:
                    if hasattr(n, 'perturbation'):
                        if n.perturbation is not None:
                            weight_perturbation = True
                if weight_perturbation:
                    node.requires_input_bounds = list(range(len(node.inputs)))
                else:
                    if isinstance(node, BoundMatMul):
                        node.requires_input_bounds = [1]
                    else:
                        node.requires_input_bounds = []

        self.splittable_activations =(
                self.get_splittable_activations())
        self.perturbed_optimizable_activations = (
                self.get_perturbed_optimizable_activations())
        return
    
    def _clear_stale_bounds(self):
        for l in self._modules.values():
            for attr in ['backup_lower', 'backup_upper']:
                if hasattr(l, attr):
                    delattr(l, attr)

    def _clear_and_set_new(
            self, intermediate_layer_bounds, clear_forward_only=False,
            reset_perturbed_nodes=True):
        for l in self._modules.values():
            if hasattr(l, 'linear'):
                if isinstance(l.linear, tuple):
                    for item in l.linear:
                        del item
                delattr(l, 'linear')

            if hasattr(l, 'patch_size'):
                l.patch_size = {}

            if clear_forward_only:
                if hasattr(l, 'forward_value'):
                    delattr(l, 'forward_value')
            else:
                for attr in [
                        'lower', 'upper', 'interval', 'forward_value', 'd',
                        'lA', 'lower_d']:
                    if attr in ['lower', 'upper'] and l.name in self.optimize_layers_bounds and self.track_bounds:
                        if hasattr(l, attr):
                            # print(f'set attribute lower')
                            if attr == 'lower':
                                l.backup_lower = l.lower.detach()
                            else:
                                l.backup_upper = l.upper.detach()
                    if hasattr(l, attr):
                        delattr(l, attr)

            for attr in [
                    'zero_backward_coeffs_l', 'zero_backward_coeffs_u',
                    'zero_lA_mtx', 'zero_uA_mtx']:
                setattr(l, attr, False)
            # Given an interval here to make IBP/CROWN start from this node                
            if (intermediate_layer_bounds is not None
                    and l.name in intermediate_layer_bounds.keys()):
                l.interval = tuple(intermediate_layer_bounds[l.name][:2])
                l.lower = intermediate_layer_bounds[l.name][0]
                l.upper = intermediate_layer_bounds[l.name][1]
                if l.lower is not None:
                    l.lower = l.lower.detach().requires_grad_(False)
                if l.upper is not None:
                    l.upper = l.upper.detach().requires_grad_(False)
            # Mark all nodes as non-perturbed except for weights.
            if reset_perturbed_nodes:
                if not hasattr(l, 'perturbation') or l.perturbation is None:
                    l.perturbed = False

            # Clear operator-specific attributes
            l.clear()

    def _set_input(
            self, *x, intermediate_layer_bounds=None,
            clear_forward_only=False, reset_perturbed_nodes=True):
        self._clear_and_set_new(
            intermediate_layer_bounds=intermediate_layer_bounds,
            clear_forward_only=clear_forward_only,
            reset_perturbed_nodes=reset_perturbed_nodes)
        inputs_unpacked = unpack_inputs(x)
        for name, index in zip(self.input_name, self.input_index):
            if index is None:
                continue
            node = self[name]
            node.value = inputs_unpacked[index]
            if isinstance(node.value, (BoundedTensor, BoundedParameter)):
                node.perturbation = node.value.ptb
            else:
                node.perturbation = None
        # Mark all perturbed nodes.
        if reset_perturbed_nodes:
            self._mark_perturbed_nodes()

    def _get_node_input(self, nodesOP, nodesIn, node):
        ret = []
        ori_names = []
        for i in range(len(node.inputs)):
            for op in nodesOP:
                if op.name == node.inputs[i]:
                    ret.append(op.bound_node)
                    break
            if len(ret) == i + 1:
                continue
            for io in nodesIn:
                if io.name == node.inputs[i]:
                    ret.append(io.bound_node)
                    ori_names.append(io.ori_name)
                    break
            if len(ret) <= i:
                raise ValueError(f'cannot find inputs of node: {node.name}')
        return ret

    def _to(self, obj, dest, inplace=False):
        """ Move all tensors in the object to a specified dest
        (device or dtype). The inplace=True option is available for dict."""
        if obj is None:
            return obj
        elif isinstance(obj, torch.Tensor):
            return obj.to(dest)
        elif isinstance(obj, Patches):
            return obj.patches.to(dest)
        elif isinstance(obj, tuple):
            return tuple([self._to(item, dest) for item in obj])
        elif isinstance(obj, list):
            return list([self._to(item, dest) for item in obj])
        elif isinstance(obj, dict):
            if inplace:
                for k, v in obj.items():
                    obj[k] = self._to(v, dest, inplace=True)
                return obj
            else:
                return {k: self._to(v, dest) for k, v in obj.items()}
        else:
            raise NotImplementedError(type(obj))

    def _convert_nodes(self, model, global_input):
        r"""
        Returns:
            nodesOP (list): List of operator nodes
            nodesIn (list): List of input nodes
            nodesOut (list): List of output nodes
            template (object): Template to specify the output format
        """
        global_input_cpu = self._to(global_input, 'cpu')
        if self.ori_training:
            model.train()
        else:
            model.eval()
        model.to('cpu')
        nodesOP, nodesIn, nodesOut, template = parse_module(
            model, global_input_cpu)
        model.to(self.device)
        for i in range(0, len(nodesIn)):
            if nodesIn[i].param is not None:
                nodesIn[i] = nodesIn[i]._replace(
                    param=nodesIn[i].param.to(self.device))
        global_input_unpacked = unpack_inputs(global_input)

        # Convert input nodes and parameters.
        for i, n in enumerate(nodesIn):
            if n.input_index is not None:
                nodesIn[i] = nodesIn[i]._replace(bound_node=BoundInput(
                    ori_name=nodesIn[i].ori_name,
                    value=global_input_unpacked[nodesIn[i].input_index],
                    perturbation=nodesIn[i].perturbation,
                    input_index=n.input_index))
            else:
                bound_class = BoundParams if isinstance(
                    nodesIn[i].param, nn.Parameter) else BoundBuffers
                nodesIn[i] = nodesIn[i]._replace(bound_node=bound_class(
                    ori_name=nodesIn[i].ori_name,
                    value=nodesIn[i].param,
                    perturbation=nodesIn[i].perturbation))

        unsupported_ops = []

        # Convert other operation nodes.
        for n in range(len(nodesOP)):
            attr = nodesOP[n].attr
            inputs = self._get_node_input(nodesOP, nodesIn, nodesOP[n])
            try:
                if nodesOP[n].op in self.custom_ops:
                    op = self.custom_ops[nodesOP[n].op]
                elif nodesOP[n].op in bound_op_map:
                    op = bound_op_map[nodesOP[n].op]
                elif nodesOP[n].op.startswith('aten::ATen'):
                    op = globals()[f'BoundATen{attr["operator"].capitalize()}']
                elif nodesOP[n].op.startswith('onnx::'):
                    op = globals()[f'Bound{nodesOP[n].op[6:]}']
                else:
                    raise KeyError
            except (NameError, KeyError):
                unsupported_ops.append(nodesOP[n])
                logger.error('The node has an unsupported operation: %s',
                             nodesOP[n])
                continue

            attr['device'] = self.device

            # FIXME generalize
            if (nodesOP[n].op == 'onnx::BatchNormalization'
                    or getattr(op, 'TRAINING_FLAG', False)):
                # BatchNormalization node needs model.training flag to set
                # running mean and vars set training=False to avoid wrongly
                # updating running mean/vars during bound wrapper
                nodesOP[n] = nodesOP[n]._replace(bound_node=op(
                    attr, inputs, nodesOP[n].output_index, self.bound_opts,
                    False))
            else:
                nodesOP[n] = nodesOP[n]._replace(bound_node=op(
                    attr, inputs, nodesOP[n].output_index, self.bound_opts))

        if unsupported_ops:
            logger.error('Unsupported operations:')
            for n in unsupported_ops:
                logger.error(f'Name: {n.op}, Attr: {n.attr}')
            raise NotImplementedError('There are unsupported operations')

        for node in nodesIn + nodesOP:
            node.bound_node.name = node.name

        nodes_dict = {}
        for node in nodesOP + nodesIn:
            nodes_dict[node.name] = node.bound_node
        nodesOP = [n.bound_node for n in nodesOP]
        nodesIn = [n.bound_node for n in nodesIn]
        nodesOut = [nodes_dict[n] for n in nodesOut]

        return nodesOP, nodesIn, nodesOut, template

    def _build_graph(self, nodesOP, nodesIn, nodesOut, template):
        # We were assuming that the original model had only one output node.
        # When there are multiple output nodes, this seems to be the first
        # output element. In this case, we are assuming that we aim to compute
        # the bounds for the first output element by default.
        self.final_name = nodesOut[0].name
        self.input_name, self.input_index, self.root_name = [], [], []
        self.output_name = [n.name for n in nodesOut]
        self.output_template = template
        for node in nodesIn:
            self.add_input_node(node, index=node.input_index)
        self.add_nodes(nodesOP)
        if self.conv_mode == 'patches':
            self.root_name = [node.name for node in nodesIn]

    # Make sure the nodes already have `name` and `input_name`
    def add_nodes(self, nodes):
        nodes = [(node if isinstance(node, Bound) else node.bound_node)
                  for node in nodes]
        for node in nodes:
            self._modules[node.name] = node
            node.output_name = []
            if len(node.inputs) == 0:
                self.root_name.append(node.name)
        for node in nodes:
            for l_pre in node.inputs:
                l_pre.output_name.append(node.name)
        for node in nodes:
            if isinstance(node, BoundOptimizableActivation):
                self.optimizable_activations.append(node)
            if isinstance(node, BoundRelu):
                self.relus.append(node)

    def add_input_node(self, node, index=None):
        self.add_nodes([node])
        self.input_name.append(node.name)
        # default value for input_index
        if index == 'auto':
            index = max([0] + [(i + 1)
                        for i in self.input_index if i is not None])
        self.input_index.append(index)

    def rename_nodes(self, nodesOP, nodesIn, rename_dict):
        def rename(node):
            node.name = rename_dict[node.name]
            return node
        for i in range(len(nodesOP)):
            nodesOP[i] = rename(nodesOP[i])
        for i in range(len(nodesIn)):
            nodesIn[i] = rename(nodesIn[i])

    def _split_complex(self, nodesOP, nodesIn):
        finished = True
        for n in range(len(nodesOP)):
            if hasattr(nodesOP[n], 'complex') and nodesOP[n].complex:
                finished = False
                _nodesOP, _nodesIn, _nodesOut, _ = self._convert_nodes(
                    nodesOP[n].model, nodesOP[n].input)
                # assuming each supported complex operation only has one output
                assert len(_nodesOut) == 1

                name_base = nodesOP[n].name + '/split'
                rename_dict = {}
                for node in _nodesOP + _nodesIn:
                    rename_dict[node.name] = name_base + node.name
                num_inputs = len(nodesOP[n].inputs)
                for i in range(num_inputs):
                    rename_dict[_nodesIn[i].name] = nodesOP[n].input_name[i]
                rename_dict[_nodesOP[-1].name] = nodesOP[n].name

                self.rename_nodes(_nodesOP, _nodesIn, rename_dict)

                output_name = _nodesOP[-1].name
                # Any input node of some node within the complex node should be
                # replaced with the complex node's corresponding input node.
                for node in _nodesOP:
                    for i in range(len(node.inputs)):
                        if node.input_name[i] in nodesOP[n].input_name:
                            index = nodesOP[n].input_name.index(
                                node.input_name[i])
                            node.inputs[i] = nodesOP[n].inputs[index]
                # For any output node of this complex node,
                # modify its input node.
                for node in nodesOP:
                    if output_name in node.input_name:
                        index = node.input_name.index(output_name)
                        node.inputs[index] = _nodesOP[-1]

                nodesOP = nodesOP[:n] + _nodesOP + nodesOP[(n + 1):]
                nodesIn = nodesIn + _nodesIn[num_inputs:]

                break

        return nodesOP, nodesIn, finished

    def _get_node_name_map(self):
        """Build a dict with {ori_name: name, name: ori_name}"""
        self.node_name_map = {}
        for node in self._modules.values():
            if isinstance(node, BoundInput) or isinstance(node, BoundParams):
                for p in list(node.named_parameters()):
                    if node.ori_name not in self.node_name_map:
                        name = f'{node.name}.{p[0]}'
                        self.node_name_map[node.ori_name] = name
                        self.node_name_map[name] = node.ori_name
                for p in list(node.named_buffers()):
                    if node.ori_name not in self.node_name_map:
                        name = f'{node.name}.{p[0]}'
                        self.node_name_map[node.ori_name] = name
                        self.node_name_map[name] = node.ori_name

    # convert a Pytorch model to a model with bounds
    def _convert(self, model, global_input):
        if self.verbose:
            logger.info('Converting the model...')

        if not isinstance(global_input, tuple):
            global_input = (global_input,)
        self.num_global_inputs = len(global_input)

        nodesOP, nodesIn, nodesOut, template = self._convert_nodes(
            model, global_input)
        global_input = self._to(global_input, self.device)

        while True:
            self._build_graph(nodesOP, nodesIn, nodesOut, template)
            self.forward(*global_input)  # running means/vars changed
            nodesOP, nodesIn, finished = self._split_complex(nodesOP, nodesIn)
            if finished:
                break

        self._get_node_name_map()

        # Load self.ori_state_dict again to avoid the running means/vars changed
        # during forward().
        self.load_state_dict(self.ori_state_dict)
        if self.ori_training:
            model.load_state_dict(self.ori_state_dict)
        delattr(self, 'ori_state_dict')

        # The final node used in the last time calling `compute_bounds`
        self.last_final_node = None
        self.used_nodes = []

        if self.verbose:
            logger.info('Model converted to support bounds')

    def check_prior_bounds(self, node, baseline_refined_bound={}, 
                           intermediate_bound_refinement=False):
        if node.prior_checked or not (node.used and node.perturbed):
            return
        # print(f'prior bounds {node.name}')
        for n in node.inputs:
            self.check_prior_bounds(n, 
                                baseline_refined_bound=baseline_refined_bound,
                                intermediate_bound_refinement=intermediate_bound_refinement)
        for i in getattr(node, 'requires_input_bounds', []):
            self.compute_intermediate_bounds(
                node.inputs[i], prior_checked=True, 
                baseline_refined_bound=baseline_refined_bound,
                intermediate_bound_refinement=intermediate_bound_refinement)
        node.prior_checked = True

    def compute_intermediate_bounds(self, node, prior_checked=False, 
                                    baseline_refined_bound={}, intermediate_bound_refinement=False):
        if getattr(node, 'lower', None) is not None:
            return

        logger.debug(f'Getting the bounds of {node}')

        if not prior_checked:
            self.check_prior_bounds(node, 
                                    baseline_refined_bound=baseline_refined_bound,
                                     intermediate_bound_refinement=intermediate_bound_refinement)

        if not node.perturbed:
            fv = self.get_forward_value(node)
            node.interval = node.lower, node.upper = fv, fv
            return

        # FIXME check that weight perturbation is not affected
        #      (from_input=True should be set for weights)
        if not node.from_input and hasattr(node, 'forward_value'):
            node.lower = node.upper = self.get_forward_value(node)
            return

        reference_bounds = self.reference_bounds


        if self.use_forward:
            node.lower, node.upper = self.forward_general(
                node=node, concretize=True)
            return

        #FIXME need clean up

        # assign concretized bound for ReLU layer to save computational cost
        # FIXME: Put ReLU after reshape will cause problem!
        if self.check_IBP_intermediate(node):
            # Intermediate bounds for some operators are directly
            # computed from their input nodes by IBP
            # (such as BoundRelu, BoundNeg)
            logger.debug('IBP propagation for intermediate bounds on %s', node)
        elif (isinstance(node, BoundReshape)
                and hasattr(node.inputs[0], 'lower')
                and hasattr(node.inputs[1], 'value')):
            # TODO merge this with `check_IBP_intermediate`
            # Node for input value.
            val_input = node.inputs[0]
            # Node for input parameter (e.g., shape, permute)
            arg_input = node.inputs[1]
            node.lower = node.forward(val_input.lower, arg_input.value)
            node.upper = node.forward(val_input.upper, arg_input.value)
            node.interval = (node.lower, node.upper)
        else:
            # For the first linear layer, IBP can give the same tightness
            # as CROWN.
            if not self.check_IBP_first_linear(node):
                sparse_intermediate_bounds_with_ibp = self.bound_opts.get(
                    'sparse_intermediate_bounds_with_ibp', True)
                # Sparse intermediate bounds can be enabled
                # if aux_reference_bounds are given.
                # (this is enabled for ReLU only, and not for other
                # activations.)
                sparse_intermediate_bounds = (self.bound_opts.get(
                    'sparse_intermediate_bounds', False)
                    and isinstance(self[node.output_name[0]], BoundRelu))

                ref_intermediate_lb, ref_intermediate_ub = None, None
                if sparse_intermediate_bounds:
                    if node.name not in self.aux_reference_bounds:
                        # If aux_reference_bounds are not available,
                        # we can use IBP to compute these bounds.
                        if sparse_intermediate_bounds_with_ibp:
                            with torch.no_grad():
                                # Get IBP bounds for this layer;
                                # we set delete_bounds_after_use=True which does
                                # not save extra intermediate bound tensors.
                                ret_ibp = self.IBP_general(
                                    node=node, delete_bounds_after_use=True)
                                ref_intermediate_lb = ret_ibp[0]
                                ref_intermediate_ub = ret_ibp[1]
                        else:
                            sparse_intermediate_bounds = False
                    else:
                        aux_bounds = self.aux_reference_bounds[node.name]
                        ref_intermediate_lb, ref_intermediate_ub = aux_bounds

                sparse_C = self.get_sparse_C(
                    node, sparse_intermediate_bounds,
                    ref_intermediate_lb, ref_intermediate_ub)
                newC, reduced_dim, unstable_idx, unstable_size = sparse_C
                
                if unstable_idx is None or unstable_size > 0:
                    if self.return_A:
                        low_bnd, upper_bound, _ = self.backward_general(
                            C=newC, node=node, unstable_idx=unstable_idx,
                            unstable_size=unstable_size)
                    else:
                        # Compute backward bounds only when there are unstable
                        # neurons, or when we don't know which neurons are unstable.
                        low_bnd, upper_bound = self.backward_general(
                            C=newC, node=node, unstable_idx=unstable_idx,
                            unstable_size=unstable_size)

                    if not intermediate_bound_refinement:
                        low_bnd, upper_bound = low_bnd.detach(), upper_bound.detach()

                    if hasattr(node, 'backup_lower') and hasattr(node, 'backup_upper'):
                        # print(f'Updated bounds {node.name}')
                        # print(f'backup shape {node.backup_lower.shape} shape {low_bnd.shape}')
                        node.lower = torch.max(node.backup_lower, low_bnd)
                        node.upper = torch.min(node.backup_upper, upper_bound)
                    else:
                        node.lower, node.upper = low_bnd, upper_bound
                    if node.name in baseline_refined_bound.keys():
                        bnds = baseline_refined_bound[node.name]
                        node.lower = torch.max(bnds[0], node.lower)
                        node.upper = torch.min(bnds[1], node.upper)
                if reduced_dim:
                    self.restore_sparse_bounds(
                        node, unstable_idx, unstable_size,
                        ref_intermediate_lb, ref_intermediate_ub)

        # node.lower and node.upper (intermediate bounds) are computed in
        # the above function. If we have bound references, we set them here
        # to always obtain a better set of bounds.
        if node.name in reference_bounds:
            ref_bounds = reference_bounds[node.name]
            # Initially, the reference bound and the computed bound can be
            # exactly the same when intermediate layer beta is 0. This will
            # prevent gradients flow. So we need a small guard here.
            if self.intermediate_constr is not None:
                # Intermediate layer beta is used.
                # Note that we cannot just take the reference bounds if
                # they are better - this makes alphas have zero gradients.
                new_lower = 0.9 * ref_bounds[0] + 0.1 * node.lower
                new_upper = 0.9 * ref_bounds[1] + 0.1 * node.upper
                node.lower = torch.max(new_lower, node.lower)
                node.upper = torch.min(new_upper, node.upper)
                # Additionally, if the reference bounds say a neuron is
                # stable, we always keep it. (FIXME: this is for ReLU only).
                lower_stable = ref_bounds[0] >= 0.
                node.lower[lower_stable] = ref_bounds[0][lower_stable]
                upper_stable = ref_bounds[1] <= 0.
                node.upper[upper_stable] = ref_bounds[1][upper_stable]
            else:
                # Set the intermediate layer bounds using reference bounds,
                # always choosing the tighter one.
                node.lower = (
                    torch.max(ref_bounds[0], node.lower).detach()
                    - node.lower.detach() + node.lower)
                node.upper = (
                    node.upper - (node.upper.detach()
                    - torch.min(ref_bounds[1], node.upper).detach()))
            # Otherwise, we only use reference bounds to check which neurons
            # are unstable.

        # FIXME (12/28): we should be consistent, and only use
        # node.interval, do not use node.lower or node.upper!
        node.interval = (node.lower, node.upper)

    def merge_A_dict(self, lA_dict, uA_dict):
        merged_A = {}
        for output_node_name in lA_dict:
            merged_A[output_node_name] = {}
            lA_dict_ = lA_dict[output_node_name]
            uA_dict_ = uA_dict[output_node_name]
            for input_node_name in lA_dict_:
                merged_A[output_node_name][input_node_name] = {
                    'lA': lA_dict_[input_node_name]['lA'],
                    'uA': uA_dict_[input_node_name]['uA'],
                    'lbias': lA_dict_[input_node_name]['lbias'],
                    'ubias': uA_dict_[input_node_name]['ubias'],
                }
        return merged_A

    def detach_bounds(self):
        for _, node in self._modules.items():
            if hasattr(node, 'lower'):
                if type(node.lower) is torch.Tensor:
                    node.lower = node.lower.detach()

            if hasattr(node, 'upper'):
                if type(node.upper) is torch.Tensor:
                    node.upper = node.upper.detach()


    def compute_bounds(
            self, x=None, aux=None, C=None, method='backward', IBP=False,
            forward=False, bound_lower=True, bound_upper=True, reuse_ibp=False,
            reuse_alpha=False, return_A=False, needed_A_dict=None,
            final_node_name=None, average_A=False,
            intermediate_layer_bounds=None, reference_bounds=None,
            intermediate_constr=None, alpha_idx=None,
            aux_reference_bounds=None, need_A_only=False,
            cutter=None, decision_thresh=None,
            update_mask=None, multiple_execution=False, execution_count=1, 
            ptb=None, unperturbed_images=None, iteration=None, 
            baseline_refined_bound={}, intermediate_bound_refinement=False, 
            always_correct_cross_execution=False, cross_refinement_results={}, 
            populate_trace=False, target_lAs=None, target_biases=None, parent_loss=None):
        r"""Main function for computing bounds.

        Args:
            x (tuple or None): Input to the model. If it is None, the input
            from the last `forward` or `compute_bounds` call is reused.
            Otherwise: the number of elements in the tuple should be
            equal to the number of input nodes in the model, and each element in
            the tuple corresponds to the value for each input node respectively.
            It should look similar as the `global_input` argument when used for
            creating a `BoundedModule`.

            aux (object, optional): Auxliary information that can be passed to
            `Perturbation` classes for initializing and concretizing bounds,
            e.g., additional information for supporting synonym word subsitution
            perturbaiton.

            C (Tensor): The specification matrix that can map the output of the
            model with an additional linear layer. This is usually used for
            maping the logits output of the model to classification margins.

            method (str): The main method for bound computation. Choices:
                * `IBP`: purely use Interval Bound Propagation (IBP) bounds.
                * `CROWN-IBP`: use IBP to compute intermediate bounds,
                but use CROWN (backward mode LiRPA) to compute the bounds of the
                final node.
                * `CROWN`: purely use CROWN to compute bounds for intermediate
                nodes and the final node.
                * `Forward`: purely use forward mode LiRPA.
                * `Forward+Backward`: use forward mode LiRPA for intermediate
                nodes, but further use CROWN for the final node.
                * `CROWN-Optimized` or `alpha-CROWN`: use CROWN, and also
                optimize the linear relaxation parameters for activations.
                * `forward-optimized`: use forward bounds with optimized linear
                relaxation.

            IBP (bool, optional): If `True`, use IBP to compute the bounds of
            intermediate nodes. It can be automatically set according to
            `method`.

            forward (bool, optional): If `True`, use the forward mode bound
            propagation to compute the bounds of intermediate nodes. It can be
            automatically set according to `method`.

            bound_lower (bool, default `True`): If `True`, the lower bounds of
            the output needs to be computed.

            bound_upper (bool, default `True`): If `True`, the upper bounds of
            the output needs to be computed.

            reuse_ibp (bool, optional): If `True` and `method` is None, reuse
            the previously saved IBP bounds.

            final_node_name (str, optional): Set the final node in the
            computational graph for bound computation. By default, the final
            node of the originally built computational graph is used.

            return_A (bool, optional): If `True`, return linear coefficients
            in bound propagation (`A` tensors) with `needed_A_dict` set.

            needed_A_dict (dict, optional): A dictionary specifying linear
            coefficients (`A` tensors) that are needed and should be returned.
            Each key in the dictionary is the name of a starting node in
            backward bound propagation, with a list as the value for the key,
            which specifies the names of the ending nodes in backward bound
            propagation, and the linear coefficients of the starting node w.r.t.
            the specified ending nodes are returned. By default, it is empty.

            reuse_alpha (bool, optional): If `True`, reuse previously saved
            alpha values when they are not being optimized.

            decision_thresh (float, optional): In CROWN-optimized mode, we will
            use this decision_thresh to dynamically optimize those domains that
            <= the threshold.

            intermediate_layer_bounds: A dictionary of 2-element tuple/list
            containing lower and upper bounds for intermediate layers.
            The dictionary keys should include the names of the layers whose
            bounds should be set without recomputation. The layer names can be
            viewed by setting environment variable AUTOLIRPA_DEBUG_GRAPH=1.
            The values of each dictionary elements are (lower_bounds,
            upper_bounds) where "lower_bounds" and "upper_bounds" are two
            tensors with the same shape as the output shape of this layer. If
            you only need to set intermediate layer bounds for certain layers,
            then just include these layers' names in the dictionary.

            reference_bounds: Format is similar to "intermediate_layer_bounds".
            However, these bounds are only used as a reference, and the bounds
            for intermediate layers will still be computed (e.g., using CROWN,
            IBP or other specified methods). The computed bounds will be
            compared to "reference_bounds" and the tighter one between the two
            will be used.

            aux_reference_bounds: Format is similar to intermediate layer
            bounds. However, these bounds are only used for determine which
            neurons are stable and which neurons are unstable for ReLU networks.
            Unstable neurons' intermediate layer bounds will be recomputed.

        Returns:
            bound (tuple): When `return_A` is `False`, return a tuple of
            the computed lower bound and upper bound. When `return_A`
            is `True`, return a tuple of lower bound, upper bound, and
            `A` dictionary.
        """
        logger.debug(f'Compute bounds with {method}')

        if method != 'backward' or not self.track_bounds:
            self._clear_stale_bounds()

        if iteration is not None:
            self.bound_opts['optimize_bound_args']['iteration'] = iteration
            # it = self.bound_opts['optimize_bound_args']['iteration']
            # print(f'iteration {it}')

        if needed_A_dict is None: needed_A_dict = {}
        if not bound_lower and not bound_upper:
            raise ValueError(
                'At least one of bound_lower and bound_upper must be True')

        # Several shortcuts.
        method = method.lower() if method is not None else method
        if method == 'ibp':
            # Pure IBP bounds.
            method = None
            IBP = True
        elif method in ['ibp+backward', 'ibp+crown', 'crown-ibp']:
            method = 'backward'
            IBP = True
        elif method == 'crown':
            method = 'backward'
        elif method == 'forward':
            forward = True
        elif method == 'forward+backward' or method == 'forward+crown':
            method = 'backward'
            forward = True
        elif method in ['crown-optimized', 'alpha-crown', 'forward-optimized']:
            # Lower and upper bounds need two separate rounds of optimization.
            if method == 'forward-optimized':
                method = 'forward'
            else:
                method = 'backward'
            if bound_lower:
                ret1 = self.get_optimized_bounds(
                    x=x, C=C, method=method,
                    intermediate_layer_bounds=intermediate_layer_bounds,
                    reference_bounds=reference_bounds, bound_lower=bound_lower,
                    bound_upper=False, return_A=return_A,
                    aux_reference_bounds=aux_reference_bounds,
                    needed_A_dict=needed_A_dict,
                    final_node_name=final_node_name,
                    cutter=cutter, decision_thresh=decision_thresh, 
                    multiple_execution=multiple_execution, execution_count=execution_count, 
                    ptb=ptb, unperturbed_images=unperturbed_images, 
                    baseline_refined_bound=baseline_refined_bound, 
                    intermediate_bound_refinement=intermediate_bound_refinement,
                    always_correct_cross_execution=always_correct_cross_execution,
                    cross_refinement_results=cross_refinement_results, populate_trace=populate_trace, 
                    target_lAs=target_lAs, target_biases=target_biases, parent_loss=parent_loss)
            if bound_upper:
                ret2 = self.get_optimized_bounds(
                    x=x, C=C, method=method,
                    intermediate_layer_bounds=intermediate_layer_bounds,
                    reference_bounds=reference_bounds, bound_lower=False,
                    bound_upper=bound_upper, return_A=return_A,
                    aux_reference_bounds=aux_reference_bounds,
                    needed_A_dict=needed_A_dict,
                    final_node_name=final_node_name,
                    cutter=cutter, decision_thresh=decision_thresh, 
                    multiple_execution=multiple_execution, execution_count=execution_count,
                    ptb=ptb, unperturbed_images=unperturbed_images, 
                    baseline_refined_bound=baseline_refined_bound,
                    intermediate_bound_refinement=intermediate_bound_refinement,
                    always_correct_cross_execution=always_correct_cross_execution, 
                    cross_refinement_results=cross_refinement_results, 
                    populate_trace=populate_trace, target_lAs=target_lAs, 
                    target_biases=target_biases, parent_loss=parent_loss)

            if bound_lower and bound_upper:
                if return_A:
                    # Needs to merge the A dictionary.
                    lA_dict = ret1[2]
                    uA_dict = ret2[2]
                    merged_A = self.merge_A_dict(lA_dict, uA_dict)
                    return ret1[0], ret2[1], merged_A
                else:
                    return ret1[0], ret2[1]
            elif bound_lower:
                return ret1  # ret1[1] is None.
            elif bound_upper:
                return ret2  # ret2[0] is None.

        if reference_bounds is None:
            reference_bounds = {}
        if aux_reference_bounds is None:
            aux_reference_bounds = {}
        

        # If y in self.backward_node_pairs[x], then node y is visited when
        # doing backward bound propagation starting from node x.
        self.backward_from = dict([(node, []) for node in self._modules])

        if not bound_lower and not bound_upper:
            raise ValueError(
                'At least one of bound_lower and bound_upper in compute_bounds '
                'should be True')
        A_dict = {} if return_A else None

        if x is not None:
            self._set_input(
                *x, intermediate_layer_bounds=intermediate_layer_bounds)

        if IBP and method is None and reuse_ibp:
            # directly return the previously saved ibp bounds
            return self.ibp_lower, self.ibp_upper
        root = [self[name] for name in self.root_name]
        batch_size = root[0].value.shape[0]
        dim_in = 0

        for i in range(len(root)):
            value = root[i].forward()
            if getattr(root[i], 'perturbation', None) is not None:
                ret_init = root[i].perturbation.init(
                    value, aux=aux, forward=forward)
                root[i].linear, root[i].center, root[i].aux = ret_init
                # This input/parameter has perturbation.
                # Create an interval object.
                root[i].interval = Interval(
                    root[i].linear.lower, root[i].linear.upper,
                    ptb=root[i].perturbation)
                if forward:
                    root[i].dim = root[i].linear.lw.shape[1]
                    dim_in += root[i].dim
            else:
                # This input/parameter does not has perturbation.
                # Use plain tuple defaulting to Linf perturbation.
                root[i].interval = (value, value)
                root[i].forward_value = root[i].value = value
                root[i].center = root[i].lower = root[i].upper = value

            root[i].lower, root[i].upper = root[i].interval

        if forward:
            self.init_forward(root, dim_in)

        final = self.final_node(
        ) if final_node_name is None else self[final_node_name]
        logger.debug(f'Final node {final.__class__.__name__}({final.name})')

        if IBP:
            self.ibp_lower, self.ibp_upper = self.IBP_general(node=final, C=C)

        if method is None:
            return self.ibp_lower, self.ibp_upper

        if C is None:
            # C is an identity matrix by default
            if final.output_shape is None:
                raise ValueError(
                    f'C is not missing while node {final} has no default shape')
            dim_output = int(prod(final.output_shape[1:]))
            # TODO: use an eyeC object here.
            C = torch.eye(dim_output, device=self.device).expand(
                batch_size, dim_output, dim_output)

        # Reuse previously saved alpha values,
        # even if they are not optimized now
        if reuse_alpha:
            for node in self.optimizable_activations:
                node.opt_reuse()
        else:
            for node in self.optimizable_activations:
                node.opt_no_reuse()

        # Inject update mask inside the activations
        # update_mask: None or  bool tensor([batch_size])
        # If set to a tensor, only update the alpha and beta of selected
        # element (with element=1).

        if update_mask is None:
            for node in self.optimizable_activations:
                node.clean_alpha_beta_update_mask()
        else:
            for node in self.optimizable_activations:
                node.set_alpha_beta_update_mask(update_mask)

        for n in self._modules.values():
            # Check whether all prior intermediate bounds already exist
            n.prior_checked = False
            if isinstance(i, BoundRelu):
                for node in i.inputs:
                    if isinstance(node, BoundConv):
                        # whether this Conv is followed by a ReLU
                        node.relu_followed = True

        # BFS to find out whether each node is used given the current final node
        self._set_used_nodes(final)

        # FIXME clean
        self.use_forward = forward
        self.root = root
        self.batch_size = batch_size
        self.dim_in = dim_in
        self.return_A = return_A
        self.A_dict = A_dict
        self.needed_A_dict = needed_A_dict
        self.intermediate_constr = intermediate_constr
        self.reference_bounds = reference_bounds
        self.aux_reference_bounds = aux_reference_bounds
        self.final_node_name = final.name

        # print(f'bound dict keys {baseline_refined_bound.keys()}')
        self.check_prior_bounds(final, 
                                baseline_refined_bound=baseline_refined_bound,
                                intermediate_bound_refinement=intermediate_bound_refinement)

        if method == 'backward':
            # This is for the final output bound.
            # No need to pass in intermediate layer beta constraints.
            ret = self.backward_general(
                C=C, node=final,
                bound_lower=bound_lower, bound_upper=bound_upper,
                average_A=average_A, need_A_only=need_A_only,
                unstable_idx=alpha_idx, update_mask=update_mask, 
                multiple_execution=multiple_execution, targeted_optimization_enabled=(target_lAs is not None))
            # FIXME when C is specified, lower and upper should not be saved to
            # final.lower and final.upper, because they are not the bounds for
            # the node.
            final.lower, final.upper = ret[0], ret[1]
            return ret
        elif method == 'forward':
            return self.forward_general(C=C, node=final, concretize=True)
        else:
            raise NotImplementedError

    def _set_used_nodes(self, final):
        if final.name != self.last_final_node:
            self.last_final_node = final.name
            self.used_nodes = []
            for i in self._modules.values():
                i.used = False
            final.used = True
            queue = deque([final])
            while len(queue) > 0:
                n = queue.popleft()
                self.used_nodes.append(n)
                for n_pre in n.inputs:
                    if not n_pre.used:
                        n_pre.used = True
                        queue.append(n_pre)
        # Based on "used" and "perturbed" properties, find out which
        # layer requires intermediate layer bounds.
        self.layers_requiring_bounds = (
                self.get_layers_requiring_bounds())

    def add_intermediate_perturbation(self, node, perturbation):
        """Add perturbation to an intermediate node and it is treated as an
        independent node in bound computation."""
        node.perturbation = perturbation
        node.perturbed = True
        # NOTE This change is currently inreversible
        if not node.name in self.root_name:
            self.root_name.append(node.name)

    from .interval_bound import (
        IBP_general, _IBP_loss_fusion, check_IBP_intermediate,
        check_IBP_first_linear)
    from .forward_bound import (
        forward_general, forward_general_dynamic, init_forward)
    from .backward_bound import (
        backward_general, get_sparse_C, check_optimized_variable_sparsity,
        restore_sparse_bounds, get_alpha_crown_start_nodes,
        get_unstable_locations, batched_backward)
    from .optimized_bounds import (
        get_optimized_bounds, init_slope, get_cross_execution_params,
        extract_mininum_coef, get_cross_execution_loss, 
        cross_execution_loss_helper, print_bound_updates, get_final_cross_executional_loss, get_targeted_loss) 
    from .beta_crown import (
        beta_bias, save_best_intermediate_betas,
        print_optimized_beta)
    from .jacobian import augment_gradient_graph, compute_jacobian_bounds


    from .solver_module import build_solver_module, _build_solver_input, _build_solver_general
