mirror of
				https://github.com/python/cpython.git
				synced 2025-10-22 06:32:43 +00:00 
			
		
		
		
	 91445fbeb0
			
		
	
	
		91445fbeb0
		
	
	
	
	
		
			
			* read_buffer can only be used for TYPE_READ and TYPE_ACCEPT types * write_buffer can only be used for TYPE_WRITE type
		
			
				
	
	
		
			1380 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1380 lines
		
	
	
	
		
			38 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Support for overlapped IO
 | |
|  *
 | |
|  * Some code borrowed from Modules/_winapi.c of CPython
 | |
|  */
 | |
| 
 | |
| /* XXX check overflow and DWORD <-> Py_ssize_t conversions
 | |
|    Check itemsize */
 | |
| 
 | |
| #include "Python.h"
 | |
| #include "structmember.h"
 | |
| 
 | |
| #define WINDOWS_LEAN_AND_MEAN
 | |
| #include <winsock2.h>
 | |
| #include <ws2tcpip.h>
 | |
| #include <mswsock.h>
 | |
| 
 | |
| #if defined(MS_WIN32) && !defined(MS_WIN64)
 | |
| #  define F_POINTER "k"
 | |
| #  define T_POINTER T_ULONG
 | |
| #else
 | |
| #  define F_POINTER "K"
 | |
| #  define T_POINTER T_ULONGLONG
 | |
| #endif
 | |
| 
 | |
| #define F_HANDLE F_POINTER
 | |
| #define F_ULONG_PTR F_POINTER
 | |
| #define F_DWORD "k"
 | |
| #define F_BOOL "i"
 | |
| #define F_UINT "I"
 | |
| 
 | |
| #define T_HANDLE T_POINTER
 | |
| 
 | |
| enum {TYPE_NONE, TYPE_NOT_STARTED, TYPE_READ, TYPE_WRITE, TYPE_ACCEPT,
 | |
|       TYPE_CONNECT, TYPE_DISCONNECT, TYPE_CONNECT_NAMED_PIPE,
 | |
|       TYPE_WAIT_NAMED_PIPE_AND_CONNECT};
 | |
| 
 | |
| typedef struct {
 | |
|     PyObject_HEAD
 | |
|     OVERLAPPED overlapped;
 | |
|     /* For convenience, we store the file handle too */
 | |
|     HANDLE handle;
 | |
|     /* Error returned by last method call */
 | |
|     DWORD error;
 | |
|     /* Type of operation */
 | |
|     DWORD type;
 | |
|     union {
 | |
|         /* Buffer used for reading: TYPE_READ and TYPE_ACCEPT */
 | |
|         PyObject *read_buffer;
 | |
|         /* Buffer used for writing: TYPE_WRITE */
 | |
|         Py_buffer write_buffer;
 | |
|     };
 | |
| } OverlappedObject;
 | |
| 
 | |
| typedef struct {
 | |
|     OVERLAPPED *Overlapped;
 | |
|     HANDLE IocpHandle;
 | |
|     char Address[1];
 | |
| } WaitNamedPipeAndConnectContext;
 | |
| 
 | |
| /*
 | |
|  * Map Windows error codes to subclasses of OSError
 | |
|  */
 | |
| 
 | |
| static PyObject *
 | |
| SetFromWindowsErr(DWORD err)
 | |
