blob: efb6d6a3639ae4755fecd8b5606cfdd8d6e3e7ee [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * Callbacks for the FSM
3 *
4 * Copyright (C) 1996 Universidade de Lisboa
Joe Perches475be4d2012-02-19 19:52:38 -08005 *
Linus Torvalds1da177e2005-04-16 15:20:36 -07006 * Written by Pedro Roque Marques (roque@di.fc.ul.pt)
7 *
Joe Perches475be4d2012-02-19 19:52:38 -08008 * This software may be used and distributed according to the terms of
Linus Torvalds1da177e2005-04-16 15:20:36 -07009 * the GNU General Public License, incorporated herein by reference.
10 */
11
12/*
13 * Fix: 19981230 - Carlos Morgado <chbm@techie.com>
Joe Perches475be4d2012-02-19 19:52:38 -080014 * Port of Nelson Escravana's <nelson.escravana@usa.net> fix to CalledPN
Linus Torvalds1da177e2005-04-16 15:20:36 -070015 * NULL pointer dereference in cb_in_1 (originally fixed in 2.0)
16 */
17
Linus Torvalds1da177e2005-04-16 15:20:36 -070018#include <linux/string.h>
19#include <linux/kernel.h>
20
21#include <linux/types.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070022#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
35ushort last_ref_num = 1;
36
37/*
38 * send_conn_req
39 *
40 */
41
Joe Perches475be4d2012-02-19 19:52:38 -080042void cb_out_1(struct pcbit_dev *dev, struct pcbit_chan *chan,
43 struct callb_data *cbdata)
Linus Torvalds1da177e2005-04-16 15:20:36 -070044{
45 struct sk_buff *skb;
46 int len;
Joe Perches475be4d2012-02-19 19:52:38 -080047 ushort refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -070048
49
50#ifdef DEBUG
Joe Perches475be4d2012-02-19 19:52:38 -080051 printk(KERN_DEBUG "Called Party Number: %s\n",
52 cbdata->data.setup.CalledPN);
Linus Torvalds1da177e2005-04-16 15:20:36 -070053#endif
Joe Perches475be4d2012-02-19 19:52:38 -080054 /*
55 * hdr - kmalloc in capi_conn_req
56 * - kfree when msg has been sent
57 */
Linus Torvalds1da177e2005-04-16 15:20:36 -070058
Joe Perches475be4d2012-02-19 19:52:38 -080059 if ((len = capi_conn_req(cbdata->data.setup.CalledPN, &skb,
Linus Torvalds1da177e2005-04-16 15:20:36 -070060 chan->proto)) < 0)
Joe Perches475be4d2012-02-19 19:52:38 -080061 {
62 printk("capi_conn_req failed\n");
63 return;
64 }
Linus Torvalds1da177e2005-04-16 15:20:36 -070065
66
Joe Perches475be4d2012-02-19 19:52:38 -080067 refnum = last_ref_num++ & 0x7fffU;
Linus Torvalds1da177e2005-04-16 15:20:36 -070068
Joe Perches475be4d2012-02-19 19:52:38 -080069 chan->callref = 0;
70 chan->layer2link = 0;
71 chan->snum = 0;
72 chan->s_refnum = refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -070073
Joe Perches475be4d2012-02-19 19:52:38 -080074 pcbit_l2_write(dev, MSG_CONN_REQ, refnum, skb, len);
Linus Torvalds1da177e2005-04-16 15:20:36 -070075}
76
77/*
78 * rcv CONNECT
79 * will go into ACTIVE state
80 * send CONN_ACTIVE_RESP
Joe Perches475be4d2012-02-19 19:52:38 -080081 * send Select protocol request
Linus Torvalds1da177e2005-04-16 15:20:36 -070082 */
83
Joe Perches475be4d2012-02-19 19:52:38 -080084void cb_out_2(struct pcbit_dev *dev, struct pcbit_chan *chan,
85 struct callb_data *data)
Linus Torvalds1da177e2005-04-16 15:20:36 -070086{
Joe Perches475be4d2012-02-19 19:52:38 -080087 isdn_ctrl ictl;
88 struct sk_buff *skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -070089 int len;
Joe Perches475be4d2012-02-19 19:52:38 -080090 ushort refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -070091
Joe Perches475be4d2012-02-19 19:52:38 -080092 if ((len = capi_conn_active_resp(chan, &skb)) < 0)
93 {
94 printk("capi_conn_active_req failed\n");
95 return;
96 }
Linus Torvalds1da177e2005-04-16 15:20:36 -070097
Joe Perches475be4d2012-02-19 19:52:38 -080098 refnum = last_ref_num++ & 0x7fffU;
99 chan->s_refnum = refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700100
Joe Perches475be4d2012-02-19 19:52:38 -0800101 pcbit_l2_write(dev, MSG_CONN_ACTV_RESP, refnum, skb, len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700102
103
Joe Perches475be4d2012-02-19 19:52:38 -0800104 ictl.command = ISDN_STAT_DCONN;
105 ictl.driver = dev->id;
106 ictl.arg = chan->id;
107 dev->dev_if->statcallb(&ictl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700108
Joe Perches475be4d2012-02-19 19:52:38 -0800109 /* ACTIVE D-channel */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700110
Joe Perches475be4d2012-02-19 19:52:38 -0800111 /* Select protocol */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700112
Joe Perches475be4d2012-02-19 19:52:38 -0800113 if ((len = capi_select_proto_req(chan, &skb, 1 /*outgoing*/)) < 0) {
114 printk("capi_select_proto_req failed\n");
115 return;
116 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700117
Joe Perches475be4d2012-02-19 19:52:38 -0800118 refnum = last_ref_num++ & 0x7fffU;
119 chan->s_refnum = refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700120
Joe Perches475be4d2012-02-19 19:52:38 -0800121 pcbit_l2_write(dev, MSG_SELP_REQ, refnum, skb, len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700122}
123
124
125/*
Linus Torvalds1da177e2005-04-16 15:20:36 -0700126 * Incoming call received
127 * inform user
128 */
129
Joe Perches475be4d2012-02-19 19:52:38 -0800130void cb_in_1(struct pcbit_dev *dev, struct pcbit_chan *chan,
131 struct callb_data *cbdata)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700132{
Joe Perches475be4d2012-02-19 19:52:38 -0800133 isdn_ctrl ictl;
134 unsigned short refnum;
135 struct sk_buff *skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700136 int len;
137
138
Joe Perches475be4d2012-02-19 19:52:38 -0800139 ictl.command = ISDN_STAT_ICALL;
140 ictl.driver = dev->id;
141 ictl.arg = chan->id;
142
143 /*
144 * ictl.num >= strlen() + strlen() + 5
145 */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700146
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
Joe Perches475be4d2012-02-19 19:52:38 -0800170 dev->dev_if->statcallb(&ictl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700171
Joe Perches475be4d2012-02-19 19:52:38 -0800172
173 if ((len = capi_conn_resp(chan, &skb)) < 0) {
174 printk(KERN_DEBUG "capi_conn_resp failed\n");
175 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700176 }
177
Joe Perches475be4d2012-02-19 19:52:38 -0800178 refnum = last_ref_num++ & 0x7fffU;
179 chan->s_refnum = refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700180
Joe Perches475be4d2012-02-19 19:52:38 -0800181 pcbit_l2_write(dev, MSG_CONN_RESP, refnum, skb, len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700182}
183
184/*
185 * user has replied
186 * open the channel
187 * send CONNECT message CONNECT_ACTIVE_REQ in CAPI
188 */
189
Joe Perches475be4d2012-02-19 19:52:38 -0800190void cb_in_2(struct pcbit_dev *dev, struct pcbit_chan *chan,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700191 struct callb_data *data)
192{
Joe Perches475be4d2012-02-19 19:52:38 -0800193 unsigned short refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700194 struct sk_buff *skb;
Joe Perches475be4d2012-02-19 19:52:38 -0800195 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 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700201
202
Joe Perches475be4d2012-02-19 19:52:38 -0800203 refnum = last_ref_num++ & 0x7fffU;
204 chan->s_refnum = refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700205
206 printk(KERN_DEBUG "sending MSG_CONN_ACTV_REQ\n");
Joe Perches475be4d2012-02-19 19:52:38 -0800207 pcbit_l2_write(dev, MSG_CONN_ACTV_REQ, refnum, skb, len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700208}
209
210/*
211 * CONN_ACK arrived
212 * start b-proto selection
213 *
214 */
215
Joe Perches475be4d2012-02-19 19:52:38 -0800216void cb_in_3(struct pcbit_dev *dev, struct pcbit_chan *chan,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700217 struct callb_data *data)
218{
Joe Perches475be4d2012-02-19 19:52:38 -0800219 unsigned short refnum;
220 struct sk_buff *skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700221 int len;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700222
Joe Perches475be4d2012-02-19 19:52:38 -0800223 if ((len = capi_select_proto_req(chan, &skb, 0 /*incoming*/)) < 0)
224 {
225 printk("capi_select_proto_req failed\n");
226 return;
227 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700228
Joe Perches475be4d2012-02-19 19:52:38 -0800229 refnum = last_ref_num++ & 0x7fffU;
230 chan->s_refnum = refnum;
231
232 pcbit_l2_write(dev, MSG_SELP_REQ, refnum, skb, len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700233
234}
235
236
237/*
238 * Received disconnect ind on active state
239 * send disconnect resp
240 * send msg to user
241 */
Joe Perches475be4d2012-02-19 19:52:38 -0800242void cb_disc_1(struct pcbit_dev *dev, struct pcbit_chan *chan,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700243 struct callb_data *data)
244{
Joe Perches475be4d2012-02-19 19:52:38 -0800245 struct sk_buff *skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700246 int len;
Joe Perches475be4d2012-02-19 19:52:38 -0800247 ushort refnum;
248 isdn_ctrl ictl;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700249
Joe Perches475be4d2012-02-19 19:52:38 -0800250 if ((len = capi_disc_resp(chan, &skb)) < 0) {
251 printk("capi_disc_resp failed\n");
252 return;
253 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700254
Joe Perches475be4d2012-02-19 19:52:38 -0800255 refnum = last_ref_num++ & 0x7fffU;
256 chan->s_refnum = refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700257
Joe Perches475be4d2012-02-19 19:52:38 -0800258 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);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700264}
265
Joe Perches475be4d2012-02-19 19:52:38 -0800266
Linus Torvalds1da177e2005-04-16 15:20:36 -0700267/*
268 * User HANGUP on active/call proceeding state
269 * send disc.req
270 */
Joe Perches475be4d2012-02-19 19:52:38 -0800271void cb_disc_2(struct pcbit_dev *dev, struct pcbit_chan *chan,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700272 struct callb_data *data)
273{
Joe Perches475be4d2012-02-19 19:52:38 -0800274 struct sk_buff *skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700275 int len;
Joe Perches475be4d2012-02-19 19:52:38 -0800276 ushort refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700277
Joe Perches475be4d2012-02-19 19:52:38 -0800278 if ((len = capi_disc_req(chan->callref, &skb, CAUSE_NORMAL)) < 0)
279 {
280 printk("capi_disc_req failed\n");
281 return;
282 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700283
Joe Perches475be4d2012-02-19 19:52:38 -0800284 refnum = last_ref_num++ & 0x7fffU;
285 chan->s_refnum = refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700286
Joe Perches475be4d2012-02-19 19:52:38 -0800287 pcbit_l2_write(dev, MSG_DISC_REQ, refnum, skb, len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700288}
289
290/*
291 * Disc confirm received send BHUP
292 * Problem: when the HL driver sends the disc req itself
293 * LL receives BHUP
294 */
Joe Perches475be4d2012-02-19 19:52:38 -0800295void cb_disc_3(struct pcbit_dev *dev, struct pcbit_chan *chan,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700296 struct callb_data *data)
297{
Joe Perches475be4d2012-02-19 19:52:38 -0800298 isdn_ctrl ictl;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700299
Joe Perches475be4d2012-02-19 19:52:38 -0800300 ictl.command = ISDN_STAT_BHUP;
301 ictl.driver = dev->id;
302 ictl.arg = chan->id;
303 dev->dev_if->statcallb(&ictl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700304}
305
Joe Perches475be4d2012-02-19 19:52:38 -0800306void cb_notdone(struct pcbit_dev *dev, struct pcbit_chan *chan,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700307 struct callb_data *data)
308{
309}
310
311/*
312 * send activate b-chan protocol
313 */
Joe Perches475be4d2012-02-19 19:52:38 -0800314void cb_selp_1(struct pcbit_dev *dev, struct pcbit_chan *chan,
315 struct callb_data *data)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700316{
Joe Perches475be4d2012-02-19 19:52:38 -0800317 struct sk_buff *skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700318 int len;
Joe Perches475be4d2012-02-19 19:52:38 -0800319 ushort refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700320
Joe Perches475be4d2012-02-19 19:52:38 -0800321 if ((len = capi_activate_transp_req(chan, &skb)) < 0)
322 {
323 printk("capi_conn_activate_transp_req failed\n");
324 return;
325 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700326
Joe Perches475be4d2012-02-19 19:52:38 -0800327 refnum = last_ref_num++ & 0x7fffU;
328 chan->s_refnum = refnum;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700329
Joe Perches475be4d2012-02-19 19:52:38 -0800330 pcbit_l2_write(dev, MSG_ACT_TRANSP_REQ, refnum, skb, len);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700331}
332
333/*
334 * Inform User that the B-channel is available
335 */
Joe Perches475be4d2012-02-19 19:52:38 -0800336void cb_open(struct pcbit_dev *dev, struct pcbit_chan *chan,
337 struct callb_data *data)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700338{
Joe Perches475be4d2012-02-19 19:52:38 -0800339 isdn_ctrl ictl;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700340
Joe Perches475be4d2012-02-19 19:52:38 -0800341 ictl.command = ISDN_STAT_BCONN;
342 ictl.driver = dev->id;
343 ictl.arg = chan->id;
344 dev->dev_if->statcallb(&ictl);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700345}