| Index: pylib/mozrunner/qijo.py | 
| =================================================================== | 
| --- pylib/mozrunner/qijo.py	(revision 0) | 
| +++ pylib/mozrunner/qijo.py	(revision 0) | 
| @@ -0,0 +1,162 @@ | 
| +from ctypes import c_void_p, POINTER, sizeof, Structure, windll, WinError, WINFUNCTYPE, addressof, c_size_t, c_ulong | 
| +from ctypes.wintypes import BOOL, BYTE, DWORD, HANDLE, LARGE_INTEGER | 
| + | 
| +LPVOID = c_void_p | 
| +LPDWORD = POINTER(DWORD) | 
| +SIZE_T = c_size_t | 
| +ULONG_PTR = POINTER(c_ulong) | 
| + | 
| +# A ULONGLONG is a 64-bit unsigned integer. | 
| +# Thus there are 8 bytes in a ULONGLONG. | 
| +# XXX why not import c_ulonglong ? | 
| +ULONGLONG = BYTE * 8 | 
| + | 
| +class IO_COUNTERS(Structure): | 
| +    # The IO_COUNTERS struct is 6 ULONGLONGs. | 
| +    # TODO: Replace with non-dummy fields. | 
| +    _fields_ = [('dummy', ULONGLONG * 6)] | 
| + | 
| +class JOBOBJECT_BASIC_ACCOUNTING_INFORMATION(Structure): | 
| +    _fields_ = [('TotalUserTime', LARGE_INTEGER), | 
| +                ('TotalKernelTime', LARGE_INTEGER), | 
| +                ('ThisPeriodTotalUserTime', LARGE_INTEGER), | 
| +                ('ThisPeriodTotalKernelTime', LARGE_INTEGER), | 
| +                ('TotalPageFaultCount', DWORD), | 
| +                ('TotalProcesses', DWORD), | 
| +                ('ActiveProcesses', DWORD), | 
| +                ('TotalTerminatedProcesses', DWORD)] | 
| + | 
| +class JOBOBJECT_BASIC_AND_IO_ACCOUNTING_INFORMATION(Structure): | 
| +    _fields_ = [('BasicInfo', JOBOBJECT_BASIC_ACCOUNTING_INFORMATION), | 
| +                ('IoInfo', IO_COUNTERS)] | 
| + | 
| +# see http://msdn.microsoft.com/en-us/library/ms684147%28VS.85%29.aspx | 
| +class JOBOBJECT_BASIC_LIMIT_INFORMATION(Structure): | 
| +    _fields_ = [('PerProcessUserTimeLimit', LARGE_INTEGER), | 
| +                ('PerJobUserTimeLimit', LARGE_INTEGER), | 
| +                ('LimitFlags', DWORD), | 
| +                ('MinimumWorkingSetSize', SIZE_T), | 
| +                ('MaximumWorkingSetSize', SIZE_T), | 
| +                ('ActiveProcessLimit', DWORD), | 
| +                ('Affinity', ULONG_PTR), | 
| +                ('PriorityClass', DWORD), | 
| +                ('SchedulingClass', DWORD) | 
| +                ] | 
| + | 
| +# see http://msdn.microsoft.com/en-us/library/ms684156%28VS.85%29.aspx | 
| +class JOBOBJECT_EXTENDED_LIMIT_INFORMATION(Structure): | 
| +    _fields_ = [('BasicLimitInformation', JOBOBJECT_BASIC_LIMIT_INFORMATION), | 
| +                ('IoInfo', IO_COUNTERS), | 
| +                ('ProcessMemoryLimit', SIZE_T), | 
| +                ('JobMemoryLimit', SIZE_T), | 
| +                ('PeakProcessMemoryUsed', SIZE_T), | 
| +                ('PeakJobMemoryUsed', SIZE_T)] | 
| + | 
| +# XXX Magical numbers like 8 should be documented | 
| +JobObjectBasicAndIoAccountingInformation = 8 | 
| + | 
| +# ...like magical number 9 comes from | 
| +# http://community.flexerasoftware.com/archive/index.php?t-181670.html | 
| +# I wish I had a more canonical source | 
| +JobObjectExtendedLimitInformation = 9 | 
| + | 
| +class JobObjectInfo(object): | 
| +    mapping = { 'JobObjectBasicAndIoAccountingInformation': 8, | 
| +                'JobObjectExtendedLimitInformation': 9 | 
| +                } | 
| +    structures = { 8: JOBOBJECT_BASIC_AND_IO_ACCOUNTING_INFORMATION, | 
| +                   9: JOBOBJECT_EXTENDED_LIMIT_INFORMATION | 
| +                   } | 
| +    def __init__(self, _class): | 
| +        if isinstance(_class, basestring): | 
| +            assert _class in self.mapping, 'Class should be one of %s; you gave %s' % (self.mapping, _class) | 
| +            _class = self.mapping[_class] | 
| +        assert _class in self.structures, 'Class should be one of %s; you gave %s' % (self.structures, _class) | 
| +        self.code = _class | 
| +        self.info = self.structures[_class]() | 
| + | 
| + | 
| +QueryInformationJobObjectProto = WINFUNCTYPE( | 
| +    BOOL,        # Return type | 
| +    HANDLE,      # hJob | 
| +    DWORD,       # JobObjectInfoClass | 
| +    LPVOID,      # lpJobObjectInfo | 
| +    DWORD,       # cbJobObjectInfoLength | 
| +    LPDWORD      # lpReturnLength | 
| +    ) | 
| + | 
| +QueryInformationJobObjectFlags = ( | 
| +    (1, 'hJob'), | 
| +    (1, 'JobObjectInfoClass'), | 
| +    (1, 'lpJobObjectInfo'), | 
| +    (1, 'cbJobObjectInfoLength'), | 
| +    (1, 'lpReturnLength', None) | 
| +    ) | 
| + | 
| +_QueryInformationJobObject = QueryInformationJobObjectProto( | 
| +    ('QueryInformationJobObject', windll.kernel32), | 
| +    QueryInformationJobObjectFlags | 
| +    ) | 
| + | 
| +class SubscriptableReadOnlyStruct(object): | 
| +    def __init__(self, struct): | 
| +        self._struct = struct | 
| + | 
| +    def _delegate(self, name): | 
| +        result = getattr(self._struct, name) | 
| +        if isinstance(result, Structure): | 
| +            return SubscriptableReadOnlyStruct(result) | 
| +        return result | 
| + | 
| +    def __getitem__(self, name): | 
| +        match = [fname for fname, ftype in self._struct._fields_ | 
| +                 if fname == name] | 
| +        if match: | 
| +            return self._delegate(name) | 
| +        raise KeyError(name) | 
| + | 
| +    def __getattr__(self, name): | 
| +        return self._delegate(name) | 
| + | 
| +def QueryInformationJobObject(hJob, JobObjectInfoClass): | 
| +    jobinfo = JobObjectInfo(JobObjectInfoClass) | 
| +    result = _QueryInformationJobObject( | 
| +        hJob=hJob, | 
| +        JobObjectInfoClass=jobinfo.code, | 
| +        lpJobObjectInfo=addressof(jobinfo.info), | 
| +        cbJobObjectInfoLength=sizeof(jobinfo.info) | 
| +        ) | 
| +    if not result: | 
| +        raise WinError() | 
| +    return SubscriptableReadOnlyStruct(jobinfo.info) | 
| + | 
| +def test_qijo(): | 
| +    from killableprocess import Popen | 
| + | 
| +    popen = Popen('c:\\windows\\notepad.exe') | 
| + | 
| +    try: | 
| +        result = QueryInformationJobObject(0, 8) | 
| +        raise AssertionError('throw should occur') | 
| +    except WindowsError, e: | 
| +        pass | 
| + | 
| +    try: | 
| +        result = QueryInformationJobObject(0, 1) | 
| +        raise AssertionError('throw should occur') | 
| +    except NotImplementedError, e: | 
| +        pass | 
| + | 
| +    result = QueryInformationJobObject(popen._job, 8) | 
| +    if result['BasicInfo']['ActiveProcesses'] != 1: | 
| +        raise AssertionError('expected ActiveProcesses to be 1') | 
| +    popen.kill() | 
| + | 
| +    result = QueryInformationJobObject(popen._job, 8) | 
| +    if result.BasicInfo.ActiveProcesses != 0: | 
| +        raise AssertionError('expected ActiveProcesses to be 0') | 
| + | 
| +if __name__ == '__main__': | 
| +    print "testing." | 
| +    test_qijo() | 
| +    print "success!" | 
|  |