| {
 | |
|     PyObject *exception_type;
 | |
| 
 | |
|     if (err == 0)
 | |
|         err = GetLastError();
 | |
|     switch (err) {
 | |
|         case ERROR_CONNECTION_REFUSED:
 | |
|             exception_type = PyExc_ConnectionRefusedError;
 | |
|             break;
 | |
|         case ERROR_CONNECTION_ABORTED:
 | |
|             exception_type = PyExc_ConnectionAbortedError;
 | |
|             break;
 | |
|         default:
 | |
|             exception_type = PyExc_OSError;
 | |
|     }
 | |
|     return PyErr_SetExcFromWindowsErr(exception_type, err);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Some functions should be loaded at runtime
 | |
|  */
 | |
| 
 | |
| static LPFN_ACCEPTEX Py_AcceptEx = NULL;
 | |
| static LPFN_CONNECTEX Py_ConnectEx = NULL;
 | |
| static LPFN_DISCONNECTEX Py_DisconnectEx = NULL;
 | |
| static BOOL (CALLBACK *Py_CancelIoEx)(HANDLE, LPOVERLAPPED) = NULL;
 | |
| 
 | |
| #define GET_WSA_POINTER(s, x)                                           \
 | |
|     (SOCKET_ERROR != WSAIoctl(s, SIO_GET_EXTENSION_FUNCTION_POINTER,    \
 | |
|                               &Guid##x, sizeof(Guid##x), &Py_##x,       \
 | |
|                               sizeof(Py_##x), &dwBytes, NULL, NULL))
 | |
| 
 | |
| static int
 | |
| initialize_function_pointers(void)
 | |
| {
 | |
|     GUID GuidAcceptEx = WSAID_ACCEPTEX;
 | |
|     GUID GuidConnectEx = WSAID_CONNECTEX;
 | |
|     GUID GuidDisconnectEx = WSAID_DISCONNECTEX;
 | |
|     HINSTANCE hKernel32;
 | |
|     SOCKET s;
 | |
|     DWORD dwBytes;
 | |
| 
 | |
|     s = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
 | |
|     if (s == INVALID_SOCKET) {
 | |
|         SetFromWindowsErr(WSAGetLastError());
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     if (!GET_WSA_POINTER(s, AcceptEx) ||
 | |
|         !GET_WSA_POINTER(s, ConnectEx) ||
 | |
|         !GET_WSA_POINTER(s, DisconnectEx))
 | |
|     {
 | |
|         closesocket(s);
 | |
|         SetFromWindowsErr(WSAGetLastError());
 | |
|         return -1;
 | |
|     }
 | |
| 
 | |
|     closesocket(s);
 | |
| 
 | |
|     /* On WinXP we will have Py_CancelIoEx == NULL */
 | |
|     hKernel32 = GetModuleHandle("KERNEL32");
 | |
|     *(FARPROC *)&Py_CancelIoEx = GetProcAddress(hKernel32, "CancelIoEx");
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Completion port stuff
 | |
|  */
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     CreateIoCompletionPort_doc,
 | |
|     "CreateIoCompletionPort(handle, port, key, concurrency) -> port\n\n"
 | |
|     "Create a completion port or register a handle with a port.");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_CreateIoCompletionPort(PyObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE FileHandle;
 | |
|     HANDLE ExistingCompletionPort;
 | |
|     ULONG_PTR CompletionKey;
 | |
|     DWORD NumberOfConcurrentThreads;
 | |
|     HANDLE ret;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE F_HANDLE F_ULONG_PTR F_DWORD,
 | |
|                           &FileHandle, &ExistingCompletionPort, &CompletionKey,
 | |
|                           &NumberOfConcurrentThreads))
 | |
|         return NULL;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = CreateIoCompletionPort(FileHandle, ExistingCompletionPort,
 | |
|                                  CompletionKey, NumberOfConcurrentThreads);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     if (ret == NULL)
 | |
|         return SetFromWindowsErr(0);
 | |
|     return Py_BuildValue(F_HANDLE, ret);
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     GetQueuedCompletionStatus_doc,
 | |
|     "GetQueuedCompletionStatus(port, msecs) -> (err, bytes, key, address)\n\n"
 | |
|     "Get a message from completion port.  Wait for up to msecs milliseconds.");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_GetQueuedCompletionStatus(PyObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE CompletionPort = NULL;
 | |
|     DWORD NumberOfBytes = 0;
 | |
|     ULONG_PTR CompletionKey = 0;
 | |
|     OVERLAPPED *Overlapped = NULL;
 | |
|     DWORD Milliseconds;
 | |
|     DWORD err;
 | |
|     BOOL ret;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE F_DWORD,
 | |
|                           &CompletionPort, &Milliseconds))
 | |
|         return NULL;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = GetQueuedCompletionStatus(CompletionPort, &NumberOfBytes,
 | |
|                                     &CompletionKey, &Overlapped, Milliseconds);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     err = ret ? ERROR_SUCCESS : GetLastError();
 | |
|     if (Overlapped == NULL) {
 | |
|         if (err == WAIT_TIMEOUT)
 | |
|             Py_RETURN_NONE;
 | |
|         else
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
|     return Py_BuildValue(F_DWORD F_DWORD F_ULONG_PTR F_POINTER,
 | |
|                          err, NumberOfBytes, CompletionKey, Overlapped);
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     PostQueuedCompletionStatus_doc,
 | |
|     "PostQueuedCompletionStatus(port, bytes, key, address) -> None\n\n"
 | |
|     "Post a message to completion port.");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_PostQueuedCompletionStatus(PyObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE CompletionPort;
 | |
|     DWORD NumberOfBytes;
 | |
|     ULONG_PTR CompletionKey;
 | |
|     OVERLAPPED *Overlapped;
 | |
|     BOOL ret;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE F_DWORD F_ULONG_PTR F_POINTER,
 | |
|                           &CompletionPort, &NumberOfBytes, &CompletionKey,
 | |
|                           &Overlapped))
 | |
|         return NULL;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = PostQueuedCompletionStatus(CompletionPort, NumberOfBytes,
 | |
|                                      CompletionKey, Overlapped);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     if (!ret)
 | |
|         return SetFromWindowsErr(0);
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Wait for a handle
 | |
|  */
 | |
| 
 | |
| struct PostCallbackData {
 | |
|     HANDLE CompletionPort;
 | |
|     LPOVERLAPPED Overlapped;
 | |
| };
 | |
| 
 | |
| static VOID CALLBACK
 | |
| PostToQueueCallback(PVOID lpParameter, BOOL TimerOrWaitFired)
 | |
| {
 | |
|     struct PostCallbackData *p = (struct PostCallbackData*) lpParameter;
 | |
| 
 | |
|     PostQueuedCompletionStatus(p->CompletionPort, TimerOrWaitFired,
 | |
|                                0, p->Overlapped);
 | |
|     /* ignore possible error! */
 | |
|     PyMem_Free(p);
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     RegisterWaitWithQueue_doc,
 | |
|     "RegisterWaitWithQueue(Object, CompletionPort, Overlapped, Timeout)\n"
 | |
|     "    -> WaitHandle\n\n"
 | |
|     "Register wait for Object; when complete CompletionPort is notified.\n");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_RegisterWaitWithQueue(PyObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE NewWaitObject;
 | |
|     HANDLE Object;
 | |
|     ULONG Milliseconds;
 | |
|     struct PostCallbackData data, *pdata;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE F_HANDLE F_POINTER F_DWORD,
 | |
|                           &Object,
 | |
|                           &data.CompletionPort,
 | |
|                           &data.Overlapped,
 | |
|                           &Milliseconds))
 | |
|         return NULL;
 | |
| 
 | |
|     pdata = PyMem_Malloc(sizeof(struct PostCallbackData));
 | |
|     if (pdata == NULL)
 | |
|         return SetFromWindowsErr(0);
 | |
| 
 | |
|     *pdata = data;
 | |
| 
 | |
|     if (!RegisterWaitForSingleObject(
 | |
|             &NewWaitObject, Object, (WAITORTIMERCALLBACK)PostToQueueCallback,
 | |
|             pdata, Milliseconds,
 | |
|             WT_EXECUTEINWAITTHREAD | WT_EXECUTEONLYONCE))
 | |
|     {
 | |
|         PyMem_Free(pdata);
 | |
|         return SetFromWindowsErr(0);
 | |
|     }
 | |
| 
 | |
|     return Py_BuildValue(F_HANDLE, NewWaitObject);
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     UnregisterWait_doc,
 | |
|     "UnregisterWait(WaitHandle) -> None\n\n"
 | |
|     "Unregister wait handle.\n");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_UnregisterWait(PyObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE WaitHandle;
 | |
|     BOOL ret;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE, &WaitHandle))
 | |
|         return NULL;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = UnregisterWait(WaitHandle);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     if (!ret)
 | |
|         return SetFromWindowsErr(0);
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Event functions -- currently only used by tests
 | |
|  */
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     CreateEvent_doc,
 | |
|     "CreateEvent(EventAttributes, ManualReset, InitialState, Name)"
 | |
|     " -> Handle\n\n"
 | |
|     "Create an event.  EventAttributes must be None.\n");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_CreateEvent(PyObject *self, PyObject *args)
 | |
