Socket options SO_RCVBUF, SO_SNDBUF, and TCP_NODELAY.
This commit is contained in:
parent
38804a964b
commit
1991246020
@ -5,21 +5,27 @@
|
|||||||
#include <sys/ioctl.h>
|
#include <sys/ioctl.h>
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
|
#include <netinet/tcp.h>
|
||||||
|
|
||||||
#include <net/gen/in.h>
|
#include <net/gen/in.h>
|
||||||
#include <net/gen/tcp.h>
|
#include <net/gen/tcp.h>
|
||||||
#include <net/gen/tcp_io.h>
|
#include <net/gen/tcp_io.h>
|
||||||
|
#include <net/gen/udp.h>
|
||||||
|
#include <net/gen/udp_io.h>
|
||||||
|
|
||||||
#define DEBUG 0
|
#define DEBUG 0
|
||||||
|
|
||||||
static int _tcp_getsockopt(int socket, int level, int option_name,
|
static int _tcp_getsockopt(int socket, int level, int option_name,
|
||||||
void *_RESTRICT option_value, socklen_t *_RESTRICT option_len);
|
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,
|
int getsockopt(int socket, int level, int option_name,
|
||||||
void *_RESTRICT option_value, socklen_t *_RESTRICT option_len)
|
void *_RESTRICT option_value, socklen_t *_RESTRICT option_len)
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
nwio_tcpopt_t tcpopt;
|
nwio_tcpopt_t tcpopt;
|
||||||
|
nwio_udpopt_t udpopt;
|
||||||
|
|
||||||
r= ioctl(socket, NWIOGTCPOPT, &tcpopt);
|
r= ioctl(socket, NWIOGTCPOPT, &tcpopt);
|
||||||
if (r != -1 || errno != ENOTTY)
|
if (r != -1 || errno != ENOTTY)
|
||||||
@ -33,10 +39,22 @@ int getsockopt(int socket, int level, int option_name,
|
|||||||
option_value, option_len);
|
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
|
#if DEBUG
|
||||||
fprintf(stderr, "getsockopt: not implemented for fd %d\n", socket);
|
fprintf(stderr, "getsockopt: not implemented for fd %d\n", socket);
|
||||||
#endif
|
#endif
|
||||||
errno= ENOSYS;
|
errno= ENOTSOCK;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -55,12 +73,57 @@ static int _tcp_getsockopt(int socket, int level, int option_name,
|
|||||||
*option_len= sizeof(i);
|
*option_len= sizeof(i);
|
||||||
return 0;
|
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
|
#if DEBUG
|
||||||
fprintf(stderr, "_tcp_getsocketopt: level %d, name %d\n",
|
fprintf(stderr, "_tcp_getsocketopt: level %d, name %d\n",
|
||||||
level, option_name);
|
level, option_name);
|
||||||
#endif
|
#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;
|
errno= ENOSYS;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4,21 +4,28 @@
|
|||||||
#include <sys/ioctl.h>
|
#include <sys/ioctl.h>
|
||||||
#include <sys/socket.h>
|
#include <sys/socket.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
|
#include <netinet/tcp.h>
|
||||||
|
|
||||||
#include <net/gen/in.h>
|
#include <net/gen/in.h>
|
||||||
#include <net/gen/tcp.h>
|
#include <net/gen/tcp.h>
|
||||||
#include <net/gen/tcp_io.h>
|
#include <net/gen/tcp_io.h>
|
||||||
|
#include <net/gen/udp.h>
|
||||||
|
#include <net/gen/udp_io.h>
|
||||||
|
|
||||||
#define DEBUG 0
|
#define DEBUG 0
|
||||||
|
|
||||||
static int _tcp_setsockopt(int socket, int level, int option_name,
|
static int _tcp_setsockopt(int socket, int level, int option_name,
|
||||||
const void *option_value, socklen_t option_len);
|
const void *option_value, socklen_t option_len);
|
||||||
|
|
||||||
|
static int _udp_setsockopt(int socket, int level, int option_name,
|
||||||
|
const void *option_value, socklen_t option_len);
|
||||||
|
|
||||||
int setsockopt(int socket, int level, int option_name,
|
int setsockopt(int socket, int level, int option_name,
|
||||||
const void *option_value, socklen_t option_len)
|
const void *option_value, socklen_t option_len)
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
nwio_tcpopt_t tcpopt;
|
nwio_tcpopt_t tcpopt;
|
||||||
|
nwio_udpopt_t udpopt;
|
||||||
|
|
||||||
r= ioctl(socket, NWIOGTCPOPT, &tcpopt);
|
r= ioctl(socket, NWIOGTCPOPT, &tcpopt);
|
||||||
if (r != -1 || errno != ENOTTY)
|
if (r != -1 || errno != ENOTTY)
|
||||||
@ -32,10 +39,22 @@ int setsockopt(int socket, int level, int option_name,
|
|||||||
option_value, option_len);
|
option_value, option_len);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
r= ioctl(socket, NWIOGUDPOPT, &udpopt);
|
||||||
|
if (r != -1 || errno != ENOTTY)
|
||||||
|
{
|
||||||
|
if (r == -1)
|
||||||
|
{
|
||||||
|
/* Bad file descriptor */
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return _udp_setsockopt(socket, level, option_name,
|
||||||
|
option_value, option_len);
|
||||||
|
}
|
||||||
|
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
fprintf(stderr, "setsockopt: not implemented for fd %d\n", socket);
|
fprintf(stderr, "setsockopt: not implemented for fd %d\n", socket);
|
||||||
#endif
|
#endif
|
||||||
errno= ENOSYS;
|
errno= ENOTSOCK;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -62,6 +81,64 @@ static int _tcp_setsockopt(int socket, int level, int option_name,
|
|||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
if (level == SOL_SOCKET && option_name == SO_RCVBUF)
|
||||||
|
{
|
||||||
|
if (option_len != sizeof(i))
|
||||||
|
{
|
||||||
|
errno= EINVAL;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
i= *(int *)option_value;
|
||||||
|
if (i > 32*1024)
|
||||||
|
{
|
||||||
|
/* The receive buffer is limited to 32K at the moment.
|
||||||
|
*/
|
||||||
|
errno= ENOSYS;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
/* There is no way to reduce the receive buffer, do we have to
|
||||||
|
* let this call fail for smaller buffers?
|
||||||
|
*/
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (level == SOL_SOCKET && option_name == SO_SNDBUF)
|
||||||
|
{
|
||||||
|
if (option_len != sizeof(i))
|
||||||
|
{
|
||||||
|
errno= EINVAL;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
i= *(int *)option_value;
|
||||||
|
if (i > 32*1024)
|
||||||
|
{
|
||||||
|
/* The send buffer is limited to 32K at the moment.
|
||||||
|
*/
|
||||||
|
errno= ENOSYS;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
/* There is no way to reduce the send buffer, do we have to
|
||||||
|
* let this call fail for smaller buffers?
|
||||||
|
*/
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (level == IPPROTO_TCP && option_name == TCP_NODELAY)
|
||||||
|
{
|
||||||
|
if (option_len != sizeof(i))
|
||||||
|
{
|
||||||
|
errno= EINVAL;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
i= *(int *)option_value;
|
||||||
|
if (i)
|
||||||
|
{
|
||||||
|
/* At the moment there is no way to turn on
|
||||||
|
* nodelay.
|
||||||
|
*/
|
||||||
|
errno= ENOSYS;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
#if DEBUG
|
#if DEBUG
|
||||||
fprintf(stderr, "_tcp_setsocketopt: level %d, name %d\n",
|
fprintf(stderr, "_tcp_setsocketopt: level %d, name %d\n",
|
||||||
level, option_name);
|
level, option_name);
|
||||||
@ -71,3 +148,17 @@ static int _tcp_setsockopt(int socket, int level, int option_name,
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int _udp_setsockopt(int socket, int level, int option_name,
|
||||||
|
const void *option_value, socklen_t option_len)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
#if DEBUG
|
||||||
|
fprintf(stderr, "_udp_setsocketopt: level %d, name %d\n",
|
||||||
|
level, option_name);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
errno= ENOSYS;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user