_lwp.c revision 1.1.6.1 1 1.1.6.1 yamt /* $NetBSD: _lwp.c,v 1.1.6.1 2012/04/17 00:05:15 yamt Exp $ */
2 1.1 christos
3 1.1 christos /*-
4 1.1 christos * Copyright (c) 2009 The NetBSD Foundation, Inc.
5 1.1 christos * All rights reserved.
6 1.1 christos *
7 1.1 christos * This code is derived from software contributed to The NetBSD Foundation
8 1.1 christos * by Christos Zoulas
9 1.1 christos *
10 1.1 christos * Redistribution and use in source and binary forms, with or without
11 1.1 christos * modification, are permitted provided that the following conditions
12 1.1 christos * are met:
13 1.1 christos * 1. Redistributions of source code must retain the above copyright
14 1.1 christos * notice, this list of conditions and the following disclaimer.
15 1.1 christos * 2. Redistributions in binary form must reproduce the above copyright
16 1.1 christos * notice, this list of conditions and the following disclaimer in the
17 1.1 christos * documentation and/or other materials provided with the distribution.
18 1.1 christos *
19 1.1 christos * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 1.1 christos * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 1.1 christos * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 1.1 christos * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 1.1 christos * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 1.1 christos * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 1.1 christos * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 1.1 christos * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 1.1 christos * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 1.1 christos * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 1.1 christos * POSSIBILITY OF SUCH DAMAGE.
30 1.1 christos */
31 1.1 christos
32 1.1 christos #include <sys/cdefs.h>
33 1.1 christos #if defined(LIBC_SCCS) && !defined(lint)
34 1.1.6.1 yamt __RCSID("$NetBSD: _lwp.c,v 1.1.6.1 2012/04/17 00:05:15 yamt Exp $");
35 1.1 christos #endif /* LIBC_SCCS and not lint */
36 1.1 christos
37 1.1 christos #include "namespace.h"
38 1.1 christos #include <sys/types.h>
39 1.1 christos #include <inttypes.h>
40 1.1 christos #include <ucontext.h>
41 1.1 christos #include <lwp.h>
42 1.1 christos #include <stdlib.h>
43 1.1 christos
44 1.1 christos void
45 1.1 christos _lwp_makecontext(ucontext_t *u, void (*start)(void *),
46 1.1 christos void *arg, void *private, caddr_t stack_base, size_t stack_size)
47 1.1 christos {
48 1.1 christos __greg_t *gr = u->uc_mcontext.__gregs;
49 1.1 christos int *sp;
50 1.1 christos
51 1.1 christos getcontext(u);
52 1.1 christos u->uc_link = NULL;
53 1.1 christos
54 1.1 christos u->uc_stack.ss_sp = stack_base;
55 1.1 christos u->uc_stack.ss_size = stack_size;
56 1.1 christos
57 1.1 christos /* Align to a word */
58 1.1 christos /* LINTED uintptr_t is safe */
59 1.1 christos sp = (int *)((uintptr_t)(stack_base + stack_size) & ~0x3);
60 1.1 christos
61 1.1 christos /*
62 1.1 christos * Allocate necessary stack space for arguments including arg count
63 1.1 christos * and call frame
64 1.1 christos */
65 1.1 christos sp -= 1 + 1 + 5;
66 1.1 christos
67 1.1 christos sp[0] = 0; /* condition handler is null */
68 1.1 christos sp[1] = 0x20000000; /* make this a CALLS frame */
69 1.1 christos sp[2] = 0; /* saved argument pointer */
70 1.1 christos sp[3] = 0; /* saved frame pointer */
71 1.1.6.1 yamt sp[4] = (int)(uintptr_t)_lwp_exit + 2;/* return via _lwp_exit */
72 1.1 christos sp[5] = 1; /* argc */
73 1.1.6.1 yamt sp[6] = (int)(uintptr_t)arg; /* argv */
74 1.1 christos
75 1.1.6.1 yamt gr[_REG_AP] = (__greg_t)(uintptr_t)(sp + 5);
76 1.1.6.1 yamt gr[_REG_SP] = (__greg_t)(uintptr_t)sp;
77 1.1.6.1 yamt gr[_REG_FP] = (__greg_t)(uintptr_t)sp;
78 1.1.6.1 yamt gr[_REG_PC] = (__greg_t)(uintptr_t)start + 2;
79 1.1.6.1 yamt
80 1.1.6.1 yamt /*
81 1.1.6.1 yamt * Push the TLS pointer onto the new stack also.
82 1.1.6.1 yamt * The _UC_TLSBASE flag tells the kernel to pop it and use it.
83 1.1.6.1 yamt */
84 1.1.6.1 yamt *--sp = (int)(intptr_t)private;
85 1.1.6.1 yamt gr[_REG_SP] = (__greg_t)(uintptr_t)sp;
86 1.1.6.1 yamt u->uc_flags |= _UC_TLSBASE;
87 1.1 christos }
88