/Modules/_winapi.c
C | 1583 lines | 1007 code | 194 blank | 382 comment | 150 complexity | c4a817c2807cac06133bb209c65ba57d MD5 | raw file
Possible License(s): BSD-3-Clause, Unlicense, CC-BY-SA-3.0, 0BSD
- /*
- * Support routines from the Windows API
- *
- * This module was originally created by merging PC/_subprocess.c with
- * Modules/_multiprocessing/win32_functions.c.
- *
- * Copyright (c) 2004 by Fredrik Lundh <fredrik@pythonware.com>
- * Copyright (c) 2004 by Secret Labs AB, http://www.pythonware.com
- * Copyright (c) 2004 by Peter Astrand <astrand@lysator.liu.se>
- *
- * By obtaining, using, and/or copying this software and/or its
- * associated documentation, you agree that you have read, understood,
- * and will comply with the following terms and conditions:
- *
- * Permission to use, copy, modify, and distribute this software and
- * its associated documentation for any purpose and without fee is
- * hereby granted, provided that the above copyright notice appears in
- * all copies, and that both that copyright notice and this permission
- * notice appear in supporting documentation, and that the name of the
- * authors not be used in advertising or publicity pertaining to
- * distribution of the software without specific, written prior
- * permission.
- *
- * THE AUTHORS DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
- * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS.
- * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
- * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
- * OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
- * NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
- * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
- *
- */
- /* Licensed to PSF under a Contributor Agreement. */
- /* See http://www.python.org/2.4/license for licensing details. */
- #include "Python.h"
- #include "structmember.h"
- #define WINDOWS_LEAN_AND_MEAN
- #include "windows.h"
- #include <crtdbg.h>
- #include "winreparse.h"
- #if defined(MS_WIN32) && !defined(MS_WIN64)
- #define HANDLE_TO_PYNUM(handle) \
- PyLong_FromUnsignedLong((unsigned long) handle)
- #define PYNUM_TO_HANDLE(obj) ((HANDLE)PyLong_AsUnsignedLong(obj))
- #define F_POINTER "k"
- #define T_POINTER T_ULONG
- #else
- #define HANDLE_TO_PYNUM(handle) \
- PyLong_FromUnsignedLongLong((unsigned long long) handle)
- #define PYNUM_TO_HANDLE(obj) ((HANDLE)PyLong_AsUnsignedLongLong(obj))
- #define F_POINTER "K"
- #define T_POINTER T_ULONGLONG
- #endif
- #define F_HANDLE F_POINTER
- #define F_DWORD "k"
- #define T_HANDLE T_POINTER
- #define DWORD_MAX 4294967295U
- /* Grab CancelIoEx dynamically from kernel32 */
- static int has_CancelIoEx = -1;
- static BOOL (CALLBACK *Py_CancelIoEx)(HANDLE, LPOVERLAPPED);
- static int
- check_CancelIoEx()
- {
- if (has_CancelIoEx == -1)
- {
- HINSTANCE hKernel32 = GetModuleHandle("KERNEL32");
- * (FARPROC *) &Py_CancelIoEx = GetProcAddress(hKernel32,
- "CancelIoEx");
- has_CancelIoEx = (Py_CancelIoEx != NULL);
- }
- return has_CancelIoEx;
- }
- /*
- * A Python object wrapping an OVERLAPPED structure and other useful data
- * for overlapped I/O
- */
- typedef struct {
- PyObject_HEAD
- OVERLAPPED overlapped;
- /* For convenience, we store the file handle too */
- HANDLE handle;
- /* Whether there's I/O in flight */
- int pending;
- /* Whether I/O completed successfully */
- int completed;
- /* Buffer used for reading (optional) */
- PyObject *read_buffer;
- /* Buffer used for writing (optional) */
- Py_buffer write_buffer;
- } OverlappedObject;
- static void
- overlapped_dealloc(OverlappedObject *self)
- {
- DWORD bytes;
- int err = GetLastError();
- if (self->pending) {
- if (check_CancelIoEx() &&
- Py_CancelIoEx(self->handle, &self->overlapped) &&
- GetOverlappedResult(self->handle, &self->overlapped, &bytes, TRUE))
- {
- /* The operation is no longer pending -- nothing to do. */
- }
- else if (_Py_Finalizing == NULL)
- {
- /* The operation is still pending -- give a warning. This
- will probably only happen on Windows XP. */
- PyErr_SetString(PyExc_RuntimeError,
- "I/O operations still in flight while destroying "
- "Overlapped object, the process may crash");
- PyErr_WriteUnraisable(NULL);
- }
- else
- {
- /* The operation is still pending, but the process is
- probably about to exit, so we need not worry too much
- about memory leaks. Leaking self prevents a potential
- crash. This can happen when a daemon thread is cleaned
- up at exit -- see #19565. We only expect to get here
- on Windows XP. */
- CloseHandle(self->overlapped.hEvent);
- SetLastError(err);
- return;
- }
- }
- CloseHandle(self->overlapped.hEvent);
- SetLastError(err);
- if (self->write_buffer.obj)
- PyBuffer_Release(&self->write_buffer);
- Py_CLEAR(self->read_buffer);
- PyObject_Del(self);
- }
- /*[clinic input]
- module _winapi
- class _winapi.Overlapped "OverlappedObject *" "&OverlappedType"
- [clinic start generated code]*/
- /*[clinic end generated code: output=da39a3ee5e6b4b0d input=c13d3f5fd1dabb84]*/
- /*[python input]
- def create_converter(type_, format_unit):
- name = type_ + '_converter'
- # registered upon creation by CConverter's metaclass
- type(name, (CConverter,), {'type': type_, 'format_unit': format_unit})
- # format unit differs between platforms for these
- create_converter('HANDLE', '" F_HANDLE "')
- create_converter('HMODULE', '" F_HANDLE "')
- create_converter('LPSECURITY_ATTRIBUTES', '" F_POINTER "')
- create_converter('BOOL', 'i') # F_BOOL used previously (always 'i')
- create_converter('DWORD', 'k') # F_DWORD is always "k" (which is much shorter)
- create_converter('LPCTSTR', 's')
- create_converter('LPWSTR', 'u')
- create_converter('UINT', 'I') # F_UINT used previously (always 'I')
- class HANDLE_return_converter(CReturnConverter):
- type = 'HANDLE'
- def render(self, function, data):
- self.declare(data)
- self.err_occurred_if("_return_value == INVALID_HANDLE_VALUE", data)
- data.return_conversion.append(
- 'if (_return_value == NULL)\n Py_RETURN_NONE;\n')
- data.return_conversion.append(
- 'return_value = HANDLE_TO_PYNUM(_return_value);\n')
- class DWORD_return_converter(CReturnConverter):
- type = 'DWORD'
- def render(self, function, data):
- self.declare(data)
- self.err_occurred_if("_return_value == DWORD_MAX", data)
- data.return_conversion.append(
- 'return_value = Py_BuildValue("k", _return_value);\n')
- [python start generated code]*/
- /*[python end generated code: output=da39a3ee5e6b4b0d input=374076979596ebba]*/
- #include "clinic/_winapi.c.h"
- /*[clinic input]
- _winapi.Overlapped.GetOverlappedResult
- wait: bool
- /
- [clinic start generated code]*/
- static PyObject *
- _winapi_Overlapped_GetOverlappedResult_impl(OverlappedObject *self, int wait)
- /*[clinic end generated code: output=bdd0c1ed6518cd03 input=194505ee8e0e3565]*/
- {
- BOOL res;
- DWORD transferred = 0;
- DWORD err;
- Py_BEGIN_ALLOW_THREADS
- res = GetOverlappedResult(self->handle, &self->overlapped, &transferred,
- wait != 0);
- Py_END_ALLOW_THREADS
- err = res ? ERROR_SUCCESS : GetLastError();
- switch (err) {
- case ERROR_SUCCESS:
- case ERROR_MORE_DATA:
- case ERROR_OPERATION_ABORTED:
- self->completed = 1;
- self->pending = 0;
- break;
- case ERROR_IO_INCOMPLETE:
- break;
- default:
- self->pending = 0;
- return PyErr_SetExcFromWindowsErr(PyExc_IOError, err);
- }
- if (self->completed && self->read_buffer != NULL) {
- assert(PyBytes_CheckExact(self->read_buffer));
- if (transferred != PyBytes_GET_SIZE(self->read_buffer) &&
- _PyBytes_Resize(&self->read_buffer, transferred))
- return NULL;
- }
- return Py_BuildValue("II", (unsigned) transferred, (unsigned) err);
- }
- /*[clinic input]
- _winapi.Overlapped.getbuffer
- [clinic start generated code]*/
- static PyObject *
- _winapi_Overlapped_getbuffer_impl(OverlappedObject *self)
- /*[clinic end generated code: output=95a3eceefae0f748 input=347fcfd56b4ceabd]*/
- {
- PyObject *res;
- if (!self->completed) {
- PyErr_SetString(PyExc_ValueError,
- "can't get read buffer before GetOverlappedResult() "
- "signals the operation completed");
- return NULL;
- }
- res = self->read_buffer ? self->read_buffer : Py_None;
- Py_INCREF(res);
- return res;
- }
- /*[clinic input]
- _winapi.Overlapped.cancel
- [clinic start generated code]*/
- static PyObject *
- _winapi_Overlapped_cancel_impl(OverlappedObject *self)
- /*[clinic end generated code: output=fcb9ab5df4ebdae5 input=cbf3da142290039f]*/
- {
- BOOL res = TRUE;
- if (self->pending) {
- Py_BEGIN_ALLOW_THREADS
- if (check_CancelIoEx())
- res = Py_CancelIoEx(self->handle, &self->overlapped);
- else
- res = CancelIo(self->handle);
- Py_END_ALLOW_THREADS
- }
- /* CancelIoEx returns ERROR_NOT_FOUND if the I/O completed in-between */
- if (!res && GetLastError() != ERROR_NOT_FOUND)
- return PyErr_SetExcFromWindowsErr(PyExc_IOError, 0);
- self->pending = 0;
- Py_RETURN_NONE;
- }
- static PyMethodDef overlapped_methods[] = {
- _WINAPI_OVERLAPPED_GETOVERLAPPEDRESULT_METHODDEF
- _WINAPI_OVERLAPPED_GETBUFFER_METHODDEF
- _WINAPI_OVERLAPPED_CANCEL_METHODDEF
- {NULL}
- };
- static PyMemberDef overlapped_members[] = {
- {"event", T_HANDLE,
- offsetof(OverlappedObject, overlapped) + offsetof(OVERLAPPED, hEvent),
- READONLY, "overlapped event handle"},
- {NULL}
- };
- PyTypeObject OverlappedType = {
- PyVarObject_HEAD_INIT(NULL, 0)
- /* tp_name */ "_winapi.Overlapped",
- /* tp_basicsize */ sizeof(OverlappedObject),
- /* tp_itemsize */ 0,
- /* tp_dealloc */ (destructor) overlapped_dealloc,
- /* tp_print */ 0,
- /* tp_getattr */ 0,
- /* tp_setattr */ 0,
- /* tp_reserved */ 0,
- /* tp_repr */ 0,
- /* tp_as_number */ 0,
- /* tp_as_sequence */ 0,
- /* tp_as_mapping */ 0,
- /* tp_hash */ 0,
- /* tp_call */ 0,
- /* tp_str */ 0,
- /* tp_getattro */ 0,
- /* tp_setattro */ 0,
- /* tp_as_buffer */ 0,
- /* tp_flags */ Py_TPFLAGS_DEFAULT,
- /* tp_doc */ "OVERLAPPED structure wrapper",
- /* tp_traverse */ 0,
- /* tp_clear */ 0,
- /* tp_richcompare */ 0,
- /* tp_weaklistoffset */ 0,
- /* tp_iter */ 0,
- /* tp_iternext */ 0,
- /* tp_methods */ overlapped_methods,
- /* tp_members */ overlapped_members,
- /* tp_getset */ 0,
- /* tp_base */ 0,
- /* tp_dict */ 0,
- /* tp_descr_get */ 0,
- /* tp_descr_set */ 0,
- /* tp_dictoffset */ 0,
- /* tp_init */ 0,
- /* tp_alloc */ 0,
- /* tp_new */ 0,
- };
- static OverlappedObject *
- new_overlapped(HANDLE handle)
- {
- OverlappedObject *self;
- self = PyObject_New(OverlappedObject, &OverlappedType);
- if (!self)
- return NULL;
- self->handle = handle;
- self->read_buffer = NULL;
- self->pending = 0;
- self->completed = 0;
- memset(&self->overlapped, 0, sizeof(OVERLAPPED));
- memset(&self->write_buffer, 0, sizeof(Py_buffer));
- /* Manual reset, initially non-signalled */
- self->overlapped.hEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
- return self;
- }
- /* -------------------------------------------------------------------- */
- /* windows API functions */
- /*[clinic input]
- _winapi.CloseHandle
- handle: HANDLE
- /
- Close handle.
- [clinic start generated code]*/
- static PyObject *
- _winapi_CloseHandle_impl(PyModuleDef *module, HANDLE handle)
- /*[clinic end generated code: output=0548595c71cb4bf7 input=7f0e4ac36e0352b8]*/
- {
- BOOL success;
- Py_BEGIN_ALLOW_THREADS
- success = CloseHandle(handle);
- Py_END_ALLOW_THREADS
- if (!success)
- return PyErr_SetFromWindowsErr(0);
- Py_RETURN_NONE;
- }
- /*[clinic input]
- _winapi.ConnectNamedPipe
- handle: HANDLE
- overlapped as use_overlapped: int(c_default='0') = False
- [clinic start generated code]*/
- static PyObject *
- _winapi_ConnectNamedPipe_impl(PyModuleDef *module, HANDLE handle,
- int use_overlapped)
- /*[clinic end generated code: output=fed3b165d1bca95a input=edc83da007ebf3be]*/
- {
- BOOL success;
- OverlappedObject *overlapped = NULL;
- if (use_overlapped) {
- overlapped = new_overlapped(handle);
- if (!overlapped)
- return NULL;
- }
- Py_BEGIN_ALLOW_THREADS
- success = ConnectNamedPipe(handle,
- overlapped ? &overlapped->overlapped : NULL);
- Py_END_ALLOW_THREADS
- if (overlapped) {
- int err = GetLastError();
- /* Overlapped ConnectNamedPipe never returns a success code */
- assert(success == 0);
- if (err == ERROR_IO_PENDING)
- overlapped->pending = 1;
- else if (err == ERROR_PIPE_CONNECTED)
- SetEvent(overlapped->overlapped.hEvent);
- else {
- Py_DECREF(overlapped);
- return PyErr_SetFromWindowsErr(err);
- }
- return (PyObject *) overlapped;
- }
- if (!success)
- return PyErr_SetFromWindowsErr(0);
- Py_RETURN_NONE;
- }
- /*[clinic input]
- _winapi.CreateFile -> HANDLE
- file_name: LPCTSTR
- desired_access: DWORD
- share_mode: DWORD
- security_attributes: LPSECURITY_ATTRIBUTES
- creation_disposition: DWORD
- flags_and_attributes: DWORD
- template_file: HANDLE
- /
- [clinic start generated code]*/
- static HANDLE
- _winapi_CreateFile_impl(PyModuleDef *module, LPCTSTR file_name,
- DWORD desired_access, DWORD share_mode,
- LPSECURITY_ATTRIBUTES security_attributes,
- DWORD creation_disposition,
- DWORD flags_and_attributes, HANDLE template_file)
- /*[clinic end generated code: output=c6e1d78f8affd10c input=6423c3e40372dbd5]*/
- {
- HANDLE handle;
- Py_BEGIN_ALLOW_THREADS
- handle = CreateFile(file_name, desired_access,
- share_mode, security_attributes,
- creation_disposition,
- flags_and_attributes, template_file);
- Py_END_ALLOW_THREADS
- if (handle == INVALID_HANDLE_VALUE)
- PyErr_SetFromWindowsErr(0);
- return handle;
- }
- /*[clinic input]
- _winapi.CreateJunction
- src_path: LPWSTR
- dst_path: LPWSTR
- /
- [clinic start generated code]*/
- static PyObject *
- _winapi_CreateJunction_impl(PyModuleDef *module, LPWSTR src_path,
- LPWSTR dst_path)
- /*[clinic end generated code: output=eccae9364e46f6da input=8cd1f9964b6e3d36]*/
- {
- /* Privilege adjustment */
- HANDLE token = NULL;
- TOKEN_PRIVILEGES tp;
- /* Reparse data buffer */
- const USHORT prefix_len = 4;
- USHORT print_len = 0;
- USHORT rdb_size = 0;
- PREPARSE_DATA_BUFFER rdb = NULL;
- /* Junction point creation */
- HANDLE junction = NULL;
- DWORD ret = 0;
- if (src_path == NULL || dst_path == NULL)
- return PyErr_SetFromWindowsErr(ERROR_INVALID_PARAMETER);
- if (wcsncmp(src_path, L"\\??\\", prefix_len) == 0)
- return PyErr_SetFromWindowsErr(ERROR_INVALID_PARAMETER);
- /* Adjust privileges to allow rewriting directory entry as a
- junction point. */
- if (!OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES, &token))
- goto cleanup;
- if (!LookupPrivilegeValue(NULL, SE_RESTORE_NAME, &tp.Privileges[0].Luid))
- goto cleanup;
- tp.PrivilegeCount = 1;
- tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
- if (!AdjustTokenPrivileges(token, FALSE, &tp, sizeof(TOKEN_PRIVILEGES),
- NULL, NULL))
- goto cleanup;
- if (GetFileAttributesW(src_path) == INVALID_FILE_ATTRIBUTES)
- goto cleanup;
- /* Store the absolute link target path length in print_len. */
- print_len = (USHORT)GetFullPathNameW(src_path, 0, NULL, NULL);
- if (print_len == 0)
- goto cleanup;
- /* NUL terminator should not be part of print_len. */
- --print_len;
- /* REPARSE_DATA_BUFFER usage is heavily under-documented, especially for
- junction points. Here's what I've learned along the way:
- - A junction point has two components: a print name and a substitute
- name. They both describe the link target, but the substitute name is
- the physical target and the print name is shown in directory listings.
- - The print name must be a native name, prefixed with "\??\".
- - Both names are stored after each other in the same buffer (the
- PathBuffer) and both must be NUL-terminated.
- - There are four members defining their respective offset and length
- inside PathBuffer: SubstituteNameOffset, SubstituteNameLength,
- PrintNameOffset and PrintNameLength.
- - The total size we need to allocate for the REPARSE_DATA_BUFFER, thus,
- is the sum of:
- - the fixed header size (REPARSE_DATA_BUFFER_HEADER_SIZE)
- - the size of the MountPointReparseBuffer member without the PathBuffer
- - the size of the prefix ("\??\") in bytes
- - the size of the print name in bytes
- - the size of the substitute name in bytes
- - the size of two NUL terminators in bytes */
- rdb_size = REPARSE_DATA_BUFFER_HEADER_SIZE +
- sizeof(rdb->MountPointReparseBuffer) -
- sizeof(rdb->MountPointReparseBuffer.PathBuffer) +
- /* Two +1's for NUL terminators. */
- (prefix_len + print_len + 1 + print_len + 1) * sizeof(WCHAR);
- rdb = (PREPARSE_DATA_BUFFER)PyMem_RawMalloc(rdb_size);
- if (rdb == NULL)
- goto cleanup;
- memset(rdb, 0, rdb_size);
- rdb->ReparseTag = IO_REPARSE_TAG_MOUNT_POINT;
- rdb->ReparseDataLength = rdb_size - REPARSE_DATA_BUFFER_HEADER_SIZE;
- rdb->MountPointReparseBuffer.SubstituteNameOffset = 0;
- rdb->MountPointReparseBuffer.SubstituteNameLength =
- (prefix_len + print_len) * sizeof(WCHAR);
- rdb->MountPointReparseBuffer.PrintNameOffset =
- rdb->MountPointReparseBuffer.SubstituteNameLength + sizeof(WCHAR);
- rdb->MountPointReparseBuffer.PrintNameLength = print_len * sizeof(WCHAR);
- /* Store the full native path of link target at the substitute name
- offset (0). */
- wcscpy(rdb->MountPointReparseBuffer.PathBuffer, L"\\??\\");
- if (GetFullPathNameW(src_path, print_len + 1,
- rdb->MountPointReparseBuffer.PathBuffer + prefix_len,
- NULL) == 0)
- goto cleanup;
- /* Copy everything but the native prefix to the print name offset. */
- wcscpy(rdb->MountPointReparseBuffer.PathBuffer +
- prefix_len + print_len + 1,
- rdb->MountPointReparseBuffer.PathBuffer + prefix_len);
- /* Create a directory for the junction point. */
- if (!CreateDirectoryW(dst_path, NULL))
- goto cleanup;
- junction = CreateFileW(dst_path, GENERIC_READ | GENERIC_WRITE, 0, NULL,
- OPEN_EXISTING,
- FILE_FLAG_OPEN_REPARSE_POINT | FILE_FLAG_BACKUP_SEMANTICS, NULL);
- if (junction == INVALID_HANDLE_VALUE)
- goto cleanup;
- /* Make the directory entry a junction point. */
- if (!DeviceIoControl(junction, FSCTL_SET_REPARSE_POINT, rdb, rdb_size,
- NULL, 0, &ret, NULL))
- goto cleanup;
- cleanup:
- ret = GetLastError();
- CloseHandle(token);
- CloseHandle(junction);
- PyMem_RawFree(rdb);
- if (ret != 0)
- return PyErr_SetFromWindowsErr(ret);
- Py_RETURN_NONE;
- }
- /*[clinic input]
- _winapi.CreateNamedPipe -> HANDLE
- name: LPCTSTR
- open_mode: DWORD
- pipe_mode: DWORD
- max_instances: DWORD
- out_buffer_size: DWORD
- in_buffer_size: DWORD
- default_timeout: DWORD
- security_attributes: LPSECURITY_ATTRIBUTES
- /
- [clinic start generated code]*/
- static HANDLE
- _winapi_CreateNamedPipe_impl(PyModuleDef *module, LPCTSTR name,
- DWORD open_mode, DWORD pipe_mode,
- DWORD max_instances, DWORD out_buffer_size,
- DWORD in_buffer_size, DWORD default_timeout,
- LPSECURITY_ATTRIBUTES security_attributes)
- /*[clinic end generated code: output=44ca2a06a219b523 input=5a73530b84d8bc37]*/
- {
- HANDLE handle;
- Py_BEGIN_ALLOW_THREADS
- handle = CreateNamedPipe(name, open_mode, pipe_mode,
- max_instances, out_buffer_size,
- in_buffer_size, default_timeout,
- security_attributes);
- Py_END_ALLOW_THREADS
- if (handle == INVALID_HANDLE_VALUE)
- PyErr_SetFromWindowsErr(0);
- return handle;
- }
- /*[clinic input]
- _winapi.CreatePipe
- pipe_attrs: object
- Ignored internally, can be None.
- size: DWORD
- /
- Create an anonymous pipe.
- Returns a 2-tuple of handles, to the read and write ends of the pipe.
- [clinic start generated code]*/
- static PyObject *
- _winapi_CreatePipe_impl(PyModuleDef *module, PyObject *pipe_attrs,
- DWORD size)
- /*[clinic end generated code: output=fef99f3b4222bc78 input=c4f2cfa56ef68d90]*/
- {
- HANDLE read_pipe;
- HANDLE write_pipe;
- BOOL result;
- Py_BEGIN_ALLOW_THREADS
- result = CreatePipe(&read_pipe, &write_pipe, NULL, size);
- Py_END_ALLOW_THREADS
- if (! result)
- return PyErr_SetFromWindowsErr(GetLastError());
- return Py_BuildValue(
- "NN", HANDLE_TO_PYNUM(read_pipe), HANDLE_TO_PYNUM(write_pipe));
- }
- /* helpers for createprocess */
- static unsigned long
- getulong(PyObject* obj, const char* name)
- {
- PyObject* value;
- unsigned long ret;
- value = PyObject_GetAttrString(obj, name);
- if (! value) {
- PyErr_Clear(); /* FIXME: propagate error? */
- return 0;
- }
- ret = PyLong_AsUnsignedLong(value);
- Py_DECREF(value);
- return ret;
- }
- static HANDLE
- gethandle(PyObject* obj, const char* name)
- {
- PyObject* value;
- HANDLE ret;
- value = PyObject_GetAttrString(obj, name);
- if (! value) {
- PyErr_Clear(); /* FIXME: propagate error? */
- return NULL;
- }
- if (value == Py_None)
- ret = NULL;
- else
- ret = PYNUM_TO_HANDLE(value);
- Py_DECREF(value);
- return ret;
- }
- static PyObject*
- getenvironment(PyObject* environment)
- {
- Py_ssize_t i, envsize, totalsize;
- Py_UCS4 *buffer = NULL, *p, *end;
- PyObject *keys, *values, *res;
- /* convert environment dictionary to windows environment string */
- if (! PyMapping_Check(environment)) {
- PyErr_SetString(
- PyExc_TypeError, "environment must be dictionary or None");
- return NULL;
- }
- envsize = PyMapping_Length(environment);
- keys = PyMapping_Keys(environment);
- values = PyMapping_Values(environment);
- if (!keys || !values)
- goto error;
- totalsize = 1; /* trailing null character */
- for (i = 0; i < envsize; i++) {
- PyObject* key = PyList_GET_ITEM(keys, i);
- PyObject* value = PyList_GET_ITEM(values, i);
- if (! PyUnicode_Check(key) || ! PyUnicode_Check(value)) {
- PyErr_SetString(PyExc_TypeError,
- "environment can only contain strings");
- goto error;
- }
- if (totalsize > PY_SSIZE_T_MAX - PyUnicode_GET_LENGTH(key) - 1) {
- PyErr_SetString(PyExc_OverflowError, "environment too long");
- goto error;
- }
- totalsize += PyUnicode_GET_LENGTH(key) + 1; /* +1 for '=' */
- if (totalsize > PY_SSIZE_T_MAX - PyUnicode_GET_LENGTH(value) - 1) {
- PyErr_SetString(PyExc_OverflowError, "environment too long");
- goto error;
- }
- totalsize += PyUnicode_GET_LENGTH(value) + 1; /* +1 for '\0' */
- }
- buffer = PyMem_NEW(Py_UCS4, totalsize);
- if (! buffer) {
- PyErr_NoMemory();
- goto error;
- }
- p = buffer;
- end = buffer + totalsize;
- for (i = 0; i < envsize; i++) {
- PyObject* key = PyList_GET_ITEM(keys, i);
- PyObject* value = PyList_GET_ITEM(values, i);
- if (!PyUnicode_AsUCS4(key, p, end - p, 0))
- goto error;
- p += PyUnicode_GET_LENGTH(key);
- *p++ = '=';
- if (!PyUnicode_AsUCS4(value, p, end - p, 0))
- goto error;
- p += PyUnicode_GET_LENGTH(value);
- *p++ = '\0';
- }
- /* add trailing null byte */
- *p++ = '\0';
- assert(p == end);
- Py_XDECREF(keys);
- Py_XDECREF(values);
- res = PyUnicode_FromKindAndData(PyUnicode_4BYTE_KIND, buffer, p - buffer);
- PyMem_Free(buffer);
- return res;
- error:
- PyMem_Free(buffer);
- Py_XDECREF(keys);
- Py_XDECREF(values);
- return NULL;
- }
- /*[clinic input]
- _winapi.CreateProcess
- application_name: Py_UNICODE(accept={str, NoneType})
- command_line: Py_UNICODE(accept={str, NoneType})
- proc_attrs: object
- Ignored internally, can be None.
- thread_attrs: object
- Ignored internally, can be None.
- inherit_handles: BOOL
- creation_flags: DWORD
- env_mapping: object
- current_directory: Py_UNICODE(accept={str, NoneType})
- startup_info: object
- /
- Create a new process and its primary thread.
- The return value is a tuple of the process handle, thread handle,
- process ID, and thread ID.
- [clinic start generated code]*/
- static PyObject *
- _winapi_CreateProcess_impl(PyModuleDef *module, Py_UNICODE *application_name,
- Py_UNICODE *command_line, PyObject *proc_attrs,
- PyObject *thread_attrs, BOOL inherit_handles,
- DWORD creation_flags, PyObject *env_mapping,
- Py_UNICODE *current_directory,
- PyObject *startup_info)
- /*[clinic end generated code: output=874bb350ff9ed4ef input=4a43b05038d639bb]*/
- {
- BOOL result;
- PROCESS_INFORMATION pi;
- STARTUPINFOW si;
- PyObject* environment;
- wchar_t *wenvironment;
- ZeroMemory(&si, sizeof(si));
- si.cb = sizeof(si);
- /* note: we only support a small subset of all SI attributes */
- si.dwFlags = getulong(startup_info, "dwFlags");
- si.wShowWindow = (WORD)getulong(startup_info, "wShowWindow");
- si.hStdInput = gethandle(startup_info, "hStdInput");
- si.hStdOutput = gethandle(startup_info, "hStdOutput");
- si.hStdError = gethandle(startup_info, "hStdError");
- if (PyErr_Occurred())
- return NULL;
- if (env_mapping != Py_None) {
- environment = getenvironment(env_mapping);
- if (! environment)
- return NULL;
- wenvironment = PyUnicode_AsUnicode(environment);
- if (wenvironment == NULL)
- {
- Py_XDECREF(environment);
- return NULL;
- }
- }
- else {
- environment = NULL;
- wenvironment = NULL;
- }
- Py_BEGIN_ALLOW_THREADS
- result = CreateProcessW(application_name,
- command_line,
- NULL,
- NULL,
- inherit_handles,
- creation_flags | CREATE_UNICODE_ENVIRONMENT,
- wenvironment,
- current_directory,
- &si,
- &pi);
- Py_END_ALLOW_THREADS
- Py_XDECREF(environment);
- if (! result)
- return PyErr_SetFromWindowsErr(GetLastError());
- return Py_BuildValue("NNkk",
- HANDLE_TO_PYNUM(pi.hProcess),
- HANDLE_TO_PYNUM(pi.hThread),
- pi.dwProcessId,
- pi.dwThreadId);
- }
- /*[clinic input]
- _winapi.DuplicateHandle -> HANDLE
- source_process_handle: HANDLE
- source_handle: HANDLE
- target_process_handle: HANDLE
- desired_access: DWORD
- inherit_handle: BOOL
- options: DWORD = 0
- /
- Return a duplicate handle object.
- The duplicate handle refers to the same object as the original
- handle. Therefore, any changes to the object are reflected
- through both handles.
- [clinic start generated code]*/
- static HANDLE
- _winapi_DuplicateHandle_impl(PyModuleDef *module,
- HANDLE source_process_handle,
- HANDLE source_handle,
- HANDLE target_process_handle,
- DWORD desired_access, BOOL inherit_handle,
- DWORD options)
- /*[clinic end generated code: output=0799515b68b5237b input=b933e3f2356a8c12]*/
- {
- HANDLE target_handle;
- BOOL result;
- Py_BEGIN_ALLOW_THREADS
- result = DuplicateHandle(
- source_process_handle,
- source_handle,
- target_process_handle,
- &target_handle,
- desired_access,
- inherit_handle,
- options
- );
- Py_END_ALLOW_THREADS
- if (! result) {
- PyErr_SetFromWindowsErr(GetLastError());
- return INVALID_HANDLE_VALUE;
- }
- return target_handle;
- }
- /*[clinic input]
- _winapi.ExitProcess
- ExitCode: UINT
- /
- [clinic start generated code]*/
- static PyObject *
- _winapi_ExitProcess_impl(PyModuleDef *module, UINT ExitCode)
- /*[clinic end generated code: output=25f3b499c24cedc8 input=4f05466a9406c558]*/
- {
- #if defined(Py_DEBUG)
- SetErrorMode(SEM_FAILCRITICALERRORS|SEM_NOALIGNMENTFAULTEXCEPT|
- SEM_NOGPFAULTERRORBOX|SEM_NOOPENFILEERRORBOX);
- _CrtSetReportMode(_CRT_ASSERT, _CRTDBG_MODE_DEBUG);
- #endif
- ExitProcess(ExitCode);
- return NULL;
- }
- /*[clinic input]
- _winapi.GetCurrentProcess -> HANDLE
- Return a handle object for the current process.
- [clinic start generated code]*/
- static HANDLE
- _winapi_GetCurrentProcess_impl(PyModuleDef *module)
- /*[clinic end generated code: output=be29ac3ad5f8291e input=b213403fd4b96b41]*/
- {
- return GetCurrentProcess();
- }
- /*[clinic input]
- _winapi.GetExitCodeProcess -> DWORD
- process: HANDLE
- /
- Return the termination status of the specified process.
- [clinic start generated code]*/
- static DWORD
- _winapi_GetExitCodeProcess_impl(PyModuleDef *module, HANDLE process)
- /*[clinic end generated code: output=0b10f0848a410f65 input=61b6bfc7dc2ee374]*/
- {
- DWORD exit_code;
- BOOL result;
- result = GetExitCodeProcess(process, &exit_code);
- if (! result) {
- PyErr_SetFromWindowsErr(GetLastError());
- exit_code = DWORD_MAX;
- }
- return exit_code;
- }
- /*[clinic input]
- _winapi.GetLastError -> DWORD
- [clinic start generated code]*/
- static DWORD
- _winapi_GetLastError_impl(PyModuleDef *module)
- /*[clinic end generated code: output=0ea00d8e67bdd056 input=62d47fb9bce038ba]*/
- {
- return GetLastError();
- }
- /*[clinic input]
- _winapi.GetModuleFileName
- module_handle: HMODULE
- /
- Return the fully-qualified path for the file that contains module.
- The module must have been loaded by the current process.
- The module parameter should be a handle to the loaded module
- whose path is being requested. If this parameter is 0,
- GetModuleFileName retrieves the path of the executable file
- of the current process.
- [clinic start generated code]*/
- static PyObject *
- _winapi_GetModuleFileName_impl(PyModuleDef *module, HMODULE module_handle)
- /*[clinic end generated code: output=90063dc63bdbfa18 input=6d66ff7deca5d11f]*/
- {
- BOOL result;
- WCHAR filename[MAX_PATH];
- result = GetModuleFileNameW(module_handle, filename, MAX_PATH);
- filename[MAX_PATH-1] = '\0';
- if (! result)
- return PyErr_SetFromWindowsErr(GetLastError());
- return PyUnicode_FromWideChar(filename, wcslen(filename));
- }
- /*[clinic input]
- _winapi.GetStdHandle -> HANDLE
- std_handle: DWORD
- One of STD_INPUT_HANDLE, STD_OUTPUT_HANDLE, or STD_ERROR_HANDLE.
- /
- Return a handle to the specified standard device.
- The integer associated with the handle object is returned.
- [clinic start generated code]*/
- static HANDLE
- _winapi_GetStdHandle_impl(PyModuleDef *module, DWORD std_handle)
- /*[clinic end generated code: output=5f5ca28b28c6fad2 input=07016b06a2fc8826]*/
- {
- HANDLE handle;
- Py_BEGIN_ALLOW_THREADS
- handle = GetStdHandle(std_handle);
- Py_END_ALLOW_THREADS
- if (handle == INVALID_HANDLE_VALUE)
- PyErr_SetFromWindowsErr(GetLastError());
- return handle;
- }
- /*[clinic input]
- _winapi.GetVersion -> long
- Return the version number of the current operating system.
- [clinic start generated code]*/
- static long
- _winapi_GetVersion_impl(PyModuleDef *module)
- /*[clinic end generated code: output=95a2f8ad3b948ca8 input=e21dff8d0baeded2]*/
- /* Disable deprecation warnings about GetVersionEx as the result is
- being passed straight through to the caller, who is responsible for
- using it correctly. */
- #pragma warning(push)
- #pragma warning(disable:4996)
- {
- return GetVersion();
- }
- #pragma warning(pop)
- /*[clinic input]
- _winapi.OpenProcess -> HANDLE
- desired_access: DWORD
- inherit_handle: BOOL
- process_id: DWORD
- /
- [clinic start generated code]*/
- static HANDLE
- _winapi_OpenProcess_impl(PyModuleDef *module, DWORD desired_access,
- BOOL inherit_handle, DWORD process_id)
- /*[clinic end generated code: output=6bc52eda82a3d226 input=ec98c4cf4ea2ec36]*/
- {
- HANDLE handle;
- handle = OpenProcess(desired_access, inherit_handle, process_id);
- if (handle == NULL) {
- PyErr_SetFromWindowsErr(0);
- handle = INVALID_HANDLE_VALUE;
- }
- return handle;
- }
- /*[clinic input]
- _winapi.PeekNamedPipe
- handle: HANDLE
- size: int = 0
- /
- [clinic start generated code]*/
- static PyObject *
- _winapi_PeekNamedPipe_impl(PyModuleDef *module, HANDLE handle, int size)
- /*[clinic end generated code: output=e6c908e2fb63c798 input=c7aa53bfbce69d70]*/
- {
- PyObject *buf = NULL;
- DWORD nread, navail, nleft;
- BOOL ret;
- if (size < 0) {
- PyErr_SetString(PyExc_ValueError, "negative size");
- return NULL;
- }
- if (size) {
- buf = PyBytes_FromStringAndSize(NULL, size);
- if (!buf)
- return NULL;
- Py_BEGIN_ALLOW_THREADS
- ret = PeekNamedPipe(handle, PyBytes_AS_STRING(buf), size, &nread,
- &navail, &nleft);
- Py_END_ALLOW_THREADS
- if (!ret) {
- Py_DECREF(buf);
- return PyErr_SetExcFromWindowsErr(PyExc_IOError, 0);
- }
- if (_PyBytes_Resize(&buf, nread))
- return NULL;
- return Py_BuildValue("Nii", buf, navail, nleft);
- }
- else {
- Py_BEGIN_ALLOW_THREADS
- ret = PeekNamedPipe(handle, NULL, 0, NULL, &navail, &nleft);
- Py_END_ALLOW_THREADS
- if (!ret) {
- return PyErr_SetExcFromWindowsErr(PyExc_IOError, 0);
- }
- return Py_BuildValue("ii", navail, nleft);
- }
- }
- /*[clinic input]
- _winapi.ReadFile
- handle: HANDLE
- size: int
- overlapped as use_overlapped: int(c_default='0') = False
- [clinic start generated code]*/
- static PyObject *
- _winapi_ReadFile_impl(PyModuleDef *module, HANDLE handle, int size,
- int use_overlapped)
- /*[clinic end generated code: output=d7695db4db97b135 input=8dd810194e86ac7d]*/
- {
- DWORD nread;
- PyObject *buf;
- BOOL ret;
- DWORD err;
- OverlappedObject *overlapped = NULL;
- buf = PyBytes_FromStringAndSize(NULL, size);
- if (!buf)
- return NULL;
- if (use_overlapped) {
- overlapped = new_overlapped(handle);
- if (!overlapped) {
- Py_DECREF(buf);
- return NULL;
- }
- /* Steals reference to buf */
- overlapped->read_buffer = buf;
- }
- Py_BEGIN_ALLOW_THREADS
- ret = ReadFile(handle, PyBytes_AS_STRING(buf), size, &nread,
- overlapped ? &overlapped->overlapped : NULL);
- Py_END_ALLOW_THREADS
- err = ret ? 0 : GetLastError();
- if (overlapped) {
- if (!ret) {
- if (err == ERROR_IO_PENDING)
- overlapped->pending = 1;
- else if (err != ERROR_MORE_DATA) {
- Py_DECREF(overlapped);
- return PyErr_SetExcFromWindowsErr(PyExc_IOError, 0);
- }
- }
- return Py_BuildValue("NI", (PyObject *) overlapped, err);
- }
- if (!ret && err != ERROR_MORE_DATA) {
- Py_DECREF(buf);
- return PyErr_SetExcFromWindowsErr(PyExc_IOError, 0);
- }
- if (_PyBytes_Resize(&buf, nread))
- return NULL;
- return Py_BuildValue("NI", buf, err);
- }
- /*[clinic input]
- _winapi.SetNamedPipeHandleState
- named_pipe: HANDLE
- mode: object
- max_collection_count: object
- collect_data_timeout: object
- /
- [clinic start generated code]*/
- static PyObject *
- _winapi_SetNamedPipeHandleState_impl(PyModuleDef *module, HANDLE named_pipe,
- PyObject *mode,
- PyObject *max_collection_count,
- PyObject *collect_data_timeout)
- /*[clinic end generated code: output=25aa3c28dee223ce input=9142d72163d0faa6]*/
- {
- PyObject *oArgs[3] = {mode, max_collection_count, collect_data_timeout};
- DWORD dwArgs[3], *pArgs[3] = {NULL, NULL, NULL};
- int i;
- PyErr_Clear();
- for (i = 0 ; i < 3 ; i++) {
- if (oArgs[i] != Py_None) {
- dwArgs[i] = PyLong_AsUnsignedLongMask(oArgs[i]);
- if (PyErr_Occurred())
- return NULL;
- pArgs[i] = &dwArgs[i];
- }
- }
- if (!SetNamedPipeHandleState(named_pipe, pArgs[0], pArgs[1], pArgs[2]))
- return PyErr_SetFromWindowsErr(0);
- Py_RETURN_NONE;
- }
- /*[clinic input]
- _winapi.TerminateProcess
- handle: HANDLE
- exit_code: UINT
- /
- Terminate the specified process and all of its threads.
- [clinic start generated code]*/
- static PyObject *
- _winapi_TerminateProcess_impl(PyModuleDef *module, HANDLE handle,
- UINT exit_code)
- /*[clinic end generated code: output=937c1bb6219aca8b input=d6bc0aa1ee3bb4df]*/
- {
- BOOL result;
- result = TerminateProcess(handle, exit_code);
- if (! result)
- return PyErr_SetFromWindowsErr(GetLastError());
- Py_RETURN_NONE;
- }
- /*[clinic input]
- _winapi.WaitNamedPipe
- name: LPCTSTR
- timeout: DWORD
- /
- [clinic start generated code]*/
- static PyObject *
- _winapi_WaitNamedPipe_impl(PyModuleDef *module, LPCTSTR name, DWORD timeout)
- /*[clinic end generated code: output=5bca5e02f448c9d7 input=36fc781291b1862c]*/
- {
- BOOL success;
- Py_BEGIN_ALLOW_THREADS
- success = WaitNamedPipe(name, timeout);
- Py_END_ALLOW_THREADS
- if (!success)
- return PyErr_SetFromWindowsErr(0);
- Py_RETURN_NONE;
- }
- /*[clinic input]
- _winapi.WaitForMultipleObjects
- handle_seq: object
- wait_flag: BOOL
- milliseconds: DWORD(c_default='INFINITE') = _winapi.INFINITE
- /
- [clinic start generated code]*/
- static PyObject *
- _winapi_WaitForMultipleObjects_impl(PyModuleDef *module,
- PyObject *handle_seq, BOOL wait_flag,
- DWORD milliseconds)
- /*[clinic end generated code: output=acb440728d06d130 input=36f76ca057cd28a0]*/
- {
- DWORD result;
- HANDLE handles[MAXIMUM_WAIT_OBJECTS];
- HANDLE sigint_event = NULL;
- Py_ssize_t nhandles, i;
- if (!PySequence_Check(handle_seq)) {
- PyErr_Format(PyExc_TypeError,
- "sequence type expected, got '%s'",
- Py_TYPE(handle_seq)->tp_name);
- return NULL;
- }
- nhandles = PySequence_Length(handle_seq);
- if (nhandles == -1)
- return NULL;
- if (nhandles < 0 || nhandles >= MAXIMUM_WAIT_OBJECTS - 1) {
- PyErr_Format(PyExc_ValueError,
- "need at most %zd handles, got a sequence of length %zd",
- MAXIMUM_WAIT_OBJECTS - 1, nhandles);
- return NULL;
- }
- for (i = 0; i < nhandles; i++) {
- HANDLE h;
- PyObject *v = PySequence_GetItem(handle_seq, i);
- if (v == NULL)
- return NULL;
- if (!PyArg_Parse(v, F_HANDLE, &h)) {
- Py_DECREF(v);
- return NULL;
- }
- handles[i] = h;
- Py_DECREF(v);
- }
- /* If this is the main thread then make the wait interruptible
- by Ctrl-C unless we are waiting for *all* handles */
- if (!wait_flag && _PyOS_IsMainThread()) {
- sigint_event = _PyOS_SigintEvent();
- assert(sigint_event != NULL);
- handles[nhandles++] = sigint_event;
- }
- Py_BEGIN_ALLOW_THREADS
- if (sigint_event != NULL)
- ResetEvent(sigint_event);
- result = WaitForMultipleObjects((DWORD) nhandles, handles,
- wait_flag, milliseconds);
- Py_END_ALLOW_THREADS
- if (result == WAIT_FAILED)
- return PyErr_SetExcFromWindowsErr(PyExc_IOError, 0);
- else if (sigint_event != NULL && result == WAIT_OBJECT_0 + nhandles - 1) {
- errno = EINTR;
- return PyErr_SetFromErrno(PyExc_IOError);
- }
- return PyLong_FromLong((int) result);
- }
- /*[clinic input]
- _winapi.WaitForSingleObject -> long
- handle: HANDLE
- milliseconds: DWORD
- /
- Wait for a single object.
- Wait until the specified object is in the signaled state or
- the time-out interval elapses. The timeout value is specified
- in milliseconds.
- [clinic start generated code]*/
- static long
- _winapi_WaitForSingleObject_impl(PyModuleDef *module, HANDLE handle,
- DWORD milliseconds)
- /*[clinic end generated code: output=34ae40c269749c48 input=443d1ab076edc7b1]*/
- {
- DWORD result;
- Py_BEGIN_ALLOW_THREADS
- result = WaitForSingleObject(handle, milliseconds);
- Py_END_ALLOW_THREADS
- if (result == WAIT_FAILED) {
- PyErr_SetFromWindowsErr(GetLastError());
- return -1;
- }
- return result;
- }
- /*[clinic input]
- _winapi.WriteFile
- handle: HANDLE
- buffer: object
- overlapped as use_overlapped: int(c_default='0') = False
- [clinic start generated code]*/
- static PyObject *
- _winapi_WriteFile_impl(PyModuleDef *module, HANDLE handle, PyObject *buffer,
- int use_overlapped)
- /*[clinic end generated code: output=65e70ea41f4d2a1d input=51846a5af52053fd]*/
- {
- Py_buffer _buf, *buf;
- DWORD len, written;
- BOOL ret;
- DWORD err;
- OverlappedObject *overlapped = NULL;
- if (use_overlapped) {
- overlapped = new_overlapped(handle);
- if (!overlapped)
- return NULL;
- buf = &overlapped->write_buffer;
- }
- else
- buf = &_buf;
- if (!PyArg_Parse(buffer, "y*", buf)) {
- Py_XDECREF(overlapped);
- return NULL;
- }
- Py_BEGIN_ALLOW_THREADS
- len = (DWORD)Py_MIN(buf->len, DWORD_MAX);
- ret = WriteFile(handle, buf->buf, len, &written,
- overlapped ? &overlapped->overlapped : NULL);
- Py_END_ALLOW_THREADS
- err = ret ? 0 : GetLastError();
- if (overlapped) {
- if (!ret) {
- if (err == ERROR_IO_PENDING)
- overlapped->pending = 1;
- else {
- Py_DECREF(overlapped);
- return PyErr_SetExcFromWindowsErr(PyExc_IOError, 0);
- }
- }
- return Py_BuildValue("NI", (PyObject *) overlapped, err);
- }
- PyBuffer_Release(buf);
- if (!ret)
- return PyErr_SetExcFromWindowsErr(PyExc_IOError, 0);
- return Py_BuildValue("II", written, err);
- }
- static PyMethodDef winapi_functions[] = {
- _WINAPI_CLOSEHANDLE_METHODDEF
- _WINAPI_CONNECTNAMEDPIPE_METHODDEF
- _WINAPI_CREATEFILE_METHODDEF
- _WINAPI_CREATENAMEDPIPE_METHODDEF
- _WINAPI_CREATEPIPE_METHODDEF
- _WINAPI_CREATEPROCESS_METHODDEF
- _WINAPI_CREATEJUNCTION_METHODDEF
- _WINAPI_DUPLICATEHANDLE_METHODDEF
- _WINAPI_EXITPROCESS_METHODDEF
- _WINAPI_GETCURRENTPROCESS_METHODDEF
- _WINAPI_GETEXITCODEPROCESS_METHODDEF
- _WINAPI_GETLASTERROR_METHODDEF
- _WINAPI_GETMODULEFILENAME_METHODDEF
- _WINAPI_GETSTDHANDLE_METHODDEF
- _WINAPI_GETVERSION_METHODDEF
- _WINAPI_OPENPROCESS_METHODDEF
- _WINAPI_PEEKNAMEDPIPE_METHODDEF
- _WINAPI_READFILE_METHODDEF
- _WINAPI_SETNAMEDPIPEHANDLESTATE_METHODDEF
- _WINAPI_TERMINATEPROCESS_METHODDEF
- _WINAPI_WAITNAMEDPIPE_METHODDEF
- _WINAPI_WAITFORMULTIPLEOBJECTS_METHODDEF
- _WINAPI_WAITFORSINGLEOBJECT_METHODDEF
- _WINAPI_WRITEFILE_METHODDEF
- {NULL, NULL}
- };
- static struct PyModuleDef winapi_module = {
- PyModuleDef_HEAD_INIT,
- "_winapi",
- NULL,
- -1,
- winapi_functions,
- NULL,
- NULL,
- NULL,
- NULL
- };
- #define WINAPI_CONSTANT(fmt, con) \
- PyDict_SetItemString(d, #con, Py_BuildValue(fmt, con))
- PyMODINIT_FUNC
- PyInit__winapi(void)
- {
- PyObject *d;
- PyObject *m;
- if (PyType_Ready(&OverlappedType) < 0)
- return NULL;
- m = PyModule_Create(&winapi_module);
- if (m == NULL)
- return NULL;
- d = PyModule_GetDict(m);
- PyDict_SetItemString(d, "Overlapped", (PyObject *) &OverlappedType);
- /* constants */
- WINAPI_CONSTANT(F_DWORD, CREATE_NEW_CONSOLE);
- WINAPI_CONSTANT(F_DWORD, CREATE_NEW_PROCESS_GROUP);
- WINAPI_CONSTANT(F_DWORD, DUPLICATE_SAME_ACCESS);
- WINAPI_CONSTANT(F_DWORD, DUPLICATE_CLOSE_SOURCE);
- WINAPI_CONSTANT(F_DWORD, ERROR_ALREADY_EXISTS);
- WINAPI_CONSTANT(F_DWORD, ERROR_BROKEN_PIPE);
- WINAPI_CONSTANT(F_DWORD, ERROR_IO_PENDING);
- WINAPI_CONSTANT(F_DWORD, ERROR_MORE_DATA);
- WINAPI_CONSTANT(F_DWORD, ERROR_NETNAME_DELETED);
- WINAPI_CONSTANT(F_DWORD, ERROR_NO_SYSTEM_RESOURCES);
- WINAPI_CONSTANT(F_DWORD, ERROR_MORE_DATA);
- WINAPI_CONSTANT(F_DWORD, ERROR_NETNAME_DELETED);
- WINAPI_CONSTANT(F_DWORD, ERROR_NO_DATA);
- WINAPI_CONSTANT(F_DWORD, ERROR_NO_SYSTEM_RESOURCES);
- WINAPI_CONSTANT(F_DWORD, ERROR_OPERATION_ABORTED);
- WINAPI_CONSTANT(F_DWORD, ERROR_PIPE_BUSY);
- WINAPI_CONSTANT(F_DWORD, ERROR_PIPE_CONNECTED);
- WINAPI_CONSTANT(F_DWORD, ERROR_SEM_TIMEOUT);
- WINAPI_CONSTANT(F_DWORD, FILE_FLAG_FIRST_PIPE_INSTANCE);
- WINAPI_CONSTANT(F_DWORD, FILE_FLAG_OVERLAPPED);
- WINAPI_CONSTANT(F_DWORD, FILE_GENERIC_READ);
- WINAPI_CONSTANT(F_DWORD, FILE_GENERIC_WRITE);
- WINAPI_CONSTANT(F_DWORD, GENERIC_READ);
- WINAPI_CONSTANT(F_DWORD, GENERIC_WRITE);
- WINAPI_CONSTANT(F_DWORD, INFINITE);
- WINAPI_CONSTANT(F_DWORD, NMPWAIT_WAIT_FOREVER);
- WINAPI_CONSTANT(F_DWORD, OPEN_EXISTING);
- WINAPI_CONSTANT(F_DWORD, PIPE_ACCESS_DUPLEX);
- WINAPI_CONSTANT(F_DWORD, PIPE_ACCESS_INBOUND);
- WINAPI_CONSTANT(F_DWORD, PIPE_READMODE_MESSAGE);
- WINAPI_CONSTANT(F_DWORD, PIPE_TYPE_MESSAGE);
- WINAPI_CONSTANT(F_DWORD, PIPE_UNLIMITED_INSTANCES);
- WINAPI_CONSTANT(F_DWORD, PIPE_WAIT);
- WINAPI_CONSTANT(F_DWORD, PROCESS_ALL_ACCESS);
- WINAPI_CONSTANT(F_DWORD, PROCESS_DUP_HANDLE);
- WINAPI_CONSTANT(F_DWORD, STARTF_USESHOWWINDOW);
- WINAPI_CONSTANT(F_DWORD, STARTF_USESTDHANDLES);
- WINAPI_CONSTANT(F_DWORD, STD_INPUT_HANDLE);
- WINAPI_CONSTANT(F_DWORD, STD_OUTPUT_HANDLE);
- WINAPI_CONSTANT(F_DWORD, STD_ERROR_HANDLE);
- WINAPI_CONSTANT(F_DWORD, STILL_ACTIVE);
- WINAPI_CONSTANT(F_DWORD, SW_HIDE);
- WINAPI_CONSTANT(F_DWORD, WAIT_OBJECT_0);
- WINAPI_CONSTANT(F_DWORD, WAIT_ABANDONED_0);
- WINAPI_CONSTANT(F_DWORD, WAIT_TIMEOUT);
- WINAPI_CONSTANT("i", NULL);
- return m;
- }