Newer
Older
# ==========================================================================
# --------------------------------------------------------------------------
# Copyright (C) Organisation europeenne pour la Recherche nucleaire (CERN)
# All rights reserved.
#
# For the licensing terms see $DD4hepINSTALL/LICENSE.
# For the list of contributors see $DD4hepINSTALL/doc/CREDITS.
#
# ==========================================================================
from __future__ import absolute_import, unicode_literals
def loadDDDigi():
from ROOT import gSystem
# Try to load libglapi to avoid issues with TLS Static
# Turn off all errors from ROOT about the library missing
if('libglapi' not in gSystem.GetLibraries()):
orgLevel = ROOT.gErrorIgnoreLevel
ROOT.gErrorIgnoreLevel = 6000
gSystem.Load("libglapi")
ROOT.gErrorIgnoreLevel = orgLevel
import os
gSystem.SetDynamicPath(os.environ['DD4HEP_LIBRARY_PATH'])
#
# load with ROOT the DDDigi plugin library, which in turn loads the DDigi module
result = gSystem.Load("libDDDigiPlugins")
if result < 0:
raise Exception('DDDigi.py: Failed to load the DDDigi library libDDDigiPlugins: ' + gSystem.GetErrorStr())
logger.info('DDDigi.py: Successfully loaded DDDigi plugin library libDDDigiPlugins!')
#
# import with ROOT the I/O module to read DDG4 output
result = gSystem.Load("libDDDigi_DDG4_IO")
raise Exception('DDDigi.py: Failed to load the DDG4 IO library libDDDigi_DDG4_IO: ' + gSystem.GetErrorStr())
logger.info('DDDigi.py: Successfully loaded DDG4 IO plugin library libDDDigi_DDG4_IO!')
#
# import the main dd4hep module from ROOT
from ROOT import dd4hep as module
return module
# We are nearly there ....
current = __import__(__name__)
def _import_class(ns, nam):
scope = getattr(current, ns)
setattr(current, nam, getattr(scope, nam))
# ---------------------------------------------------------------------------
#
try:
except Exception as X:
logger.error('+--%-100s--+' % (100 * '-',))
logger.error('| %-100s |' % ('Failed to load DDDigi library:',))
logger.error('| %-100s |' % (str(X),))
logger.error('+--%-100s--+' % (100 * '-',))
exit(1)
core = dd4hep
digi = dd4hep.digi
Kernel = digi.KernelHandle
Interface = digi.DigiActionCreation
# ---------------------------------------------------------------------------
return self.constantAsString(name)
Detector.globalVal = _constant
# ---------------------------------------------------------------------------
def importConstants(description, namespace=None, debug=False):
"""
Import the Detector constants into the dddigi namespace
"""
ns = current
if namespace is not None and not hasattr(current, namespace):
import imp
m = imp.new_module('dddigi.' + namespace)
ns = m
evaluator = dd4hep.g4Evaluator()
cnt = 0
num = 0
todo = {}
strings = {}
for c in description.constants():
if c.second.dataType == 'string':
strings[c.first] = c.second.GetTitle()
else:
todo[c.first] = c.second.GetTitle().replace('(int)', '')
while len(todo) and cnt < 100:
cnt = cnt + 1
if cnt == 100:
logger.info('%s %d out of %d %s "%s": [%s]\n+++ %s' %
('+++ FAILED to import',
len(todo), len(todo) + num,
'global values into namespace',
ns.__name__, 'Try to continue anyway', 100 * '=',))
for k, v in todo.items():
if not hasattr(ns, k):
logger.info('+++ FAILED to import: "' + k + '" = "' + str(v) + '"')
logger.info('+++ %s' % (100 * '=',))
for k, v in todo.items():
if not hasattr(ns, k):
val = evaluator.evaluate(v)
status = evaluator.status()
if status == 0:
evaluator.setVariable(k, val)
setattr(ns, k, val)
if debug:
logger.info('Imported global value: "' + k + '" = "' + str(val) + '" into namespace' + ns.__name__)
del todo[k]
num = num + 1
logger.info('+++ Imported %d global values to namespace:%s' % (num, ns.__name__),)
# ---------------------------------------------------------------------------
def _getKernelProperty(self, name):
if ret.status > 0:
return ret.data
elif hasattr(self.get(), name):
return getattr(self.get(), name)
elif hasattr(self, name):
return getattr(self, name)
msg = 'DigiKernel::GetProperty [Unhandled]: Cannot access Kernel.' + name
raise KeyError(msg)
# ---------------------------------------------------------------------------
def _setKernelProperty(self, name, value):
if Interface.setPropertyKernel(self.get(), str(name), str(value)):
return
msg = 'DigiKernel::SetProperty [Unhandled]: Cannot set Kernel.' + name + ' = ' + str(value)
raise KeyError(msg)
# ---------------------------------------------------------------------------
def TestAction(kernel, nam, sleep=0):
obj = Interface.createAction(kernel, str('DigiTestAction/' + nam))
if sleep != 0:
obj.sleep = sleep
return obj
# ---------------------------------------------------------------------------
def Action(kernel, nam, **options):
action = Interface.createAction(kernel, str(nam))
for option in options.items():
setattr(action, option[0], option[1])
return action
# ---------------------------------------------------------------------------
def _get_action(self):
return Interface.toAction(self)
# if hasattr(self, 'I_am_a_ROOT_interface_handle'):
# return Interface.toAction(self.get())
# return self
# ---------------------------------------------------------------------------
def _adopt_property(self, action, foreign_name, local_name):
proc = _get_action(action)
_get_action(self).adopt_property(proc, str(foreign_name), str(local_name))
# ---------------------------------------------------------------------------
def _add_property(self, name, value):
Interface.addProperty(_get_action(self), str(name), value)
# ---------------------------------------------------------------------------
def _add_position_property(self, name, value):
Interface.addPositionProperty(_get_action(self), str(name), str(value))
# ---------------------------------------------------------------------------
def _add_set_property(self, name, value):
Interface.addSetProperty(_get_action(self), str(name), value)
# ---------------------------------------------------------------------------
def _add_list_property(self, name, value):
Interface.addListProperty(_get_action(self), str(name), value)
# ---------------------------------------------------------------------------
def _add_vector_property(self, name, value):
Interface.addVectorProperty(_get_action(self), str(name), value)
# ---------------------------------------------------------------------------
def _add_mapped_property(self, name, value):
Interface.addMappedProperty(_get_action(self), str(name), value)
# ---------------------------------------------------------------------------
def _kernel_terminate(self):
return self.get().terminate()
# ---------------------------------------------------------------------------
Kernel.__getattr__ = _getKernelProperty
Kernel.__setattr__ = _setKernelProperty
# ---------------------------------------------------------------------------
def _default_adopt(self, action):
getattr(self, '__adopt')(action.get())
# ---------------------------------------------------------------------------
def _adopt_event_action(self, action):
" Helper to convert DigiActions objects to DigiEventAction "
proc = Interface.toEventAction(_get_action(action))
attr = getattr(self, '__adopt')
attr(proc)
# ---------------------------------------------------------------------------
def _adopt_container_processor(self, action, container):
" Helper to convert DigiActions objects to DigiEventAction "
attr = getattr(_get_action(self), 'adopt_processor')
proc = Interface.toContainerProcessor(_get_action(action))
# ---------------------------------------------------------------------------
def _adopt_sequence_action(self, name, **options):
" Helper to adopt DigiAction objects for DigiSynchronize "
kernel = Interface.createKernel(_get_action(self))
action = Action(kernel, name)
for option in options.items():
setattr(action, option[0], option[1])
self.adopt(action)
return action
# ---------------------------------------------------------------------------
proc = Interface.toContainerProcessor(_get_action(action))
attr = getattr(_get_action(self), '__adopt_processor')
attr(proc, containers)
# ---------------------------------------------------------------------------
def _get(self, name):
a = Interface.toAction(self)
if ret.status > 0:
return ret.data
elif hasattr(self.action, name):
return getattr(self.action, name)
elif a.__class__ != self.__class__ and hasattr(a, name):
msg = 'DigiAction::GetProperty [Unhandled]: Cannot access property ' + a.name() + '.' + name
raise KeyError(msg)
# ---------------------------------------------------------------------------
def _set(self, name, value):
"""This function is called when properties are passed to the c++ objects."""
name = dd4hep.unicode_2_string(name)
value = dd4hep.unicode_2_string(value)
if Interface.setProperty(action, name, value):
return
msg = 'DigiAction::SetProperty [Unhandled]: Cannot set ' + action.name() + '.' + name + ' = ' + value
raise KeyError(msg)
# ---------------------------------------------------------------------------
def _props(obj, **extensions):
cls = getattr(current, obj)
for extension in extensions.items():
call = extension[0]
if hasattr(cls, call):
setattr(cls, '__' + call, getattr(cls, call))
setattr(cls, call, extension[1])
cls.__getattr__ = _get
cls.__setattr__ = _set
return cls
# ---------------------------------------------------------------------------
#
# Import unmodified classes from C++
_import_class('digi', 'DigiKernel')
_import_class('digi', 'DigiContext')
_import_class('digi', 'DigiAction')
_import_class('digi', 'DigiEventAction')
_import_class('digi', 'DigiInputAction')
#
# Import classes with specialized python extensions
_props('ActionHandle',
adopt_property=_adopt_property,
add_property=_add_property,
add_position_property=_add_position_property,
add_set_property=_add_set_property,
add_list_property=_add_list_property,
add_vector_property=_add_vector_property,
add_mapped_property=_add_mapped_property)
_props('DigiSynchronize', adopt=_adopt_event_action, adopt_action=_adopt_sequence_action)
_props('DigiActionSequence', adopt=_adopt_event_action, adopt_action=_adopt_sequence_action)
_props('DigiParallelActionSequence', adopt_action=_adopt_sequence_action)
_props('DigiSequentialActionSequence', adopt_action=_adopt_sequence_action)
_props('DigiContainerSequenceAction', adopt_container_processor=_adopt_container_processor)
_props('DigiMultiContainerProcessor', adopt_processor=_adopt_processor)
#
# Need to import digitize late, since it cross includes dddigi
Digitize = None
try:
import digitize
Digitize = digitize.Digitize
except Exception as X:
logger.error('Failed to import digitize: ' + str(X))