Source code for pywarpx.callbacks

# Copyright 2017 David Grote
#
# This file is part of WarpX.
#
# License: BSD-3-Clause-LBNL

"""call back operations
=====================

These are the functions which allow installing user created functions so that
they are called at various places along the time step.

For each call back type, the following three functions are defined.
 - install___: Installs a function to be called at that specified time
 - uninstall___: Uninstalls the function (so it won't be called anymore)
 - isinstalled___: Checks if the function is installed

These functions all take a function or instance method as an argument. Note that
if an instance method is used, an extra reference to the method's object is saved.

The install can be done using a decorator, which has the prefix "callfrom". See example below.

Functions can be called at the following times:
 - afterinit <installafterinit>: immediately after the init is complete
 - beforeEsolve <installbeforeEsolve>: before the solve for E fields
 - poissonsolver <installpoissonsolver>: In place of the computePhi call but only in an electrostatic simulation
 - afterEsolve <installafterEsolve>: after the solve for E fields
 - beforedeposition <installbeforedeposition>: before the particle deposition (for charge and/or current)
 - afterdeposition <installafterdeposition>: after particle deposition (for charge and/or current)
 - beforestep <installbeforestep>: before the time step
 - afterstep <installafterstep>: after the time step
 - afterdiagnostics <installafterdiagnostics>: after diagnostic output
 - particlescraper <installparticlescraper>: just after the particle boundary conditions are applied
                                             but before lost particles are processed
 - particleloader <installparticleloader>: at the time that the standard particle loader is called
 - particleinjection <installparticleinjection>: called when particle injection happens, after the position
                                                  advance and before deposition is called, allowing a user defined
                                                  particle distribution to be injected each time step
 - appliedfields <installappliedfields>: allows directly specifying any fields to be applied to the particles
                                         during the advance

To use a decorator, the syntax is as follows. This will install the function myplots to be called after each step.

@callfromafterstep
def myplots():
  ppzx()

This is equivalent to the following:

def myplots():
  ppzx()
installafterstep(myplots)

"""
from __future__ import generators

import copy
import ctypes
import sys
import time
import types

import numpy

from ._libwarpx import libwarpx


