blob: 31b42216b4f5afa8e7281cb623e02c41a22eef91 [file] [log] [blame]
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +01001/*
Holger Hans Peter Freyther1b6901e2011-01-17 16:13:28 +01002 * (C) 2010-2011 by Holger Hans Peter Freyther <zecke@selfish.org>
3 * (C) 2010-2011 by On-Waves
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +01004 * All Rights Reserved
5 *
Holger Hans Peter Freytherde56c222011-01-16 17:45:14 +01006 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU Affero General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +01009 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Holger Hans Peter Freytherde56c222011-01-16 17:45:14 +010014 * GNU Affero General Public License for more details.
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +010015 *
Holger Hans Peter Freytherde56c222011-01-16 17:45:14 +010016 * You should have received a copy of the GNU Affero General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +010018 *
19 */
20
21#include <isup_types.h>
22#include <cellmgr_debug.h>
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +010023#include <mtp_data.h>
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +010024
Harald Welteff397ed2011-05-08 10:29:23 +020025#include <osmocom/core/msgb.h>
26#include <osmocom/gsm/tlv.h>
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +010027
Holger Hans Peter Freyther8f6eb5e2011-01-19 09:27:41 +010028static struct msgb *isup_status_alloc(int cic, int msg_type, uint8_t *extra, int range, int val)
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +010029{
30 struct isup_msg_hdr *hdr;
31 struct msgb *msg;
32 int bits, len;
Holger Hans Peter Freyther8f6eb5e2011-01-19 09:27:41 +010033 uint8_t *data;
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +010034
Holger Hans Peter Freytherecd83552011-01-19 09:19:17 +010035 msg = msgb_alloc_headroom(4096, 128, "ISUP Simple MSG");
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +010036 if (!msg) {
Holger Hans Peter Freytherecd83552011-01-19 09:19:17 +010037 LOGP(DISUP, LOGL_ERROR, "Allocation of status message failed.\n");
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +010038 return NULL;
39 }
40
41 msg->l2h = msgb_put(msg, sizeof(*hdr));
42
43 /* write the ISUP header */
44 hdr = (struct isup_msg_hdr *) msg->l2h;
45 hdr->cic = cic;
Holger Hans Peter Freytherecd83552011-01-19 09:19:17 +010046 hdr->msg_type = msg_type;
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +010047
Holger Hans Peter Freyther8f6eb5e2011-01-19 09:27:41 +010048 if (extra)
49 msgb_v_put(msg, *extra);
50
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +010051 /*
52 * place the pointers here.
53 * 1.) place the variable start after us
54 * 2.) place the length
55 */
56 msgb_v_put(msg, 1);
57
58 bits = range + 1;
59 len = (bits / 8) + 1;
60 msgb_v_put(msg, len + 1);
61 msgb_v_put(msg, range);
62
Holger Hans Peter Freyther8f6eb5e2011-01-19 09:27:41 +010063 data = msgb_put(msg, len);
64
65 /* set the status bits to val... FIXME this sets the extra bits too */
66 memset(data, val, len);
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +010067
68 return msg;
69}
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +010070
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +010071static struct msgb *isup_simple_alloc(int cic, int msg_type)
Holger Hans Peter Freyther6e11bee2011-01-18 23:34:19 +010072{
73 struct isup_msg_hdr *hdr;
74 struct msgb *msg;
75
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +010076 msg = msgb_alloc_headroom(4096, 128, "ISUP Simple MSG");
Holger Hans Peter Freyther6e11bee2011-01-18 23:34:19 +010077 if (!msg) {
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +010078 LOGP(DISUP, LOGL_ERROR, "Allocation of Simple message failed.\n");
Holger Hans Peter Freyther6e11bee2011-01-18 23:34:19 +010079 return NULL;
80 }
81
82 msg->l2h = msgb_put(msg, sizeof(*hdr));
83
84 /* write the ISUP header */
85 hdr = (struct isup_msg_hdr *) msg->l2h;
86 hdr->cic = cic;
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +010087 hdr->msg_type = msg_type;
Holger Hans Peter Freyther6e11bee2011-01-18 23:34:19 +010088
89 msgb_v_put(msg, 0);
90 return msg;
91}
92
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +010093/* this message contains the range */
Holger Hans Peter Freytherecd83552011-01-19 09:19:17 +010094int isup_parse_status(const uint8_t *data, uint8_t in_length)
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +010095{
96 uint8_t ptr;
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +010097
Holger Hans Peter Freythera4661fb2011-01-19 09:43:01 +010098 if (in_length < 3) {
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +010099 LOGP(DISUP, LOGL_ERROR, "This needs three bytes.\n");
100 return -1;
101 }
102
103 ptr = data[0];
104 if (1 + ptr > in_length) {
105 LOGP(DISUP, LOGL_ERROR, "Pointing outside the packet.\n");
106 return -1;
107 }
108
Holger Hans Peter Freytherc21cfaa2010-12-10 03:53:28 +0100109 if (1 + ptr + 1 > in_length) {
110 LOGP(DISUP, LOGL_ERROR, "No space for the data.\n");
111 return -1;
112 }
113
114 return data[0 + ptr + 1];
115}
116
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100117
118/* Handle incoming ISUP data */
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100119static int handle_circuit_reset_grs(struct mtp_link_set *set, int sls, int cic,
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100120 const uint8_t *data, int size)
121{
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +0100122 struct msgb *resp;
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100123 int range;
124
Holger Hans Peter Freytherecd83552011-01-19 09:19:17 +0100125 range = isup_parse_status(data, size);
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100126 if (range < 0)
127 return -1;
128
Holger Hans Peter Freyther8f6eb5e2011-01-19 09:27:41 +0100129 resp = isup_status_alloc(cic, ISUP_MSG_GRA, NULL, range, 0);
130 if (!resp)
131 return -1;
132
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100133 mtp_link_set_submit_isup_data(set, sls, resp->l2h, msgb_l2len(resp));
Holger Hans Peter Freyther8f6eb5e2011-01-19 09:27:41 +0100134 msgb_free(resp);
135 return 0;
136}
137
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100138static int handle_circuit_reset_cgb(struct mtp_link_set *set, int sls, int cic,
Holger Hans Peter Freyther8f6eb5e2011-01-19 09:27:41 +0100139 const uint8_t *data, int size)
140{
141 struct msgb *resp;
142 int range;
143 uint8_t val;
144
145 if (size < 1)
146 return -1;
147
148 range = isup_parse_status(data + 1, size - 1);
149 if (range < 0)
150 return -1;
151
152 val = 0;
153 resp = isup_status_alloc(cic, ISUP_MSG_CGBA, &val, range, 0xff);
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +0100154 if (!resp)
155 return -1;
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100156
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100157 mtp_link_set_submit_isup_data(set, sls, resp->l2h, msgb_l2len(resp));
Holger Hans Peter Freyther3aad7762010-12-10 12:18:57 +0100158 msgb_free(resp);
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100159 return 0;
160}
161
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100162static int send_cgu(struct mtp_link_set *set, int sls, int cic, int range)
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +0100163{
164 struct msgb *resp;
165 uint8_t val;
166
167 val = 0;
168 resp = isup_status_alloc(cic, ISUP_MSG_CGU, &val, range, 0);
169 if (!resp)
170 return -1;
171
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100172 mtp_link_set_submit_isup_data(set, sls, resp->l2h, msgb_l2len(resp));
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +0100173 msgb_free(resp);
174 return 0;
175}
176
Holger Hans Peter Freythere0dc3e52011-03-05 09:50:47 +0100177static int handle_cgu(struct mtp_link_set *set, int sls, int cic,
178 uint8_t *data, uint16_t size)
179{
180 uint8_t *out;
181 struct isup_msg_hdr *hdr;
182 struct msgb *resp;
183
184 resp = msgb_alloc_headroom(4096, 128, "ISUP CGUA MSG");
185 if (!resp) {
186 LOGP(DISUP, LOGL_ERROR, "Allocation of CGUA message failed.\n");
187 return -1;
188 }
189
190 resp->l2h = msgb_put(resp, sizeof(*hdr));
191
192 /* write the ISUP header */
193 hdr = (struct isup_msg_hdr *) resp->l2h;
194 hdr->cic = cic;
195 hdr->msg_type = ISUP_MSG_CGUA;
196
197 out = msgb_put(resp, size);
198 memcpy(out, data, size);
199
200 mtp_link_set_submit_isup_data(set, sls, resp->l2h, msgb_l2len(resp));
201 msgb_free(resp);
202 return 0;
203}
204
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +0100205static int handle_simple_resp(struct mtp_link_set *set, int sls, int cic, int msg_type)
Holger Hans Peter Freyther6e11bee2011-01-18 23:34:19 +0100206{
207 struct msgb *resp;
208
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +0100209 resp = isup_simple_alloc(cic, msg_type);
Holger Hans Peter Freyther6e11bee2011-01-18 23:34:19 +0100210 if (!resp)
211 return -1;
212 mtp_link_set_submit_isup_data(set, sls, resp->l2h, msgb_l2len(resp));
213 msgb_free(resp);
214 return 0;
215}
216
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100217int mtp_link_set_isup(struct mtp_link_set *set, struct msgb *msg, int sls)
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100218{
219 int rc = -1;
220 int payload_size;
221 struct isup_msg_hdr *hdr;
222
223 if (msgb_l3len(msg) < sizeof(*hdr)) {
224 LOGP(DISUP, LOGL_ERROR, "ISUP header is too short.\n");
225 return -1;
226 }
227
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100228 if (set->pass_all_isup) {
229 mtp_link_set_forward_isup(set, msg, sls);
Holger Hans Peter Freytherefbd8c22011-01-17 20:21:45 +0100230 return 0;
231 }
232
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100233 hdr = (struct isup_msg_hdr *) msg->l3h;
234 payload_size = msgb_l3len(msg) - sizeof(*hdr);
235
236 switch (hdr->msg_type) {
237 case ISUP_MSG_GRS:
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100238 rc = handle_circuit_reset_grs(set, sls, hdr->cic, hdr->data, payload_size);
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100239 break;
Holger Hans Peter Freyther8f6eb5e2011-01-19 09:27:41 +0100240 case ISUP_MSG_CGB:
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100241 rc = handle_circuit_reset_cgb(set, sls, hdr->cic, hdr->data, payload_size);
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +0100242 if (rc == 0)
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100243 rc = send_cgu(set, sls, hdr->cic, 28);
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +0100244 break;
Holger Hans Peter Freythere0dc3e52011-03-05 09:50:47 +0100245 case ISUP_MSG_CGU:
246 rc = handle_cgu(set, sls, hdr->cic, hdr->data, payload_size);
247 break;
Holger Hans Peter Freyther69e71e02011-01-19 09:33:00 +0100248 case ISUP_MSG_CGUA:
Holger Hans Peter Freyther8941c1d2011-03-03 08:18:27 +0100249 LOGP(DISUP, LOGL_NOTICE, "CIC %d is now unblocked on linkset %d/%s.\n",
250 hdr->cic, set->nr, set->name);
Holger Hans Peter Freyther8f6eb5e2011-01-19 09:27:41 +0100251 break;
Holger Hans Peter Freyther6e11bee2011-01-18 23:34:19 +0100252 case ISUP_MSG_RSC:
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100253 rc = handle_simple_resp(set, sls, hdr->cic, ISUP_MSG_RLC);
Holger Hans Peter Freyther6e11bee2011-01-18 23:34:19 +0100254 break;
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100255 default:
Holger Hans Peter Freytherd062f832011-02-23 16:58:15 +0100256 mtp_link_set_forward_isup(set, msg, sls);
Holger Hans Peter Freyther1b6901e2011-01-17 16:13:28 +0100257 rc = 0;
Holger Hans Peter Freyther433ea2f2010-12-10 07:34:01 +0100258 break;
259 }
260
261 return rc;
262}