| {
 | |
|     PyObject *EventAttributes;
 | |
|     BOOL ManualReset;
 | |
|     BOOL InitialState;
 | |
|     Py_UNICODE *Name;
 | |
|     HANDLE Event;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, "O" F_BOOL F_BOOL "Z",
 | |
|                           &EventAttributes, &ManualReset,
 | |
|                           &InitialState, &Name))
 | |
|         return NULL;
 | |
| 
 | |
|     if (EventAttributes != Py_None) {
 | |
|         PyErr_SetString(PyExc_ValueError, "EventAttributes must be None");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     Event = CreateEventW(NULL, ManualReset, InitialState, Name);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     if (Event == NULL)
 | |
|         return SetFromWindowsErr(0);
 | |
|     return Py_BuildValue(F_HANDLE, Event);
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     SetEvent_doc,
 | |
|     "SetEvent(Handle) -> None\n\n"
 | |
|     "Set event.\n");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_SetEvent(PyObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE Handle;
 | |
|     BOOL ret;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE, &Handle))
 | |
|         return NULL;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = SetEvent(Handle);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     if (!ret)
 | |
|         return SetFromWindowsErr(0);
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     ResetEvent_doc,
 | |
|     "ResetEvent(Handle) -> None\n\n"
 | |
|     "Reset event.\n");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_ResetEvent(PyObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE Handle;
 | |
|     BOOL ret;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE, &Handle))
 | |
|         return NULL;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = ResetEvent(Handle);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     if (!ret)
 | |
|         return SetFromWindowsErr(0);
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Bind socket handle to local port without doing slow getaddrinfo()
 | |
|  */
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     BindLocal_doc,
 | |
|     "BindLocal(handle, family) -> None\n\n"
 | |
|     "Bind a socket handle to an arbitrary local port.\n"
 | |
|     "family should AF_INET or AF_INET6.\n");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_BindLocal(PyObject *self, PyObject *args)
 | |
| {
 | |
|     SOCKET Socket;
 | |
|     int Family;
 | |
|     BOOL ret;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE "i", &Socket, &Family))
 | |
|         return NULL;
 | |
| 
 | |
|     if (Family == AF_INET) {
 | |
|         struct sockaddr_in addr;
 | |
|         memset(&addr, 0, sizeof(addr));
 | |
|         addr.sin_family = AF_INET;
 | |
|         addr.sin_port = 0;
 | |
|         addr.sin_addr.S_un.S_addr = INADDR_ANY;
 | |
|         ret = bind(Socket, (SOCKADDR*)&addr, sizeof(addr)) != SOCKET_ERROR;
 | |
|     } else if (Family == AF_INET6) {
 | |
|         struct sockaddr_in6 addr;
 | |
|         memset(&addr, 0, sizeof(addr));
 | |
|         addr.sin6_family = AF_INET6;
 | |
|         addr.sin6_port = 0;
 | |
|         addr.sin6_addr = in6addr_any;
 | |
|         ret = bind(Socket, (SOCKADDR*)&addr, sizeof(addr)) != SOCKET_ERROR;
 | |
|     } else {
 | |
|         PyErr_SetString(PyExc_ValueError, "expected tuple of length 2 or 4");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!ret)
 | |
|         return SetFromWindowsErr(WSAGetLastError());
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Windows equivalent of os.strerror() -- compare _ctypes/callproc.c
 | |
|  */
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     FormatMessage_doc,
 | |
|     "FormatMessage(error_code) -> error_message\n\n"
 | |
|     "Return error message for an error code.");
 | |
| 
 | |
| static PyObject *
 | |
| overlapped_FormatMessage(PyObject *ignore, PyObject *args)
 | |
| {
 | |
|     DWORD code, n;
 | |
|     WCHAR *lpMsgBuf;
 | |
|     PyObject *res;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_DWORD, &code))
 | |
|         return NULL;
 | |
| 
 | |
|     n = FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER |
 | |
|                        FORMAT_MESSAGE_FROM_SYSTEM,
 | |
|                        NULL,
 | |
|                        code,
 | |
|                        MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
 | |
|                        (LPWSTR) &lpMsgBuf,
 | |
|                        0,
 | |
|                        NULL);
 | |
|     if (n) {
 | |
|         while (iswspace(lpMsgBuf[n-1]))
 | |
|             --n;
 | |
|         lpMsgBuf[n] = L'\0';
 | |
|         res = Py_BuildValue("u", lpMsgBuf);
 | |
|     } else {
 | |
|         res = PyUnicode_FromFormat("unknown error code %u", code);
 | |
|     }
 | |
|     LocalFree(lpMsgBuf);
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
|  * Mark operation as completed - used when reading produces ERROR_BROKEN_PIPE
 | |
|  */
 | |
| 
 | |
| static void
 | |
| mark_as_completed(OVERLAPPED *ov)
 | |
| {
 | |
|     ov->Internal = 0;
 | |
|     if (ov->hEvent != NULL)
 | |
|         SetEvent(ov->hEvent);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * A Python object wrapping an OVERLAPPED structure and other useful data
 | |
|  * for overlapped I/O
 | |
|  */
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_doc,
 | |
|     "Overlapped object");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_new(PyTypeObject *type, PyObject *args, PyObject *kwds)
 | |
