Difficult-Rocket/pyglet/com.py
2021-05-24 22:28:11 +08:00

376 lines
13 KiB
Python

# ----------------------------------------------------------------------------
# pyglet
# Copyright (c) 2006-2008 Alex Holkner
# Copyright (c) 2008-2021 pyglet contributors
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# * Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in
# the documentation and/or other materials provided with the
# distribution.
# * Neither the name of pyglet nor the names of its
# contributors may be used to endorse or promote products
# derived from this software without specific prior written
# permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.
# ----------------------------------------------------------------------------
"""Minimal Windows COM interface.
Allows pyglet to use COM interfaces on Windows without comtypes. Unlike
comtypes, this module does not provide property interfaces, read typelibs,
nice-ify return values. We don't need anything that sophisticated to work with COM's.
Interfaces should derive from pIUnknown if their implementation is returned by the COM.
The Python COM interfaces are actually pointers to the implementation (take note
when translating methods that take an interface as argument).
(example: A Double Pointer is simply POINTER(MyInterface) as pInterface is already a POINTER.)
Interfaces can define methods::
class IDirectSound8(com.pIUnknown):
_methods_ = [
('CreateSoundBuffer', com.STDMETHOD()),
('GetCaps', com.STDMETHOD(LPDSCAPS)),
...
]
Only use STDMETHOD or METHOD for the method types (not ordinary ctypes
function types). The 'this' pointer is bound automatically... e.g., call::
device = IDirectSound8()
DirectSoundCreate8(None, ctypes.byref(device), None)
caps = DSCAPS()
device.GetCaps(caps)
Because STDMETHODs use HRESULT as the return type, there is no need to check
the return value.
Don't forget to manually manage memory... call Release() when you're done with
an interface.
"""
import sys
import ctypes
from pyglet.util import debug_print
_debug_com = debug_print('debug_com')
if sys.platform != 'win32':
raise ImportError('pyglet.com requires a Windows build of Python')
class GUID(ctypes.Structure):
_fields_ = [
('Data1', ctypes.c_ulong),
('Data2', ctypes.c_ushort),
('Data3', ctypes.c_ushort),
('Data4', ctypes.c_ubyte * 8)
]
def __init__(self, l, w1, w2, b1, b2, b3, b4, b5, b6, b7, b8):
self.Data1 = l
self.Data2 = w1
self.Data3 = w2
self.Data4[:] = (b1, b2, b3, b4, b5, b6, b7, b8)
def __repr__(self):
b1, b2, b3, b4, b5, b6, b7, b8 = self.Data4
return 'GUID(%x, %x, %x, %x, %x, %x, %x, %x, %x, %x, %x)' % (
self.Data1, self.Data2, self.Data3, b1, b2, b3, b4, b5, b6, b7, b8)
def __cmp__(self, other):
if isinstance(other, GUID):
return ctypes.cmp(bytes(self), bytes(other))
return -1
def __eq__(self, other):
return isinstance(other, GUID) and bytes(self) == bytes(other)
def __hash__(self):
return hash(bytes(self))
LPGUID = ctypes.POINTER(GUID)
IID = GUID
REFIID = ctypes.POINTER(IID)
class METHOD:
"""COM method."""
def __init__(self, restype, *args):
self.restype = restype
self.argtypes = args
def get_field(self):
# ctypes caches WINFUNCTYPE's so this should be ok.
return ctypes.WINFUNCTYPE(self.restype, *self.argtypes)
class STDMETHOD(METHOD):
"""COM method with HRESULT return value."""
def __init__(self, *args):
super(STDMETHOD, self).__init__(ctypes.HRESULT, *args)
class COMMethodInstance:
"""Binds a COM interface method."""
def __init__(self, name, i, method):
self.name = name
self.i = i
self.method = method
def __get__(self, obj, tp):
if obj is not None:
def _call(*args):
assert _debug_com('COM: #{} IN {}({}, {})'.format(self.i, self.name, obj.__class__.__name__, args))
ret = self.method.get_field()(self.i, self.name)(obj, *args)
assert _debug_com('COM: #{} OUT {}({}, {})'.format(self.i, self.name, obj.__class__.__name__, args))
assert _debug_com('COM: RETURN {}'.format(ret))
return ret
return _call
raise AttributeError()
class COMInterface(ctypes.Structure):
"""Dummy struct to serve as the type of all COM pointers."""
_fields_ = [
('lpVtbl', ctypes.c_void_p),
]
class InterfacePtrMeta(type(ctypes.POINTER(COMInterface))):
"""Allows interfaces to be subclassed as ctypes POINTER and expects to be populated with data from a COM object.
TODO: Phase this out and properly use POINTER(Interface) where applicable.
"""
def __new__(cls, name, bases, dct):
methods = []
for base in bases[::-1]:
methods.extend(base.__dict__.get('_methods_', ()))
methods.extend(dct.get('_methods_', ()))
for i, (n, method) in enumerate(methods):
dct[n] = COMMethodInstance(n, i, method)
dct['_type_'] = COMInterface
return super(InterfacePtrMeta, cls).__new__(cls, name, bases, dct)
# pyglet.util.with_metaclass does not work here, as the base class is from _ctypes.lib
# See https://wiki.python.org/moin/PortingToPy3k/BilingualQuickRef
pInterface = InterfacePtrMeta(str('Interface'),
(ctypes.POINTER(COMInterface),),
{'__doc__': 'Base COM interface pointer.'})
class COMInterfaceMeta(type):
"""This differs in the original as an implemented interface object, not a POINTER object.
Used when the user must implement their own functions within an interface rather than
being created and generated by the COM object itself. The types are automatically inserted in the ctypes type
cache so it can recognize the type arguments.
"""
def __new__(mcs, name, bases, dct):
methods = dct.pop("_methods_", None)
cls = type.__new__(mcs, name, bases, dct)
if methods is not None:
cls._methods_ = methods
if not bases:
_ptr_bases = (cls, COMPointer)
else:
_ptr_bases = (cls, ctypes.POINTER(bases[0]))
# Class type is dynamically created inside __new__ based on metaclass inheritence; update ctypes cache manually.
from ctypes import _pointer_type_cache
_pointer_type_cache[cls] = type(COMPointer)("POINTER({})".format(cls.__name__),
_ptr_bases,
{"__interface__": cls})
return cls
def __get_subclassed_methodcount(self):
"""Returns the amount of COM methods in all subclasses to determine offset of methods.
Order must be exact from the source when calling COM methods.
"""
try:
result = 0
for itf in self.mro()[1:-1]:
result += len(itf.__dict__["_methods_"])
return result
except KeyError as err:
(name,) = err.args
if name == "_methods_":
raise TypeError("Interface '{}' requires a _methods_ attribute.".format(itf.__name__))
raise
class COMPointerMeta(type(ctypes.c_void_p), COMInterfaceMeta):
"""Required to prevent metaclass conflicts with inheritance."""
class COMPointer(ctypes.c_void_p, metaclass=COMPointerMeta):
"""COM Pointer base, could use c_void_p but need to override from_param ."""
@classmethod
def from_param(cls, obj):
"""Allows obj to return ctypes pointers, even if it's base is not a ctype.
In this case, all we simply want is a ctypes pointer matching the cls interface from the obj.
"""
if obj is None:
return
try:
ptr_dct = obj._pointers
except AttributeError:
raise Exception("Interface method argument specified incorrectly, or passed wrong argument.", cls)
else:
try:
return ptr_dct[cls.__interface__]
except KeyError:
raise TypeError("Interface {} doesn't have a pointer in this class.".format(cls.__name__))
def _missing_impl(interface_name, method_name):
"""Functions that are not implemented use this to prevent errors when called."""
def missing_cb_func(*args):
"""Return E_NOTIMPL because the method is not implemented."""
assert _debug_com("Undefined method: {0} was called in interface: {1}".format(method_name, interface_name))
return 0
return missing_cb_func
def _found_impl(interface_name, method_name, method_func):
"""If a method was found in class, we can set it as a callback."""
def cb_func(*args, **kw):
try:
result = method_func(*args, **kw)
except Exception as err:
raise err
if not result: # QOL so callbacks don't need to specify a return for assumed OK's.
return 0
return result
return cb_func
def _make_callback_func(interface, name, method_func):
"""Create a callback function for ctypes if possible."""
if method_func is None:
return _missing_impl(interface, name)
return _found_impl(interface, name, method_func)
# Store structures with same fields to prevent duplicate table creations.
_cached_structures = {}
def create_vtbl_structure(fields, interface):
"""Create virtual table structure with fields for use in COM's."""
try:
return _cached_structures[fields]
except KeyError:
Vtbl = type("Vtbl_{}".format(interface.__name__), (ctypes.Structure,), {"_fields_": fields})
_cached_structures[fields] = Vtbl
return Vtbl
class COMObject:
"""A base class for defining a COM object for use with callbacks and custom implementations."""
_interfaces_ = []
def __new__(cls, *args, **kw):
new_cls = super(COMObject, cls).__new__(cls)
assert len(cls._interfaces_) > 0, "Atleast one interface must be defined to use a COMObject."
new_cls._pointers = {}
new_cls.__create_interface_pointers()
return new_cls
def __create_interface_pointers(cls):
"""Create a custom ctypes structure to handle COM functions in a COM Object."""
interfaces = tuple(cls._interfaces_)
for itf in interfaces[::-1]:
methods = []
fields = []
for interface in itf.__mro__[-2::-1]:
for method in interface._methods_:
name, com_method = method
found_method = getattr(cls, name, None)
mth = _make_callback_func(itf.__name__, name, found_method)
proto = ctypes.WINFUNCTYPE(com_method.restype, *com_method.argtypes)
fields.append((name, proto))
methods.append(proto(mth))
# Make a structure dynamically with the fields given.
itf_structure = create_vtbl_structure(tuple(fields), interface)
# Assign the methods to the fields
vtbl = itf_structure(*methods)
cls._pointers[itf] = ctypes.pointer(ctypes.pointer(vtbl))
@property
def pointers(self):
"""Returns pointers to the implemented interfaces in this COMObject. Read-only.
:type: dict
"""
return self._pointers
class Interface(metaclass=COMInterfaceMeta):
_methods_ = []
class IUnknown(metaclass=COMInterfaceMeta):
"""These methods are not implemented by default yet. Strictly for COM method ordering."""
_methods_ = [
('QueryInterface', STDMETHOD(ctypes.c_void_p, REFIID, ctypes.c_void_p)),
('AddRef', METHOD(ctypes.c_int, ctypes.c_void_p)),
('Release', METHOD(ctypes.c_int, ctypes.c_void_p))
]
class pIUnknown(pInterface):
_methods_ = [
('QueryInterface', STDMETHOD(REFIID, ctypes.c_void_p)),
('AddRef', METHOD(ctypes.c_int)),
('Release', METHOD(ctypes.c_int))
]