aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kernel/process.c
diff options
context:
space:
mode:
Diffstat (limited to 'arch/x86/kernel/process.c')
-rw-r--r--arch/x86/kernel/process.c35
1 files changed, 35 insertions, 0 deletions
diff --git a/arch/x86/kernel/process.c b/arch/x86/kernel/process.c
index f3c1a6b3a65e..8705ccedd447 100644
--- a/arch/x86/kernel/process.c
+++ b/arch/x86/kernel/process.c
@@ -243,6 +243,41 @@ sys_clone(unsigned long clone_flags, unsigned long newsp,
243 return do_fork(clone_flags, newsp, regs, 0, parent_tid, child_tid); 243 return do_fork(clone_flags, newsp, regs, 0, parent_tid, child_tid);
244} 244}
245 245
246/*
247 * This gets run with %si containing the
248 * function to call, and %di containing
249 * the "args".
250 */
251extern void kernel_thread_helper(void);
252
253/*
254 * Create a kernel thread
255 */
256int kernel_thread(int (*fn)(void *), void *arg, unsigned long flags)
257{
258 struct pt_regs regs;
259
260 memset(&regs, 0, sizeof(regs));
261
262 regs.si = (unsigned long) fn;
263 regs.di = (unsigned long) arg;
264
265#ifdef CONFIG_X86_32
266 regs.ds = __USER_DS;
267 regs.es = __USER_DS;
268 regs.fs = __KERNEL_PERCPU;
269 regs.gs = __KERNEL_STACK_CANARY;
270#endif
271
272 regs.orig_ax = -1;
273 regs.ip = (unsigned long) kernel_thread_helper;
274 regs.cs = __KERNEL_CS | get_kernel_rpl();
275 regs.flags = X86_EFLAGS_IF | 0x2;
276
277 /* Ok, create the new process.. */
278 return do_fork(flags | CLONE_VM | CLONE_UNTRACED, 0, &regs, 0, NULL, NULL);
279}
280EXPORT_SYMBOL(kernel_thread);
246 281
247/* 282/*
248 * sys_execve() executes a new program. 283 * sys_execve() executes a new program.