| {
 | |
|     OverlappedObject *self;
 | |
|     HANDLE event = INVALID_HANDLE_VALUE;
 | |
|     static char *kwlist[] = {"event", NULL};
 | |
| 
 | |
|     if (!PyArg_ParseTupleAndKeywords(args, kwds, "|" F_HANDLE, kwlist, &event))
 | |
|         return NULL;
 | |
| 
 | |
|     if (event == INVALID_HANDLE_VALUE) {
 | |
|         event = CreateEvent(NULL, TRUE, FALSE, NULL);
 | |
|         if (event == NULL)
 | |
|             return SetFromWindowsErr(0);
 | |
|     }
 | |
| 
 | |
|     self = PyObject_New(OverlappedObject, type);
 | |
|     if (self == NULL) {
 | |
|         if (event != NULL)
 | |
|             CloseHandle(event);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     self->handle = NULL;
 | |
|     self->error = 0;
 | |
|     self->type = TYPE_NONE;
 | |
|     self->read_buffer = NULL;
 | |
|     memset(&self->overlapped, 0, sizeof(OVERLAPPED));
 | |
|     memset(&self->write_buffer, 0, sizeof(Py_buffer));
 | |
|     if (event)
 | |
|         self->overlapped.hEvent = event;
 | |
|     return (PyObject *)self;
 | |
| }
 | |
| 
 | |
| static void
 | |
| Overlapped_dealloc(OverlappedObject *self)
 | |
| {
 | |
|     DWORD bytes;
 | |
|     DWORD olderr = GetLastError();
 | |
|     BOOL wait = FALSE;
 | |
|     BOOL ret;
 | |
| 
 | |
|     if (!HasOverlappedIoCompleted(&self->overlapped) &&
 | |
|         self->type != TYPE_NOT_STARTED)
 | |
|     {
 | |
|         if (Py_CancelIoEx && Py_CancelIoEx(self->handle, &self->overlapped))
 | |
|             wait = TRUE;
 | |
| 
 | |
|         Py_BEGIN_ALLOW_THREADS
 | |
|         ret = GetOverlappedResult(self->handle, &self->overlapped,
 | |
|                                   &bytes, wait);
 | |
|         Py_END_ALLOW_THREADS
 | |
| 
 | |
|         switch (ret ? ERROR_SUCCESS : GetLastError()) {
 | |
|             case ERROR_SUCCESS:
 | |
|             case ERROR_NOT_FOUND:
 | |
|             case ERROR_OPERATION_ABORTED:
 | |
|                 break;
 | |
|             default:
 | |
|                 PyErr_Format(
 | |
|                     PyExc_RuntimeError,
 | |
|                     "%R still has pending operation at "
 | |
|                     "deallocation, the process may crash", self);
 | |
|                 PyErr_WriteUnraisable(NULL);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (self->overlapped.hEvent != NULL)
 | |
|         CloseHandle(self->overlapped.hEvent);
 | |
| 
 | |
|     switch (self->type) {
 | |
|     case TYPE_READ:
 | |
|     case TYPE_ACCEPT:
 | |
|         Py_CLEAR(self->read_buffer);
 | |
|         break;
 | |
|     case TYPE_WRITE:
 | |
|         if (self->write_buffer.obj)
 | |
|             PyBuffer_Release(&self->write_buffer);
 | |
|         break;
 | |
|     }
 | |
|     PyObject_Del(self);
 | |
|     SetLastError(olderr);
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_cancel_doc,
 | |
|     "cancel() -> None\n\n"
 | |
|     "Cancel overlapped operation");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_cancel(OverlappedObject *self)
 | |
| {
 | |
|     BOOL ret = TRUE;
 | |
| 
 | |
|     if (self->type == TYPE_NOT_STARTED
 | |
|         || self->type == TYPE_WAIT_NAMED_PIPE_AND_CONNECT)
 | |
|         Py_RETURN_NONE;
 | |
| 
 | |
|     if (!HasOverlappedIoCompleted(&self->overlapped)) {
 | |
|         Py_BEGIN_ALLOW_THREADS
 | |
|         if (Py_CancelIoEx)
 | |
|             ret = Py_CancelIoEx(self->handle, &self->overlapped);
 | |
|         else
 | |
|             ret = CancelIo(self->handle);
 | |
|         Py_END_ALLOW_THREADS
 | |
|     }
 | |
| 
 | |
|     /* CancelIoEx returns ERROR_NOT_FOUND if the I/O completed in-between */
 | |
|     if (!ret && GetLastError() != ERROR_NOT_FOUND)
 | |
|         return SetFromWindowsErr(0);
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_getresult_doc,
 | |
|     "getresult(wait=False) -> result\n\n"
 | |
|     "Retrieve result of operation.  If wait is true then it blocks\n"
 | |
|     "until the operation is finished.  If wait is false and the\n"
 | |
|     "operation is still pending then an error is raised.");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_getresult(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     BOOL wait = FALSE;
 | |
|     DWORD transferred = 0;
 | |
|     BOOL ret;
 | |
|     DWORD err;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, "|" F_BOOL, &wait))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type == TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation not yet attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (self->type == TYPE_NOT_STARTED) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation failed to start");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = GetOverlappedResult(self->handle, &self->overlapped, &transferred,
 | |
|                               wait);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     self->error = err = ret ? ERROR_SUCCESS : GetLastError();
 | |
|     switch (err) {
 | |
|         case ERROR_SUCCESS:
 | |
|         case ERROR_MORE_DATA:
 | |
|             break;
 | |
|         case ERROR_BROKEN_PIPE:
 | |
|             if ((self->type == TYPE_READ || self->type == TYPE_ACCEPT) && self->read_buffer != NULL)
 | |
|                 break;
 | |
|             /* fall through */
 | |
|         default:
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
| 
 | |
|     switch (self->type) {
 | |
|         case TYPE_READ:
 | |
|             assert(PyBytes_CheckExact(self->read_buffer));
 | |
|             if (transferred != PyBytes_GET_SIZE(self->read_buffer) &&
 | |
|                 _PyBytes_Resize(&self->read_buffer, transferred))
 | |
|                 return NULL;
 | |
|             Py_INCREF(self->read_buffer);
 | |
|             return self->read_buffer;
 | |
|         default:
 | |
|             return PyLong_FromUnsignedLong((unsigned long) transferred);
 | |
|     }
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_ReadFile_doc,
 | |
|     "ReadFile(handle, size) -> Overlapped[message]\n\n"
 | |
|     "Start overlapped read");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_ReadFile(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE handle;
 | |
|     DWORD size;
 | |
|     DWORD nread;
 | |
|     PyObject *buf;
 | |
|     BOOL ret;
 | |
|     DWORD err;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE F_DWORD, &handle, &size))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type != TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation already attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
| #if SIZEOF_SIZE_T <= SIZEOF_LONG
 | |
|     size = Py_MIN(size, (DWORD)PY_SSIZE_T_MAX);
 | |
| #endif
 | |
|     buf = PyBytes_FromStringAndSize(NULL, Py_MAX(size, 1));
 | |
|     if (buf == NULL)
 | |
|         return NULL;
 | |
| 
 | |
|     self->type = TYPE_READ;
 | |
|     self->handle = handle;
 | |
|     self->read_buffer = buf;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = ReadFile(handle, PyBytes_AS_STRING(buf), size, &nread,
 | |
|                    &self->overlapped);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     self->error = err = ret ? ERROR_SUCCESS : GetLastError();
 | |
