0

我想在两个进程之间发送消息。但是EACCES当我尝试发送消息时出现错误msgsnd()

创建消息队列

const char* MSG_QUEUE = "/tmp/msg_queue";

int file = open(MSG_QUEUE, O_CREAT | O_RDWR | O_APPEND, 0755);
close(file);
key_t key = ftok(MSG_QUEUE, 1);
errno = 0;
msg_queue = msgget(key, IPC_CREAT);
if(msg_queue == -1) {
    M_DEBUG("Error %s\r\n", strerror(errno));
}

消息结构

struct feld_msg_s {
    long id;
    char mtext[5];
};

发送消息

struct feld_msg_s a_msg = {1, "Test"};
errno = 0;
int ret = msgsnd(msg_queue, &a_msg, sizeof(a_msg.mtext), 0);
if(ret == -1) {
    if(errno == EACCES) {
        printf("\r\n NO PERMISSION\r\n");
    } else {
        printf("msgsnd ERROR!: %s\r\n", strerror(errno));
    }
}

在 msgsnd 的手册页中写着

EACCES调用进程没有消息队列的读权限,也没有CAP_IPC_OWNER能力。

所以我用setcap命令添加了以下功能

sudo setcap CAP_SETFCAP,CAP_IPC_OWNER+epi /home/mvollmer/build-APP-Desktop_Qt_5_6_1_GCC_64bit-Debug/APP 

我已经检查getcap了应用程序是否具有这些功能。没关系。但我仍然收到无权限错误。

当以root权限执行应用程序时,它的工作!

有一件事很奇怪,尽管 msgget 成功了,但ipcs没有显示任何消息队列。

那么我的错在哪里?

我正在使用 Linux Mint

附加问题:是否可以在 msg 结构中使用另一种数据类型然后是 char 或者消息是否仅限于字符串?

4

1 回答 1

0

您需要阅读手册页。根据POSIXmsgget()标准

概要

#include <sys/msg.h>

int msgget(key_t key, int msgflg); [Option End]

描述

...

  • msg_perm.mode的低 9 位应设置为msgflg的低 9 位。

因此,这段代码

msg_queue = msgget(key, IPC_CREAT);

将“ msgflg的低 9 位”全部设置为零。因此消息队列模式也是全部0- 任何人都没有权限。

于 2017-03-03T10:21:49.790 回答