/synapse/source/lib/sslinux.pas

http://transmisson-remote-gui.googlecode.com/ · Pascal · 1314 lines · 1118 code · 115 blank · 81 comment · 55 complexity · c5ebdde98b919920c8830e527d22e2be MD5 · raw file

  1. {==============================================================================|
  2. | Project : Ararat Synapse | 002.000.009 |
  3. |==============================================================================|
  4. | Content: Socket Independent Platform Layer - Linux definition include |
  5. |==============================================================================|
  6. | Copyright (c)1999-2010, Lukas Gebauer |
  7. | All rights reserved. |
  8. | |
  9. | Redistribution and use in source and binary forms, with or without |
  10. | modification, are permitted provided that the following conditions are met: |
  11. | |
  12. | Redistributions of source code must retain the above copyright notice, this |
  13. | list of conditions and the following disclaimer. |
  14. | |
  15. | Redistributions in binary form must reproduce the above copyright notice, |
  16. | this list of conditions and the following disclaimer in the documentation |
  17. | and/or other materials provided with the distribution. |
  18. | |
  19. | Neither the name of Lukas Gebauer nor the names of its contributors may |
  20. | be used to endorse or promote products derived from this software without |
  21. | specific prior written permission. |
  22. | |
  23. | THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
  24. | AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
  25. | IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
  26. | ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR |
  27. | ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
  28. | DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR |
  29. | SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER |
  30. | CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT |
  31. | LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY |
  32. | OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH |
  33. | DAMAGE. |
  34. |==============================================================================|
  35. | The Initial Developer of the Original Code is Lukas Gebauer (Czech Republic).|
  36. | Portions created by Lukas Gebauer are Copyright (c)2003-2010. |
  37. | All Rights Reserved. |
  38. |==============================================================================|
  39. | Contributor(s): |
  40. |==============================================================================|
  41. | History: see HISTORY.HTM from distribution package |
  42. | (Found at URL: http://www.ararat.cz/synapse/) |
  43. |==============================================================================}
  44. {:@exclude}
  45. {$IFDEF LINUX}
  46. //{$DEFINE FORCEOLDAPI}
  47. {Note about define FORCEOLDAPI:
  48. If you activate this compiler directive, then is allways used old socket API
  49. for name resolution. If you leave this directive inactive, then the new API
  50. is used, when running system allows it.
  51. For IPv6 support you must have new API!
  52. }
  53. {$IFDEF FPC}
  54. {$MODE DELPHI}
  55. {$ENDIF}
  56. {$H+}
  57. interface
  58. uses
  59. SyncObjs, SysUtils, Classes,
  60. synafpc,
  61. Libc;
  62. function InitSocketInterface(stack: string): Boolean;
  63. function DestroySocketInterface: Boolean;
  64. const
  65. WinsockLevel = $0202;
  66. type
  67. u_char = Char;
  68. u_short = Word;
  69. u_int = Integer;
  70. u_long = Longint;
  71. pu_long = ^u_long;
  72. pu_short = ^u_short;
  73. TSocket = u_int;
  74. TAddrFamily = integer;
  75. TMemory = pointer;
  76. const
  77. DLLStackName = 'libc.so.6';
  78. cLocalhost = '127.0.0.1';
  79. cAnyHost = '0.0.0.0';
  80. cBroadcast = '255.255.255.255';
  81. c6Localhost = '::1';
  82. c6AnyHost = '::0';
  83. c6Broadcast = 'ffff::1';
  84. cAnyPort = '0';
  85. type
  86. DWORD = Integer;
  87. __fd_mask = LongWord;
  88. const
  89. __FD_SETSIZE = 1024;
  90. __NFDBITS = 8 * sizeof(__fd_mask);
  91. type
  92. __fd_set = {packed} record
  93. fds_bits: packed array[0..(__FD_SETSIZE div __NFDBITS)-1] of __fd_mask;
  94. end;
  95. TFDSet = __fd_set;
  96. PFDSet = ^TFDSet;
  97. const
  98. FIONREAD = $541B;
  99. FIONBIO = $5421;
  100. FIOASYNC = $5452;
  101. type
  102. PTimeVal = ^TTimeVal;
  103. TTimeVal = packed record
  104. tv_sec: Longint;
  105. tv_usec: Longint;
  106. end;
  107. const
  108. IPPROTO_IP = 0; { Dummy }
  109. IPPROTO_ICMP = 1; { Internet Control Message Protocol }
  110. IPPROTO_IGMP = 2; { Internet Group Management Protocol}
  111. IPPROTO_TCP = 6; { TCP }
  112. IPPROTO_UDP = 17; { User Datagram Protocol }
  113. IPPROTO_IPV6 = 41;
  114. IPPROTO_ICMPV6 = 58;
  115. IPPROTO_RM = 113;
  116. IPPROTO_RAW = 255;
  117. IPPROTO_MAX = 256;
  118. type
  119. PInAddr = ^TInAddr;
  120. TInAddr = packed record
  121. case integer of
  122. 0: (S_bytes: packed array [0..3] of byte);
  123. 1: (S_addr: u_long);
  124. end;
  125. PSockAddrIn = ^TSockAddrIn;
  126. TSockAddrIn = packed record
  127. case Integer of
  128. 0: (sin_family: u_short;
  129. sin_port: u_short;
  130. sin_addr: TInAddr;
  131. sin_zero: array[0..7] of Char);
  132. 1: (sa_family: u_short;
  133. sa_data: array[0..13] of Char)
  134. end;
  135. TIP_mreq = record
  136. imr_multiaddr: TInAddr; { IP multicast address of group }
  137. imr_interface: TInAddr; { local IP address of interface }
  138. end;
  139. PInAddr6 = ^TInAddr6;
  140. TInAddr6 = packed record
  141. case integer of
  142. 0: (S6_addr: packed array [0..15] of byte);
  143. 1: (u6_addr8: packed array [0..15] of byte);
  144. 2: (u6_addr16: packed array [0..7] of word);
  145. 3: (u6_addr32: packed array [0..3] of integer);
  146. end;
  147. PSockAddrIn6 = ^TSockAddrIn6;
  148. TSockAddrIn6 = packed record
  149. sin6_family: u_short; // AF_INET6
  150. sin6_port: u_short; // Transport level port number
  151. sin6_flowinfo: u_long; // IPv6 flow information
  152. sin6_addr: TInAddr6; // IPv6 address
  153. sin6_scope_id: u_long; // Scope Id: IF number for link-local
  154. // SITE id for site-local
  155. end;
  156. TIPv6_mreq = record
  157. ipv6mr_multiaddr: TInAddr6; // IPv6 multicast address.
  158. ipv6mr_interface: integer; // Interface index.
  159. padding: u_long;
  160. end;
  161. PHostEnt = ^THostEnt;
  162. THostent = record
  163. h_name: PChar;
  164. h_aliases: PPChar;
  165. h_addrtype: Integer;
  166. h_length: Cardinal;
  167. case Byte of
  168. 0: (h_addr_list: PPChar);
  169. 1: (h_addr: PPChar);
  170. end;
  171. PNetEnt = ^TNetEnt;
  172. TNetEnt = record
  173. n_name: PChar;
  174. n_aliases: PPChar;
  175. n_addrtype: Integer;
  176. n_net: uint32_t;
  177. end;
  178. PServEnt = ^TServEnt;
  179. TServEnt = record
  180. s_name: PChar;
  181. s_aliases: PPChar;
  182. s_port: Integer;
  183. s_proto: PChar;
  184. end;
  185. PProtoEnt = ^TProtoEnt;
  186. TProtoEnt = record
  187. p_name: PChar;
  188. p_aliases: ^PChar;
  189. p_proto: u_short;
  190. end;
  191. const
  192. INADDR_ANY = $00000000;
  193. INADDR_LOOPBACK = $7F000001;
  194. INADDR_BROADCAST = $FFFFFFFF;
  195. INADDR_NONE = $FFFFFFFF;
  196. ADDR_ANY = INADDR_ANY;
  197. INVALID_SOCKET = TSocket(NOT(0));
  198. SOCKET_ERROR = -1;
  199. Const
  200. IP_TOS = 1; { int; IP type of service and precedence. }
  201. IP_TTL = 2; { int; IP time to live. }
  202. IP_HDRINCL = 3; { int; Header is included with data. }
  203. IP_OPTIONS = 4; { ip_opts; IP per-packet options. }
  204. IP_ROUTER_ALERT = 5; { bool }
  205. IP_RECVOPTS = 6; { bool }
  206. IP_RETOPTS = 7; { bool }
  207. IP_PKTINFO = 8; { bool }
  208. IP_PKTOPTIONS = 9;
  209. IP_PMTUDISC = 10; { obsolete name? }
  210. IP_MTU_DISCOVER = 10; { int; see below }
  211. IP_RECVERR = 11; { bool }
  212. IP_RECVTTL = 12; { bool }
  213. IP_RECVTOS = 13; { bool }
  214. IP_MULTICAST_IF = 32; { in_addr; set/get IP multicast i/f }
  215. IP_MULTICAST_TTL = 33; { u_char; set/get IP multicast ttl }
  216. IP_MULTICAST_LOOP = 34; { i_char; set/get IP multicast loopback }
  217. IP_ADD_MEMBERSHIP = 35; { ip_mreq; add an IP group membership }
  218. IP_DROP_MEMBERSHIP = 36; { ip_mreq; drop an IP group membership }
  219. SOL_SOCKET = 1;
  220. SO_DEBUG = 1;
  221. SO_REUSEADDR = 2;
  222. SO_TYPE = 3;
  223. SO_ERROR = 4;
  224. SO_DONTROUTE = 5;
  225. SO_BROADCAST = 6;
  226. SO_SNDBUF = 7;
  227. SO_RCVBUF = 8;
  228. SO_KEEPALIVE = 9;
  229. SO_OOBINLINE = 10;
  230. SO_NO_CHECK = 11;
  231. SO_PRIORITY = 12;
  232. SO_LINGER = 13;
  233. SO_BSDCOMPAT = 14;
  234. SO_REUSEPORT = 15;
  235. SO_PASSCRED = 16;
  236. SO_PEERCRED = 17;
  237. SO_RCVLOWAT = 18;
  238. SO_SNDLOWAT = 19;
  239. SO_RCVTIMEO = 20;
  240. SO_SNDTIMEO = 21;
  241. { Security levels - as per NRL IPv6 - don't actually do anything }
  242. SO_SECURITY_AUTHENTICATION = 22;
  243. SO_SECURITY_ENCRYPTION_TRANSPORT = 23;
  244. SO_SECURITY_ENCRYPTION_NETWORK = 24;
  245. SO_BINDTODEVICE = 25;
  246. { Socket filtering }
  247. SO_ATTACH_FILTER = 26;
  248. SO_DETACH_FILTER = 27;
  249. SOMAXCONN = 128;
  250. IPV6_UNICAST_HOPS = 16;
  251. IPV6_MULTICAST_IF = 17;
  252. IPV6_MULTICAST_HOPS = 18;
  253. IPV6_MULTICAST_LOOP = 19;
  254. IPV6_JOIN_GROUP = 20;
  255. IPV6_LEAVE_GROUP = 21;
  256. MSG_NOSIGNAL = $4000; // Do not generate SIGPIPE.
  257. // getnameinfo constants
  258. NI_MAXHOST = 1025;
  259. NI_MAXSERV = 32;
  260. NI_NOFQDN = $4;
  261. NI_NUMERICHOST = $1;
  262. NI_NAMEREQD = $8;
  263. NI_NUMERICSERV = $2;
  264. NI_DGRAM = $10;
  265. const
  266. SOCK_STREAM = 1; { stream socket }
  267. SOCK_DGRAM = 2; { datagram socket }
  268. SOCK_RAW = 3; { raw-protocol interface }
  269. SOCK_RDM = 4; { reliably-delivered message }
  270. SOCK_SEQPACKET = 5; { sequenced packet stream }
  271. { TCP options. }
  272. TCP_NODELAY = $0001;
  273. { Address families. }
  274. AF_UNSPEC = 0; { unspecified }
  275. AF_INET = 2; { internetwork: UDP, TCP, etc. }
  276. AF_INET6 = 10; { Internetwork Version 6 }
  277. AF_MAX = 24;
  278. { Protocol families, same as address families for now. }
  279. PF_UNSPEC = AF_UNSPEC;
  280. PF_INET = AF_INET;
  281. PF_INET6 = AF_INET6;
  282. PF_MAX = AF_MAX;
  283. type
  284. { Structure used by kernel to store most addresses. }
  285. PSockAddr = ^TSockAddr;
  286. TSockAddr = TSockAddrIn;
  287. { Structure used by kernel to pass protocol information in raw sockets. }
  288. PSockProto = ^TSockProto;
  289. TSockProto = packed record
  290. sp_family: u_short;
  291. sp_protocol: u_short;
  292. end;
  293. type
  294. PAddrInfo = ^TAddrInfo;
  295. TAddrInfo = record
  296. ai_flags: integer; // AI_PASSIVE, AI_CANONNAME, AI_NUMERICHOST.
  297. ai_family: integer; // PF_xxx.
  298. ai_socktype: integer; // SOCK_xxx.
  299. ai_protocol: integer; // 0 or IPPROTO_xxx for IPv4 and IPv6.
  300. ai_addrlen: u_int; // Length of ai_addr.
  301. ai_addr: PSockAddr; // Binary address.
  302. ai_canonname: PChar; // Canonical name for nodename.
  303. ai_next: PAddrInfo; // Next structure in linked list.
  304. end;
  305. const
  306. // Flags used in "hints" argument to getaddrinfo().
  307. AI_PASSIVE = $1; // Socket address will be used in bind() call.
  308. AI_CANONNAME = $2; // Return canonical name in first ai_canonname.
  309. AI_NUMERICHOST = $4; // Nodename must be a numeric address string.
  310. type
  311. { Structure used for manipulating linger option. }
  312. PLinger = ^TLinger;
  313. TLinger = packed record
  314. l_onoff: integer;
  315. l_linger: integer;
  316. end;
  317. const
  318. MSG_OOB = $01; // Process out-of-band data.
  319. MSG_PEEK = $02; // Peek at incoming messages.
  320. const
  321. WSAEINTR = EINTR;
  322. WSAEBADF = EBADF;
  323. WSAEACCES = EACCES;
  324. WSAEFAULT = EFAULT;
  325. WSAEINVAL = EINVAL;
  326. WSAEMFILE = EMFILE;
  327. WSAEWOULDBLOCK = EWOULDBLOCK;
  328. WSAEINPROGRESS = EINPROGRESS;
  329. WSAEALREADY = EALREADY;
  330. WSAENOTSOCK = ENOTSOCK;
  331. WSAEDESTADDRREQ = EDESTADDRREQ;
  332. WSAEMSGSIZE = EMSGSIZE;
  333. WSAEPROTOTYPE = EPROTOTYPE;
  334. WSAENOPROTOOPT = ENOPROTOOPT;
  335. WSAEPROTONOSUPPORT = EPROTONOSUPPORT;
  336. WSAESOCKTNOSUPPORT = ESOCKTNOSUPPORT;
  337. WSAEOPNOTSUPP = EOPNOTSUPP;
  338. WSAEPFNOSUPPORT = EPFNOSUPPORT;
  339. WSAEAFNOSUPPORT = EAFNOSUPPORT;
  340. WSAEADDRINUSE = EADDRINUSE;
  341. WSAEADDRNOTAVAIL = EADDRNOTAVAIL;
  342. WSAENETDOWN = ENETDOWN;
  343. WSAENETUNREACH = ENETUNREACH;
  344. WSAENETRESET = ENETRESET;
  345. WSAECONNABORTED = ECONNABORTED;
  346. WSAECONNRESET = ECONNRESET;
  347. WSAENOBUFS = ENOBUFS;
  348. WSAEISCONN = EISCONN;
  349. WSAENOTCONN = ENOTCONN;
  350. WSAESHUTDOWN = ESHUTDOWN;
  351. WSAETOOMANYREFS = ETOOMANYREFS;
  352. WSAETIMEDOUT = ETIMEDOUT;
  353. WSAECONNREFUSED = ECONNREFUSED;
  354. WSAELOOP = ELOOP;
  355. WSAENAMETOOLONG = ENAMETOOLONG;
  356. WSAEHOSTDOWN = EHOSTDOWN;
  357. WSAEHOSTUNREACH = EHOSTUNREACH;
  358. WSAENOTEMPTY = ENOTEMPTY;
  359. WSAEPROCLIM = -1;
  360. WSAEUSERS = EUSERS;
  361. WSAEDQUOT = EDQUOT;
  362. WSAESTALE = ESTALE;
  363. WSAEREMOTE = EREMOTE;
  364. WSASYSNOTREADY = -2;
  365. WSAVERNOTSUPPORTED = -3;
  366. WSANOTINITIALISED = -4;
  367. WSAEDISCON = -5;
  368. WSAHOST_NOT_FOUND = HOST_NOT_FOUND;
  369. WSATRY_AGAIN = TRY_AGAIN;
  370. WSANO_RECOVERY = NO_RECOVERY;
  371. WSANO_DATA = -6;
  372. EAI_BADFLAGS = -1; { Invalid value for `ai_flags' field. }
  373. EAI_NONAME = -2; { NAME or SERVICE is unknown. }
  374. EAI_AGAIN = -3; { Temporary failure in name resolution. }
  375. EAI_FAIL = -4; { Non-recoverable failure in name res. }
  376. EAI_NODATA = -5; { No address associated with NAME. }
  377. EAI_FAMILY = -6; { `ai_family' not supported. }
  378. EAI_SOCKTYPE = -7; { `ai_socktype' not supported. }
  379. EAI_SERVICE = -8; { SERVICE not supported for `ai_socktype'. }
  380. EAI_ADDRFAMILY = -9; { Address family for NAME not supported. }
  381. EAI_MEMORY = -10; { Memory allocation failure. }
  382. EAI_SYSTEM = -11; { System error returned in `errno'. }
  383. const
  384. WSADESCRIPTION_LEN = 256;
  385. WSASYS_STATUS_LEN = 128;
  386. type
  387. PWSAData = ^TWSAData;
  388. TWSAData = packed record
  389. wVersion: Word;
  390. wHighVersion: Word;
  391. szDescription: array[0..WSADESCRIPTION_LEN] of Char;
  392. szSystemStatus: array[0..WSASYS_STATUS_LEN] of Char;
  393. iMaxSockets: Word;
  394. iMaxUdpDg: Word;
  395. lpVendorInfo: PChar;
  396. end;
  397. function IN6_IS_ADDR_UNSPECIFIED(const a: PInAddr6): boolean;
  398. function IN6_IS_ADDR_LOOPBACK(const a: PInAddr6): boolean;
  399. function IN6_IS_ADDR_LINKLOCAL(const a: PInAddr6): boolean;
  400. function IN6_IS_ADDR_SITELOCAL(const a: PInAddr6): boolean;
  401. function IN6_IS_ADDR_MULTICAST(const a: PInAddr6): boolean;
  402. function IN6_ADDR_EQUAL(const a: PInAddr6; const b: PInAddr6):boolean;
  403. procedure SET_IN6_IF_ADDR_ANY (const a: PInAddr6);
  404. procedure SET_LOOPBACK_ADDR6 (const a: PInAddr6);
  405. var
  406. in6addr_any, in6addr_loopback : TInAddr6;
  407. procedure FD_CLR(Socket: TSocket; var FDSet: TFDSet);
  408. function FD_ISSET(Socket: TSocket; var FDSet: TFDSet): Boolean;
  409. procedure FD_SET(Socket: TSocket; var FDSet: TFDSet);
  410. procedure FD_ZERO(var FDSet: TFDSet);
  411. {=============================================================================}
  412. type
  413. TWSAStartup = function(wVersionRequired: Word; var WSData: TWSAData): Integer;
  414. cdecl;
  415. TWSACleanup = function: Integer;
  416. cdecl;
  417. TWSAGetLastError = function: Integer;
  418. cdecl;
  419. TGetServByName = function(name, proto: PChar): PServEnt;
  420. cdecl;
  421. TGetServByPort = function(port: Integer; proto: PChar): PServEnt;
  422. cdecl;
  423. TGetProtoByName = function(name: PChar): PProtoEnt;
  424. cdecl;
  425. TGetProtoByNumber = function(proto: Integer): PProtoEnt;
  426. cdecl;
  427. TGetHostByName = function(name: PChar): PHostEnt;
  428. cdecl;
  429. TGetHostByAddr = function(addr: Pointer; len, Struc: Integer): PHostEnt;
  430. cdecl;
  431. TGetHostName = function(name: PChar; len: Integer): Integer;
  432. cdecl;
  433. TShutdown = function(s: TSocket; how: Integer): Integer;
  434. cdecl;
  435. TSetSockOpt = function(s: TSocket; level, optname: Integer; optval: PChar;
  436. optlen: Integer): Integer;
  437. cdecl;
  438. TGetSockOpt = function(s: TSocket; level, optname: Integer; optval: PChar;
  439. var optlen: Integer): Integer;
  440. cdecl;
  441. TSendTo = function(s: TSocket; const Buf; len, flags: Integer; addrto: PSockAddr;
  442. tolen: Integer): Integer;
  443. cdecl;
  444. TSend = function(s: TSocket; const Buf; len, flags: Integer): Integer;
  445. cdecl;
  446. TRecv = function(s: TSocket; var Buf; len, flags: Integer): Integer;
  447. cdecl;
  448. TRecvFrom = function(s: TSocket; var Buf; len, flags: Integer; from: PSockAddr;
  449. var fromlen: Integer): Integer;
  450. cdecl;
  451. Tntohs = function(netshort: u_short): u_short;
  452. cdecl;
  453. Tntohl = function(netlong: u_long): u_long;
  454. cdecl;
  455. TListen = function(s: TSocket; backlog: Integer): Integer;
  456. cdecl;
  457. TIoctlSocket = function(s: TSocket; cmd: DWORD; var arg: integer): Integer;
  458. cdecl;
  459. TInet_ntoa = function(inaddr: TInAddr): PChar;
  460. cdecl;
  461. TInet_addr = function(cp: PChar): u_long;
  462. cdecl;
  463. Thtons = function(hostshort: u_short): u_short;
  464. cdecl;
  465. Thtonl = function(hostlong: u_long): u_long;
  466. cdecl;
  467. TGetSockName = function(s: TSocket; name: PSockAddr; var namelen: Integer): Integer;
  468. cdecl;
  469. TGetPeerName = function(s: TSocket; name: PSockAddr; var namelen: Integer): Integer;
  470. cdecl;
  471. TConnect = function(s: TSocket; name: PSockAddr; namelen: Integer): Integer;
  472. cdecl;
  473. TCloseSocket = function(s: TSocket): Integer;
  474. cdecl;
  475. TBind = function(s: TSocket; addr: PSockAddr; namelen: Integer): Integer;
  476. cdecl;
  477. TAccept = function(s: TSocket; addr: PSockAddr; var addrlen: Integer): TSocket;
  478. cdecl;
  479. TTSocket = function(af, Struc, Protocol: Integer): TSocket;
  480. cdecl;
  481. TSelect = function(nfds: Integer; readfds, writefds, exceptfds: PFDSet;
  482. timeout: PTimeVal): Longint;
  483. cdecl;
  484. TGetAddrInfo = function(NodeName: PChar; ServName: PChar; Hints: PAddrInfo;
  485. var Addrinfo: PAddrInfo): integer;
  486. cdecl;
  487. TFreeAddrInfo = procedure(ai: PAddrInfo);
  488. cdecl;
  489. TGetNameInfo = function( addr: PSockAddr; namelen: Integer; host: PChar;
  490. hostlen: DWORD; serv: PChar; servlen: DWORD; flags: integer): integer;
  491. cdecl;
  492. var
  493. WSAStartup: TWSAStartup = nil;
  494. WSACleanup: TWSACleanup = nil;
  495. WSAGetLastError: TWSAGetLastError = nil;
  496. GetServByName: TGetServByName = nil;
  497. GetServByPort: TGetServByPort = nil;
  498. GetProtoByName: TGetProtoByName = nil;
  499. GetProtoByNumber: TGetProtoByNumber = nil;
  500. GetHostByName: TGetHostByName = nil;
  501. GetHostByAddr: TGetHostByAddr = nil;
  502. ssGetHostName: TGetHostName = nil;
  503. Shutdown: TShutdown = nil;
  504. SetSockOpt: TSetSockOpt = nil;
  505. GetSockOpt: TGetSockOpt = nil;
  506. ssSendTo: TSendTo = nil;
  507. ssSend: TSend = nil;
  508. ssRecv: TRecv = nil;
  509. ssRecvFrom: TRecvFrom = nil;
  510. ntohs: Tntohs = nil;
  511. ntohl: Tntohl = nil;
  512. Listen: TListen = nil;
  513. IoctlSocket: TIoctlSocket = nil;
  514. Inet_ntoa: TInet_ntoa = nil;
  515. Inet_addr: TInet_addr = nil;
  516. htons: Thtons = nil;
  517. htonl: Thtonl = nil;
  518. ssGetSockName: TGetSockName = nil;
  519. ssGetPeerName: TGetPeerName = nil;
  520. ssConnect: TConnect = nil;
  521. CloseSocket: TCloseSocket = nil;
  522. ssBind: TBind = nil;
  523. ssAccept: TAccept = nil;
  524. Socket: TTSocket = nil;
  525. Select: TSelect = nil;
  526. GetAddrInfo: TGetAddrInfo = nil;
  527. FreeAddrInfo: TFreeAddrInfo = nil;
  528. GetNameInfo: TGetNameInfo = nil;
  529. function LSWSAStartup(wVersionRequired: Word; var WSData: TWSAData): Integer; cdecl;
  530. function LSWSACleanup: Integer; cdecl;
  531. function LSWSAGetLastError: Integer; cdecl;
  532. var
  533. SynSockCS: SyncObjs.TCriticalSection;
  534. SockEnhancedApi: Boolean;
  535. SockWship6Api: Boolean;
  536. type
  537. TVarSin = packed record
  538. case integer of
  539. 0: (AddressFamily: u_short);
  540. 1: (
  541. case sin_family: u_short of
  542. AF_INET: (sin_port: u_short;
  543. sin_addr: TInAddr;
  544. sin_zero: array[0..7] of Char);
  545. AF_INET6: (sin6_port: u_short;
  546. sin6_flowinfo: u_long;
  547. sin6_addr: TInAddr6;
  548. sin6_scope_id: u_long);
  549. );
  550. end;
  551. function SizeOfVarSin(sin: TVarSin): integer;
  552. function Bind(s: TSocket; const addr: TVarSin): Integer;
  553. function Connect(s: TSocket; const name: TVarSin): Integer;
  554. function GetSockName(s: TSocket; var name: TVarSin): Integer;
  555. function GetPeerName(s: TSocket; var name: TVarSin): Integer;
  556. function GetHostName: string;
  557. function Send(s: TSocket; Buf: TMemory; len, flags: Integer): Integer;
  558. function Recv(s: TSocket; Buf: TMemory; len, flags: Integer): Integer;
  559. function SendTo(s: TSocket; Buf: TMemory; len, flags: Integer; addrto: TVarSin): Integer;
  560. function RecvFrom(s: TSocket; Buf: TMemory; len, flags: Integer; var from: TVarSin): Integer;
  561. function Accept(s: TSocket; var addr: TVarSin): TSocket;
  562. function IsNewApi(Family: integer): Boolean;
  563. function SetVarSin(var Sin: TVarSin; IP, Port: string; Family, SockProtocol, SockType: integer; PreferIP4: Boolean): integer;
  564. function GetSinIP(Sin: TVarSin): string;
  565. function GetSinPort(Sin: TVarSin): Integer;
  566. procedure ResolveNameToIP(Name: string; Family, SockProtocol, SockType: integer; const IPList: TStrings);
  567. function ResolveIPToName(IP: string; Family, SockProtocol, SockType: integer): string;
  568. function ResolvePort(Port: string; Family, SockProtocol, SockType: integer): Word;
  569. {==============================================================================}
  570. implementation
  571. var
  572. SynSockCount: Integer = 0;
  573. LibHandle: TLibHandle = 0;
  574. Libwship6Handle: TLibHandle = 0;
  575. function IN6_IS_ADDR_UNSPECIFIED(const a: PInAddr6): boolean;
  576. begin
  577. Result := ((a^.u6_addr32[0] = 0) and (a^.u6_addr32[1] = 0) and
  578. (a^.u6_addr32[2] = 0) and (a^.u6_addr32[3] = 0));
  579. end;
  580. function IN6_IS_ADDR_LOOPBACK(const a: PInAddr6): boolean;
  581. begin
  582. Result := ((a^.u6_addr32[0] = 0) and (a^.u6_addr32[1] = 0) and
  583. (a^.u6_addr32[2] = 0) and
  584. (a^.u6_addr8[12] = 0) and (a^.u6_addr8[13] = 0) and
  585. (a^.u6_addr8[14] = 0) and (a^.u6_addr8[15] = 1));
  586. end;
  587. function IN6_IS_ADDR_LINKLOCAL(const a: PInAddr6): boolean;
  588. begin
  589. Result := ((a^.u6_addr8[0] = $FE) and (a^.u6_addr8[1] = $80));
  590. end;
  591. function IN6_IS_ADDR_SITELOCAL(const a: PInAddr6): boolean;
  592. begin
  593. Result := ((a^.u6_addr8[0] = $FE) and (a^.u6_addr8[1] = $C0));
  594. end;
  595. function IN6_IS_ADDR_MULTICAST(const a: PInAddr6): boolean;
  596. begin
  597. Result := (a^.u6_addr8[0] = $FF);
  598. end;
  599. function IN6_ADDR_EQUAL(const a: PInAddr6; const b: PInAddr6): boolean;
  600. begin
  601. Result := (CompareMem( a, b, sizeof(TInAddr6)));
  602. end;
  603. procedure SET_IN6_IF_ADDR_ANY (const a: PInAddr6);
  604. begin
  605. FillChar(a^, sizeof(TInAddr6), 0);
  606. end;
  607. procedure SET_LOOPBACK_ADDR6 (const a: PInAddr6);
  608. begin
  609. FillChar(a^, sizeof(TInAddr6), 0);
  610. a^.u6_addr8[15] := 1;
  611. end;
  612. {=============================================================================}
  613. var
  614. {$IFNDEF VER1_0} //FTP version 1.0.x
  615. errno_loc: function: PInteger cdecl = nil;
  616. {$ELSE}
  617. errno_loc: function: PInteger = nil; cdecl;
  618. {$ENDIF}
  619. function LSWSAStartup(wVersionRequired: Word; var WSData: TWSAData): Integer;
  620. begin
  621. with WSData do
  622. begin
  623. wVersion := wVersionRequired;
  624. wHighVersion := $202;
  625. szDescription := 'Synsock - Synapse Platform Independent Socket Layer';
  626. szSystemStatus := 'Running on Linux';
  627. iMaxSockets := 32768;
  628. iMaxUdpDg := 8192;
  629. end;
  630. Result := 0;
  631. end;
  632. function LSWSACleanup: Integer;
  633. begin
  634. Result := 0;
  635. end;
  636. function LSWSAGetLastError: Integer;
  637. var
  638. p: PInteger;
  639. begin
  640. p := errno_loc;
  641. Result := p^;
  642. end;
  643. function __FDELT(Socket: TSocket): Integer;
  644. begin
  645. Result := Socket div __NFDBITS;
  646. end;
  647. function __FDMASK(Socket: TSocket): __fd_mask;
  648. begin
  649. Result := LongWord(1) shl (Socket mod __NFDBITS);
  650. end;
  651. function FD_ISSET(Socket: TSocket; var fdset: TFDSet): Boolean;
  652. begin
  653. Result := (fdset.fds_bits[__FDELT(Socket)] and __FDMASK(Socket)) <> 0;
  654. end;
  655. procedure FD_SET(Socket: TSocket; var fdset: TFDSet);
  656. begin
  657. fdset.fds_bits[__FDELT(Socket)] := fdset.fds_bits[__FDELT(Socket)] or __FDMASK(Socket);
  658. end;
  659. procedure FD_CLR(Socket: TSocket; var fdset: TFDSet);
  660. begin
  661. fdset.fds_bits[__FDELT(Socket)] := fdset.fds_bits[__FDELT(Socket)] and (not __FDMASK(Socket));
  662. end;
  663. procedure FD_ZERO(var fdset: TFDSet);
  664. var
  665. I: Integer;
  666. begin
  667. with fdset do
  668. for I := Low(fds_bits) to High(fds_bits) do
  669. fds_bits[I] := 0;
  670. end;
  671. {=============================================================================}
  672. function SizeOfVarSin(sin: TVarSin): integer;
  673. begin
  674. case sin.sin_family of
  675. AF_INET:
  676. Result := SizeOf(TSockAddrIn);
  677. AF_INET6:
  678. Result := SizeOf(TSockAddrIn6);
  679. else
  680. Result := 0;
  681. end;
  682. end;
  683. {=============================================================================}
  684. function Bind(s: TSocket; const addr: TVarSin): Integer;
  685. begin
  686. Result := ssBind(s, @addr, SizeOfVarSin(addr));
  687. end;
  688. function Connect(s: TSocket; const name: TVarSin): Integer;
  689. begin
  690. Result := ssConnect(s, @name, SizeOfVarSin(name));
  691. end;
  692. function GetSockName(s: TSocket; var name: TVarSin): Integer;
  693. var
  694. len: integer;
  695. begin
  696. len := SizeOf(name);
  697. FillChar(name, len, 0);
  698. Result := ssGetSockName(s, @name, Len);
  699. end;
  700. function GetPeerName(s: TSocket; var name: TVarSin): Integer;
  701. var
  702. len: integer;
  703. begin
  704. len := SizeOf(name);
  705. FillChar(name, len, 0);
  706. Result := ssGetPeerName(s, @name, Len);
  707. end;
  708. function GetHostName: string;
  709. var
  710. s: string;
  711. begin
  712. Result := '';
  713. setlength(s, 255);
  714. ssGetHostName(pchar(s), Length(s) - 1);
  715. Result := Pchar(s);
  716. end;
  717. function Send(s: TSocket; Buf: TMemory; len, flags: Integer): Integer;
  718. begin
  719. Result := ssSend(s, Buf^, len, flags);
  720. end;
  721. function Recv(s: TSocket; Buf: TMemory; len, flags: Integer): Integer;
  722. begin
  723. Result := ssRecv(s, Buf^, len, flags);
  724. end;
  725. function SendTo(s: TSocket; Buf: TMemory; len, flags: Integer; addrto: TVarSin): Integer;
  726. begin
  727. Result := ssSendTo(s, Buf^, len, flags, @addrto, SizeOfVarSin(addrto));
  728. end;
  729. function RecvFrom(s: TSocket; Buf: TMemory; len, flags: Integer; var from: TVarSin): Integer;
  730. var
  731. x: integer;
  732. begin
  733. x := SizeOf(from);
  734. Result := ssRecvFrom(s, Buf^, len, flags, @from, x);
  735. end;
  736. function Accept(s: TSocket; var addr: TVarSin): TSocket;
  737. var
  738. x: integer;
  739. begin
  740. x := SizeOf(addr);
  741. Result := ssAccept(s, @addr, x);
  742. end;
  743. {=============================================================================}
  744. function IsNewApi(Family: integer): Boolean;
  745. begin
  746. Result := SockEnhancedApi;
  747. if not Result then
  748. Result := (Family = AF_INET6) and SockWship6Api;
  749. end;
  750. function SetVarSin(var Sin: TVarSin; IP, Port: string; Family, SockProtocol, SockType: integer; PreferIP4: Boolean): integer;
  751. type
  752. pu_long = ^u_long;
  753. var
  754. ProtoEnt: PProtoEnt;
  755. ServEnt: PServEnt;
  756. HostEnt: PHostEnt;
  757. r: integer;
  758. Hints1, Hints2: TAddrInfo;
  759. Sin1, Sin2: TVarSin;
  760. TwoPass: boolean;
  761. function GetAddr(const IP, port: string; Hints: TAddrInfo; var Sin: TVarSin): integer;
  762. var
  763. Addr: PAddrInfo;
  764. begin
  765. Addr := nil;
  766. try
  767. FillChar(Sin, Sizeof(Sin), 0);
  768. if Hints.ai_socktype = SOCK_RAW then
  769. begin
  770. Hints.ai_socktype := 0;
  771. Hints.ai_protocol := 0;
  772. Result := synsock.GetAddrInfo(PChar(IP), nil, @Hints, Addr);
  773. end
  774. else
  775. begin
  776. if (IP = cAnyHost) or (IP = c6AnyHost) then
  777. begin
  778. Hints.ai_flags := AI_PASSIVE;
  779. Result := synsock.GetAddrInfo(nil, PChar(Port), @Hints, Addr);
  780. end
  781. else
  782. if (IP = cLocalhost) or (IP = c6Localhost) then
  783. begin
  784. Result := synsock.GetAddrInfo(nil, PChar(Port), @Hints, Addr);
  785. end
  786. else
  787. begin
  788. Result := synsock.GetAddrInfo(PChar(IP), PChar(Port), @Hints, Addr);
  789. end;
  790. end;
  791. if Result = 0 then
  792. if (Addr <> nil) then
  793. Move(Addr^.ai_addr^, Sin, Addr^.ai_addrlen);
  794. finally
  795. if Assigned(Addr) then
  796. synsock.FreeAddrInfo(Addr);
  797. end;
  798. end;
  799. begin
  800. Result := 0;
  801. FillChar(Sin, Sizeof(Sin), 0);
  802. if not IsNewApi(family) then
  803. begin
  804. SynSockCS.Enter;
  805. try
  806. Sin.sin_family := AF_INET;
  807. ProtoEnt := synsock.GetProtoByNumber(SockProtocol);
  808. ServEnt := nil;
  809. if ProtoEnt <> nil then
  810. ServEnt := synsock.GetServByName(PChar(Port), ProtoEnt^.p_name);
  811. if ServEnt = nil then
  812. Sin.sin_port := synsock.htons(StrToIntDef(Port, 0))
  813. else
  814. Sin.sin_port := ServEnt^.s_port;
  815. if IP = cBroadcast then
  816. Sin.sin_addr.s_addr := u_long(INADDR_BROADCAST)
  817. else
  818. begin
  819. Sin.sin_addr.s_addr := synsock.inet_addr(PChar(IP));
  820. if Sin.sin_addr.s_addr = u_long(INADDR_NONE) then
  821. begin
  822. HostEnt := synsock.GetHostByName(PChar(IP));
  823. Result := synsock.WSAGetLastError;
  824. if HostEnt <> nil then
  825. Sin.sin_addr.S_addr := u_long(Pu_long(HostEnt^.h_addr_list^)^);
  826. end;
  827. end;
  828. finally
  829. SynSockCS.Leave;
  830. end;
  831. end
  832. else
  833. begin
  834. FillChar(Hints1, Sizeof(Hints1), 0);
  835. FillChar(Hints2, Sizeof(Hints2), 0);
  836. TwoPass := False;
  837. if Family = AF_UNSPEC then
  838. begin
  839. if PreferIP4 then
  840. begin
  841. Hints1.ai_family := AF_INET;
  842. Hints2.ai_family := AF_INET6;
  843. TwoPass := True;
  844. end
  845. else
  846. begin
  847. Hints2.ai_family := AF_INET;
  848. Hints1.ai_family := AF_INET6;
  849. TwoPass := True;
  850. end;
  851. end
  852. else
  853. Hints1.ai_family := Family;
  854. Hints1.ai_socktype := SockType;
  855. Hints1.ai_protocol := SockProtocol;
  856. Hints2.ai_socktype := Hints1.ai_socktype;
  857. Hints2.ai_protocol := Hints1.ai_protocol;
  858. r := GetAddr(IP, Port, Hints1, Sin1);
  859. Result := r;
  860. sin := sin1;
  861. if r <> 0 then
  862. if TwoPass then
  863. begin
  864. r := GetAddr(IP, Port, Hints2, Sin2);
  865. Result := r;
  866. if r = 0 then
  867. sin := sin2;
  868. end;
  869. end;
  870. end;
  871. function GetSinIP(Sin: TVarSin): string;
  872. var
  873. p: PChar;
  874. host, serv: string;
  875. hostlen, servlen: integer;
  876. r: integer;
  877. begin
  878. Result := '';
  879. if not IsNewApi(Sin.AddressFamily) then
  880. begin
  881. p := synsock.inet_ntoa(Sin.sin_addr);
  882. if p <> nil then
  883. Result := p;
  884. end
  885. else
  886. begin
  887. hostlen := NI_MAXHOST;
  888. servlen := NI_MAXSERV;
  889. setlength(host, hostlen);
  890. setlength(serv, servlen);
  891. r := getnameinfo(@sin, SizeOfVarSin(sin), PChar(host), hostlen,
  892. PChar(serv), servlen, NI_NUMERICHOST + NI_NUMERICSERV);
  893. if r = 0 then
  894. Result := PChar(host);
  895. end;
  896. end;
  897. function GetSinPort(Sin: TVarSin): Integer;
  898. begin
  899. if (Sin.sin_family = AF_INET6) then
  900. Result := synsock.ntohs(Sin.sin6_port)
  901. else
  902. Result := synsock.ntohs(Sin.sin_port);
  903. end;
  904. procedure ResolveNameToIP(Name: string; Family, SockProtocol, SockType: integer; const IPList: TStrings);
  905. type
  906. TaPInAddr = array[0..250] of PInAddr;
  907. PaPInAddr = ^TaPInAddr;
  908. var
  909. Hints: TAddrInfo;
  910. Addr: PAddrInfo;
  911. AddrNext: PAddrInfo;
  912. r: integer;
  913. host, serv: string;
  914. hostlen, servlen: integer;
  915. RemoteHost: PHostEnt;
  916. IP: u_long;
  917. PAdrPtr: PaPInAddr;
  918. i: Integer;
  919. s: string;
  920. InAddr: TInAddr;
  921. begin
  922. IPList.Clear;
  923. if not IsNewApi(Family) then
  924. begin
  925. IP := synsock.inet_addr(PChar(Name));
  926. if IP = u_long(INADDR_NONE) then
  927. begin
  928. SynSockCS.Enter;
  929. try
  930. RemoteHost := synsock.GetHostByName(PChar(Name));
  931. if RemoteHost <> nil then
  932. begin
  933. PAdrPtr := PAPInAddr(RemoteHost^.h_addr_list);
  934. i := 0;
  935. while PAdrPtr^[i] <> nil do
  936. begin
  937. InAddr := PAdrPtr^[i]^;
  938. s := Format('%d.%d.%d.%d', [InAddr.S_bytes[0], InAddr.S_bytes[1],
  939. InAddr.S_bytes[2], InAddr.S_bytes[3]]);
  940. IPList.Add(s);
  941. Inc(i);
  942. end;
  943. end;
  944. finally
  945. SynSockCS.Leave;
  946. end;
  947. end
  948. else
  949. IPList.Add(Name);
  950. end
  951. else
  952. begin
  953. Addr := nil;
  954. try
  955. FillChar(Hints, Sizeof(Hints), 0);
  956. Hints.ai_family := AF_UNSPEC;
  957. Hints.ai_socktype := SockType;
  958. Hints.ai_protocol := SockProtocol;
  959. Hints.ai_flags := 0;
  960. r := synsock.GetAddrInfo(PChar(Name), nil, @Hints, Addr);
  961. if r = 0 then
  962. begin
  963. AddrNext := Addr;
  964. while not(AddrNext = nil) do
  965. begin
  966. if not(((Family = AF_INET6) and (AddrNext^.ai_family = AF_INET))
  967. or ((Family = AF_INET) and (AddrNext^.ai_family = AF_INET6))) then
  968. begin
  969. hostlen := NI_MAXHOST;
  970. servlen := NI_MAXSERV;
  971. setlength(host, hostlen);
  972. setlength(serv, servlen);
  973. r := getnameinfo(AddrNext^.ai_addr, AddrNext^.ai_addrlen,
  974. PChar(host), hostlen, PChar(serv), servlen,
  975. NI_NUMERICHOST + NI_NUMERICSERV);
  976. if r = 0 then
  977. begin
  978. host := PChar(host);
  979. IPList.Add(host);
  980. end;
  981. end;
  982. AddrNext := AddrNext^.ai_next;
  983. end;
  984. end;
  985. finally
  986. if Assigned(Addr) then
  987. synsock.FreeAddrInfo(Addr);
  988. end;
  989. end;
  990. if IPList.Count = 0 then
  991. IPList.Add(cAnyHost);
  992. end;
  993. function ResolvePort(Port: string; Family, SockProtocol, SockType: integer): Word;
  994. var
  995. ProtoEnt: PProtoEnt;
  996. ServEnt: PServEnt;
  997. Hints: TAddrInfo;
  998. Addr: PAddrInfo;
  999. r: integer;
  1000. begin
  1001. Result := 0;
  1002. if not IsNewApi(Family) then
  1003. begin
  1004. SynSockCS.Enter;
  1005. try
  1006. ProtoEnt := synsock.GetProtoByNumber(SockProtocol);
  1007. ServEnt := nil;
  1008. if ProtoEnt <> nil then
  1009. ServEnt := synsock.GetServByName(PChar(Port), ProtoEnt^.p_name);
  1010. if ServEnt = nil then
  1011. Result := StrToIntDef(Port, 0)
  1012. else
  1013. Result := synsock.htons(ServEnt^.s_port);
  1014. finally
  1015. SynSockCS.Leave;
  1016. end;
  1017. end
  1018. else
  1019. begin
  1020. Addr := nil;
  1021. try
  1022. FillChar(Hints, Sizeof(Hints), 0);
  1023. Hints.ai_family := AF_UNSPEC;
  1024. Hints.ai_socktype := SockType;
  1025. Hints.ai_protocol := Sockprotocol;
  1026. Hints.ai_flags := AI_PASSIVE;
  1027. r := synsock.GetAddrInfo(nil, PChar(Port), @Hints, Addr);
  1028. if (r = 0) and Assigned(Addr) then
  1029. begin
  1030. if Addr^.ai_family = AF_INET then
  1031. Result := synsock.htons(Addr^.ai_addr^.sin_port);
  1032. if Addr^.ai_family = AF_INET6 then
  1033. Result := synsock.htons(PSockAddrIn6(Addr^.ai_addr)^.sin6_port);
  1034. end;
  1035. finally
  1036. if Assigned(Addr) then
  1037. synsock.FreeAddrInfo(Addr);
  1038. end;
  1039. end;
  1040. end;
  1041. function ResolveIPToName(IP: string; Family, SockProtocol, SockType: integer): string;
  1042. var
  1043. Hints: TAddrInfo;
  1044. Addr: PAddrInfo;
  1045. r: integer;
  1046. host, serv: string;
  1047. hostlen, servlen: integer;
  1048. RemoteHost: PHostEnt;
  1049. IPn: u_long;
  1050. begin
  1051. Result := IP;
  1052. if not IsNewApi(Family) then
  1053. begin
  1054. IPn := synsock.inet_addr(PChar(IP));
  1055. if IPn <> u_long(INADDR_NONE) then
  1056. begin
  1057. SynSockCS.Enter;
  1058. try
  1059. RemoteHost := GetHostByAddr(@IPn, SizeOf(IPn), AF_INET);
  1060. if RemoteHost <> nil then
  1061. Result := RemoteHost^.h_name;
  1062. finally
  1063. SynSockCS.Leave;
  1064. end;
  1065. end;
  1066. end
  1067. else
  1068. begin
  1069. Addr := nil;
  1070. try
  1071. FillChar(Hints, Sizeof(Hints), 0);
  1072. Hints.ai_family := AF_UNSPEC;
  1073. Hints.ai_socktype := SockType;
  1074. Hints.ai_protocol := SockProtocol;
  1075. Hints.ai_flags := 0;
  1076. r := synsock.GetAddrInfo(PChar(IP), nil, @Hints, Addr);
  1077. if (r = 0) and Assigned(Addr)then
  1078. begin
  1079. hostlen := NI_MAXHOST;
  1080. servlen := NI_MAXSERV;
  1081. setlength(host, hostlen);
  1082. setlength(serv, servlen);
  1083. r := getnameinfo(Addr^.ai_addr, Addr^.ai_addrlen,
  1084. PChar(host), hostlen, PChar(serv), servlen,
  1085. NI_NUMERICSERV);
  1086. if r = 0 then
  1087. Result := PChar(host);
  1088. end;
  1089. finally
  1090. if Assigned(Addr) then
  1091. synsock.FreeAddrInfo(Addr);
  1092. end;
  1093. end;
  1094. end;
  1095. {=============================================================================}
  1096. function InitSocketInterface(stack: string): Boolean;
  1097. begin
  1098. Result := False;
  1099. SockEnhancedApi := False;
  1100. if stack = '' then
  1101. stack := DLLStackName;
  1102. SynSockCS.Enter;
  1103. try
  1104. if SynSockCount = 0 then
  1105. begin
  1106. SockEnhancedApi := False;
  1107. SockWship6Api := False;
  1108. Libc.Signal(Libc.SIGPIPE, TSignalHandler(Libc.SIG_IGN));
  1109. LibHandle := LoadLibrary(PChar(Stack));
  1110. if LibHandle <> 0 then
  1111. begin
  1112. errno_loc := GetProcAddress(LibHandle, PChar('__errno_location'));
  1113. CloseSocket := GetProcAddress(LibHandle, PChar('close'));
  1114. IoctlSocket := GetProcAddress(LibHandle, PChar('ioctl'));
  1115. WSAGetLastError := LSWSAGetLastError;
  1116. WSAStartup := LSWSAStartup;
  1117. WSACleanup := LSWSACleanup;
  1118. ssAccept := GetProcAddress(LibHandle, PChar('accept'));
  1119. ssBind := GetProcAddress(LibHandle, PChar('bind'));
  1120. ssConnect := GetProcAddress(LibHandle, PChar('connect'));
  1121. ssGetPeerName := GetProcAddress(LibHandle, PChar('getpeername'));
  1122. ssGetSockName := GetProcAddress(LibHandle, PChar('getsockname'));
  1123. GetSockOpt := GetProcAddress(LibHandle, PChar('getsockopt'));
  1124. Htonl := GetProcAddress(LibHandle, PChar('htonl'));
  1125. Htons := GetProcAddress(LibHandle, PChar('htons'));
  1126. Inet_Addr := GetProcAddress(LibHandle, PChar('inet_addr'));
  1127. Inet_Ntoa := GetProcAddress(LibHandle, PChar('inet_ntoa'));
  1128. Listen := GetProcAddress(LibHandle, PChar('listen'));
  1129. Ntohl := GetProcAddress(LibHandle, PChar('ntohl'));
  1130. Ntohs := GetProcAddress(LibHandle, PChar('ntohs'));
  1131. ssRecv := GetProcAddress(LibHandle, PChar('recv'));
  1132. ssRecvFrom := GetProcAddress(LibHandle, PChar('recvfrom'));
  1133. Select := GetProcAddress(LibHandle, PChar('select'));
  1134. ssSend := GetProcAddress(LibHandle, PChar('send'));
  1135. ssSendTo := GetProcAddress(LibHandle, PChar('sendto'));
  1136. SetSockOpt := GetProcAddress(LibHandle, PChar('setsockopt'));
  1137. ShutDown := GetProcAddress(LibHandle, PChar('shutdown'));
  1138. Socket := GetProcAddress(LibHandle, PChar('socket'));
  1139. GetHostByAddr := GetProcAddress(LibHandle, PChar('gethostbyaddr'));
  1140. GetHostByName := GetProcAddress(LibHandle, PChar('gethostbyname'));
  1141. GetProtoByName := GetProcAddress(LibHandle, PChar('getprotobyname'));
  1142. GetProtoByNumber := GetProcAddress(LibHandle, PChar('getprotobynumber'));
  1143. GetServByName := GetProcAddress(LibHandle, PChar('getservbyname'));
  1144. GetServByPort := GetProcAddress(LibHandle, PChar('getservbyport'));
  1145. ssGetHostName := GetProcAddress(LibHandle, PChar('gethostname'));
  1146. {$IFNDEF FORCEOLDAPI}
  1147. GetAddrInfo := GetProcAddress(LibHandle, PChar('getaddrinfo'));
  1148. FreeAddrInfo := GetProcAddress(LibHandle, PChar('freeaddrinfo'));
  1149. GetNameInfo := GetProcAddress(LibHandle, PChar('getnameinfo'));
  1150. SockEnhancedApi := Assigned(GetAddrInfo) and Assigned(FreeAddrInfo)
  1151. and Assigned(GetNameInfo);
  1152. {$ENDIF}
  1153. Result := True;
  1154. end;
  1155. end
  1156. else Result := True;
  1157. if Result then
  1158. Inc(SynSockCount);
  1159. finally
  1160. SynSockCS.Leave;
  1161. end;
  1162. end;
  1163. function DestroySocketInterface: Boolean;
  1164. begin
  1165. SynSockCS.Enter;
  1166. try
  1167. Dec(SynSockCount);
  1168. if SynSockCount < 0 then
  1169. SynSockCount := 0;
  1170. if SynSockCount = 0 then
  1171. begin
  1172. if LibHandle <> 0 then
  1173. begin
  1174. FreeLibrary(libHandle);
  1175. LibHandle := 0;
  1176. end;
  1177. if LibWship6Handle <> 0 then
  1178. begin
  1179. FreeLibrary(LibWship6Handle);
  1180. LibWship6Handle := 0;
  1181. end;
  1182. end;
  1183. finally
  1184. SynSockCS.Leave;
  1185. end;
  1186. Result := True;
  1187. end;
  1188. initialization
  1189. begin
  1190. SynSockCS := SyncObjs.TCriticalSection.Create;
  1191. SET_IN6_IF_ADDR_ANY (@in6addr_any);
  1192. SET_LOOPBACK_ADDR6 (@in6addr_loopback);
  1193. end;
  1194. finalization
  1195. begin
  1196. SynSockCS.Free;
  1197. end;
  1198. {$ENDIF}