source: rtems-libbsd/rtems/freebsd/rtems/rtems-bsd-thread.c @ a9153ec

4.1155-freebsd-126-freebsd-12freebsd-9.3
Last change on this file since a9153ec was a9153ec, checked in by Joel Sherrill <joel.sherrill@…>, on 03/07/12 at 15:52:04

Initial import

Code is based on FreeBSD 8.2 with USB support from Sebastian Huber
and Thomas Doerfler. Initial TCP/IP stack work is from Kevel Kirspel.

  • Property mode set to 100644
File size: 4.3 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 * The license and distribution terms for this file may be
19 * found in the file LICENSE in this distribution or at
20 * http://www.rtems.com/license/LICENSE.
21 */
22
23#include <rtems/freebsd/machine/rtems-bsd-config.h>
24
25#include <rtems/freebsd/sys/param.h>
26#include <rtems/freebsd/sys/types.h>
27#include <rtems/freebsd/sys/systm.h>
28#include <rtems/freebsd/sys/proc.h>
29#include <rtems/freebsd/sys/kthread.h>
30#include <rtems/freebsd/sys/malloc.h>
31
32RTEMS_CHAIN_DEFINE_EMPTY(rtems_bsd_thread_chain);
33
34static int
35rtems_bsd_thread_start(struct thread **td_ptr, void (*func)(void *), void *arg, int flags, int pages, const char *fmt, va_list ap)
36{
37  struct proc *p = &proc0;;
38        struct thread *td = malloc(sizeof(struct thread), M_TEMP, M_WAITOK | M_ZERO);
39
40        if (td != NULL) {
41                rtems_status_code sc = RTEMS_SUCCESSFUL;
42                rtems_id id = RTEMS_ID_NONE;
43                unsigned index = 0;
44                char name [5] = "_???";
45
46                BSD_ASSERT(pages >= 0);
47
48                sc = rtems_task_create(
49                        rtems_build_name('_', 'T', 'S', 'K'),
50                        BSD_TASK_PRIORITY_NORMAL,
51                        BSD_MINIMUM_TASK_STACK_SIZE + (size_t) pages * PAGE_SIZE,
52                        RTEMS_DEFAULT_ATTRIBUTES,
53                        RTEMS_DEFAULT_ATTRIBUTES,
54                        &id
55                );
56                if (sc != RTEMS_SUCCESSFUL) {
57                        free(td, M_TEMP);
58
59                        return ENOMEM;
60                }
61
62    sc = rtems_task_set_note( id, RTEMS_NOTEPAD_0, ( uint32_t )td );
63    if (sc != RTEMS_SUCCESSFUL) {
64      free(td, M_TEMP);
65
66      return ENOMEM;
67    }
68
69                index = rtems_object_id_get_index(id);
70                snprintf(name + 1, sizeof(name) - 1, "%03u", index);
71                sc = rtems_object_set_name(id, name);
72                if (sc != RTEMS_SUCCESSFUL) {
73                        rtems_task_delete(id);
74                        free(td, M_TEMP);
75
76                        return ENOMEM;
77                }
78
79                sc = rtems_task_start(id, (rtems_task_entry) func, (rtems_task_argument) arg);
80                if (sc != RTEMS_SUCCESSFUL) {
81                        rtems_task_delete(id);
82                        free(td, M_TEMP);
83
84                        return ENOMEM;
85                }
86
87                td->td_id = id;
88                vsnprintf(td->td_name, sizeof(td->td_name), fmt, ap);
89    bzero(&td->td_ru, sizeof(td->td_ru));
90    td->td_ucred = crhold(p->p_ucred);
91    td->td_proc = p;
92
93                rtems_chain_append(&rtems_bsd_thread_chain, &td->td_node);
94
95                if (td_ptr != NULL) {
96                        *td_ptr = td;
97                }
98
99                return 0;
100        }
101
102        return ENOMEM;
103}
104
105static void rtems_bsd_thread_delete(void) __dead2;
106
107static void
108rtems_bsd_thread_delete(void)
109{
110        rtems_chain_control *chain = &rtems_bsd_thread_chain;
111        rtems_chain_node *node = rtems_chain_first(chain);
112        rtems_id id = rtems_task_self();
113        struct thread *td = NULL;
114
115        while (!rtems_chain_is_tail(chain, node)) {
116                struct thread *cur = (struct thread *) node;
117
118                if (cur->td_id == id) {
119                        td = cur;
120                        break;
121                }
122
123                node = rtems_chain_next(node);
124        }
125
126        if (td != NULL) {
127                rtems_chain_extract(&td->td_node);
128
129                free(td, M_TEMP);
130        } else {
131                BSD_PANIC("cannot find task entry");
132        }
133
134        rtems_task_delete(RTEMS_SELF);
135
136        while (true) {
137                /* Do nothing */
138        }
139}
140
141void
142kproc_start(const void *udata)
143{
144        const struct kproc_desc *pd = udata;
145        int eno = kproc_create((void (*)(void *))pd->func, NULL, pd->global_procpp, 0, 0, "%s", pd->arg0);
146
147        BSD_ASSERT(eno == 0);
148}
149
150int
151kproc_create(void (*func)(void *), void *arg, struct proc **newpp, int flags, int pages, const char *fmt, ...)
152{
153        int eno = 0;
154        va_list ap;
155
156        va_start(ap, fmt);
157        eno = rtems_bsd_thread_start(newpp, func, arg, flags, pages, fmt, ap);
158        va_end(ap);
159
160        return eno;
161}
162
163void
164kproc_exit(int ecode)
165{
166        rtems_bsd_thread_delete();
167}
168
169void
170kthread_start(const void *udata)
171{
172        const struct kthread_desc *td = udata;
173        int eno = kthread_add((void (*)(void *)) td->func, NULL, NULL, td->global_threadpp, 0, 0, "%s", td->arg0);
174
175        BSD_ASSERT(eno == 0);
176}
177
178int
179kthread_add(void (*func)(void *), void *arg, struct proc *p, struct thread **newtdp, int flags, int pages, const char *fmt, ...)
180{
181        int eno = 0;
182        va_list ap;
183
184        va_start(ap, fmt);
185        eno = rtems_bsd_thread_start(newtdp, func, arg, flags, pages, fmt, ap);
186        va_end(ap);
187
188        return eno;
189}
190
191void
192kthread_exit(void)
193{
194        rtems_bsd_thread_delete();
195}
196
197int
198kproc_kthread_add(void (*func)(void *), void *arg, struct proc **procptr, struct thread **tdptr, int flags, int pages, const char * procname, const char *fmt, ...)
199{
200        int eno = 0;
201        va_list ap;
202
203        va_start(ap, fmt);
204        eno = rtems_bsd_thread_start(tdptr, func, arg, flags, pages, fmt, ap);
205        va_end(ap);
206
207        return eno;
208}
Note: See TracBrowser for help on using the repository browser.