2 * Copyright (c) 1997, 1999 Hellmuth Michaelis. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25 *---------------------------------------------------------------------------
27 * layer3.c - decode and print layer 3 (Q.931) information
28 * -------------------------------------------------------
30 * $Id: layer3.c,v 1.9 2000/02/21 15:17:17 hm Exp $
32 * $FreeBSD: src/usr.sbin/i4b/isdndecode/layer3.c,v 1.6.2.2 2001/08/01 17:45:05 obrien Exp $
34 * last edit-date: [Mon Feb 21 15:54:58 2000]
36 *---------------------------------------------------------------------------*/
42 /* 0x00 */ /* call establishment group */
59 "CONNECT ACKNOWLEDGE",
81 "USER INFORMATION", /* call information phase */
94 "SUSPEND ACKNOWLEDGE",
103 "RETRIEVE ACKNOWLEDGE",
119 "DETACH", /* call clearing */
127 "DETACH ACKNOWLEDGE",
133 "RESTART ACKNOWLEDGE",
157 "SEGMENT", /* misc messages */
165 "CANCEL ACKNOWLEDGE",
167 "FACILITY ACKNOWLEDGE",
169 "REGISTER ACKNOWLEDGE",
185 "CONGESTION CONTROL",
194 #define MTTAB_MAX 0x7f
196 extern int f_null(char *pbuf, unsigned char *buf, int off);
197 extern int f_bc(char *pbuf, unsigned char *buf, int off);
198 extern int f_cause(char *pbuf, unsigned char *buf, int off);
199 extern int f_cstat(char *pbuf, unsigned char *buf, int off);
200 extern int f_chid(char *pbuf, unsigned char *buf, int off);
201 extern int f_fac(char *pbuf, unsigned char *buf, int off);
202 extern int f_progi(char *pbuf, unsigned char *buf, int off);
203 extern int f_displ(char *pbuf, unsigned char *buf, int off);
204 extern int f_date(char *pbuf, unsigned char *buf, int off);
205 extern int f_cnu(char *pbuf, unsigned char *buf, int off);
206 extern int f_cgpn(char *pbuf, unsigned char *buf, int off);
207 extern int f_cdpn(char *pbuf, unsigned char *buf, int off);
208 extern int f_hlc(char *pbuf, unsigned char *buf, int off);
209 extern int f_uu(char *pbuf, unsigned char *buf, int off);
212 unsigned char code; /* information element identifier code */
213 char *name; /* ie name */
214 int (*func) (char *pbuf, unsigned char *buf, int off); /* decode function */
216 { 0x00, "segmented message", f_null },
217 { 0x04, "bearer capability", f_bc },
218 { 0x08, "cause", f_cause },
219 { 0x0c, "connected address", f_null },
220 { 0x0d, "extended facility", f_null },
221 { 0x10, "call identity", f_null },
222 { 0x14, "call state", f_cstat },
223 { 0x18, "channel id", f_chid },
224 { 0x19, "data link connection id", f_null },
225 { 0x1c, "facility", f_fac },
226 { 0x1e, "progress indicator", f_progi },
227 { 0x20, "network specific facilities", f_null },
228 { 0x24, "terminal capabilities", f_null },
229 { 0x27, "notification indicator", f_null },
230 { 0x28, "display", f_displ },
231 { 0x29, "date/time", f_date },
232 { 0x2c, "keypad", f_null },
233 { 0x30, "keypad echo", f_null },
234 { 0x32, "information request", f_null },
235 { 0x34, "signal", f_null },
236 { 0x36, "switchhook", f_null },
237 { 0x38, "feature activation", f_null },
238 { 0x39, "feature indication", f_null },
239 { 0x3a, "service profile id", f_null },
240 { 0x3b, "endpoint identifier", f_null },
241 { 0x40, "information rate", f_null },
242 { 0x41, "precedence level", f_null },
243 { 0x42, "end-to-end transit delay", f_null },
244 { 0x43, "transit delay detection", f_null },
245 { 0x44, "packet layer binary parms", f_null },
246 { 0x45, "packet layer window size", f_null },
247 { 0x46, "packet size", f_null },
248 { 0x47, "closed user group", f_null },
249 { 0x48, "link layer core parameters", f_null },
250 { 0x49, "link layer protocol parms", f_null },
251 { 0x4a, "reverse charging information", f_null },
252 { 0x4c, "connected number", f_cnu },
253 { 0x4d, "connected subaddress", f_null },
254 { 0x50, "X.213 priority", f_null },
255 { 0x51, "report type", f_null },
256 { 0x53, "link integrity verification", f_null },
257 { 0x57, "PVC status", f_null },
258 { 0x6c, "calling party number", f_cnu },
259 { 0x6d, "calling party subaddress", f_null },
260 { 0x70, "called party number", f_cnu },
261 { 0x71, "called party subaddress", f_null },
262 { 0x74, "redirecting number", f_null },
263 { 0x78, "transit network selection", f_null },
264 { 0x79, "restart indicator", f_null },
265 { 0x7c, "low layer compatibility", f_null },
266 { 0x7d, "high layer compatibility", f_hlc },
267 { 0x7e, "user-user", f_uu },
268 { 0x7f, "escape for extension", f_null },
269 { 0xff, "unknown information element", f_null }
272 /*---------------------------------------------------------------------------*
273 * decode Q.931 protocol
274 *---------------------------------------------------------------------------*/
276 layer3(char *pbuf, int n, int off, unsigned char *buf)
295 /* protocol discriminator */
299 if(pd >= 0x00 && pd <= 0x07)
300 sprintf(buffer, "User-User IE (0x%02x)",pd);
302 sprintf(buffer, "Q.931/I.451");
303 else if(pd >= 0x10 && pd <= 0x3f)
304 sprintf(buffer, "Other Layer 3 or X.25 (0x%02x)",pd);
305 else if(pd >= 0x40 && pd <= 0x4f)
306 sprintf(buffer, "National Use (0x%02x)",pd);
307 else if(pd >= 0x50 && pd <= 0xfe)
308 sprintf(buffer, "Other Layer 3 or X.25 (0x%02x)",pd);
310 sprintf(buffer, "Reserved (0x%02x)",pd);
312 sprintline(3, (pbuf+strlen(pbuf)), off+i, pd, 0xff, "Protocol discriminator = %s", buffer);
318 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0xff, "-");
326 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0xf0, "Call Reference");
331 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x0f, "Length of Call Reference = 0 (Dummy CR)");
334 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x0f, "Length of Call Reference = 1");
336 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x80, "Call Reference sent %s origination side", (buf[i] & 0x80) ? "to" : "from");
337 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x7f, "Call Reference = %d = 0x%02x", (buf[i] & 0x7f), (buf[i] & 0x7f));
340 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x0f, "Length of Call Reference = 2");
342 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x80, "Call Reference sent %s origination side", (buf[i] & 0x80) ? "to" : "from");
343 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x7f, "Call reference = %d = %02x", (buf[i] & 0x7f), (buf[i] & 0x7f));
345 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0xff, "Call reference = %d = %02x", (buf[i]), (buf[i]));
352 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x80, "Message type extension = %d", buf[i] & 0x80 ? 1 : 0);
354 if(buf[i] <= MTTAB_MAX)
355 strcpy(buffer, mttab[buf[i]]);
357 sprintf(buffer, "unknown (0x%02x)", buf[i]);
359 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x7f, "Message type = %s", buffer);
362 /* information elements */
366 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x80, "%s Information element", buf[i] & 0x80 ? "Single octet" : "Variable length");
370 /* single octett info element type 1 */
372 if((buf[i] & 0x70) == 0x00)
374 strcpy(buffer, "Reserved");
375 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x70, "Reserved");
376 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x0f, "Reserved, content of IE");
378 else if((buf[i] & 0x70) == 0x10)
380 strcpy(buffer, "Shift");
381 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x70, "Shift");
382 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x08, "%s shift", buf[i] & 0x08 ? "Non-locking" : "Locking");
384 switch(buf[i] & 0x07)
387 strcpy(buffer, "Not applicable");
392 sprintf(buffer, "Reserved (%d)", buf[i] & 0x07);
395 strcpy(buffer, "Codeset 4 (ISO/IEC)");
398 strcpy(buffer, "Codeset 5 (National use)");
401 strcpy(buffer, "Codeset 6 (Local network specific)");
404 strcpy(buffer, "Codeset 7 (User specific)");
407 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x07, "%s", buffer);
410 else if((buf[i] & 0x70) == 0x30)
412 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x70, "Congestion Level");
413 switch(buf[i] & 0x0f)
416 strcpy(buffer, "receiver ready");
419 strcpy(buffer, "receiver not ready");
422 sprintf(buffer, "reserved (0x%02x)", buf[i] & 0x0f);
425 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x0f, "Congestion Level = ", buffer);
428 else if((buf[i] & 0x70) == 0x50)
430 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x70, "Repeat Indicator");
431 switch(buf[i] & 0x0f)
434 strcpy(buffer, "Prioritized list for selecting one possibility");
437 sprintf(buffer, "reserved (0x%02x)", buf[i] & 0x0f);
440 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x0f, "Repeat indication = ", buffer);
444 /* single octett info element type 2 */
446 else if((buf[i] & 0x7f) == 0x20)
448 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x7f, "More data");
450 else if((buf[i] & 0x7f) == 0x21)
452 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x7f, "Sending complete");
456 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0xff, "UNKNOWN single octet IE = 0x%02x", buf[i]);
464 struct ie *iep = &ietab[0];
468 if((iep->code == buf[i]) ||
474 sprintline(3, (pbuf+strlen(pbuf)), off+i, buf[i], 0x7f, "IE = %s", iep->name);
475 sprintline(3, pbuf+strlen(pbuf), off+i+1, buf[i+1], 0xff, "IE Length = %d", buf[i+1]);
477 if(iep->func == f_null)
482 i += (iep->func)(pbuf, &buf[i], off+i);
488 sprintf((pbuf+strlen(pbuf)), "UNKNOWN CODESET=%d, IE=0x%02x", codeset, buf[i]);
491 i++; /* index -> length */
495 sprintf((pbuf+strlen(pbuf)), "LEN=0x%02x, DATA=", len);
497 i++; /* index -> 1st param */
499 for(j = 0; j < len; j++)
501 sprintf((pbuf+strlen(pbuf)),"0x%02x ", buf[j+i]);
504 sprintf((pbuf+strlen(pbuf)),"]");
510 if(!codelock && (codeset != oldcodeset))
511 codeset = oldcodeset;
514 /* sprintf((pbuf+strlen(pbuf)),"\n"); */