winlean.nim 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110
  1. #
  2. #
  3. # Nim's Runtime Library
  4. # (c) Copyright 2012 Andreas Rumpf
  5. #
  6. # See the file "copying.txt", included in this
  7. # distribution, for details about the copyright.
  8. #
  9. ## This module implements a small wrapper for some needed Win API procedures,
  10. ## so that the Nim compiler does not depend on the huge Windows module.
  11. {.deadCodeElim: on.} # dce option deprecated
  12. import dynlib
  13. {.passC: "-DWIN32_LEAN_AND_MEAN".}
  14. const
  15. useWinUnicode* = not defined(useWinAnsi)
  16. when useWinUnicode:
  17. type WinChar* = Utf16Char
  18. else:
  19. type WinChar* = char
  20. type
  21. Handle* = int
  22. LONG* = int32
  23. ULONG* = int32
  24. PULONG* = ptr int
  25. WINBOOL* = int32
  26. DWORD* = int32
  27. PDWORD* = ptr DWORD
  28. LPINT* = ptr int32
  29. ULONG_PTR* = uint
  30. PULONG_PTR* = ptr uint
  31. HDC* = Handle
  32. HGLRC* = Handle
  33. SECURITY_ATTRIBUTES* {.final, pure.} = object
  34. nLength*: int32
  35. lpSecurityDescriptor*: pointer
  36. bInheritHandle*: WINBOOL
  37. STARTUPINFO* {.final, pure.} = object
  38. cb*: int32
  39. lpReserved*: cstring
  40. lpDesktop*: cstring
  41. lpTitle*: cstring
  42. dwX*: int32
  43. dwY*: int32
  44. dwXSize*: int32
  45. dwYSize*: int32
  46. dwXCountChars*: int32
  47. dwYCountChars*: int32
  48. dwFillAttribute*: int32
  49. dwFlags*: int32
  50. wShowWindow*: int16
  51. cbReserved2*: int16
  52. lpReserved2*: pointer
  53. hStdInput*: Handle
  54. hStdOutput*: Handle
  55. hStdError*: Handle
  56. PROCESS_INFORMATION* {.final, pure.} = object
  57. hProcess*: Handle
  58. hThread*: Handle
  59. dwProcessId*: int32
  60. dwThreadId*: int32
  61. FILETIME* {.final, pure.} = object ## CANNOT BE int64 BECAUSE OF ALIGNMENT
  62. dwLowDateTime*: DWORD
  63. dwHighDateTime*: DWORD
  64. BY_HANDLE_FILE_INFORMATION* {.final, pure.} = object
  65. dwFileAttributes*: DWORD
  66. ftCreationTime*: FILETIME
  67. ftLastAccessTime*: FILETIME
  68. ftLastWriteTime*: FILETIME
  69. dwVolumeSerialNumber*: DWORD
  70. nFileSizeHigh*: DWORD
  71. nFileSizeLow*: DWORD
  72. nNumberOfLinks*: DWORD
  73. nFileIndexHigh*: DWORD
  74. nFileIndexLow*: DWORD
  75. OSVERSIONINFO* {.final, pure.} = object
  76. dwOSVersionInfoSize*: DWORD
  77. dwMajorVersion*: DWORD
  78. dwMinorVersion*: DWORD
  79. dwBuildNumber*: DWORD
  80. dwPlatformId*: DWORD
  81. szCSDVersion*: array[0..127, WinChar]
  82. const
  83. STARTF_USESHOWWINDOW* = 1'i32
  84. STARTF_USESTDHANDLES* = 256'i32
  85. HIGH_PRIORITY_CLASS* = 128'i32
  86. IDLE_PRIORITY_CLASS* = 64'i32
  87. NORMAL_PRIORITY_CLASS* = 32'i32
  88. REALTIME_PRIORITY_CLASS* = 256'i32
  89. WAIT_OBJECT_0* = 0'i32
  90. WAIT_TIMEOUT* = 0x00000102'i32
  91. WAIT_FAILED* = 0xFFFFFFFF'i32
  92. INFINITE* = -1'i32
  93. STILL_ACTIVE* = 0x00000103'i32
  94. STD_INPUT_HANDLE* = -10'i32
  95. STD_OUTPUT_HANDLE* = -11'i32
  96. STD_ERROR_HANDLE* = -12'i32
  97. DETACHED_PROCESS* = 8'i32
  98. SW_SHOWNORMAL* = 1'i32
  99. INVALID_HANDLE_VALUE* = Handle(-1)
  100. CREATE_UNICODE_ENVIRONMENT* = 1024'i32
  101. PIPE_ACCESS_DUPLEX* = 0x00000003'i32
  102. PIPE_ACCESS_INBOUND* = 1'i32
  103. PIPE_ACCESS_OUTBOUND* = 2'i32
  104. PIPE_NOWAIT* = 0x00000001'i32
  105. SYNCHRONIZE* = 0x00100000'i32
  106. CREATE_NO_WINDOW* = 0x08000000'i32
  107. when useWinUnicode:
  108. proc getVersionExW*(lpVersionInfo: ptr OSVERSIONINFO): WINBOOL {.stdcall, dynlib: "kernel32", importc: "GetVersionExW".}
  109. else:
  110. proc getVersionExA*(lpVersionInfo: ptr OSVERSIONINFO): WINBOOL {.stdcall, dynlib: "kernel32", importc: "GetVersionExA".}
  111. proc getVersion*(): DWORD {.stdcall, dynlib: "kernel32", importc: "GetVersion".}
  112. proc closeHandle*(hObject: Handle): WINBOOL {.stdcall, dynlib: "kernel32",
  113. importc: "CloseHandle".}
  114. proc readFile*(hFile: Handle, Buffer: pointer, nNumberOfBytesToRead: int32,
  115. lpNumberOfBytesRead: ptr int32, lpOverlapped: pointer): WINBOOL{.
  116. stdcall, dynlib: "kernel32", importc: "ReadFile".}
  117. proc writeFile*(hFile: Handle, Buffer: pointer, nNumberOfBytesToWrite: int32,
  118. lpNumberOfBytesWritten: ptr int32,
  119. lpOverlapped: pointer): WINBOOL{.
  120. stdcall, dynlib: "kernel32", importc: "WriteFile".}
  121. proc createPipe*(hReadPipe, hWritePipe: var Handle,
  122. lpPipeAttributes: var SECURITY_ATTRIBUTES,
  123. nSize: int32): WINBOOL{.
  124. stdcall, dynlib: "kernel32", importc: "CreatePipe".}
  125. proc createNamedPipe*(lpName: WideCString,
  126. dwOpenMode, dwPipeMode, nMaxInstances, nOutBufferSize,
  127. nInBufferSize, nDefaultTimeOut: int32,
  128. lpSecurityAttributes: ptr SECURITY_ATTRIBUTES): Handle {.
  129. stdcall, dynlib: "kernel32", importc: "CreateNamedPipeW".}
  130. proc peekNamedPipe*(hNamedPipe: Handle, lpBuffer: pointer=nil,
  131. nBufferSize: int32 = 0,
  132. lpBytesRead: ptr int32 = nil,
  133. lpTotalBytesAvail: ptr int32 = nil,
  134. lpBytesLeftThisMessage: ptr int32 = nil): bool {.
  135. stdcall, dynlib: "kernel32", importc: "PeekNamedPipe".}
  136. when useWinUnicode:
  137. proc createProcessW*(lpApplicationName, lpCommandLine: WideCString,
  138. lpProcessAttributes: ptr SECURITY_ATTRIBUTES,
  139. lpThreadAttributes: ptr SECURITY_ATTRIBUTES,
  140. bInheritHandles: WINBOOL, dwCreationFlags: int32,
  141. lpEnvironment, lpCurrentDirectory: WideCString,
  142. lpStartupInfo: var STARTUPINFO,
  143. lpProcessInformation: var PROCESS_INFORMATION): WINBOOL{.
  144. stdcall, dynlib: "kernel32", importc: "CreateProcessW".}
  145. else:
  146. proc createProcessA*(lpApplicationName, lpCommandLine: cstring,
  147. lpProcessAttributes: ptr SECURITY_ATTRIBUTES,
  148. lpThreadAttributes: ptr SECURITY_ATTRIBUTES,
  149. bInheritHandles: WINBOOL, dwCreationFlags: int32,
  150. lpEnvironment: pointer, lpCurrentDirectory: cstring,
  151. lpStartupInfo: var STARTUPINFO,
  152. lpProcessInformation: var PROCESS_INFORMATION): WINBOOL{.
  153. stdcall, dynlib: "kernel32", importc: "CreateProcessA".}
  154. proc suspendThread*(hThread: Handle): int32 {.stdcall, dynlib: "kernel32",
  155. importc: "SuspendThread".}
  156. proc resumeThread*(hThread: Handle): int32 {.stdcall, dynlib: "kernel32",
  157. importc: "ResumeThread".}
  158. proc waitForSingleObject*(hHandle: Handle, dwMilliseconds: int32): int32 {.
  159. stdcall, dynlib: "kernel32", importc: "WaitForSingleObject".}
  160. proc terminateProcess*(hProcess: Handle, uExitCode: int): WINBOOL {.stdcall,
  161. dynlib: "kernel32", importc: "TerminateProcess".}
  162. proc getExitCodeProcess*(hProcess: Handle, lpExitCode: var int32): WINBOOL {.
  163. stdcall, dynlib: "kernel32", importc: "GetExitCodeProcess".}
  164. proc getStdHandle*(nStdHandle: int32): Handle {.stdcall, dynlib: "kernel32",
  165. importc: "GetStdHandle".}
  166. proc setStdHandle*(nStdHandle: int32, hHandle: Handle): WINBOOL {.stdcall,
  167. dynlib: "kernel32", importc: "SetStdHandle".}
  168. proc flushFileBuffers*(hFile: Handle): WINBOOL {.stdcall, dynlib: "kernel32",
  169. importc: "FlushFileBuffers".}
  170. proc getLastError*(): int32 {.importc: "GetLastError",
  171. stdcall, dynlib: "kernel32".}
  172. proc setLastError*(error: int32) {.importc: "SetLastError",
  173. stdcall, dynlib: "kernel32".}
  174. when useWinUnicode:
  175. proc formatMessageW*(dwFlags: int32, lpSource: pointer,
  176. dwMessageId, dwLanguageId: int32,
  177. lpBuffer: pointer, nSize: int32,
  178. Arguments: pointer): int32 {.
  179. importc: "FormatMessageW", stdcall, dynlib: "kernel32".}
  180. else:
  181. proc formatMessageA*(dwFlags: int32, lpSource: pointer,
  182. dwMessageId, dwLanguageId: int32,
  183. lpBuffer: pointer, nSize: int32,
  184. Arguments: pointer): int32 {.
  185. importc: "FormatMessageA", stdcall, dynlib: "kernel32".}
  186. proc localFree*(p: pointer) {.
  187. importc: "LocalFree", stdcall, dynlib: "kernel32".}
  188. when useWinUnicode:
  189. proc getCurrentDirectoryW*(nBufferLength: int32,
  190. lpBuffer: WideCString): int32 {.
  191. importc: "GetCurrentDirectoryW", dynlib: "kernel32", stdcall.}
  192. proc setCurrentDirectoryW*(lpPathName: WideCString): int32 {.
  193. importc: "SetCurrentDirectoryW", dynlib: "kernel32", stdcall.}
  194. proc createDirectoryW*(pathName: WideCString, security: pointer=nil): int32 {.
  195. importc: "CreateDirectoryW", dynlib: "kernel32", stdcall.}
  196. proc removeDirectoryW*(lpPathName: WideCString): int32 {.
  197. importc: "RemoveDirectoryW", dynlib: "kernel32", stdcall.}
  198. proc setEnvironmentVariableW*(lpName, lpValue: WideCString): int32 {.
  199. stdcall, dynlib: "kernel32", importc: "SetEnvironmentVariableW".}
  200. proc getModuleFileNameW*(handle: Handle, buf: WideCString,
  201. size: int32): int32 {.importc: "GetModuleFileNameW",
  202. dynlib: "kernel32", stdcall.}
  203. else:
  204. proc getCurrentDirectoryA*(nBufferLength: int32, lpBuffer: cstring): int32 {.
  205. importc: "GetCurrentDirectoryA", dynlib: "kernel32", stdcall.}
  206. proc setCurrentDirectoryA*(lpPathName: cstring): int32 {.
  207. importc: "SetCurrentDirectoryA", dynlib: "kernel32", stdcall.}
  208. proc createDirectoryA*(pathName: cstring, security: pointer=nil): int32 {.
  209. importc: "CreateDirectoryA", dynlib: "kernel32", stdcall.}
  210. proc removeDirectoryA*(lpPathName: cstring): int32 {.
  211. importc: "RemoveDirectoryA", dynlib: "kernel32", stdcall.}
  212. proc setEnvironmentVariableA*(lpName, lpValue: cstring): int32 {.
  213. stdcall, dynlib: "kernel32", importc: "SetEnvironmentVariableA".}
  214. proc getModuleFileNameA*(handle: Handle, buf: cstring, size: int32): int32 {.
  215. importc: "GetModuleFileNameA", dynlib: "kernel32", stdcall.}
  216. when useWinUnicode:
  217. proc createSymbolicLinkW*(lpSymlinkFileName, lpTargetFileName: WideCString,
  218. flags: DWORD): int32 {.
  219. importc:"CreateSymbolicLinkW", dynlib: "kernel32", stdcall.}
  220. proc createHardLinkW*(lpFileName, lpExistingFileName: WideCString,
  221. security: pointer=nil): int32 {.
  222. importc:"CreateHardLinkW", dynlib: "kernel32", stdcall.}
  223. else:
  224. proc createSymbolicLinkA*(lpSymlinkFileName, lpTargetFileName: cstring,
  225. flags: DWORD): int32 {.
  226. importc:"CreateSymbolicLinkA", dynlib: "kernel32", stdcall.}
  227. proc createHardLinkA*(lpFileName, lpExistingFileName: cstring,
  228. security: pointer=nil): int32 {.
  229. importc:"CreateHardLinkA", dynlib: "kernel32", stdcall.}
  230. const
  231. FILE_ATTRIBUTE_READONLY* = 0x00000001'i32
  232. FILE_ATTRIBUTE_HIDDEN* = 0x00000002'i32
  233. FILE_ATTRIBUTE_SYSTEM* = 0x00000004'i32
  234. FILE_ATTRIBUTE_DIRECTORY* = 0x00000010'i32
  235. FILE_ATTRIBUTE_ARCHIVE* = 0x00000020'i32
  236. FILE_ATTRIBUTE_DEVICE* = 0x00000040'i32
  237. FILE_ATTRIBUTE_NORMAL* = 0x00000080'i32
  238. FILE_ATTRIBUTE_TEMPORARY* = 0x00000100'i32
  239. FILE_ATTRIBUTE_SPARSE_FILE* = 0x00000200'i32
  240. FILE_ATTRIBUTE_REPARSE_POINT* = 0x00000400'i32
  241. FILE_ATTRIBUTE_COMPRESSED* = 0x00000800'i32
  242. FILE_ATTRIBUTE_OFFLINE* = 0x00001000'i32
  243. FILE_ATTRIBUTE_NOT_CONTENT_INDEXED* = 0x00002000'i32
  244. FILE_FLAG_FIRST_PIPE_INSTANCE* = 0x00080000'i32
  245. FILE_FLAG_OPEN_NO_RECALL* = 0x00100000'i32
  246. FILE_FLAG_OPEN_REPARSE_POINT* = 0x00200000'i32
  247. FILE_FLAG_POSIX_SEMANTICS* = 0x01000000'i32
  248. FILE_FLAG_BACKUP_SEMANTICS* = 0x02000000'i32
  249. FILE_FLAG_DELETE_ON_CLOSE* = 0x04000000'i32
  250. FILE_FLAG_SEQUENTIAL_SCAN* = 0x08000000'i32
  251. FILE_FLAG_RANDOM_ACCESS* = 0x10000000'i32
  252. FILE_FLAG_NO_BUFFERING* = 0x20000000'i32
  253. FILE_FLAG_OVERLAPPED* = 0x40000000'i32
  254. FILE_FLAG_WRITE_THROUGH* = 0x80000000'i32
  255. MAX_PATH* = 260
  256. MOVEFILE_COPY_ALLOWED* = 0x2'i32
  257. MOVEFILE_CREATE_HARDLINK* = 0x10'i32
  258. MOVEFILE_DELAY_UNTIL_REBOOT* = 0x4'i32
  259. MOVEFILE_FAIL_IF_NOT_TRACKABLE* = 0x20'i32
  260. MOVEFILE_REPLACE_EXISTING* = 0x1'i32
  261. MOVEFILE_WRITE_THROUGH* = 0x8'i32
  262. type
  263. WIN32_FIND_DATA* {.pure.} = object
  264. dwFileAttributes*: int32
  265. ftCreationTime*: FILETIME
  266. ftLastAccessTime*: FILETIME
  267. ftLastWriteTime*: FILETIME
  268. nFileSizeHigh*: int32
  269. nFileSizeLow*: int32
  270. dwReserved0: int32
  271. dwReserved1: int32
  272. cFileName*: array[0..(MAX_PATH) - 1, WinChar]
  273. cAlternateFileName*: array[0..13, WinChar]
  274. when useWinUnicode:
  275. proc findFirstFileW*(lpFileName: WideCString,
  276. lpFindFileData: var WIN32_FIND_DATA): Handle {.
  277. stdcall, dynlib: "kernel32", importc: "FindFirstFileW".}
  278. proc findNextFileW*(hFindFile: Handle,
  279. lpFindFileData: var WIN32_FIND_DATA): int32 {.
  280. stdcall, dynlib: "kernel32", importc: "FindNextFileW".}
  281. else:
  282. proc findFirstFileA*(lpFileName: cstring,
  283. lpFindFileData: var WIN32_FIND_DATA): Handle {.
  284. stdcall, dynlib: "kernel32", importc: "FindFirstFileA".}
  285. proc findNextFileA*(hFindFile: Handle,
  286. lpFindFileData: var WIN32_FIND_DATA): int32 {.
  287. stdcall, dynlib: "kernel32", importc: "FindNextFileA".}
  288. proc findClose*(hFindFile: Handle) {.stdcall, dynlib: "kernel32",
  289. importc: "FindClose".}
  290. when useWinUnicode:
  291. proc getFullPathNameW*(lpFileName: WideCString, nBufferLength: int32,
  292. lpBuffer: WideCString,
  293. lpFilePart: var WideCString): int32 {.
  294. stdcall, dynlib: "kernel32",
  295. importc: "GetFullPathNameW".}
  296. proc getFileAttributesW*(lpFileName: WideCString): int32 {.
  297. stdcall, dynlib: "kernel32",
  298. importc: "GetFileAttributesW".}
  299. proc setFileAttributesW*(lpFileName: WideCString,
  300. dwFileAttributes: int32): WINBOOL {.
  301. stdcall, dynlib: "kernel32", importc: "SetFileAttributesW".}
  302. proc copyFileW*(lpExistingFileName, lpNewFileName: WideCString,
  303. bFailIfExists: WINBOOL): WINBOOL {.
  304. importc: "CopyFileW", stdcall, dynlib: "kernel32".}
  305. proc moveFileW*(lpExistingFileName, lpNewFileName: WideCString): WINBOOL {.
  306. importc: "MoveFileW", stdcall, dynlib: "kernel32".}
  307. proc moveFileExW*(lpExistingFileName, lpNewFileName: WideCString,
  308. flags: DWORD): WINBOOL {.
  309. importc: "MoveFileExW", stdcall, dynlib: "kernel32".}
  310. proc getEnvironmentStringsW*(): WideCString {.
  311. stdcall, dynlib: "kernel32", importc: "GetEnvironmentStringsW".}
  312. proc freeEnvironmentStringsW*(para1: WideCString): int32 {.
  313. stdcall, dynlib: "kernel32", importc: "FreeEnvironmentStringsW".}
  314. proc getCommandLineW*(): WideCString {.importc: "GetCommandLineW",
  315. stdcall, dynlib: "kernel32".}
  316. else:
  317. proc getFullPathNameA*(lpFileName: cstring, nBufferLength: int32,
  318. lpBuffer: cstring, lpFilePart: var cstring): int32 {.
  319. stdcall, dynlib: "kernel32",
  320. importc: "GetFullPathNameA".}
  321. proc getFileAttributesA*(lpFileName: cstring): int32 {.
  322. stdcall, dynlib: "kernel32",
  323. importc: "GetFileAttributesA".}
  324. proc setFileAttributesA*(lpFileName: cstring,
  325. dwFileAttributes: int32): WINBOOL {.
  326. stdcall, dynlib: "kernel32", importc: "SetFileAttributesA".}
  327. proc copyFileA*(lpExistingFileName, lpNewFileName: cstring,
  328. bFailIfExists: cint): cint {.
  329. importc: "CopyFileA", stdcall, dynlib: "kernel32".}
  330. proc moveFileA*(lpExistingFileName, lpNewFileName: cstring): WINBOOL {.
  331. importc: "MoveFileA", stdcall, dynlib: "kernel32".}
  332. proc moveFileExA*(lpExistingFileName, lpNewFileName: cstring,
  333. flags: DWORD): WINBOOL {.
  334. importc: "MoveFileExA", stdcall, dynlib: "kernel32".}
  335. proc getEnvironmentStringsA*(): cstring {.
  336. stdcall, dynlib: "kernel32", importc: "GetEnvironmentStringsA".}
  337. proc freeEnvironmentStringsA*(para1: cstring): int32 {.
  338. stdcall, dynlib: "kernel32", importc: "FreeEnvironmentStringsA".}
  339. proc getCommandLineA*(): cstring {.
  340. importc: "GetCommandLineA", stdcall, dynlib: "kernel32".}
  341. proc rdFileTime*(f: FILETIME): int64 =
  342. result = ze64(f.dwLowDateTime) or (ze64(f.dwHighDateTime) shl 32)
  343. proc rdFileSize*(f: WIN32_FIND_DATA): int64 =
  344. result = ze64(f.nFileSizeLow) or (ze64(f.nFileSizeHigh) shl 32)
  345. proc getSystemTimeAsFileTime*(lpSystemTimeAsFileTime: var FILETIME) {.
  346. importc: "GetSystemTimeAsFileTime", dynlib: "kernel32", stdcall.}
  347. proc sleep*(dwMilliseconds: int32){.stdcall, dynlib: "kernel32",
  348. importc: "Sleep".}
  349. when useWinUnicode:
  350. proc shellExecuteW*(HWND: Handle, lpOperation, lpFile,
  351. lpParameters, lpDirectory: WideCString,
  352. nShowCmd: int32): Handle{.
  353. stdcall, dynlib: "shell32.dll", importc: "ShellExecuteW".}
  354. else:
  355. proc shellExecuteA*(HWND: Handle, lpOperation, lpFile,
  356. lpParameters, lpDirectory: cstring,
  357. nShowCmd: int32): Handle{.
  358. stdcall, dynlib: "shell32.dll", importc: "ShellExecuteA".}
  359. proc getFileInformationByHandle*(hFile: Handle,
  360. lpFileInformation: ptr BY_HANDLE_FILE_INFORMATION): WINBOOL{.
  361. stdcall, dynlib: "kernel32", importc: "GetFileInformationByHandle".}
  362. const
  363. WSADESCRIPTION_LEN* = 256
  364. WSASYS_STATUS_LEN* = 128
  365. FD_SETSIZE* = 64
  366. MSG_PEEK* = 2
  367. INADDR_ANY* = 0'u32
  368. INADDR_LOOPBACK* = 0x7F000001
  369. INADDR_BROADCAST* = -1
  370. INADDR_NONE* = -1
  371. ws2dll = "Ws2_32.dll"
  372. proc wsaGetLastError*(): cint {.importc: "WSAGetLastError", dynlib: ws2dll.}
  373. type
  374. SocketHandle* = distinct int
  375. type
  376. WSAData* {.importc: "WSADATA", header: "winsock2.h".} = object
  377. wVersion, wHighVersion: int16
  378. szDescription: array[0..WSADESCRIPTION_LEN, char]
  379. szSystemStatus: array[0..WSASYS_STATUS_LEN, char]
  380. iMaxSockets, iMaxUdpDg: int16
  381. lpVendorInfo: cstring
  382. SockAddr* {.importc: "SOCKADDR", header: "winsock2.h".} = object
  383. sa_family*: uint16
  384. sa_data*: array[0..13, char]
  385. PSockAddr = ptr SockAddr
  386. InAddr* {.importc: "IN_ADDR", header: "winsock2.h".} = object
  387. s_addr*: uint32 # IP address
  388. Sockaddr_in* {.importc: "SOCKADDR_IN",
  389. header: "winsock2.h".} = object
  390. sin_family*: uint16
  391. sin_port*: uint16
  392. sin_addr*: InAddr
  393. sin_zero*: array[0..7, char]
  394. In6_addr* {.importc: "IN6_ADDR", header: "winsock2.h".} = object
  395. bytes* {.importc: "u.Byte".}: array[0..15, char]
  396. Sockaddr_in6* {.importc: "SOCKADDR_IN6",
  397. header: "ws2tcpip.h".} = object
  398. sin6_family*: uint16
  399. sin6_port*: uint16
  400. sin6_flowinfo*: int32 # unsigned
  401. sin6_addr*: In6_addr
  402. sin6_scope_id*: int32 # unsigned
  403. Sockaddr_in6_old* = object
  404. sin6_family*: uint16
  405. sin6_port*: int16 # unsigned
  406. sin6_flowinfo*: int32 # unsigned
  407. sin6_addr*: In6_addr
  408. Sockaddr_storage* {.importc: "SOCKADDR_STORAGE",
  409. header: "winsock2.h".} = object
  410. ss_family*: uint16
  411. ss_pad1: array[6, byte]
  412. ss_align: int64
  413. ss_pad2: array[112, byte]
  414. Servent* = object
  415. s_name*: cstring
  416. s_aliases*: cstringArray
  417. when defined(cpu64):
  418. s_proto*: cstring
  419. s_port*: int16
  420. else:
  421. s_port*: int16
  422. s_proto*: cstring
  423. Hostent* = object
  424. h_name*: cstring
  425. h_aliases*: cstringArray
  426. h_addrtype*: int16
  427. h_length*: int16
  428. h_addr_list*: cstringArray
  429. TFdSet* = object
  430. fd_count*: cint # unsigned
  431. fd_array*: array[0..FD_SETSIZE-1, SocketHandle]
  432. AddrInfo* = object
  433. ai_flags*: cint ## Input flags.
  434. ai_family*: cint ## Address family of socket.
  435. ai_socktype*: cint ## Socket type.
  436. ai_protocol*: cint ## Protocol of socket.
  437. ai_addrlen*: csize ## Length of socket address.
  438. ai_canonname*: cstring ## Canonical name of service location.
  439. ai_addr*: ptr SockAddr ## Socket address of socket.
  440. ai_next*: ptr AddrInfo ## Pointer to next in list.
  441. SockLen* = cuint
  442. when defined(cpp):
  443. type
  444. Timeval* {.importc: "timeval", header: "<time.h>".} = object
  445. tv_sec*, tv_usec*: int32
  446. else:
  447. type
  448. Timeval* = object
  449. tv_sec*, tv_usec*: int32
  450. var
  451. SOMAXCONN* {.importc, header: "winsock2.h".}: cint
  452. INVALID_SOCKET* {.importc, header: "winsock2.h".}: SocketHandle
  453. SOL_SOCKET* {.importc, header: "winsock2.h".}: cint
  454. SO_DEBUG* {.importc, header: "winsock2.h".}: cint ## turn on debugging info recording
  455. SO_ACCEPTCONN* {.importc, header: "winsock2.h".}: cint # socket has had listen()
  456. SO_REUSEADDR* {.importc, header: "winsock2.h".}: cint # allow local address reuse
  457. SO_REUSEPORT* {.importc: "SO_REUSEADDR", header: "winsock2.h".}: cint # allow port reuse. Since Windows does not really support it, mapped to SO_REUSEADDR. This shouldn't cause problems.
  458. SO_KEEPALIVE* {.importc, header: "winsock2.h".}: cint # keep connections alive
  459. SO_DONTROUTE* {.importc, header: "winsock2.h".}: cint # just use interface addresses
  460. SO_BROADCAST* {.importc, header: "winsock2.h".}: cint # permit sending of broadcast msgs
  461. SO_USELOOPBACK* {.importc, header: "winsock2.h".}: cint # bypass hardware when possible
  462. SO_LINGER* {.importc, header: "winsock2.h".}: cint # linger on close if data present
  463. SO_OOBINLINE* {.importc, header: "winsock2.h".}: cint # leave received OOB data in line
  464. SO_DONTLINGER* {.importc, header: "winsock2.h".}: cint
  465. SO_EXCLUSIVEADDRUSE* {.importc, header: "winsock2.h".}: cint # disallow local address reuse
  466. SO_ERROR* {.importc, header: "winsock2.h".}: cint
  467. TCP_NODELAY* {.importc, header: "winsock2.h".}: cint
  468. proc `==`*(x, y: SocketHandle): bool {.borrow.}
  469. proc getservbyname*(name, proto: cstring): ptr Servent {.
  470. stdcall, importc: "getservbyname", dynlib: ws2dll.}
  471. proc getservbyport*(port: cint, proto: cstring): ptr Servent {.
  472. stdcall, importc: "getservbyport", dynlib: ws2dll.}
  473. proc gethostbyaddr*(ip: ptr InAddr, len: cuint, theType: cint): ptr Hostent {.
  474. stdcall, importc: "gethostbyaddr", dynlib: ws2dll.}
  475. proc gethostbyname*(name: cstring): ptr Hostent {.
  476. stdcall, importc: "gethostbyname", dynlib: ws2dll.}
  477. proc gethostname*(hostname: cstring, len: cint): cint {.
  478. stdcall, importc: "gethostname", dynlib: ws2dll.}
  479. proc socket*(af, typ, protocol: cint): SocketHandle {.
  480. stdcall, importc: "socket", dynlib: ws2dll.}
  481. proc closesocket*(s: SocketHandle): cint {.
  482. stdcall, importc: "closesocket", dynlib: ws2dll.}
  483. proc accept*(s: SocketHandle, a: ptr SockAddr, addrlen: ptr SockLen): SocketHandle {.
  484. stdcall, importc: "accept", dynlib: ws2dll.}
  485. proc bindSocket*(s: SocketHandle, name: ptr SockAddr, namelen: SockLen): cint {.
  486. stdcall, importc: "bind", dynlib: ws2dll.}
  487. proc connect*(s: SocketHandle, name: ptr SockAddr, namelen: SockLen): cint {.
  488. stdcall, importc: "connect", dynlib: ws2dll.}
  489. proc getsockname*(s: SocketHandle, name: ptr SockAddr,
  490. namelen: ptr SockLen): cint {.
  491. stdcall, importc: "getsockname", dynlib: ws2dll.}
  492. proc getpeername*(s: SocketHandle, name: ptr SockAddr,
  493. namelen: ptr SockLen): cint {.
  494. stdcall, importc, dynlib: ws2dll.}
  495. proc getsockopt*(s: SocketHandle, level, optname: cint, optval: pointer,
  496. optlen: ptr SockLen): cint {.
  497. stdcall, importc: "getsockopt", dynlib: ws2dll.}
  498. proc setsockopt*(s: SocketHandle, level, optname: cint, optval: pointer,
  499. optlen: SockLen): cint {.
  500. stdcall, importc: "setsockopt", dynlib: ws2dll.}
  501. proc listen*(s: SocketHandle, backlog: cint): cint {.
  502. stdcall, importc: "listen", dynlib: ws2dll.}
  503. proc recv*(s: SocketHandle, buf: pointer, len, flags: cint): cint {.
  504. stdcall, importc: "recv", dynlib: ws2dll.}
  505. proc recvfrom*(s: SocketHandle, buf: cstring, len, flags: cint,
  506. fromm: ptr SockAddr, fromlen: ptr SockLen): cint {.
  507. stdcall, importc: "recvfrom", dynlib: ws2dll.}
  508. proc select*(nfds: cint, readfds, writefds, exceptfds: ptr TFdSet,
  509. timeout: ptr Timeval): cint {.
  510. stdcall, importc: "select", dynlib: ws2dll.}
  511. proc send*(s: SocketHandle, buf: pointer, len, flags: cint): cint {.
  512. stdcall, importc: "send", dynlib: ws2dll.}
  513. proc sendto*(s: SocketHandle, buf: pointer, len, flags: cint,
  514. to: ptr SockAddr, tolen: SockLen): cint {.
  515. stdcall, importc: "sendto", dynlib: ws2dll.}
  516. proc shutdown*(s: SocketHandle, how: cint): cint {.
  517. stdcall, importc: "shutdown", dynlib: ws2dll.}
  518. proc getnameinfo*(a1: ptr SockAddr, a2: SockLen,
  519. a3: cstring, a4: SockLen, a5: cstring,
  520. a6: SockLen, a7: cint): cint {.
  521. stdcall, importc: "getnameinfo", dynlib: ws2dll.}
  522. proc inet_addr*(cp: cstring): uint32 {.
  523. stdcall, importc: "inet_addr", dynlib: ws2dll.}
  524. proc WSAFDIsSet(s: SocketHandle, set: var TFdSet): bool {.
  525. stdcall, importc: "__WSAFDIsSet", dynlib: ws2dll, noSideEffect.}
  526. proc FD_ISSET*(socket: SocketHandle, set: var TFdSet): cint =
  527. result = if WSAFDIsSet(socket, set): 1'i32 else: 0'i32
  528. proc FD_SET*(socket: SocketHandle, s: var TFdSet) =
  529. if s.fd_count < FD_SETSIZE:
  530. s.fd_array[int(s.fd_count)] = socket
  531. inc(s.fd_count)
  532. proc FD_ZERO*(s: var TFdSet) =
  533. s.fd_count = 0
  534. proc wsaStartup*(wVersionRequired: int16, WSData: ptr WSAData): cint {.
  535. stdcall, importc: "WSAStartup", dynlib: ws2dll.}
  536. proc getaddrinfo*(nodename, servname: cstring, hints: ptr AddrInfo,
  537. res: var ptr AddrInfo): cint {.
  538. stdcall, importc: "getaddrinfo", dynlib: ws2dll.}
  539. proc freeaddrinfo*(ai: ptr AddrInfo) {.
  540. stdcall, importc: "freeaddrinfo", dynlib: ws2dll.}
  541. proc inet_ntoa*(i: InAddr): cstring {.
  542. stdcall, importc, dynlib: ws2dll.}
  543. const
  544. MAXIMUM_WAIT_OBJECTS* = 0x00000040
  545. type
  546. WOHandleArray* = array[0..MAXIMUM_WAIT_OBJECTS - 1, Handle]
  547. PWOHandleArray* = ptr WOHandleArray
  548. proc waitForMultipleObjects*(nCount: DWORD, lpHandles: PWOHandleArray,
  549. bWaitAll: WINBOOL, dwMilliseconds: DWORD): DWORD{.
  550. stdcall, dynlib: "kernel32", importc: "WaitForMultipleObjects".}
  551. # for memfiles.nim:
  552. const
  553. GENERIC_READ* = 0x80000000'i32
  554. GENERIC_WRITE* = 0x40000000'i32
  555. GENERIC_ALL* = 0x10000000'i32
  556. FILE_SHARE_READ* = 1'i32
  557. FILE_SHARE_DELETE* = 4'i32
  558. FILE_SHARE_WRITE* = 2'i32
  559. CREATE_ALWAYS* = 2'i32
  560. CREATE_NEW* = 1'i32
  561. OPEN_EXISTING* = 3'i32
  562. OPEN_ALWAYS* = 4'i32
  563. FILE_BEGIN* = 0'i32
  564. INVALID_SET_FILE_POINTER* = -1'i32
  565. NO_ERROR* = 0'i32
  566. PAGE_NOACCESS* = 0x01'i32
  567. PAGE_EXECUTE* = 0x10'i32
  568. PAGE_EXECUTE_READ* = 0x20'i32
  569. PAGE_EXECUTE_READWRITE* = 0x40'i32
  570. PAGE_READONLY* = 2'i32
  571. PAGE_READWRITE* = 4'i32
  572. FILE_MAP_READ* = 4'i32
  573. FILE_MAP_WRITE* = 2'i32
  574. INVALID_FILE_SIZE* = -1'i32
  575. DUPLICATE_SAME_ACCESS* = 2
  576. FILE_READ_DATA* = 0x00000001 # file & pipe
  577. FILE_WRITE_DATA* = 0x00000002 # file & pipe
  578. # Error Constants
  579. const
  580. ERROR_FILE_NOT_FOUND* = 2
  581. ERROR_PATH_NOT_FOUND* = 3
  582. ERROR_ACCESS_DENIED* = 5
  583. ERROR_NO_MORE_FILES* = 18
  584. ERROR_LOCK_VIOLATION* = 33
  585. ERROR_HANDLE_EOF* = 38
  586. ERROR_BAD_ARGUMENTS* = 165
  587. proc duplicateHandle*(hSourceProcessHandle: HANDLE, hSourceHandle: HANDLE,
  588. hTargetProcessHandle: HANDLE,
  589. lpTargetHandle: ptr HANDLE,
  590. dwDesiredAccess: DWORD, bInheritHandle: WINBOOL,
  591. dwOptions: DWORD): WINBOOL{.stdcall, dynlib: "kernel32",
  592. importc: "DuplicateHandle".}
  593. proc setHandleInformation*(hObject: HANDLE, dwMask: DWORD,
  594. dwFlags: DWORD): WINBOOL {.stdcall,
  595. dynlib: "kernel32", importc: "SetHandleInformation".}
  596. proc getCurrentProcess*(): HANDLE{.stdcall, dynlib: "kernel32",
  597. importc: "GetCurrentProcess".}
  598. when useWinUnicode:
  599. proc createFileW*(lpFileName: WideCString, dwDesiredAccess, dwShareMode: DWORD,
  600. lpSecurityAttributes: pointer,
  601. dwCreationDisposition, dwFlagsAndAttributes: DWORD,
  602. hTemplateFile: Handle): Handle {.
  603. stdcall, dynlib: "kernel32", importc: "CreateFileW".}
  604. proc deleteFileW*(pathName: WideCString): int32 {.
  605. importc: "DeleteFileW", dynlib: "kernel32", stdcall.}
  606. else:
  607. proc createFileA*(lpFileName: cstring, dwDesiredAccess, dwShareMode: DWORD,
  608. lpSecurityAttributes: pointer,
  609. dwCreationDisposition, dwFlagsAndAttributes: DWORD,
  610. hTemplateFile: Handle): Handle {.
  611. stdcall, dynlib: "kernel32", importc: "CreateFileA".}
  612. proc deleteFileA*(pathName: cstring): int32 {.
  613. importc: "DeleteFileA", dynlib: "kernel32", stdcall.}
  614. proc setEndOfFile*(hFile: Handle): WINBOOL {.stdcall, dynlib: "kernel32",
  615. importc: "SetEndOfFile".}
  616. proc setFilePointer*(hFile: Handle, lDistanceToMove: LONG,
  617. lpDistanceToMoveHigh: ptr LONG,
  618. dwMoveMethod: DWORD): DWORD {.
  619. stdcall, dynlib: "kernel32", importc: "SetFilePointer".}
  620. proc getFileSize*(hFile: Handle, lpFileSizeHigh: ptr DWORD): DWORD{.stdcall,
  621. dynlib: "kernel32", importc: "GetFileSize".}
  622. proc mapViewOfFileEx*(hFileMappingObject: Handle, dwDesiredAccess: DWORD,
  623. dwFileOffsetHigh, dwFileOffsetLow: DWORD,
  624. dwNumberOfBytesToMap: DWORD,
  625. lpBaseAddress: pointer): pointer{.
  626. stdcall, dynlib: "kernel32", importc: "MapViewOfFileEx".}
  627. proc createFileMappingW*(hFile: Handle,
  628. lpFileMappingAttributes: pointer,
  629. flProtect, dwMaximumSizeHigh: DWORD,
  630. dwMaximumSizeLow: DWORD,
  631. lpName: pointer): Handle {.
  632. stdcall, dynlib: "kernel32", importc: "CreateFileMappingW".}
  633. when not useWinUnicode:
  634. proc createFileMappingA*(hFile: Handle,
  635. lpFileMappingAttributes: pointer,
  636. flProtect, dwMaximumSizeHigh: DWORD,
  637. dwMaximumSizeLow: DWORD, lpName: cstring): Handle {.
  638. stdcall, dynlib: "kernel32", importc: "CreateFileMappingA".}
  639. proc unmapViewOfFile*(lpBaseAddress: pointer): WINBOOL {.stdcall,
  640. dynlib: "kernel32", importc: "UnmapViewOfFile".}
  641. proc flushViewOfFile*(lpBaseAddress: pointer, dwNumberOfBytesToFlush: DWORD): WINBOOL {.
  642. stdcall, dynlib: "kernel32", importc: "FlushViewOfFile".}
  643. type
  644. OVERLAPPED* {.pure, inheritable.} = object
  645. internal*: PULONG
  646. internalHigh*: PULONG
  647. offset*: DWORD
  648. offsetHigh*: DWORD
  649. hEvent*: Handle
  650. POVERLAPPED* = ptr OVERLAPPED
  651. POVERLAPPED_COMPLETION_ROUTINE* = proc (para1: DWORD, para2: DWORD,
  652. para3: POVERLAPPED){.stdcall.}
  653. GUID* {.final, pure.} = object
  654. D1*: int32
  655. D2*: int16
  656. D3*: int16
  657. D4*: array[0..7, int8]
  658. const
  659. ERROR_IO_PENDING* = 997 # a.k.a WSA_IO_PENDING
  660. WSAECONNABORTED* = 10053
  661. WSAEADDRINUSE* = 10048
  662. WSAECONNRESET* = 10054
  663. WSAEDISCON* = 10101
  664. WSAENETRESET* = 10052
  665. WSAETIMEDOUT* = 10060
  666. WSANOTINITIALISED* = 10093
  667. WSAENOTSOCK* = 10038
  668. WSAEINPROGRESS* = 10036
  669. WSAEINTR* = 10004
  670. WSAEWOULDBLOCK* = 10035
  671. ERROR_NETNAME_DELETED* = 64
  672. STATUS_PENDING* = 0x103
  673. proc createIoCompletionPort*(FileHandle: Handle, ExistingCompletionPort: Handle,
  674. CompletionKey: ULONG_PTR,
  675. NumberOfConcurrentThreads: DWORD): Handle{.stdcall,
  676. dynlib: "kernel32", importc: "CreateIoCompletionPort".}
  677. proc getQueuedCompletionStatus*(CompletionPort: Handle,
  678. lpNumberOfBytesTransferred: PDWORD, lpCompletionKey: PULONG_PTR,
  679. lpOverlapped: ptr POVERLAPPED,
  680. dwMilliseconds: DWORD): WINBOOL{.stdcall,
  681. dynlib: "kernel32", importc: "GetQueuedCompletionStatus".}
  682. proc getOverlappedResult*(hFile: Handle, lpOverlapped: POVERLAPPED,
  683. lpNumberOfBytesTransferred: var DWORD, bWait: WINBOOL): WINBOOL{.
  684. stdcall, dynlib: "kernel32", importc: "GetOverlappedResult".}
  685. # this is copy of HasOverlappedIoCompleted() macro from <winbase.h>
  686. # because we have declared own OVERLAPPED structure with member names not
  687. # compatible with original names.
  688. template hasOverlappedIoCompleted*(lpOverlapped): bool =
  689. (cast[uint](lpOverlapped.internal) != STATUS_PENDING)
  690. const
  691. IOC_OUT* = 0x40000000'i32
  692. IOC_IN* = 0x80000000'i32
  693. IOC_WS2* = 0x08000000'i32
  694. IOC_INOUT* = IOC_IN or IOC_OUT
  695. template WSAIORW*(x,y): untyped = (IOC_INOUT or x or y)
  696. const
  697. SIO_GET_EXTENSION_FUNCTION_POINTER* = WSAIORW(IOC_WS2,6).DWORD
  698. SO_UPDATE_ACCEPT_CONTEXT* = 0x700B
  699. AI_V4MAPPED* = 0x0008
  700. AF_UNSPEC* = 0
  701. AF_INET* = 2
  702. AF_INET6* = 23
  703. var
  704. WSAID_CONNECTEX*: GUID = GUID(D1: 0x25a207b9, D2: 0xddf3'i16, D3: 0x4660, D4: [
  705. 0x8e'i8, 0xe9'i8, 0x76'i8, 0xe5'i8, 0x8c'i8, 0x74'i8, 0x06'i8, 0x3e'i8])
  706. WSAID_ACCEPTEX*: GUID = GUID(D1: 0xb5367df1'i32, D2: 0xcbac'i16, D3: 0x11cf, D4: [
  707. 0x95'i8, 0xca'i8, 0x00'i8, 0x80'i8, 0x5f'i8, 0x48'i8, 0xa1'i8, 0x92'i8])
  708. WSAID_GETACCEPTEXSOCKADDRS*: GUID = GUID(D1: 0xb5367df2'i32, D2: 0xcbac'i16, D3: 0x11cf, D4: [
  709. 0x95'i8, 0xca'i8, 0x00'i8, 0x80'i8, 0x5f'i8, 0x48'i8, 0xa1'i8, 0x92'i8])
  710. proc WSAIoctl*(s: SocketHandle, dwIoControlCode: DWORD, lpvInBuffer: pointer,
  711. cbInBuffer: DWORD, lpvOutBuffer: pointer, cbOutBuffer: DWORD,
  712. lpcbBytesReturned: PDWORD, lpOverlapped: POVERLAPPED,
  713. lpCompletionRoutine: POVERLAPPED_COMPLETION_ROUTINE): cint
  714. {.stdcall, importc: "WSAIoctl", dynlib: "Ws2_32.dll".}
  715. type
  716. TWSABuf* {.importc: "WSABUF", header: "winsock2.h".} = object
  717. len*: ULONG
  718. buf*: cstring
  719. proc WSARecv*(s: SocketHandle, buf: ptr TWSABuf, bufCount: DWORD,
  720. bytesReceived, flags: PDWORD, lpOverlapped: POVERLAPPED,
  721. completionProc: POVERLAPPED_COMPLETION_ROUTINE): cint {.
  722. stdcall, importc: "WSARecv", dynlib: "Ws2_32.dll".}
  723. proc WSARecvFrom*(s: SocketHandle, buf: ptr TWSABuf, bufCount: DWORD,
  724. bytesReceived: PDWORD, flags: PDWORD, name: ptr SockAddr,
  725. namelen: ptr cint, lpOverlapped: POVERLAPPED,
  726. completionProc: POVERLAPPED_COMPLETION_ROUTINE): cint {.
  727. stdcall, importc: "WSARecvFrom", dynlib: "Ws2_32.dll".}
  728. proc WSASend*(s: SocketHandle, buf: ptr TWSABuf, bufCount: DWORD,
  729. bytesSent: PDWORD, flags: DWORD, lpOverlapped: POVERLAPPED,
  730. completionProc: POVERLAPPED_COMPLETION_ROUTINE): cint {.
  731. stdcall, importc: "WSASend", dynlib: "Ws2_32.dll".}
  732. proc WSASendTo*(s: SocketHandle, buf: ptr TWSABuf, bufCount: DWORD,
  733. bytesSent: PDWORD, flags: DWORD, name: ptr SockAddr,
  734. namelen: cint, lpOverlapped: POVERLAPPED,
  735. completionProc: POVERLAPPED_COMPLETION_ROUTINE): cint {.
  736. stdcall, importc: "WSASendTo", dynlib: "Ws2_32.dll".}
  737. proc get_osfhandle*(fd:FileHandle): Handle {.
  738. importc: "_get_osfhandle", header:"<io.h>".}
  739. proc getSystemTimes*(lpIdleTime, lpKernelTime,
  740. lpUserTime: var FILETIME): WINBOOL {.stdcall,
  741. dynlib: "kernel32", importc: "GetSystemTimes".}
  742. proc getProcessTimes*(hProcess: Handle; lpCreationTime, lpExitTime,
  743. lpKernelTime, lpUserTime: var FILETIME): WINBOOL {.stdcall,
  744. dynlib: "kernel32", importc: "GetProcessTimes".}
  745. type inet_ntop_proc = proc(family: cint, paddr: pointer, pStringBuffer: cstring,
  746. stringBufSize: int32): cstring {.gcsafe, stdcall, tags: [].}
  747. var inet_ntop_real: inet_ntop_proc = nil
  748. let ws2 = loadLib(ws2dll)
  749. if ws2 != nil:
  750. inet_ntop_real = cast[inet_ntop_proc](symAddr(ws2, "inet_ntop"))
  751. proc WSAAddressToStringA(pAddr: ptr SockAddr, addrSize: DWORD, unused: pointer, pBuff: cstring, pBuffSize: ptr DWORD): cint {.stdcall, importc, dynlib: ws2dll.}
  752. proc inet_ntop_emulated(family: cint, paddr: pointer, pStringBuffer: cstring,
  753. stringBufSize: int32): cstring {.stdcall.} =
  754. case family
  755. of AF_INET:
  756. var sa: Sockaddr_in
  757. sa.sin_family = AF_INET
  758. sa.sin_addr = cast[ptr InAddr](paddr)[]
  759. var bs = stringBufSize.DWORD
  760. let r = WSAAddressToStringA(cast[ptr SockAddr](sa.addr), sa.sizeof.DWORD, nil, pStringBuffer, bs.addr)
  761. if r != 0:
  762. result = nil
  763. else:
  764. result = pStringBuffer
  765. of AF_INET6:
  766. var sa: Sockaddr_in6
  767. sa.sin6_family = AF_INET6
  768. sa.sin6_addr = cast[ptr In6_addr](paddr)[]
  769. var bs = stringBufSize.DWORD
  770. let r = WSAAddressToStringA(cast[ptr SockAddr](sa.addr), sa.sizeof.DWORD, nil, pStringBuffer, bs.addr)
  771. if r != 0:
  772. result = nil
  773. else:
  774. result = pStringBuffer
  775. else:
  776. setLastError(ERROR_BAD_ARGUMENTS)
  777. result = nil
  778. proc inet_ntop*(family: cint, paddr: pointer, pStringBuffer: cstring,
  779. stringBufSize: int32): cstring {.stdcall.} =
  780. var ver: OSVERSIONINFO
  781. ver.dwOSVersionInfoSize = sizeof(ver).DWORD
  782. let res = when useWinUnicode: getVersionExW(ver.addr) else: getVersionExA(ver.addr)
  783. if res == 0:
  784. result = nil
  785. elif ver.dwMajorVersion >= 6:
  786. if inet_ntop_real == nil:
  787. quit("Can't load inet_ntop proc from " & ws2dll)
  788. result = inet_ntop_real(family, paddr, pStringBuffer, stringBufSize)
  789. else:
  790. result = inet_ntop_emulated(family, paddr, pStringBuffer, stringBufSize)
  791. type
  792. WSAPROC_ACCEPTEX* = proc (sListenSocket: SocketHandle,
  793. sAcceptSocket: SocketHandle,
  794. lpOutputBuffer: pointer, dwReceiveDataLength: DWORD,
  795. dwLocalAddressLength: DWORD,
  796. dwRemoteAddressLength: DWORD,
  797. lpdwBytesReceived: ptr DWORD,
  798. lpOverlapped: POVERLAPPED): bool {.
  799. stdcall,gcsafe.}
  800. WSAPROC_CONNECTEX* = proc (s: SocketHandle, name: ptr SockAddr, namelen: cint,
  801. lpSendBuffer: pointer, dwSendDataLength: DWORD,
  802. lpdwBytesSent: ptr DWORD,
  803. lpOverlapped: POVERLAPPED): bool {.
  804. stdcall,gcsafe.}
  805. WSAPROC_GETACCEPTEXSOCKADDRS* = proc(lpOutputBuffer: pointer,
  806. dwReceiveDataLength: DWORD,
  807. dwLocalAddressLength: DWORD,
  808. dwRemoteAddressLength: DWORD,
  809. LocalSockaddr: ptr PSockAddr,
  810. LocalSockaddrLength: ptr cint,
  811. RemoteSockaddr: ptr PSockAddr,
  812. RemoteSockaddrLength: ptr cint) {.
  813. stdcall,gcsafe.}
  814. const
  815. WT_EXECUTEDEFAULT* = 0x00000000'i32
  816. WT_EXECUTEINIOTHREAD* = 0x00000001'i32
  817. WT_EXECUTEINUITHREAD* = 0x00000002'i32
  818. WT_EXECUTEINWAITTHREAD* = 0x00000004'i32
  819. WT_EXECUTEONLYONCE* = 0x00000008'i32
  820. WT_EXECUTELONGFUNCTION* = 0x00000010'i32
  821. WT_EXECUTEINTIMERTHREAD* = 0x00000020'i32
  822. WT_EXECUTEINPERSISTENTIOTHREAD* = 0x00000040'i32
  823. WT_EXECUTEINPERSISTENTTHREAD* = 0x00000080'i32
  824. WT_TRANSFER_IMPERSONATION* = 0x00000100'i32
  825. PROCESS_TERMINATE* = 0x00000001'i32
  826. PROCESS_CREATE_THREAD* = 0x00000002'i32
  827. PROCESS_SET_SESSIONID* = 0x00000004'i32
  828. PROCESS_VM_OPERATION* = 0x00000008'i32
  829. PROCESS_VM_READ* = 0x00000010'i32
  830. PROCESS_VM_WRITE* = 0x00000020'i32
  831. PROCESS_DUP_HANDLE* = 0x00000040'i32
  832. PROCESS_CREATE_PROCESS* = 0x00000080'i32
  833. PROCESS_SET_QUOTA* = 0x00000100'i32
  834. PROCESS_SET_INFORMATION* = 0x00000200'i32
  835. PROCESS_QUERY_INFORMATION* = 0x00000400'i32
  836. PROCESS_SUSPEND_RESUME* = 0x00000800'i32
  837. PROCESS_QUERY_LIMITED_INFORMATION* = 0x00001000'i32
  838. PROCESS_SET_LIMITED_INFORMATION* = 0x00002000'i32
  839. type
  840. WAITORTIMERCALLBACK* = proc(para1: pointer, para2: int32): void {.stdcall.}
  841. proc postQueuedCompletionStatus*(CompletionPort: HANDLE,
  842. dwNumberOfBytesTransferred: DWORD,
  843. dwCompletionKey: ULONG_PTR,
  844. lpOverlapped: pointer): bool
  845. {.stdcall, dynlib: "kernel32", importc: "PostQueuedCompletionStatus".}
  846. proc registerWaitForSingleObject*(phNewWaitObject: ptr Handle, hObject: Handle,
  847. Callback: WAITORTIMERCALLBACK,
  848. Context: pointer,
  849. dwMilliseconds: ULONG,
  850. dwFlags: ULONG): bool
  851. {.stdcall, dynlib: "kernel32", importc: "RegisterWaitForSingleObject".}
  852. proc unregisterWait*(WaitHandle: HANDLE): DWORD
  853. {.stdcall, dynlib: "kernel32", importc: "UnregisterWait".}
  854. proc openProcess*(dwDesiredAccess: DWORD, bInheritHandle: WINBOOL,
  855. dwProcessId: DWORD): Handle
  856. {.stdcall, dynlib: "kernel32", importc: "OpenProcess".}
  857. when defined(useWinAnsi):
  858. proc createEvent*(lpEventAttributes: ptr SECURITY_ATTRIBUTES,
  859. bManualReset: DWORD, bInitialState: DWORD,
  860. lpName: cstring): Handle
  861. {.stdcall, dynlib: "kernel32", importc: "CreateEventA".}
  862. else:
  863. proc createEvent*(lpEventAttributes: ptr SECURITY_ATTRIBUTES,
  864. bManualReset: DWORD, bInitialState: DWORD,
  865. lpName: ptr Utf16Char): Handle
  866. {.stdcall, dynlib: "kernel32", importc: "CreateEventW".}
  867. proc setEvent*(hEvent: Handle): cint
  868. {.stdcall, dynlib: "kernel32", importc: "SetEvent".}
  869. const
  870. FD_READ* = 0x00000001'i32
  871. FD_WRITE* = 0x00000002'i32
  872. FD_OOB* = 0x00000004'i32
  873. FD_ACCEPT* = 0x00000008'i32
  874. FD_CONNECT* = 0x00000010'i32
  875. FD_CLOSE* = 0x00000020'i32
  876. FD_QQS* = 0x00000040'i32
  877. FD_GROUP_QQS* = 0x00000080'i32
  878. FD_ROUTING_INTERFACE_CHANGE* = 0x00000100'i32
  879. FD_ADDRESS_LIST_CHANGE* = 0x00000200'i32
  880. FD_ALL_EVENTS* = 0x000003FF'i32
  881. proc wsaEventSelect*(s: SocketHandle, hEventObject: Handle,
  882. lNetworkEvents: clong): cint
  883. {.stdcall, importc: "WSAEventSelect", dynlib: "ws2_32.dll".}
  884. proc wsaCreateEvent*(): Handle
  885. {.stdcall, importc: "WSACreateEvent", dynlib: "ws2_32.dll".}
  886. proc wsaCloseEvent*(hEvent: Handle): bool
  887. {.stdcall, importc: "WSACloseEvent", dynlib: "ws2_32.dll".}
  888. proc wsaResetEvent*(hEvent: Handle): bool
  889. {.stdcall, importc: "WSAResetEvent", dynlib: "ws2_32.dll".}
  890. type
  891. KEY_EVENT_RECORD* {.final, pure.} = object
  892. eventType*: int16
  893. bKeyDown*: WINBOOL
  894. wRepeatCount*: int16
  895. wVirtualKeyCode*: int16
  896. wVirtualScanCode*: int16
  897. uChar*: int16
  898. dwControlKeyState*: DWORD
  899. when defined(useWinAnsi):
  900. proc readConsoleInput*(hConsoleInput: Handle, lpBuffer: pointer, nLength: cint,
  901. lpNumberOfEventsRead: ptr cint): cint
  902. {.stdcall, dynlib: "kernel32", importc: "ReadConsoleInputA".}
  903. else:
  904. proc readConsoleInput*(hConsoleInput: Handle, lpBuffer: pointer, nLength: cint,
  905. lpNumberOfEventsRead: ptr cint): cint
  906. {.stdcall, dynlib: "kernel32", importc: "ReadConsoleInputW".}
  907. type
  908. LPFIBER_START_ROUTINE* = proc (param: pointer): void {.stdcall.}
  909. const
  910. FIBER_FLAG_FLOAT_SWITCH* = 0x01
  911. proc CreateFiber*(stackSize: int, fn: LPFIBER_START_ROUTINE, param: pointer): pointer {.stdcall, discardable, dynlib: "kernel32", importc.}
  912. proc CreateFiberEx*(stkCommit: int, stkReserve: int, flags: int32, fn: LPFIBER_START_ROUTINE, param: pointer): pointer {.stdcall, discardable, dynlib: "kernel32", importc.}
  913. proc ConvertThreadToFiber*(param: pointer): pointer {.stdcall, discardable, dynlib: "kernel32", importc.}
  914. proc ConvertThreadToFiberEx*(param: pointer, flags: int32): pointer {.stdcall, discardable, dynlib: "kernel32", importc.}
  915. proc DeleteFiber*(fiber: pointer): void {.stdcall, discardable, dynlib: "kernel32", importc.}
  916. proc SwitchToFiber*(fiber: pointer): void {.stdcall, discardable, dynlib: "kernel32", importc.}
  917. proc GetCurrentFiber*(): pointer {.stdcall, importc, header: "windows.h".}
  918. proc toFILETIME*(t: int64): FILETIME =
  919. ## Convert the Windows file time timestamp ``t`` to ``FILETIME``.
  920. result = FILETIME(dwLowDateTime: cast[DWORD](t), dwHighDateTime: DWORD(t shr 32))
  921. type
  922. LPFILETIME* = ptr FILETIME
  923. proc setFileTime*(hFile: HANDLE, lpCreationTime: LPFILETIME,
  924. lpLastAccessTime: LPFILETIME, lpLastWriteTime: LPFILETIME): WINBOOL
  925. {.stdcall, dynlib: "kernel32", importc: "SetFileTime".}