2010-07-15 16:05:23 +02:00
|
|
|
#undef NDEBUG
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <net/ioctl.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/un.h>
|
|
|
|
|
|
|
|
#define DEBUG 0
|
|
|
|
|
|
|
|
static ssize_t _uds_recvmsg_conn(int socket, struct msghdr *msg, int flags);
|
|
|
|
static ssize_t _uds_recvmsg_dgram(int socket, struct msghdr *msg, int flags);
|
|
|
|
|
|
|
|
ssize_t recvmsg(int socket, struct msghdr *msg, int flags)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
int uds_sotype;
|
|
|
|
|
|
|
|
if (msg == NULL) {
|
|
|
|
errno= EFAULT;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
r= ioctl(socket, NWIOGUDSSOTYPE, &uds_sotype);
|
2010-08-30 15:46:44 +02:00
|
|
|
if (r != -1 || (errno != ENOTTY && errno != EBADIOCTL)) {
|
2010-07-15 16:05:23 +02:00
|
|
|
if (r == -1) {
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (uds_sotype == SOCK_DGRAM) {
|
|
|
|
return _uds_recvmsg_dgram(socket, msg, flags);
|
|
|
|
} else {
|
|
|
|
return _uds_recvmsg_conn(socket, msg, flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if DEBUG
|
|
|
|
fprintf(stderr, "recvmsg: not implemented for fd %d\n", socket);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
errno= ENOSYS;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t _uds_recvmsg_conn(int socket, struct msghdr *msg, int flags)
|
|
|
|
{
|
2010-08-30 15:46:44 +02:00
|
|
|
int r, rc;
|
2010-07-15 16:05:23 +02:00
|
|
|
|
|
|
|
if (flags != 0) {
|
|
|
|
#if DEBUG
|
|
|
|
fprintf(stderr, "recvmsg(uds): flags not implemented\n");
|
|
|
|
#endif
|
|
|
|
errno= ENOSYS;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-08-30 15:46:44 +02:00
|
|
|
r= readv(socket, msg->msg_iov, msg->msg_iovlen);
|
2010-07-15 16:05:23 +02:00
|
|
|
|
2010-08-30 15:46:44 +02:00
|
|
|
if (r >= 0 && msg->msg_name && msg->msg_namelen > 0) {
|
2010-07-15 16:05:23 +02:00
|
|
|
getpeername(socket, msg->msg_name, &msg->msg_namelen);
|
|
|
|
}
|
|
|
|
|
2010-08-30 15:46:44 +02:00
|
|
|
/* get control data */
|
|
|
|
if (r >= 0 && msg->msg_control && msg->msg_controllen > 0) {
|
|
|
|
struct msg_control msg_ctrl;
|
|
|
|
|
|
|
|
memset(&msg_ctrl, '\0', sizeof(struct msg_control));
|
|
|
|
msg_ctrl.msg_controllen = msg->msg_controllen;
|
|
|
|
rc = ioctl(socket, NWIOGUDSCTRL, &msg_ctrl);
|
|
|
|
if (rc == -1) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msg_ctrl.msg_controllen <= msg->msg_controllen) {
|
|
|
|
memcpy(msg->msg_control, msg_ctrl.msg_control,
|
|
|
|
msg_ctrl.msg_controllen);
|
|
|
|
msg->msg_controllen = msg_ctrl.msg_controllen;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-15 16:05:23 +02:00
|
|
|
msg->msg_flags = 0;
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ssize_t _uds_recvmsg_dgram(int socket, struct msghdr *msg, int flags)
|
|
|
|
{
|
2010-08-30 15:46:44 +02:00
|
|
|
int r, rc;
|
2010-07-15 16:05:23 +02:00
|
|
|
|
|
|
|
if (flags != 0) {
|
|
|
|
#if DEBUG
|
|
|
|
fprintf(stderr, "recvmsg(uds): flags not implemented\n");
|
|
|
|
#endif
|
|
|
|
errno= ENOSYS;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-08-30 15:46:44 +02:00
|
|
|
r= readv(socket, msg->msg_iov, msg->msg_iovlen);
|
2010-07-15 16:05:23 +02:00
|
|
|
|
2010-08-30 15:46:44 +02:00
|
|
|
if (r >= 0 && msg->msg_name &&
|
|
|
|
msg->msg_namelen >= sizeof(struct sockaddr_un))
|
2010-07-15 16:05:23 +02:00
|
|
|
{
|
2010-08-30 15:46:44 +02:00
|
|
|
rc= ioctl(socket, NWIOGUDSFADDR, msg->msg_name);
|
|
|
|
if (rc == -1) {
|
|
|
|
return rc;
|
|
|
|
}
|
2010-07-15 16:05:23 +02:00
|
|
|
msg->msg_namelen= sizeof(struct sockaddr_un);
|
2010-08-30 15:46:44 +02:00
|
|
|
}
|
2010-07-15 16:05:23 +02:00
|
|
|
|
2010-08-30 15:46:44 +02:00
|
|
|
/* get control data */
|
|
|
|
if (r >= 0 && msg->msg_control && msg->msg_controllen > 0) {
|
|
|
|
struct msg_control msg_ctrl;
|
|
|
|
|
|
|
|
memset(&msg_ctrl, '\0', sizeof(struct msg_control));
|
|
|
|
msg_ctrl.msg_controllen = msg->msg_controllen;
|
|
|
|
rc = ioctl(socket, NWIOGUDSCTRL, &msg_ctrl);
|
|
|
|
if (rc == -1) {
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (msg_ctrl.msg_controllen <= msg->msg_controllen) {
|
|
|
|
memcpy(msg->msg_control, msg_ctrl.msg_control,
|
|
|
|
msg_ctrl.msg_controllen);
|
|
|
|
msg->msg_controllen = msg_ctrl.msg_controllen;
|
|
|
|
}
|
2010-07-15 16:05:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
msg->msg_flags = 0;
|
|
|
|
|
|
|
|
return r;
|
|
|
|
}
|