#include <kern/clock.h>
#include <mach/kern_return.h>
+#define f_flag fp_glob->fg_flag
+#define f_ops fp_glob->fg_ops
+#define f_data fp_glob->fg_data
-#define f_flag f_fglob->fg_flag
-#define f_type f_fglob->fg_ops->fo_type
-#define f_msgcount f_fglob->fg_msgcount
-#define f_cred f_fglob->fg_cred
-#define f_ops f_fglob->fg_ops
-#define f_offset f_fglob->fg_offset
-#define f_data f_fglob->fg_data
#define PSEMNAMLEN 31 /* maximum name segment length we bother with */
struct pseminfo {
struct psemcache {
LIST_ENTRY(psemcache) psem_hash; /* hash chain */
struct pseminfo *pseminfo; /* vnode the name refers to */
- int psem_nlen; /* length of name */
+ size_t psem_nlen; /* length of name */
char psem_name[PSEMNAMLEN + 1]; /* segment name */
};
#define PSEMCACHE_NULL (struct psemcache *)0
struct psemname {
char *psem_nameptr; /* pointer to looked up name */
- long psem_namelen; /* length of looked up component */
+ size_t psem_namelen; /* length of looked up component */
u_int32_t psem_hash; /* hash value of looked up name */
};
struct psemstats psemstats; /* cache effectiveness statistics */
-static int psem_access(struct pseminfo *pinfo, int mode, kauth_cred_t cred);
+static int psem_access(struct pseminfo *pinfo, mode_t mode, kauth_cred_t cred);
static int psem_cache_search(struct pseminfo **,
struct psemname *, struct psemcache **);
static int psem_delete(struct pseminfo * pinfo);
-static int psem_read(struct fileproc *fp, struct uio *uio,
- int flags, vfs_context_t ctx);
-static int psem_write(struct fileproc *fp, struct uio *uio,
- int flags, vfs_context_t ctx);
-static int psem_ioctl(struct fileproc *fp, u_long com,
- caddr_t data, vfs_context_t ctx);
-static int psem_select(struct fileproc *fp, int which, void *wql, vfs_context_t ctx);
static int psem_closefile(struct fileglob *fp, vfs_context_t ctx);
static int psem_unlink_internal(struct pseminfo *pinfo, struct psemcache *pcache);
-static int psem_kqfilter(struct fileproc *fp, struct knote *kn,
- struct kevent_internal_s *kev, vfs_context_t ctx);
-
static const struct fileops psemops = {
- .fo_type = DTYPE_PSXSEM,
- .fo_read = psem_read,
- .fo_write = psem_write,
- .fo_ioctl = psem_ioctl,
- .fo_select = psem_select,
- .fo_close = psem_closefile,
- .fo_kqfilter = psem_kqfilter,
- .fo_drain = NULL,
+ .fo_type = DTYPE_PSXSEM,
+ .fo_read = fo_no_read,
+ .fo_write = fo_no_write,
+ .fo_ioctl = fo_no_ioctl,
+ .fo_select = fo_no_select,
+ .fo_close = psem_closefile,
+ .fo_drain = fo_no_drain,
+ .fo_kqfilter = fo_no_kqfilter,
};
static lck_grp_t *psx_sem_subsys_lck_grp;
for (pcp = pcpp->lh_first; pcp != 0; pcp = nnp) {
nnp = pcp->psem_hash.le_next;
if (pcp->psem_nlen == pnp->psem_namelen &&
- !bcmp(pcp->psem_name, pnp->psem_nameptr, (u_int)pcp->psem_nlen)) {
+ !bcmp(pcp->psem_name, pnp->psem_nameptr, pcp->psem_nlen)) {
break;
}
}
*/
pcp->pseminfo = psemp;
pcp->psem_nlen = pnp->psem_namelen;
- bcopy(pnp->psem_nameptr, pcp->psem_name, (unsigned)pcp->psem_nlen);
+ bcopy(pnp->psem_nameptr, pcp->psem_name, pcp->psem_nlen);
pcpp = PSEMHASH(pnp);
#if DIAGNOSTIC
{
void
psem_cache_init(void)
{
- psemhashtbl = hashinit(posix_sem_max / 2, M_SHM, &psemhash);
+ psemhashtbl = hashinit((int)(posix_sem_max / 2), M_SHM, &psemhash);
}
static void
return error;
}
+/*
+ * In order to support unnamed POSIX semaphores, the named
+ * POSIX semaphores will have to move out of the per-process
+ * open filetable, and into a global table that is shared with
+ * unnamed POSIX semaphores, since unnamed POSIX semaphores
+ * are typically used by declaring instances in shared memory,
+ * and there's no other way to do this without changing the
+ * underlying type, which would introduce binary compatibility
+ * issues.
+ */
int
sem_open(proc_t p, struct sem_open_args *uap, user_addr_t *retval)
{
char * nameptr;
char * cp;
size_t pathlen, plen;
- int fmode;
- int cmode = uap->mode;
+ mode_t fmode;
+ mode_t cmode = (mode_t)uap->mode;
int value = uap->value;
int incache = 0;
struct psemcache *pcp = PSEMCACHE_NULL;
kern_return_t kret = KERN_INVALID_ADDRESS; /* default fail */
AUDIT_ARG(fflags, uap->oflag);
- AUDIT_ARG(mode, uap->mode);
+ AUDIT_ARG(mode, (mode_t)uap->mode);
AUDIT_ARG(value32, uap->value);
pinfo = PSEMINFO_NULL;
* Preallocate everything we might need up front to avoid taking
* and dropping the lock, opening us up to race conditions.
*/
- MALLOC_ZONE(pnbuf, caddr_t, MAXPATHLEN, M_NAMEI, M_WAITOK | M_ZERO);
- if (pnbuf == NULL) {
- error = ENOSPC;
- goto bad;
- }
+ pnbuf = zalloc_flags(ZV_NAMEI, Z_WAITOK | Z_ZERO);
pathlen = MAXPATHLEN;
error = copyinstr(uap->name, pnbuf, MAXPATHLEN, &pathlen);
* to KERN_INVALID_ADDRESS, above.
*/
- fmode = FFLAGS(uap->oflag);
+ fmode = (mode_t)FFLAGS(uap->oflag);
if ((fmode & O_CREAT)) {
if ((value < 0) || (value > SEM_VALUE_MAX)) {
proc_fdunlock(p);
*retval = CAST_USER_ADDR_T(indx);
- FREE_ZONE(pnbuf, MAXPATHLEN, M_NAMEI);
+ zfree(ZV_NAMEI, pnbuf);
return 0;
bad_locked:
}
if (pnbuf != NULL) {
- FREE_ZONE(pnbuf, MAXPATHLEN, M_NAMEI);
+ zfree(ZV_NAMEI, pnbuf);
}
return error;
}
* XXX This code is repeated in several places
*/
static int
-psem_access(struct pseminfo *pinfo, int mode, kauth_cred_t cred)
+psem_access(struct pseminfo *pinfo, mode_t mode, kauth_cred_t cred)
{
- int mode_req = ((mode & FREAD) ? S_IRUSR : 0) |
+ mode_t mode_req = ((mode & FREAD) ? S_IRUSR : 0) |
((mode & FWRITE) ? S_IWUSR : 0);
/* Otherwise, user id 0 always gets access. */
pinfo = PSEMINFO_NULL;
- MALLOC_ZONE(pnbuf, caddr_t, MAXPATHLEN, M_NAMEI, M_WAITOK);
- if (pnbuf == NULL) {
- return ENOSPC; /* XXX non-standard */
- }
+ pnbuf = zalloc(ZV_NAMEI);
+
pathlen = MAXPATHLEN;
error = copyinstr(uap->name, pnbuf, MAXPATHLEN, &pathlen);
if (error) {
if (error != PSEMCACHE_FOUND) {
PSEM_SUBSYS_UNLOCK();
- error = EINVAL;
+ error = ENOENT;
goto bad;
}
PSEM_SUBSYS_UNLOCK();
bad:
- FREE_ZONE(pnbuf, MAXPATHLEN, M_NAMEI);
+ zfree(ZV_NAMEI, pnbuf);
return error;
}
{
int fd = CAST_DOWN_EXPLICIT(int, uap->sem);
struct fileproc *fp;
- int error = 0;
AUDIT_ARG(fd, fd); /* XXX This seems wrong; uap->sem is a pointer */
proc_fdlock(p);
- error = fp_lookup(p, fd, &fp, 1);
- if (error) {
+ if ((fp = fp_get_noref_locked(p, fd)) == NULL) {
proc_fdunlock(p);
- return error;
+ return EBADF;
}
- if (fp->f_type != DTYPE_PSXSEM) {
- fp_drop(p, fd, fp, 1);
+ if (FILEGLOB_DTYPE(fp->fp_glob) != DTYPE_PSXSEM) {
proc_fdunlock(p);
return EBADF;
}
- procfdtbl_markclosefd(p, fd);
- fileproc_drain(p, fp);
- fdrelse(p, fd);
- error = closef_locked(fp, fp->f_fglob, p);
- fileproc_free(fp);
- proc_fdunlock(p);
- return error;
+ return fp_close_and_unlock(p, fd, fp, 0);
}
int
kern_return_t kret;
int error;
- error = fp_getfpsem(p, fd, &fp, &pnode);
+ error = fp_get_ftype(p, fd, DTYPE_PSXSEM, EBADF, &fp);
if (error) {
return error;
}
- if (((pnode = (struct psemnode *)fp->f_data)) == PSEMNODE_NULL) {
- error = EINVAL;
- goto out;
- }
+ pnode = (struct psemnode *)fp->f_data;
+
PSEM_SUBSYS_LOCK();
if ((pinfo = pnode->pinfo) == PSEMINFO_NULL) {
PSEM_SUBSYS_UNLOCK();
mach_timespec_t wait_time;
int error;
- error = fp_getfpsem(p, fd, &fp, &pnode);
+ error = fp_get_ftype(p, fd, DTYPE_PSXSEM, EBADF, &fp);
if (error) {
return error;
}
- if (((pnode = (struct psemnode *)fp->f_data)) == PSEMNODE_NULL) {
- error = EINVAL;
- goto out;
- }
+ pnode = (struct psemnode *)fp->f_data;
+
PSEM_SUBSYS_LOCK();
if ((pinfo = pnode->pinfo) == PSEMINFO_NULL) {
PSEM_SUBSYS_UNLOCK();
kern_return_t kret;
int error;
- error = fp_getfpsem(p, fd, &fp, &pnode);
+ error = fp_get_ftype(p, fd, DTYPE_PSXSEM, EBADF, &fp);
if (error) {
return error;
}
- if (((pnode = (struct psemnode *)fp->f_data)) == PSEMNODE_NULL) {
- error = EINVAL;
- goto out;
- }
+ pnode = (struct psemnode *)fp->f_data;
+
PSEM_SUBSYS_LOCK();
if ((pinfo = pnode->pinfo) == PSEMINFO_NULL) {
PSEM_SUBSYS_UNLOCK();
}
static int
-psem_close(struct psemnode *pnode, __unused int flags)
+psem_close(struct psemnode *pnode)
{
int error = 0;
struct pseminfo *pinfo;
static int
psem_closefile(struct fileglob *fg, __unused vfs_context_t ctx)
{
- int error;
-
/*
* Not locked as psem_close is called only from here and is locked
* properly
*/
- error = psem_close(((struct psemnode *)fg->fg_data), fg->fg_flag);
-
- return error;
+ return psem_close((struct psemnode *)fg->fg_data);
}
static int
}
}
-static int
-psem_read(__unused struct fileproc *fp, __unused struct uio *uio,
- __unused int flags, __unused vfs_context_t ctx)
-{
- return ENOTSUP;
-}
-
-static int
-psem_write(__unused struct fileproc *fp, __unused struct uio *uio,
- __unused int flags, __unused vfs_context_t ctx)
-{
- return ENOTSUP;
-}
-
-static int
-psem_ioctl(__unused struct fileproc *fp, __unused u_long com,
- __unused caddr_t data, __unused vfs_context_t ctx)
-{
- return ENOTSUP;
-}
-
-static int
-psem_select(__unused struct fileproc *fp, __unused int which,
- __unused void *wql, __unused vfs_context_t ctx)
-{
- return ENOTSUP;
-}
-
-static int
-psem_kqfilter(__unused struct fileproc *fp, struct knote *kn,
- __unused struct kevent_internal_s *kev, __unused vfs_context_t ctx)
-{
- kn->kn_flags = EV_ERROR;
- kn->kn_data = ENOTSUP;
- return 0;
-}
-
int
fill_pseminfo(struct psemnode *pnode, struct psem_info * info)
{
struct pseminfo *psem;
PSEM_SUBSYS_LOCK();
- pnode = (struct psemnode *)fp->f_fglob->fg_data;
+ pnode = (struct psemnode *)fp->fp_glob->fg_data;
if (pnode != NULL) {
psem = pnode->pinfo;
if (psem != NULL) {