source: rtems-libbsd/rtemsbsd/src/rtems-bsd-thread.c @ b41a358

4.1155-freebsd-126-freebsd-12freebsd-9.3
Last change on this file since b41a358 was b41a358, checked in by Sebastian Huber <sebastian.huber@…>, on 04/27/12 at 07:38:25

Fix ucred reference counting

  • Property mode set to 100644
File size: 5.4 KB
Line 
1/**
2 * @file
3 *
4 * @ingroup rtems_bsd_rtems
5 *
6 * @brief TODO.
7 */
8
9/*
10 * Copyright (c) 2009, 2010 embedded brains GmbH.  All rights reserved.
11 *
12 *  embedded brains GmbH
13 *  Obere Lagerstr. 30
14 *  82178 Puchheim
15 *  Germany
16 *  <rtems@embedded-brains.de>
17 *
18 * Redistribution and use in source and binary forms, with or without
19 * modification, are permitted provided that the following conditions
20 * are met:
21 * 1. Redistributions of source code must retain the above copyright
22 *    notice, this list of conditions and the following disclaimer.
23 * 2. Redistributions in binary form must reproduce the above copyright
24 *    notice, this list of conditions and the following disclaimer in the
25 *    documentation and/or other materials provided with the distribution.
26 *
27 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
28 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
29 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
30 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
31 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
32 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
33 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
34 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
35 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
36 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
37 * SUCH DAMAGE.
38 */
39
40#include <freebsd/machine/rtems-bsd-config.h>
41
42#include <freebsd/sys/param.h>
43#include <freebsd/sys/types.h>
44#include <freebsd/sys/systm.h>
45#include <freebsd/sys/proc.h>
46#include <freebsd/sys/kthread.h>
47#include <freebsd/sys/malloc.h>
48
49RTEMS_CHAIN_DEFINE_EMPTY(rtems_bsd_thread_chain);
50
51/* FIXME: What to do with the credentials? */
52static struct ucred FIXME_ucred = {
53  .cr_ref = 1
54};
55
56static int
57rtems_bsd_thread_start(struct thread **td_ptr, void (*func)(void *), void *arg, int flags, int pages, const char *fmt, va_list ap)
58{
59        struct thread *td = malloc(sizeof(struct thread), M_TEMP, M_WAITOK | M_ZERO);
60
61        if (td != NULL) {
62                rtems_status_code sc = RTEMS_SUCCESSFUL;
63                rtems_id id = RTEMS_ID_NONE;
64                unsigned index = 0;
65                char name [5] = "_???";
66
67                BSD_ASSERT(pages >= 0);
68
69                sc = rtems_task_create(
70                        rtems_build_name('_', 'T', 'S', 'K'),
71                        BSD_TASK_PRIORITY_NORMAL,
72                        BSD_MINIMUM_TASK_STACK_SIZE + (size_t) pages * PAGE_SIZE,
73                        RTEMS_DEFAULT_ATTRIBUTES,
74                        RTEMS_DEFAULT_ATTRIBUTES,
75                        &id
76                );
77                if (sc != RTEMS_SUCCESSFUL) {
78                        free(td, M_TEMP);
79
80                        return ENOMEM;
81                }
82
83    sc = rtems_task_set_note( id, RTEMS_NOTEPAD_0, ( uint32_t )td );
84    if (sc != RTEMS_SUCCESSFUL) {
85      free(td, M_TEMP);
86
87      return ENOMEM;
88    }
89
90                index = rtems_object_id_get_index(id);
91                snprintf(name + 1, sizeof(name) - 1, "%03u", index);
92                sc = rtems_object_set_name(id, name);
93                if (sc != RTEMS_SUCCESSFUL) {
94                        rtems_task_delete(id);
95                        free(td, M_TEMP);
96
97                        return ENOMEM;
98                }
99
100                sc = rtems_task_start(id, (rtems_task_entry) func, (rtems_task_argument) arg);
101                if (sc != RTEMS_SUCCESSFUL) {
102                        rtems_task_delete(id);
103                        free(td, M_TEMP);
104
105                        return ENOMEM;
106                }
107
108                td->td_id = id;
109                vsnprintf(td->td_name, sizeof(td->td_name), fmt, ap);
110                td->td_ucred = crhold(&FIXME_ucred);
111
112                rtems_chain_append(&rtems_bsd_thread_chain, &td->td_node);
113
114                if (td_ptr != NULL) {
115                        *td_ptr = td;
116                }
117
118                return 0;
119        }
120
121        return ENOMEM;
122}
123
124static void rtems_bsd_thread_delete(void) __dead2;
125
126static void
127rtems_bsd_thread_delete(void)
128{
129        rtems_chain_control *chain = &rtems_bsd_thread_chain;
130        rtems_chain_node *node = rtems_chain_first(chain);
131        rtems_id id = rtems_task_self();
132        struct thread *td = NULL;
133
134        while (!rtems_chain_is_tail(chain, node)) {
135                struct thread *cur = (struct thread *) node;
136
137                if (cur->td_id == id) {
138                        td = cur;
139                        break;
140                }
141
142                node = rtems_chain_next(node);
143        }
144
145        if (td != NULL) {
146                rtems_chain_extract(&td->td_node);
147
148                free(td, M_TEMP);
149        } else {
150                BSD_PANIC("cannot find task entry");
151        }
152
153        rtems_task_delete(RTEMS_SELF);
154
155        while (true) {
156                /* Do nothing */
157        }
158}
159
160void
161kproc_start(const void *udata)
162{
163        const struct kproc_desc *pd = udata;
164        int eno = kproc_create((void (*)(void *))pd->func, NULL, pd->global_procpp, 0, 0, "%s", pd->arg0);
165
166        BSD_ASSERT(eno == 0);
167}
168
169int
170kproc_create(void (*func)(void *), void *arg, struct proc **newpp, int flags, int pages, const char *fmt, ...)
171{
172        int eno = 0;
173        va_list ap;
174
175        va_start(ap, fmt);
176        eno = rtems_bsd_thread_start(newpp, func, arg, flags, pages, fmt, ap);
177        va_end(ap);
178
179        return eno;
180}
181
182void
183kproc_exit(int ecode)
184{
185        rtems_bsd_thread_delete();
186}
187
188void
189kthread_start(const void *udata)
190{
191        const struct kthread_desc *td = udata;
192        int eno = kthread_add((void (*)(void *)) td->func, NULL, NULL, td->global_threadpp, 0, 0, "%s", td->arg0);
193
194        BSD_ASSERT(eno == 0);
195}
196
197int
198kthread_add(void (*func)(void *), void *arg, struct proc *p, struct thread **newtdp, int flags, int pages, const char *fmt, ...)
199{
200        int eno = 0;
201        va_list ap;
202
203        va_start(ap, fmt);
204        eno = rtems_bsd_thread_start(newtdp, func, arg, flags, pages, fmt, ap);
205        va_end(ap);
206
207        return eno;
208}
209
210void
211kthread_exit(void)
212{
213        rtems_bsd_thread_delete();
214}
215
216int
217kproc_kthread_add(void (*func)(void *), void *arg, struct proc **procptr, struct thread **tdptr, int flags, int pages, const char * procname, const char *fmt, ...)
218{
219        int eno = 0;
220        va_list ap;
221
222        va_start(ap, fmt);
223        eno = rtems_bsd_thread_start(tdptr, func, arg, flags, pages, fmt, ap);
224        va_end(ap);
225
226        return eno;
227}
Note: See TracBrowser for help on using the repository browser.