Source code for pybamm.discretisations.discretisation

#
# Interface for discretisation
#
import itertools
from collections import OrderedDict, defaultdict

import numpy as np
from scipy.sparse import block_diag, csr_matrix

import pybamm
from pybamm.models.base_model import ModelSolutionObservability


def has_bc_of_form(symbol, side, bcs, form):
    return (symbol in bcs) and (bcs[symbol][side][1] == form)


[docs] class Discretisation: """The discretisation class, with methods to process a model and replace Spatial Operators with Matrices and Variables with StateVectors Parameters ---------- mesh : pybamm.Mesh contains all submeshes to be used on each domain spatial_methods : dict a dictionary of the spatial methods to be used on each domain. The keys correspond to the model domains and the values to the spatial method. check_model : bool, optional If True, model checks are performed after discretisation. For large systems these checks can be slow, so can be skipped by setting this option to False. When developing, testing or debugging it is recommended to leave this option as True as it may help to identify any errors. Default is True. remove_independent_variables_from_rhs : bool, optional If True, model checks to see whether any variables from the RHS are used in any other equation. If a variable meets all of the following criteria (not used anywhere in the model, len(rhs)>1), then the variable is moved to be explicitly integrated when called by the solution object. Default is False. resolve_coupled_variables : bool, optional If True, resolve CoupledVariables in rhs, algebraic, initial_conditions, and boundary_conditions before processing. Default is False. """ def __init__( self, mesh=None, spatial_methods=None, check_model=True, remove_independent_variables_from_rhs=False, resolve_coupled_variables=False, ): self._mesh = mesh if mesh is None: self._spatial_methods = {} else: # Unpack macroscale to the constituent subdomains if "macroscale" in spatial_methods.keys(): method = spatial_methods["macroscale"] spatial_methods["negative electrode"] = method spatial_methods["separator"] = method spatial_methods["positive electrode"] = method self._spatial_methods = spatial_methods for domain, method in self._spatial_methods.items(): method.build(mesh) # Check zero-dimensional methods are only applied to zero-dimensional # meshes if isinstance(method, pybamm.ZeroDimensionalSpatialMethod): if not isinstance(mesh[domain], pybamm.SubMesh0D): raise pybamm.DiscretisationError( "Zero-dimensional spatial method for the " f"{domain} domain requires a zero-dimensional submesh" ) self._bcs = {} self.y_slices = {} self._discretised_symbols = {} self._check_model_flag = check_model self._remove_independent_variables_from_rhs_flag = ( remove_independent_variables_from_rhs ) self._resolve_coupled_variables = resolve_coupled_variables @property def mesh(self): return self._mesh @property def y_slices(self): return self._y_slices @y_slices.setter def y_slices(self, value): if not isinstance(value, dict): raise TypeError(f"y_slices should be dict, not {type(value)}") self._y_slices = value @property def spatial_methods(self): return self._spatial_methods @property def bcs(self): return self._bcs @bcs.setter def bcs(self, value): self._bcs = value # reset discretised_symbols self._discretised_symbols = {}
[docs] def process_model( self, model, inplace=True, delayed_variable_processing=None, ): """ Discretise a model. Currently inplace, could be changed to return a new model. Parameters ---------- model : :class:`pybamm.BaseModel` Model to dicretise. Must have attributes rhs, initial_conditions and boundary_conditions (all dicts of {variable: equation}) inplace : bool, optional If True, discretise the model in place. Otherwise, return a new discretised model. Default is True. delayed_variable_processing: bool, optional If True, make variable processing a post-processing step. Default is False. Returns ------- model_disc : :class:`pybamm.BaseModel` The discretised model. Note that if ``inplace`` is True, model will have also been discretised in place so model == model_disc. If ``inplace`` is False, model != model_disc Raises ------ :class:`pybamm.ModelError` If an empty model is passed (`model.rhs = {}` and `model.algebraic = {}` and `model.variables = {}`) """ if model.is_discretised is True: raise pybamm.ModelError( "Cannot re-discretise a model. " "Set 'inplace=False' when first discretising a model to then be able " "to discretise it more times (e.g. for convergence studies)." ) if delayed_variable_processing is None: delayed_variable_processing = False pybamm.logger.info(f"Start discretising {model.name}") # Make sure model isn't empty if ( len(model.rhs) == 0 and len(model.algebraic) == 0 and len(model.variables) == 0 ): raise pybamm.ModelError("Cannot discretise empty model") # Check well-posedness to avoid obscure errors model.check_well_posedness() # Prepare discretisation # set variables (we require the full variable not just id) # Search Equations for Independence if self._remove_independent_variables_from_rhs_flag: model = self.remove_independent_variables_from_rhs(model) variables = list(model.rhs.keys()) + list(model.algebraic.keys()) # Find those RHS's that are constant if self.spatial_methods == {} and any(var.domain != [] for var in variables): for var in variables: if var.domain != []: raise pybamm.DiscretisationError( "Spatial method has not been given " f"for variable {var.name} with domain {var.domain}" ) # Set the y split for variables pybamm.logger.verbose(f"Set variable slices for {model.name}") self.set_variable_slices(variables) # set boundary conditions (only need key ids for boundary_conditions) pybamm.logger.verbose(f"Discretise boundary conditions for {model.name}") self._bcs = self.process_boundary_conditions(model) pybamm.logger.verbose(f"Set internal boundary conditions for {model.name}") self.set_internal_boundary_conditions(model) # set up inplace vs not inplace if inplace: # any changes to model_disc attributes will change model attributes # since they point to the same object model_disc = model else: # create a copy of the original model model_disc = model.new_copy() if self._resolve_coupled_variables: self._resolve_coupled_variables_in_model(model) # Keep a record of y_slices in the model model_disc.y_slices = self.y_slices_explicit # Keep a record of the bounds in the model model_disc.bounds = self.bounds model_disc.bcs = self.bcs pybamm.logger.verbose(f"Discretise initial conditions for {model.name}") ics, concat_ics = self.process_initial_conditions(model) model_disc.initial_conditions = ics model_disc.concatenated_initial_conditions = concat_ics # Discretise variables (applying boundary conditions) # Note that we **do not** discretise the keys of model.rhs, # model.initial_conditions and model.boundary_conditions pybamm.logger.verbose(f"Discretise variables for {model.name}") # pre-process variables so that all state variables are included # This is the ONLY place where model.variables should be modified pre_processed_variables = self._pre_process_variables( model.variables, model.initial_conditions ) model_disc.variables = pybamm.FuzzyDict(pre_processed_variables) if not delayed_variable_processing: # Process variables and store in _variables_processed variables_to_process = model.get_processed_variables_dict() for name, var in pre_processed_variables.items(): if name not in variables_to_process: # New variable (e.g., added by _pre_process_variables) variables_to_process[name] = var processed_variables = self.process_dict(variables_to_process) model_disc.update_processed_variables(processed_variables) # Process parabolic and elliptic equations pybamm.logger.verbose(f"Discretise model equations for {model.name}") rhs, concat_rhs, alg, concat_alg = self.process_rhs_and_algebraic(model) model_disc.rhs, model_disc.concatenated_rhs = rhs, concat_rhs model_disc.algebraic, model_disc.concatenated_algebraic = alg, concat_alg # Save length of rhs and algebraic model_disc.len_rhs = model_disc.concatenated_rhs.size model_disc.len_alg = model_disc.concatenated_algebraic.size model_disc.len_rhs_and_alg = model_disc.len_rhs + model_disc.len_alg # Process events processed_events = [] pybamm.logger.verbose(f"Discretise events for {model.name}") for event in model.events: pybamm.logger.debug(f"Discretise event '{event.name}'") processed_event = pybamm.Event( event.name, self.process_symbol(event.expression), event.event_type ) processed_events.append(processed_event) model_disc.events = processed_events # Create mass matrix pybamm.logger.verbose(f"Create mass matrix for {model.name}") model_disc.mass_matrix = self.create_mass_matrix(model_disc) # Save geometry pybamm.logger.verbose(f"Save geometry for {model.name}") model_disc._geometry = getattr(self.mesh, "_geometry", None) # Check that resulting model makes sense if self._check_model_flag: pybamm.logger.verbose(f"Performing model checks for {model.name}") self.check_model(model_disc) pybamm.logger.info(f"Finish discretising {model.name}") # Re-discretising the model means it can no longer safely process symbols. # Not currently reachable, but keeping the check for safety if model.is_discretised: pybamm.logger.debug( f"Model '{model.name}' is being re-discretised, " "which makes it unable to process symbols using `model.process_symbol`" ) # pragma: no cover model_disc.disable_symbol_processing( ModelSolutionObservability.REDISCRETISED_MODEL ) # pragma: no cover pybamm.logger.debug("Attaching the `discretisation` to the `symbol_processor`") model_disc.symbol_processor.discretisation = self model_disc.is_discretised = True return model_disc
def _resolve_coupled_variables_in_model(self, model): """Resolve CoupledVariables in rhs, algebraic, initial_conditions, and boundary_conditions.""" def resolve_symbol(symbol): if isinstance(symbol, pybamm.CoupledVariable): if symbol.name not in model.variables: raise pybamm.DiscretisationError( f"CoupledVariable '{symbol.name}' not found in model.variables." ) return resolve_symbol(model.variables[symbol.name]) elif hasattr(symbol, "children") and symbol.children: new_children = [] changed = False for child in symbol.children: new_child = resolve_symbol(child) new_children.append(new_child) if new_child is not child: changed = True if changed: return symbol.create_copy(new_children=new_children) return symbol for var, expr in list(model.rhs.items()): resolved = resolve_symbol(expr) if resolved is not expr: model.rhs[var] = resolved for var, expr in list(model.algebraic.items()): resolved = resolve_symbol(expr) if resolved is not expr: model.algebraic[var] = resolved for var, expr in list(model.initial_conditions.items()): resolved = resolve_symbol(expr) if resolved is not expr: model.initial_conditions[var] = resolved for var, bcs in list(model.boundary_conditions.items()): for side, (expr, bc_type) in list(bcs.items()): resolved = resolve_symbol(expr) if resolved is not expr: model.boundary_conditions[var][side] = (resolved, bc_type)
[docs] def set_variable_slices(self, variables): """ Sets the slicing for variables. Parameters ---------- variables : iterable of :class:`pybamm.Variables` The variables for which to set slices """ # Set up y_slices and bounds y_slices = defaultdict(list) y_slices_explicit = defaultdict(list) start = 0 end = 0 lower_bounds = [] upper_bounds = [] # Iterate through unpacked variables, adding appropriate slices to y_slices for variable in variables: if variable in y_slices: continue # Add up the size of all the domains in variable.domain if isinstance(variable, pybamm.ConcatenationVariable): spatial_method = self.spatial_methods[variable.domain[0]] dimension = spatial_method.mesh[variable.domain[0]].dimension start_ = start children = variable.children meshes = OrderedDict() lr_points = OrderedDict() tb_points = OrderedDict() for child in children: meshes[child] = [spatial_method.mesh[dom] for dom in child.domain] if dimension == 2: lr_points[child] = sum( spatial_method.mesh[dom].npts_lr for dom in child.domain ) tb_points[child] = sum( spatial_method.mesh[dom].npts_tb for dom in child.domain ) sec_points = spatial_method._get_auxiliary_domain_repeats( variable.domains ) for _ in range(sec_points): start_this_child = start_ for child, mesh in meshes.items(): for domain_mesh in mesh: end += domain_mesh.npts_for_broadcast_to_nodes # Add to slices if dimension == 2: other_children = set(meshes.keys()) - {child} num_pts_to_skip = sum( lr_points[other_child] for other_child in other_children ) for row in range(tb_points[child]): start_this_row = ( start_this_child + (lr_points[child] + num_pts_to_skip) * row ) end_this_row = start_this_row + lr_points[child] y_slices[child].append( slice(start_this_row, end_this_row) ) y_slices_explicit[child].append( slice(start_this_row, end_this_row) ) start_this_child += lr_points[child] else: y_slices[child].append(slice(start_, end)) y_slices_explicit[child].append(slice(start_, end)) # Increment start_ start_ = end else: end += self._get_variable_size(variable) # Add to slices y_slices[variable].append(slice(start, end)) y_slices_explicit[variable].append(slice(start, end)) # Add to bounds def evaluate_bound(bound, side): if bound.has_symbol_of_classes(pybamm.InputParameter): if side == "lower": return -np.inf elif side == "upper": return np.inf else: return bound.evaluate() lower_bounds.extend( [evaluate_bound(variable.bounds[0], "lower")] * (end - start) ) upper_bounds.extend( [evaluate_bound(variable.bounds[1], "upper")] * (end - start) ) # Increment start start = end # Convert y_slices back to normal dictionary self.y_slices = dict(y_slices) # Also keep a record of what the y_slices are, to be stored in the model self.y_slices_explicit = dict(y_slices_explicit) # Also keep a record of bounds self.bounds = (np.array(lower_bounds), np.array(upper_bounds)) # reset discretised_symbols self._discretised_symbols = {}
def _get_variable_size(self, variable): """Helper function to determine what size a variable should be""" # If domain is empty then variable has size 1 if variable.domain == []: return 1 else: size = 0 spatial_method = self.spatial_methods[variable.domain[0]] repeats = spatial_method._get_auxiliary_domain_repeats(variable.domains) for dom in variable.domain: size += spatial_method.mesh[dom].npts_for_broadcast_to_nodes * repeats return size
[docs] def set_internal_boundary_conditions(self, model): """ A method to set the internal boundary conditions for the submodel. These are required to properly calculate the gradient. Note: this method modifies the state of self.boundary_conditions. """ def boundary_gradient(left_symbol, right_symbol): pybamm.logger.debug( f"Calculate boundary gradient ({left_symbol} and {right_symbol})" ) left_domain = left_symbol.domain[0] right_domain = right_symbol.domain[0] left_mesh = self.spatial_methods[left_domain].mesh[left_domain] right_mesh = self.spatial_methods[right_domain].mesh[right_domain] left_symbol_disc = self.process_symbol(left_symbol) right_symbol_disc = self.process_symbol(right_symbol) return self.spatial_methods[left_domain].internal_neumann_condition( left_symbol_disc, right_symbol_disc, left_mesh, right_mesh ) bc_keys = list(self.bcs.keys()) internal_bcs = {} for var in model.boundary_conditions.keys(): if not isinstance(var, pybamm.Concatenation): continue children = var.orphans first_child = children[0] next_child = children[1] lbc = self.bcs[var]["left"] rbc = (boundary_gradient(first_child, next_child), "Neumann") if first_child not in bc_keys: internal_bcs.update({first_child: {"left": lbc, "right": rbc}}) for current_child, next_child in itertools.pairwise(children[1:]): lbc = rbc rbc = (boundary_gradient(current_child, next_child), "Neumann") if current_child not in bc_keys: internal_bcs.update({current_child: {"left": lbc, "right": rbc}}) lbc = rbc rbc = self.bcs[var]["right"] if children[-1] not in bc_keys: internal_bcs.update({children[-1]: {"left": lbc, "right": rbc}}) self.bcs.update(internal_bcs)
[docs] def process_initial_conditions(self, model): """Discretise model initial_conditions. Parameters ---------- model : :class:`pybamm.BaseModel` Model to dicretise. Must have attributes rhs, initial_conditions and boundary_conditions (all dicts of {variable: equation}) Returns ------- tuple Tuple of processed_initial_conditions (dict of initial conditions) and concatenated_initial_conditions (numpy array of concatenated initial conditions) """ # Discretise initial conditions processed_initial_conditions = self.process_dict( model.initial_conditions, ics=True ) # Concatenate initial conditions into a single vector # check that all initial conditions are set processed_concatenated_initial_conditions = self._concatenate_in_order( processed_initial_conditions, check_complete=True ) return processed_initial_conditions, processed_concatenated_initial_conditions
[docs] def process_boundary_conditions(self, model): """Discretise model boundary_conditions, also converting keys to ids Parameters ---------- model : :class:`pybamm.BaseModel` Model to dicretise. Must have attributes rhs, initial_conditions and boundary_conditions (all dicts of {variable: equation}) Returns ------- dict Dictionary of processed boundary conditions """ processed_bcs = {} # process and set pybamm.variables first incase required # in discrisation of other boundary conditions for key, bcs in model.boundary_conditions.items(): processed_bcs[key] = {} # check if the boundary condition at the origin for sphere domains is other # than no flux for subdomain in key.domain: if ( self.mesh[subdomain].coord_sys in ["spherical polar", "cylindrical polar"] and next(iter(self.mesh.geometry[subdomain].values()))["min"] == 0 ): if bcs["left"][0].value != 0 or bcs["left"][1] != "Neumann": raise pybamm.ModelError( "Boundary condition at r = 0 must be a homogeneous " f"Neumann condition for {self.mesh[subdomain].coord_sys} coordinates" ) # Handle any boundary conditions applied on the tabs if any("tab" in side for side in list(bcs.keys())): bcs = self.check_tab_conditions(key, bcs) # Process boundary conditions for side, bc in bcs.items(): eqn, typ = bc pybamm.logger.debug(f"Discretise {key} ({side} bc)") processed_eqn = self.process_symbol(eqn) processed_bcs[key][side] = (processed_eqn, typ) return processed_bcs
[docs] def check_tab_conditions(self, symbol, bcs): """ Check any boundary conditions applied on "negative tab", "positive tab" and "no tab". For 1D current collector meshes, these conditions are converted into boundary conditions on "left" (tab at z=0) or "right" (tab at z=l_z) depending on the tab location stored in the mesh. For 2D current collector meshes, the boundary conditions can be applied on the tabs directly. Parameters ---------- symbol : :class:`pybamm.expression_tree.symbol.Symbol` The symbol on which the boundary conditions are applied. bcs : dict The dictionary of boundary conditions (a dict of {side: equation}). Returns ------- dict The dictionary of boundary conditions, with the keys changed to "left" and "right" where necessary. """ # Check symbol domain domain = symbol.domain[0] mesh = self.mesh[domain] if domain != "current collector": raise pybamm.ModelError( "Boundary conditions can only be applied on the tabs in the domain " f"'current collector', but {symbol} has domain {domain}" ) # Replace keys with "left" and "right" as appropriate for 1D meshes if isinstance(mesh, pybamm.SubMesh1D): # send boundary conditions applied on the tabs to "left" or "right" # depending on the tab location stored in the mesh for tab in ["negative tab", "positive tab"]: if any(tab in side for side in list(bcs.keys())): bcs[mesh.tabs[tab]] = bcs.pop(tab) # if there was a tab at either end, then the boundary conditions # have now been set on "left" and "right" as required by the spatial # method, so there is no need to further modify the bcs dict if all(side in list(bcs.keys()) for side in ["left", "right"]): pass # if both tabs are located at z=0 then the "right" boundary condition # (at z=1) is the condition for "no tab" elif "left" in list(bcs.keys()): bcs["right"] = bcs.pop("no tab") # else if both tabs are located at z=1, the "left" boundary condition # (at z=0) is the condition for "no tab" else: bcs["left"] = bcs.pop("no tab") return bcs
[docs] def process_rhs_and_algebraic(self, model): """Discretise model equations - differential ('rhs') and algebraic. Parameters ---------- model : :class:`pybamm.BaseModel` Model to dicretise. Must have attributes rhs, initial_conditions and boundary_conditions (all dicts of {variable: equation}) Returns ------- tuple Tuple of processed_rhs (dict of processed differential equations), processed_concatenated_rhs, processed_algebraic (dict of processed algebraic equations) and processed_concatenated_algebraic """ # Discretise right-hand sides, passing domain from variable processed_rhs = self.process_dict(model.rhs) # Concatenate rhs into a single state vector # Need to concatenate in order as the ordering of equations could be different # in processed_rhs and model.rhs processed_concatenated_rhs = self._concatenate_in_order(processed_rhs) # Discretise and concatenate algebraic equations processed_algebraic = self.process_dict(model.algebraic) # Concatenate algebraic into a single state vector # Need to concatenate in order as the ordering of equations could be different # in processed_algebraic and model.algebraic processed_concatenated_algebraic = self._concatenate_in_order( processed_algebraic ) return ( processed_rhs, processed_concatenated_rhs, processed_algebraic, processed_concatenated_algebraic, )
[docs] def create_mass_matrix(self, model): """Creates mass matrix of the discretised model. Note that the model is assumed to be of the form M*y_dot = f(t,y), where M is the (possibly singular) mass matrix. Parameters ---------- model : :class:`pybamm.BaseModel` Discretised model. Must have attributes rhs, initial_conditions and boundary_conditions (all dicts of {variable: equation}) Returns ------- :class:`pybamm.Matrix` The mass matrix """ mass_list = [] # get a list of model rhs variables that are sorted according to # where they are in the state vector model_variables = model.rhs.keys() model_slices = [] for v in model_variables: model_slices.append(self.y_slices[v][0]) sorted_model_variables = [ v for _, v in sorted(zip(model_slices, model_variables, strict=False)) ] # Process mass matrices for the differential equations for var in sorted_model_variables: if var.domain == []: mass = 1.0 else: mass = ( self.spatial_methods[var.domain[0]] .mass_matrix(var, self.bcs) .entries ) mass_list.append(mass) # Create lumped mass matrix (of zeros) of the correct shape for the # discretised algebraic equations if model.algebraic.keys(): mass_algebraic_size = model.concatenated_algebraic.shape[0] mass_algebraic = csr_matrix((mass_algebraic_size, mass_algebraic_size)) mass_list.append(mass_algebraic) # Create block diagonal (sparse) mass matrix (if model is not empty) N_rhs = len(model.rhs) N_alg = len(model.algebraic) if N_rhs > 0 or N_alg > 0: mass_matrix = pybamm.Matrix(block_diag(mass_list, format="csr")) else: mass_matrix = None return mass_matrix
def _pre_process_variables( self, variables: dict[str, pybamm.Symbol], initial_conditions: dict[pybamm.Variable, pybamm.Symbol], ): """ Pre-process variables before discretisation. This involves: - ensuring that all the state variables are included in the variables, any missing are added Parameters ---------- variables : dict Dictionary of variables to pre-process initial_conditions : dict Dictionary of initial conditions Returns ------- dict Pre-processed variables (copy of input variables with any missing state) Raises ------ :class:`pybamm.ModelError` If any state variable names are already included but with incorrect expressions """ new_variables = dict(variables) for var in initial_conditions.keys(): if var.name not in new_variables: new_variables[var.name] = var else: existing_var = new_variables[var.name] # Compare by name and domains, not identity (handles unpickling case) if existing_var.name != var.name or existing_var.domains != var.domains: raise pybamm.ModelError( f"Variable '{var.name}' should have expression " f"'{var}', but has expression '{existing_var}'" ) return new_variables
[docs] def process_dict(self, var_eqn_dict, ics=False): """Discretise a dictionary of {variable: equation}, broadcasting if necessary (can be model.rhs, model.algebraic, model.initial_conditions or model.variables). Parameters ---------- var_eqn_dict : dict Equations ({variable: equation} dict) to dicretise (can be model.rhs, model.algebraic, model.initial_conditions or model.variables) ics : bool, optional Whether the equations are initial conditions. If True, the equations are scaled by the reference value of the variable, if given Returns ------- new_var_eqn_dict : dict Discretised equations """ return { k: self.process_equation(k, v, ics=ics) for k, v in var_eqn_dict.items() }
[docs] def process_equation(self, name, eqn, ics=False): """Discretise a dictionary of {variable: equation}, broadcasting if necessary (can be model.rhs, model.algebraic, model.initial_conditions or model.variables). Parameters ---------- var_eqn_dict : dict Equations ({variable: equation} dict) to dicretise (can be model.rhs, model.algebraic, model.initial_conditions or model.variables) ics : bool, optional Whether the equations are initial conditions. If True, the equations are scaled by the reference value of the variable, if given Returns ------- processed_eqn Discretised equation """ # Broadcast if the equation evaluates to a number (e.g. Scalar) if np.prod(eqn.shape_for_testing) == 1 and not isinstance(name, str): if name.domain == []: eqn = eqn * pybamm.Vector([1]) else: eqn = pybamm.FullBroadcast(eqn, broadcast_domains=name.domains) pybamm.logger.debug(f"Discretise {name!r}") processed_eqn = self.process_symbol(eqn) if ics and (reference := getattr(name, "reference", 0)) != 0: processed_eqn = processed_eqn - reference # Calculate scale if the key has a scale scale = getattr(name, "scale", 1) if scale != 1: processed_eqn = processed_eqn / scale return processed_eqn
[docs] def process_symbol(self, symbol): """Discretise operators in model equations. If a symbol has already been discretised, the stored value is returned. Parameters ---------- symbol : :class:`pybamm.expression_tree.symbol.Symbol` Symbol to discretise Returns ------- :class:`pybamm.expression_tree.symbol.Symbol` Discretised symbol """ _discretised_symbol = self._discretised_symbols.get(symbol) if _discretised_symbol is not None: return _discretised_symbol discretised_symbol = self._process_symbol(symbol) self._discretised_symbols[symbol] = discretised_symbol discretised_symbol.test_shape() # Assign mesh as an attribute to the processed variable if symbol.domain != []: discretised_symbol.mesh = self.mesh[symbol.domain] else: discretised_symbol.mesh = None # Assign secondary mesh if symbol.domains["secondary"] != []: discretised_symbol.secondary_mesh = self.mesh[symbol.domains["secondary"]] else: discretised_symbol.secondary_mesh = None # Assign tertiary mesh if symbol.domains["tertiary"] != []: discretised_symbol.tertiary_mesh = self.mesh[symbol.domains["tertiary"]] else: discretised_symbol.tertiary_mesh = None return discretised_symbol
def _process_symbol(self, symbol): """See :meth:`Discretisation.process_symbol()`.""" if symbol.domain != []: spatial_method = self.spatial_methods[symbol.domain[0]] # If boundary conditions are provided, need to check for BCs on tabs if self.bcs: key_id = next(iter(self.bcs.keys())) if any("tab" in side for side in list(self.bcs[key_id].keys())): self.bcs[key_id] = self.check_tab_conditions( symbol, self.bcs[key_id] ) else: spatial_method = None if isinstance(symbol, pybamm.BinaryOperator): # Pre-process children left, right = symbol.children # Catch case where diffusion is a scalar and turn it into an identity matrix vector field. if isinstance(spatial_method, pybamm.FiniteVolume2D): if isinstance(left, pybamm.Scalar) and ( isinstance(right, pybamm.VectorField) or isinstance(right, pybamm.Gradient) ): left = pybamm.VectorField(left, left) elif isinstance(right, pybamm.Scalar) and ( isinstance(left, pybamm.VectorField) or isinstance(left, pybamm.Gradient) ): right = pybamm.VectorField(right, right) disc_left = self.process_symbol(left) disc_right = self.process_symbol(right) if symbol.domain == []: if isinstance(disc_left, pybamm.VectorField) or isinstance( disc_right, pybamm.VectorField ): if not isinstance(disc_right, pybamm.VectorField): disc_right = pybamm.VectorField(disc_right, disc_right) if not isinstance(disc_left, pybamm.VectorField): disc_left = pybamm.VectorField(disc_left, disc_left) else: # both are vector fields already pass disc_lr = pybamm.simplify_if_constant( symbol.create_copy( new_children=[disc_left.lr_field, disc_right.lr_field] ) ) disc_tb = pybamm.simplify_if_constant( symbol.create_copy( new_children=[disc_left.tb_field, disc_right.tb_field] ) ) return pybamm.VectorField(disc_lr, disc_tb) return pybamm.simplify_if_constant( symbol.create_copy(new_children=[disc_left, disc_right]) ) else: return spatial_method.process_binary_operators( symbol, left, right, disc_left, disc_right ) elif isinstance(symbol, pybamm._BaseAverage): # Create a new Integral operator and process it child = symbol.orphans[0] if isinstance(symbol, pybamm.SizeAverage): R = symbol.integration_variable[0] f_a_dist = symbol.f_a_dist # take average using Integral and distribution f_a_dist average = pybamm.Integral(f_a_dist * child, R) / pybamm.Integral( f_a_dist, R ) else: x = symbol.integration_variable v = pybamm.ones_like(child) average = pybamm.Integral(child, x) / pybamm.Integral(v, x) return self.process_symbol(average) elif isinstance(symbol, pybamm.UnaryOperator): child = symbol.child disc_child = self.process_symbol(child) if child.domain != []: child_spatial_method = self.spatial_methods[child.domain[0]] if isinstance(symbol, pybamm.Gradient): return child_spatial_method.gradient(child, disc_child, self.bcs) elif isinstance(symbol, pybamm.Divergence): return child_spatial_method.divergence(child, disc_child, self.bcs) elif isinstance(symbol, pybamm.Laplacian): return child_spatial_method.laplacian(child, disc_child, self.bcs) elif isinstance(symbol, pybamm.GradientSquared): return child_spatial_method.gradient_squared( child, disc_child, self.bcs ) elif isinstance(symbol, pybamm.Mass): return child_spatial_method.mass_matrix(child, self.bcs) elif isinstance(symbol, pybamm.BoundaryMass): return child_spatial_method.boundary_mass_matrix(child, self.bcs) elif isinstance(symbol, pybamm.IndefiniteIntegral): return child_spatial_method.indefinite_integral( child, disc_child, "forward" ) elif isinstance(symbol, pybamm.BackwardIndefiniteIntegral): return child_spatial_method.indefinite_integral( child, disc_child, "backward" ) elif isinstance(symbol, pybamm.Integral): integral_spatial_method = self.spatial_methods[ symbol.integration_variable[0].domain[0] ] out = integral_spatial_method.integral( child, disc_child, symbol._integration_dimension, symbol.integration_variable, ) out.copy_domains(symbol) return out elif isinstance(symbol, pybamm.DefiniteIntegralVector): return child_spatial_method.definite_integral_matrix( child, vector_type=symbol.vector_type ) elif isinstance(symbol, pybamm.OneDimensionalIntegral): child_spatial_method = self.spatial_methods[ symbol.integration_domain[0] ] return child_spatial_method.one_dimensional_integral( symbol, child, disc_child, symbol.integration_domain, symbol.direction, ) elif isinstance(symbol, pybamm.BoundaryIntegral): return child_spatial_method.boundary_integral( child, disc_child, symbol.region ) elif isinstance(symbol, pybamm.Broadcast): # Broadcast new_child to the domain specified by symbol.domain # Different discretisations may broadcast differently return spatial_method.broadcast( disc_child, symbol.domains, symbol.broadcast_type ) elif isinstance(symbol, pybamm.DeltaFunction): return spatial_method.delta_function(symbol, disc_child) elif isinstance(symbol, pybamm.BoundaryOperator): # if boundary operator applied on "negative tab" or # "positive tab" *and* the mesh is 1D then change side to # "left" or "right" as appropriate if symbol.side in ["negative tab", "positive tab"]: mesh = self.mesh[symbol.children[0].domain[0]] if isinstance(mesh, pybamm.SubMesh1D): symbol.side = mesh.tabs[symbol.side] return child_spatial_method.boundary_value_or_flux( symbol, disc_child, self.bcs ) elif isinstance(symbol, pybamm.EvaluateAt): return child_spatial_method.evaluate_at( symbol, disc_child, symbol.position ) elif isinstance(symbol, pybamm.UpwindDownwind2D): return spatial_method.upwind_or_downwind( child, disc_child, self.bcs, symbol.lr_direction, symbol.tb_direction, ) elif isinstance(symbol, pybamm.NodeToEdge2D): return spatial_method.node_to_edge( disc_child, method="arithmetic", direction=symbol.direction, ) elif isinstance(symbol, pybamm.UpwindDownwind): direction = symbol.name # upwind or downwind return spatial_method.upwind_or_downwind( child, disc_child, self.bcs, direction ) elif isinstance(symbol, pybamm.NotConstant): # After discretisation, we can make the symbol constant return disc_child elif isinstance(symbol, pybamm.Magnitude): if not isinstance(disc_child, pybamm.VectorField): raise ValueError("Magnitude can only be applied to a vector field") direction = symbol.direction if direction == "lr": return disc_child.lr_field elif direction == "tb": return disc_child.tb_field else: raise ValueError("Invalid direction") else: if isinstance(disc_child, pybamm.VectorField): return pybamm.VectorField( symbol.create_copy(new_children=[disc_child.lr_field]), symbol.create_copy(new_children=[disc_child.tb_field]), ) else: return symbol.create_copy(new_children=[disc_child]) elif isinstance(symbol, (pybamm.Function, pybamm.Conditional)): disc_children = [self.process_symbol(child) for child in symbol.children] return symbol.create_copy(disc_children) elif isinstance(symbol, pybamm.VariableDot): # Add symbol's reference and multiply by the symbol's scale # so that the state vector is of order 1 return symbol.reference + symbol.scale * pybamm.StateVectorDot( *self.y_slices[symbol.get_variable()], domains=symbol.domains, ) elif isinstance(symbol, pybamm.Variable): # add a try except block for a more informative error if a variable # can't be found. This should usually be caught earlier by # model.check_well_posedness, but won't be if debug_mode is False try: y_slices = self.y_slices[symbol] except KeyError as error: raise pybamm.ModelError( f"No key set for variable '{symbol.name}'. Make sure it is included in either " "model.rhs or model.algebraic in an unmodified form " "(e.g. not Broadcasted)" ) from error # Add symbol's reference and multiply by the symbol's scale # so that the state vector is of order 1 return symbol.reference + symbol.scale * pybamm.StateVector( *y_slices, domains=symbol.domains ) elif isinstance(symbol, pybamm.SpatialVariable): return spatial_method.spatial_variable(symbol) elif isinstance(symbol, pybamm.ConcatenationVariable): # create new children without scale and reference # the scale and reference will be applied to the concatenation instead new_children = [] old_y_slices = self.y_slices.copy() for child in symbol.children: child_no_scale = child.create_copy() child_no_scale.scale = 1 child_no_scale.reference = 0 self.y_slices[child_no_scale] = self.y_slices[child] new_children.append(self.process_symbol(child_no_scale)) self.y_slices = old_y_slices new_symbol = spatial_method.concatenation(new_children) # apply scale to the whole concatenation return symbol.reference + symbol.scale * new_symbol elif isinstance(symbol, pybamm.Concatenation): new_children = [self.process_symbol(child) for child in symbol.children] new_symbol = spatial_method.concatenation(new_children) return new_symbol elif isinstance(symbol, pybamm.InputParameter): if symbol.domain != []: expected_size = self._get_variable_size(symbol) else: expected_size = None if symbol._expected_size is None: symbol._expected_size = expected_size return symbol.create_copy() elif isinstance(symbol, pybamm.CoupledVariable): raise pybamm.DiscretisationError( f"CoupledVariable '{symbol.name}' was not resolved before discretisation. " "Ensure the variable exists in model.variables." ) elif isinstance(symbol, pybamm.VectorField): # VectorField is a subclass of TensorField, handle it first for specificity left_symbol = self.process_symbol(symbol.lr_field) right_symbol = self.process_symbol(symbol.tb_field) return symbol.create_copy(new_children=[left_symbol, right_symbol]) elif isinstance(symbol, pybamm.TensorField): # General TensorField handling (rank-2 tensors) processed_children = [self.process_symbol(c) for c in symbol.children] return symbol.create_copy(new_children=processed_children) elif isinstance(symbol, pybamm.Constant): # after discretisation we just care about the value, not the name return self.process_symbol(pybamm.Scalar(symbol.value)) else: # Backup option: return the object return symbol def concatenate(self, *symbols, sparse=False): if sparse: return pybamm.SparseStack(*symbols) else: return pybamm.numpy_concatenation(*symbols) def _concatenate_in_order(self, var_eqn_dict, check_complete=False, sparse=False): """ Concatenate a dictionary of {variable: equation} using self.y_slices The keys/variables in `var_eqn_dict` must be the same as the ids in `self.y_slices`. The resultant concatenation is ordered according to the ordering of the slice values in `self.y_slices` Parameters ---------- var_eqn_dict : dict Equations ({variable: equation} dict) to dicretise check_complete : bool, optional Whether to check keys in var_eqn_dict against self.y_slices. Default is False sparse : bool, optional If True the concatenation will be a :class:`pybamm.SparseStack`. If False the concatenation will be a :class:`pybamm.NumpyConcatenation`. Default is False Returns ------- var_eqn_dict : dict Discretised right-hand side equations """ # Unpack symbols in variables that are concatenations of variables unpacked_variables = [] slices = [] for symbol in var_eqn_dict.keys(): if isinstance(symbol, pybamm.ConcatenationVariable): unpacked_variables.extend([symbol] + [var for var in symbol.children]) else: unpacked_variables.append(symbol) slices.append(self.y_slices[symbol][0]) if check_complete: # Check keys from the given var_eqn_dict against self.y_slices unpacked_variables_set = set(unpacked_variables) if unpacked_variables_set != set(self.y_slices.keys()): given_variable_names = [v.name for v in var_eqn_dict.keys()] raise pybamm.ModelError( "Initial conditions are insufficient. Only " f"provided for {given_variable_names} " ) equations = list(var_eqn_dict.values()) # sort equations according to slices sorted_equations = [eq for _, eq in sorted(zip(slices, equations, strict=True))] return self.concatenate(*sorted_equations, sparse=sparse)
[docs] def check_model(self, model): """Perform some basic checks to make sure the discretised model makes sense.""" self.check_initial_conditions(model)
def check_initial_conditions(self, model): # Check initial conditions are a numpy array # Individual for var, eqn in model.initial_conditions.items(): ic_eval = eqn.evaluate(t=0, inputs="shape test") if not isinstance(ic_eval, np.ndarray): raise pybamm.ModelError( "initial conditions must be numpy array after discretisation but " f"they are {type(ic_eval)} for variable '{var}'." ) # Check that the initial condition is within the bounds # Skip this check if there are input parameters in the initial conditions bounds = var.bounds if not eqn.has_symbol_of_classes(pybamm.InputParameter) and not ( all(bounds[0].value <= ic_eval) and all(ic_eval <= bounds[1].value) ): raise pybamm.ModelError( "initial condition is outside of variable bounds " f"{bounds} for variable '{var}'." ) # Check initial conditions and model equations have the same shape # Individual for var in model.rhs.keys(): if model.rhs[var].shape != model.initial_conditions[var].shape: raise pybamm.ModelError( "rhs and initial conditions must have the same shape after " "discretisation but rhs.shape = " f"{model.rhs[var].shape} and initial_conditions.shape = {model.initial_conditions[var].shape} for variable '{var}'." ) for var in model.algebraic.keys(): if model.algebraic[var].shape != model.initial_conditions[var].shape: raise pybamm.ModelError( "algebraic and initial conditions must have the same shape after " "discretisation but algebraic.shape = " f"{model.algebraic[var].shape} and initial_conditions.shape = {model.initial_conditions[var].shape} for variable '{var}'." ) def is_variable_independent(self, var, all_vars_in_eqns): pybamm.logger.verbose("Removing independent blocks.") if not isinstance(var, pybamm.Variable): return False this_var_is_independent = var.name not in all_vars_in_eqns not_in_y_slices = var not in list(self.y_slices.keys()) not_in_discretised = var not in list(self._discretised_symbols.keys()) is_0D = len(var.domain) == 0 this_var_is_independent = ( this_var_is_independent and not_in_y_slices and not_in_discretised and is_0D ) return this_var_is_independent def remove_independent_variables_from_rhs(self, model): rhs_vars_to_search_over = list(model.rhs.keys()) unpacker = pybamm.SymbolUnpacker(pybamm.Variable) eqns_to_check = ( list(model.rhs.values()) + list(model.algebraic.values()) + [ x[side][0] for x in model.boundary_conditions.values() for side in x.keys() ] # only check children of variables, this will skip the variable itself # and catch any other cases + [child for var in model.variables.values() for child in var.children] + [event.expression for event in model.events] ) all_vars_in_eqns = unpacker.unpack_list_of_symbols(eqns_to_check) all_vars_in_eqns = [var.name for var in all_vars_in_eqns] for var in rhs_vars_to_search_over: this_var_is_independent = self.is_variable_independent( var, all_vars_in_eqns ) if this_var_is_independent: if len(model.rhs) != 1: pybamm.logger.info(f"removing variable {var} from rhs") my_initial_condition = model.initial_conditions[var] explicit_integral = pybamm.ExplicitTimeIntegral( model.rhs[var], my_initial_condition ) # Collect variables to update in _variables_processed # Do NOT modify model.variables - only update _variables_processed vars_to_update = {var.name: explicit_integral} # edge case where a variable appears # in variables twice under different names for key in model.variables: if model.variables[key] == var: vars_to_update[key] = explicit_integral # Update _variables_processed using the proper method model.update_processed_variables(vars_to_update) del model.rhs[var] del model.initial_conditions[var] else: break return model