aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorOleg Nesterov <oleg@redhat.com>2010-05-26 17:43:00 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2010-05-27 12:12:45 -0400
commit685bfd2c48bb3284d31e73ff3151c957d76deda9 (patch)
tree177210787515f48c0eaad5216bd012f4a2fb2149
parent898b374af6f71041bd3bceebe257e564f3f1d458 (diff)
umh: creds: convert call_usermodehelper_keys() to use subprocess_info->init()
call_usermodehelper_keys() uses call_usermodehelper_setkeys() to change subprocess_info->cred in advance. Now that we have info->init() we can change this code to set tgcred->session_keyring in context of execing kernel thread. Note: since currently call_usermodehelper_keys() is never called with UMH_NO_WAIT, call_usermodehelper_keys()->key_get() and umh_keys_cleanup() are not really needed, we could rely on install_session_keyring_to_cred() which does key_get() on success. Signed-off-by: Oleg Nesterov <oleg@redhat.com> Acked-by: Neil Horman <nhorman@tuxdriver.com> Acked-by: David Howells <dhowells@redhat.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r--include/linux/kmod.h17
-rw-r--r--kernel/kmod.c18
-rw-r--r--security/keys/internal.h1
-rw-r--r--security/keys/process_keys.c3
-rw-r--r--security/keys/request_key.c32
5 files changed, 34 insertions, 37 deletions
diff --git a/include/linux/kmod.h b/include/linux/kmod.h
index 5c058778ad35..d876dce217f0 100644
--- a/include/linux/kmod.h
+++ b/include/linux/kmod.h
@@ -72,8 +72,6 @@ struct subprocess_info *call_usermodehelper_setup(char *path, char **argv,
72 char **envp, gfp_t gfp_mask); 72 char **envp, gfp_t gfp_mask);
73 73
74/* Set various pieces of state into the subprocess_info structure */ 74/* Set various pieces of state into the subprocess_info structure */
75void call_usermodehelper_setkeys(struct subprocess_info *info,
76 struct key *session_keyring);
77void call_usermodehelper_setfns(struct subprocess_info *info, 75void call_usermodehelper_setfns(struct subprocess_info *info,
78 int (*init)(struct subprocess_info *info), 76 int (*init)(struct subprocess_info *info),
79 void (*cleanup)(struct subprocess_info *info), 77 void (*cleanup)(struct subprocess_info *info),
@@ -112,21 +110,6 @@ call_usermodehelper(char *path, char **argv, char **envp, enum umh_wait wait)
112 NULL, NULL, NULL); 110 NULL, NULL, NULL);
113} 111}
114 112
115static inline int
116call_usermodehelper_keys(char *path, char **argv, char **envp,
117 struct key *session_keyring, enum umh_wait wait)
118{
119 struct subprocess_info *info;
120 gfp_t gfp_mask = (wait == UMH_NO_WAIT) ? GFP_ATOMIC : GFP_KERNEL;
121
122 info = call_usermodehelper_setup(path, argv, envp, gfp_mask);
123 if (info == NULL)
124 return -ENOMEM;
125
126 call_usermodehelper_setkeys(info, session_keyring);
127 return call_usermodehelper_exec(info, wait);
128}
129
130extern void usermodehelper_init(void); 113extern void usermodehelper_init(void);
131 114
132extern int usermodehelper_disable(void); 115extern int usermodehelper_disable(void);
diff --git a/kernel/kmod.c b/kernel/kmod.c
index d1544547fea6..743d3883517c 100644
--- a/kernel/kmod.c
+++ b/kernel/kmod.c
@@ -367,24 +367,6 @@ struct subprocess_info *call_usermodehelper_setup(char *path, char **argv,
367EXPORT_SYMBOL(call_usermodehelper_setup); 367EXPORT_SYMBOL(call_usermodehelper_setup);
368 368
369/** 369/**
370 * call_usermodehelper_setkeys - set the session keys for usermode helper
371 * @info: a subprocess_info returned by call_usermodehelper_setup
372 * @session_keyring: the session keyring for the process
373 */
374void call_usermodehelper_setkeys(struct subprocess_info *info,
375 struct key *session_keyring)
376{
377#ifdef CONFIG_KEYS
378 struct thread_group_cred *tgcred = info->cred->tgcred;
379 key_put(tgcred->session_keyring);
380 tgcred->session_keyring = key_get(session_keyring);
381#else
382 BUG();
383#endif
384}
385EXPORT_SYMBOL(call_usermodehelper_setkeys);
386
387/**
388 * call_usermodehelper_setfns - set a cleanup/init function 370 * call_usermodehelper_setfns - set a cleanup/init function
389 * @info: a subprocess_info returned by call_usermodehelper_setup 371 * @info: a subprocess_info returned by call_usermodehelper_setup
390 * @cleanup: a cleanup function 372 * @cleanup: a cleanup function
diff --git a/security/keys/internal.h b/security/keys/internal.h
index 5d4402a1161a..38783dcf6c61 100644
--- a/security/keys/internal.h
+++ b/security/keys/internal.h
@@ -124,6 +124,7 @@ extern struct key *find_keyring_by_name(const char *name, bool skip_perm_check);
124extern int install_user_keyrings(void); 124extern int install_user_keyrings(void);
125extern int install_thread_keyring_to_cred(struct cred *); 125extern int install_thread_keyring_to_cred(struct cred *);
126extern int install_process_keyring_to_cred(struct cred *); 126extern int install_process_keyring_to_cred(struct cred *);
127extern int install_session_keyring_to_cred(struct cred *, struct key *);
127 128
128extern struct key *request_key_and_link(struct key_type *type, 129extern struct key *request_key_and_link(struct key_type *type,
129 const char *description, 130 const char *description,
diff --git a/security/keys/process_keys.c b/security/keys/process_keys.c
index 20a38fed61b1..6b8e4ff4cc68 100644
--- a/security/keys/process_keys.c
+++ b/security/keys/process_keys.c
@@ -216,8 +216,7 @@ static int install_process_keyring(void)
216/* 216/*
217 * install a session keyring directly to a credentials struct 217 * install a session keyring directly to a credentials struct
218 */ 218 */
219static int install_session_keyring_to_cred(struct cred *cred, 219int install_session_keyring_to_cred(struct cred *cred, struct key *keyring)
220 struct key *keyring)
221{ 220{
222 unsigned long flags; 221 unsigned long flags;
223 struct key *old; 222 struct key *old;
diff --git a/security/keys/request_key.c b/security/keys/request_key.c
index f656e9c069e3..f5ec9ac5d57c 100644
--- a/security/keys/request_key.c
+++ b/security/keys/request_key.c
@@ -58,6 +58,38 @@ void complete_request_key(struct key_construction *cons, int error)
58} 58}
59EXPORT_SYMBOL(complete_request_key); 59EXPORT_SYMBOL(complete_request_key);
60 60
61static int umh_keys_init(struct subprocess_info *info)
62{
63 struct cred *cred = (struct cred*)current_cred();
64 struct key *keyring = info->data;
65 /*
66 * This is called in context of freshly forked kthread before
67 * kernel_execve(), we can just change our ->session_keyring.
68 */
69 return install_session_keyring_to_cred(cred, keyring);
70}
71
72static void umh_keys_cleanup(struct subprocess_info *info)
73{
74 struct key *keyring = info->data;
75 key_put(keyring);
76}
77
78static int call_usermodehelper_keys(char *path, char **argv, char **envp,
79 struct key *session_keyring, enum umh_wait wait)
80{
81 gfp_t gfp_mask = (wait == UMH_NO_WAIT) ? GFP_ATOMIC : GFP_KERNEL;
82 struct subprocess_info *info =
83 call_usermodehelper_setup(path, argv, envp, gfp_mask);
84
85 if (!info)
86 return -ENOMEM;
87
88 call_usermodehelper_setfns(info, umh_keys_init, umh_keys_cleanup,
89 key_get(session_keyring));
90 return call_usermodehelper_exec(info, wait);
91}
92
61/* 93/*
62 * request userspace finish the construction of a key 94 * request userspace finish the construction of a key
63 * - execute "/sbin/request-key <op> <key> <uid> <gid> <keyring> <keyring> <keyring>" 95 * - execute "/sbin/request-key <op> <key> <uid> <gid> <keyring> <keyring> <keyring>"