|     switch (err) {
 | |
|         case ERROR_BROKEN_PIPE:
 | |
|             mark_as_completed(&self->overlapped);
 | |
|             Py_RETURN_NONE;
 | |
|         case ERROR_SUCCESS:
 | |
|         case ERROR_MORE_DATA:
 | |
|         case ERROR_IO_PENDING:
 | |
|             Py_RETURN_NONE;
 | |
|         default:
 | |
|             self->type = TYPE_NOT_STARTED;
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_WSARecv_doc,
 | |
|     "RecvFile(handle, size, flags) -> Overlapped[message]\n\n"
 | |
|     "Start overlapped receive");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_WSARecv(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE handle;
 | |
|     DWORD size;
 | |
|     DWORD flags = 0;
 | |
|     DWORD nread;
 | |
|     PyObject *buf;
 | |
|     WSABUF wsabuf;
 | |
|     int ret;
 | |
|     DWORD err;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE F_DWORD "|" F_DWORD,
 | |
|                           &handle, &size, &flags))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type != TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation already attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
| #if SIZEOF_SIZE_T <= SIZEOF_LONG
 | |
|     size = Py_MIN(size, (DWORD)PY_SSIZE_T_MAX);
 | |
| #endif
 | |
|     buf = PyBytes_FromStringAndSize(NULL, Py_MAX(size, 1));
 | |
|     if (buf == NULL)
 | |
|         return NULL;
 | |
| 
 | |
|     self->type = TYPE_READ;
 | |
|     self->handle = handle;
 | |
|     self->read_buffer = buf;
 | |
|     wsabuf.len = size;
 | |
|     wsabuf.buf = PyBytes_AS_STRING(buf);
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = WSARecv((SOCKET)handle, &wsabuf, 1, &nread, &flags,
 | |
|                   &self->overlapped, NULL);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     self->error = err = (ret < 0 ? WSAGetLastError() : ERROR_SUCCESS);
 | |
|     switch (err) {
 | |
|         case ERROR_BROKEN_PIPE:
 | |
|             mark_as_completed(&self->overlapped);
 | |
|             Py_RETURN_NONE;
 | |
|         case ERROR_SUCCESS:
 | |
|         case ERROR_MORE_DATA:
 | |
|         case ERROR_IO_PENDING:
 | |
|             Py_RETURN_NONE;
 | |
|         default:
 | |
|             self->type = TYPE_NOT_STARTED;
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_WriteFile_doc,
 | |
|     "WriteFile(handle, buf) -> Overlapped[bytes_transferred]\n\n"
 | |
|     "Start overlapped write");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_WriteFile(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE handle;
 | |
|     PyObject *bufobj;
 | |
|     DWORD written;
 | |
|     BOOL ret;
 | |
|     DWORD err;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE "O", &handle, &bufobj))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type != TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation already attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!PyArg_Parse(bufobj, "y*", &self->write_buffer))
 | |
|         return NULL;
 | |
| 
 | |
| #if SIZEOF_SIZE_T > SIZEOF_LONG
 | |
|     if (self->write_buffer.len > (Py_ssize_t)ULONG_MAX) {
 | |
|         PyBuffer_Release(&self->write_buffer);
 | |
|         PyErr_SetString(PyExc_ValueError, "buffer to large");
 | |
|         return NULL;
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     self->type = TYPE_WRITE;
 | |
|     self->handle = handle;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = WriteFile(handle, self->write_buffer.buf,
 | |
|                     (DWORD)self->write_buffer.len,
 | |
|                     &written, &self->overlapped);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     self->error = err = ret ? ERROR_SUCCESS : GetLastError();
 | |
|     switch (err) {
 | |
|         case ERROR_SUCCESS:
 | |
|         case ERROR_IO_PENDING:
 | |
|             Py_RETURN_NONE;
 | |
|         default:
 | |
|             self->type = TYPE_NOT_STARTED;
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_WSASend_doc,
 | |
|     "WSASend(handle, buf, flags) -> Overlapped[bytes_transferred]\n\n"
 | |
|     "Start overlapped send");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_WSASend(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE handle;
 | |
|     PyObject *bufobj;
 | |
|     DWORD flags;
 | |
|     DWORD written;
 | |
|     WSABUF wsabuf;
 | |
|     int ret;
 | |
|     DWORD err;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE "O" F_DWORD,
 | |
|                           &handle, &bufobj, &flags))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type != TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation already attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!PyArg_Parse(bufobj, "y*", &self->write_buffer))
 | |
|         return NULL;
 | |
| 
 | |
| #if SIZEOF_SIZE_T > SIZEOF_LONG
 | |
|     if (self->write_buffer.len > (Py_ssize_t)ULONG_MAX) {
 | |
|         PyBuffer_Release(&self->write_buffer);
 | |
|         PyErr_SetString(PyExc_ValueError, "buffer to large");
 | |
|         return NULL;
 | |
|     }
 | |
| #endif
 | |
| 
 | |
|     self->type = TYPE_WRITE;
 | |
|     self->handle = handle;
 | |
|     wsabuf.len = (DWORD)self->write_buffer.len;
 | |
|     wsabuf.buf = self->write_buffer.buf;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = WSASend((SOCKET)handle, &wsabuf, 1, &written, flags,
 | |
|                   &self->overlapped, NULL);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     self->error = err = (ret < 0 ? WSAGetLastError() : ERROR_SUCCESS);
 | |
|     switch (err) {
 | |
|         case ERROR_SUCCESS:
 | |
|         case ERROR_IO_PENDING:
 | |
|             Py_RETURN_NONE;
 | |
|         default:
 | |
|             self->type = TYPE_NOT_STARTED;
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_AcceptEx_doc,
 | |
|     "AcceptEx(listen_handle, accept_handle) -> Overlapped[address_as_bytes]\n\n"
 | |
|     "Start overlapped wait for client to connect");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_AcceptEx(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     SOCKET ListenSocket;
 | |
|     SOCKET AcceptSocket;
 | |
|     DWORD BytesReceived;
 | |
|     DWORD size;
 | |
|     PyObject *buf;
 | |
|     BOOL ret;
 | |
|     DWORD err;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE F_HANDLE,
 | |
|                           &ListenSocket, &AcceptSocket))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type != TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation already attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     size = sizeof(struct sockaddr_in6) + 16;
 | |
