Прохождение структур в сообщениях mqueue
Я пытаюсь прочитать пакет сокета IP/RAW с его параметрами (такими как символьный указатель на IP-пакет и его размер), поместить его в mqueue в режиме WRONLY и получить те же параметры из очереди в режиме RDONLY. Я использую структуры для сообщения (MSG) для отправки и получения. При отправке сообщения оба параметра успешно помещаются в очередь, но при получении он возвращает ошибку сегментации. Мои файлы отправки и получения приведены ниже:
mq_send.c
#include<stdlib.h>
#include<stdio.h>
#include<string.h>
#include<sys/stat.h>
#include<sys/types.h>
#include<mqueue.h>
#include<errno.h>
#include<netinet/ip_icmp.h>
#include<netinet/udp.h>
#include<netinet/tcp.h>
#include<netinet/ip.h>
#include<netinet/in.h>
#include<netinet/if_ether.h>
#include<net/ethernet.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<sys/ioctl.h>
#include<sys/time.h>
#include<sys/types.h>
#include<unistd.h>
#include"external.h"
#define QUEUE_NAME "/test_queue"
#define MAX_SIZE 71680
#define CHECK(x) \
do { \
if (!(x)) { \
fprintf(stderr, "%s:%d: ", __func__, __LINE__); \
perror(#x); \
exit(-1); \
} \
} while (0) \
struct sockaddr_in source,dest;
int tcp=0,udp=0,icmp=0,others=0,igmp=0,total=0,i,j;
struct mymsg
{
char buff[MAX_SIZE];
int size;
};
int main(int argc, char **argv)
{
mqd_t mq;
struct mymsg m;
struct mq_attr attr;
//char buff[MAX_SIZE];
unsigned char* buffer = (unsigned char*) malloc(sizeof(65536));
int saddr_size,sock_raw;
struct sockaddr saddr;
int data_size;
attr.mq_flags = 0;
attr.mq_maxmsg = 10;
attr.mq_msgsize = MAX_SIZE;
attr.mq_curmsgs = 0;
mq = mq_open(QUEUE_NAME, O_CREAT | O_WRONLY, 0644, &attr);
CHECK((mqd_t)-1 != mq);
memset(buffer, 0, MAX_SIZE);
sock_raw = socket( AF_PACKET , SOCK_RAW , htons(ETH_P_ALL)) ;
if(sock_raw < 0)
{
perror("Socket Error\n");
return 1;
}
saddr_size = sizeof saddr;
data_size = recvfrom(sock_raw , buffer ,65536 , 0 , &saddr , (socklen_t*)&saddr_size);
if(data_size <0 )
{
printf("Recvfrom error , failed to get packets\n");
return 1;
}
memcpy(m.buff,buffer,65536);
m.size=data_size;
//char *a="hi";
//CHECK(0 <= mq_send(mq, (char *)&a, MAX_SIZE, 0));
CHECK(0 <= mq_send(mq, (char *)&m, MAX_SIZE, 0));
printf("Packet msg:%s size:%d\n",buffer,data_size);
printf("Sent msg:%s size:%d\n",m.buff,m.size);
ProcessPacket(m.buff , m.size);
CHECK((mqd_t)-1 != mq_close(mq));
close(sock_raw);
return 0;
}
mq_receive.c
#include<stdlib.h>
#include<stdio.h>
#include<string.h>
#include<sys/stat.h>
#include<sys/types.h>
#include<mqueue.h>
#include<errno.h>
#include<netinet/ip_icmp.h>
#include<netinet/udp.h>
#include<netinet/tcp.h>
#include<netinet/ip.h>
#include<netinet/in.h>
#include<netinet/if_ether.h>
#include<net/ethernet.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<sys/ioctl.h>
#include<sys/time.h>
#include<sys/types.h>
#include<unistd.h>
#include"external.h"
#define QUEUE_NAME "/test_queue"
#define MAX_SIZE 71680
#define CHECK(x) \
do { \
if (!(x)) { \
fprintf(stderr, "%s:%d: ", __func__, __LINE__); \
perror(#x); \
exit(-1); \
} \
} while (0) \
struct sockaddr_in source,dest;
int tcp=0,udp=0,icmp=0,others=0,igmp=0,total=0,i,j;
typedef struct mymsg
{
unsigned char *buff;
int size;
}msg;
int main(int argc, char *argv[])
{
mqd_t mq;
msg *m=(msg *)malloc(sizeof(msg));
m->buff=(unsigned char*) malloc(sizeof(MAX_SIZE));
int data_size;
mq = mq_open(QUEUE_NAME, O_RDONLY);
CHECK((mqd_t)-1 != mq);
memset(m->buff, 0, MAX_SIZE+1);
ssize_t bytes_read;
bytes_read = mq_receive(mq, (char *)m, MAX_SIZE, NULL);
CHECK(bytes_read >= 0);
printf("Received buff:%s size:%d\n", m->buff,m->size);
//ProcessPacket(m.buff , data_size);
/* cleanup */
CHECK((mqd_t)-1 != mq_close(mq));
CHECK((mqd_t)-1 != mq_unlink(QUEUE_NAME));
return 0;
}
Вывод для mqueue_send.c:
bcg @ BCGA53: ~ / Рабочий стол /mqueue$ sudo ./mq_send Пакет MSG: M Размер:60 Отправлено MSG: M размер:60
Вывод для mq_receive.c:
bcg @ BCGA53: ~ / Рабочий стол /mqueue$ sudo ./mq_receive Ошибка сегментации
1 ответ
unsigned char* buffer = (unsigned char*) malloc(sizeof(65536));
Код выше выделяет память, которая может содержать sizeof(65536)
байт. Проблема в том, что sizeof(65536)
такой же как sizeof(int)
и обычно это 4 или 8 байт.
Вы также должны использовать это:
unsigned char* buffer = (unsigned char*) malloc(sizeof(char[65536]));
или это
unsigned char* buffer = (unsigned char*) malloc(65536);
Есть и другие ошибки - например, вы используете MAX_SIZE
в mq_send()
в то время как вы должны использовать sizeof(m)
, Основной причиной вашей ошибки является проблема, изложенная выше.
На самом деле, это распределение не имеет смысла (по крайней мере, в коде, который вы представили выше), так как вы можете получить его непосредственно в структуру сообщения - тогда вы не увидите эту проблему.