[docs]class CallbackFunctions(object): """ Class to handle call back function lists. Note that for functions passed in that are methods of a class instance, a full reference of the instance is saved. This extra reference means that the object will not actually be deleted if the user deletes the original reference. This is good since the user does not need to keep the reference to the object (for example it can be created using a local variable in a function). It may be bad if the user thinks an object was deleted, but it actually isn't since it had (unkown to the user) installed a method in one of the call back lists. """ def __init__(self,name=None,lcallonce=0): self.funcs = [] self.time = 0. self.timers = {} self.name = name self.lcallonce = lcallonce def __call__(self,*args,**kw): """Call all of the functions in the list""" tt = self.callfuncsinlist(*args,**kw) self.time = self.time + tt if self.lcallonce: self.funcs = [] def clearlist(self): """Unregister/clear out all registered C callbacks""" self.funcs = [] libwarpx.libwarpx_so.warpx_clear_callback_py( ctypes.c_char_p(self.name.encode('utf-8')) ) def __bool__(self): """Returns True if functions are installed, otherwise False""" return self.hasfuncsinstalled() def __len__(self): """Returns number of functions installed""" return len(self.funcs) def hasfuncsinstalled(self): """Checks if there are any functions installed""" return len(self.funcs) > 0 def _getmethodobject(self,func): """For call backs that are methods, returns the method's instance""" return func[0] def callbackfunclist(self): """Generator returning callable functions from the list""" funclistcopy = copy.copy(self.funcs) for f in funclistcopy: if isinstance(f,list): object = self._getmethodobject(f) if object is None: self.funcs.remove(f) continue result = getattr(object,f[1]) elif isinstance(f,str): import __main__ if f in __main__.__dict__: result = __main__.__dict__[f] # --- If the function with the name is found, then replace the # --- name in the list with the function. self.funcs[self.funcs.index(f)] = result else: continue else: result = f if not callable(result): print("\n\nWarning: a call back was found that is not callable.") if self.name is not None: print("For %s"%self.name) print("Only callable objects can be installed.") print("It is possible that the callable's name has been overwritten") print("by something not callable. This can happen during restart") print("if a function name had later been used as a variable name.") print(self.name) if isinstance(f,str): print("The name of the call back is %s"%f) print("\n\n") continue yield result def installfuncinlist(self,f): """Check if the specified function is installed""" if len(self.funcs) == 0: # If this is the first function installed, set the callback in the C++ # to call this class instance. # Note that the _c_func must be saved. _CALLBACK_FUNC_0 = ctypes.CFUNCTYPE(None) self._c_func = _CALLBACK_FUNC_0(self) libwarpx.libwarpx_so.warpx_set_callback_py( ctypes.c_char_p(self.name.encode('utf-8')), self._c_func ) if isinstance(f,types.MethodType): # --- If the function is a method of a class instance, then save a full # --- reference to that instance and the method name. finstance = f.__self__ fname = f.__name__ self.funcs.append([finstance,fname]) elif callable(f): # --- If a function had already been installed by name, then skip the install. # --- This is problematic, since no warning message is given, but it is unlikely # --- to arise under normal circumstances. # --- The purpose of this check is to avoid redundant installation of functions # --- during a restore from a dump file. Without the check, functions that had been # --- installed via a decorator would be installed an extra time since the source # --- of the function contains the decoration (which is activated when the source # --- is exec'd). if f.__name__ not in self.funcs: self.funcs.append(f) else: self.funcs.append(f) def uninstallfuncinlist(self,f): """Uninstall the specified function""" # --- An element by element search is needed # --- f can be a function or method object, or a name (string). # --- Note that method objects can not be removed by name. funclistcopy = copy.copy(self.funcs) for func in funclistcopy: if f == func: self.funcs.remove(f) break elif isinstance(func,list) and isinstance(f,types.MethodType): object = self._getmethodobject(func) if f.__self__ is object and f.__name__ == func[1]: self.funcs.remove(func) break elif isinstance(func,str): if f.__name__ == func: self.funcs.remove(func) break elif isinstance(f,str): if isinstance(func,str): funcname = func elif isinstance(func,list): funcname = None else: funcname = func.__name__ if f == funcname: self.funcs.remove(func) break # check that a function was removed if len(self.funcs) == len(funclistcopy): raise Exception(f'Warning: no function, {f}, had been installed') # if there are no functions left, remove the C callback if not self.hasfuncsinstalled(): self.clearlist() def isinstalledfuncinlist(self,f): """Checks if the specified function is installed""" # --- An element by element search is needed funclistcopy = copy.copy(self.funcs) for func in funclistcopy: if f == func: return 1 elif isinstance(func,list) and isinstance(f,types.MethodType): object = self._getmethodobject(func) if f.__self__ is object and f.__name__ == func[1]: return 1 elif isinstance(func,str): if f.__name__ == func: return 1 return 0 def callfuncsinlist(self,*args,**kw): """Call the functions in the list""" bb = time.time() for f in self.callbackfunclist(): #barrier() t1 = time.time() f(*args,**kw) #barrier() t2 = time.time() # --- For the timers, use the function (or method) name as the key. self.timers[f.__name__] = self.timers.get(f.__name__,0.) + (t2 - t1) aa = time.time() return aa - bb
#============================================================================= # --- Now create the actual instances. _afterinit = CallbackFunctions('afterinit') _beforecollisions = CallbackFunctions('beforecollisions') _aftercollisions = CallbackFunctions('aftercollisions') _beforeEsolve = CallbackFunctions('beforeEsolve') _poissonsolver = CallbackFunctions('poissonsolver') _afterEsolve = CallbackFunctions('afterEsolve') _beforedeposition = CallbackFunctions('beforedeposition') _afterdeposition = CallbackFunctions('afterdeposition') _particlescraper = CallbackFunctions('particlescraper') _particleloader = CallbackFunctions('particleloader') _beforestep = CallbackFunctions('beforestep') _afterstep = CallbackFunctions('afterstep') _afterdiagnostics = CallbackFunctions('afterdiagnostics') _afterrestart = CallbackFunctions('afterrestart',lcallonce=1) _oncheckpointsignal = CallbackFunctions('oncheckpointsignal') _particleinjection = CallbackFunctions('particleinjection') _appliedfields = CallbackFunctions('appliedfields') #============================================================================= def printcallbacktimers(tmin=1.,lminmax=False,ff=None): """Prints timings of installed functions. - tmin=1.: only functions with time greater than tmin will be printed - lminmax=False: If True, prints the min and max times over all processors - ff=None: If given, timings will be written to the file object instead of stdout """ if ff is None: ff = sys.stdout for c in [_afterinit,_beforeEsolve,_poissonsolver,_afterEsolve, _beforedeposition,_afterdeposition, _particlescraper, _particleloader, _beforestep,_afterstep, _afterdiagnostics, _afterrestart, _particleinjection, _appliedfields]: for fname, time in c.timers.items(): #vlist = numpy.array(gather(time)) vlist = numpy.array([time]) #if me > 0: continue vsum = numpy.sum(vlist) if vsum <= tmin: continue vrms = numpy.sqrt(max(0.,numpy.sum(vlist**2)/len(vlist) - (numpy.sum(vlist)/len(vlist))**2)) npes = 1. # Only works for one processor ff.write('%20s %s %10.4f %10.4f %10.4f'%(c.name,fname,vsum,vsum/npes,vrms)) if lminmax: vmin = numpy.min(vlist) vmax = numpy.max(vlist) ff.write(' %10.4f %10.4f'%(vmin,vmax)) it = libwarpx.libwarpx_so.warpx_getistep(0) if it > 0: ff.write(' %10.4f'%(vsum/npes/(it))) ff.write('\n') #============================================================================= # ---------------------------------------------------------------------------- def callfromafterinit(f): installafterinit(f) return f def installafterinit(f): "Adds a function to the list of functions called after the init" _afterinit.installfuncinlist(f) def uninstallafterinit(f): "Removes the function from the list of functions called after the init" _afterinit.uninstallfuncinlist(f) def isinstalledafterinit(f): "Checks if the function is called after a init" return _afterinit.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfrombeforecollisions(f): installbeforecollisions(f) return f def installbeforecollisions(f): "Adds a function to the list of functions called before collisions" _beforecollisions.installfuncinlist(f) def uninstallbeforecollisions(f): "Removes the function from the list of functions called before collisions" _beforecollisions.uninstallfuncinlist(f) def isinstalledbeforecollisions(f): "Checks if the function is called before collisions" return _beforecollisions.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromaftercollisions(f): installaftercollisions(f) return f def installaftercollisions(f): "Adds a function to the list of functions called after collisions" _aftercollisions.installfuncinlist(f) def uninstallaftercollisions(f): "Removes the function from the list of functions called after collisions" _aftercollisions.uninstallfuncinlist(f) def isinstalledaftercollisions(f): "Checks if the function is called after collisions" return _aftercollisions.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfrombeforeEsolve(f): installbeforeEsolve(f) return f def installbeforeEsolve(f): "Adds a function to the list of functions called before an E solve" _beforeEsolve.installfuncinlist(f) def uninstallbeforeEsolve(f): "Removes the function from the list of functions called before an E solve" _beforeEsolve.uninstallfuncinlist(f) def isinstalledbeforeEsolve(f): "Checks if the function is called before an E solve" return _beforeEsolve.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfrompoissonsolver(f): installpoissonsolver(f) return f def installpoissonsolver(f): """Installs an external function to solve Poisson's equation""" if _poissonsolver.hasfuncsinstalled(): raise RuntimeError("Only one external Poisson solver can be installed.") _poissonsolver.installfuncinlist(f) def uninstallpoissonsolver(f): """Removes the external function to solve Poisson's equation""" _poissonsolver.uninstallfuncinlist(f) def isinstalledpoissonsolver(f): """Checks if the function is called to solve Poisson's equation""" return _poissonsolver.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromafterEsolve(f): installafterEsolve(f) return f def installafterEsolve(f): "Adds a function to the list of functions called after an E solve" _afterEsolve.installfuncinlist(f) def uninstallafterEsolve(f): "Removes the function from the list of functions called after an E solve" _afterEsolve.uninstallfuncinlist(f) def isinstalledafterEsolve(f): "Checks if the function is called after an E solve" return _afterEsolve.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfrombeforedeposition(f): installbeforedeposition(f) return f def installbeforedeposition(f): "Adds a function to the list of functions called before a particle deposition" _beforedeposition.installfuncinlist(f) def uninstallbeforedeposition(f): "Removes the function from the list of functions called before a particle deposition" _beforedeposition.uninstallfuncinlist(f) def isinstalledbeforedeposition(f): "Checks if the function is called before a particle deposition" return _beforedeposition.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromafterdeposition(f): installafterdeposition(f) return f def installafterdeposition(f): "Adds a function to the list of functions called after a particle deposition" _afterdeposition.installfuncinlist(f) def uninstallafterdeposition(f): "Removes the function from the list of functions called after a particle deposition" _afterdeposition.uninstallfuncinlist(f) def isinstalledafterdeposition(f): "Checks if the function is called after a particle deposition" return _afterdeposition.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromparticlescraper(f): installparticlescraper(f) return f def installparticlescraper(f): "Adds a function to the list of functions called to scrape particles" _particlescraper.installfuncinlist(f) def uninstallparticlescraper(f): "Removes the function from the list of functions called to scrape particles" _particlescraper.uninstallfuncinlist(f) def isinstalledparticlescraper(f): "Checks if the function is called to scrape particles" return _particlescraper.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromparticleloader(f): installparticleloader(f) return f def installparticleloader(f): "Adds a function to the list of functions called to load particles" _particleloader.installfuncinlist(f) def uninstallparticleloader(f): "Removes the function from the list of functions called to load particles" _particleloader.uninstallfuncinlist(f) def isinstalledparticleloader(f): "Checks if the function is called to load particles" return _particleloader.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfrombeforestep(f): installbeforestep(f) return f def installbeforestep(f): "Adds a function to the list of functions called before a step" _beforestep.installfuncinlist(f) def uninstallbeforestep(f): "Removes the function from the list of functions called before a step" _beforestep.uninstallfuncinlist(f) def isinstalledbeforestep(f): "Checks if the function is called before a step" return _beforestep.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromafterstep(f): installafterstep(f) return f def installafterstep(f): "Adds a function to the list of functions called after a step" _afterstep.installfuncinlist(f) def uninstallafterstep(f): "Removes the function from the list of functions called after a step" _afterstep.uninstallfuncinlist(f) def isinstalledafterstep(f): "Checks if the function is called after a step" return _afterstep.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromafterdiagnostics(f): installafterdiagnostics(f) return f def installafterdiagnostics(f): "Adds a function to the list of functions called after diagnostic output" _afterdiagnostics.installfuncinlist(f) def uninstallafterdiagnostics(f): "Removes the function from the list of functions called after diagnostic output" _afterdiagnostics.uninstallfuncinlist(f) def isinstalledafterdiagnostics(f): "Checks if the function is called after diagnostic output" return _afterdiagnostics.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromafterrestart(f): raise Exception('restart call back not implemented yet') installafterrestart(f) return f def installafterrestart(f): "Adds a function to the list of functions called immediately after a restart" raise Exception('restart call back not implemented yet') _afterrestart.installfuncinlist(f) def uninstallafterrestart(f): "Removes the function from the list of functions called immediately after a restart" raise Exception('restart call back not implemented yet') _afterrestart.uninstallfuncinlist(f) def isinstalledafterrestart(f): "Checks if the function is called immediately after a restart" raise Exception('restart call back not implemented yet') return _afterrestart.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def oncheckpointsignal(f): installoncheckpointsignal(f) return f def installoncheckpointsignal(f): "Adds a function to the list of functions called on checkpoint signal" _oncheckpointsignal.installfuncinlist(f) def uninstalloncheckpointsignal(f): "Removes the function from the list of functions called on checkpoint signal" _oncheckpointsignal.uninstallfuncinlist(f) def isinstalledoncheckpointsignal(f): "Checks if the function is called on checkpoint signal" return _oncheckpointsignal.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromparticleinjection(f): installparticleinjection(f) return f def installparticleinjection(f): """ Adds a user defined function that is to be called when particle injection happens, after the position advance and before deposition is called, allowing a user defined particle distribution to be injected each time step""" _particleinjection.installfuncinlist(f) def uninstallparticleinjection(f): "Removes the function installed by installparticleinjection" _particleinjection.uninstallfuncinlist(f) def isinstalledparticleinjection(f): "Checks if the function is called when particles injection happens" return _particleinjection.isinstalledfuncinlist(f) # ---------------------------------------------------------------------------- def callfromappliedfields(f): raise Exception('applied fields call back not implemented yet') installappliedfields(f) return f def installappliedfields(f): """ Adds a user defined function which can specify E and B fields which are applied to the particles during the particle advance. """ raise Exception('applied fields call back not implemented yet') _appliedfields.installfuncinlist(f) def uninstallappliedfields(f): "Removes the function installed by installappliedfields" raise Exception('applied fields call back not implemented yet') _appliedfields.uninstallfuncinlist(f) def isinstalledappliedfields(f): "Checks if the function is called when which applies fields" raise Exception('applied fields call back not implemented yet') return _appliedfields.isinstalledfuncinlist(f)