2

我有一个测试程序:

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>
#include <mqueue.h>
#include <errno.h>
#include <fcntl.h>

int main() {
  struct mq_attr attrs;
  attrs.mq_maxmsg = 10;
  attrs.mq_msgsize = sizeof(int);

  const char name[] = "/test-queue";

  mqd_t q = mq_open(name, O_CREAT | O_RDWR, 0600, &attrs);
  if (q == (mqd_t)-1) {
    perror("mq_open");
    exit(EXIT_FAILURE);
  }

  mq_unlink(name); // it doesn't matter if I do this at the end or not

  if (fork()) {
    int msg = 666;
    if (mq_send(q, (const char *)&msg, sizeof(msg), 1)) {
      perror("mq_send");
      exit(EXIT_FAILURE);
    }
  } else {
    int msg;
    unsigned priority;
    if (mq_receive(q, (char *)&msg, sizeof(msg), &priority) == -1) {
      perror("mq_receive");
      exit(EXIT_FAILURE);
    }
    printf("%d\n", msg);
  }

  mq_close(q);

  return 0;
}

gcc -std=c99 -Wall -o mqtest mqtest.c -lrt我在两个平台上编译这个程序:

  • Linux kallikanzarid-desktop 3.8.0-31-generic #46-Ubuntu SMP Tue Sep 10 20:03:44 UTC 2013 x86_64 x86_64 x86_64 GNU/Linux
  • FreeBSD bsd.localhost 9.2-RELEASE FreeBSD 9.2-RELEASE #0 r255898: Thu Sep 26 22:50:31 UTC 2013 root@bake.isc.freebsd.org:/usr/obj/usr/src/sys/GENERIC amd64

在 Linux 上,一切正常。在 FreeBSD 上,我得到mq_receive: Bad file descriptor. 将mq_unlink呼叫移至末尾main()无济于事。有没有办法解决这个问题,还是我必须推迟将队列标记为删除并在分叉后重新打开它?

4

1 回答 1

0

FreeBSD确实保留了消息队列描述符。参见 mq_open(2):

FreeBSD 基于文件描述符实现消息队列。描述符在 fork(2) 之后由 child 继承。在 exec(3) 之后,描述符在新图像中关闭。消息队列描述符支持 select(2) 和 kevent(2) 系统调用。

编辑

mqd_t指向的结构确实包含一个描述符。fork()但是,如果您在using之后测试该文件描述符fcntl(),它也会返回 EBADF。

这是 FreeBSD 中的一个错误。但是我不能说这个错误是在文档中还是在实现中。

于 2014-04-22T12:05:11.017 回答