/**************************************************************************** * fs/mqueue/mq_open.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance with the * License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include "inode/inode.h" #include "mqueue/mqueue.h" /**************************************************************************** * Private Functions Prototypes ****************************************************************************/ static int nxmq_file_close(FAR struct file *filep); static int nxmq_file_poll(FAR struct file *filep, struct pollfd *fds, bool setup); /**************************************************************************** * Private Data ****************************************************************************/ static const struct file_operations g_nxmq_fileops = { NULL, /* open */ nxmq_file_close, /* close */ NULL, /* read */ NULL, /* write */ NULL, /* seek */ NULL, /* ioctl */ nxmq_file_poll, /* poll */ #ifndef CONFIG_DISABLE_PSEUDOFS_OPERATIONS NULL, /* unlink */ #endif }; /**************************************************************************** * Private Functions ****************************************************************************/ static int nxmq_file_close(FAR struct file *filep) { FAR struct inode *inode = filep->f_inode; if (inode->i_crefs <= 1 && (inode->i_flags & FSNODEFLAG_DELETED)) { FAR struct mqueue_inode_s *msgq = inode->i_private; if (msgq) { nxmq_free_msgq(msgq); inode->i_private = NULL; } } return 0; } static int nxmq_file_poll(FAR struct file *filep, struct pollfd *fds, bool setup) { FAR struct inode *inode = filep->f_inode; FAR struct mqueue_inode_s *msgq = inode->i_private; pollevent_t eventset = 0; irqstate_t flags; int ret = 0; int i; flags = enter_critical_section(); if (setup) { for (i = 0; i < CONFIG_FS_MQUEUE_NPOLLWAITERS; i++) { /* Find an available slot */ if (!msgq->fds[i]) { /* Bind the poll structure and this slot */ msgq->fds[i] = fds; fds->priv = &msgq->fds[i]; break; } } if (i >= CONFIG_FS_MQUEUE_NPOLLWAITERS) { fds->priv = NULL; ret = -EBUSY; goto errout; } /* Immediately notify on any of the requested events */ if (msgq->nmsgs < msgq->maxmsgs) { eventset |= (fds->events & POLLOUT); } if (msgq->nmsgs) { eventset |= (fds->events & POLLIN); } if (eventset) { nxmq_pollnotify(msgq, eventset); } } else if (fds->priv != NULL) { for (i = 0; i < CONFIG_FS_MQUEUE_NPOLLWAITERS; i++) { if (fds == msgq->fds[i]) { msgq->fds[i] = NULL; fds->priv = NULL; break; } } } errout: leave_critical_section(flags); return ret; } static int file_mq_vopen(FAR struct file *mq, FAR const char *mq_name, int oflags, va_list ap, int *created) { FAR struct inode *inode; FAR struct mqueue_inode_s *msgq; FAR struct mq_attr *attr = NULL; struct inode_search_s desc; char fullpath[MAX_MQUEUE_PATH]; mode_t mode = 0; int ret; /* Make sure that a non-NULL name is supplied */ if (!mq || !mq_name || *mq_name == '\0') { ret = -EINVAL; goto errout; } /* Were we asked to create it? */ if ((oflags & O_CREAT) != 0) { /* We have to extract the additional * parameters from the variable argument list. */ mode = va_arg(ap, mode_t); attr = va_arg(ap, FAR struct mq_attr *); } /* Skip over any leading '/'. All message queue paths are relative to * CONFIG_FS_MQUEUE_MPATH. */ while (*mq_name == '/') { mq_name++; } /* Get the full path to the message queue */ snprintf(fullpath, MAX_MQUEUE_PATH, CONFIG_FS_MQUEUE_MPATH "/%s", mq_name); /* Make sure that the check for the existence of the message queue * and the creation of the message queue are atomic with respect to * other processes executing mq_open(). A simple sched_lock() should * be sufficient. */ sched_lock(); /* Get the inode for this mqueue. This should succeed if the message * queue has already been created. In this case, inode_find() will * have incremented the reference count on the inode. */ SETUP_SEARCH(&desc, fullpath, false); ret = inode_find(&desc); if (ret >= 0) { /* Something exists at this path. Get the search results */ inode = desc.node; DEBUGASSERT(inode != NULL); /* Verify that the inode is a message queue */ if (!INODE_IS_MQUEUE(inode)) { ret = -ENXIO; goto errout_with_inode; } /* It exists and is a message queue. Check if the caller wanted to * create a new mqueue with this name. */ if ((oflags & (O_CREAT | O_EXCL)) == (O_CREAT | O_EXCL)) { ret = -EEXIST; goto errout_with_inode; } /* Associate the inode with a file structure */ mq->f_oflags = oflags; mq->f_pos = 0; mq->f_inode = inode; mq->f_priv = NULL; if (created) { *created = 1; } } else { /* The mqueue does not exists. Were we asked to create it? */ if ((oflags & O_CREAT) == 0) { /* The mqueue does not exist and O_CREAT is not set */ ret = -ENOENT; goto errout_with_lock; } /* Create an inode in the pseudo-filesystem at this path */ ret = inode_semtake(); if (ret < 0) { goto errout_with_lock; } ret = inode_reserve(fullpath, &inode); inode_semgive(); if (ret < 0) { goto errout_with_lock; } /* Allocate memory for the new message queue. The new inode will * be created with a reference count of zero. */ msgq = (FAR struct mqueue_inode_s *)nxmq_alloc_msgq(mode, attr); if (!msgq) { ret = -ENOSPC; goto errout_with_inode; } /* Associate the inode with a file structure */ mq->f_oflags = oflags; mq->f_pos = 0; mq->f_inode = inode; mq->f_priv = NULL; INODE_SET_MQUEUE(inode); inode->u.i_ops = &g_nxmq_fileops; inode->i_private = msgq; msgq->inode = inode; /* Set the initial reference count on this inode to one */ inode->i_crefs = 1; if (created) { *created = 0; } } RELEASE_SEARCH(&desc); sched_unlock(); return OK; errout_with_inode: inode_release(inode); errout_with_lock: RELEASE_SEARCH(&desc); sched_unlock(); errout: return ret; } static mqd_t nxmq_vopen(FAR const char *mq_name, int oflags, va_list ap) { struct file mq; int created; int ret; ret = file_mq_vopen(&mq, mq_name, oflags, ap, &created); if (ret < 0) { return ret; } ret = files_allocate(mq.f_inode, mq.f_oflags, mq.f_pos, mq.f_priv, 0); if (ret < 0) { file_mq_close(&mq); if (created) { file_mq_unlink(mq_name); } } return ret; } /**************************************************************************** * Public Functions ****************************************************************************/ void nxmq_pollnotify(FAR struct mqueue_inode_s *msgq, pollevent_t eventset) { int i; for (i = 0; i < CONFIG_FS_MQUEUE_NPOLLWAITERS; i++) { FAR struct pollfd *fds = msgq->fds[i]; if (fds) { fds->revents |= (fds->events & eventset); if (fds->revents != 0) { int semcount; nxsem_get_value(fds->sem, &semcount); if (semcount < 1) { nxsem_post(fds->sem); } } } } } /**************************************************************************** * Name: file_mq_open * * Description: * This function establish a connection between a named message queue and * the calling task. This is an internal OS interface. It is functionally * equivalent to mq_open() except that: * * - It is not a cancellation point, and * - It does not modify the errno value. * * See comments with mq_open() for a more complete description of the * behavior of this function * * Input Parameters: * mq_name - Name of the queue to open * oflags - open flags * Optional parameters. When the O_CREAT flag is specified, two optional * parameters are expected: * * 1. mode_t mode (ignored), and * 2. struct mq_attr *attr. The mq_maxmsg attribute * is used at the time that the message queue is * created to determine the maximum number of * messages that may be placed in the message queue. * * Returned Value: * This is an internal OS interface and should not be used by applications. * It follows the NuttX internal error return policy: Zero (OK) is * returned on success, mqdes point to the new message queue descriptor. * A negated errno value is returned on failure. * ****************************************************************************/ int file_mq_open(FAR struct file *mq, FAR const char *mq_name, int oflags, ...) { va_list ap; int ret; va_start(ap, oflags); ret = file_mq_vopen(mq, mq_name, oflags, ap, NULL); va_end(ap); return ret; } /**************************************************************************** * Name: nxmq_open * * Description: * This function establish a connection between a named message queue and * the calling task. This is an internal OS interface. It is functionally * equivalent to mq_open() except that: * * - It is not a cancellation point, and * - It does not modify the errno value. * * See comments with mq_open() for a more complete description of the * behavior of this function * * Input Parameters: * mq_name - Name of the queue to open * oflags - open flags * Optional parameters. When the O_CREAT flag is specified, two optional * parameters are expected: * * 1. mode_t mode (ignored), and * 2. struct mq_attr *attr. The mq_maxmsg attribute * is used at the time that the message queue is * created to determine the maximum number of * messages that may be placed in the message queue. * * Returned Value: * This is an internal OS interface and should not be used by applications. * It follows the NuttX internal error return policy: Zero (OK) is * returned on success, mqdes point to the new message queue descriptor. * A negated errno value is returned on failure. * ****************************************************************************/ mqd_t nxmq_open(FAR const char *mq_name, int oflags, ...) { va_list ap; mqd_t ret; va_start(ap, oflags); ret = nxmq_vopen(mq_name, oflags, ap); va_end(ap); return ret; } /**************************************************************************** * Name: mq_open * * Description: * This function establish a connection between a named message queue and * the calling task. After a successful call of mq_open(), the task can * reference the message queue using the address returned by the call. The * message queue remains usable until it is closed by a successful call to * mq_close(). * * Input Parameters: * mq_name - Name of the queue to open * oflags - open flags * Optional parameters. When the O_CREAT flag is specified, two optional * parameters are expected: * * 1. mode_t mode (ignored), and * 2. struct mq_attr *attr. The mq_maxmsg attribute * is used at the time that the message queue is * created to determine the maximum number of * messages that may be placed in the message queue. * * Returned Value: * A message queue descriptor or (mqd_t)-1 (ERROR) * * Assumptions: * ****************************************************************************/ mqd_t mq_open(FAR const char *mq_name, int oflags, ...) { va_list ap; mqd_t ret; va_start(ap, oflags); ret = nxmq_vopen(mq_name, oflags, ap); va_end(ap); if (ret < 0) { set_errno(-ret); return ERROR; } return ret; }