/* * lib/af.c This file contains the top-level part of the protocol * support functions module for the NET-2 base distribution. * * Version: $Id: af.c,v 1.14 2007/12/01 17:49:35 ecki Exp $ * * Author: Fred N. van Kempen, * Copyright 1993 MicroWalt Corporation * * This program is free software; you can redistribute it * and/or modify it under the terms of the GNU General * Public License as published by the Free Software * Foundation; either version 2 of the License, or (at * your option) any later version. */ #include #include #include #include #include #include #include #include #include "config.h" #include "net-support.h" #include "pathnames.h" #include "intl.h" #include "util.h" int flag_unx; int flag_ipx; int flag_ax25; int flag_ddp; int flag_netrom; int flag_inet; int flag_inet6; int flag_econet; int flag_rose; int flag_x25 = 0; int flag_ash; int flag_bluetooth; static const struct aftrans_t { char *alias; char *name; int *flag; } aftrans[] = { { "ax25", "ax25", &flag_ax25 }, { "ip", "inet", &flag_inet }, { "ip6", "inet6", &flag_inet6 }, { "ipx", "ipx", &flag_ipx }, { "rose", "rose", &flag_rose }, { "appletalk", "ddp", &flag_ddp }, { "netrom", "netrom", &flag_netrom }, { "inet", "inet", &flag_inet }, { "inet6", "inet6", &flag_inet6 }, { "ddp", "ddp", &flag_ddp }, { "unix", "unix", &flag_unx }, { "tcpip", "inet", &flag_inet }, { "econet", "ec", &flag_econet }, { "x25", "x25", &flag_x25 }, { "ash", "ash", &flag_ash }, { "bluetooth", "bluetooth", &flag_bluetooth }, { 0, 0, 0 } }; char afname[256] = ""; extern struct aftype unspec_aftype; extern struct aftype unix_aftype; extern struct aftype inet_aftype; extern struct aftype inet6_aftype; extern struct aftype ax25_aftype; extern struct aftype netrom_aftype; extern struct aftype ipx_aftype; extern struct aftype ddp_aftype; extern struct aftype ec_aftype; extern struct aftype x25_aftype; extern struct aftype rose_aftype; extern struct aftype ash_aftype; static short sVafinit = 0; struct aftype * const aftypes[] = { #if HAVE_AFUNIX &unix_aftype, #endif #if HAVE_AFINET &inet_aftype, #endif #if HAVE_AFINET6 &inet6_aftype, #endif #if HAVE_AFAX25 &ax25_aftype, #endif #if HAVE_AFNETROM &netrom_aftype, #endif #if HAVE_AFROSE &rose_aftype, #endif #if HAVE_AFIPX &ipx_aftype, #endif #if HAVE_AFATALK &ddp_aftype, #endif #if HAVE_AFECONET &ec_aftype, #endif #if HAVE_AFASH &ash_aftype, #endif #if HAVE_AFX25 &x25_aftype, #endif &unspec_aftype, NULL }; static void afinit(void) { unspec_aftype.title = _("UNSPEC"); #if HAVE_AFUNIX unix_aftype.title = _("UNIX Domain"); #endif #if HAVE_AFINET inet_aftype.title = _("DARPA Internet"); #endif #if HAVE_AFINET6 inet6_aftype.title = _("IPv6"); #endif #if HAVE_AFAX25 ax25_aftype.title = _("AMPR AX.25"); #endif #if HAVE_AFNETROM netrom_aftype.title = _("AMPR NET/ROM"); #endif #if HAVE_AFIPX ipx_aftype.title = _("Novell IPX"); #endif #if HAVE_AFATALK ddp_aftype.title = _("Appletalk DDP"); #endif #if HAVE_AFECONET ec_aftype.title = _("Econet"); #endif #if HAVE_AFX25 x25_aftype.title = _("CCITT X.25"); #endif #if HAVE_AFROSE rose_aftype.title = _("AMPR ROSE"); #endif #if HAVE_AFASH ash_aftype.title = _("Ash"); #endif sVafinit = 1; } /* set the default AF list from the program name or a constant value */ void aftrans_def(char *tool, char *argv0, char *dflt) { char *tmp; char *buf; safe_strncpy(afname, dflt, sizeof(afname)); if (!(tmp = strrchr(argv0, '/'))) tmp = argv0; /* no slash?! */ else tmp++; buf = xstrdup(tmp); if (strlen(tool) >= strlen(tmp)) { free(buf); return; } tmp = buf + (strlen(tmp) - strlen(tool)); if (strcmp(tmp, tool) != 0) { free(buf); return; } *tmp = '\0'; if ((tmp = strchr(buf, '_'))) *tmp = '\0'; afname[0] = '\0'; if (aftrans_opt(buf)) safe_strncpy(afname, buf, sizeof(afname)); free(buf); } /* Check our protocol family table for this family. */ const struct aftype *get_aftype(const char *name) { struct aftype * const *afp; if (!sVafinit) afinit(); afp = aftypes; while (*afp != NULL) { if (!strcmp((*afp)->name, name)) return (*afp); afp++; } if (index(name, ',')) fprintf(stderr, _("Please don't supply more than one address family.\n")); return (NULL); } /* Check our protocol family table for this family. */ const struct aftype *get_afntype(int af) { struct aftype * const *afp; if (!sVafinit) afinit(); afp = aftypes; while (*afp != NULL) { if ((*afp)->af == af) return (*afp); afp++; } return (NULL); } /* Check our protocol family table for this family and return its socket */ int get_socket_for_af(int af) { const struct aftype *afp = get_afntype(af); return afp ? afp->fd : -1; } int aftrans_opt(const char *arg) { const struct aftrans_t *paft; char *tmp1, *tmp2; char buf[256]; safe_strncpy(buf, arg, sizeof(buf)); tmp1 = buf; while (tmp1) { tmp2 = index(tmp1, ','); if (tmp2) *(tmp2++) = '\0'; for (paft = aftrans; paft->alias; paft++) { if (strcmp(tmp1, paft->alias)) continue; if (strlen(paft->name) + strlen(afname) + 1 >= sizeof(afname)) { fprintf(stderr, _("Too much address family arguments.\n")); return (0); } if (paft->flag) (*paft->flag)++; if (afname[0]) strcat(afname, ","); strcat(afname, paft->name); break; } if (!paft->alias) { fprintf(stderr, _("Unknown address family `%s'.\n"), tmp1); return (1); } tmp1 = tmp2; } return (0); } /* type: 0=all, 1=getroute */ void print_aflist(int type) { int count = 0; const char * txt; struct aftype * const *afp; if (!sVafinit) afinit(); afp = aftypes; while (*afp != NULL) { if ((type == 1 && ((*afp)->rprint == NULL)) || ((*afp)->af == 0)) { afp++; continue; } if ((count % 3) == 0) fprintf(stderr,count?"\n ":" "); txt = (*afp)->name; if (!txt) txt = ".."; fprintf(stderr,"%s (%s) ",txt,(*afp)->title); count++; afp++; } fprintf(stderr,"\n"); }