mirror of
https://github.com/gnif/LookingGlass.git
synced 2024-11-25 23:07:18 +00:00
281 lines
5.3 KiB
C
281 lines
5.3 KiB
C
#include "ivshmem.h"
|
|
|
|
#define DEBUG
|
|
#define DEBUG_IVSHMEM
|
|
#include "debug.h"
|
|
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
|
|
#include <sys/stat.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/un.h>
|
|
|
|
#include <sys/mman.h>
|
|
|
|
struct IVSHMEMServer
|
|
{
|
|
int64_t version;
|
|
int64_t clientID;
|
|
int sharedFD;
|
|
};
|
|
|
|
struct IVSHMEM
|
|
{
|
|
bool connected;
|
|
int socket;
|
|
struct IVSHMEMServer server;
|
|
|
|
off_t mapSize;
|
|
void * map;
|
|
};
|
|
|
|
struct IVSHMEM ivshmem =
|
|
{
|
|
.connected = false,
|
|
.socket = -1
|
|
};
|
|
|
|
// ============================================================================
|
|
// internal functions
|
|
|
|
void ivshmem_cleanup();
|
|
bool ivshmem_read(void * buffer, const ssize_t size);
|
|
bool ivshmem_read_msg(int64_t * index, int *fd);
|
|
|
|
// ============================================================================
|
|
|
|
bool ivshmem_connect(const char * unix_socket)
|
|
{
|
|
ivshmem.socket = socket(AF_UNIX, SOCK_STREAM, 0);
|
|
if (ivshmem.socket < 0)
|
|
{
|
|
DEBUG_ERROR("socket creation failed");
|
|
return false;
|
|
}
|
|
|
|
struct sockaddr_un addr;
|
|
addr.sun_family = AF_UNIX;
|
|
strncpy(addr.sun_path, unix_socket, sizeof(addr.sun_path));
|
|
|
|
if (connect(ivshmem.socket, (struct sockaddr *)&addr, sizeof(struct sockaddr_un)) < 0)
|
|
{
|
|
DEBUG_ERROR("socket connect failed");
|
|
ivshmem_cleanup();
|
|
return false;
|
|
}
|
|
|
|
ivshmem.connected = true;
|
|
|
|
if (!ivshmem_read(&ivshmem.server.version, sizeof(ivshmem.server.version)))
|
|
{
|
|
DEBUG_ERROR("read protocol version failed");
|
|
ivshmem_cleanup();
|
|
return false;
|
|
}
|
|
|
|
if (ivshmem.server.version != 0)
|
|
{
|
|
DEBUG_ERROR("unsupported protocol version %ld", ivshmem.server.version);
|
|
ivshmem_cleanup();
|
|
return false;
|
|
}
|
|
|
|
if (!ivshmem_read(&ivshmem.server.clientID, sizeof(ivshmem.server.clientID)))
|
|
{
|
|
DEBUG_ERROR("read client id failed");
|
|
ivshmem_cleanup();
|
|
return false;
|
|
}
|
|
|
|
DEBUG_PROTO("Protocol : %ld", ivshmem.server.version );
|
|
DEBUG_PROTO("Client ID: %ld", ivshmem.server.clientID);
|
|
|
|
if (!ivshmem_read_msg(NULL, &ivshmem.server.sharedFD))
|
|
{
|
|
DEBUG_ERROR("failed to read shared memory file descriptor");
|
|
ivshmem_cleanup();
|
|
return false;
|
|
}
|
|
|
|
struct stat stat;
|
|
if (fstat(ivshmem.server.sharedFD, &stat) != 0)
|
|
{
|
|
DEBUG_ERROR("failed to stat shared FD");
|
|
ivshmem_cleanup();
|
|
return false;
|
|
}
|
|
|
|
ivshmem.mapSize = stat.st_size;
|
|
|
|
DEBUG_INFO("RAM Size : %ld", ivshmem.mapSize);
|
|
ivshmem.map = mmap(
|
|
NULL,
|
|
stat.st_size,
|
|
PROT_READ | PROT_WRITE,
|
|
MAP_SHARED,
|
|
ivshmem.server.sharedFD,
|
|
0);
|
|
|
|
if (!ivshmem.map)
|
|
{
|
|
DEBUG_ERROR("failed to map memory");
|
|
ivshmem_cleanup();
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// ============================================================================
|
|
|
|
void ivshmem_cleanup()
|
|
{
|
|
if (ivshmem.map)
|
|
munmap(ivshmem.map, ivshmem.mapSize);
|
|
ivshmem.map = NULL;
|
|
ivshmem.mapSize = 0;
|
|
|
|
if (ivshmem.socket >= 0)
|
|
{
|
|
close(ivshmem.socket);
|
|
ivshmem.socket = -1;
|
|
}
|
|
|
|
ivshmem.connected = false;
|
|
}
|
|
|
|
// ============================================================================
|
|
|
|
void ivshmem_close()
|
|
{
|
|
if (!ivshmem.connected)
|
|
{
|
|
DEBUG_WARN("socket not connected");
|
|
return;
|
|
}
|
|
|
|
ivshmem_cleanup();
|
|
}
|
|
|
|
// ============================================================================
|
|
|
|
bool ivshmem_read(void * buffer, const ssize_t size)
|
|
{
|
|
if (!ivshmem.connected)
|
|
{
|
|
DEBUG_ERROR("not connected");
|
|
return false;
|
|
}
|
|
|
|
ssize_t len = read(ivshmem.socket, buffer, size);
|
|
if (len != size)
|
|
{
|
|
DEBUG_ERROR("incomplete read");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// ============================================================================
|
|
|
|
bool ivshmem_read_msg(int64_t * index, int * fd)
|
|
{
|
|
if (!ivshmem.connected)
|
|
{
|
|
DEBUG_ERROR("not connected");
|
|
return false;
|
|
}
|
|
|
|
struct msghdr msg;
|
|
struct iovec iov[1];
|
|
union {
|
|
struct cmsghdr cmsg;
|
|
char control[CMSG_SPACE(sizeof(int))];
|
|
} msg_control;
|
|
|
|
int64_t tmp;
|
|
if (!index)
|
|
index = &tmp;
|
|
|
|
iov[0].iov_base = index;
|
|
iov[0].iov_len = sizeof(*index);
|
|
|
|
memset(&msg, 0, sizeof(msg));
|
|
msg.msg_iov = iov;
|
|
msg.msg_iovlen = 1;
|
|
msg.msg_control = &msg_control;
|
|
msg.msg_controllen = sizeof(msg_control);
|
|
|
|
int ret = recvmsg(ivshmem.socket, &msg, 0);
|
|
if (ret < sizeof(*index))
|
|
{
|
|
DEBUG_ERROR("failed ot read message\n");
|
|
return false;
|
|
}
|
|
|
|
if (ret == 0)
|
|
{
|
|
DEBUG_ERROR("lost connetion to server\n");
|
|
return false;
|
|
}
|
|
|
|
if (!fd)
|
|
return true;
|
|
|
|
struct cmsghdr *cmsg;
|
|
for(cmsg = CMSG_FIRSTHDR(&msg); cmsg; cmsg = CMSG_NXTHDR(&msg, cmsg))
|
|
{
|
|
if (cmsg->cmsg_len != CMSG_LEN(sizeof(int)) ||
|
|
cmsg->cmsg_level != SOL_SOCKET ||
|
|
cmsg->cmsg_type != SCM_RIGHTS)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
memcpy(fd, CMSG_DATA(cmsg), sizeof(*fd));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// ============================================================================
|
|
|
|
void * ivshmem_get_map()
|
|
{
|
|
if (!ivshmem.connected)
|
|
{
|
|
DEBUG_ERROR("not connected");
|
|
return NULL;
|
|
}
|
|
|
|
if (!ivshmem.map)
|
|
{
|
|
DEBUG_ERROR("not mapped");
|
|
return NULL;
|
|
}
|
|
|
|
return ivshmem.map;
|
|
}
|
|
|
|
// ============================================================================
|
|
|
|
size_t ivshmem_get_map_size()
|
|
{
|
|
if (!ivshmem.connected)
|
|
{
|
|
DEBUG_ERROR("not connected");
|
|
return 0;
|
|
}
|
|
|
|
if (!ivshmem.map)
|
|
{
|
|
DEBUG_ERROR("not mapped");
|
|
return 0;
|
|
}
|
|
|
|
return ivshmem.mapSize;
|
|
} |