Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
awilliam
GitHub Repository: awilliam/linux-vfio
Path: blob/master/drivers/isdn/pcbit/callbacks.c
15111 views
1
/*
2
* Callbacks for the FSM
3
*
4
* Copyright (C) 1996 Universidade de Lisboa
5
*
6
* Written by Pedro Roque Marques ([email protected])
7
*
8
* This software may be used and distributed according to the terms of
9
* the GNU General Public License, incorporated herein by reference.
10
*/
11
12
/*
13
* Fix: 19981230 - Carlos Morgado <[email protected]>
14
* Port of Nelson Escravana's <[email protected]> fix to CalledPN
15
* NULL pointer dereference in cb_in_1 (originally fixed in 2.0)
16
*/
17
18
#include <linux/string.h>
19
#include <linux/kernel.h>
20
21
#include <linux/types.h>
22
#include <linux/mm.h>
23
#include <linux/skbuff.h>
24
25
#include <asm/io.h>
26
27
#include <linux/isdnif.h>
28
29
#include "pcbit.h"
30
#include "layer2.h"
31
#include "edss1.h"
32
#include "callbacks.h"
33
#include "capi.h"
34
35
ushort last_ref_num = 1;
36
37
/*
38
* send_conn_req
39
*
40
*/
41
42
void cb_out_1(struct pcbit_dev * dev, struct pcbit_chan* chan,
43
struct callb_data *cbdata)
44
{
45
struct sk_buff *skb;
46
int len;
47
ushort refnum;
48
49
50
#ifdef DEBUG
51
printk(KERN_DEBUG "Called Party Number: %s\n",
52
cbdata->data.setup.CalledPN);
53
#endif
54
/*
55
* hdr - kmalloc in capi_conn_req
56
* - kfree when msg has been sent
57
*/
58
59
if ((len = capi_conn_req(cbdata->data.setup.CalledPN, &skb,
60
chan->proto)) < 0)
61
{
62
printk("capi_conn_req failed\n");
63
return;
64
}
65
66
67
refnum = last_ref_num++ & 0x7fffU;
68
69
chan->callref = 0;
70
chan->layer2link = 0;
71
chan->snum = 0;
72
chan->s_refnum = refnum;
73
74
pcbit_l2_write(dev, MSG_CONN_REQ, refnum, skb, len);
75
}
76
77
/*
78
* rcv CONNECT
79
* will go into ACTIVE state
80
* send CONN_ACTIVE_RESP
81
* send Select protocol request
82
*/
83
84
void cb_out_2(struct pcbit_dev * dev, struct pcbit_chan* chan,
85
struct callb_data *data)
86
{
87
isdn_ctrl ictl;
88
struct sk_buff *skb;
89
int len;
90
ushort refnum;
91
92
if ((len=capi_conn_active_resp(chan, &skb)) < 0)
93
{
94
printk("capi_conn_active_req failed\n");
95
return;
96
}
97
98
refnum = last_ref_num++ & 0x7fffU;
99
chan->s_refnum = refnum;
100
101
pcbit_l2_write(dev, MSG_CONN_ACTV_RESP, refnum, skb, len);
102
103
104
ictl.command = ISDN_STAT_DCONN;
105
ictl.driver=dev->id;
106
ictl.arg=chan->id;
107
dev->dev_if->statcallb(&ictl);
108
109
/* ACTIVE D-channel */
110
111
/* Select protocol */
112
113
if ((len=capi_select_proto_req(chan, &skb, 1 /*outgoing*/)) < 0) {
114
printk("capi_select_proto_req failed\n");
115
return;
116
}
117
118
refnum = last_ref_num++ & 0x7fffU;
119
chan->s_refnum = refnum;
120
121
pcbit_l2_write(dev, MSG_SELP_REQ, refnum, skb, len);
122
}
123
124
125
/*
126
* Incoming call received
127
* inform user
128
*/
129
130
void cb_in_1(struct pcbit_dev * dev, struct pcbit_chan* chan,
131
struct callb_data *cbdata)
132
{
133
isdn_ctrl ictl;
134
unsigned short refnum;
135
struct sk_buff *skb;
136
int len;
137
138
139
ictl.command = ISDN_STAT_ICALL;
140
ictl.driver=dev->id;
141
ictl.arg=chan->id;
142
143
/*
144
* ictl.num >= strlen() + strlen() + 5
145
*/
146
147
if (cbdata->data.setup.CallingPN == NULL) {
148
printk(KERN_DEBUG "NULL CallingPN to phone; using 0\n");
149
strcpy(ictl.parm.setup.phone, "0");
150
}
151
else {
152
strcpy(ictl.parm.setup.phone, cbdata->data.setup.CallingPN);
153
}
154
if (cbdata->data.setup.CalledPN == NULL) {
155
printk(KERN_DEBUG "NULL CalledPN to eazmsn; using 0\n");
156
strcpy(ictl.parm.setup.eazmsn, "0");
157
}
158
else {
159
strcpy(ictl.parm.setup.eazmsn, cbdata->data.setup.CalledPN);
160
}
161
ictl.parm.setup.si1 = 7;
162
ictl.parm.setup.si2 = 0;
163
ictl.parm.setup.plan = 0;
164
ictl.parm.setup.screen = 0;
165
166
#ifdef DEBUG
167
printk(KERN_DEBUG "statstr: %s\n", ictl.num);
168
#endif
169
170
dev->dev_if->statcallb(&ictl);
171
172
173
if ((len=capi_conn_resp(chan, &skb)) < 0) {
174
printk(KERN_DEBUG "capi_conn_resp failed\n");
175
return;
176
}
177
178
refnum = last_ref_num++ & 0x7fffU;
179
chan->s_refnum = refnum;
180
181
pcbit_l2_write(dev, MSG_CONN_RESP, refnum, skb, len);
182
}
183
184
/*
185
* user has replied
186
* open the channel
187
* send CONNECT message CONNECT_ACTIVE_REQ in CAPI
188
*/
189
190
void cb_in_2(struct pcbit_dev * dev, struct pcbit_chan* chan,
191
struct callb_data *data)
192
{
193
unsigned short refnum;
194
struct sk_buff *skb;
195
int len;
196
197
if ((len = capi_conn_active_req(chan, &skb)) < 0) {
198
printk(KERN_DEBUG "capi_conn_active_req failed\n");
199
return;
200
}
201
202
203
refnum = last_ref_num++ & 0x7fffU;
204
chan->s_refnum = refnum;
205
206
printk(KERN_DEBUG "sending MSG_CONN_ACTV_REQ\n");
207
pcbit_l2_write(dev, MSG_CONN_ACTV_REQ, refnum, skb, len);
208
}
209
210
/*
211
* CONN_ACK arrived
212
* start b-proto selection
213
*
214
*/
215
216
void cb_in_3(struct pcbit_dev * dev, struct pcbit_chan* chan,
217
struct callb_data *data)
218
{
219
unsigned short refnum;
220
struct sk_buff *skb;
221
int len;
222
223
if ((len = capi_select_proto_req(chan, &skb, 0 /*incoming*/)) < 0)
224
{
225
printk("capi_select_proto_req failed\n");
226
return;
227
}
228
229
refnum = last_ref_num++ & 0x7fffU;
230
chan->s_refnum = refnum;
231
232
pcbit_l2_write(dev, MSG_SELP_REQ, refnum, skb, len);
233
234
}
235
236
237
/*
238
* Received disconnect ind on active state
239
* send disconnect resp
240
* send msg to user
241
*/
242
void cb_disc_1(struct pcbit_dev * dev, struct pcbit_chan* chan,
243
struct callb_data *data)
244
{
245
struct sk_buff *skb;
246
int len;
247
ushort refnum;
248
isdn_ctrl ictl;
249
250
if ((len = capi_disc_resp(chan, &skb)) < 0) {
251
printk("capi_disc_resp failed\n");
252
return;
253
}
254
255
refnum = last_ref_num++ & 0x7fffU;
256
chan->s_refnum = refnum;
257
258
pcbit_l2_write(dev, MSG_DISC_RESP, refnum, skb, len);
259
260
ictl.command = ISDN_STAT_BHUP;
261
ictl.driver=dev->id;
262
ictl.arg=chan->id;
263
dev->dev_if->statcallb(&ictl);
264
}
265
266
267
/*
268
* User HANGUP on active/call proceeding state
269
* send disc.req
270
*/
271
void cb_disc_2(struct pcbit_dev * dev, struct pcbit_chan* chan,
272
struct callb_data *data)
273
{
274
struct sk_buff *skb;
275
int len;
276
ushort refnum;
277
278
if ((len = capi_disc_req(chan->callref, &skb, CAUSE_NORMAL)) < 0)
279
{
280
printk("capi_disc_req failed\n");
281
return;
282
}
283
284
refnum = last_ref_num++ & 0x7fffU;
285
chan->s_refnum = refnum;
286
287
pcbit_l2_write(dev, MSG_DISC_REQ, refnum, skb, len);
288
}
289
290
/*
291
* Disc confirm received send BHUP
292
* Problem: when the HL driver sends the disc req itself
293
* LL receives BHUP
294
*/
295
void cb_disc_3(struct pcbit_dev * dev, struct pcbit_chan* chan,
296
struct callb_data *data)
297
{
298
isdn_ctrl ictl;
299
300
ictl.command = ISDN_STAT_BHUP;
301
ictl.driver=dev->id;
302
ictl.arg=chan->id;
303
dev->dev_if->statcallb(&ictl);
304
}
305
306
void cb_notdone(struct pcbit_dev * dev, struct pcbit_chan* chan,
307
struct callb_data *data)
308
{
309
}
310
311
/*
312
* send activate b-chan protocol
313
*/
314
void cb_selp_1(struct pcbit_dev * dev, struct pcbit_chan* chan,
315
struct callb_data *data)
316
{
317
struct sk_buff *skb;
318
int len;
319
ushort refnum;
320
321
if ((len = capi_activate_transp_req(chan, &skb)) < 0)
322
{
323
printk("capi_conn_activate_transp_req failed\n");
324
return;
325
}
326
327
refnum = last_ref_num++ & 0x7fffU;
328
chan->s_refnum = refnum;
329
330
pcbit_l2_write(dev, MSG_ACT_TRANSP_REQ, refnum, skb, len);
331
}
332
333
/*
334
* Inform User that the B-channel is available
335
*/
336
void cb_open(struct pcbit_dev * dev, struct pcbit_chan* chan,
337
struct callb_data *data)
338
{
339
isdn_ctrl ictl;
340
341
ictl.command = ISDN_STAT_BCONN;
342
ictl.driver=dev->id;
343
ictl.arg=chan->id;
344
dev->dev_if->statcallb(&ictl);
345
}
346
347
348
349
350