ecore-drm: Add code for SPARTACUS !! ;)

@ferature: This adds code for the ecore-drm auth process to open
restricted inputs/cards/etc by the user.

Signed-off-by: Chris Michael <cp.michael@samsung.com>
This commit is contained in:
Chris Michael 2014-03-06 11:15:29 +00:00
parent 8e30a4eeab
commit 3d5f93a9cf
1 changed files with 407 additions and 0 deletions

View File

@ -0,0 +1,407 @@
#ifdef HAVE_CONFIG_H
# include "config.h"
#endif
/* standard headers */
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <limits.h>
#include <fcntl.h>
#include <errno.h>
#include <signal.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/epoll.h>
#include <sys/ioctl.h>
/* #include <syslog.h> */
/* #include <pwd.h> */
/* #include <linux/major.h> */
/* #include <linux/vt.h> */
/* #include <linux/kd.h> */
#include <xf86drm.h>
#include <xf86drmMode.h>
#include <drm_fourcc.h>
#include <Eina.h>
#include <Ecore_Drm.h>
#define RIGHTS_LEN CMSG_LEN(sizeof(int))
#define IOVSET(_iov, _addr, _len) \
(_iov)->iov_base = (void *)(_addr); \
(_iov)->iov_len = (_len);
/* local prototypes */
static int _send_msg(int opcode, int fd, void *data, size_t bytes);
/* local variables */
static struct cmsghdr *cmsgptr = NULL;
static int _read_fd = -1;
static int _write_fd = -1;
static int
_open_device(const char *device)
{
int fd = -1, ret = ECORE_DRM_OP_SUCCESS;
if (!device)
{
ret = ECORE_DRM_OP_FAILURE;
_send_msg(ECORE_DRM_OP_DEVICE_OPEN, fd, &ret, sizeof(int));
return ret;
}
fprintf(stderr, "Launcher Trying to Open Device: %s\n", device);
if ((fd = open(device, O_RDWR | O_NONBLOCK)) < 0)
{
fprintf(stderr, "Failed to Open Device: %s: %m\n", device);
ret = ECORE_DRM_OP_FAILURE;
}
else
fprintf(stderr, "Launcher Opened Device: %s %d\n", device, fd);
_send_msg(ECORE_DRM_OP_DEVICE_OPEN, fd, &ret, sizeof(int));
return ret;
}
static int
_close_device(int fd)
{
int ret = ECORE_DRM_OP_SUCCESS;
if (!fd)
{
ret = ECORE_DRM_OP_FAILURE;
_send_msg(ECORE_DRM_OP_DEVICE_CLOSE, fd, &ret, sizeof(int));
return ret;
}
close(fd);
_send_msg(ECORE_DRM_OP_DEVICE_CLOSE, fd, &ret, sizeof(int));
return ret;
}
static int
_open_tty(const char *name)
{
int fd = -1, ret = ECORE_DRM_OP_SUCCESS;
/* struct stat st; */
if (!name) goto fail;
fprintf(stderr, "Launcher Trying to Open Tty: %s\n", name);
if ((fd = open(name, O_RDWR | O_NOCTTY)) < 0)
{
fprintf(stderr, "Failed to Open Tty: %s: %m\n", name);
goto fail;
}
else
fprintf(stderr, "Launcher Opened Tty: %s %d\n", name, fd);
/* if ((fstat(fd, &st) == -1) || */
/* (major(st.st_rdev) != TTY_MAJOR) || (minor(st.st_rdev) == 0)) */
/* { */
/* fprintf(stderr, "%d is Not a Tty\n", fd); */
/* goto fail; */
/* } */
_send_msg(ECORE_DRM_OP_TTY_OPEN, fd, &ret, sizeof(int));
return ret;
fail:
if (fd > -1) close(fd);
fd = -1;
ret = ECORE_DRM_OP_FAILURE;
_send_msg(ECORE_DRM_OP_DEVICE_OPEN, fd, &ret, sizeof(int));
return ret;
}
static int
_drop_master(int fd)
{
int ret = ECORE_DRM_OP_SUCCESS;
fprintf(stderr, "Drop Master: %d\n", fd);
if (drmDropMaster(fd) != 0)
{
ret = ECORE_DRM_OP_FAILURE;
fprintf(stderr, "\tFailed to drop master: %m\n");
}
_send_msg(ECORE_DRM_OP_DEVICE_MASTER_DROP, fd, &ret, sizeof(int));
close(fd);
return ret;
}
static int
_set_master(int fd)
{
int ret = ECORE_DRM_OP_SUCCESS;
fprintf(stderr, "Set Master: %d\n", fd);
if (drmSetMaster(fd) != 0)
{
ret = ECORE_DRM_OP_FAILURE;
fprintf(stderr, "\tFailed to set master: %m\n");
}
_send_msg(ECORE_DRM_OP_DEVICE_MASTER_SET, fd, &ret, sizeof(int));
close(fd);
return ret;
}
static int
_read_fd_get(void)
{
char *ev, *end;
int fd = -1, flags = -1;
if (!(ev = getenv("ECORE_DRM_LAUNCHER_SOCKET_READ")))
return -1;
fd = strtol(ev, &end, 0);
if (*end != '\0') return -1;
flags = fcntl(fd, F_GETFD);
if (flags < 0) return -1;
fprintf(stderr, "Got Read FD: %d\n", fd);
return fd;
}
static int
_write_fd_get(void)
{
char *ev, *end;
int fd = -1, flags = -1;
if (!(ev = getenv("ECORE_DRM_LAUNCHER_SOCKET_WRITE")))
return -1;
fd = strtol(ev, &end, 0);
if (*end != '\0') return -1;
flags = fcntl(fd, F_GETFD);
if (flags < 0) return -1;
fprintf(stderr, "Got Write FD: %d\n", fd);
return fd;
}
static int
_send_msg(int opcode, int fd, void *data, size_t bytes)
{
Ecore_Drm_Message dmsg;
struct iovec iov[2];
struct msghdr msg;
ssize_t size;
/* send a message to the calling process */
/* 'fd' is the fd to send */
memset(&dmsg, 0, sizeof(dmsg));
IOVSET(iov + 0, &dmsg, sizeof(dmsg));
IOVSET(iov + 1, &data, bytes);
dmsg.opcode = opcode;
dmsg.size = bytes;
msg.msg_name = NULL;
msg.msg_namelen = 0;
msg.msg_iov = iov;
msg.msg_iovlen = 2;
msg.msg_flags = 0;
if ((!cmsgptr) && (!(cmsgptr = malloc(RIGHTS_LEN))))
return -1;
cmsgptr->cmsg_level = SOL_SOCKET;
cmsgptr->cmsg_type = SCM_RIGHTS;
cmsgptr->cmsg_len = RIGHTS_LEN;
msg.msg_control = cmsgptr;
msg.msg_controllen = RIGHTS_LEN;
fprintf(stderr, "Launcher Sending FD: %d\n", fd);
*((int *)CMSG_DATA(cmsgptr)) = fd;
errno = 0;
size = sendmsg(_write_fd, &msg, MSG_EOR);
if (errno != 0)
{
fprintf(stderr, "Failed to send message: %s", strerror(errno));
return -1;
}
fprintf(stderr, "Launcher Wrote %li to %d\n", size, _write_fd);
return size;
}
static int
_recv_msg(void)
{
int fd = -1;
Ecore_Drm_Message dmsg;
struct iovec iov[2];
struct msghdr msg;
struct cmsghdr *cmsg = NULL;
char data[BUFSIZ];
ssize_t size;
fprintf(stderr, "Received Message\n");
memset(&dmsg, 0, sizeof(dmsg));
memset(&data, 0, sizeof(data));
IOVSET(iov + 0, &dmsg, sizeof(dmsg));
IOVSET(iov + 1, &data, sizeof(data));
msg.msg_name = NULL;
msg.msg_namelen = 0;
msg.msg_iov = iov;
msg.msg_iovlen = 2;
msg.msg_flags = 0;
if ((!cmsgptr) && (!(cmsgptr = malloc(RIGHTS_LEN))))
return -1;
msg.msg_control = cmsgptr;
msg.msg_controllen = RIGHTS_LEN;
errno = 0;
size = recvmsg(_read_fd, &msg, 0);
if (errno != 0)
{
fprintf(stderr, "Failed to receive message: %m\n");
return -1;
}
fprintf(stderr, "\tReceived %li bytes from %d\n", size, _read_fd);
for (cmsg = CMSG_FIRSTHDR(&msg); cmsg != NULL;
cmsg = CMSG_NXTHDR(&msg, cmsg))
{
if (cmsg->cmsg_level != SOL_SOCKET)
continue;
switch (cmsg->cmsg_type)
{
case SCM_RIGHTS:
fd = *((int *)CMSG_DATA(cmsg));
switch (dmsg.opcode)
{
case ECORE_DRM_OP_DEVICE_OPEN:
fprintf(stderr, "Open Device: %s\n", (char *)data);
_open_device((char *)data);
break;
case ECORE_DRM_OP_DEVICE_CLOSE:
fprintf(stderr, "Close Device: %d\n", fd);
_close_device(fd);
case ECORE_DRM_OP_TTY_OPEN:
fprintf(stderr, "Open Tty: %s\n", (char *)data);
_open_tty((char *)data);
break;
case ECORE_DRM_OP_DEVICE_MASTER_DROP:
fprintf(stderr, "Drop Master: %d\n", fd);
_drop_master(fd);
break;
case ECORE_DRM_OP_DEVICE_MASTER_SET:
fprintf(stderr, "Set Master\n");
_set_master(fd);
break;
default:
fprintf(stderr, "Unhandled Opcode: %d\n", dmsg.opcode);
break;
}
break;
default:
fprintf(stderr, "Unhandled message type: %d\n", cmsg->cmsg_type);
return -1;
break;
}
}
return size;
}
int
main(int argc EINA_UNUSED, char **argv EINA_UNUSED)
{
struct epoll_event ev, events[1];
int ret, i, _epoll_fd = -1;
setvbuf(stdout, NULL, _IONBF, 0);
setvbuf(stderr, NULL, _IONBF, 0);
fprintf(stderr, "Spartacus Is Alive\n");
_read_fd = _read_fd_get();
if (_read_fd < 0) return EXIT_FAILURE;
_write_fd = _write_fd_get();
if (_write_fd < 0) return EXIT_FAILURE;
fprintf(stderr, "Creating Epoll\n");
_epoll_fd = epoll_create(1);
memset(&ev, 0, sizeof(ev));
ev.events = EPOLLIN;
ev.data.fd = _read_fd;
if (epoll_ctl(_epoll_fd, EPOLL_CTL_ADD, _read_fd, &ev) < 0)
{
return EXIT_FAILURE;
}
memset(&events, 0, sizeof(events));
while (1)
{
ret = epoll_wait(_epoll_fd, events, sizeof(events) / sizeof(struct epoll_event), -1);
if (ret < 0)
{
fprintf(stderr, "Epoll Failed: %m\n");
return EXIT_FAILURE;
}
for (i = 0; i < ret; i++)
{
fprintf(stderr, "Epoll Event on: %d\n", events[i].data.fd);
if (events[i].data.fd != _read_fd) continue;
if (events[i].events & EPOLLIN)
{
fprintf(stderr, "Epoll Data In\n");
_recv_msg();
}
else if (events[i].events & EPOLLERR)
{
fprintf(stderr, "Epoll Data Error\n");
}
}
}
fprintf(stderr, "Spartacus Is Dead\n");
return EXIT_SUCCESS;
}