195 lines
6.8 KiB
Python
195 lines
6.8 KiB
Python
|
"""
|
||
|
>>> from direct.showbase import ObjectReport
|
||
|
|
||
|
>>> o=ObjectReport.ObjectReport('baseline')
|
||
|
>>> run()
|
||
|
...
|
||
|
|
||
|
>>> o2=ObjectReport.ObjectReport('')
|
||
|
>>> o.diff(o2)
|
||
|
"""
|
||
|
|
||
|
__all__ = ['ExclusiveObjectPool', 'ObjectReport']
|
||
|
|
||
|
from direct.directnotify.DirectNotifyGlobal import directNotify
|
||
|
from direct.showbase import DirectObject, ObjectPool, GarbageReport
|
||
|
from direct.showbase.PythonUtil import makeList, Sync
|
||
|
import gc
|
||
|
import sys
|
||
|
|
||
|
if sys.version_info >= (3, 0):
|
||
|
import builtins
|
||
|
else:
|
||
|
import __builtin__ as builtins
|
||
|
|
||
|
class ExclusiveObjectPool(DirectObject.DirectObject):
|
||
|
# ObjectPool specialization that excludes particular objects
|
||
|
# IDs of objects to globally exclude from reporting
|
||
|
_ExclObjs = []
|
||
|
_ExclObjIds = {}
|
||
|
_SyncMaster = Sync('ExclusiveObjectPool.ExcludedObjectList')
|
||
|
_SerialNumGen = SerialNumGen()
|
||
|
|
||
|
@classmethod
|
||
|
def addExclObjs(cls, *objs):
|
||
|
for obj in makeList(objs):
|
||
|
if id(obj) not in cls._ExclObjIds:
|
||
|
cls._ExclObjs.append(obj)
|
||
|
cls._ExclObjIds.setdefault(id(obj), 0)
|
||
|
cls._ExclObjIds[id(obj)] += 1
|
||
|
cls._SyncMaster.change()
|
||
|
@classmethod
|
||
|
def removeExclObjs(cls, *objs):
|
||
|
for obj in makeList(objs):
|
||
|
assert id(obj) in cls._ExclObjIds
|
||
|
cls._ExclObjIds[id(obj)] -= 1
|
||
|
if cls._ExclObjIds[id(obj)] == 0:
|
||
|
del cls._ExclObjIds[id(obj)]
|
||
|
cls._ExclObjs.remove(obj)
|
||
|
cls._SyncMaster.change()
|
||
|
|
||
|
def __init__(self, objects):
|
||
|
self._objects = list(objects)
|
||
|
self._postFilterObjs = []
|
||
|
self._sync = Sync('%s-%s' % (self.__class__.__name__,
|
||
|
self._SerialNumGen.next()),
|
||
|
self._SyncMaster)
|
||
|
self._sync.invalidate()
|
||
|
ExclusiveObjectPool.addExclObjs(self._objects, self._postFilterObjs,
|
||
|
self._sync)
|
||
|
|
||
|
def destroy(self):
|
||
|
self.ignoreAll()
|
||
|
ExclusiveObjectPool.removeExclObjs(self._objects, self._postFilterObjs,
|
||
|
self._sync)
|
||
|
del self._objects
|
||
|
del self._postFilterObjs
|
||
|
del self._sync
|
||
|
|
||
|
def _resync(self):
|
||
|
if self._sync.isSynced(self._SyncMaster):
|
||
|
return
|
||
|
if hasattr(self, '_filteredPool'):
|
||
|
ExclusiveObjectPool.removeExclObjs(*self._filteredPool._getInternalObjs())
|
||
|
ExclusiveObjectPool.removeExclObjs(self._filteredPool)
|
||
|
del self._filteredPool
|
||
|
del self._postFilterObjs[:]
|
||
|
for obj in self._objects:
|
||
|
if id(obj) not in ExclusiveObjectPool._ExclObjIds:
|
||
|
self._postFilterObjs.append(obj)
|
||
|
self._filteredPool = ExclusiveObjectPool(self._postFilterObjs)
|
||
|
ExclusiveObjectPool.addExclObjs(self._filteredPool)
|
||
|
ExclusiveObjectPool.addExclObjs(*self._filteredPool._getInternalObjs())
|
||
|
self._sync.sync(self._SyncMaster)
|
||
|
|
||
|
def getObjsOfType(self, type):
|
||
|
self._resync()
|
||
|
return self._filteredPool.getObjsOfType(type)
|
||
|
def printObjsOfType(self, type):
|
||
|
self._resync()
|
||
|
return self._filteredPool.printObjsOfType(type)
|
||
|
def diff(self, other):
|
||
|
self._resync()
|
||
|
return self._filteredPool.diff(other._filteredPool)
|
||
|
def typeFreqStr(self):
|
||
|
self._resync()
|
||
|
return self._filteredPool.typeFreqStr()
|
||
|
def __len__(self):
|
||
|
self._resync()
|
||
|
return len(self._filteredPool)
|
||
|
|
||
|
class ObjectReport:
|
||
|
"""report on every Python object in the current process"""
|
||
|
notify = directNotify.newCategory('ObjectReport')
|
||
|
|
||
|
def __init__(self, name, log=True):
|
||
|
gr = GarbageReport.GarbageReport('ObjectReport\'s GarbageReport: %s' % name, log=log)
|
||
|
gr.destroy()
|
||
|
del gr
|
||
|
self._name = name
|
||
|
self._pool = ObjectPool.ObjectPool(self._getObjectList())
|
||
|
#ExclusiveObjectPool.addExclObjs(self, self._pool, self._name)
|
||
|
if log:
|
||
|
self.notify.info('===== ObjectReport: \'%s\' =====\n%s' % (self._name, self.typeFreqStr()))
|
||
|
|
||
|
def destroy(self):
|
||
|
#ExclusiveObjectPool.removeExclObjs(self, self._pool, self._name)
|
||
|
self._pool.destroy()
|
||
|
del self._pool
|
||
|
del self._name
|
||
|
|
||
|
def typeFreqStr(self):
|
||
|
return self._pool.typeFreqStr()
|
||
|
|
||
|
def diff(self, other):
|
||
|
return self._pool.diff(other._pool)
|
||
|
|
||
|
def getObjectPool(self):
|
||
|
return self._pool
|
||
|
|
||
|
def _getObjectList(self):
|
||
|
if hasattr(sys, 'getobjects'):
|
||
|
return sys.getobjects(0)
|
||
|
else:
|
||
|
gc.collect()
|
||
|
# grab gc's object list
|
||
|
gc_objects = gc.get_objects()
|
||
|
# use get_referents to find everything else
|
||
|
objects = gc_objects
|
||
|
objects.append(builtins.__dict__)
|
||
|
nextObjList = gc_objects
|
||
|
found = set()
|
||
|
found.add(id(objects))
|
||
|
found.add(id(found))
|
||
|
found.add(id(gc_objects))
|
||
|
for obj in objects:
|
||
|
found.add(id(obj))
|
||
|
# repeatedly call get_referents until we can't find any more objects
|
||
|
while len(nextObjList):
|
||
|
curObjList = nextObjList
|
||
|
nextObjList = []
|
||
|
for obj in curObjList:
|
||
|
refs = gc.get_referents(obj)
|
||
|
for ref in refs:
|
||
|
if id(ref) not in found:
|
||
|
found.add(id(ref))
|
||
|
objects.append(ref)
|
||
|
nextObjList.append(ref)
|
||
|
return objects
|
||
|
"""
|
||
|
if hasattr(sys, 'getobjects'):
|
||
|
return sys.getobjects(0)
|
||
|
else:
|
||
|
objs = []
|
||
|
stateStack = Stack()
|
||
|
root = builtins
|
||
|
objIds = set([id(root)])
|
||
|
stateStack.push((root, None, 0))
|
||
|
while True:
|
||
|
if len(stateStack) == 0:
|
||
|
break
|
||
|
obj, adjacents, resumeIndex = stateStack.pop()
|
||
|
objs.append(obj)
|
||
|
print id(obj)
|
||
|
if adjacents is None:
|
||
|
adjacents = gc.get_referents(obj)
|
||
|
adjacents.extend(gc.get_referrers(obj))
|
||
|
# pare the list down to the ones that have not already been visited
|
||
|
# to minimize calls to get_referents/get_referrers
|
||
|
newObjs = []
|
||
|
for adj in adjacents:
|
||
|
adjId = id(adj)
|
||
|
if adjId not in objIds:
|
||
|
objIds.add(adjId)
|
||
|
newObjs.append(adj)
|
||
|
adjacents = newObjs
|
||
|
if len(adjacents) == 0:
|
||
|
print 'DEAD END'
|
||
|
for i in range(resumeIndex, len(adjacents)):
|
||
|
adj = adjacents[i]
|
||
|
stateStack.push((obj, adjacents, i+1))
|
||
|
stateStack.push((adj, None, 0))
|
||
|
continue
|
||
|
"""
|
||
|
|