Merge pull request #175 from Eximius/master

PyAssimp optimize: load Armadillo: 33.8s -> 6.7s
pull/181/head
Alexander Gessler 2013-11-06 14:09:12 -08:00
commit 686ecf7d34
1 changed files with 55 additions and 44 deletions

View File

@ -57,18 +57,33 @@ def make_tuple(ai_obj, type = None):
return res return res
# It is faster and more correct to have an init function for each assimp class
def _init_face(aiFace):
aiFace.indices = [aiFace.mIndices[i] for i in range(aiFace.mNumIndices)]
assimp_struct_inits = \
{ structs.Face : _init_face }
def call_init(obj, caller = None): def call_init(obj, caller = None):
# init children if helper.hasattr_silent(obj,'contents'): #pointer
if helper.hasattr_silent(obj, '_init'): _init(obj.contents, obj, caller)
obj._init(parent = caller) else:
_init(obj,parent=caller)
# pointers
elif helper.hasattr_silent(obj, 'contents'):
if helper.hasattr_silent(obj.contents, '_init'):
obj.contents._init(target = obj, parent = caller)
def _is_init_type(obj):
if helper.hasattr_silent(obj,'contents'): #pointer
return _is_init_type(obj[0])
# null-pointer case that arises when we reach a mesh attribute
# like mBitangents which use mNumVertices rather than mNumBitangents
# so it breaks the 'is iterable' check.
# Basically:
# FIXME!
elif not bool(obj):
return False
tname = obj.__class__.__name__
return not (tname[:2] == 'c_' or tname == 'Structure' \
or tname == 'POINTER') and not isinstance(obj,int)
def _init(self, target = None, parent = None): def _init(self, target = None, parent = None):
""" """
Custom initialize() for C structs, adds safely accessable member functionality. Custom initialize() for C structs, adds safely accessable member functionality.
@ -76,29 +91,35 @@ def _init(self, target = None, parent = None):
:param target: set the object which receive the added methods. Useful when manipulating :param target: set the object which receive the added methods. Useful when manipulating
pointers, to skip the intermediate 'contents' deferencing. pointers, to skip the intermediate 'contents' deferencing.
""" """
if helper.hasattr_silent(self, '_is_init'):
return self
self._is_init = True
if not target: if not target:
target = self target = self
for m in dir(self): dirself = dir(self)
for m in dirself:
name = m[1:].lower()
if m.startswith("_"): if m.startswith("_"):
continue continue
obj = getattr(self, m)
if m.startswith('mNum'): if m.startswith('mNum'):
if 'm' + m[4:] in dir(self): if 'm' + m[4:] in dirself:
continue # will be processed later on continue # will be processed later on
else: else:
name = m[1:].lower()
obj = getattr(self, m)
setattr(target, name, obj) setattr(target, name, obj)
continue
if m == 'mName':
obj = self.mName
target.name = str(obj.data.decode("utf-8"))
target.__class__.__repr__ = lambda x: str(x.__class__) + "(" + x.name + ")"
target.__class__.__str__ = lambda x: x.name
continue
name = m[1:].lower()
obj = getattr(self, m)
# Create tuples # Create tuples
if isinstance(obj, assimp_structs_as_tuple): if isinstance(obj, assimp_structs_as_tuple):
@ -106,14 +127,6 @@ def _init(self, target = None, parent = None):
logger.debug(str(self) + ": Added array " + str(getattr(target, name)) + " as self." + name.lower()) logger.debug(str(self) + ": Added array " + str(getattr(target, name)) + " as self." + name.lower())
continue continue
if isinstance(obj, structs.String):
setattr(target, 'name', obj.data.decode("utf-8"))
setattr(target.__class__, '__repr__', lambda x: str(x.__class__) + "(" + x.name + ")")
setattr(target.__class__, '__str__', lambda x: x.name)
continue
if m.startswith('m'): if m.startswith('m'):
if name == "parent": if name == "parent":
@ -150,8 +163,15 @@ def _init(self, target = None, parent = None):
logger.debug(str(self) + ": Added list of " + str(obj) + " " + name + " as self." + name + " (type: " + str(type(obj)) + ")") logger.debug(str(self) + ": Added list of " + str(obj) + " " + name + " as self." + name + " (type: " + str(type(obj)) + ")")
# initialize array elements # initialize array elements
for e in getattr(target, name): try:
call_init(e, caller = target) init = assimp_struct_inits[type(obj[0])]
except KeyError:
if _is_init_type(obj[0]):
for e in getattr(target, name):
call_init(e, target)
else:
for e in getattr(target, name):
init(e)
except IndexError: except IndexError:
@ -174,8 +194,9 @@ def _init(self, target = None, parent = None):
setattr(target, name, obj) setattr(target, name, obj)
logger.debug("Added " + name + " as self." + name + " (type: " + str(type(obj)) + ")") logger.debug("Added " + name + " as self." + name + " (type: " + str(type(obj)) + ")")
call_init(obj, caller = target) if _is_init_type(obj):
call_init(obj, target)
@ -189,16 +210,6 @@ def _init(self, target = None, parent = None):
return self return self
"""
Python magic to add the _init() function to all C struct classes.
"""
for struct in dir(structs):
if not (struct.startswith('_') or struct.startswith('c_') or struct == "Structure" or struct == "POINTER") and not isinstance(getattr(structs, struct),int):
setattr(getattr(structs, struct), '_init', _init)
class AssimpLib(object): class AssimpLib(object):
""" """
Assimp-Singleton Assimp-Singleton
@ -281,7 +292,7 @@ def load(filename, processing=0):
#Uhhh, something went wrong! #Uhhh, something went wrong!
raise AssimpError("could not import file: %s" % filename) raise AssimpError("could not import file: %s" % filename)
scene = model.contents._init() scene = _init(model.contents)
recur_pythonize(scene.rootnode, scene) recur_pythonize(scene.rootnode, scene)