2020-05-16 21:31:45 +03:00
|
|
|
/* $NetBSD: kern_auth.c,v 1.78 2020/05/16 18:31:50 christos Exp $ */
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/*-
|
|
|
|
* Copyright (c) 2005, 2006 Elad Efrat <elad@NetBSD.org>
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
2007-01-09 15:49:36 +03:00
|
|
|
* 3. The name of the author may not be used to endorse or promote products
|
2006-05-15 01:12:38 +04:00
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
2006-09-09 01:57:38 +04:00
|
|
|
#include <sys/cdefs.h>
|
2020-05-16 21:31:45 +03:00
|
|
|
__KERNEL_RCSID(0, "$NetBSD: kern_auth.c,v 1.78 2020/05/16 18:31:50 christos Exp $");
|
2006-09-09 01:57:38 +04:00
|
|
|
|
2007-02-24 23:41:33 +03:00
|
|
|
#include <sys/types.h>
|
2006-05-15 01:12:38 +04:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/queue.h>
|
|
|
|
#include <sys/proc.h>
|
|
|
|
#include <sys/ucred.h>
|
|
|
|
#include <sys/pool.h>
|
2015-10-07 01:13:39 +03:00
|
|
|
#define __KAUTH_PRIVATE
|
2006-05-15 01:12:38 +04:00
|
|
|
#include <sys/kauth.h>
|
2006-12-23 11:38:00 +03:00
|
|
|
#include <sys/kmem.h>
|
2007-02-10 00:55:00 +03:00
|
|
|
#include <sys/rwlock.h>
|
2009-12-31 05:20:36 +03:00
|
|
|
#include <sys/sysctl.h>
|
2007-11-29 20:48:27 +03:00
|
|
|
#include <sys/atomic.h>
|
2007-02-24 23:41:33 +03:00
|
|
|
#include <sys/specificdata.h>
|
2009-09-03 08:45:27 +04:00
|
|
|
#include <sys/vnode.h>
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
|
Implement the register/deregister/evaluation API for secmodel(9). It
allows registration of callbacks that can be used later for
cross-secmodel "safe" communication.
When a secmodel wishes to know a property maintained by another
secmodel, it has to submit a request to it so the other secmodel can
proceed to evaluating the request. This is done through the
secmodel_eval(9) call; example:
bool isroot;
error = secmodel_eval("org.netbsd.secmodel.suser", "is-root",
cred, &isroot);
if (error == 0 && !isroot)
result = KAUTH_RESULT_DENY;
This one asks the suser module if the credentials are assumed to be root
when evaluated by suser module. If the module is present, it will
respond. If absent, the call will return an error.
Args and command are arbitrarily defined; it's up to the secmodel(9) to
document what it expects.
Typical example is securelevel testing: when someone wants to know
whether securelevel is raised above a certain level or not, the caller
has to request this property to the secmodel_securelevel(9) module.
Given that securelevel module may be absent from system's context (thus
making access to the global "securelevel" variable impossible or
unsafe), this API can cope with this absence and return an error.
We are using secmodel_eval(9) to implement a secmodel_extensions(9)
module, which plugs with the bsd44, suser and securelevel secmodels
to provide the logic behind curtain, usermount and user_set_cpu_affinity
modes, without adding hooks to traditional secmodels. This solves a
real issue with the current secmodel(9) code, as usermount or
user_set_cpu_affinity are not really tied to secmodel_suser(9).
The secmodel_eval(9) is also used to restrict security.models settings
when securelevel is above 0, through the "is-securelevel-above"
evaluation:
- curtain can be enabled any time, but cannot be disabled if
securelevel is above 0.
- usermount/user_set_cpu_affinity can be disabled any time, but cannot
be enabled if securelevel is above 0.
Regarding sysctl(7) entries:
curtain and usermount are now found under security.models.extensions
tree. The security.curtain and vfs.generic.usermount are still
accessible for backwards compat.
Documentation is incoming, I am proof-reading my writings.
Written by elad@, reviewed and tested (anita test + interact for rights
tests) by me. ok elad@.
See also
http://mail-index.netbsd.org/tech-security/2011/11/29/msg000422.html
XXX might consider va0 mapping too.
XXX Having a secmodel(9) specific printf (like aprint_*) for reporting
secmodel(9) errors might be a good idea, but I am not sure on how
to design such a function right now.
2011-12-04 23:24:58 +04:00
|
|
|
#include <secmodel/secmodel.h>
|
|
|
|
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
/*
|
|
|
|
* Secmodel-specific credentials.
|
|
|
|
*/
|
|
|
|
struct kauth_key {
|
Implement the register/deregister/evaluation API for secmodel(9). It
allows registration of callbacks that can be used later for
cross-secmodel "safe" communication.
When a secmodel wishes to know a property maintained by another
secmodel, it has to submit a request to it so the other secmodel can
proceed to evaluating the request. This is done through the
secmodel_eval(9) call; example:
bool isroot;
error = secmodel_eval("org.netbsd.secmodel.suser", "is-root",
cred, &isroot);
if (error == 0 && !isroot)
result = KAUTH_RESULT_DENY;
This one asks the suser module if the credentials are assumed to be root
when evaluated by suser module. If the module is present, it will
respond. If absent, the call will return an error.
Args and command are arbitrarily defined; it's up to the secmodel(9) to
document what it expects.
Typical example is securelevel testing: when someone wants to know
whether securelevel is raised above a certain level or not, the caller
has to request this property to the secmodel_securelevel(9) module.
Given that securelevel module may be absent from system's context (thus
making access to the global "securelevel" variable impossible or
unsafe), this API can cope with this absence and return an error.
We are using secmodel_eval(9) to implement a secmodel_extensions(9)
module, which plugs with the bsd44, suser and securelevel secmodels
to provide the logic behind curtain, usermount and user_set_cpu_affinity
modes, without adding hooks to traditional secmodels. This solves a
real issue with the current secmodel(9) code, as usermount or
user_set_cpu_affinity are not really tied to secmodel_suser(9).
The secmodel_eval(9) is also used to restrict security.models settings
when securelevel is above 0, through the "is-securelevel-above"
evaluation:
- curtain can be enabled any time, but cannot be disabled if
securelevel is above 0.
- usermount/user_set_cpu_affinity can be disabled any time, but cannot
be enabled if securelevel is above 0.
Regarding sysctl(7) entries:
curtain and usermount are now found under security.models.extensions
tree. The security.curtain and vfs.generic.usermount are still
accessible for backwards compat.
Documentation is incoming, I am proof-reading my writings.
Written by elad@, reviewed and tested (anita test + interact for rights
tests) by me. ok elad@.
See also
http://mail-index.netbsd.org/tech-security/2011/11/29/msg000422.html
XXX might consider va0 mapping too.
XXX Having a secmodel(9) specific printf (like aprint_*) for reporting
secmodel(9) errors might be a good idea, but I am not sure on how
to design such a function right now.
2011-12-04 23:24:58 +04:00
|
|
|
secmodel_t ks_secmodel; /* secmodel */
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
specificdata_key_t ks_key; /* key */
|
|
|
|
};
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2007-02-24 23:41:33 +03:00
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
/*
|
|
|
|
* Listener.
|
|
|
|
*/
|
|
|
|
struct kauth_listener {
|
|
|
|
kauth_scope_callback_t func; /* callback */
|
|
|
|
kauth_scope_t scope; /* scope backpointer */
|
2006-07-17 18:37:20 +04:00
|
|
|
u_int refcnt; /* reference count */
|
2006-05-15 01:12:38 +04:00
|
|
|
SIMPLEQ_ENTRY(kauth_listener) listener_next; /* listener list */
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Scope.
|
|
|
|
*/
|
|
|
|
struct kauth_scope {
|
|
|
|
const char *id; /* scope name */
|
|
|
|
void *cookie; /* user cookie */
|
2006-07-17 18:37:20 +04:00
|
|
|
u_int nlisteners; /* # of listeners */
|
2006-05-15 01:12:38 +04:00
|
|
|
SIMPLEQ_HEAD(, kauth_listener) listenq; /* listener list */
|
|
|
|
SIMPLEQ_ENTRY(kauth_scope) next_scope; /* scope list */
|
|
|
|
};
|
|
|
|
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
static int kauth_cred_hook(kauth_cred_t, kauth_action_t, void *, void *);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/* List of scopes and its lock. */
|
2007-11-12 02:22:23 +03:00
|
|
|
static SIMPLEQ_HEAD(, kauth_scope) scope_list =
|
|
|
|
SIMPLEQ_HEAD_INITIALIZER(scope_list);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/* Built-in scopes: generic, process. */
|
|
|
|
static kauth_scope_t kauth_builtin_scope_generic;
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
static kauth_scope_t kauth_builtin_scope_system;
|
2006-05-15 01:12:38 +04:00
|
|
|
static kauth_scope_t kauth_builtin_scope_process;
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
static kauth_scope_t kauth_builtin_scope_network;
|
|
|
|
static kauth_scope_t kauth_builtin_scope_machdep;
|
2006-10-01 00:05:57 +04:00
|
|
|
static kauth_scope_t kauth_builtin_scope_device;
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
static kauth_scope_t kauth_builtin_scope_cred;
|
2009-09-03 08:45:27 +04:00
|
|
|
static kauth_scope_t kauth_builtin_scope_vnode;
|
2006-05-15 01:12:38 +04:00
|
|
|
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
static specificdata_domain_t kauth_domain;
|
2007-11-07 03:23:13 +03:00
|
|
|
static pool_cache_t kauth_cred_cache;
|
2009-12-31 05:20:36 +03:00
|
|
|
|
2007-02-10 00:55:00 +03:00
|
|
|
krwlock_t kauth_lock;
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
/* Allocate new, empty kauth credentials. */
|
|
|
|
kauth_cred_t
|
2006-05-23 04:43:30 +04:00
|
|
|
kauth_cred_alloc(void)
|
|
|
|
{
|
2006-05-15 01:12:38 +04:00
|
|
|
kauth_cred_t cred;
|
|
|
|
|
2007-11-07 03:23:13 +03:00
|
|
|
cred = pool_cache_get(kauth_cred_cache, PR_WAITOK);
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
cred->cr_refcnt = 1;
|
2007-11-07 03:23:13 +03:00
|
|
|
cred->cr_uid = 0;
|
|
|
|
cred->cr_euid = 0;
|
|
|
|
cred->cr_svuid = 0;
|
|
|
|
cred->cr_gid = 0;
|
|
|
|
cred->cr_egid = 0;
|
|
|
|
cred->cr_svgid = 0;
|
|
|
|
cred->cr_ngroups = 0;
|
|
|
|
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
specificdata_init(kauth_domain, &cred->cr_sd);
|
|
|
|
kauth_cred_hook(cred, KAUTH_CRED_INIT, NULL, NULL);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (cred);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Increment reference count to cred. */
|
|
|
|
void
|
|
|
|
kauth_cred_hold(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt > 0);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2012-06-27 14:06:55 +04:00
|
|
|
atomic_inc_uint(&cred->cr_refcnt);
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Decrease reference count to cred. If reached zero, free it. */
|
|
|
|
void
|
2006-05-23 04:43:30 +04:00
|
|
|
kauth_cred_free(kauth_cred_t cred)
|
|
|
|
{
|
2006-07-17 18:37:20 +04:00
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt > 0);
|
2009-08-16 15:01:12 +04:00
|
|
|
ASSERT_SLEEPABLE();
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2007-11-29 22:50:28 +03:00
|
|
|
if (atomic_dec_uint_nv(&cred->cr_refcnt) > 0)
|
2007-11-29 20:48:27 +03:00
|
|
|
return;
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2007-11-29 20:48:27 +03:00
|
|
|
kauth_cred_hook(cred, KAUTH_CRED_FREE, NULL, NULL);
|
|
|
|
specificdata_fini(kauth_domain, &cred->cr_sd);
|
|
|
|
pool_cache_put(kauth_cred_cache, cred);
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
2007-06-30 17:32:14 +04:00
|
|
|
static void
|
2007-06-23 13:02:12 +04:00
|
|
|
kauth_cred_clone1(kauth_cred_t from, kauth_cred_t to, bool copy_groups)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
|
|
|
KASSERT(from != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(from != NOCRED);
|
|
|
|
KASSERT(from != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(to != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(to != NOCRED);
|
|
|
|
KASSERT(to != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(from->cr_refcnt > 0);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
to->cr_uid = from->cr_uid;
|
|
|
|
to->cr_euid = from->cr_euid;
|
|
|
|
to->cr_svuid = from->cr_svuid;
|
|
|
|
to->cr_gid = from->cr_gid;
|
|
|
|
to->cr_egid = from->cr_egid;
|
|
|
|
to->cr_svgid = from->cr_svgid;
|
2007-06-23 13:02:12 +04:00
|
|
|
if (copy_groups) {
|
|
|
|
to->cr_ngroups = from->cr_ngroups;
|
|
|
|
memcpy(to->cr_groups, from->cr_groups, sizeof(to->cr_groups));
|
|
|
|
}
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
|
|
|
|
kauth_cred_hook(from, KAUTH_CRED_COPY, to, NULL);
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
2007-06-30 17:32:14 +04:00
|
|
|
void
|
|
|
|
kauth_cred_clone(kauth_cred_t from, kauth_cred_t to)
|
|
|
|
{
|
|
|
|
kauth_cred_clone1(from, to, true);
|
|
|
|
}
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
/*
|
|
|
|
* Duplicate cred and return a new kauth_cred_t.
|
|
|
|
*/
|
|
|
|
kauth_cred_t
|
|
|
|
kauth_cred_dup(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
kauth_cred_t new_cred;
|
|
|
|
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt > 0);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
new_cred = kauth_cred_alloc();
|
|
|
|
|
|
|
|
kauth_cred_clone(cred, new_cred);
|
|
|
|
|
|
|
|
return (new_cred);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Similar to crcopy(), only on a kauth_cred_t.
|
|
|
|
* XXX: Is this even needed? [kauth_cred_copy]
|
|
|
|
*/
|
|
|
|
kauth_cred_t
|
|
|
|
kauth_cred_copy(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
kauth_cred_t new_cred;
|
|
|
|
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt > 0);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/* If the provided credentials already have one reference, use them. */
|
|
|
|
if (cred->cr_refcnt == 1)
|
|
|
|
return (cred);
|
|
|
|
|
|
|
|
new_cred = kauth_cred_alloc();
|
|
|
|
|
|
|
|
kauth_cred_clone(cred, new_cred);
|
|
|
|
|
|
|
|
kauth_cred_free(cred);
|
|
|
|
|
|
|
|
return (new_cred);
|
|
|
|
}
|
|
|
|
|
2007-01-15 20:45:32 +03:00
|
|
|
void
|
|
|
|
kauth_proc_fork(struct proc *parent, struct proc *child)
|
|
|
|
{
|
2007-02-10 00:55:00 +03:00
|
|
|
|
2008-04-24 22:39:20 +04:00
|
|
|
mutex_enter(parent->p_lock);
|
2007-01-15 20:45:32 +03:00
|
|
|
kauth_cred_hold(parent->p_cred);
|
|
|
|
child->p_cred = parent->p_cred;
|
2008-04-24 22:39:20 +04:00
|
|
|
mutex_exit(parent->p_lock);
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
|
2007-02-10 00:55:00 +03:00
|
|
|
/* XXX: relies on parent process stalling during fork() */
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
kauth_cred_hook(parent->p_cred, KAUTH_CRED_FORK, parent,
|
|
|
|
child);
|
2007-01-15 20:45:32 +03:00
|
|
|
}
|
|
|
|
|
2012-06-27 16:28:28 +04:00
|
|
|
void
|
|
|
|
kauth_proc_chroot(kauth_cred_t cred, struct cwdinfo *cwdi)
|
|
|
|
{
|
|
|
|
kauth_cred_hook(cred, KAUTH_CRED_CHROOT, cwdi, NULL);
|
|
|
|
}
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
uid_t
|
|
|
|
kauth_cred_getuid(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (cred->cr_uid);
|
|
|
|
}
|
|
|
|
|
|
|
|
uid_t
|
|
|
|
kauth_cred_geteuid(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (cred->cr_euid);
|
|
|
|
}
|
|
|
|
|
|
|
|
uid_t
|
|
|
|
kauth_cred_getsvuid(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (cred->cr_svuid);
|
|
|
|
}
|
|
|
|
|
|
|
|
gid_t
|
|
|
|
kauth_cred_getgid(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (cred->cr_gid);
|
|
|
|
}
|
|
|
|
|
|
|
|
gid_t
|
|
|
|
kauth_cred_getegid(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (cred->cr_egid);
|
|
|
|
}
|
|
|
|
|
|
|
|
gid_t
|
|
|
|
kauth_cred_getsvgid(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (cred->cr_svgid);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
kauth_cred_setuid(kauth_cred_t cred, uid_t uid)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt == 1);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
cred->cr_uid = uid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
kauth_cred_seteuid(kauth_cred_t cred, uid_t uid)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt == 1);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
cred->cr_euid = uid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
kauth_cred_setsvuid(kauth_cred_t cred, uid_t uid)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt == 1);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
cred->cr_svuid = uid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
kauth_cred_setgid(kauth_cred_t cred, gid_t gid)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt == 1);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
cred->cr_gid = gid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
kauth_cred_setegid(kauth_cred_t cred, gid_t gid)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt == 1);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
cred->cr_egid = gid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
kauth_cred_setsvgid(kauth_cred_t cred, gid_t gid)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt == 1);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
cred->cr_svgid = gid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Checks if gid is a member of the groups in cred. */
|
|
|
|
int
|
|
|
|
kauth_cred_ismember_gid(kauth_cred_t cred, gid_t gid, int *resultp)
|
|
|
|
{
|
2009-04-05 15:48:02 +04:00
|
|
|
uint32_t i;
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(resultp != NULL);
|
|
|
|
|
|
|
|
*resultp = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < cred->cr_ngroups; i++)
|
|
|
|
if (cred->cr_groups[i] == gid) {
|
|
|
|
*resultp = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2006-07-17 18:37:20 +04:00
|
|
|
u_int
|
2006-05-15 01:12:38 +04:00
|
|
|
kauth_cred_ngroups(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (cred->cr_ngroups);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return the group at index idx from the groups in cred.
|
|
|
|
*/
|
|
|
|
gid_t
|
2006-07-17 18:37:20 +04:00
|
|
|
kauth_cred_group(kauth_cred_t cred, u_int idx)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(idx < cred->cr_ngroups);
|
|
|
|
|
|
|
|
return (cred->cr_groups[idx]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* XXX elad: gmuid is unused for now. */
|
|
|
|
int
|
2007-06-30 17:32:14 +04:00
|
|
|
kauth_cred_setgroups(kauth_cred_t cred, const gid_t *grbuf, size_t len,
|
2007-09-23 20:00:08 +04:00
|
|
|
uid_t gmuid, enum uio_seg seg)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
2007-06-30 17:32:14 +04:00
|
|
|
int error = 0;
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-07-17 18:37:20 +04:00
|
|
|
KASSERT(cred->cr_refcnt == 1);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2008-08-15 05:31:02 +04:00
|
|
|
if (len > __arraycount(cred->cr_groups))
|
2007-06-30 17:32:14 +04:00
|
|
|
return EINVAL;
|
|
|
|
|
|
|
|
if (len) {
|
2007-09-23 20:00:08 +04:00
|
|
|
if (seg == UIO_SYSSPACE) {
|
2007-06-30 17:32:14 +04:00
|
|
|
memcpy(cred->cr_groups, grbuf,
|
|
|
|
len * sizeof(cred->cr_groups[0]));
|
2007-09-23 20:00:08 +04:00
|
|
|
} else {
|
2007-06-30 17:32:14 +04:00
|
|
|
error = copyin(grbuf, cred->cr_groups,
|
|
|
|
len * sizeof(cred->cr_groups[0]));
|
|
|
|
if (error != 0)
|
|
|
|
len = 0;
|
|
|
|
}
|
|
|
|
}
|
2006-05-15 01:12:38 +04:00
|
|
|
memset(cred->cr_groups + len, 0xff,
|
|
|
|
sizeof(cred->cr_groups) - (len * sizeof(cred->cr_groups[0])));
|
|
|
|
|
|
|
|
cred->cr_ngroups = len;
|
|
|
|
|
2007-06-30 17:32:14 +04:00
|
|
|
return error;
|
2007-06-23 13:02:12 +04:00
|
|
|
}
|
|
|
|
|
2007-06-30 17:32:14 +04:00
|
|
|
/* This supports sys_setgroups() */
|
2007-06-23 13:02:12 +04:00
|
|
|
int
|
|
|
|
kauth_proc_setgroups(struct lwp *l, kauth_cred_t ncred)
|
|
|
|
{
|
|
|
|
kauth_cred_t cred;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* At this point we could delete duplicate groups from ncred,
|
|
|
|
* and plausibly sort the list - but in general the later is
|
|
|
|
* a bad idea.
|
|
|
|
*/
|
|
|
|
proc_crmod_enter();
|
|
|
|
/* Maybe we should use curproc here ? */
|
|
|
|
cred = l->l_proc->p_cred;
|
|
|
|
|
|
|
|
kauth_cred_clone1(cred, ncred, false);
|
|
|
|
|
|
|
|
error = kauth_authorize_process(cred, KAUTH_PROCESS_SETID,
|
|
|
|
l->l_proc, NULL, NULL, NULL);
|
|
|
|
if (error != 0) {
|
|
|
|
proc_crmod_leave(cred, ncred, false);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Broadcast our credentials to the process and other LWPs. */
|
|
|
|
proc_crmod_leave(ncred, cred, true);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
int
|
2007-06-30 17:32:14 +04:00
|
|
|
kauth_cred_getgroups(kauth_cred_t cred, gid_t *grbuf, size_t len,
|
2007-09-23 20:00:08 +04:00
|
|
|
enum uio_seg seg)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
|
|
|
|
2007-06-30 17:32:14 +04:00
|
|
|
if (len > cred->cr_ngroups)
|
|
|
|
return EINVAL;
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2007-09-23 20:00:08 +04:00
|
|
|
if (seg == UIO_USERSPACE)
|
2007-06-30 17:32:14 +04:00
|
|
|
return copyout(cred->cr_groups, grbuf, sizeof(*grbuf) * len);
|
|
|
|
memcpy(grbuf, cred->cr_groups, sizeof(*grbuf) * len);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2007-06-30 17:32:14 +04:00
|
|
|
return 0;
|
2007-06-23 13:02:12 +04:00
|
|
|
}
|
|
|
|
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
int
|
Implement the register/deregister/evaluation API for secmodel(9). It
allows registration of callbacks that can be used later for
cross-secmodel "safe" communication.
When a secmodel wishes to know a property maintained by another
secmodel, it has to submit a request to it so the other secmodel can
proceed to evaluating the request. This is done through the
secmodel_eval(9) call; example:
bool isroot;
error = secmodel_eval("org.netbsd.secmodel.suser", "is-root",
cred, &isroot);
if (error == 0 && !isroot)
result = KAUTH_RESULT_DENY;
This one asks the suser module if the credentials are assumed to be root
when evaluated by suser module. If the module is present, it will
respond. If absent, the call will return an error.
Args and command are arbitrarily defined; it's up to the secmodel(9) to
document what it expects.
Typical example is securelevel testing: when someone wants to know
whether securelevel is raised above a certain level or not, the caller
has to request this property to the secmodel_securelevel(9) module.
Given that securelevel module may be absent from system's context (thus
making access to the global "securelevel" variable impossible or
unsafe), this API can cope with this absence and return an error.
We are using secmodel_eval(9) to implement a secmodel_extensions(9)
module, which plugs with the bsd44, suser and securelevel secmodels
to provide the logic behind curtain, usermount and user_set_cpu_affinity
modes, without adding hooks to traditional secmodels. This solves a
real issue with the current secmodel(9) code, as usermount or
user_set_cpu_affinity are not really tied to secmodel_suser(9).
The secmodel_eval(9) is also used to restrict security.models settings
when securelevel is above 0, through the "is-securelevel-above"
evaluation:
- curtain can be enabled any time, but cannot be disabled if
securelevel is above 0.
- usermount/user_set_cpu_affinity can be disabled any time, but cannot
be enabled if securelevel is above 0.
Regarding sysctl(7) entries:
curtain and usermount are now found under security.models.extensions
tree. The security.curtain and vfs.generic.usermount are still
accessible for backwards compat.
Documentation is incoming, I am proof-reading my writings.
Written by elad@, reviewed and tested (anita test + interact for rights
tests) by me. ok elad@.
See also
http://mail-index.netbsd.org/tech-security/2011/11/29/msg000422.html
XXX might consider va0 mapping too.
XXX Having a secmodel(9) specific printf (like aprint_*) for reporting
secmodel(9) errors might be a good idea, but I am not sure on how
to design such a function right now.
2011-12-04 23:24:58 +04:00
|
|
|
kauth_register_key(secmodel_t secmodel, kauth_key_t *result)
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
{
|
|
|
|
kauth_key_t k;
|
|
|
|
specificdata_key_t key;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
KASSERT(result != NULL);
|
|
|
|
|
|
|
|
error = specificdata_key_create(kauth_domain, &key, NULL);
|
|
|
|
if (error)
|
|
|
|
return (error);
|
|
|
|
|
|
|
|
k = kmem_alloc(sizeof(*k), KM_SLEEP);
|
|
|
|
k->ks_secmodel = secmodel;
|
|
|
|
k->ks_key = key;
|
|
|
|
|
|
|
|
*result = k;
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
kauth_deregister_key(kauth_key_t key)
|
|
|
|
{
|
|
|
|
KASSERT(key != NULL);
|
|
|
|
|
|
|
|
specificdata_key_delete(kauth_domain, key->ks_key);
|
|
|
|
kmem_free(key, sizeof(*key));
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *
|
|
|
|
kauth_cred_getdata(kauth_cred_t cred, kauth_key_t key)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
KASSERT(key != NULL);
|
|
|
|
|
|
|
|
return (specificdata_getspecific(kauth_domain, &cred->cr_sd,
|
|
|
|
key->ks_key));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
kauth_cred_setdata(kauth_cred_t cred, kauth_key_t key, void *data)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
KASSERT(key != NULL);
|
|
|
|
|
|
|
|
specificdata_setspecific(kauth_domain, &cred->cr_sd, key->ks_key, data);
|
|
|
|
}
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
/*
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
* Match uids in two credentials.
|
2006-05-15 01:12:38 +04:00
|
|
|
*/
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
int
|
2006-05-15 01:12:38 +04:00
|
|
|
kauth_cred_uidmatch(kauth_cred_t cred1, kauth_cred_t cred2)
|
|
|
|
{
|
|
|
|
KASSERT(cred1 != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred1 != NOCRED);
|
|
|
|
KASSERT(cred1 != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(cred2 != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred2 != NOCRED);
|
|
|
|
KASSERT(cred2 != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
if (cred1->cr_uid == cred2->cr_uid ||
|
|
|
|
cred1->cr_euid == cred2->cr_uid ||
|
|
|
|
cred1->cr_uid == cred2->cr_euid ||
|
|
|
|
cred1->cr_euid == cred2->cr_euid)
|
|
|
|
return (1);
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2006-07-17 18:37:20 +04:00
|
|
|
u_int
|
2006-05-15 01:12:38 +04:00
|
|
|
kauth_cred_getrefcnt(kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (cred->cr_refcnt);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert userland credentials (struct uucred) to kauth_cred_t.
|
2006-10-22 17:07:15 +04:00
|
|
|
* XXX: For NFS & puffs
|
2006-05-15 01:12:38 +04:00
|
|
|
*/
|
2006-10-22 17:07:15 +04:00
|
|
|
void
|
|
|
|
kauth_uucred_to_cred(kauth_cred_t cred, const struct uucred *uuc)
|
|
|
|
{
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(uuc != NULL);
|
2006-10-22 17:07:15 +04:00
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
cred->cr_refcnt = 1;
|
|
|
|
cred->cr_uid = uuc->cr_uid;
|
|
|
|
cred->cr_euid = uuc->cr_uid;
|
|
|
|
cred->cr_svuid = uuc->cr_uid;
|
|
|
|
cred->cr_gid = uuc->cr_gid;
|
|
|
|
cred->cr_egid = uuc->cr_gid;
|
|
|
|
cred->cr_svgid = uuc->cr_gid;
|
Rename min/max -> uimin/uimax for better honesty.
These functions are defined on unsigned int. The generic name
min/max should not silently truncate to 32 bits on 64-bit systems.
This is purely a name change -- no functional change intended.
HOWEVER! Some subsystems have
#define min(a, b) ((a) < (b) ? (a) : (b))
#define max(a, b) ((a) > (b) ? (a) : (b))
even though our standard name for that is MIN/MAX. Although these
may invite multiple evaluation bugs, these do _not_ cause integer
truncation.
To avoid `fixing' these cases, I first changed the name in libkern,
and then compile-tested every file where min/max occurred in order to
confirm that it failed -- and thus confirm that nothing shadowed
min/max -- before changing it.
I have left a handful of bootloaders that are too annoying to
compile-test, and some dead code:
cobalt ews4800mips hp300 hppa ia64 luna68k vax
acorn32/if_ie.c (not included in any kernels)
macppc/if_gm.c (superseded by gem(4))
It should be easy to fix the fallout once identified -- this way of
doing things fails safe, and the goal here, after all, is to _avoid_
silent integer truncations, not introduce them.
Maybe one day we can reintroduce min/max as type-generic things that
never silently truncate. But we should avoid doing that for a while,
so that existing code has a chance to be detected by the compiler for
conversion to uimin/uimax without changing the semantics until we can
properly audit it all. (Who knows, maybe in some cases integer
truncation is actually intended!)
2018-09-03 19:29:22 +03:00
|
|
|
cred->cr_ngroups = uimin(uuc->cr_ngroups, NGROUPS);
|
2006-05-15 01:12:38 +04:00
|
|
|
kauth_cred_setgroups(cred, __UNCONST(uuc->cr_groups),
|
2007-06-30 17:32:14 +04:00
|
|
|
cred->cr_ngroups, -1, UIO_SYSSPACE);
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
2006-10-22 17:07:15 +04:00
|
|
|
/*
|
|
|
|
* Convert kauth_cred_t to userland credentials (struct uucred).
|
|
|
|
* XXX: For NFS & puffs
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
kauth_cred_to_uucred(struct uucred *uuc, const kauth_cred_t cred)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-10-22 17:07:15 +04:00
|
|
|
KASSERT(uuc != NULL);
|
|
|
|
int ng;
|
|
|
|
|
Rename min/max -> uimin/uimax for better honesty.
These functions are defined on unsigned int. The generic name
min/max should not silently truncate to 32 bits on 64-bit systems.
This is purely a name change -- no functional change intended.
HOWEVER! Some subsystems have
#define min(a, b) ((a) < (b) ? (a) : (b))
#define max(a, b) ((a) > (b) ? (a) : (b))
even though our standard name for that is MIN/MAX. Although these
may invite multiple evaluation bugs, these do _not_ cause integer
truncation.
To avoid `fixing' these cases, I first changed the name in libkern,
and then compile-tested every file where min/max occurred in order to
confirm that it failed -- and thus confirm that nothing shadowed
min/max -- before changing it.
I have left a handful of bootloaders that are too annoying to
compile-test, and some dead code:
cobalt ews4800mips hp300 hppa ia64 luna68k vax
acorn32/if_ie.c (not included in any kernels)
macppc/if_gm.c (superseded by gem(4))
It should be easy to fix the fallout once identified -- this way of
doing things fails safe, and the goal here, after all, is to _avoid_
silent integer truncations, not introduce them.
Maybe one day we can reintroduce min/max as type-generic things that
never silently truncate. But we should avoid doing that for a while,
so that existing code has a chance to be detected by the compiler for
conversion to uimin/uimax without changing the semantics until we can
properly audit it all. (Who knows, maybe in some cases integer
truncation is actually intended!)
2018-09-03 19:29:22 +03:00
|
|
|
ng = uimin(cred->cr_ngroups, NGROUPS);
|
2006-10-22 17:07:15 +04:00
|
|
|
uuc->cr_uid = cred->cr_euid;
|
|
|
|
uuc->cr_gid = cred->cr_egid;
|
|
|
|
uuc->cr_ngroups = ng;
|
2007-06-30 17:32:14 +04:00
|
|
|
kauth_cred_getgroups(cred, uuc->cr_groups, ng, UIO_SYSSPACE);
|
2006-10-22 17:07:15 +04:00
|
|
|
}
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
/*
|
|
|
|
* Compare kauth_cred_t and uucred credentials.
|
|
|
|
* XXX: Modelled after crcmp() for NFS.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
kauth_cred_uucmp(kauth_cred_t cred, const struct uucred *uuc)
|
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(uuc != NULL);
|
|
|
|
|
|
|
|
if (cred->cr_euid == uuc->cr_uid &&
|
|
|
|
cred->cr_egid == uuc->cr_gid &&
|
2009-04-05 15:48:02 +04:00
|
|
|
cred->cr_ngroups == (uint32_t)uuc->cr_ngroups) {
|
2006-05-15 01:12:38 +04:00
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Check if all groups from uuc appear in cred. */
|
|
|
|
for (i = 0; i < uuc->cr_ngroups; i++) {
|
|
|
|
int ismember;
|
|
|
|
|
|
|
|
ismember = 0;
|
|
|
|
if (kauth_cred_ismember_gid(cred, uuc->cr_groups[i],
|
|
|
|
&ismember) != 0 || !ismember)
|
2006-05-25 03:00:49 +04:00
|
|
|
return (1);
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
2006-05-25 03:00:49 +04:00
|
|
|
return (0);
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
2006-05-25 03:00:49 +04:00
|
|
|
return (1);
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2006-07-17 18:47:02 +04:00
|
|
|
* Make a struct ucred out of a kauth_cred_t. For compatibility.
|
2006-05-15 01:12:38 +04:00
|
|
|
*/
|
|
|
|
void
|
2007-02-18 18:20:34 +03:00
|
|
|
kauth_cred_toucred(kauth_cred_t cred, struct ki_ucred *uc)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(uc != NULL);
|
|
|
|
|
2006-07-17 18:47:02 +04:00
|
|
|
uc->cr_ref = cred->cr_refcnt;
|
2006-05-15 01:12:38 +04:00
|
|
|
uc->cr_uid = cred->cr_euid;
|
|
|
|
uc->cr_gid = cred->cr_egid;
|
Rename min/max -> uimin/uimax for better honesty.
These functions are defined on unsigned int. The generic name
min/max should not silently truncate to 32 bits on 64-bit systems.
This is purely a name change -- no functional change intended.
HOWEVER! Some subsystems have
#define min(a, b) ((a) < (b) ? (a) : (b))
#define max(a, b) ((a) > (b) ? (a) : (b))
even though our standard name for that is MIN/MAX. Although these
may invite multiple evaluation bugs, these do _not_ cause integer
truncation.
To avoid `fixing' these cases, I first changed the name in libkern,
and then compile-tested every file where min/max occurred in order to
confirm that it failed -- and thus confirm that nothing shadowed
min/max -- before changing it.
I have left a handful of bootloaders that are too annoying to
compile-test, and some dead code:
cobalt ews4800mips hp300 hppa ia64 luna68k vax
acorn32/if_ie.c (not included in any kernels)
macppc/if_gm.c (superseded by gem(4))
It should be easy to fix the fallout once identified -- this way of
doing things fails safe, and the goal here, after all, is to _avoid_
silent integer truncations, not introduce them.
Maybe one day we can reintroduce min/max as type-generic things that
never silently truncate. But we should avoid doing that for a while,
so that existing code has a chance to be detected by the compiler for
conversion to uimin/uimax without changing the semantics until we can
properly audit it all. (Who knows, maybe in some cases integer
truncation is actually intended!)
2018-09-03 19:29:22 +03:00
|
|
|
uc->cr_ngroups = uimin(cred->cr_ngroups, __arraycount(uc->cr_groups));
|
2006-05-15 01:12:38 +04:00
|
|
|
memcpy(uc->cr_groups, cred->cr_groups,
|
|
|
|
uc->cr_ngroups * sizeof(uc->cr_groups[0]));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2006-07-17 18:47:02 +04:00
|
|
|
* Make a struct pcred out of a kauth_cred_t. For compatibility.
|
2006-05-15 01:12:38 +04:00
|
|
|
*/
|
|
|
|
void
|
2007-02-18 18:20:34 +03:00
|
|
|
kauth_cred_topcred(kauth_cred_t cred, struct ki_pcred *pc)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
|
|
|
KASSERT(cred != NULL);
|
2015-08-08 10:53:51 +03:00
|
|
|
KASSERT(cred != NOCRED);
|
|
|
|
KASSERT(cred != FSCRED);
|
2006-05-15 01:12:38 +04:00
|
|
|
KASSERT(pc != NULL);
|
|
|
|
|
2007-02-18 18:20:34 +03:00
|
|
|
pc->p_pad = NULL;
|
2006-05-15 01:12:38 +04:00
|
|
|
pc->p_ruid = cred->cr_uid;
|
|
|
|
pc->p_svuid = cred->cr_svuid;
|
|
|
|
pc->p_rgid = cred->cr_gid;
|
|
|
|
pc->p_svgid = cred->cr_svgid;
|
|
|
|
pc->p_refcnt = cred->cr_refcnt;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2006-07-24 02:06:03 +04:00
|
|
|
* Return kauth_cred_t for the current LWP.
|
2006-05-15 01:12:38 +04:00
|
|
|
*/
|
|
|
|
kauth_cred_t
|
|
|
|
kauth_cred_get(void)
|
|
|
|
{
|
2006-07-24 02:06:03 +04:00
|
|
|
return (curlwp->l_cred);
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns a scope matching the provided id.
|
|
|
|
* Requires the scope list lock to be held by the caller.
|
|
|
|
*/
|
|
|
|
static kauth_scope_t
|
2006-05-23 04:43:30 +04:00
|
|
|
kauth_ifindscope(const char *id)
|
|
|
|
{
|
2006-05-15 01:12:38 +04:00
|
|
|
kauth_scope_t scope;
|
|
|
|
|
2007-02-10 00:55:00 +03:00
|
|
|
KASSERT(rw_lock_held(&kauth_lock));
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
scope = NULL;
|
|
|
|
SIMPLEQ_FOREACH(scope, &scope_list, next_scope) {
|
|
|
|
if (strcmp(scope->id, id) == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (scope);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Register a new scope.
|
|
|
|
*
|
|
|
|
* id - identifier for the scope
|
|
|
|
* callback - the scope's default listener
|
|
|
|
* cookie - cookie to be passed to the listener(s)
|
|
|
|
*/
|
|
|
|
kauth_scope_t
|
|
|
|
kauth_register_scope(const char *id, kauth_scope_callback_t callback,
|
2006-08-16 21:57:26 +04:00
|
|
|
void *cookie)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
|
|
|
kauth_scope_t scope;
|
2006-09-14 15:37:07 +04:00
|
|
|
kauth_listener_t listener = NULL; /* XXX gcc */
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/* Sanitize input */
|
2006-08-16 21:57:26 +04:00
|
|
|
if (id == NULL)
|
2006-05-15 01:12:38 +04:00
|
|
|
return (NULL);
|
|
|
|
|
|
|
|
/* Allocate space for a new scope and listener. */
|
2006-12-23 11:38:00 +03:00
|
|
|
scope = kmem_alloc(sizeof(*scope), KM_SLEEP);
|
2017-06-01 05:45:05 +03:00
|
|
|
if (callback != NULL)
|
2006-12-23 11:38:00 +03:00
|
|
|
listener = kmem_alloc(sizeof(*listener), KM_SLEEP);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2006-12-23 11:38:00 +03:00
|
|
|
/*
|
|
|
|
* Acquire scope list lock.
|
|
|
|
*/
|
2007-02-10 00:55:00 +03:00
|
|
|
rw_enter(&kauth_lock, RW_WRITER);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/* Check we don't already have a scope with the same id */
|
|
|
|
if (kauth_ifindscope(id) != NULL) {
|
2007-02-10 00:55:00 +03:00
|
|
|
rw_exit(&kauth_lock);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2006-12-23 11:38:00 +03:00
|
|
|
kmem_free(scope, sizeof(*scope));
|
|
|
|
if (callback != NULL)
|
|
|
|
kmem_free(listener, sizeof(*listener));
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialize new scope with parameters */
|
|
|
|
scope->id = id;
|
|
|
|
scope->cookie = cookie;
|
|
|
|
scope->nlisteners = 1;
|
|
|
|
|
|
|
|
SIMPLEQ_INIT(&scope->listenq);
|
2006-08-16 21:57:26 +04:00
|
|
|
|
|
|
|
/* Add default listener */
|
|
|
|
if (callback != NULL) {
|
|
|
|
listener->func = callback;
|
|
|
|
listener->scope = scope;
|
|
|
|
listener->refcnt = 0;
|
|
|
|
SIMPLEQ_INSERT_HEAD(&scope->listenq, listener, listener_next);
|
|
|
|
}
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/* Insert scope to scopes list */
|
2006-08-16 21:57:26 +04:00
|
|
|
SIMPLEQ_INSERT_TAIL(&scope_list, scope, next_scope);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2007-02-10 00:55:00 +03:00
|
|
|
rw_exit(&kauth_lock);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
return (scope);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize the kernel authorization subsystem.
|
|
|
|
*
|
|
|
|
* Initialize the scopes list lock.
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
* Create specificdata domain.
|
|
|
|
* Register the credentials scope, used in kauth(9) internally.
|
|
|
|
* Register built-in scopes: generic, system, process, network, machdep, device.
|
2006-05-15 01:12:38 +04:00
|
|
|
*/
|
|
|
|
void
|
|
|
|
kauth_init(void)
|
|
|
|
{
|
2007-02-10 00:55:00 +03:00
|
|
|
rw_init(&kauth_lock);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2007-11-07 03:23:13 +03:00
|
|
|
kauth_cred_cache = pool_cache_init(sizeof(struct kauth_cred),
|
2008-03-27 21:30:15 +03:00
|
|
|
coherency_unit, 0, 0, "kcredpl", NULL, IPL_NONE,
|
2007-11-29 20:48:27 +03:00
|
|
|
NULL, NULL, NULL);
|
2007-11-07 03:23:13 +03:00
|
|
|
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
/* Create specificdata domain. */
|
|
|
|
kauth_domain = specificdata_domain_create();
|
|
|
|
|
|
|
|
/* Register credentials scope. */
|
|
|
|
kauth_builtin_scope_cred =
|
|
|
|
kauth_register_scope(KAUTH_SCOPE_CRED, NULL, NULL);
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
/* Register generic scope. */
|
|
|
|
kauth_builtin_scope_generic = kauth_register_scope(KAUTH_SCOPE_GENERIC,
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
NULL, NULL);
|
|
|
|
|
|
|
|
/* Register system scope. */
|
|
|
|
kauth_builtin_scope_system = kauth_register_scope(KAUTH_SCOPE_SYSTEM,
|
|
|
|
NULL, NULL);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/* Register process scope. */
|
|
|
|
kauth_builtin_scope_process = kauth_register_scope(KAUTH_SCOPE_PROCESS,
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
NULL, NULL);
|
|
|
|
|
|
|
|
/* Register network scope. */
|
|
|
|
kauth_builtin_scope_network = kauth_register_scope(KAUTH_SCOPE_NETWORK,
|
|
|
|
NULL, NULL);
|
|
|
|
|
|
|
|
/* Register machdep scope. */
|
|
|
|
kauth_builtin_scope_machdep = kauth_register_scope(KAUTH_SCOPE_MACHDEP,
|
|
|
|
NULL, NULL);
|
2006-10-01 00:05:57 +04:00
|
|
|
|
|
|
|
/* Register device scope. */
|
|
|
|
kauth_builtin_scope_device = kauth_register_scope(KAUTH_SCOPE_DEVICE,
|
|
|
|
NULL, NULL);
|
2009-09-03 08:45:27 +04:00
|
|
|
|
|
|
|
/* Register vnode scope. */
|
|
|
|
kauth_builtin_scope_vnode = kauth_register_scope(KAUTH_SCOPE_VNODE,
|
|
|
|
NULL, NULL);
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Deregister a scope.
|
|
|
|
* Requires scope list lock to be held by the caller.
|
|
|
|
*
|
|
|
|
* scope - the scope to deregister
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
kauth_deregister_scope(kauth_scope_t scope)
|
|
|
|
{
|
|
|
|
if (scope != NULL) {
|
|
|
|
/* Remove scope from list */
|
|
|
|
SIMPLEQ_REMOVE(&scope_list, scope, kauth_scope, next_scope);
|
2007-01-02 02:33:03 +03:00
|
|
|
kmem_free(scope, sizeof(*scope));
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Register a listener.
|
|
|
|
*
|
|
|
|
* id - scope identifier.
|
|
|
|
* callback - the callback routine for the listener.
|
|
|
|
* cookie - cookie to pass unmoidfied to the callback.
|
|
|
|
*/
|
|
|
|
kauth_listener_t
|
|
|
|
kauth_listen_scope(const char *id, kauth_scope_callback_t callback,
|
2006-11-01 13:17:58 +03:00
|
|
|
void *cookie)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
|
|
|
kauth_scope_t scope;
|
|
|
|
kauth_listener_t listener;
|
|
|
|
|
2007-02-10 00:55:00 +03:00
|
|
|
listener = kmem_alloc(sizeof(*listener), KM_SLEEP);
|
|
|
|
rw_enter(&kauth_lock, RW_WRITER);
|
|
|
|
|
2006-12-23 11:38:00 +03:00
|
|
|
/*
|
|
|
|
* Find scope struct.
|
|
|
|
*/
|
2006-05-15 01:12:38 +04:00
|
|
|
scope = kauth_ifindscope(id);
|
2007-02-10 00:55:00 +03:00
|
|
|
if (scope == NULL) {
|
|
|
|
rw_exit(&kauth_lock);
|
|
|
|
kmem_free(listener, sizeof(*listener));
|
2006-05-15 01:12:38 +04:00
|
|
|
return (NULL);
|
2007-02-10 00:55:00 +03:00
|
|
|
}
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/* Allocate listener */
|
|
|
|
|
|
|
|
/* Initialize listener with parameters */
|
|
|
|
listener->func = callback;
|
|
|
|
listener->refcnt = 0;
|
|
|
|
|
|
|
|
/* Add listener to scope */
|
|
|
|
SIMPLEQ_INSERT_TAIL(&scope->listenq, listener, listener_next);
|
|
|
|
|
|
|
|
/* Raise number of listeners on scope. */
|
|
|
|
scope->nlisteners++;
|
|
|
|
listener->scope = scope;
|
|
|
|
|
2007-02-10 00:55:00 +03:00
|
|
|
rw_exit(&kauth_lock);
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
return (listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Deregister a listener.
|
|
|
|
*
|
|
|
|
* listener - listener reference as returned from kauth_listen_scope().
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
kauth_unlisten_scope(kauth_listener_t listener)
|
|
|
|
{
|
2007-02-10 00:55:00 +03:00
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
if (listener != NULL) {
|
2007-02-10 00:55:00 +03:00
|
|
|
rw_enter(&kauth_lock, RW_WRITER);
|
2006-05-23 04:43:30 +04:00
|
|
|
SIMPLEQ_REMOVE(&listener->scope->listenq, listener,
|
|
|
|
kauth_listener, listener_next);
|
2006-05-15 01:12:38 +04:00
|
|
|
listener->scope->nlisteners--;
|
2007-02-10 00:55:00 +03:00
|
|
|
rw_exit(&kauth_lock);
|
2007-01-02 02:33:03 +03:00
|
|
|
kmem_free(listener, sizeof(*listener));
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Authorize a request.
|
|
|
|
*
|
|
|
|
* scope - the scope of the request as defined by KAUTH_SCOPE_* or as
|
|
|
|
* returned from kauth_register_scope().
|
|
|
|
* credential - credentials of the user ("actor") making the request.
|
|
|
|
* action - request identifier.
|
|
|
|
* arg[0-3] - passed unmodified to listener(s).
|
2009-09-03 08:45:27 +04:00
|
|
|
*
|
|
|
|
* Returns the aggregated result:
|
|
|
|
* - KAUTH_RESULT_ALLOW if there is at least one KAUTH_RESULT_ALLOW and
|
|
|
|
* zero KAUTH_DESULT_DENY
|
|
|
|
* - KAUTH_RESULT_DENY if there is at least one KAUTH_RESULT_DENY
|
|
|
|
* - KAUTH_RESULT_DEFER if there is nothing but KAUTH_RESULT_DEFER
|
2006-05-15 01:12:38 +04:00
|
|
|
*/
|
2009-09-03 08:45:27 +04:00
|
|
|
static int
|
|
|
|
kauth_authorize_action_internal(kauth_scope_t scope, kauth_cred_t cred,
|
|
|
|
kauth_action_t action, void *arg0, void *arg1, void *arg2, void *arg3)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
|
|
|
kauth_listener_t listener;
|
|
|
|
int error, allow, fail;
|
|
|
|
|
2006-10-02 20:29:57 +04:00
|
|
|
KASSERT(cred != NULL);
|
|
|
|
KASSERT(action != 0);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2006-08-20 19:05:14 +04:00
|
|
|
/* Short-circuit requests coming from the kernel. */
|
|
|
|
if (cred == NOCRED || cred == FSCRED)
|
2012-09-16 18:35:26 +04:00
|
|
|
return KAUTH_RESULT_ALLOW;
|
2006-08-20 19:05:14 +04:00
|
|
|
|
2006-10-02 20:29:57 +04:00
|
|
|
KASSERT(scope != NULL);
|
|
|
|
|
2006-05-15 01:12:38 +04:00
|
|
|
fail = 0;
|
|
|
|
allow = 0;
|
2007-01-16 14:51:22 +03:00
|
|
|
|
2007-02-10 00:55:00 +03:00
|
|
|
/* rw_enter(&kauth_lock, RW_READER); XXX not yet */
|
2006-05-15 01:12:38 +04:00
|
|
|
SIMPLEQ_FOREACH(listener, &scope->listenq, listener_next) {
|
|
|
|
error = listener->func(cred, action, scope->cookie, arg0,
|
2007-02-10 00:55:00 +03:00
|
|
|
arg1, arg2, arg3);
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
if (error == KAUTH_RESULT_ALLOW)
|
|
|
|
allow = 1;
|
|
|
|
else if (error == KAUTH_RESULT_DENY)
|
|
|
|
fail = 1;
|
|
|
|
}
|
2007-02-10 00:55:00 +03:00
|
|
|
/* rw_exit(&kauth_lock); */
|
2006-05-15 01:12:38 +04:00
|
|
|
|
2007-01-16 14:51:22 +03:00
|
|
|
if (fail)
|
2009-09-03 08:45:27 +04:00
|
|
|
return (KAUTH_RESULT_DENY);
|
2007-01-16 14:51:22 +03:00
|
|
|
|
|
|
|
if (allow)
|
2009-09-03 08:45:27 +04:00
|
|
|
return (KAUTH_RESULT_ALLOW);
|
|
|
|
|
|
|
|
return (KAUTH_RESULT_DEFER);
|
|
|
|
};
|
|
|
|
|
|
|
|
int
|
|
|
|
kauth_authorize_action(kauth_scope_t scope, kauth_cred_t cred,
|
|
|
|
kauth_action_t action, void *arg0, void *arg1, void *arg2, void *arg3)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
|
|
|
|
r = kauth_authorize_action_internal(scope, cred, action, arg0, arg1,
|
|
|
|
arg2, arg3);
|
|
|
|
|
|
|
|
if (r == KAUTH_RESULT_DENY)
|
|
|
|
return (EPERM);
|
|
|
|
|
|
|
|
if (r == KAUTH_RESULT_ALLOW)
|
2007-01-16 14:51:22 +03:00
|
|
|
return (0);
|
|
|
|
|
Implement the register/deregister/evaluation API for secmodel(9). It
allows registration of callbacks that can be used later for
cross-secmodel "safe" communication.
When a secmodel wishes to know a property maintained by another
secmodel, it has to submit a request to it so the other secmodel can
proceed to evaluating the request. This is done through the
secmodel_eval(9) call; example:
bool isroot;
error = secmodel_eval("org.netbsd.secmodel.suser", "is-root",
cred, &isroot);
if (error == 0 && !isroot)
result = KAUTH_RESULT_DENY;
This one asks the suser module if the credentials are assumed to be root
when evaluated by suser module. If the module is present, it will
respond. If absent, the call will return an error.
Args and command are arbitrarily defined; it's up to the secmodel(9) to
document what it expects.
Typical example is securelevel testing: when someone wants to know
whether securelevel is raised above a certain level or not, the caller
has to request this property to the secmodel_securelevel(9) module.
Given that securelevel module may be absent from system's context (thus
making access to the global "securelevel" variable impossible or
unsafe), this API can cope with this absence and return an error.
We are using secmodel_eval(9) to implement a secmodel_extensions(9)
module, which plugs with the bsd44, suser and securelevel secmodels
to provide the logic behind curtain, usermount and user_set_cpu_affinity
modes, without adding hooks to traditional secmodels. This solves a
real issue with the current secmodel(9) code, as usermount or
user_set_cpu_affinity are not really tied to secmodel_suser(9).
The secmodel_eval(9) is also used to restrict security.models settings
when securelevel is above 0, through the "is-securelevel-above"
evaluation:
- curtain can be enabled any time, but cannot be disabled if
securelevel is above 0.
- usermount/user_set_cpu_affinity can be disabled any time, but cannot
be enabled if securelevel is above 0.
Regarding sysctl(7) entries:
curtain and usermount are now found under security.models.extensions
tree. The security.curtain and vfs.generic.usermount are still
accessible for backwards compat.
Documentation is incoming, I am proof-reading my writings.
Written by elad@, reviewed and tested (anita test + interact for rights
tests) by me. ok elad@.
See also
http://mail-index.netbsd.org/tech-security/2011/11/29/msg000422.html
XXX might consider va0 mapping too.
XXX Having a secmodel(9) specific printf (like aprint_*) for reporting
secmodel(9) errors might be a good idea, but I am not sure on how
to design such a function right now.
2011-12-04 23:24:58 +04:00
|
|
|
if (secmodel_nsecmodels() == 0)
|
2007-01-16 14:51:22 +03:00
|
|
|
return (0);
|
|
|
|
|
|
|
|
return (EPERM);
|
2009-09-03 08:45:27 +04:00
|
|
|
}
|
2006-05-15 01:12:38 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Generic scope authorization wrapper.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
kauth_authorize_generic(kauth_cred_t cred, kauth_action_t action, void *arg0)
|
|
|
|
{
|
|
|
|
return (kauth_authorize_action(kauth_builtin_scope_generic, cred,
|
|
|
|
action, arg0, NULL, NULL, NULL));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
* System scope authorization wrapper.
|
2006-05-15 01:12:38 +04:00
|
|
|
*/
|
|
|
|
int
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
kauth_authorize_system(kauth_cred_t cred, kauth_action_t action,
|
|
|
|
enum kauth_system_req req, void *arg1, void *arg2, void *arg3)
|
2006-05-15 01:12:38 +04:00
|
|
|
{
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
return (kauth_authorize_action(kauth_builtin_scope_system, cred,
|
|
|
|
action, (void *)req, arg1, arg2, arg3));
|
2006-05-15 01:12:38 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Process scope authorization wrapper.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
kauth_authorize_process(kauth_cred_t cred, kauth_action_t action,
|
|
|
|
struct proc *p, void *arg1, void *arg2, void *arg3)
|
|
|
|
{
|
|
|
|
return (kauth_authorize_action(kauth_builtin_scope_process, cred,
|
|
|
|
action, p, arg1, arg2, arg3));
|
|
|
|
}
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Network scope authorization wrapper.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
kauth_authorize_network(kauth_cred_t cred, kauth_action_t action,
|
2006-09-20 01:42:29 +04:00
|
|
|
enum kauth_network_req req, void *arg1, void *arg2, void *arg3)
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
{
|
|
|
|
return (kauth_authorize_action(kauth_builtin_scope_network, cred,
|
2006-09-20 01:42:29 +04:00
|
|
|
action, (void *)req, arg1, arg2, arg3));
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
kauth_authorize_machdep(kauth_cred_t cred, kauth_action_t action,
|
2006-12-26 13:43:43 +03:00
|
|
|
void *arg0, void *arg1, void *arg2, void *arg3)
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
{
|
|
|
|
return (kauth_authorize_action(kauth_builtin_scope_machdep, cred,
|
2006-12-26 13:43:43 +03:00
|
|
|
action, arg0, arg1, arg2, arg3));
|
First take at security model abstraction.
- Add a few scopes to the kernel: system, network, and machdep.
- Add a few more actions/sub-actions (requests), and start using them as
opposed to the KAUTH_GENERIC_ISSUSER place-holders.
- Introduce a basic set of listeners that implement our "traditional"
security model, called "bsd44". This is the default (and only) model we
have at the moment.
- Update all relevant documentation.
- Add some code and docs to help folks who want to actually use this stuff:
* There's a sample overlay model, sitting on-top of "bsd44", for
fast experimenting with tweaking just a subset of an existing model.
This is pretty cool because it's *really* straightforward to do stuff
you had to use ugly hacks for until now...
* And of course, documentation describing how to do the above for quick
reference, including code samples.
All of these changes were tested for regressions using a Python-based
testsuite that will be (I hope) available soon via pkgsrc. Information
about the tests, and how to write new ones, can be found on:
http://kauth.linbsd.org/kauthwiki
NOTE FOR DEVELOPERS: *PLEASE* don't add any code that does any of the
following:
- Uses a KAUTH_GENERIC_ISSUSER kauth(9) request,
- Checks 'securelevel' directly,
- Checks a uid/gid directly.
(or if you feel you have to, contact me first)
This is still work in progress; It's far from being done, but now it'll
be a lot easier.
Relevant mailing list threads:
http://mail-index.netbsd.org/tech-security/2006/01/25/0011.html
http://mail-index.netbsd.org/tech-security/2006/03/24/0001.html
http://mail-index.netbsd.org/tech-security/2006/04/18/0000.html
http://mail-index.netbsd.org/tech-security/2006/05/15/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/01/0000.html
http://mail-index.netbsd.org/tech-security/2006/08/25/0000.html
Many thanks to YAMAMOTO Takashi, Matt Thomas, and Christos Zoulas for help
stablizing kauth(9).
Full credit for the regression tests, making sure these changes didn't break
anything, goes to Matt Fleming and Jaime Fournier.
Happy birthday Randi! :)
2006-09-09 00:58:56 +04:00
|
|
|
}
|
2006-10-01 00:05:57 +04:00
|
|
|
|
2006-11-19 03:11:29 +03:00
|
|
|
int
|
|
|
|
kauth_authorize_device(kauth_cred_t cred, kauth_action_t action,
|
|
|
|
void *arg0, void *arg1, void *arg2, void *arg3)
|
|
|
|
{
|
|
|
|
return (kauth_authorize_action(kauth_builtin_scope_device, cred,
|
|
|
|
action, arg0, arg1, arg2, arg3));
|
|
|
|
}
|
|
|
|
|
2006-10-01 00:05:57 +04:00
|
|
|
int
|
|
|
|
kauth_authorize_device_tty(kauth_cred_t cred, kauth_action_t action,
|
|
|
|
struct tty *tty)
|
|
|
|
{
|
|
|
|
return (kauth_authorize_action(kauth_builtin_scope_device, cred,
|
|
|
|
action, tty, NULL, NULL, NULL));
|
|
|
|
}
|
2006-11-04 12:30:00 +03:00
|
|
|
|
|
|
|
int
|
|
|
|
kauth_authorize_device_spec(kauth_cred_t cred, enum kauth_device_req req,
|
|
|
|
struct vnode *vp)
|
|
|
|
{
|
|
|
|
return (kauth_authorize_action(kauth_builtin_scope_device, cred,
|
|
|
|
KAUTH_DEVICE_RAWIO_SPEC, (void *)req, vp, NULL, NULL));
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2006-12-02 06:10:42 +03:00
|
|
|
kauth_authorize_device_passthru(kauth_cred_t cred, dev_t dev, u_long bits,
|
|
|
|
void *data)
|
2006-11-04 12:30:00 +03:00
|
|
|
{
|
|
|
|
return (kauth_authorize_action(kauth_builtin_scope_device, cred,
|
2006-12-02 06:10:42 +03:00
|
|
|
KAUTH_DEVICE_RAWIO_PASSTHRU, (void *)bits, (void *)(u_long)dev,
|
|
|
|
data, NULL));
|
2006-11-04 12:30:00 +03:00
|
|
|
}
|
2007-01-16 14:51:22 +03:00
|
|
|
|
2009-09-03 08:45:27 +04:00
|
|
|
kauth_action_t
|
2020-05-16 21:31:45 +03:00
|
|
|
kauth_accmode_to_action(accmode_t accmode)
|
2009-09-03 08:45:27 +04:00
|
|
|
{
|
|
|
|
kauth_action_t action = 0;
|
|
|
|
|
2020-05-16 21:31:45 +03:00
|
|
|
// XXX: Revisit we need to have a richer set of kauth primitives
|
|
|
|
// We also get only the Unix perms here sometimes
|
|
|
|
if (accmode & (VSTAT_PERMS|VREAD))
|
2009-09-03 08:45:27 +04:00
|
|
|
action |= KAUTH_VNODE_READ_DATA;
|
2020-05-16 21:31:45 +03:00
|
|
|
if (accmode & (VMODIFY_PERMS|VADMIN_PERMS))
|
2009-09-03 08:45:27 +04:00
|
|
|
action |= KAUTH_VNODE_WRITE_DATA;
|
2020-05-16 21:31:45 +03:00
|
|
|
if (accmode & VEXEC)
|
2009-09-03 08:45:27 +04:00
|
|
|
action |= KAUTH_VNODE_EXECUTE;
|
2020-05-16 21:31:45 +03:00
|
|
|
return action == 0 ? KAUTH_VNODE_ACCESS : action;
|
2009-09-03 08:45:27 +04:00
|
|
|
}
|
|
|
|
|
2012-03-13 22:40:26 +04:00
|
|
|
kauth_action_t
|
|
|
|
kauth_extattr_action(mode_t access_mode)
|
|
|
|
{
|
|
|
|
kauth_action_t action = 0;
|
|
|
|
|
|
|
|
if (access_mode & VREAD)
|
|
|
|
action |= KAUTH_VNODE_READ_EXTATTRIBUTES;
|
|
|
|
if (access_mode & VWRITE)
|
|
|
|
action |= KAUTH_VNODE_WRITE_EXTATTRIBUTES;
|
|
|
|
|
|
|
|
return action;
|
|
|
|
}
|
|
|
|
|
2009-09-03 08:45:27 +04:00
|
|
|
int
|
|
|
|
kauth_authorize_vnode(kauth_cred_t cred, kauth_action_t action,
|
|
|
|
struct vnode *vp, struct vnode *dvp, int fs_decision)
|
|
|
|
{
|
|
|
|
int error;
|
|
|
|
|
|
|
|
error = kauth_authorize_action_internal(kauth_builtin_scope_vnode, cred,
|
|
|
|
action, vp, dvp, NULL, NULL);
|
|
|
|
|
|
|
|
if (error == KAUTH_RESULT_DENY)
|
|
|
|
return (EACCES);
|
|
|
|
|
|
|
|
if (error == KAUTH_RESULT_ALLOW)
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the file-system does not support decision-before-action, we can
|
|
|
|
* only short-circuit the operation (deny). If we're here, it means no
|
|
|
|
* listener denied it, so our only alternative is to supposedly-allow
|
|
|
|
* it and let the file-system have the last word.
|
|
|
|
*/
|
|
|
|
if (fs_decision == KAUTH_VNODE_REMOTEFS)
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
return (fs_decision);
|
|
|
|
}
|
|
|
|
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
static int
|
|
|
|
kauth_cred_hook(kauth_cred_t cred, kauth_action_t action, void *arg0,
|
|
|
|
void *arg1)
|
|
|
|
{
|
|
|
|
int r;
|
|
|
|
|
|
|
|
r = kauth_authorize_action(kauth_builtin_scope_cred, cred, action,
|
|
|
|
arg0, arg1, NULL, NULL);
|
|
|
|
|
2007-01-31 19:30:09 +03:00
|
|
|
#ifdef DIAGNOSTIC
|
|
|
|
if (!SIMPLEQ_EMPTY(&kauth_builtin_scope_cred->listenq))
|
|
|
|
KASSERT(r == 0);
|
|
|
|
#endif /* DIAGNOSTIC */
|
Add a new scope, the credentials scope, which is internal to the kauth(9)
implementation and meant to be used by security models to hook credential
related operations (init, fork, copy, free -- hooked in kauth_cred_alloc(),
kauth_proc_fork(), kauth_cred_clone(), and kauth_cred_free(), respectively)
and document it.
Add specificdata to credentials, and routines to register/deregister new
"keys", as well as set/get routines. This allows security models to add
their own private data to a kauth_cred_t.
The above two, combined, allow security models to control inheritance of
their own private data in credentials which is a requirement for doing
stuff like, I dunno, capabilities?
2007-01-31 13:08:23 +03:00
|
|
|
|
|
|
|
return (r);
|
|
|
|
}
|