#include "message.h" #include "error.h" #include "usocket.h" #include void ipc_message_print (const struct ipc_message *m) { assert (m != NULL); #if defined(IPC_WITH_ERRORS) && IPC_WITH_ERRORS > 2 printf ("msg: type %d len %d\n", m->type, m->length); #endif } int32_t ipc_message_format_read (struct ipc_message *m, const char *buf, ssize_t msize) { assert (m != NULL); assert (buf != NULL); assert (msize <= IPC_MAX_MESSAGE_SIZE); if (m == NULL) return -1; m->type = buf[0]; memcpy (&m->length, buf+1, sizeof m->length); assert (m->length <= IPC_MAX_MESSAGE_SIZE); #if defined(IPC_WITH_ERRORS) && IPC_WITH_ERRORS > 2 printf ("type %d, paylen = %u, total = %lu\n", m->type, m->length, msize); #endif assert (m->length == msize - IPC_HEADER_SIZE || m->length == 0); if (m->payload != NULL) { free (m->payload); m->payload = NULL; } if (m->length > 0) { m->payload = malloc (m->length); memcpy (m->payload, buf+IPC_HEADER_SIZE, m->length); } return 0; } int32_t ipc_message_format_write (const struct ipc_message *m, char **buf, ssize_t *msize) { assert (m != NULL); assert (buf != NULL); assert (msize != NULL); assert (m->length <= IPC_MAX_MESSAGE_SIZE); if (m == NULL) return -1; if (buf == NULL) return -2; if (msize == NULL) return -3; if (*buf == NULL) { *buf = malloc (IPC_HEADER_SIZE + m->length); memset (*buf, 0, IPC_HEADER_SIZE + m->length); } char *buffer = *buf; buffer[0] = m->type; memcpy (buffer + 1, &m->length, sizeof m->length); if (m->payload != NULL) { memcpy (buffer + IPC_HEADER_SIZE, m->payload, m->length); } *msize = IPC_HEADER_SIZE + m->length; #if defined(IPC_WITH_ERRORS) && IPC_WITH_ERRORS > 2 printf ("sending msg: type %u, size %d, msize %ld\n", m->type, m->length, *msize); #endif return 0; } // 1 on a recipient socket close int32_t ipc_message_read (int32_t fd, struct ipc_message *m) { assert (m != NULL); char *buf = NULL; ssize_t msize = IPC_MAX_MESSAGE_SIZE; int32_t ret = usock_recv (fd, &buf, &msize); if (ret < 0) { // on error, buffer already freed handle_err ("msg_read", "usock_recv"); return -1; } // closed recipient, buffer already freed if (ret == 1) { return 1; } if (ipc_message_format_read (m, buf, msize) < 0) { return -1; } free (buf); return 0; } int32_t ipc_message_write (int32_t fd, const struct ipc_message *m) { assert (m != NULL); char *buf = NULL; ssize_t msize = 0; ipc_message_format_write (m, &buf, &msize); ssize_t nbytes_sent = 0; int32_t ret = usock_send (fd, buf, msize, &nbytes_sent); if (ret < 0) { if (buf != NULL) { free (buf); } handle_err ("msg_write", "usock_send ret < 0"); return -1; } // what was sent != what should have been sent if (nbytes_sent != msize) { if (buf != NULL) { free (buf); } handle_err ("msg_write", "usock_send did not send enough data"); return -1; } if (buf != NULL) { free (buf); } return 0; } // MSG FORMAT int32_t ipc_message_format (struct ipc_message *m, char type, const char *payload, ssize_t length) { assert (m != NULL); assert (length <= IPC_MAX_MESSAGE_SIZE); assert ((length == 0 && payload == NULL) || (length > 0 && payload != NULL)); if (length > IPC_MAX_MESSAGE_SIZE) { handle_err ("msg_format_con", "msgsize > BUFSIZ"); return -1; } m->type = type; m->length = (uint32_t) length; if (payload != NULL) { if (m->payload != NULL) { free (m->payload); } m->payload = malloc (length); if (m->payload == NULL) { return IPC_ERROR_NOT_ENOUGH_MEMORY; } memset (m->payload, 0, length); } if (payload != NULL) { memcpy (m->payload, payload, length); } return 0; } int32_t ipc_message_format_data (struct ipc_message *m, const char *payload, ssize_t length) { return ipc_message_format (m, MSG_TYPE_DATA, payload, length); } int32_t ipc_message_format_server_close (struct ipc_message *m) { return ipc_message_format (m, MSG_TYPE_SERVER_CLOSE, NULL, 0); } int32_t ipc_message_empty (struct ipc_message *m) { assert (m != NULL); if (m == NULL) return -1; if (m->payload != NULL) { free (m->payload); m->payload = NULL; } m->length = 0; return 0; }