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
|
/* Quagga Pthreads support -- header
* Copyright (C) 2009 Chris Hall (GMCH), Highwayman
*
* This file is part of GNU Zebra.
*
* GNU Zebra is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published
* by the Free Software Foundation; either version 2, or (at your
* option) any later version.
*
* GNU Zebra is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with GNU Zebra; see the file COPYING. If not, write to the
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
#ifndef _ZEBRA_QPNEXUS_H
#define _ZEBRA_QPNEXUS_H
#include "misc.h"
#include <time.h>
#include <unistd.h>
#include <errno.h>
#include <signal.h>
#include "zassert.h"
#include "qpthreads.h"
#include "qtimers.h"
#include "mqueue.h"
#include "qpselect.h"
/*==============================================================================
* Quagga Nexus Interface -- qpn_xxxx
*
* Object to hold, a qpthread, a qps_selection, a qtimer_pile, a mqueue_queue
* together with the thread routine to poll and dispatch their respective
* action routines.
*
*/
enum
{
/* maximum time in seconds to sit in a pselect */
MAX_PSELECT_WAIT = 10,
/* signal for message queues */
SIG_INTERRUPT = SIGUSR2,
/* number of hooks per hook list */
qpn_hooks_max = 4,
} ;
/*==============================================================================
* Data Structures.
*/
typedef int qpn_hook_function(void) ; /* dispatch of tasks */
typedef int qpn_init_function(void) ; /* start/stop work */
typedef struct qpn_hook_list* qpn_hook_list ;
struct qpn_hook_list
{
void* hooks[qpn_hooks_max] ;
unsigned count ;
} ;
typedef struct qpn_nexus* qpn_nexus ;
struct qpn_nexus
{
/* set true to terminate the thread (eventually) */
bool terminate;
/* true if this is the main thread */
bool main_thread;
/* thread ID */
qpt_thread_t thread_id;
/* Signal mask for pselect */
sigset_t pselect_mask[1] ;
int pselect_signal ;
/* pselect handler */
qps_selection selection;
/* timer pile */
qtimer_pile pile;
/* message queue */
mqueue_queue queue;
mqueue_thread_signal mts;
/* qpthread routine, can override */
void* (*start)(void*);
/* in-thread initialise, can override. Called within the thread after all
* other initialisation just before thread loop
*
* These are typedef int qpn_init_function(void).
*
* These are executed in the order given.
*/
struct qpn_hook_list in_thread_init ;
/* in-thread finalise, can override. Called within thread just before
* thread dies. Nexus components all exist but thread loop is no longer
* executed
*
* These are typedef int qpn_init_function(void).
*
* These are executed in the reverse of the order given.
*/
struct qpn_hook_list in_thread_final ;
/* in-thread queue(s) of events or other work.
*
* The hook function(s) are called in the qpnexus loop, at the top of the
* loop. So in addition to the mqueue, I/O, timers and any background stuff,
* the thread may have other queue(s) of things to be done.
*
* These are typedef int qpn_hook_function(void).
*
* Hook function can process some queue(s) of things to be done. It does not
* have to empty its queues, but it MUST only return 0 if all queues are now
* empty.
*/
struct qpn_hook_list foreground ;
/* in-thread background queue(s) of events or other work.
*
* The hook functions are called at the bottom of the qpnexus loop, but only
* when there is absolutely nothing else to do.
*
* These are typedef int qpn_hook_function(void).
*
* The hook function should do some unit of background work (if there is any)
* and return. MUST return 0 iff there is no more work to do.
*/
struct qpn_hook_list background ;
};
/*------------------------------------------------------------------------------
* Each thread that has a qpnexus uses this piece of thread specific data in
* order to be able to find its own nexus.
*/
qpt_data_t qpn_self ; /* thread specific data */
Inline qpn_nexus
qpn_find_self(void)
{
return qpt_data_get_value(qpn_self) ;
} ;
/*==============================================================================
* Functions
*/
extern void qpn_init(void) ;
extern qpn_nexus qpn_init_new(qpn_nexus qpn, bool main_thread);
extern void qpn_add_hook_function(qpn_hook_list list, void* hook) ;
extern void qpn_exec(qpn_nexus qpn);
extern void qpn_terminate(qpn_nexus qpn);
extern qpn_nexus qpn_reset(qpn_nexus qpn, free_keep_b free_structure);
#endif /* _ZEBRA_QPNEXUS_H */
|