-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathydThreads.h
192 lines (154 loc) · 6.17 KB
/
ydThreads.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
/*
* File: yd_threads.h
* Authors: Danni Friedland & Yaniv Dudu
*
* Created on March 26, 2010, 11:32 PM
*/
#ifndef _YDTHREADS_H
#define _YDTHREADS_H
#include <ucontext.h>
#include "globalDefs.h"
#include "linkedList.h"
boolean worked;
typedef enum state_e {
RUN_THREAD,
ENQ_THREAD,
TERM_THREAD,
} state_t;
typedef enum thread_stats_e {
UNBORN_THREAD,
ALIVE_THREAD,
DEAD_THREAD
} thread_stats_t;
typedef enum run_e {
RR = 1,
PB = 2,
YD = 3,
} run_t;
typedef struct mctx_st {
int priority;
int initPriority;
ucontext_t uc;
tID id;
} mctx_t, *mctx_t_p;
typedef struct threads_stats {
int max_switch_wait;
int curr_switch_wait;
int max_jobs_wait;
int curr_jobs_wait;
tID id;
thread_stats_t thread_state;
} threads_stats_t, *threads_stats_t_p;
typedef struct th_container_s {
tID lastRunThreadID;
int indexOfLastThreadInContainer;
node_t_p container;
node_t_p stats;
} th_container_t, *th_container_t_p;
typedef int PB_priority;
typedef PB_priority* PB_priority_array;
/* save machine context */
#define MCTX_SAVE(mctx) (void)getcontext(&(mctx)->uc)
/* restore machine context */
#define MCTX_RESTORE(mctx) (void)setcontext(&(mctx)->uc);
#define THREAD_DATA(nodePointer) ((mctx_t_p)nodePointer->data)
#define THREAD_STATS(nodePointer) ((threads_stats_t_p)nodePointer->data)
/* save and restore context */
#define MCTX_SWAP(mctx_out, mctx_in) \
do { \
MCTX_SAVE(mctx_out); \
if (state == RUN_THREAD) { \
MCTX_RESTORE(mctx_in); \
} \
state = RUN_THREAD; \
} while (0)\
/* contain the next numeric id to give the next new thread */
static volatile int next_id = 0;
/* index for the round rubin scheduling method - temporary untill part 2-3 */
static volatile int scheduler_index = 0;
run_t* runType;
/* array that holds priorities for the threads */
PB_priority_array PB_array;
/* create machine context which can later be used to save & restore threads
* Returns:new thread ID or -1 if error
*/
tID mctx_create(mctx_t_p const mctx, void (*sf_addr)(), const void *sf_arg, void *sk_addr, const size_t sk_size);
op_status delete_thread(const tID threadID);
/* This function receives as arguments a pointer to the thread’s main function and a pointer to
* its argument list. The function should allocate the thread and prepare it but not run it. Once
* the thread’s fields are all set, the thread is inserted into the threads container.
* The function returns the newly created thread’s id or -1 in case of a failure to create a
* thread
* RETURNS: OP_CODE or new thread ID
*/
int create_thread(void (*sf_addr)(), void *sf_arg);
/* This function saves the current thread’s context and resumes the manager (restores the
* manager’s context). The argument pInfo is related to the requested priority upon yielding
* (see part three) and the statInfo is an argument which relates to the user’s application, and
* allows for statistics gathering (part three) */
void thread_yield(int pInfo, int statInfo);
/* Terminates the thread and transfer’s control to the manager. Note that this is the graceful
* means to terminate a thread – an alternate means to terminate all threads is by terminating
* the manager (we assume that the program completes after the manager exits) */
void thread_term();
/* Initializes the manager and the thread container data structure */
void thread_manager_init(void* arg, ucontext_t* ret_func, int arg_count);
/* This function starts the user space threads. All it does is simply let the manager run
* (restores the manager’s context) */
void threads_start();
void threads_start_with_ui(mctx_t_p ui_thread);
//The largest number of context switched performed between any two
//consecutive runs of the thread t is as accessed by the switches_wait(t) value of the statistics.
int switches_wait(tID threadID);
/* scheduler is responsable to choose one thread from the container list*/
mctx_t_p scheduler();
mctx_t_p scheduler_rr();
mctx_t_p scheduler_pb();
mctx_t_p scheduler_yd();
/* manager is the main function of the manager_thread
* all threads are comunicating with this the manager_thread thread
* to schedule next thread to run.
* the manager uses scheduler method to determine that order */
void manager();
/* Returns the ID of the currently running thread */
tID current_thread_id();
/* generate id by using an itarator */
tID id_itarator();
//searches the thread container for the specified thread id
mctx_t_p get_thread_byID(const tID threadID);
threads_stats_t_p get_thread_stats_byID(const IN tID threadID);
//prints thread info
void containerToString(const th_container_t_p const threadContainer);
void increase_switch_wait_for_all_except(tID threadID);
void increase_jobs_wait_for_all_except(tID threadID, int amount);
int maximal_switch_wait();
int jobs_wait(tID threadID);
int maximal_jobs_wait();
float avarage_jobs_wait();
int total_jobs_wait();
float avarage_switch_wait();
int total_switch_wait();
/* get the next thread by round robin algorithm.
* the procidure will return the next thread by the given offset */
mctx_t_p get_by_rr_with_offset_next_thread(tID lastRunThreadID);
op_status delete_statistics();
mctx_t_p search_for_highest_priority_thread(int offset);
/* global manager and current thread*/
/* return all threads that have a job with higher priority on the graph (parent)
* and NULL_TID in the end of the array */
tID* find_all_parents(tID lastRunThreadID);
/* return all threads that have a job with lower priority on the graph (child)
* and NULL_TID in the end of the array */
tID* find_all_children(tID lastRunThreadID);
/* get list of ids and retrun the thread with the highest priority */
mctx_t_p find_highest_priority_thread_by_id(tID* children);
/* return the thread index in the container list */
int thread_index_in_list(tID threadID);
/* global manager and current thread*/
mctx_t_p manager_thread, current_thread;
/* the thread container holds all thread in a list data type*/
th_container_t_p container;
/* Flag indicating that the current_thread has completed. */
static volatile state_t state;
void reset_iterator(int num);
#endif /* _YD_THREADS_H */