Date: Wed, 25 Nov 2009 15:14:54 GMT From: Jonathan Anderson <jona@FreeBSD.org> To: Perforce Change Reviews <perforce@FreeBSD.org> Subject: PERFORCE change 171017 for review Message-ID: <200911251514.nAPFEsiF088564@repoman.freebsd.org>
next in thread | raw e-mail | index | archive | help
http://p4web.freebsd.org/chv.cgi?CH=171017 Change 171017 by jona@jona-capsicum-kent on 2009/11/25 15:14:52 Add openat(2) in capability mode. openat(2) is now permitted in capability mode, subject to the constraint that the relative path must not "escape" the FD that the lookup is being conducted relative to. This results in EPERM when in capability mode (no change otherwise). openat(2) also now wraps the resulting FD with a capability if the directory FD was a capability. The rights of the new capability are identical to those of the original. Affected files ... .. //depot/projects/trustedbsd/capabilities/src/sys/amd64/conf/CAPABILITIES#2 edit .. //depot/projects/trustedbsd/capabilities/src/sys/kern/capabilities.conf#22 edit .. //depot/projects/trustedbsd/capabilities/src/sys/kern/init_sysent.c#42 edit .. //depot/projects/trustedbsd/capabilities/src/sys/kern/kern_descrip.c#30 edit .. //depot/projects/trustedbsd/capabilities/src/sys/kern/sys_capability.c#27 edit .. //depot/projects/trustedbsd/capabilities/src/sys/kern/vfs_lookup.c#13 edit .. //depot/projects/trustedbsd/capabilities/src/sys/kern/vfs_syscalls.c#18 edit .. //depot/projects/trustedbsd/capabilities/src/sys/sys/capability.h#26 edit .. //depot/projects/trustedbsd/capabilities/src/sys/sys/filedesc.h#4 edit .. //depot/projects/trustedbsd/capabilities/src/sys/sys/namei.h#6 edit Differences ... ==== //depot/projects/trustedbsd/capabilities/src/sys/amd64/conf/CAPABILITIES#2 (text+ko) ==== @@ -1,4 +1,9 @@ include GENERIC + options CAPABILITIES options PROCDESC options KDTRACE_HOOKS +options WITNESS +options KDB +options DDB + ==== //depot/projects/trustedbsd/capabilities/src/sys/kern/capabilities.conf#22 (text+ko) ==== @@ -38,7 +38,7 @@ ## - sys_exit(2), abort2(2) and close(2) are very important. ## - Sorted alphabetically, please keep it that way. ## -## $P4: //depot/projects/trustedbsd/capabilities/src/sys/kern/capabilities.conf#21 $ +## $P4: //depot/projects/trustedbsd/capabilities/src/sys/kern/capabilities.conf#22 $ ## ## @@ -453,6 +453,12 @@ olio_listio ## +## Allow openat(2), which we have constrained to prevent accessing files +## which are not "under" the directory FD given to the syscall. +## +openat + +## ## Allow poll(2), which will be scoped by capability rights. ## ## XXXRW: Perhaps we don't need the OpenBSD version? ==== //depot/projects/trustedbsd/capabilities/src/sys/kern/init_sysent.c#42 (text+ko) ==== @@ -533,7 +533,7 @@ { AS(mkdirat_args), (sy_call_t *)mkdirat, AUE_MKDIRAT, NULL, 0, 0, 0 }, /* 496 = mkdirat */ { AS(mkfifoat_args), (sy_call_t *)mkfifoat, AUE_MKFIFOAT, NULL, 0, 0, 0 }, /* 497 = mkfifoat */ { AS(mknodat_args), (sy_call_t *)mknodat, AUE_MKNODAT, NULL, 0, 0, 0 }, /* 498 = mknodat */ - { AS(openat_args), (sy_call_t *)openat, AUE_OPENAT_RWTC, NULL, 0, 0, 0 }, /* 499 = openat */ + { AS(openat_args), (sy_call_t *)openat, AUE_OPENAT_RWTC, NULL, 0, 0, SYF_CAPENABLED }, /* 499 = openat */ { AS(readlinkat_args), (sy_call_t *)readlinkat, AUE_READLINKAT, NULL, 0, 0, 0 }, /* 500 = readlinkat */ { AS(renameat_args), (sy_call_t *)renameat, AUE_RENAMEAT, NULL, 0, 0, 0 }, /* 501 = renameat */ { AS(symlinkat_args), (sy_call_t *)symlinkat, AUE_SYMLINKAT, NULL, 0, 0, 0 }, /* 502 = symlinkat */ ==== //depot/projects/trustedbsd/capabilities/src/sys/kern/kern_descrip.c#30 (text+ko) ==== @@ -1543,13 +1543,31 @@ int falloc(struct thread *td, struct file **resultfp, int *resultfd) { + return _falloc(td, resultfp, resultfd, 1); +} + +/* + * Create a new open file structure and, optionally, allocate a file decriptor + * for the process that refers to it. + */ +int +_falloc(struct thread *td, struct file **resultfp, int *resultfd, + int addfd) +{ struct proc *p = td->td_proc; struct file *fp; - int error, i; + int error, i = -1; int maxuserfiles = maxfiles - (maxfiles / 20); static struct timeval lastfail; static int curfail; + /* + * Cowardly refuse to create a referenceless file: if we're not adding + * the file to the process descriptor array, then the calling code + * MUST expect a pointer to be returned. + */ + if (!addfd && !resultfp) return (error = EINVAL); + fp = uma_zalloc(file_zone, M_WAITOK | M_ZERO); if ((openfiles >= maxuserfiles && priv_check(td, PRIV_MAXFILES) != 0) || @@ -1561,14 +1579,16 @@ uma_zfree(file_zone, fp); return (ENFILE); } - atomic_add_int(&openfiles, 1); + if (addfd) + atomic_add_int(&openfiles, 1); /* + * If addfd: * If the process has file descriptor zero open, add the new file * descriptor to the list of open files at that point, otherwise * put it at the front of the list of open files. */ - refcount_init(&fp->f_count, 1); + refcount_init(&fp->f_count, (addfd > 0)); if (resultfp) fhold(fp); fp->f_cred = crhold(td->td_ucred); @@ -1577,16 +1597,20 @@ fp->f_vnode = NULL; LIST_INIT(&fp->f_caps); fp->f_capcount = 0; - FILEDESC_XLOCK(p->p_fd); - if ((error = fdalloc(td, 0, &i))) { + + if (addfd) { + FILEDESC_XLOCK(p->p_fd); + if ((error = fdalloc(td, 0, &i))) { + FILEDESC_XUNLOCK(p->p_fd); + fdrop(fp, td); + if (resultfp) + fdrop(fp, td); + return (error); + } + p->p_fd->fd_ofiles[i] = fp; FILEDESC_XUNLOCK(p->p_fd); - fdrop(fp, td); - if (resultfp) - fdrop(fp, td); - return (error); } - p->p_fd->fd_ofiles[i] = fp; - FILEDESC_XUNLOCK(p->p_fd); + if (resultfp) *resultfp = fp; if (resultfd) ==== //depot/projects/trustedbsd/capabilities/src/sys/kern/sys_capability.c#27 (text+ko) ==== @@ -50,7 +50,7 @@ #include "opt_capabilities.h" #include <sys/cdefs.h> -__FBSDID("$P4: //depot/projects/trustedbsd/capabilities/src/sys/kern/sys_capability.c#26 $"); +__FBSDID("$P4: //depot/projects/trustedbsd/capabilities/src/sys/kern/sys_capability.c#27 $"); #include <sys/param.h> #include <sys/capability.h> @@ -278,28 +278,52 @@ int cap_new(struct thread *td, struct cap_new_args *uap) { - struct capability *c, *c_old; - struct file *fp, *fp_cap, *fp_object; - int error, fd_cap; + int error, capfd; + int fd = uap->fd; + struct file *fp, *cap; + cap_rights_t rights = uap->rights; - AUDIT_ARG_FD(uap->fd); - AUDIT_ARG_RIGHTS(uap->rights); - if ((uap->rights | CAP_MASK_VALID) != CAP_MASK_VALID) - return (EINVAL); + AUDIT_ARG_FD(fd); + AUDIT_ARG_RIGHTS(rights); - c = uma_zalloc(capability_zone, M_WAITOK | M_ZERO); - /* * We always allow creating a capability referencing an existing * descriptor or capability, even if it's not of much use to the * application. */ - error = fget(td, uap->fd, 0, &fp); - if (error) - goto fail; + error = fget(td, fd, 0, &fp); + if (error) return (error); AUDIT_ARG_FILE(td->td_proc, fp); + error = kern_capwrap(td, fp, rights, &cap, &capfd); + + /* + * Release our reference to the file (another one has been taken for + * the capability's sake if necessary). + */ + fdrop(fp, td); + + return error; +} + + +/* + * Create a capability to wrap around an existing file. + */ +int kern_capwrap(struct thread *td, struct file *fp, cap_rights_t rights, + struct file **cap, int *capfd) +{ + struct capability *c, *c_old; + struct file *fp_object; + int error; + + if ((rights | CAP_MASK_VALID) != CAP_MASK_VALID) + return (EINVAL); + + c = uma_zalloc(capability_zone, M_WAITOK | M_ZERO); + + /* * If a new capability is being derived from an existing capability, * then the new capability rights must be a subset of the existing @@ -307,18 +331,18 @@ */ if (fp->f_type == DTYPE_CAPABILITY) { c_old = fp->f_data; - if ((c_old->cap_rights | uap->rights) != c_old->cap_rights) { + if ((c_old->cap_rights | rights) != c_old->cap_rights) { error = ENOTCAPABLE; - goto fail2; + goto fail; } } /* * Allocate a new file descriptor to hang the capability off. */ - error = falloc(td, &fp_cap, &fd_cap); + error = falloc(td, cap, capfd); if (error) - goto fail2; + goto fail; /* * Rather than nesting capabilities, directly reference the object an @@ -332,10 +356,10 @@ else fp_object = fp; fhold(fp_object); - c->cap_rights = uap->rights; + c->cap_rights = rights; c->cap_object = fp_object; - c->cap_file = fp_cap; - finit(fp_cap, fp->f_flag, DTYPE_CAPABILITY, c, &capability_ops); + c->cap_file = *cap; + finit(*cap, fp->f_flag, DTYPE_CAPABILITY, c, &capability_ops); /* * Add this capability to the per-file list of referencing @@ -345,13 +369,15 @@ LIST_INSERT_HEAD(&fp_object->f_caps, c, cap_filelist); fp_object->f_capcount++; mtx_pool_unlock(mtxpool_sleep, fp_object); - td->td_retval[0] = fd_cap; - fdrop(fp, td); - fdrop(fp_cap, td); + td->td_retval[0] = *capfd; + + /* + * Release our private reference (the proc filedesc still has one). + */ + fdrop(*cap, td); + return (0); -fail2: - fdrop(fp, td); fail: uma_zfree(capability_zone, c); return (error); ==== //depot/projects/trustedbsd/capabilities/src/sys/kern/vfs_lookup.c#13 (text+ko) ==== @@ -140,9 +140,11 @@ int vfslocked; #ifdef KDB - if (td->td_ucred->cr_flags & CRED_FLAG_CAPMODE) { + if ((td->td_ucred->cr_flags & CRED_FLAG_CAPMODE) + && (ndp->ni_dirfd == AT_FDCWD)) + { printf("namei: pid %d proc %s performed namei in capability " - "mode\n", p->p_pid, p->p_comm); + "mode (and it's not *at())\n", p->p_pid, p->p_comm); kdb_backtrace(); } #endif @@ -478,6 +480,7 @@ int dvfslocked; /* VFS Giant state for parent */ int tvfslocked; int lkflags_save; + int insidebasedir = 0; /* we're under the *at() base */ /* * Setup: break out flag bits into variables. @@ -504,6 +507,11 @@ cnp->cn_lkflags = LK_SHARED; else cnp->cn_lkflags = LK_EXCLUSIVE; + + /* we do not allow absolute lookups in capability mode */ + if(ndp->ni_basedir && (ndp->ni_startdir == ndp->ni_rootdir)) + return (error = EPERM); + dp = ndp->ni_startdir; ndp->ni_startdir = NULLVP; vn_lock(dp, @@ -572,6 +580,11 @@ goto bad; } + + /* Check to see if we're at the *at directory */ + if(dp == ndp->ni_basedir) insidebasedir = 1; + + /* * Check for degenerate name (e.g. / or "") * which is a way of talking about a directory, @@ -626,6 +639,13 @@ goto bad; } for (;;) { + /* attempting to wander out of the *at root */ + if(dp == ndp->ni_basedir) + { + error = EPERM; + goto bad; + } + for (pr = cnp->cn_cred->cr_prison; pr != NULL; pr = pr->pr_parent) if (dp == pr->pr_root) @@ -886,6 +906,16 @@ VOP_UNLOCK(dp, 0); success: /* + * If we're in capability mode and the syscall was *at(), ensure + * that the *at() base was part of the path + */ + if(ndp->ni_basedir && !insidebasedir) + { + error = EPERM; + goto bad; + } + + /* * Because of lookup_shared we may have the vnode shared locked, but * the caller may want it to be exclusively locked. */ ==== //depot/projects/trustedbsd/capabilities/src/sys/kern/vfs_syscalls.c#18 (text+ko) ==== @@ -1083,6 +1083,8 @@ return (kern_openat(td, AT_FDCWD, path, pathseg, flags, mode)); } + + int kern_openat(struct thread *td, int fd, char *path, enum uio_seg pathseg, int flags, int mode) @@ -1090,7 +1092,7 @@ struct proc *p = td->td_proc; struct filedesc *fdp = p->p_fd; struct file *fp; - struct vnode *vp; + struct vnode *vp, *base = 0; struct vattr vat; struct mount *mp; int cmode; @@ -1099,6 +1101,7 @@ struct flock lf; struct nameidata nd; int vfslocked; + cap_rights_t baserights = -1; AUDIT_ARG_FFLAGS(flags); AUDIT_ARG_MODE(mode); @@ -1115,16 +1118,69 @@ else flags = FFLAGS(flags); - error = falloc(td, &nfp, &indx); + /* get capability info of base FD */ + if (fd >= 0) + { + struct file *f; + const cap_rights_t LOOKUP_RIGHTS = CAP_LOOKUP | CAP_ATBASE; + + FILEDESC_SLOCK(fdp); + + error = fgetcap(td, fd, &f); + if (error == 0) { + /* FD is a capability; get rights and unwrap */ + struct file *real_fp = NULL; + + baserights = cap_rights(f); + error = cap_fextract(f, LOOKUP_RIGHTS, &real_fp); + + /* hold the underlying file, not the capability */ + if (error == 0) fhold(real_fp); + fdrop(f, td); + + f = real_fp; + } + else if (error == EINVAL) + /* not a capability; get the real file pointer */ + error = fget(td, fd, LOOKUP_RIGHTS, &f); + + + + /* if in capability mode, get base vnode (for namei) */ + if (!error && (td->td_ucred->cr_flags & CRED_FLAG_CAPMODE)) { + base = f->f_vnode; + vref(base); + } + + + /* don't need to hold the base any more */ + if (f != NULL) fdrop(f, td); + + if (error) { + FILEDESC_SUNLOCK(fdp); + return (error); + } + else + FILEDESC_SUNLOCK(fdp); + } + + + /* + * allocate the file descriptor, but only add it to the descriptor + * array if fd isn't a capability (in which case we'll add the + * capability instead, later) + */ + error = _falloc(td, &nfp, &indx, (baserights == -1)); if (error) return (error); - /* An extra reference on `nfp' has been held for us by falloc(). */ + + /* An extra reference on `nfp' has been held for us by _falloc(). */ fp = nfp; /* Set the flags early so the finit in devfs can pick them up. */ fp->f_flag = flags & FMASK; cmode = ((mode &~ fdp->fd_cmask) & ALLPERMS) &~ S_ISTXT; - NDINIT_AT(&nd, LOOKUP, FOLLOW | AUDITVNODE1 | MPSAFE, pathseg, path, fd, - td); + NDINIT_ATBASE(&nd, LOOKUP, FOLLOW | AUDITVNODE1 | MPSAFE, pathseg, + path, fd, base, td); td->td_dupfd = -1; /* XXX check for fdopen */ error = vn_open(&nd, &flags, cmode, fp); if (error) { @@ -1133,11 +1189,8 @@ * wonderous happened deep below and we just pass it up * pretending we know what we do. */ - if (error == ENXIO && fp->f_ops != &badfileops) { - fdrop(fp, td); - td->td_retval[0] = indx; - return (0); - } + if (error == ENXIO && fp->f_ops != &badfileops) + goto success; /* * handle special fdopen() case. bleh. dupfdopen() is @@ -1147,15 +1200,14 @@ if ((error == ENODEV || error == ENXIO) && td->td_dupfd >= 0 && /* XXX from fdopen */ (error = - dupfdopen(td, fdp, indx, td->td_dupfd, flags, error)) == 0) { - td->td_retval[0] = indx; - fdrop(fp, td); - return (0); - } + dupfdopen(td, fdp, indx, td->td_dupfd, flags, error)) == 0) + goto success; + /* * Clean up the descriptor, but only if another thread hadn't * replaced or closed it. */ + if (base) vrele(base); fdclose(fdp, fp, indx, td); fdrop(fp, td); @@ -1213,15 +1265,28 @@ goto bad; } VFS_UNLOCK_GIANT(vfslocked); + +success: + if (baserights != -1) { + /* wrap the result in a capability */ + struct file *cap; + + error = kern_capwrap(td, fp, baserights, &cap, &indx); + if (error) goto bad_unlocked; + } + /* * Release our private reference, leaving the one associated with * the descriptor table intact. */ + if (base) vrele(base); fdrop(fp, td); td->td_retval[0] = indx; return (0); bad: VFS_UNLOCK_GIANT(vfslocked); +bad_unlocked: + if (base) vrele(base); fdclose(fdp, fp, indx, td); fdrop(fp, td); return (error); ==== //depot/projects/trustedbsd/capabilities/src/sys/sys/capability.h#26 (text+ko) ==== @@ -30,7 +30,7 @@ * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * - * $P4: //depot/projects/trustedbsd/capabilities/src/sys/sys/capability.h#25 $ + * $P4: //depot/projects/trustedbsd/capabilities/src/sys/sys/capability.h#26 $ */ /* @@ -96,7 +96,8 @@ #define CAP_TTYHOOK 0x0001000000000000ULL /* register tty hook */ #define CAP_FCHDIR 0x0002000000000000ULL /* fchdir(2) */ #define CAP_FSCK 0x0004000000000000ULL /* sysctl_ffs_fsck */ -#define CAP_MASK_VALID 0x0007ffffffffffffULL +#define CAP_ATBASE 0x0008000000000000ULL /* openat(2), etc. */ +#define CAP_MASK_VALID 0x000fffffffffffffULL /* * Notes: @@ -138,6 +139,13 @@ #ifdef _KERNEL struct file; +struct thread; + +/* + * Create a capability to wrap a file object. + */ +int kern_capwrap(struct thread *td, struct file *fp, cap_rights_t rights, + struct file **cap, int *capfd); /* * Given a file descriptor that may be a capability, check the requested ==== //depot/projects/trustedbsd/capabilities/src/sys/sys/filedesc.h#4 (text+ko) ==== @@ -112,6 +112,8 @@ int dupfdopen(struct thread *td, struct filedesc *fdp, int indx, int dfd, int mode, int error); int falloc(struct thread *td, struct file **resultfp, int *resultfd); +int _falloc(struct thread *td, struct file **resultfp, int *resultfd, + int addfd); int fdalloc(struct thread *td, int minfd, int *result); int fdavail(struct thread *td, int n); int fdcheckstd(struct thread *td); ==== //depot/projects/trustedbsd/capabilities/src/sys/sys/namei.h#6 (text+ko) ==== @@ -70,6 +70,7 @@ struct vnode *ni_rootdir; /* logical root directory */ struct vnode *ni_topdir; /* logical top directory */ int ni_dirfd; /* starting directory for *at functions */ + struct vnode *ni_basedir; /* root for capability-mode *at */ /* * Results: returned from/manipulated by lookup */ @@ -151,11 +152,13 @@ * Initialization of a nameidata structure. */ #define NDINIT(ndp, op, flags, segflg, namep, td) \ - NDINIT_ALL(ndp, op, flags, segflg, namep, AT_FDCWD, NULL, td) -#define NDINIT_AT(ndp, op, flags, segflg, namep, dirfd, td) \ - NDINIT_ALL(ndp, op, flags, segflg, namep, dirfd, NULL, td) -#define NDINIT_ATVP(ndp, op, flags, segflg, namep, vp, td) \ - NDINIT_ALL(ndp, op, flags, segflg, namep, AT_FDCWD, vp, td) + NDINIT_ALL(ndp, op, flags, segflg, namep, AT_FDCWD, NULL, NULL, td) +#define NDINIT_AT(ndp, op, flags, segflg, namep, dirfd, td) \ + NDINIT_ALL(ndp, op, flags, segflg, namep, dirfd, NULL, NULL, td) +#define NDINIT_ATBASE(ndp, op, flags, segflg, namep, dirfd, base, td) \ + NDINIT_ALL(ndp, op, flags, segflg, namep, dirfd, NULL, base, td) +#define NDINIT_ATVP(ndp, op, flags, segflg, namep, vp, td) \ + NDINIT_ALL(ndp, op, flags, segflg, namep, AT_FDCWD, vp, NULL, td) static __inline void NDINIT_ALL(struct nameidata *ndp, @@ -164,6 +167,7 @@ const char *namep, int dirfd, struct vnode *startdir, + struct vnode *basedir, struct thread *td) { ndp->ni_cnd.cn_nameiop = op; @@ -172,6 +176,7 @@ ndp->ni_dirp = namep; ndp->ni_dirfd = dirfd; ndp->ni_startdir = startdir; + ndp->ni_basedir = basedir; ndp->ni_cnd.cn_thread = td; }
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200911251514.nAPFEsiF088564>