Added new option to interface.conf: "nonotify" to disable notify messages.
[lcr.git] / interface.c
index 24bc731..a4bac2d 100644 (file)
@@ -114,6 +114,10 @@ static int inter_block(struct interface *interface, char *filename, int line, ch
 }
 static int inter_extension(struct interface *interface, char *filename, int line, char *parameter, char *value)
 {
+       if (interface->external) {
+               SPRINT(interface_error, "Error in %s (line %d): parameter '%s' not allowed, because interface is external interface.\n", filename, line, parameter);
+               return(-1);
+       }
        if (value[0]) {
                SPRINT(interface_error, "Error in %s (line %d): parameter '%s' expects no value.\n", filename, line, parameter);
                return(-1);
@@ -121,6 +125,19 @@ static int inter_extension(struct interface *interface, char *filename, int line
        interface->extension = 1;
        return(0);
 }
+static int inter_extern(struct interface *interface, char *filename, int line, char *parameter, char *value)
+{
+       if (interface->extension) {
+               SPRINT(interface_error, "Error in %s (line %d): parameter '%s' not allowed, because interface is an extension.\n", filename, line, parameter);
+               return(-1);
+       }
+       if (value[0]) {
+               SPRINT(interface_error, "Error in %s (line %d): parameter '%s' expects no value.\n", filename, line, parameter);
+               return(-1);
+       }
+       interface->external = 1;
+       return(0);
+}
 static int inter_ptp(struct interface *interface, char *filename, int line, char *parameter, char *value)
 {
        struct interface_port *ifport;
@@ -899,6 +916,72 @@ static int inter_gsm(struct interface *interface, char *filename, int line, char
        return(0);
 #endif
 }
+static int inter_nonotify(struct interface *interface, char *filename, int line, char *parameter, char *value)
+{
+       struct interface_port *ifport;
+
+       /* port in chain ? */
+       if (!interface->ifport) {
+               SPRINT(interface_error, "Error in %s (line %d): parameter '%s' expects previous 'port' definition.\n", filename, line, parameter);
+               return(-1);
+       }
+       /* goto end of chain */
+       ifport = interface->ifport;
+       while(ifport->next)
+               ifport = ifport->next;
+       ifport->nonotify = 1;
+       return(0);
+}
+#ifdef WITH_SS5
+static int inter_ss5(struct interface *interface, char *filename, int line, char *parameter, char *value)
+{
+       struct interface_port *ifport;
+       char *element;
+
+       /* port in chain ? */
+       if (!interface->ifport) {
+               SPRINT(interface_error, "Error in %s (line %d): parameter '%s' expects previous 'port' definition.\n", filename, line, parameter);
+               return(-1);
+       }
+       /* goto end of chain */
+       ifport = interface->ifport;
+       while(ifport->next)
+               ifport = ifport->next;
+       ifport->ss5 |= SS5_ENABLE;
+       while((element = strsep(&value, " "))) {
+               if (element[0] == '\0')
+                       continue;
+               if (!strcasecmp(element, "connect"))
+                       ifport->ss5 |= SS5_FEATURE_CONNECT;
+               else
+               if (!strcasecmp(element, "nodisconnect"))
+                       ifport->ss5 |= SS5_FEATURE_NODISCONNECT;
+               else
+               if (!strcasecmp(element, "releaseguardtimer"))
+                       ifport->ss5 |= SS5_FEATURE_RELEASEGUARDTIMER;
+               else
+               if (!strcasecmp(element, "bell"))
+                       ifport->ss5 |= SS5_FEATURE_BELL;
+               else
+               if (!strcasecmp(element, "pulsedialing"))
+                       ifport->ss5 |= SS5_FEATURE_PULSEDIALING;
+               else
+               if (!strcasecmp(element, "delay"))
+                       ifport->ss5 |= SS5_FEATURE_DELAY;
+               else
+               if (!strcasecmp(element, "starrelease"))
+                       ifport->ss5 |= SS5_FEATURE_STAR_RELEASE;
+               else
+               if (!strcasecmp(element, "suppress"))
+                       ifport->ss5 |= SS5_FEATURE_SUPPRESS;
+               else {
+                       SPRINT(interface_error, "Error in %s (line %d): parameter '%s' does not allow value element '%s'.\n", filename, line, parameter, element);
+                       return(-1);
+               }
+       }
+       return(0);
+}
+#endif
 
 
 /*
@@ -907,6 +990,12 @@ static int inter_gsm(struct interface *interface, char *filename, int line, char
 struct interface_param interface_param[] = {
        { "extension", &inter_extension, "",
        "If keyword is given, calls to interface are handled as internal extensions."},
+
+       { "extern", &inter_extern, "",
+       "If keyword is given, this interface will be used for external calls.\n"
+       "Calls require an external interface, if the routing action 'extern' is used\nwithout specific interface given.\n"
+       "Calls forwarded by extension's 'settings' also require an external interface."},
+
        {"tones", &inter_tones, "yes | no",
        "Interface generates tones during call setup and release, or not.\nBy default only NT-mode ports generate tones."},
 
@@ -1024,7 +1113,28 @@ struct interface_param interface_param[] = {
        {"gsm", &inter_gsm, "",
        "Sets up GSM interface for using OpenBSC.\n"
        "This interface must be a loopback interface. The second loopback interface\n"
-       "must be assigned to OpenBSC"},
+       "must be assigned to OpenBSC."},
+
+       {"nonotify", &inter_nonotify, "",
+       "Prevents sending notify messages to this interface. A call placed on hold will\n"
+       "Not affect the remote end (phone or telcom switch).\n"
+       "This parameter must follow a 'port' parameter."},
+
+#ifdef WITH_SS5
+       {"ccitt5", &inter_ss5, "[<feature> [feature ...]]",
+       "Interface uses CCITT No. 5 inband signalling rather than D-channel.\n"
+       "This feature causes CPU load to rise and has no practical intend.\n"
+       "If you don't know what it is, you don't need it.\n"
+       "Features apply to protocol behaviour and blueboxing specials, they are:\n"
+       " connect - Connect incomming call to throughconnect audio, if required.\n"
+       " nodisconnect - Don't disconnect if incomming exchange disconnects.\n"
+       " releaseguardtimer - Tries to prevent Blueboxing by a longer release-guard.\n"
+       " bell - Allow releasing and pulse-dialing via 2600 Hz like old Bell systems.\n"
+       " pulsedialing - Use pulse dialing on outgoing exchange. (takes long!)\n"
+       " delay - Use on incomming exchange, to make you feel a delay when blueboxing.\n"
+       " starrelease - Pulse dialing a star (11 pulses per digit) clears current call.\n"
+       " suppress - Suppress received tones, as they will be recognized."},
+#endif
 
        {NULL, NULL, NULL, NULL}
 };
@@ -1059,9 +1169,7 @@ struct interface *read_interfaces(void)
        }
 
        line=0;
-       while((fgets(buffer,sizeof(buffer),fp))) {
-               buffer[sizeof(buffer)-1]=0;
-               if (buffer[0]) buffer[strlen(buffer)-1]=0;
+       while((GETLINE(buffer, fp))) {
                p=buffer;
                line++;
 
@@ -1252,12 +1360,12 @@ static void set_defaults(struct interface_port *ifport)
        if (ifport->interface->is_tones)
                ifport->mISDNport->tones = (ifport->interface->is_tones==IS_YES);
        else
-               ifport->mISDNport->tones = (ifport->mISDNport->ntmode)?1:0;
+               ifport->mISDNport->tones = (ifport->mISDNport->ntmode || ifport->mISDNport->ss5)?1:0;
        /* default is_earlyb */
        if (ifport->interface->is_earlyb)
                ifport->mISDNport->earlyb = (ifport->interface->is_earlyb==IS_YES);
        else
-               ifport->mISDNport->earlyb = (ifport->mISDNport->ntmode)?0:1;
+               ifport->mISDNport->earlyb = (ifport->mISDNport->ntmode && !ifport->mISDNport->ss5)?0:1;
        /* set locally flag */
        if (ifport->interface->extension)
                ifport->mISDNport->locally = 1;
@@ -1294,7 +1402,7 @@ void relink_interfaces(void)
                                if (!strcmp(mISDNport->name, ifport->portname))
                                        ifport->portnum = mISDNport->portnum; /* same name, so we use same number */
                                if (mISDNport->portnum == ifport->portnum) {
-                                       PDEBUG(DEBUG_ISDN, "Port %d:%s relinking!\n", mISDNport->portnum);
+                                       PDEBUG(DEBUG_ISDN, "Port %d:%s relinking!\n", ifport->portnum, ifport->portname);
                                        ifport->mISDNport = mISDNport;
                                        mISDNport->ifport = ifport;
                                        set_defaults(ifport);
@@ -1343,7 +1451,7 @@ void load_port(struct interface_port *ifport)
        struct mISDNport *mISDNport;
 
        /* open new port */
-       mISDNport = mISDNport_open(ifport->portnum, ifport->portname, ifport->ptp, ifport->nt, ifport->tespecial, ifport->l1hold, ifport->l2hold, ifport->interface, ifport->gsm);
+       mISDNport = mISDNport_open(ifport);
        if (mISDNport) {
                /* link port */
                ifport->mISDNport = mISDNport;
@@ -1353,6 +1461,8 @@ void load_port(struct interface_port *ifport)
                SCPY(ifport->portname, mISDNport->name);
                /* set defaults */
                set_defaults(ifport);
+               /* load static port instances */
+               mISDNport_static(mISDNport);
        } else {
                ifport->block = 2; /* not available */
        }