winnet.c
来自「大名鼎鼎的远程登录软件putty的Symbian版源码」· C语言 代码 · 共 1,373 行 · 第 1/3 页
C
1,373 行
ret->address = p_ntohl(a); realhost[lenof(realhost)-1] = '\0'; *canonicalname = snewn(1+strlen(realhost), char); strcpy(*canonicalname, realhost); return ret;}SockAddr sk_nonamelookup(const char *host){ SockAddr ret = snew(struct SockAddr_tag); ret->error = NULL; ret->family = AF_UNSPEC; strncpy(ret->hostname, host, lenof(ret->hostname)); ret->hostname[lenof(ret->hostname)-1] = '\0'; return ret;}void sk_getaddr(SockAddr addr, char *buf, int buflen){#ifdef IPV6 if (addr->family == AF_INET6) { FIXME; /* I don't know how to get a text form of an IPv6 address. */ } else#endif if (addr->family == AF_INET) { struct in_addr a; a.s_addr = p_htonl(addr->address); strncpy(buf, p_inet_ntoa(a), buflen); buf[buflen-1] = '\0'; } else { assert(addr->family == AF_UNSPEC); strncpy(buf, addr->hostname, buflen); buf[buflen-1] = '\0'; }}int sk_hostname_is_local(char *name){ return !strcmp(name, "localhost");}static INTERFACE_INFO local_interfaces[16];static int n_local_interfaces; /* 0=not yet, -1=failed, >0=number */static int ipv4_is_local_addr(struct in_addr addr){ if (ipv4_is_loopback(addr)) return 1; /* loopback addresses are local */ if (!n_local_interfaces) { SOCKET s = p_socket(AF_INET, SOCK_DGRAM, 0); DWORD retbytes; if (p_WSAIoctl && p_WSAIoctl(s, SIO_GET_INTERFACE_LIST, NULL, 0, local_interfaces, sizeof(local_interfaces), &retbytes, NULL, NULL) == 0) n_local_interfaces = retbytes / sizeof(INTERFACE_INFO); else logevent(NULL, "Unable to get list of local IP addresses"); } if (n_local_interfaces > 0) { int i; for (i = 0; i < n_local_interfaces; i++) { SOCKADDR_IN *address = (SOCKADDR_IN *)&local_interfaces[i].iiAddress; if (address->sin_addr.s_addr == addr.s_addr) return 1; /* this address is local */ } } return 0; /* this address is not local */}int sk_address_is_local(SockAddr addr){#ifdef IPV6 if (addr->family == AF_INET6) { FIXME; /* someone who can compile for IPV6 had better do this bit */ } else#endif if (addr->family == AF_INET) { struct in_addr a; a.s_addr = p_htonl(addr->address); return ipv4_is_local_addr(a); } else { assert(addr->family == AF_UNSPEC); return 0; /* we don't know; assume not */ }}int sk_addrtype(SockAddr addr){ return (addr->family == AF_INET ? ADDRTYPE_IPV4 :#ifdef IPV6 addr->family == AF_INET6 ? ADDRTYPE_IPV6 :#endif ADDRTYPE_NAME);}void sk_addrcopy(SockAddr addr, char *buf){ assert(addr->family != AF_UNSPEC);#ifdef IPV6 if (addr->family == AF_INET6) { memcpy(buf, (char*) addr->ai, 16); } else#endif if (addr->family == AF_INET) { struct in_addr a; a.s_addr = p_htonl(addr->address); memcpy(buf, (char*) &a.s_addr, 4); }}void sk_addr_free(SockAddr addr){ sfree(addr);}static Plug sk_tcp_plug(Socket sock, Plug p){ Actual_Socket s = (Actual_Socket) sock; Plug ret = s->plug; if (p) s->plug = p; return ret;}static void sk_tcp_flush(Socket s){ /* * We send data to the socket as soon as we can anyway, * so we don't need to do anything here. :-) */}static void sk_tcp_close(Socket s);static int sk_tcp_write(Socket s, const char *data, int len);static int sk_tcp_write_oob(Socket s, const char *data, int len);static void sk_tcp_set_private_ptr(Socket s, void *ptr);static void *sk_tcp_get_private_ptr(Socket s);static void sk_tcp_set_frozen(Socket s, int is_frozen);static const char *sk_tcp_socket_error(Socket s);extern char *do_select(SOCKET skt, int startup);Socket sk_register(void *sock, Plug plug){ static const struct socket_function_table fn_table = { sk_tcp_plug, sk_tcp_close, sk_tcp_write, sk_tcp_write_oob, sk_tcp_flush, sk_tcp_set_private_ptr, sk_tcp_get_private_ptr, sk_tcp_set_frozen, sk_tcp_socket_error }; DWORD err; char *errstr; Actual_Socket ret; /* * Create Socket structure. */ ret = snew(struct Socket_tag); ret->fn = &fn_table; ret->error = NULL; ret->plug = plug; bufchain_init(&ret->output_data); ret->writable = 1; /* to start with */ ret->sending_oob = 0; ret->frozen = 1; ret->frozen_readable = 0; ret->localhost_only = 0; /* unused, but best init anyway */ ret->pending_error = 0; ret->s = (SOCKET)sock; if (ret->s == INVALID_SOCKET) { err = p_WSAGetLastError(); ret->error = winsock_error_string(err); return (Socket) ret; } ret->oobinline = 0; /* Set up a select mechanism. This could be an AsyncSelect on a * window, or an EventSelect on an event object. */ errstr = do_select(ret->s, 1); if (errstr) { ret->error = errstr; return (Socket) ret; } add234(sktree, ret); return (Socket) ret;}Socket sk_new(SockAddr addr, int port, int privport, int oobinline, int nodelay, int keepalive, Plug plug){ static const struct socket_function_table fn_table = { sk_tcp_plug, sk_tcp_close, sk_tcp_write, sk_tcp_write_oob, sk_tcp_flush, sk_tcp_set_private_ptr, sk_tcp_get_private_ptr, sk_tcp_set_frozen, sk_tcp_socket_error }; SOCKET s;#ifdef IPV6 SOCKADDR_IN6 a6;#endif SOCKADDR_IN a; DWORD err; char *errstr; Actual_Socket ret; short localport; /* * Create Socket structure. */ ret = snew(struct Socket_tag); ret->fn = &fn_table; ret->error = NULL; ret->plug = plug; bufchain_init(&ret->output_data); ret->connected = 0; /* to start with */ ret->writable = 0; /* to start with */ ret->sending_oob = 0; ret->frozen = 0; ret->frozen_readable = 0; ret->localhost_only = 0; /* unused, but best init anyway */ ret->pending_error = 0; /* * Open socket. */ assert(addr->family != AF_UNSPEC); s = p_socket(addr->family, SOCK_STREAM, 0); ret->s = s; if (s == INVALID_SOCKET) { err = p_WSAGetLastError(); ret->error = winsock_error_string(err); return (Socket) ret; } ret->oobinline = oobinline; if (oobinline) { BOOL b = TRUE; p_setsockopt(s, SOL_SOCKET, SO_OOBINLINE, (void *) &b, sizeof(b)); } if (nodelay) { BOOL b = TRUE; p_setsockopt(s, IPPROTO_TCP, TCP_NODELAY, (void *) &b, sizeof(b)); } if (keepalive) { BOOL b = TRUE; p_setsockopt(s, SOL_SOCKET, SO_KEEPALIVE, (void *) &b, sizeof(b)); } /* * Bind to local address. */ if (privport) localport = 1023; /* count from 1023 downwards */ else localport = 0; /* just use port 0 (ie winsock picks) */ /* Loop round trying to bind */ while (1) { int retcode;#ifdef IPV6 if (addr->family == AF_INET6) { memset(&a6, 0, sizeof(a6)); a6.sin6_family = AF_INET6;/*a6.sin6_addr = in6addr_any; *//* == 0 */ a6.sin6_port = p_htons(localport); } else#endif { a.sin_family = AF_INET; a.sin_addr.s_addr = p_htonl(INADDR_ANY); a.sin_port = p_htons(localport); }#ifdef IPV6 retcode = p_bind(s, (addr->family == AF_INET6 ? (struct sockaddr *) &a6 : (struct sockaddr *) &a), (addr->family == AF_INET6 ? sizeof(a6) : sizeof(a)));#else retcode = p_bind(s, (struct sockaddr *) &a, sizeof(a));#endif if (retcode != SOCKET_ERROR) { err = 0; break; /* done */ } else { err = p_WSAGetLastError(); if (err != WSAEADDRINUSE) /* failed, for a bad reason */ break; } if (localport == 0) break; /* we're only looping once */ localport--; if (localport == 0) break; /* we might have got to the end */ } if (err) { ret->error = winsock_error_string(err); return (Socket) ret; } /* * Connect to remote address. */#ifdef IPV6 if (addr->family == AF_INET6) { memset(&a, 0, sizeof(a)); a6.sin6_family = AF_INET6; a6.sin6_port = p_htons((short) port); a6.sin6_addr = ((struct sockaddr_in6 *) addr->ai->ai_addr)->sin6_addr; } else#endif { a.sin_family = AF_INET; a.sin_addr.s_addr = p_htonl(addr->address); a.sin_port = p_htons((short) port); } /* Set up a select mechanism. This could be an AsyncSelect on a * window, or an EventSelect on an event object. */ errstr = do_select(s, 1); if (errstr) { ret->error = errstr; return (Socket) ret; } if ((#ifdef IPV6 p_connect(s, ((addr->family == AF_INET6) ? (struct sockaddr *) &a6 : (struct sockaddr *) &a), (addr->family == AF_INET6) ? sizeof(a6) : sizeof(a))#else p_connect(s, (struct sockaddr *) &a, sizeof(a))#endif ) == SOCKET_ERROR) { err = p_WSAGetLastError(); /* * We expect a potential EWOULDBLOCK here, because the * chances are the front end has done a select for * FD_CONNECT, so that connect() will complete * asynchronously. */ if ( err != WSAEWOULDBLOCK ) { ret->error = winsock_error_string(err); return (Socket) ret; } } else { /* * If we _don't_ get EWOULDBLOCK, the connect has completed * and we should set the socket as writable. */ ret->writable = 1; } add234(sktree, ret); /* We're done with 'addr' now. */ sk_addr_free(addr); return (Socket) ret;}Socket sk_newlistener(char *srcaddr, int port, Plug plug, int local_host_only){ static const struct socket_function_table fn_table = { sk_tcp_plug, sk_tcp_close, sk_tcp_write, sk_tcp_write_oob, sk_tcp_flush, sk_tcp_set_private_ptr, sk_tcp_get_private_ptr, sk_tcp_set_frozen, sk_tcp_socket_error }; SOCKET s;#ifdef IPV6 SOCKADDR_IN6 a6;#endif SOCKADDR_IN a; DWORD err; char *errstr; Actual_Socket ret; int retcode; int on = 1; /* * Create Socket structure. */ ret = snew(struct Socket_tag); ret->fn = &fn_table; ret->error = NULL; ret->plug = plug; bufchain_init(&ret->output_data); ret->writable = 0; /* to start with */ ret->sending_oob = 0; ret->frozen = 0; ret->frozen_readable = 0; ret->localhost_only = local_host_only; ret->pending_error = 0; /* * Open socket. */ s = p_socket(AF_INET, SOCK_STREAM, 0); ret->s = s; if (s == INVALID_SOCKET) { err = p_WSAGetLastError(); ret->error = winsock_error_string(err); return (Socket) ret; } ret->oobinline = 0; p_setsockopt(s, SOL_SOCKET, SO_REUSEADDR, (const char *)&on, sizeof(on));#ifdef IPV6 if (addr->family == AF_INET6) { memset(&a6, 0, sizeof(a6)); a6.sin6_family = AF_INET6; /* FIXME: srcaddr is ignored for IPv6, because I (SGT) don't * know how to do it. :-) */ if (local_host_only) a6.sin6_addr = in6addr_loopback; else a6.sin6_addr = in6addr_any; a6.sin6_port = p_htons(port); } else#endif {
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?