forked from LineageOS/android_kernel_lge_mako
-
Notifications
You must be signed in to change notification settings - Fork 13
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[PATCH] Keys: Make request-key create an authorisation key
The attached patch makes the following changes: (1) There's a new special key type called ".request_key_auth". This is an authorisation key for when one process requests a key and another process is started to construct it. This type of key cannot be created by the user; nor can it be requested by kernel services. Authorisation keys hold two references: (a) Each refers to a key being constructed. When the key being constructed is instantiated the authorisation key is revoked, rendering it of no further use. (b) The "authorising process". This is either: (i) the process that called request_key(), or: (ii) if the process that called request_key() itself had an authorisation key in its session keyring, then the authorising process referred to by that authorisation key will also be referred to by the new authorisation key. This means that the process that initiated a chain of key requests will authorise the lot of them, and will, by default, wind up with the keys obtained from them in its keyrings. (2) request_key() creates an authorisation key which is then passed to /sbin/request-key in as part of a new session keyring. (3) When request_key() is searching for a key to hand back to the caller, if it comes across an authorisation key in the session keyring of the calling process, it will also search the keyrings of the process specified therein and it will use the specified process's credentials (fsuid, fsgid, groups) to do that rather than the calling process's credentials. This allows a process started by /sbin/request-key to find keys belonging to the authorising process. (4) A key can be read, even if the process executing KEYCTL_READ doesn't have direct read or search permission if that key is contained within the keyrings of a process specified by an authorisation key found within the calling process's session keyring, and is searchable using the credentials of the authorising process. This allows a process started by /sbin/request-key to read keys belonging to the authorising process. (5) The magic KEY_SPEC_*_KEYRING key IDs when passed to KEYCTL_INSTANTIATE or KEYCTL_NEGATE will specify a keyring of the authorising process, rather than the process doing the instantiation. (6) One of the process keyrings can be nominated as the default to which request_key() should attach new keys if not otherwise specified. This is done with KEYCTL_SET_REQKEY_KEYRING and one of the KEY_REQKEY_DEFL_* constants. The current setting can also be read using this call. (7) request_key() is partially interruptible. If it is waiting for another process to finish constructing a key, it can be interrupted. This permits a request-key cycle to be broken without recourse to rebooting. Signed-Off-By: David Howells <[email protected]> Signed-Off-By: Benoit Boissinot <[email protected]> Signed-off-by: Andrew Morton <[email protected]> Signed-off-by: Linus Torvalds <[email protected]>
- Loading branch information
Showing
15 changed files
with
779 additions
and
191 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,4 @@ | ||
/* key-ui.h: key userspace interface stuff for use by keyfs | ||
/* key-ui.h: key userspace interface stuff | ||
* | ||
* Copyright (C) 2004 Red Hat, Inc. All Rights Reserved. | ||
* Written by David Howells ([email protected]) | ||
|
@@ -84,8 +84,45 @@ static inline int key_any_permission(const struct key *key, key_perm_t perm) | |
return kperm != 0; | ||
} | ||
|
||
static inline int key_task_groups_search(struct task_struct *tsk, gid_t gid) | ||
{ | ||
int ret; | ||
|
||
task_lock(tsk); | ||
ret = groups_search(tsk->group_info, gid); | ||
task_unlock(tsk); | ||
return ret; | ||
} | ||
|
||
static inline int key_task_permission(const struct key *key, | ||
struct task_struct *context, | ||
key_perm_t perm) | ||
{ | ||
key_perm_t kperm; | ||
|
||
if (key->uid == context->fsuid) { | ||
kperm = key->perm >> 16; | ||
} | ||
else if (key->gid != -1 && | ||
key->perm & KEY_GRP_ALL && ( | ||
key->gid == context->fsgid || | ||
key_task_groups_search(context, key->gid) | ||
) | ||
) { | ||
kperm = key->perm >> 8; | ||
} | ||
else { | ||
kperm = key->perm; | ||
} | ||
|
||
kperm = kperm & perm & KEY_ALL; | ||
|
||
return kperm == perm; | ||
|
||
} | ||
|
||
extern struct key *lookup_user_key(key_serial_t id, int create, int part, | ||
extern struct key *lookup_user_key(struct task_struct *context, | ||
key_serial_t id, int create, int partial, | ||
key_perm_t perm); | ||
|
||
extern long join_session_keyring(const char *name); | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,6 @@ | ||
/* compat.c: 32-bit compatibility syscall for 64-bit systems | ||
* | ||
* Copyright (C) 2004 Red Hat, Inc. All Rights Reserved. | ||
* Copyright (C) 2004-5 Red Hat, Inc. All Rights Reserved. | ||
* Written by David Howells ([email protected]) | ||
* | ||
* This program is free software; you can redistribute it and/or | ||
|
@@ -24,7 +24,7 @@ | |
* - if you can, you should call sys_keyctl directly | ||
*/ | ||
asmlinkage long compat_sys_keyctl(u32 option, | ||
u32 arg2, u32 arg3, u32 arg4, u32 arg5) | ||
u32 arg2, u32 arg3, u32 arg4, u32 arg5) | ||
{ | ||
switch (option) { | ||
case KEYCTL_GET_KEYRING_ID: | ||
|
@@ -71,6 +71,9 @@ asmlinkage long compat_sys_keyctl(u32 option, | |
case KEYCTL_NEGATE: | ||
return keyctl_negate_key(arg2, arg3, arg4); | ||
|
||
case KEYCTL_SET_REQKEY_KEYRING: | ||
return keyctl_set_reqkey_keyring(arg2); | ||
|
||
default: | ||
return -EOPNOTSUPP; | ||
} | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,6 @@ | ||
/* internal.h: authentication token and access key management internal defs | ||
* | ||
* Copyright (C) 2003 Red Hat, Inc. All Rights Reserved. | ||
* Copyright (C) 2003-5 Red Hat, Inc. All Rights Reserved. | ||
* Written by David Howells ([email protected]) | ||
* | ||
* This program is free software; you can redistribute it and/or | ||
|
@@ -15,6 +15,16 @@ | |
#include <linux/key.h> | ||
#include <linux/key-ui.h> | ||
|
||
#if 0 | ||
#define kenter(FMT, a...) printk("==> %s("FMT")\n",__FUNCTION__ , ## a) | ||
#define kleave(FMT, a...) printk("<== %s()"FMT"\n",__FUNCTION__ , ## a) | ||
#define kdebug(FMT, a...) printk(FMT"\n" , ## a) | ||
#else | ||
#define kenter(FMT, a...) do {} while(0) | ||
#define kleave(FMT, a...) do {} while(0) | ||
#define kdebug(FMT, a...) do {} while(0) | ||
#endif | ||
|
||
extern struct key_type key_type_dead; | ||
extern struct key_type key_type_user; | ||
|
||
|
@@ -66,20 +76,46 @@ extern struct key *__keyring_search_one(struct key *keyring, | |
const char *description, | ||
key_perm_t perm); | ||
|
||
extern struct key *keyring_search_instkey(struct key *keyring, | ||
key_serial_t target_id); | ||
|
||
typedef int (*key_match_func_t)(const struct key *, const void *); | ||
|
||
extern struct key *keyring_search_aux(struct key *keyring, | ||
struct task_struct *tsk, | ||
struct key_type *type, | ||
const void *description, | ||
key_match_func_t match); | ||
|
||
extern struct key *search_process_keyrings_aux(struct key_type *type, | ||
const void *description, | ||
key_match_func_t match); | ||
extern struct key *search_process_keyrings(struct key_type *type, | ||
const void *description, | ||
key_match_func_t match, | ||
struct task_struct *tsk); | ||
|
||
extern struct key *find_keyring_by_name(const char *name, key_serial_t bound); | ||
|
||
extern int install_thread_keyring(struct task_struct *tsk); | ||
extern int install_process_keyring(struct task_struct *tsk); | ||
|
||
extern struct key *request_key_and_link(struct key_type *type, | ||
const char *description, | ||
const char *callout_info, | ||
struct key *dest_keyring); | ||
|
||
/* | ||
* request_key authorisation | ||
*/ | ||
struct request_key_auth { | ||
struct key *target_key; | ||
struct task_struct *context; | ||
pid_t pid; | ||
}; | ||
|
||
extern struct key_type key_type_request_key_auth; | ||
extern struct key *request_key_auth_new(struct key *target, | ||
struct key **_rkakey); | ||
|
||
extern struct key *key_get_instantiation_authkey(key_serial_t target_id); | ||
|
||
/* | ||
* keyctl functions | ||
|
@@ -100,6 +136,7 @@ extern long keyctl_setperm_key(key_serial_t, key_perm_t); | |
extern long keyctl_instantiate_key(key_serial_t, const void __user *, | ||
size_t, key_serial_t); | ||
extern long keyctl_negate_key(key_serial_t, unsigned, key_serial_t); | ||
extern long keyctl_set_reqkey_keyring(int); | ||
|
||
|
||
/* | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,6 +1,6 @@ | ||
/* key.c: basic authentication token and access key management | ||
* | ||
* Copyright (C) 2004 Red Hat, Inc. All Rights Reserved. | ||
* Copyright (C) 2004-5 Red Hat, Inc. All Rights Reserved. | ||
* Written by David Howells ([email protected]) | ||
* | ||
* This program is free software; you can redistribute it and/or | ||
|
@@ -391,7 +391,8 @@ EXPORT_SYMBOL(key_payload_reserve); | |
static int __key_instantiate_and_link(struct key *key, | ||
const void *data, | ||
size_t datalen, | ||
struct key *keyring) | ||
struct key *keyring, | ||
struct key *instkey) | ||
{ | ||
int ret, awaken; | ||
|
||
|
@@ -419,6 +420,10 @@ static int __key_instantiate_and_link(struct key *key, | |
/* and link it into the destination keyring */ | ||
if (keyring) | ||
ret = __key_link(keyring, key); | ||
|
||
/* disable the authorisation key */ | ||
if (instkey) | ||
key_revoke(instkey); | ||
} | ||
} | ||
|
||
|
@@ -439,19 +444,21 @@ static int __key_instantiate_and_link(struct key *key, | |
int key_instantiate_and_link(struct key *key, | ||
const void *data, | ||
size_t datalen, | ||
struct key *keyring) | ||
struct key *keyring, | ||
struct key *instkey) | ||
{ | ||
int ret; | ||
|
||
if (keyring) | ||
down_write(&keyring->sem); | ||
|
||
ret = __key_instantiate_and_link(key, data, datalen, keyring); | ||
ret = __key_instantiate_and_link(key, data, datalen, keyring, instkey); | ||
|
||
if (keyring) | ||
up_write(&keyring->sem); | ||
|
||
return ret; | ||
|
||
} /* end key_instantiate_and_link() */ | ||
|
||
EXPORT_SYMBOL(key_instantiate_and_link); | ||
|
@@ -462,7 +469,8 @@ EXPORT_SYMBOL(key_instantiate_and_link); | |
*/ | ||
int key_negate_and_link(struct key *key, | ||
unsigned timeout, | ||
struct key *keyring) | ||
struct key *keyring, | ||
struct key *instkey) | ||
{ | ||
struct timespec now; | ||
int ret, awaken; | ||
|
@@ -495,6 +503,10 @@ int key_negate_and_link(struct key *key, | |
/* and link it into the destination keyring */ | ||
if (keyring) | ||
ret = __key_link(keyring, key); | ||
|
||
/* disable the authorisation key */ | ||
if (instkey) | ||
key_revoke(instkey); | ||
} | ||
|
||
up_write(&key_construction_sem); | ||
|
@@ -781,7 +793,7 @@ struct key *key_create_or_update(struct key *keyring, | |
} | ||
|
||
/* instantiate it and link it into the target keyring */ | ||
ret = __key_instantiate_and_link(key, payload, plen, keyring); | ||
ret = __key_instantiate_and_link(key, payload, plen, keyring, NULL); | ||
if (ret < 0) { | ||
key_put(key); | ||
key = ERR_PTR(ret); | ||
|
Oops, something went wrong.