270 lines
5.1 KiB
C
270 lines
5.1 KiB
C
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <stdio.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_setsockopt(int socket, int level, int option_name,
|
|
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);
|
|
|
|
static int _uds_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,
|
|
const void *option_value, socklen_t option_len)
|
|
{
|
|
int r;
|
|
nwio_tcpopt_t tcpopt;
|
|
nwio_udpopt_t udpopt;
|
|
struct sockaddr_un uds_addr;
|
|
|
|
r= ioctl(socket, NWIOGTCPOPT, &tcpopt);
|
|
if (r != -1 || (errno != ENOTTY && errno != EBADIOCTL))
|
|
{
|
|
if (r == -1)
|
|
{
|
|
/* Bad file descriptor */
|
|
return -1;
|
|
}
|
|
return _tcp_setsockopt(socket, level, option_name,
|
|
option_value, option_len);
|
|
}
|
|
|
|
r= ioctl(socket, NWIOGUDPOPT, &udpopt);
|
|
if (r != -1 || (errno != ENOTTY && errno != EBADIOCTL))
|
|
{
|
|
if (r == -1)
|
|
{
|
|
/* Bad file descriptor */
|
|
return -1;
|
|
}
|
|
return _udp_setsockopt(socket, level, option_name,
|
|
option_value, option_len);
|
|
}
|
|
|
|
r= ioctl(socket, NWIOGUDSADDR, &uds_addr);
|
|
if (r != -1 || (errno != ENOTTY && errno != EBADIOCTL))
|
|
{
|
|
if (r == -1)
|
|
{
|
|
/* Bad file descriptor */
|
|
return -1;
|
|
}
|
|
return _uds_setsockopt(socket, level, option_name,
|
|
option_value, option_len);
|
|
}
|
|
|
|
|
|
#if DEBUG
|
|
fprintf(stderr, "setsockopt: not implemented for fd %d\n", socket);
|
|
#endif
|
|
errno= ENOTSOCK;
|
|
return -1;
|
|
}
|
|
|
|
static int _tcp_setsockopt(int socket, int level, int option_name,
|
|
const void *option_value, socklen_t option_len)
|
|
{
|
|
int i;
|
|
|
|
if (level == SOL_SOCKET && option_name == SO_REUSEADDR)
|
|
{
|
|
if (option_len != sizeof(i))
|
|
{
|
|
errno= EINVAL;
|
|
return -1;
|
|
}
|
|
i= *(int *)option_value;
|
|
if (!i)
|
|
{
|
|
/* At the moment there is no way to turn off
|
|
* reusing addresses.
|
|
*/
|
|
errno= ENOSYS;
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
if (level == SOL_SOCKET && option_name == SO_KEEPALIVE)
|
|
{
|
|
if (option_len != sizeof(i))
|
|
{
|
|
errno= EINVAL;
|
|
return -1;
|
|
}
|
|
i= *(int *)option_value;
|
|
if (!i)
|
|
{
|
|
/* At the moment there is no way to turn off
|
|
* keepalives.
|
|
*/
|
|
errno= ENOSYS;
|
|
return -1;
|
|
}
|
|
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
|
|
fprintf(stderr, "_tcp_setsocketopt: level %d, name %d\n",
|
|
level, option_name);
|
|
#endif
|
|
|
|
errno= ENOSYS;
|
|
return -1;
|
|
}
|
|
|
|
static int _udp_setsockopt(int socket, int level, int option_name,
|
|
const void *option_value, socklen_t option_len)
|
|
{
|
|
#if DEBUG
|
|
fprintf(stderr, "_udp_setsocketopt: level %d, name %d\n",
|
|
level, option_name);
|
|
#endif
|
|
|
|
errno= ENOSYS;
|
|
return -1;
|
|
}
|
|
|
|
|
|
static int _uds_setsockopt(int socket, int level, int option_name,
|
|
const void *option_value, socklen_t option_len)
|
|
{
|
|
int i;
|
|
size_t size;
|
|
|
|
if (level == SOL_SOCKET && option_name == SO_RCVBUF)
|
|
{
|
|
if (option_len != sizeof(size))
|
|
{
|
|
errno= EINVAL;
|
|
return -1;
|
|
}
|
|
size= *(size_t *)option_value;
|
|
return ioctl(socket, NWIOSUDSRCVBUF, &size);
|
|
}
|
|
|
|
if (level == SOL_SOCKET && option_name == SO_SNDBUF)
|
|
{
|
|
if (option_len != sizeof(size))
|
|
{
|
|
errno= EINVAL;
|
|
return -1;
|
|
}
|
|
size= *(size_t *)option_value;
|
|
return ioctl(socket, NWIOSUDSSNDBUF, &size);
|
|
}
|
|
|
|
if (level == SOL_SOCKET && option_name == SO_REUSEADDR)
|
|
{
|
|
if (option_len != sizeof(i))
|
|
{
|
|
errno= EINVAL;
|
|
return -1;
|
|
}
|
|
i= *(int *)option_value;
|
|
if (!i)
|
|
{
|
|
/* At the moment there is no way to turn off
|
|
* reusing addresses.
|
|
*/
|
|
errno= ENOSYS;
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
if (level == SOL_SOCKET && option_name == SO_PASSCRED)
|
|
{
|
|
if (option_len != sizeof(i))
|
|
{
|
|
errno= EINVAL;
|
|
return -1;
|
|
}
|
|
i= *(int *)option_value;
|
|
if (!i)
|
|
{
|
|
/* credentials can always be received. */
|
|
errno= ENOSYS;
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#if DEBUG
|
|
fprintf(stderr, "_uds_setsocketopt: level %d, name %d\n",
|
|
level, option_name);
|
|
#endif
|
|
|
|
errno= ENOSYS;
|
|
return -1;
|
|
}
|