142 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			142 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include <assert.h>
 | |
| #include <errno.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <sys/ioctl.h>
 | |
| #include <sys/socket.h>
 | |
| #include <sys/types.h>
 | |
| #include <netinet/tcp.h>
 | |
| 
 | |
| #include <net/gen/in.h>
 | |
| #include <net/gen/tcp.h>
 | |
| #include <net/gen/tcp_io.h>
 | |
| #include <net/gen/udp.h>
 | |
| #include <net/gen/udp_io.h>
 | |
| 
 | |
| #define DEBUG 0
 | |
| 
 | |
| static int _tcp_getsockopt(int socket, int level, int option_name,
 | |
| 	void *_RESTRICT option_value, socklen_t *_RESTRICT option_len);
 | |
| static int _udp_getsockopt(int socket, int level, int option_name,
 | |
| 	void *_RESTRICT option_value, socklen_t *_RESTRICT option_len);
 | |
| 
 | |
| int getsockopt(int socket, int level, int option_name,
 | |
|         void *_RESTRICT option_value, socklen_t *_RESTRICT option_len)
 | |
| {
 | |
| 	int r;
 | |
| 	nwio_tcpopt_t tcpopt;
 | |
| 	nwio_udpopt_t udpopt;
 | |
| 
 | |
| 	r= ioctl(socket, NWIOGTCPOPT, &tcpopt);
 | |
| 	if (r != -1 || errno != ENOTTY)
 | |
| 	{
 | |
| 		if (r == -1)
 | |
| 		{
 | |
| 			/* Bad file descriptor */
 | |
| 			return -1;
 | |
| 		}
 | |
| 		return _tcp_getsockopt(socket, level, option_name,
 | |
| 			option_value, option_len);
 | |
| 	}
 | |
| 
 | |
| 	r= ioctl(socket, NWIOGUDPOPT, &udpopt);
 | |
| 	if (r != -1 || errno != ENOTTY)
 | |
| 	{
 | |
| 		if (r == -1)
 | |
| 		{
 | |
| 			/* Bad file descriptor */
 | |
| 			return -1;
 | |
| 		}
 | |
| 		return _udp_getsockopt(socket, level, option_name,
 | |
| 			option_value, option_len);
 | |
| 	}
 | |
| 
 | |
| #if DEBUG
 | |
| 	fprintf(stderr, "getsockopt: not implemented for fd %d\n", socket);
 | |
| #endif
 | |
| 	errno= ENOTSOCK;
 | |
| 	return -1;
 | |
| }
 | |
| 
 | |
| static int _tcp_getsockopt(int socket, int level, int option_name,
 | |
| 	void *_RESTRICT option_value, socklen_t *_RESTRICT option_len)
 | |
| {
 | |
| 	int i, r, err;
 | |
| 
 | |
| 	if (level == SOL_SOCKET && option_name == SO_KEEPALIVE)
 | |
| 	{
 | |
| 		i= 1;	/* Keepalive is always on */
 | |
| 		if (*option_len < sizeof(i))
 | |
| 			memcpy(option_value, &i, *option_len);
 | |
| 		else
 | |
| 			memcpy(option_value, &i, sizeof(i));
 | |
| 		*option_len= sizeof(i);
 | |
| 		return 0;
 | |
| 	}
 | |
| 	if (level == SOL_SOCKET && option_name == SO_ERROR)
 | |
| 	{
 | |
| 		r= ioctl(socket, NWIOTCPGERROR, &err);
 | |
| 		if (r != 0)
 | |
| 			return r;
 | |
| 		if (*option_len < sizeof(err))
 | |
| 			memcpy(option_value, &err, *option_len);
 | |
| 		else
 | |
| 			memcpy(option_value, &err, sizeof(err));
 | |
| 		*option_len= sizeof(err);
 | |
| 		return 0;
 | |
| 	}
 | |
| 	if (level == SOL_SOCKET && option_name == SO_RCVBUF)
 | |
| 	{
 | |
| 		i= 32*1024;	/* Receive buffer in the current
 | |
| 				 * implementation
 | |
| 				 */
 | |
| 		if (*option_len < sizeof(i))
 | |
| 			memcpy(option_value, &i, *option_len);
 | |
| 		else
 | |
| 			memcpy(option_value, &i, sizeof(i));
 | |
| 		*option_len= sizeof(i);
 | |
| 		return 0;
 | |
| 	}
 | |
| 	if (level == SOL_SOCKET && option_name == SO_SNDBUF)
 | |
| 	{
 | |
| 		i= 32*1024;	/* Send buffer in the current implementation */
 | |
| 		if (*option_len < sizeof(i))
 | |
| 			memcpy(option_value, &i, *option_len);
 | |
| 		else
 | |
| 			memcpy(option_value, &i, sizeof(i));
 | |
| 		*option_len= sizeof(i);
 | |
| 		return 0;
 | |
| 	}
 | |
| 	if (level == IPPROTO_TCP && option_name == TCP_NODELAY)
 | |
| 	{
 | |
| 		i= 0;	/* nodelay is always off */
 | |
| 		if (*option_len < sizeof(i))
 | |
| 			memcpy(option_value, &i, *option_len);
 | |
| 		else
 | |
| 			memcpy(option_value, &i, sizeof(i));
 | |
| 		*option_len= sizeof(i);
 | |
| 		return 0;
 | |
| 	}
 | |
| #if DEBUG
 | |
| 	fprintf(stderr, "_tcp_getsocketopt: level %d, name %d\n",
 | |
| 		level, option_name);
 | |
| #endif
 | |
| 
 | |
| 	errno= ENOPROTOOPT;
 | |
| 	return -1;
 | |
| }
 | |
| 
 | |
| static int _udp_getsockopt(int socket, int level, int option_name,
 | |
| 	void *_RESTRICT option_value, socklen_t *_RESTRICT option_len)
 | |
| {
 | |
| 	int i;
 | |
| 
 | |
| #if DEBUG
 | |
| 	fprintf(stderr, "_udp_getsocketopt: level %d, name %d\n",
 | |
| 		level, option_name);
 | |
| #endif
 | |
| 
 | |
| 	errno= ENOSYS;
 | |
| 	return -1;
 | |
| }
 | 
