Welcome to mirror list, hosted at ThFree Co, Russian Federation.

mq_send.c « linux « sys « libc « newlib - cygwin.com/git/newlib-cygwin.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: d3291fd2e80bf037985df665ca0985fcee4402ec (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
/* Copyright 2002, Red Hat Inc. */

#include <mqueue.h>
#include <fcntl.h>
#include <errno.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <string.h>
#include <stdlib.h>
#define _LIBC 1
#include <sys/lock.h>
#undef _LIBC

#include "mqlocal.h"

__LOCK_INIT(static, mq_wrbuf_lock);

int
mq_send (mqd_t msgid, const char *msg, size_t msg_len, unsigned int msg_prio)
{
  struct libc_mq *info;
  struct sembuf sb2 = {2, -1, 0};
  struct sembuf sb3 = {3, 1, 0};
  int rc;
  int ipcflag;

  info = __find_mq (msgid);

  if (info == NULL || (info->oflag & O_ACCMODE) == O_RDONLY)
    {
      errno = EBADF;
      return -1;
    }

  if (msg_len > info->attr->mq_msgsize)
    {
      errno = EMSGSIZE;
      return -1;
    }

  if (msg_prio > MQ_PRIO_MAX)
    {
      errno = EINVAL;
      return -1;
    }

  __lock_acquire (mq_wrbuf_lock);

  memcpy (info->wrbuf->text, msg, msg_len);
  info->wrbuf->type = (MQ_PRIO_MAX - msg_prio);

  ipcflag = (info->attr->mq_flags & O_NONBLOCK) ? IPC_NOWAIT : 0;
  sb2.sem_flg = ipcflag;

  /* check to see if max msgs are on queue */
  rc = semop (info->semid, &sb2, 1);

  if (rc == 0)
    rc = msgsnd (info->msgqid, info->wrbuf, msg_len, ipcflag);

  if (rc == 0)
    semop (info->semid, &sb3, 1);  /* increment number of reads */

  __lock_release (mq_wrbuf_lock);
  return rc;
}