|     buf = PyBytes_FromStringAndSize(NULL, size*2);
 | |
|     if (!buf)
 | |
|         return NULL;
 | |
| 
 | |
|     self->type = TYPE_ACCEPT;
 | |
|     self->handle = (HANDLE)ListenSocket;
 | |
|     self->read_buffer = buf;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = Py_AcceptEx(ListenSocket, AcceptSocket, PyBytes_AS_STRING(buf),
 | |
|                       0, size, size, &BytesReceived, &self->overlapped);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     self->error = err = ret ? ERROR_SUCCESS : WSAGetLastError();
 | |
|     switch (err) {
 | |
|         case ERROR_SUCCESS:
 | |
|         case ERROR_IO_PENDING:
 | |
|             Py_RETURN_NONE;
 | |
|         default:
 | |
|             self->type = TYPE_NOT_STARTED;
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| parse_address(PyObject *obj, SOCKADDR *Address, int Length)
 | |
| {
 | |
|     char *Host;
 | |
|     unsigned short Port;
 | |
|     unsigned long FlowInfo;
 | |
|     unsigned long ScopeId;
 | |
| 
 | |
|     memset(Address, 0, Length);
 | |
| 
 | |
|     if (PyArg_ParseTuple(obj, "sH", &Host, &Port))
 | |
|     {
 | |
|         Address->sa_family = AF_INET;
 | |
|         if (WSAStringToAddressA(Host, AF_INET, NULL, Address, &Length) < 0) {
 | |
|             SetFromWindowsErr(WSAGetLastError());
 | |
|             return -1;
 | |
|         }
 | |
|         ((SOCKADDR_IN*)Address)->sin_port = htons(Port);
 | |
|         return Length;
 | |
|     }
 | |
|     else if (PyArg_ParseTuple(obj, "sHkk", &Host, &Port, &FlowInfo, &ScopeId))
 | |
|     {
 | |
|         PyErr_Clear();
 | |
|         Address->sa_family = AF_INET6;
 | |
|         if (WSAStringToAddressA(Host, AF_INET6, NULL, Address, &Length) < 0) {
 | |
|             SetFromWindowsErr(WSAGetLastError());
 | |
|             return -1;
 | |
|         }
 | |
|         ((SOCKADDR_IN6*)Address)->sin6_port = htons(Port);
 | |
|         ((SOCKADDR_IN6*)Address)->sin6_flowinfo = FlowInfo;
 | |
|         ((SOCKADDR_IN6*)Address)->sin6_scope_id = ScopeId;
 | |
|         return Length;
 | |
|     }
 | |
| 
 | |
|     return -1;
 | |
| }
 | |
| 
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_ConnectEx_doc,
 | |
|     "ConnectEx(client_handle, address_as_bytes) -> Overlapped[None]\n\n"
 | |
|     "Start overlapped connect.  client_handle should be unbound.");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_ConnectEx(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     SOCKET ConnectSocket;
 | |
|     PyObject *AddressObj;
 | |
|     char AddressBuf[sizeof(struct sockaddr_in6)];
 | |
|     SOCKADDR *Address = (SOCKADDR*)AddressBuf;
 | |
|     int Length;
 | |
|     BOOL ret;
 | |
|     DWORD err;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE "O", &ConnectSocket, &AddressObj))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type != TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation already attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     Length = sizeof(AddressBuf);
 | |
|     Length = parse_address(AddressObj, Address, Length);
 | |
|     if (Length < 0)
 | |
|         return NULL;
 | |
| 
 | |
|     self->type = TYPE_CONNECT;
 | |
|     self->handle = (HANDLE)ConnectSocket;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = Py_ConnectEx(ConnectSocket, Address, Length,
 | |
|                        NULL, 0, NULL, &self->overlapped);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     self->error = err = ret ? ERROR_SUCCESS : WSAGetLastError();
 | |
|     switch (err) {
 | |
|         case ERROR_SUCCESS:
 | |
|         case ERROR_IO_PENDING:
 | |
|             Py_RETURN_NONE;
 | |
|         default:
 | |
|             self->type = TYPE_NOT_STARTED;
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_DisconnectEx_doc,
 | |
|     "DisconnectEx(handle, flags) -> Overlapped[None]\n\n"
 | |
|     "Start overlapped connect.  client_handle should be unbound.");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_DisconnectEx(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     SOCKET Socket;
 | |
|     DWORD flags;
 | |
|     BOOL ret;
 | |
|     DWORD err;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE F_DWORD, &Socket, &flags))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type != TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation already attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     self->type = TYPE_DISCONNECT;
 | |
|     self->handle = (HANDLE)Socket;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = Py_DisconnectEx(Socket, &self->overlapped, flags, 0);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     self->error = err = ret ? ERROR_SUCCESS : WSAGetLastError();
 | |
|     switch (err) {
 | |
|         case ERROR_SUCCESS:
 | |
|         case ERROR_IO_PENDING:
 | |
|             Py_RETURN_NONE;
 | |
|         default:
 | |
|             self->type = TYPE_NOT_STARTED;
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_ConnectNamedPipe_doc,
 | |
|     "ConnectNamedPipe(handle) -> Overlapped[None]\n\n"
 | |
|     "Start overlapped wait for a client to connect.");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_ConnectNamedPipe(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     HANDLE Pipe;
 | |
|     BOOL ret;
 | |
|     DWORD err;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, F_HANDLE, &Pipe))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type != TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation already attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     self->type = TYPE_CONNECT_NAMED_PIPE;
 | |
|     self->handle = Pipe;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = ConnectNamedPipe(Pipe, &self->overlapped);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     self->error = err = ret ? ERROR_SUCCESS : GetLastError();
 | |
