1240 lines
31 KiB
C
1240 lines
31 KiB
C
/*
|
|
* Copyright (c) 1997, 2000 Hellmuth Michaelis. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. 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.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR 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 AUTHOR 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.
|
|
*
|
|
*---------------------------------------------------------------------------
|
|
*
|
|
* q932_fac.c - decode Q.932 facilities
|
|
* ------------------------------------
|
|
*
|
|
* $Id: q932_fac.c,v 1.4 2009/04/16 05:56:33 lukem Exp $
|
|
*
|
|
* $FreeBSD$
|
|
*
|
|
* last edit-date: [Thu Feb 24 17:36:47 2000]
|
|
*
|
|
*---------------------------------------------------------------------------
|
|
*
|
|
* - Q.932 (03/93) Generic Procedures for the Control of
|
|
* ISDN Supplementaty Services
|
|
* - Q.950 (03/93) Supplementary Services Protocols, Structure and
|
|
* General Principles
|
|
* - ETS 300 179 (10/92) Advice Of Charge: charging information during
|
|
* the call (AOC-D) supplementary service Service description
|
|
* - ETS 300 180 (10/92) Advice Of Charge: charging information at the
|
|
* end of call (AOC-E) supplementary service Service description
|
|
* - ETS 300 181 (04/93) Advice Of Charge (AOC) supplementary service
|
|
* Functional capabilities and information flows
|
|
* - ETS 300 182 (04/93) Advice Of Charge (AOC) supplementary service
|
|
* Digital Subscriber Signalling System No. one (DSS1) protocol
|
|
* - X.208 Specification of Abstract Syntax Notation One (ASN.1)
|
|
* - X.209 Specification of Basic Encoding Rules for
|
|
* Abstract Syntax Notation One (ASN.1)
|
|
* - "ASN.1 Abstract Syntax Notation One", Walter Gora, DATACOM-Verlag
|
|
* 1992, 3rd Edition (ISBN 3-89238-062-7) (german !)
|
|
*
|
|
*---------------------------------------------------------------------------*/
|
|
|
|
#include "trace.h"
|
|
#include "q932_fac.h"
|
|
|
|
static int do_component(int length, char *pbuf);
|
|
static const char *uni_str(int code);
|
|
static const char *opval_str(int val);
|
|
static const char *bid_str(int val);
|
|
static void next_state(char *pbuf, int class, int form, int code, int val);
|
|
|
|
static void object_id(int comp_length, unsigned char *pbuf);
|
|
|
|
static int byte_len;
|
|
static unsigned char *byte_buf;
|
|
static int state;
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* decode Q.931/Q.932 facility info element
|
|
*---------------------------------------------------------------------------*/
|
|
int
|
|
q932_facility(char *pbuf, unsigned char *buf)
|
|
{
|
|
int len;
|
|
|
|
sprintf((pbuf+strlen(pbuf)), "[facility (Q.932): ");
|
|
|
|
buf++; /* length */
|
|
|
|
len = *buf;
|
|
|
|
buf++; /* protocol profile */
|
|
|
|
sprintf((pbuf+strlen(pbuf)), "Protocol=");
|
|
|
|
switch (*buf & 0x1f)
|
|
{
|
|
case FAC_PROTO_ROP:
|
|
sprintf((pbuf+strlen(pbuf)), "Remote Operations Protocol\n");
|
|
break;
|
|
|
|
case FAC_PROTO_CMIP:
|
|
sprintf((pbuf+strlen(pbuf)), "CMIP Protocol (Q.941), UNSUPPORTED!\n");
|
|
return(len+2);
|
|
break;
|
|
|
|
case FAC_PROTO_ACSE:
|
|
sprintf((pbuf+strlen(pbuf)), "ACSE Protocol (X.217/X.227), UNSUPPORTED!\n");
|
|
return(len+2);
|
|
break;
|
|
|
|
default:
|
|
sprintf((pbuf+strlen(pbuf)), "Unknown Protocol (val = 0x%x), UNSUPPORTED!\n", *buf & 0x1f);
|
|
return(len+2);
|
|
break;
|
|
}
|
|
|
|
/* next byte */
|
|
|
|
buf++;
|
|
len--;
|
|
|
|
/* initialize variables for do_component */
|
|
|
|
byte_len = 0;
|
|
byte_buf = buf;
|
|
state = ST_EXP_COMP_TYP;
|
|
|
|
/* decode facility */
|
|
|
|
do_component(len, pbuf);
|
|
|
|
sprintf((pbuf+(strlen(pbuf)-1)), "]"); /* XXX replace last newline */
|
|
|
|
return(len+3);
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* handle a component recursively
|
|
*---------------------------------------------------------------------------*/
|
|
static int
|
|
do_component(int length, char *pbuf)
|
|
{
|
|
int comp_tag_class; /* component tag class */
|
|
int comp_tag_form; /* component form: constructor or primitive */
|
|
int comp_tag_code; /* component code depending on class */
|
|
int comp_length = 0; /* component length */
|
|
|
|
#ifdef FAC_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "ENTER - comp_length = %d, byte_len = %d, length =%d\n", comp_length, byte_len, length);
|
|
#endif
|
|
|
|
again:
|
|
|
|
#ifdef FAC_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "AGAIN - comp_length = %d, byte_len = %d, length =%d\n", comp_length, byte_len, length);
|
|
#endif
|
|
|
|
/*----------------------------------------*/
|
|
/* first component element: component tag */
|
|
/*----------------------------------------*/
|
|
|
|
/* tag class bits */
|
|
|
|
sprintf((pbuf+strlen(pbuf)), "\t0x%02x Tag: ", *byte_buf);
|
|
|
|
comp_tag_class = (*byte_buf & 0xc0) >> 6;
|
|
|
|
switch (comp_tag_class)
|
|
{
|
|
case FAC_TAGCLASS_UNI:
|
|
sprintf((pbuf+strlen(pbuf)), "Universal");
|
|
break;
|
|
case FAC_TAGCLASS_APW:
|
|
sprintf((pbuf+strlen(pbuf)), "Applic-wide");
|
|
break;
|
|
case FAC_TAGCLASS_COS:
|
|
sprintf((pbuf+strlen(pbuf)), "Context-spec");
|
|
break;
|
|
case FAC_TAGCLASS_PRU:
|
|
sprintf((pbuf+strlen(pbuf)), "Private");
|
|
break;
|
|
}
|
|
|
|
/* tag form bit */
|
|
|
|
comp_tag_form = (*byte_buf & 0x20) > 5;
|
|
|
|
sprintf((pbuf+strlen(pbuf)), ", ");
|
|
|
|
if (comp_tag_form == FAC_TAGFORM_CON)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "Constructor");
|
|
}
|
|
else
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "Primitive");
|
|
}
|
|
|
|
/* tag code bits */
|
|
|
|
comp_tag_code = *byte_buf & 0x1f;
|
|
|
|
sprintf((pbuf+strlen(pbuf)), ", ");
|
|
|
|
if (comp_tag_code == 0x1f)
|
|
{
|
|
comp_tag_code = 0;
|
|
|
|
byte_buf++;
|
|
byte_len++;
|
|
|
|
while(*byte_buf & 0x80)
|
|
{
|
|
comp_tag_code += (*byte_buf & 0x7f);
|
|
byte_buf++;
|
|
byte_len++;
|
|
}
|
|
comp_tag_code += (*byte_buf & 0x7f);
|
|
sprintf((pbuf+strlen(pbuf)), "%d (ext)\n", comp_tag_code);
|
|
}
|
|
else
|
|
{
|
|
comp_tag_code = (*byte_buf & 0x1f);
|
|
|
|
if (comp_tag_class == FAC_TAGCLASS_UNI)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "%s (%d)\n", uni_str(comp_tag_code), comp_tag_code);
|
|
}
|
|
else
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "code = %d\n", comp_tag_code);
|
|
}
|
|
}
|
|
|
|
byte_buf++;
|
|
byte_len++;
|
|
|
|
/*--------------------------------------------*/
|
|
/* second component element: component length */
|
|
/*--------------------------------------------*/
|
|
|
|
sprintf((pbuf+strlen(pbuf)), "\t0x%02x Len: ", *byte_buf);
|
|
|
|
comp_length = 0;
|
|
|
|
if (*byte_buf & 0x80)
|
|
{
|
|
int i = *byte_buf & 0x7f;
|
|
|
|
byte_len += i;
|
|
|
|
for (;i > 0;i++)
|
|
{
|
|
byte_buf++;
|
|
comp_length += (*byte_buf * (i*256));
|
|
}
|
|
sprintf((pbuf+strlen(pbuf)), "%d (long form)\n", comp_length);
|
|
}
|
|
else
|
|
{
|
|
comp_length = *byte_buf & 0x7f;
|
|
sprintf((pbuf+strlen(pbuf)), "%d (short form)\n", comp_length);
|
|
}
|
|
|
|
next_state(pbuf, comp_tag_class, comp_tag_form, comp_tag_code, -1);
|
|
|
|
byte_len++;
|
|
byte_buf++;
|
|
|
|
if (comp_length)
|
|
{
|
|
|
|
/*---------------------------------------------*/
|
|
/* third component element: component contents */
|
|
/*---------------------------------------------*/
|
|
|
|
if (comp_tag_form) /* == constructor */
|
|
{
|
|
do_component(comp_length, pbuf);
|
|
}
|
|
else
|
|
{
|
|
int val = 0;
|
|
if (comp_tag_class == FAC_TAGCLASS_UNI)
|
|
{
|
|
switch (comp_tag_code)
|
|
{
|
|
case FAC_CODEUNI_INT:
|
|
case FAC_CODEUNI_ENUM:
|
|
case FAC_CODEUNI_BOOL:
|
|
if (comp_length)
|
|
{
|
|
int i;
|
|
|
|
sprintf((pbuf+strlen(pbuf)), "\t");
|
|
|
|
for (i = comp_length-1; i >= 0; i--)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "0x%02x ", *byte_buf);
|
|
val += (*byte_buf + (i*255));
|
|
byte_buf++;
|
|
byte_len++;
|
|
if (i)
|
|
sprintf((pbuf+strlen(pbuf)), "\n\t");
|
|
}
|
|
sprintf((pbuf+strlen(pbuf)), "Val: %d\n", val);
|
|
}
|
|
break;
|
|
|
|
case FAC_CODEUNI_OBJI: /* object id */
|
|
|
|
if (comp_length)
|
|
object_id(comp_length, pbuf);
|
|
break;
|
|
|
|
default:
|
|
if (comp_length)
|
|
{
|
|
int i;
|
|
|
|
sprintf((pbuf+strlen(pbuf)), "\t");
|
|
|
|
for (i = comp_length-1; i >= 0; i--)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "0x%02x = %d", *byte_buf, *byte_buf);
|
|
if (isprint(*byte_buf))
|
|
sprintf((pbuf+strlen(pbuf)), " = '%c'", *byte_buf);
|
|
byte_buf++;
|
|
byte_len++;
|
|
if (i)
|
|
sprintf((pbuf+strlen(pbuf)), "\n\t");
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
else /* comp_tag_class != FAC_TAGCLASS_UNI */
|
|
{
|
|
if (comp_length)
|
|
{
|
|
int i;
|
|
|
|
sprintf((pbuf+strlen(pbuf)), "\t");
|
|
|
|
for (i = comp_length-1; i >= 0; i--)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "0x%02x", *byte_buf);
|
|
val += (*byte_buf + (i*255));
|
|
byte_buf++;
|
|
byte_len++;
|
|
if (i)
|
|
sprintf((pbuf+strlen(pbuf)), "\n\t");
|
|
}
|
|
sprintf((pbuf+strlen(pbuf)), "\n");
|
|
}
|
|
}
|
|
next_state(pbuf, comp_tag_class, comp_tag_form, comp_tag_code, val);
|
|
}
|
|
}
|
|
|
|
#ifdef FAC_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "PREGOTO - comp_length = %d, byte_len = %d, length =%d\n", comp_length, byte_len, length);
|
|
#endif
|
|
if (byte_len < length)
|
|
goto again;
|
|
#ifdef FAC_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "RETURN - comp_length = %d, byte_len = %d, length =%d\n", comp_length, byte_len, length);
|
|
#endif
|
|
return(byte_len);
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* print universal id type
|
|
*---------------------------------------------------------------------------*/
|
|
static const char *uni_str(int code)
|
|
{
|
|
static const char *tbl[] = {
|
|
"BOOLEAN",
|
|
"INTEGER",
|
|
"BIT STRING",
|
|
"OCTET STRING",
|
|
"NULL",
|
|
"OBJECT IDENTIFIER",
|
|
"OBJECT DESCRIPTOR",
|
|
"EXTERNAL",
|
|
"REAL",
|
|
"ENUMERATED",
|
|
"RESERVED11",
|
|
"RESERVED12",
|
|
"RESERVED13",
|
|
"RESERVED14",
|
|
"RESERVED15",
|
|
"SEQUENCE",
|
|
"SET",
|
|
"NUMERIC STRING",
|
|
"PRINTABLE STRING",
|
|
"TELETEX STRING",
|
|
"ISO646 STRING",
|
|
"IA5 STRING",
|
|
"U TIME",
|
|
"G TIME",
|
|
"GRAPHIC STRING",
|
|
"VISUAL STRING",
|
|
"GENERAL STRING"
|
|
};
|
|
|
|
if (code >= 1 && code <= (int)(sizeof(tbl) / sizeof(tbl[0])))
|
|
return(tbl[code-1]);
|
|
else
|
|
return("ERROR, Value out of Range!");
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* print operation value
|
|
*---------------------------------------------------------------------------*/
|
|
static const char *opval_str(int val)
|
|
{
|
|
static char buffer[80];
|
|
const char *r;
|
|
|
|
switch (val)
|
|
{
|
|
case FAC_OPVAL_UUS:
|
|
r = "uUs";
|
|
break;
|
|
case FAC_OPVAL_CUG:
|
|
r = "cUGCall";
|
|
break;
|
|
case FAC_OPVAL_MCID:
|
|
r = "mCIDRequest";
|
|
break;
|
|
case FAC_OPVAL_BTPY:
|
|
r = "beginTPY";
|
|
break;
|
|
case FAC_OPVAL_ETPY:
|
|
r = "endTPY";
|
|
break;
|
|
case FAC_OPVAL_ECT:
|
|
r = "eCTRequest";
|
|
break;
|
|
case FAC_OPVAL_DIV_ACT:
|
|
r = "activationDiversion";
|
|
break;
|
|
case FAC_OPVAL_DIV_DEACT:
|
|
r = "deactivationDiversion";
|
|
break;
|
|
case FAC_OPVAL_DIV_ACTSN:
|
|
r = "activationStatusNotificationDiv";
|
|
break;
|
|
case FAC_OPVAL_DIV_DEACTSN:
|
|
r = "deactivationStatusNotificationDiv";
|
|
break;
|
|
case FAC_OPVAL_DIV_INTER:
|
|
r = "interrogationDiversion";
|
|
break;
|
|
case FAC_OPVAL_DIV_INFO:
|
|
r = "diversionInformation";
|
|
break;
|
|
case FAC_OPVAL_DIV_CALLDEF:
|
|
r = "callDeflection";
|
|
break;
|
|
case FAC_OPVAL_DIV_CALLRER:
|
|
r = "callRerouting";
|
|
break;
|
|
case FAC_OPVAL_DIV_LINF2:
|
|
r = "divertingLegInformation2";
|
|
break;
|
|
case FAC_OPVAL_DIV_INVS:
|
|
r = "invokeStatus";
|
|
break;
|
|
case FAC_OPVAL_DIV_INTER1:
|
|
r = "interrogationDiversion1";
|
|
break;
|
|
case FAC_OPVAL_DIV_LINF1:
|
|
r = "divertingLegInformation1";
|
|
break;
|
|
case FAC_OPVAL_DIV_LINF3:
|
|
r = "divertingLegInformation3";
|
|
break;
|
|
case FAC_OPVAL_ER_CRCO:
|
|
r = "explicitReservationCreationControl";
|
|
break;
|
|
case FAC_OPVAL_ER_MGMT:
|
|
r = "explicitReservationManagement";
|
|
break;
|
|
case FAC_OPVAL_ER_CANC:
|
|
r = "explicitReservationCancel";
|
|
break;
|
|
case FAC_OPVAL_MLPP_QUERY:
|
|
r = "mLPP lfb Query";
|
|
break;
|
|
case FAC_OPVAL_MLPP_CALLR:
|
|
r = "mLPP Call Request";
|
|
break;
|
|
case FAC_OPVAL_MLPP_CALLP:
|
|
r = "mLPP Call Preemption";
|
|
break;
|
|
case FAC_OPVAL_AOC_REQ:
|
|
r = "chargingRequest";
|
|
break;
|
|
case FAC_OPVAL_AOC_S_CUR:
|
|
r = "aOCSCurrency";
|
|
break;
|
|
case FAC_OPVAL_AOC_S_SPC:
|
|
r = "aOCSSpecialArrangement";
|
|
break;
|
|
case FAC_OPVAL_AOC_D_CUR:
|
|
r = "aOCDCurrency";
|
|
break;
|
|
case FAC_OPVAL_AOC_D_UNIT:
|
|
r = "aOCDChargingUnit";
|
|
break;
|
|
case FAC_OPVAL_AOC_E_CUR:
|
|
r = "aOCECurrency";
|
|
break;
|
|
case FAC_OPVAL_AOC_E_UNIT:
|
|
r = "aOCEChargingUnit";
|
|
break;
|
|
case FAC_OPVAL_AOC_IDOFCRG:
|
|
r = "identificationOfCharge";
|
|
break;
|
|
case FAC_OPVAL_CONF_BEG:
|
|
r = "beginConf";
|
|
break;
|
|
case FAC_OPVAL_CONF_ADD:
|
|
r = "addConf";
|
|
break;
|
|
case FAC_OPVAL_CONF_SPLIT:
|
|
r = "splitConf";
|
|
break;
|
|
case FAC_OPVAL_CONF_DROP:
|
|
r = "dropConf";
|
|
break;
|
|
case FAC_OPVAL_CONF_ISOLATE:
|
|
r = "isolateConf";
|
|
break;
|
|
case FAC_OPVAL_CONF_REATT:
|
|
r = "reattachConf";
|
|
break;
|
|
case FAC_OPVAL_CONF_PDISC:
|
|
r = "partyDISC";
|
|
break;
|
|
case FAC_OPVAL_CONF_FCONF:
|
|
r = "floatConf";
|
|
break;
|
|
case FAC_OPVAL_CONF_END:
|
|
r = "endConf";
|
|
break;
|
|
case FAC_OPVAL_CONF_IDCFE:
|
|
r = "indentifyConferee";
|
|
break;
|
|
case FAC_OPVAL_REVC_REQ:
|
|
r = "requestREV";
|
|
break;
|
|
default:
|
|
sprintf(buffer, "unknown operation value %d!", val);
|
|
r = buffer;
|
|
}
|
|
return(r);
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* billing id string
|
|
*---------------------------------------------------------------------------*/
|
|
static const char *bid_str(int val)
|
|
{
|
|
static char buffer[80];
|
|
const char *r;
|
|
|
|
switch (val)
|
|
{
|
|
case 0:
|
|
r = "normalCharging";
|
|
break;
|
|
case 1:
|
|
r = "reverseCharging";
|
|
break;
|
|
case 2:
|
|
r = "creditCardCharging";
|
|
break;
|
|
case 3:
|
|
r = "callForwardingUnconditional";
|
|
break;
|
|
case 4:
|
|
r = "callForwardingBusy";
|
|
break;
|
|
case 5:
|
|
r = "callForwardingNoReply";
|
|
break;
|
|
case 6:
|
|
r = "callDeflection";
|
|
break;
|
|
case 7:
|
|
r = "callTransfer";
|
|
break;
|
|
default:
|
|
sprintf(buffer, "unknown billing-id value %d!", val);
|
|
r = buffer;
|
|
}
|
|
return(r);
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* invoke component
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_1_1(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_1_1, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t invokeComponent\n");
|
|
state = ST_EXP_INV_ID;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* return result
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_1_2(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_1_2, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t returnResult\n");
|
|
state = ST_EXP_RR_INV_ID;
|
|
}
|
|
}
|
|
/*---------------------------------------------------------------------------*
|
|
* return error
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_1_3(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_1_3, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t returnError\n");
|
|
state = ST_EXP_NIX;
|
|
}
|
|
}
|
|
/*---------------------------------------------------------------------------*
|
|
* reject
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_1_4(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_1_4, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t reject\n");
|
|
state = ST_EXP_REJ_INV_ID;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* return result: invoke id
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_RJ2(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_RJ2, val = %d\n", val);
|
|
#endif
|
|
if (val != -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t InvokeIdentifier = %d\n", val);
|
|
state = ST_EXP_REJ_OP_VAL;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* reject, general problem
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_RJ30(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_RJ30, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t General problem\n");
|
|
}
|
|
else
|
|
{
|
|
switch (val)
|
|
{
|
|
case 0:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unrecognized component\n");
|
|
break;
|
|
case 1:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = mistyped component\n");
|
|
break;
|
|
case 2:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = badly structured component\n");
|
|
break;
|
|
default:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unknown problem code 0x%x\n", val);
|
|
break;
|
|
}
|
|
state = ST_EXP_NIX;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* reject, invoke problem
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_RJ31(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_RJ31, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t Invoke problem\n");
|
|
}
|
|
else
|
|
{
|
|
switch (val)
|
|
{
|
|
case 0:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = duplicate invocation\n");
|
|
break;
|
|
case 1:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unrecognized operation\n");
|
|
break;
|
|
case 2:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = mistyped argument\n");
|
|
break;
|
|
case 3:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = resource limitation\n");
|
|
break;
|
|
case 4:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = initiator releasing\n");
|
|
break;
|
|
case 5:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unrecognized linked identifier\n");
|
|
break;
|
|
case 6:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = linked resonse unexpected\n");
|
|
break;
|
|
case 7:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unexpected child operation\n");
|
|
break;
|
|
default:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unknown problem code 0x%x\n", val);
|
|
break;
|
|
}
|
|
state = ST_EXP_NIX;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* reject, return result problem
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_RJ32(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_RJ32, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t Return result problem\n");
|
|
}
|
|
else
|
|
{
|
|
switch (val)
|
|
{
|
|
case 0:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unrecognized invocation\n");
|
|
break;
|
|
case 1:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = return response unexpected\n");
|
|
break;
|
|
case 2:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = mistyped result\n");
|
|
break;
|
|
default:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unknown problem code 0x%x\n", val);
|
|
break;
|
|
}
|
|
state = ST_EXP_NIX;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* reject, return error problem
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_RJ33(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_RJ33, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t Return error problem\n");
|
|
}
|
|
else
|
|
{
|
|
switch (val)
|
|
{
|
|
case 0:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unrecognized invocation\n");
|
|
break;
|
|
case 1:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = error response unexpected\n");
|
|
break;
|
|
case 2:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unrecognized error\n");
|
|
break;
|
|
case 3:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unexpected error\n");
|
|
break;
|
|
case 4:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = mistyped parameter\n");
|
|
break;
|
|
default:
|
|
sprintf((pbuf+strlen(pbuf)), "\t problem = unknown problem code 0x%x\n", val);
|
|
break;
|
|
}
|
|
state = ST_EXP_NIX;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* invoke component: invoke id
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_2(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_2, val = %d\n", val);
|
|
#endif
|
|
if (val != -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t InvokeIdentifier = %d\n", val);
|
|
state = ST_EXP_OP_VAL;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* return result: invoke id
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_RR2(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_RR2, val = %d\n", val);
|
|
#endif
|
|
if (val != -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t InvokeIdentifier = %d\n", val);
|
|
state = ST_EXP_RR_OP_VAL;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* invoke component: operation value
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_3(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_3, val = %d\n", val);
|
|
#endif
|
|
if (val != -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t Operation Value = %s (%d)\n", opval_str(val), val);
|
|
state = ST_EXP_INFO;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* return result: operation value
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_RR3(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_RR3, val = %d\n", val);
|
|
#endif
|
|
if (val != -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t Operation Value = %s (%d)\n", opval_str(val), val);
|
|
state = ST_EXP_RR_RESULT;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* return result: RESULT
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_RRR(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_RRR, val = %d\n", val);
|
|
#endif
|
|
state = ST_EXP_NIX;
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
*
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_4(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_4, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t specificChargingUnits\n");
|
|
state = ST_EXP_RUL;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
*
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_4_1(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_4_1, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t freeOfCharge\n");
|
|
state = ST_EXP_NIX;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
*
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_4_2(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_4_2, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t chargeNotAvailable\n");
|
|
state = ST_EXP_NIX;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
*
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_5(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_5, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t recordedUnitsList [1]\n");
|
|
state = ST_EXP_RU;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
*
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_6(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_6, val = %d\n", val);
|
|
#endif
|
|
if (val == -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t RecordedUnits\n");
|
|
state = ST_EXP_RNOU;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
*
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_7(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_7, val = %d\n", val);
|
|
#endif
|
|
if (val != -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t NumberOfUnits = %d\n", val);
|
|
state = ST_EXP_TOCI;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
*
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_8(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_8, val = %d\n", val);
|
|
#endif
|
|
if (val != -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t typeOfChargingInfo = %s\n", val == 0 ? "subTotal" : "total");
|
|
state = ST_EXP_DBID;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
*
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
F_9(char *pbuf, int val)
|
|
{
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: exec F_9, val = %d\n", val);
|
|
#endif
|
|
if (val != -1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "\t AOCDBillingId = %s (%d)\n", bid_str(val), val);
|
|
state = ST_EXP_NIX;
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* state table
|
|
*---------------------------------------------------------------------------*/
|
|
static struct statetab {
|
|
int currstate; /* input: current state we are in */
|
|
int form; /* input: current tag form */
|
|
int class; /* input: current tag class */
|
|
int code; /* input: current tag code */
|
|
void (*func)(char *,int); /* output: func to exec */
|
|
} statetab[] = {
|
|
|
|
/* current state tag form tag class tag code function */
|
|
/* --------------------- ---------------------- ---------------------- ---------------------- ----------------*/
|
|
|
|
/* invoke */
|
|
|
|
{ST_EXP_COMP_TYP, FAC_TAGFORM_CON, FAC_TAGCLASS_COS, 1, F_1_1 },
|
|
{ST_EXP_COMP_TYP, FAC_TAGFORM_CON, FAC_TAGCLASS_COS, 2, F_1_2 },
|
|
{ST_EXP_COMP_TYP, FAC_TAGFORM_CON, FAC_TAGCLASS_COS, 3, F_1_3 },
|
|
{ST_EXP_COMP_TYP, FAC_TAGFORM_CON, FAC_TAGCLASS_COS, 4, F_1_4 },
|
|
{ST_EXP_INV_ID, FAC_TAGFORM_PRI, FAC_TAGCLASS_UNI, FAC_CODEUNI_INT, F_2 },
|
|
{ST_EXP_OP_VAL, FAC_TAGFORM_PRI, FAC_TAGCLASS_UNI, FAC_CODEUNI_INT, F_3 },
|
|
{ST_EXP_INFO, FAC_TAGFORM_CON, FAC_TAGCLASS_UNI, FAC_CODEUNI_SEQ, F_4 },
|
|
{ST_EXP_INFO, FAC_TAGFORM_PRI, FAC_TAGCLASS_UNI, FAC_CODEUNI_NULL, F_4_1 },
|
|
{ST_EXP_INFO, FAC_TAGFORM_PRI, FAC_TAGCLASS_COS, 1, F_4_2 },
|
|
{ST_EXP_RUL, FAC_TAGFORM_CON, FAC_TAGCLASS_COS, 1, F_5 },
|
|
{ST_EXP_RU, FAC_TAGFORM_CON, FAC_TAGCLASS_UNI, FAC_CODEUNI_SEQ, F_6 },
|
|
{ST_EXP_RNOU, FAC_TAGFORM_PRI, FAC_TAGCLASS_UNI, FAC_CODEUNI_INT, F_7 },
|
|
{ST_EXP_TOCI, FAC_TAGFORM_PRI, FAC_TAGCLASS_COS, 2, F_8 },
|
|
{ST_EXP_DBID, FAC_TAGFORM_PRI, FAC_TAGCLASS_COS, 3, F_9 },
|
|
|
|
/* return result */
|
|
|
|
{ST_EXP_RR_INV_ID, FAC_TAGFORM_PRI, FAC_TAGCLASS_UNI, FAC_CODEUNI_INT, F_RR2 },
|
|
{ST_EXP_RR_OP_VAL, FAC_TAGFORM_PRI, FAC_TAGCLASS_UNI, FAC_CODEUNI_INT, F_RR3 },
|
|
{ST_EXP_RR_RESULT, FAC_TAGFORM_CON, FAC_TAGCLASS_UNI, FAC_CODEUNI_SET, F_RRR },
|
|
|
|
/* current state tag form tag class tag code function */
|
|
/* --------------------- ---------------------- ---------------------- ---------------------- ----------------*/
|
|
/* reject */
|
|
|
|
{ST_EXP_REJ_INV_ID, FAC_TAGFORM_PRI, FAC_TAGCLASS_UNI, FAC_CODEUNI_INT, F_RJ2 },
|
|
{ST_EXP_REJ_OP_VAL, FAC_TAGFORM_PRI, FAC_TAGCLASS_COS, 0, F_RJ30 },
|
|
{ST_EXP_REJ_OP_VAL, FAC_TAGFORM_PRI, FAC_TAGCLASS_COS, 1, F_RJ31 },
|
|
{ST_EXP_REJ_OP_VAL, FAC_TAGFORM_PRI, FAC_TAGCLASS_COS, 2, F_RJ32 },
|
|
{ST_EXP_REJ_OP_VAL, FAC_TAGFORM_PRI, FAC_TAGCLASS_COS, 3, F_RJ33 },
|
|
|
|
/* end */
|
|
|
|
{-1, -1, -1, -1, NULL }
|
|
};
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* state decode for do_component
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
next_state(char *pbuf, int class, int form, int code, int val)
|
|
{
|
|
int i;
|
|
|
|
#ifdef ST_DEBUG
|
|
sprintf((pbuf+strlen(pbuf)), "next_state: class=%d, form=%d, code=%d, val=%d\n", class, form, code, val);
|
|
#endif
|
|
|
|
for (i=0; ; i++)
|
|
{
|
|
if ((statetab[i].currstate > state) ||
|
|
(statetab[i].currstate == -1))
|
|
{
|
|
break;
|
|
}
|
|
|
|
if ((statetab[i].currstate == state) &&
|
|
(statetab[i].form == form) &&
|
|
(statetab[i].class == class) &&
|
|
(statetab[i].code == code))
|
|
{
|
|
(*statetab[i].func)(pbuf, val);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*
|
|
* decode OBJECT IDENTIFIER
|
|
*---------------------------------------------------------------------------*/
|
|
static void
|
|
object_id(int comp_length, unsigned char *pbuf)
|
|
{
|
|
int x;
|
|
int i;
|
|
int j = 0;
|
|
int id_org = 0;
|
|
int etsi = 0;
|
|
|
|
sprintf((pbuf+strlen(pbuf)), "\t");
|
|
|
|
for (i = comp_length - 1; i >= 0; i--, j++)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), "0x%02x = %d", *byte_buf, *byte_buf);
|
|
|
|
if (j == 0)
|
|
{
|
|
x = *byte_buf;
|
|
|
|
if (x >= 0 && x <= 39)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), " ccitt/itu-t (0)");
|
|
switch (x)
|
|
{
|
|
case 0:
|
|
sprintf((pbuf+strlen(pbuf)), " recommendation (0)");
|
|
break;
|
|
case 1:
|
|
sprintf((pbuf+strlen(pbuf)), " question (1)");
|
|
break;
|
|
case 2:
|
|
sprintf((pbuf+strlen(pbuf)), " administration (2)");
|
|
break;
|
|
case 3:
|
|
sprintf((pbuf+strlen(pbuf)), " network-operator (3)");
|
|
break;
|
|
case 4:
|
|
sprintf((pbuf+strlen(pbuf)), " identified-organization (4)");
|
|
id_org = 1;
|
|
break;
|
|
default:
|
|
sprintf((pbuf+strlen(pbuf)), " error: undefined-identifier (%d)", x);
|
|
break;
|
|
}
|
|
}
|
|
else if (x >= 40 && x <= 79)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), " iso (1)");
|
|
x -= 40;
|
|
switch (x)
|
|
{
|
|
case 0:
|
|
sprintf((pbuf+strlen(pbuf)), " standard (0)");
|
|
break;
|
|
case 1:
|
|
sprintf((pbuf+strlen(pbuf)), " registration-authority (1)");
|
|
break;
|
|
case 2:
|
|
sprintf((pbuf+strlen(pbuf)), " member-body (2)");
|
|
break;
|
|
case 3:
|
|
sprintf((pbuf+strlen(pbuf)), " identified-organization (3)");
|
|
id_org = 1;
|
|
break;
|
|
default:
|
|
sprintf((pbuf+strlen(pbuf)), " error: undefined-identifier (%d)", x);
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
x -= 80;
|
|
sprintf((pbuf+strlen(pbuf)), " joint-iso-ccitt (3) ??? (%d)", x);
|
|
}
|
|
}
|
|
|
|
if (j == 1)
|
|
{
|
|
if (id_org == 1)
|
|
{
|
|
if (*byte_buf == 0)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), " etsi (0)");
|
|
etsi = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (j == 2)
|
|
{
|
|
if (etsi == 1)
|
|
{
|
|
if (*byte_buf == 0)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), " mobileDomain (0)");
|
|
}
|
|
if (*byte_buf == 1)
|
|
{
|
|
sprintf((pbuf+strlen(pbuf)), " inDomain (1)");
|
|
}
|
|
}
|
|
}
|
|
|
|
byte_buf++;
|
|
byte_len++;
|
|
|
|
if (i)
|
|
sprintf((pbuf+strlen(pbuf)), "\n\t");
|
|
else
|
|
sprintf((pbuf+strlen(pbuf)), "\n");
|
|
}
|
|
}
|
|
|
|
/* EOF */
|
|
|