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
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
|
/*
* Linux WiMax
* Internal API and declarations
*
*
* Copyright (C) 2007-2008 Intel Corporation. All rights reserved.
* Inaky Perez-Gonzalez <inaky.perez-gonzalez@intel.com>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Intel Corporation nor the names of its
* contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*
*
*/
#ifndef __lib_internal_h__
#define __lib_internal_h__
#include <wimaxll.h>
struct nl_msg;
struct nlmsgerr;
struct sockaddr_nl;
enum {
#define __WIMAXLL_IFNAME_LEN 32
/**
* WIMAXLL_IFNAME_LEN - Maximum size of a wimax interface
* name.
*/
WIMAXLL_IFNAME_LEN = __WIMAXLL_IFNAME_LEN,
};
/**
* General structure for storing callback context
*
* \ingroup callbacks
*
* Callbacks set by the user receive a user-set pointer to a context
* structure. The user can wrap this struct in a bigger context struct
* and use wimaxll_container_of() during the callback to obtain its
* pointer.
*
* Usage:
*
* \code
* ...
* struct wimaxll_handle *wmx;
* ...
* struct my_context {
* struct wimaxll_cb_ctx ctx;
* <my data>
* } my_ctx = {
* .ctx = WIMAXLL_CB_CTX_INIT(wmx),
* <my data initialization>
* };
* ...
* wimaxll_set_cb_SOMECALLBACK(wmx, my_callback, &my_ctx.ctx);
* ...
* result = wimaxll_pipe_read(wmx);
* ...
*
* // When my_callback() is called
* my_callback(wmx, ctx, ...)
* {
* struct my_context *my_ctx = wimaxll_container_of(
* ctx, struct my_callback, ctx);
* ...
* // do stuff with my_ctx
* }
* \endcode
*
* \param wmx WiMAX handle this context refers to (for usage by the
* callback).
* \param result Result of the handling of the message. For usage by
* the callback. Should not be set to -EINPROGRESS, as this will
* be interpreted by the message handler as no processing was done
* on the message.
*
* \internal
*
* \param msg_done This is used internally to mark when the acks (or
* errors) for a message have been received and the message
* receiving loop can be considered done.
*/
struct wimaxll_cb_ctx {
struct wimaxll_handle *wmx;
ssize_t result;
unsigned msg_done:1; /* internal */
};
/**
* Initialize a definition of struct wimaxll_cb_ctx
*
* \param _wmx pointer to the WiMAX device handle this will be
* associated to
*
* Use as:
*
* \code
* struct wimaxll_handle *wmx;
* ...
* struct wimaxll_cb_ctx my_context = WIMAXLL_CB_CTX_INIT(wmx);
* \endcode
*
* \ingroup callbacks
*/
#define WIMAXLL_CB_CTX_INIT(_wmx) { \
.wmx = (_wmx), \
.result = -EINPROGRESS, \
}
static inline // ugly workaround for doxygen
/**
* Initialize a struct wimaxll_cb_ctx
*
* \param ctx Pointer to the struct wimaxll_cb_ctx.
* \param wmx pointer to the WiMAX device handle this will be
* associated to
*
* Use as:
*
* \code
* struct wimaxll_handle *wmx;
* ...
* struct wimaxll_cb_ctx my_context;
* ...
* wimaxll_cb_ctx(&my_context, wmx);
* \endcode
*
* \ingroup callbacks
* \fn static void wimaxll_cb_ctx_init(struct wimaxll_cb_ctx *ctx, struct wimaxll_handle *wmx)
*/
void wimaxll_cb_ctx_init(struct wimaxll_cb_ctx *ctx, struct wimaxll_handle *wmx)
{
ctx->wmx = wmx;
ctx->result = -EINPROGRESS;
}
static inline // ugly workaround for doxygen
/**
* Set the result value in a callback context
*
* \param ctx Context where to set -- if NULL, no action will be taken
* \param val value to set for \a result
*
* \ingroup callbacks
* \fn static void wimaxll_cb_maybe_set_result(struct wimaxll_cb_ctx *ctx, int val)
*/
void wimaxll_cb_maybe_set_result(struct wimaxll_cb_ctx *ctx, int val)
{
if (ctx != NULL && ctx->result == -EINPROGRESS)
ctx->result = val;
}
/**
* A WiMax control pipe handle
*
* This type is opaque to the user
*
* \internal
*
* In order to simplify multithread support, we use to different \a
* libnl handles, one for sending to the kernel, one for receiving
* from the kernel (multicast group). This allows us to parallelize \c
* wimaxll_msg_write() and \c wimaxll_msg_read() at the same time in a
* multithreaded environment, for example.
*
* \param ifidx Interface Index (of the network interface); if 0, the
* interface name will be \c "any" and this means that this handle
* works for \e any WiMAX interface.
* \param gnl_family_id Generic Netlink Family ID assigned to the
* device; we maintain it here (for each interface) because we
* want to discover it every time we open. This solves the case of
* the WiMAX modules being reloaded (and the ID changing) while
* this library is running; this way it takes only a new open when
* the new device is discovered.
* \param mcg_id Id of the 'msg' multicast group
* \param name name of the wimax interface
* \param priv Private pointer set with wimaxll_priv_set() or other
* accessors. Use wimaxll_priv_get() to access it.
* \param nlh_tx handle for writing to the kernel.
* Internal note: You \b have \b to set the handlers for
* %NL_CB_VALID and nl_cb_err() callbacks, as each callsite will
* do it to suit their needs. See wimaxll_rfkill() for an
* example. Any other callback you are supposed to restore to what
* it was before.
* \param nlh_rx handle for reading from the kernel.
* \param nl_rx_cb Callbacks for the nlh_rx handle
*
* FIXME: add doc on callbacks
*/
struct wimaxll_handle {
unsigned ifidx;
int gnl_family_id, mcg_id;
char name[__WIMAXLL_IFNAME_LEN];
void *priv;
struct nl_handle *nlh_tx;
struct nl_handle *nlh_rx;
wimaxll_msg_to_user_cb_f msg_to_user_cb;
void *msg_to_user_priv;
wimaxll_state_change_cb_f state_change_cb;
void *state_change_priv;
};
/* Utilities */
int wimaxll_wait_for_ack(struct wimaxll_handle *);
int wimaxll_gnl_handle_msg_to_user(struct wimaxll_handle *, struct nl_msg *);
int wimaxll_gnl_handle_state_change(struct wimaxll_handle *, struct nl_msg *);
int wimaxll_gnl_error_cb(struct sockaddr_nl *, struct nlmsgerr *, void *);
int wimaxll_gnl_ack_cb(struct nl_msg *msg, void *_mch);
/*
* wimaxll_family_id - Return the associated Generic Netlink family ID
*
* @wmx: WiMax interface for which to provide the ID.
*/
static inline
int wimaxll_family_id(struct wimaxll_handle *wmx)
{
return wmx->gnl_family_id;
}
void wimaxll_msg(struct wimaxll_handle *, const char *fmt, ...)
__attribute__ ((format(printf, 2, 3)));
/* Generic Netlink utilities */
int nl_get_multicast_groups(struct nl_handle *, const char *,
void (*cb)(void *, const char *, int),
void *);
int genl_ctrl_get_version(struct nl_handle *, const char *);
#endif /* #ifndef __lib_internal_h__ */
|