1 module hunt.io.channel.iocp.Common; 2 3 4 // dfmt off 5 version (HAVE_IOCP) : 6 7 pragma(lib, "Ws2_32"); 8 // dfmt on 9 10 import hunt.io.ByteBuffer; 11 import hunt.io.channel.AbstractChannel; 12 import hunt.io.channel.Common; 13 import hunt.logging.ConsoleLogger; 14 import hunt.Functions; 15 16 import core.atomic; 17 import core.sys.windows.windows; 18 import core.sys.windows.winsock2; 19 import core.sys.windows.mswsock; 20 21 import std.conv; 22 import std.exception; 23 import std.format; 24 import std.process; 25 import std.socket; 26 import std.stdio; 27 28 29 /** 30 * 31 */ 32 mixin template CheckIocpError() { 33 void checkErro(int ret, int erro = 0) { 34 DWORD dwLastError = GetLastError(); 35 version (HUNT_IO_DEBUG) 36 infof("ret=%d, erro=%d, dwLastError=%d", ret, erro, dwLastError); 37 if (ret != erro || dwLastError == 0) 38 return; 39 40 if (ERROR_IO_PENDING != dwLastError) { // ERROR_IO_PENDING 41 import hunt.system.Error; 42 warningf("erro=%d, dwLastError=%d", erro, dwLastError); 43 this._error = true; 44 this._errorMessage = getErrorMessage(dwLastError); // format("IOCP error: code=%s", dwLastError); 45 } 46 } 47 } 48 49 enum IocpOperation { 50 accept, 51 connect, 52 read, 53 write, 54 event, 55 close 56 } 57 58 struct IocpContext { 59 OVERLAPPED overlapped; 60 IocpOperation operation; 61 AbstractChannel channel = null; 62 } 63 64 alias WSAOVERLAPPED = OVERLAPPED; 65 alias LPWSAOVERLAPPED = OVERLAPPED*; 66 67 __gshared static LPFN_ACCEPTEX AcceptEx; 68 __gshared static LPFN_CONNECTEX ConnectEx; 69 /*__gshared LPFN_DISCONNECTEX DisconnectEx; 70 __gshared LPFN_GETACCEPTEXSOCKADDRS GetAcceptexSockAddrs; 71 __gshared LPFN_TRANSMITFILE TransmitFile; 72 __gshared LPFN_TRANSMITPACKETS TransmitPackets; 73 __gshared LPFN_WSARECVMSG WSARecvMsg; 74 __gshared LPFN_WSASENDMSG WSASendMsg;*/ 75 76 shared static this() { 77 WSADATA wsaData; 78 int iResult = WSAStartup(MAKEWORD(2, 2), &wsaData); 79 if (iResult != 0) { 80 stderr.writeln("unable to load Winsock!"); 81 } 82 } 83 84 shared static ~this() { 85 WSACleanup(); 86 } 87 88 void loadWinsockExtension(SOCKET socket) { 89 if (isApiLoaded) 90 return; 91 isApiLoaded = true; 92 93 // SOCKET ListenSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); 94 // scope (exit) 95 // closesocket(ListenSocket); 96 GUID guid; 97 mixin(GET_FUNC_POINTER("WSAID_ACCEPTEX", "AcceptEx", socket.stringof)); 98 mixin(GET_FUNC_POINTER("WSAID_CONNECTEX", "ConnectEx")); 99 /* mixin(GET_FUNC_POINTER("WSAID_DISCONNECTEX", "DisconnectEx")); 100 mixin(GET_FUNC_POINTER("WSAID_GETACCEPTEXSOCKADDRS", "GetAcceptexSockAddrs")); 101 mixin(GET_FUNC_POINTER("WSAID_TRANSMITFILE", "TransmitFile")); 102 mixin(GET_FUNC_POINTER("WSAID_TRANSMITPACKETS", "TransmitPackets")); 103 mixin(GET_FUNC_POINTER("WSAID_WSARECVMSG", "WSARecvMsg"));*/ 104 } 105 106 private __gshared bool isApiLoaded = false; 107 108 private bool GetFunctionPointer(FuncPointer)(SOCKET sock, ref FuncPointer pfn, ref GUID guid) { 109 DWORD dwBytesReturned = 0; 110 if (WSAIoctl(sock, SIO_GET_EXTENSION_FUNCTION_POINTER, &guid, guid.sizeof, 111 &pfn, pfn.sizeof, &dwBytesReturned, null, null) == SOCKET_ERROR) { 112 error("Get function failed with error:", GetLastError()); 113 return false; 114 } 115 116 return true; 117 } 118 119 private string GET_FUNC_POINTER(string GuidValue, string pft, string socket = "socket") { 120 string str = " guid = " ~ GuidValue ~ ";"; 121 str ~= "if( !GetFunctionPointer( " ~ socket ~ ", " ~ pft 122 ~ ", guid ) ) { errnoEnforce(false,\"get function error!\"); } "; 123 return str; 124 } 125 126 enum : DWORD { 127 IOCPARAM_MASK = 0x7f, 128 IOC_VOID = 0x20000000, 129 IOC_OUT = 0x40000000, 130 IOC_IN = 0x80000000, 131 IOC_INOUT = IOC_IN | IOC_OUT 132 } 133 134 enum IOC_UNIX = 0x00000000; 135 enum IOC_WS2 = 0x08000000; 136 enum IOC_PROTOCOL = 0x10000000; 137 enum IOC_VENDOR = 0x18000000; 138 139 template _WSAIO(int x, int y) { 140 enum _WSAIO = IOC_VOID | x | y; 141 } 142 143 template _WSAIOR(int x, int y) { 144 enum _WSAIOR = IOC_OUT | x | y; 145 } 146 147 template _WSAIOW(int x, int y) { 148 enum _WSAIOW = IOC_IN | x | y; 149 } 150 151 template _WSAIORW(int x, int y) { 152 enum _WSAIORW = IOC_INOUT | x | y; 153 } 154 155 enum SIO_ASSOCIATE_HANDLE = _WSAIOW!(IOC_WS2, 1); 156 enum SIO_ENABLE_CIRCULAR_QUEUEING = _WSAIO!(IOC_WS2, 2); 157 enum SIO_FIND_ROUTE = _WSAIOR!(IOC_WS2, 3); 158 enum SIO_FLUSH = _WSAIO!(IOC_WS2, 4); 159 enum SIO_GET_BROADCAST_ADDRESS = _WSAIOR!(IOC_WS2, 5); 160 enum SIO_GET_EXTENSION_FUNCTION_POINTER = _WSAIORW!(IOC_WS2, 6); 161 enum SIO_GET_QOS = _WSAIORW!(IOC_WS2, 7); 162 enum SIO_GET_GROUP_QOS = _WSAIORW!(IOC_WS2, 8); 163 enum SIO_MULTIPOINT_LOOPBACK = _WSAIOW!(IOC_WS2, 9); 164 enum SIO_MULTICAST_SCOPE = _WSAIOW!(IOC_WS2, 10); 165 enum SIO_SET_QOS = _WSAIOW!(IOC_WS2, 11); 166 enum SIO_SET_GROUP_QOS = _WSAIOW!(IOC_WS2, 12); 167 enum SIO_TRANSLATE_HANDLE = _WSAIORW!(IOC_WS2, 13); 168 enum SIO_ROUTING_INTERFACE_QUERY = _WSAIORW!(IOC_WS2, 20); 169 enum SIO_ROUTING_INTERFACE_CHANGE = _WSAIOW!(IOC_WS2, 21); 170 enum SIO_ADDRESS_LIST_QUERY = _WSAIOR!(IOC_WS2, 22); 171 enum SIO_ADDRESS_LIST_CHANGE = _WSAIO!(IOC_WS2, 23); 172 enum SIO_QUERY_TARGET_PNP_HANDLE = _WSAIOR!(IOC_WS2, 24); 173 enum SIO_NSP_NOTIFY_CHANGE = _WSAIOW!(IOC_WS2, 25); 174 175 extern (Windows): 176 int WSARecv(SOCKET, LPWSABUF, DWORD, LPDWORD, LPDWORD, LPWSAOVERLAPPED, 177 LPWSAOVERLAPPED_COMPLETION_ROUTINE); 178 int WSARecvDisconnect(SOCKET, LPWSABUF); 179 int WSARecvFrom(SOCKET, LPWSABUF, DWORD, LPDWORD, LPDWORD, SOCKADDR*, LPINT, 180 LPWSAOVERLAPPED, LPWSAOVERLAPPED_COMPLETION_ROUTINE); 181 182 int WSASend(SOCKET, LPWSABUF, DWORD, LPDWORD, DWORD, LPWSAOVERLAPPED, 183 LPWSAOVERLAPPED_COMPLETION_ROUTINE); 184 int WSASendDisconnect(SOCKET, LPWSABUF); 185 int WSASendTo(SOCKET, LPWSABUF, DWORD, LPDWORD, DWORD, const(SOCKADDR)*, int, 186 LPWSAOVERLAPPED, LPWSAOVERLAPPED_COMPLETION_ROUTINE);