|     switch (err) {
 | |
|         case ERROR_PIPE_CONNECTED:
 | |
|             mark_as_completed(&self->overlapped);
 | |
|             Py_RETURN_NONE;
 | |
|         case ERROR_SUCCESS:
 | |
|         case ERROR_IO_PENDING:
 | |
|             Py_RETURN_NONE;
 | |
|         default:
 | |
|             self->type = TYPE_NOT_STARTED;
 | |
|             return SetFromWindowsErr(err);
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* Unfortunately there is no way to do an overlapped connect to a
 | |
|    pipe.  We instead use WaitNamedPipe() and CreateFile() in a thread
 | |
|    pool thread.  If a connection succeeds within a time limit (10
 | |
|    seconds) then PostQueuedCompletionStatus() is used to return the
 | |
|    pipe handle to the completion port. */
 | |
| 
 | |
| static DWORD WINAPI
 | |
| WaitNamedPipeAndConnectInThread(WaitNamedPipeAndConnectContext *ctx)
 | |
| {
 | |
|     HANDLE PipeHandle = INVALID_HANDLE_VALUE;
 | |
|     DWORD Start = GetTickCount();
 | |
|     DWORD Deadline = Start + 10*1000;
 | |
|     DWORD Error = 0;
 | |
|     DWORD Timeout;
 | |
|     BOOL Success;
 | |
| 
 | |
|     for ( ; ; ) {
 | |
|         Timeout = Deadline - GetTickCount();
 | |
|         if ((int)Timeout < 0)
 | |
|             break;
 | |
|         Success = WaitNamedPipe(ctx->Address, Timeout);
 | |
|         Error = Success ? ERROR_SUCCESS : GetLastError();
 | |
|         switch (Error) {
 | |
|             case ERROR_SUCCESS:
 | |
|                 PipeHandle = CreateFile(ctx->Address,
 | |
|                                         GENERIC_READ | GENERIC_WRITE,
 | |
|                                         0, NULL, OPEN_EXISTING,
 | |
|                                         FILE_FLAG_OVERLAPPED, NULL);
 | |
|                 if (PipeHandle == INVALID_HANDLE_VALUE)
 | |
|                     continue;
 | |
|                 break;
 | |
|             case ERROR_SEM_TIMEOUT:
 | |
|                 continue;
 | |
|         }
 | |
|         break;
 | |
|     }
 | |
|     if (!PostQueuedCompletionStatus(ctx->IocpHandle, Error,
 | |
|                                     (ULONG_PTR)PipeHandle, ctx->Overlapped))
 | |
|         CloseHandle(PipeHandle);
 | |
|     free(ctx);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| PyDoc_STRVAR(
 | |
|     Overlapped_WaitNamedPipeAndConnect_doc,
 | |
|     "WaitNamedPipeAndConnect(addr, iocp_handle) -> Overlapped[pipe_handle]\n\n"
 | |
|     "Start overlapped connection to address, notifying iocp_handle when\n"
 | |
|     "finished");
 | |
| 
 | |
| static PyObject *
 | |
| Overlapped_WaitNamedPipeAndConnect(OverlappedObject *self, PyObject *args)
 | |
| {
 | |
|     char *Address;
 | |
|     Py_ssize_t AddressLength;
 | |
|     HANDLE IocpHandle;
 | |
|     OVERLAPPED Overlapped;
 | |
|     BOOL ret;
 | |
|     DWORD err;
 | |
|     WaitNamedPipeAndConnectContext *ctx;
 | |
|     Py_ssize_t ContextLength;
 | |
| 
 | |
|     if (!PyArg_ParseTuple(args, "s#" F_HANDLE F_POINTER,
 | |
|                           &Address, &AddressLength, &IocpHandle, &Overlapped))
 | |
|         return NULL;
 | |
| 
 | |
|     if (self->type != TYPE_NONE) {
 | |
|         PyErr_SetString(PyExc_ValueError, "operation already attempted");
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     ContextLength = (AddressLength +
 | |
|                      offsetof(WaitNamedPipeAndConnectContext, Address));
 | |
|     ctx = calloc(1, ContextLength + 1);
 | |
|     if (ctx == NULL)
 | |
|         return PyErr_NoMemory();
 | |
|     memcpy(ctx->Address, Address, AddressLength + 1);
 | |
|     ctx->Overlapped = &self->overlapped;
 | |
|     ctx->IocpHandle = IocpHandle;
 | |
| 
 | |
|     self->type = TYPE_WAIT_NAMED_PIPE_AND_CONNECT;
 | |
|     self->handle = NULL;
 | |
| 
 | |
|     Py_BEGIN_ALLOW_THREADS
 | |
|     ret = QueueUserWorkItem(WaitNamedPipeAndConnectInThread, ctx,
 | |
|                             WT_EXECUTELONGFUNCTION);
 | |
|     Py_END_ALLOW_THREADS
 | |
| 
 | |
|     mark_as_completed(&self->overlapped);
 | |
| 
 | |
|     self->error = err = ret ? ERROR_SUCCESS : GetLastError();
 | |
|     if (!ret)
 | |
|         return SetFromWindowsErr(err);
 | |
|     Py_RETURN_NONE;
 | |
| }
 | |
| 
 | |
| static PyObject*
 | |
| Overlapped_getaddress(OverlappedObject *self)
 | |
| {
 | |
|     return PyLong_FromVoidPtr(&self->overlapped);
 | |
| }
 | |
| 
 | |
| static PyObject*
 | |
| Overlapped_getpending(OverlappedObject *self)
 | |
| {
 | |
|     return PyBool_FromLong(!HasOverlappedIoCompleted(&self->overlapped) &&
 | |
|                            self->type != TYPE_NOT_STARTED);
 | |
| }
 | |
| 
 | |
| static PyMethodDef Overlapped_methods[] = {
 | |
|     {"getresult", (PyCFunction) Overlapped_getresult,
 | |
|      METH_VARARGS, Overlapped_getresult_doc},
 | |
|     {"cancel", (PyCFunction) Overlapped_cancel,
 | |
|      METH_NOARGS, Overlapped_cancel_doc},
 | |
|     {"ReadFile", (PyCFunction) Overlapped_ReadFile,
 | |
|      METH_VARARGS, Overlapped_ReadFile_doc},
 | |
|     {"WSARecv", (PyCFunction) Overlapped_WSARecv,
 | |
|      METH_VARARGS, Overlapped_WSARecv_doc},
 | |
|     {"WriteFile", (PyCFunction) Overlapped_WriteFile,
 | |
|      METH_VARARGS, Overlapped_WriteFile_doc},
 | |
|     {"WSASend", (PyCFunction) Overlapped_WSASend,
 | |
|      METH_VARARGS, Overlapped_WSASend_doc},
 | |
|     {"AcceptEx", (PyCFunction) Overlapped_AcceptEx,
 | |
|      METH_VARARGS, Overlapped_AcceptEx_doc},
 | |
|     {"ConnectEx", (PyCFunction) Overlapped_ConnectEx,
 | |
|      METH_VARARGS, Overlapped_ConnectEx_doc},
 | |
|     {"DisconnectEx", (PyCFunction) Overlapped_DisconnectEx,
 | |
|      METH_VARARGS, Overlapped_DisconnectEx_doc},
 | |
|     {"ConnectNamedPipe", (PyCFunction) Overlapped_ConnectNamedPipe,
 | |
|      METH_VARARGS, Overlapped_ConnectNamedPipe_doc},
 | |
|     {"WaitNamedPipeAndConnect",
 | |
|      (PyCFunction) Overlapped_WaitNamedPipeAndConnect,
 | |
|      METH_VARARGS, Overlapped_WaitNamedPipeAndConnect_doc},
 | |
|     {NULL}
 | |
| };
 | |
| 
 | |
| static PyMemberDef Overlapped_members[] = {
 | |
|     {"error", T_ULONG,
 | |
|      offsetof(OverlappedObject, error),
 | |
|      READONLY, "Error from last operation"},
 | |
|     {"event", T_HANDLE,
 | |
|      offsetof(OverlappedObject, overlapped) + offsetof(OVERLAPPED, hEvent),
 | |
|      READONLY, "Overlapped event handle"},
 | |
|     {NULL}
 | |
| };
 | |
| 
 | |
| static PyGetSetDef Overlapped_getsets[] = {
 | |
|     {"address", (getter)Overlapped_getaddress, NULL,
 | |
|      "Address of overlapped structure"},
 | |
|     {"pending", (getter)Overlapped_getpending, NULL,
 | |
|      "Whether the operation is pending"},
 | |
|     {NULL},
 | |
| };
 | |
| 
 | |
| PyTypeObject OverlappedType = {
 | |
|     PyVarObject_HEAD_INIT(NULL, 0)
 | |
|     /* tp_name           */ "_overlapped.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         */ Overlapped_getsets,
 | |
|     /* 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            */ Overlapped_new,
 | |
| };
 | |
| 
 | |
| static PyMethodDef overlapped_functions[] = {
 | |
|     {"CreateIoCompletionPort", overlapped_CreateIoCompletionPort,
 | |
|      METH_VARARGS, CreateIoCompletionPort_doc},
 | |
|     {"GetQueuedCompletionStatus", overlapped_GetQueuedCompletionStatus,
 | |
|      METH_VARARGS, GetQueuedCompletionStatus_doc},
 | |
|     {"PostQueuedCompletionStatus", overlapped_PostQueuedCompletionStatus,
 | |
|      METH_VARARGS, PostQueuedCompletionStatus_doc},
 | |
|     {"FormatMessage", overlapped_FormatMessage,
 | |
|      METH_VARARGS, FormatMessage_doc},
 | |
|     {"BindLocal", overlapped_BindLocal,
 | |
|      METH_VARARGS, BindLocal_doc},
 | |
|     {"RegisterWaitWithQueue", overlapped_RegisterWaitWithQueue,
 | |
|      METH_VARARGS, RegisterWaitWithQueue_doc},
 | |
|     {"UnregisterWait", overlapped_UnregisterWait,
 | |
|      METH_VARARGS, UnregisterWait_doc},
 | |
|     {"CreateEvent", overlapped_CreateEvent,
 | |
|      METH_VARARGS, CreateEvent_doc},
 | |
|     {"SetEvent", overlapped_SetEvent,
 | |
|      METH_VARARGS, SetEvent_doc},
 | |
|     {"ResetEvent", overlapped_ResetEvent,
 | |
|      METH_VARARGS, ResetEvent_doc},
 | |
|     {NULL}
 | |
| };
 | |
| 
 | |
| static struct PyModuleDef overlapped_module = {
 | |
|     PyModuleDef_HEAD_INIT,
 | |
|     "_overlapped",
 | |
|     NULL,
 | |
|     -1,
 | |
|     overlapped_functions,
 | |
|     NULL,
 | |
|     NULL,
 | |
|     NULL,
 | |
|     NULL
 | |
| };
 | |
| 
 | |
| #define WINAPI_CONSTANT(fmt, con) \
 | |
|     PyDict_SetItemString(d, #con, Py_BuildValue(fmt, con))
 | |
| 
 | |
| PyMODINIT_FUNC
 | |
| PyInit__overlapped(void)
 | |
| {
 | |
|     PyObject *m, *d;
 | |
| 
 | |
|     /* Ensure WSAStartup() called before initializing function pointers */
 | |
|     m = PyImport_ImportModule("_socket");
 | |
|     if (!m)
 | |
|         return NULL;
 | |
|     Py_DECREF(m);
 | |
| 
 | |
|     if (initialize_function_pointers() < 0)
 | |
|         return NULL;
 | |
| 
 | |
|     if (PyType_Ready(&OverlappedType) < 0)
 | |
|         return NULL;
 | |
| 
 | |
|     m = PyModule_Create(&overlapped_module);
 | |
|     if (PyModule_AddObject(m, "Overlapped", (PyObject *)&OverlappedType) < 0)
 | |
|         return NULL;
 | |
| 
 | |
|     d = PyModule_GetDict(m);
 | |
| 
 | |
|     WINAPI_CONSTANT(F_DWORD,  ERROR_IO_PENDING);
 | |
|     WINAPI_CONSTANT(F_DWORD,  ERROR_NETNAME_DELETED);
 | |
|     WINAPI_CONSTANT(F_DWORD,  ERROR_SEM_TIMEOUT);
 | |
|     WINAPI_CONSTANT(F_DWORD,  INFINITE);
 | |
|     WINAPI_CONSTANT(F_HANDLE, INVALID_HANDLE_VALUE);
 | |
|     WINAPI_CONSTANT(F_HANDLE, NULL);
 | |
|     WINAPI_CONSTANT(F_DWORD,  SO_UPDATE_ACCEPT_CONTEXT);
 | |
|     WINAPI_CONSTANT(F_DWORD,  SO_UPDATE_CONNECT_CONTEXT);
 | |
|     WINAPI_CONSTANT(F_DWORD,  TF_REUSE_SOCKET);
 | |
| 
 | |
|     return m;
 | |
| }
 |