source: libcfa/src/concurrency/thread.cfa@ 959cc59

Last change on this file since 959cc59 was 5d3d281, checked in by Michael Brooks <mlbrooks@…>, 11 months ago

Remove autogen forward declarations, which are never needed, and cause warnings about static declarations without definitions.

Intended to fix the failing test from previous commit.

Autogen forward declarations are never needed because they do not depend on each other, much less with mutual recursion.

Consequences:

  • tests/.expect/(5 tests).(3 archs).txt: Accept generated code that lacks autogen forward declarations
  • libcfa/src/concurrency/thread.*: Remove unused dependency on destructor from constructor (via thrd_start), by splitting trait is_thread with is_basic_thread
  • Property mode set to 100644
File size: 7.2 KB
Line 
1//
2// Cforall Version 1.0.0 Copyright (C) 2016 University of Waterloo
3//
4// The contents of this file are covered under the licence agreement in the
5// file "LICENCE" distributed with Cforall.
6//
7// thread.c --
8//
9// Author : Thierry Delisle
10// Created On : Tue Jan 17 12:27:26 2017
11// Last Modified By : Peter A. Buhr
12// Last Modified On : Sun Sep 3 08:25:21 2023
13// Update Count : 105
14//
15
16#define __cforall_thread__
17
18#include "thread.hfa"
19
20#include "exception.hfa"
21#include "kernel/private.hfa"
22#include "limits.hfa"
23
24#define __CFA_INVOKE_PRIVATE__
25#include "invoke.h"
26
27extern size_t __global_random_seed;
28extern size_t __global_random_prime;
29extern bool __global_random_mask;
30
31#pragma GCC visibility push(default)
32
33//-----------------------------------------------------------------------------
34// Thread ctors and dtors
35void ?{}( thread$ & this, const char * const name, cluster & cl, void * storage, size_t storageSize ) with( this ) {
36 context{ 0p, 0p };
37 self_cor{ name, storage, storageSize };
38 ticket = TICKET_RUNNING;
39 state = Start;
40 preempted = __NO_PREEMPTION;
41 corctx_flag = false;
42 curr_cor = &self_cor;
43 self_mon.owner = &this;
44 self_mon.recursion = 1;
45 self_mon_p = &self_mon;
46 curr_cluster = &cl;
47 rdy_link.next = 0p;
48 rdy_link.ts = MAX;
49 user_link.next = 0p;
50 user_link.prev = 0p;
51 cltr_link.next = 0p;
52 cltr_link.prev = 0p;
53 preferred = ready_queue_new_preferred();
54 last_proc = 0p;
55 link_node = 0p;
56 PRNG_SET_SEED( random_state, __global_random_mask ? __global_random_prime : __global_random_prime ^ rdtscl() );
57 #if defined( __CFA_WITH_VERIFY__ )
58 executing = 0p;
59 canary = 0x0D15EA5E0D15EA5Ep;
60 #endif
61
62 doregister(curr_cluster, this);
63 monitors{ &self_mon_p, 1, (fptr_t)0 };
64}
65
66void ^?{}(thread$& this) with( this ) {
67 #if defined( __CFA_WITH_VERIFY__ )
68 canary = 0xDEADDEADDEADDEADp;
69 #endif
70 unregister(curr_cluster, this);
71 ^self_cor{};
72}
73
74forall(T &)
75void copy(ThreadCancelled(T) * dst, ThreadCancelled(T) * src) {
76 dst->virtual_table = src->virtual_table;
77 dst->the_thread = src->the_thread;
78 dst->the_exception = src->the_exception;
79}
80
81forall(T &)
82const char * msg(ThreadCancelled(T) *) {
83 return "ThreadCancelled(...)";
84}
85
86forall(T &)
87static void default_thread_cancel_handler(ThreadCancelled(T) & ) {
88 // Improve this error message, can I do formatting?
89 abort( "Unhandled thread cancellation.\n" );
90}
91
92forall(T & | is_thread(T) | IS_EXCEPTION(ThreadCancelled(T))
93 | { EHM_DEFAULT_VTABLE(ThreadCancelled(T)); })
94void ?{}( thread_dtor_guard_t & this,
95 T & thrd, void(*cancelHandler)(ThreadCancelled(T) &)) {
96 monitor$ * m = get_monitor(thrd);
97 thread$ * desc = get_thread(thrd);
98
99 // Setup the monitor guard
100 void (*dtor)(T& mutex this) = ^?{};
101 bool join = cancelHandler != (void(*)(ThreadCancelled(T)&))0;
102 (this.mg){&m, (void(*)())dtor, join};
103
104
105 /* paranoid */ verifyf( Halted == desc->state || Cancelled == desc->state, "Expected thread to be Halted or Cancelled, was %d\n", (int)desc->state );
106
107 // After the guard set-up and any wait, check for cancellation.
108 struct _Unwind_Exception * cancellation = desc->self_cor.cancellation;
109 if ( likely( 0p == cancellation ) ) {
110 return;
111 } else if ( Cancelled == desc->state ) {
112 return;
113 }
114 desc->state = Cancelled;
115 void(*defaultResumptionHandler)(ThreadCancelled(T) &) =
116 join ? cancelHandler : default_thread_cancel_handler;
117
118 // TODO: Remove explitate vtable set once trac#186 is fixed.
119 ThreadCancelled(T) except;
120 except.virtual_table = &_default_vtable;
121 except.the_thread = &thrd;
122 except.the_exception = __cfaehm_cancellation_exception( cancellation );
123 // Why is this cast required?
124 throwResume (ThreadCancelled(T) &)except;
125
126 except.the_exception->virtual_table->free( except.the_exception );
127 free( cancellation );
128 desc->self_cor.cancellation = 0p;
129}
130
131void ^?{}( thread_dtor_guard_t & this ) {
132 ^(this.mg){};
133}
134
135//-----------------------------------------------------------------------------
136// Starting and stopping threads
137forall( T & | is_basic_thread(T) )
138void __thrd_start( T & this, void (*main_p)(T &) ) {
139 thread$ * this_thrd = get_thread(this);
140
141 disable_interrupts();
142 __cfactx_start(main_p, get_coroutine(this), this, __cfactx_invoke_thread);
143
144 this_thrd->context.[SP, FP] = this_thrd->self_cor.context.[SP, FP];
145 /* paranoid */ verify( this_thrd->context.SP );
146
147 schedule_thread$( this_thrd, UNPARK_LOCAL );
148 enable_interrupts();
149}
150
151//-----------------------------------------------------------------------------
152// Support for threads that don't ues the thread keyword
153forall( T & | sized(T) | is_thread(T) | { void ?{}(T&); } )
154void ?{}( scoped(T)& this ) with( this ) {
155 handle{};
156 __thrd_start(handle, main);
157}
158
159forall( T &, P... | sized(T) | is_thread(T) | { void ?{}(T&, P); } )
160void ?{}( scoped(T)& this, P params ) with( this ) {
161 handle{ params };
162 __thrd_start(handle, main);
163}
164
165forall( T & | sized(T) | is_thread(T) )
166void ^?{}( scoped(T)& this ) with( this ) {
167 ^handle{};
168}
169
170//-----------------------------------------------------------------------------
171forall(T & | is_thread(T) | IS_RESUMPTION_EXCEPTION(ThreadCancelled(T))
172 | { EHM_DEFAULT_VTABLE(ThreadCancelled(T)); })
173T & join( T & this ) {
174 thread_dtor_guard_t guard = { this, defaultResumptionHandler };
175 return this;
176}
177
178//-----------------------------------------------------------------------------
179bool migrate( thread$ * thrd, struct cluster & cl ) {
180 monitor$ * tmon = get_monitor(thrd);
181 monitor$ * __monitors[] = { tmon };
182 monitor_guard_t __guard = { __monitors, 1 };
183 {
184 // if nothing needs to be done, return false
185 if( thrd->curr_cluster == &cl ) return false;
186
187 // are we migrating ourself?
188 const bool local = thrd == active_thread();
189
190 /* paranoid */ verify( !local || &cl != active_cluster() );
191 /* paranoid */ verify( !local || thrd->curr_cluster == active_cluster() );
192 /* paranoid */ verify( !local || thrd->curr_cluster == active_processor()->cltr );
193 /* paranoid */ verify( local || tmon->signal_stack.top->owner->waiting_thread == thrd );
194 /* paranoid */ verify( local || tmon->signal_stack.top );
195
196 // make sure we aren't interrupted while doing this
197 // not as important if we aren't local
198 disable_interrupts();
199
200 // actually move the thread
201 unregister( thrd->curr_cluster, *thrd );
202 thrd->curr_cluster = &cl;
203 doregister( thrd->curr_cluster, *thrd );
204
205 // restore interrupts
206 enable_interrupts();
207
208 // if this is the local thread, we are still running on the old cluster
209 if(local) yield();
210
211 /* paranoid */ verify( !local || &cl == active_cluster() );
212 /* paranoid */ verify( !local || thrd->curr_cluster == active_cluster() );
213 /* paranoid */ verify( !local || thrd->curr_cluster == active_processor()->cltr );
214 /* paranoid */ verify( local || tmon->signal_stack.top );
215 /* paranoid */ verify( local || tmon->signal_stack.top->owner->waiting_thread == thrd );
216
217 return true;
218 }
219}
220
221//-----------------------------------------------------------------------------
222
223void set_seed( size_t seed ) {
224 PRNG_STATE_T & state = active_thread()->random_state;
225 PRNG_SET_SEED( state, seed );
226 __global_random_seed = seed;
227 __global_random_prime = seed;
228 __global_random_mask = true;
229} // set_seed
230
231size_t get_seed( void ) { return __global_random_seed; }
232
233size_t prng( void ) { // [0,UINT_MAX]
234 return PRNG_NAME( active_thread()->random_state );
235} // prng
236
237// Local Variables: //
238// mode: c //
239// tab-width: 4 //
240// End: //
Note: See TracBrowser for help on using the repository browser.