home *** CD-ROM | disk | FTP | other *** search
/ Beijing Paradise BBS Backup / PARADISE.ISO / software / BBSDOORW / UUPC11XS.ZIP / LIB / HOSTABLE.C < prev    next >
C/C++ Source or Header  |  1992-12-18  |  26KB  |  661 lines

  1. /*
  2.    h o s t a b l e . c
  3.  
  4.    Remote host table routines for UUPC/extended
  5.  
  6.    Copyright (C) 1989, 1990 by Andrew H. Derbyshire
  7.  
  8.    See file README.SCR for restrictions on re-distribution.
  9.  
  10.    History:
  11.  
  12.       18 Mar 1990 Create hostable.c from router.c                    ahd
  13.                   Move code to generate localdomain to here          ahd
  14.       22 Apr 90   Perform check for full host name before examining
  15.                   name without domain.                               ahd
  16.       29 Jul 90   Only load host table based on first six characters
  17.                   of host name.                                      ahd
  18.  */
  19.  
  20.  /*
  21.   *      $Id: HOSTABLE.C 1.3 1992/12/18 12:05:57 ahd Exp $
  22.   *
  23.   *      $Log: HOSTABLE.C $
  24.  * Revision 1.3  1992/12/18  12:05:57  ahd
  25.  * Suppress duplicate machine state messages to improving OS/2 scrolling
  26.  *
  27.  * Revision 1.2  1992/11/22  20:58:55  ahd
  28.  * Use strpool to allocate const strings
  29.  *
  30.   */
  31.  
  32. #include <ctype.h>
  33. #include <limits.h>
  34. #include <stdio.h>
  35. #include <stdlib.h>
  36. #include <string.h>
  37. #include <sys/types.h>
  38.  
  39. #include "lib.h"
  40. #include "hlib.h"
  41. #include "hostable.h"
  42. #include "security.h"
  43.  
  44. currentfile();
  45.  
  46. static struct HostTable *hosts = NULL;
  47.  
  48. static size_t  HostElements = 0;
  49.  
  50. static size_t loadhost( void );
  51.  
  52. static int hostcmp( const void *a , const void *b );
  53.  
  54. static size_t localdomainl;   /* Length of localdomain               */
  55.  
  56. /*--------------------------------------------------------------------*/
  57. /*    c h e c k n a m e                                               */
  58. /*                                                                    */
  59. /*    Perform a search for a single host name                         */
  60. /*                                                                    */
  61. /*    Rewritten for release 1.10a.  The old release had               */
  62. /*    most of the same logic, but nested it inside the search         */
  63. /*    loop; adding new cases (specifically, the wildcard domain       */
  64. /*    search) was difficult.  This version is slower because it       */
  65. /*    makes multiple passes through the host table, but this          */
  66. /*    isn't really performance code for a small (under 100 hosts)     */
  67. /*    table.                                 ahd 26 April 1991        */
  68. /*                                                                    */
  69. /*    Note because we save the arguments and use a static variable    */
  70. /*    to save the result of searches, this function is not            */
  71. /*    recursive!                                                      */
  72. /*--------------------------------------------------------------------*/
  73.  
  74. struct HostTable *checkname(const char *name)
  75. {
  76.    char  hostname[MAXADDR];   /* Local copy of name to process       */
  77.    char *period;              /* Pointer "." in hostname             */
  78.    size_t namel;              /* Length of the name input            */
  79.    size_t column;             /* Length of the name input            */
  80.  
  81.    static char savename[MAXADDR] = "";
  82.                               /* Saved copy of name to make function
  83.                                  reducible                           */
  84.    static struct HostTable *hostz;
  85.  
  86. /*--------------------------------------------------------------------*/
  87. /*                       Validate the argument                        */
  88. /*--------------------------------------------------------------------*/
  89.  
  90.    if ((name == NULL) || ((namel = strlen(name)) == 0))
  91.    {
  92.       printmsg(0,"checkname: Invalid (missing) hostname passed");
  93.       panic();
  94.       return NULL;           /* Never executed                      */
  95.    }
  96.  
  97. /*--------------------------------------------------------------------*/
  98. /*    If same argument as last time, return same result; otherwise    */
  99. /*    save input for next pass                                        */
  100. /*--------------------------------------------------------------------*/
  101.  
  102.    if (equal(name, savename))
  103.       return hostz;
  104.    strcpy( savename, name);   /* Save for next pass                  */
  105.  
  106. /*--------------------------------------------------------------------*/
  107. /*                      Search for the full name                      */
  108. /*--------------------------------------------------------------------*/
  109.  
  110.    if ((hostz = searchname(name, MAXADDR)) != BADHOST)
  111.       return hostz;
  112.  
  113. /*--------------------------------------------------------------------*/
  114. /*    If the name already has the local domain attached, search for   */
  115. /*    the host name without the domain.                               */
  116. /*--------------------------------------------------------------------*/
  117.  
  118.    column = namel - localdomainl;
  119.    if ((namel > localdomainl) && equal(E_localdomain, &name[column]) &&
  120.        (name[ column - 1] == '.'))
  121.    {
  122.       if ((hostz = searchname(name,column-1 )) != BADHOST)
  123.          return hostz;
  124.    } /* if */
  125.  
  126. /*--------------------------------------------------------------------*/
  127. /*              Search for the name in the local domain               */
  128. /*--------------------------------------------------------------------*/
  129.  
  130.    if ((namel + localdomainl + 2) < MAXADDR)
  131.    {
  132.       sprintf(hostname,"%s.%s",name,E_localdomain);
  133.       if ((hostz = searchname(hostname, MAXADDR)) != BADHOST)
  134.          return hostz;
  135.    } /* if */
  136.  
  137. /*--------------------------------------------------------------------*/
  138. /*    If a simple name and not found, return search for truncated     */
  139. /*    UNIX name.                                                      */
  140. /*--------------------------------------------------------------------*/
  141.  
  142.    if ( strchr(name,'.') == NULL )
  143.       return checkreal( name );
  144.  
  145. /*--------------------------------------------------------------------*/
  146. /*               Perform a wildcard domain name search                */
  147. /*--------------------------------------------------------------------*/
  148.  
  149.    period = (char *) name;    /* Begin at front of name              */
  150.    while( period != NULL )
  151.    {
  152.       sprintf( hostname,(*period == '.') ? "*%s" : "*.%s",period);
  153.                               /* We add the missing period for the
  154.                                  first pass through the loop         */
  155.       if ((hostz = searchname(hostname, MAXADDR)) != BADHOST)
  156.          return hostz;
  157.       period = strchr(++period,'.');   /* Not found, search for next
  158.                                           higher domain              */
  159.    }
  160.  
  161. /*--------------------------------------------------------------------*/
  162. /*         We didn't find the host.  Return failure to caller         */
  163. /*--------------------------------------------------------------------*/
  164.  
  165.    return BADHOST;
  166.  
  167. }  /* checkname */
  168.  
  169. /*--------------------------------------------------------------------*/
  170. /*    c h e c k r e a l                                               */
  171. /*                                                                    */
  172. /*    Perform a search for a real (connected) simple host name        */
  173. /*--------------------------------------------------------------------*/
  174.  
  175. struct HostTable *checkreal(const char *name)
  176. {
  177.    size_t  namel = max( strlen(name), HOSTLEN);
  178.    struct HostTable *hostp = searchname( name, namel );
  179.  
  180. /*--------------------------------------------------------------------*/
  181. /*             If we didn't find the host, return failure             */
  182. /*--------------------------------------------------------------------*/
  183.  
  184.    if ((hostp == BADHOST) || (hostp->hstatus >= nocall))
  185.       return hostp;           /* Return raw information              */
  186.    else
  187.       return BADHOST;         /* Not a real host, invalid for our
  188.                                  purposes                            */
  189.  
  190. } /* searchreal */
  191.  
  192. /*--------------------------------------------------------------------*/
  193. /*    s e a r c h n a m e                                             */
  194. /*                                                                    */
  195. /*    Look up a system name in our systems (L.sys) file.              */
  196. /*    Only the first 7 characters of a system name is significant.    */
  197. /*--------------------------------------------------------------------*/
  198.  
  199. struct HostTable *searchname(const char *name, const size_t namel)
  200. {
  201.    int   lower;
  202.    int   upper;
  203.  
  204.  /*------------------------------------------------------------------*/
  205.  /*             Initialize the host name table if needed             */
  206.  /*------------------------------------------------------------------*/
  207.  
  208.    if (HostElements == 0)           /* host table initialized yet?   */
  209.       HostElements = loadhost();        /* No --> load it            */
  210.  
  211.    lower = 0;
  212.    upper = HostElements - 1;
  213.    while ( lower <= upper )
  214.    {
  215.       int midpoint = (lower + upper) / 2;
  216.       int hit;
  217.  
  218. /*--------------------------------------------------------------------*/
  219. /*    Compare for up to the specified length of the host name, but    */
  220. /*    never less than the length of the item we are comparing it      */
  221. /*    to.  In other words, the search key can be shorter than the     */
  222. /*    table entry for a simple host name.                             */
  223. /*                                                                    */
  224. /*    This mostly affects simple host names, as domain names          */
  225. /*    have a Very Large Number (VLN) passed into to insure we         */
  226. /*    compare the entire length of the name.                          */
  227. /*--------------------------------------------------------------------*/
  228.  
  229.       hit = strnicmp(name,hosts[midpoint].hostname,namel);
  230.       if (hit > 0)
  231.          lower = midpoint + 1;
  232.       else if ((hit < 0) || (strlen(hosts[midpoint].hostname) > namel))
  233.          upper = midpoint - 1;
  234.       else {
  235.             printmsg(8,"searchname: Looking for \"%s\" of length %d,\
  236.  found \"%s\"",
  237.             name, namel, hosts[midpoint].hostname);
  238.          return &hosts[midpoint];
  239.       }
  240.    }
  241.  
  242. /*--------------------------------------------------------------------*/
  243. /*         We didn't find the host.  Return failure to caller         */
  244. /*--------------------------------------------------------------------*/
  245.  
  246.    printmsg(8,"searchname: Looking for \"%s\", did not find it",
  247.             name);
  248.    return BADHOST;
  249.  
  250. }  /* searchname */
  251.  
  252.  
  253. /*--------------------------------------------------------------------*/
  254. /*    n e x t h o s t                                                 */
  255. /*                                                                    */
  256. /*    Returns next host in table with requested attribute             */
  257. /*--------------------------------------------------------------------*/
  258.  
  259. struct HostTable *nexthost( const boolean start )
  260. {
  261.    static size_t current = 0;
  262.  
  263.    if (HostElements == 0)     /* host table initialized yet?         */
  264.       HostElements = loadhost(); /* No --> load it                   */
  265.  
  266.    if (start)
  267.       current = 0;
  268.    else
  269.       current ++;
  270.  
  271.    while ( current < HostElements )
  272.    {
  273.       if (hosts[current].hstatus >= nocall)
  274.          return &hosts[current];
  275.       else
  276.          current++;
  277.    }
  278.  
  279.    return BADHOST;
  280.  
  281. }  /* nexthost */
  282.  
  283.  
  284. /*--------------------------------------------------------------------*/
  285. /*    i n i t h o s t                                                 */
  286. /*                                                                    */
  287. /*    Intializes a host table entry for for loadhost                  */
  288. /*--------------------------------------------------------------------*/
  289.  
  290. struct HostTable *inithost(char *name)
  291. {
  292.  
  293.    size_t hit = HostElements;
  294.    size_t element = 0;
  295.    static size_t max_elements = 32; /* This is automatically
  296.                                        raised if we run out of room  */
  297.  
  298.    if (hosts == NULL)
  299.    {
  300.       hosts = calloc(max_elements, sizeof(*hosts));
  301.       printmsg(5,"inithost: Allocated room for %d host entries",
  302.                max_elements);
  303.    }
  304.    else if ( max_elements == HostElements )
  305.    {
  306.       max_elements = max_elements * 2;
  307.       hosts = realloc(hosts , max_elements * sizeof(*hosts));
  308.       printmsg(5,"inithost: reallocated room for %d host entries",
  309.                max_elements);
  310.    }
  311.    checkref(hosts);
  312.  
  313. /*--------------------------------------------------------------------*/
  314. /*    Add the host to the table.  Note that we must add the host      */
  315. /*    to the table ourselves (rather than use lsearch) because we     */
  316. /*    must make a copy of the string; the *token we use for the       */
  317. /*    search is in the middle of our I/O buffer!                      */
  318. /*--------------------------------------------------------------------*/
  319.  
  320.    while ( element < hit )
  321.    {
  322.       if (equali( hosts[element].hostname , name ))
  323.          hit = element;
  324.       else
  325.          element++;
  326.    }
  327.  
  328. /*--------------------------------------------------------------------*/
  329. /*               If a new element, initialize the block               */
  330. /*--------------------------------------------------------------------*/
  331.  
  332.    if (hit == HostElements)
  333.    {
  334.       memset( &hosts[hit] , 0, sizeof hosts[hit] );
  335.       hosts[hit].hostname = newstr(name);
  336.       checkref( hosts[hit].hostname );
  337.       hosts[hit].anylogin = TRUE;   /* Allow generic login by default   */
  338.       HostElements ++ ;
  339.    } /* if */
  340.  
  341.    return &hosts[hit];
  342.  
  343. } /* inithost */
  344.  
  345. /*--------------------------------------------------------------------*/
  346. /*    l o a d h o s t                                                 */
  347. /*                                                                    */
  348. /*    Initializes table of known host names for checkname             */
  349. /*--------------------------------------------------------------------*/
  350.  
  351. static size_t loadhost()
  352. {
  353.    FILE *ff;
  354.    char buf[BUFSIZ];
  355.    char *token;
  356.    char s_systems[FILENAME_MAX]; /* full-name of systems file        */
  357.    size_t hit;
  358.  
  359.    struct HostTable *hostp;
  360.  
  361. /*--------------------------------------------------------------------*/
  362. /*                      Validate the domain name                      */
  363. /*--------------------------------------------------------------------*/
  364.  
  365.    token = strrchr(E_domain,'.');
  366.  
  367.    if (token == NULL)
  368.    {
  369.       printmsg(0,"Domain name \"%s\" is invalid, missing period",E_domain);
  370.       panic();
  371.    }
  372.  
  373. /*--------------------------------------------------------------------*/
  374. /*                  Load the local host information                   */
  375. /*--------------------------------------------------------------------*/
  376.  
  377.    hostp = inithost(E_nodename);
  378.    hostp->hstatus  = localhost;
  379.    hostp->realname = E_nodename; /* Don't let user alias our system
  380.                                     name                             */
  381.  
  382. /*--------------------------------------------------------------------*/
  383. /*                Now do the local domain information                 */
  384. /*--------------------------------------------------------------------*/
  385.  
  386.    hostp = inithost(E_domain);
  387.  
  388.    if (hostp->via == NULL )   /* Not initialized?                    */
  389.       hostp->via      = E_nodename;  /* Correct --> Route via local  */
  390.    else
  391.       panic();                /* "Houston, we a have problem" -
  392.                                  Apollo 13                           */
  393.  
  394.    hostp->realname = E_nodename;
  395.  
  396. /*--------------------------------------------------------------------*/
  397. /*    If we allow anonymous UUCP, load the dummy host we use for      */
  398. /*    connections to such hosts                                       */
  399. /*--------------------------------------------------------------------*/
  400.  
  401.    if ( E_anonymous != NULL )
  402.    {
  403.       hostp = inithost( ANONYMOUS_HOST );
  404.       hostp->hstatus = nocall;
  405.       hostp->via     = E_nodename;
  406.       hostp->hstats  = malloc( sizeof *(hostp->hstats) );
  407.       checkref( hostp->hstats );
  408.       memset( hostp->hstats, 0, sizeof *(hostp->hstats) );
  409.    } /* if */
  410.  
  411. /*--------------------------------------------------------------------*/
  412. /*                  Load names from the systems file                  */
  413. /*--------------------------------------------------------------------*/
  414.  
  415.    mkfilename(s_systems, E_confdir, SYSTEMS);
  416.  
  417.    ff = FOPEN(s_systems, "r", TEXT);
  418.    if (ff == NULL)
  419.    {
  420.       printerr(s_systems);
  421.       panic();
  422.    }
  423.  
  424.    while (! feof(ff))
  425.    {
  426.       if (fgets(buf,BUFSIZ,ff) == NULL)   /* Try to read a line      */
  427.          break;                  /* Exit if end of file              */
  428.       token = strtok(buf,WHITESPACE);
  429.       if (token == NULL)         /* Any data?                        */
  430.          continue;               /* No --> read another line         */
  431.       if (token[0] == '#')
  432.          continue;                  /* Line is a comment; loop again */
  433.       hostp = inithost(token);
  434.  
  435.       if (hostp->hstatus == phantom)
  436.       {
  437.          hostp->hstatus = nocall;
  438.          hostp->hstats  = malloc( sizeof *(hostp->hstats) );
  439.          checkref( hostp->hstats );
  440.          memset( hostp->hstats, 0, sizeof *(hostp->hstats) );
  441.       }
  442.    } /* while */
  443.  
  444.    fclose(ff);
  445.  
  446. /*--------------------------------------------------------------------*/
  447. /*               Now the load the routing file, if any.               */
  448. /*--------------------------------------------------------------------*/
  449.  
  450.    mkfilename(s_systems, E_confdir, PATHS);
  451.  
  452.    if ((ff = FOPEN(s_systems, "r", TEXT)) != NULL)
  453.    {
  454.  
  455.       while (! feof(ff))
  456.       {
  457.          boolean freeit = FALSE;
  458.  
  459.          if (fgets(buf,BUFSIZ,ff) == NULL)   /* Try to read a line      */
  460.             break;                  /* Exit if end of file              */
  461.          token = strtok(buf,WHITESPACE);
  462.  
  463.          if (token == NULL)         /* Any data?                        */
  464.             continue;               /* No --> read another line         */
  465.  
  466.          if (*token == '#')
  467.             continue;               /* Line is a comment; loop again    */
  468.  
  469.          hostp = inithost(token);
  470.          token = strtok(NULL,WHITESPACE);
  471.  
  472.          if ( token == NULL )
  473.          {
  474.             printmsg(0,"loadhost: Missing path name for host \"%s\"",
  475.                         hostp->hostname);
  476.             freeit = TRUE;
  477.          }
  478. /*--------------------------------------------------------------------*/
  479. /*                              Gate way                              */
  480. /*--------------------------------------------------------------------*/
  481.          else if (equal(token,"|"))
  482.          {
  483.             token = strtok(NULL,"\n");
  484.  
  485.             if (( hostp->via != NULL ) || ( token == NULL ))
  486.                freeit = TRUE;
  487.             else {
  488.                hostp->hstatus = gatewayed;
  489.  
  490.                while(isspace( *token ))   /* Drop leading white space only */
  491.                   token++;
  492.  
  493.                if (*token == '\0')        /* Empty string?           */
  494.                   freeit = TRUE;          /* Yes --> Flag for error  */
  495.                else
  496.                   hostp->via = token = newstr(token);
  497.             } /* else if */
  498.  
  499.             if ( freeit )
  500.                printmsg(0,"loadhost: Invalid/duplicate gateway for \"%s\"",
  501.                      hostp->hostname );
  502.  
  503.          } /* else if */
  504. /*--------------------------------------------------------------------*/
  505. /*                               Alias                                */
  506. /*--------------------------------------------------------------------*/
  507.          else if (equal(token,"="))
  508.          {
  509.             token = strtok(NULL,WHITESPACE);
  510.  
  511.             if (( hostp->realname == NULL ) && (token != NULL))
  512.                hostp->realname = token = newstr( token );
  513.             else {
  514.                printmsg(0,"loadhost: Invalid/duplicate alias of \"%s\"",
  515.                      hostp->hostname );
  516.                freeit = TRUE;
  517.             } /* else */
  518.          } /* else if (equal(token,"=")) */
  519. /*--------------------------------------------------------------------*/
  520. /*                           Routing entry                            */
  521. /*--------------------------------------------------------------------*/
  522.          else {
  523.  
  524.             if ( hostp->via == NULL )
  525.                hostp->via = token = newstr( token );
  526.             else {
  527.                printmsg(0,"loadhost: Invalid/duplicate route for \"%s\"",
  528.                      hostp->hostname );
  529.                freeit = TRUE;
  530.             } /* else */
  531.  
  532.          } /* else */
  533.  
  534.  
  535.          if ( ! freeit )
  536.          {
  537.             checkref( token );
  538.  
  539.             if (*token == '*')       /* Wildcard on right side?    */
  540.             {
  541.                printmsg(0,
  542.      "loadhost: Wildcard \"%s\" not allowed for real name of host \"%s\"",
  543.          token, hostp->hostname);
  544.                freeit = TRUE;
  545.             } /* if (*token == '*') */
  546.          } /* if ( ! freeit ) */
  547.  
  548.          if ( freeit )
  549.          {
  550.             if ( hostp->hstatus == phantom )
  551.                HostElements--;            /* Ignore the routing entry   */
  552.          }
  553.       }  /* end while */
  554.  
  555.       fclose(ff);
  556.    }
  557.    else {
  558.       if ( debuglevel > 2 )
  559.          perror( s_systems );
  560.    }
  561.  
  562. /*--------------------------------------------------------------------*/
  563. /*                   Provide default for fromdomain                   */
  564. /*--------------------------------------------------------------------*/
  565.  
  566.    if (E_fdomain != NULL)     /* If fromdomain provided ...          */
  567.    {
  568.       hostp = inithost(E_fdomain);
  569.  
  570.       if (hostp->via == NULL)    /* Uninitialized?                   */
  571.          hostp->via = E_mailserv;   /* Yes --> Use default route     */
  572.    }
  573.    else
  574.       E_fdomain = E_domain;   /* Use domain as fromdomain            */
  575.  
  576. /*--------------------------------------------------------------------*/
  577. /*    Shrink the table to whatever we actually need, then sort it     */
  578. /*--------------------------------------------------------------------*/
  579.  
  580.    hosts = realloc(hosts, HostElements *  sizeof(*hosts));
  581.    checkref(hosts);
  582.  
  583.    qsort(hosts, HostElements ,sizeof(hosts[0]) , hostcmp);
  584.  
  585. /*--------------------------------------------------------------------*/
  586. /*    If the user did not define a local domain, then generate one    */
  587. /*    based on our own domain name; the generated name will either    */
  588. /*    be of the format ".a.b.c" (incuding the trailing period) or     */
  589. /*    a null string.                                                  */
  590. /*--------------------------------------------------------------------*/
  591.  
  592.    if ( E_localdomain == NULL )
  593.    {
  594.       E_localdomain = strchr(E_domain,'.');
  595.  
  596.       if (E_localdomain == NULL)
  597.          E_localdomain = "UUCP";
  598.       else {
  599.          E_localdomain ++;    /* Step past the period                */
  600.  
  601.          if ( !equal(E_localdomain, "UUCP" ) &&
  602.               (strchr( E_localdomain, '.' ) == NULL ))
  603.                               /* Implied single level domain name?   */
  604.             E_localdomain = E_domain;
  605.                               /* Impossible, use both parts of name  */
  606.       } /* else */
  607.  
  608.       printmsg(3,"loadhost: local domain defined as \"%s\"",
  609.                  E_localdomain);
  610.  
  611.    } /* if */
  612.  
  613.    localdomainl = strlen(E_localdomain);
  614.  
  615. /*--------------------------------------------------------------------*/
  616. /*    Amend for the sin of typing the domain wrong in the sample      */
  617. /*    files for the old releases of UUPC/extended.                    */
  618. /*--------------------------------------------------------------------*/
  619.  
  620.    if (equal(E_localdomain,"UUPC"))
  621.    {
  622.       printmsg(0,"inithost: UUPC is an invalid domain name! "
  623.                  "Change it to UUCP");
  624.       panic();
  625.    }
  626.  
  627. /*--------------------------------------------------------------------*/
  628. /*                      Display the final table                       */
  629. /*--------------------------------------------------------------------*/
  630.  
  631.    for (hit = 0; hit < HostElements; hit++)
  632.    {
  633.       printmsg(8,"loadhost: entry[%02d] %-20s\tvia %s\talias %s",
  634.                   hit,
  635.                   hosts[hit].hostname,
  636.                   (hosts[hit].via == NULL) ? "(self)" : hosts[hit].via,
  637.                   (hosts[hit].realname == NULL)
  638.                                     ? "(self)" : hosts[hit].realname);
  639.    } /* for */
  640.  
  641. /*--------------------------------------------------------------------*/
  642. /*                          Return to caller                          */
  643. /*--------------------------------------------------------------------*/
  644.  
  645.    return (HostElements) ;
  646. } /*loadhost*/
  647.  
  648.  
  649. /*--------------------------------------------------------------------*/
  650. /*    h o s t c m p                                                   */
  651. /*                                                                    */
  652. /*    Accepts indirect pointers to two strings and compares           */
  653. /*    them using stricmp (case insensitive string compare)            */
  654. /*--------------------------------------------------------------------*/
  655.  
  656. int hostcmp( const void *a , const void *b )
  657. {
  658.    return stricmp(((struct HostTable*) a)->hostname,
  659.         ((struct HostTable*) b)->hostname);
  660. }  /*hostcmp*/
  661.