chiark / gitweb /
Merge branches 'idx/verh' and 'idx/qmqpc'
[qmail] / ipme.c
1 #include <sys/types.h>
2 #include <sys/param.h>
3 #include <sys/time.h>
4 #include <sys/ioctl.h>
5 #include <sys/socket.h>
6 #include <net/if.h>
7 #include <netinet/in.h>
8 #ifndef SIOCGIFCONF /* whatever works */
9 #include <sys/sockio.h>
10 #endif
11 #include "hassalen.h"
12 #include "byte.h"
13 #include "ip.h"
14 #include "ipalloc.h"
15 #include "stralloc.h"
16 #include "ipme.h"
17
18 static int ipmeok = 0;
19 ipalloc ipme = {0};
20
21 int ipme_is(ip)
22 struct ip_address *ip;
23 {
24   int i;
25   if (ipme_init() != 1) return -1;
26   for (i = 0;i < ipme.len;++i)
27     if (byte_equal(&ipme.ix[i].ip,4,ip))
28       return 1;
29   return 0;
30 }
31
32 static stralloc buf = {0};
33
34 int ipme_init()
35 {
36   struct ifconf ifc;
37   char *x;
38   struct ifreq *ifr;
39   struct sockaddr_in *sin;
40   int len;
41   int s;
42   struct ip_mx ix;
43  
44   if (ipmeok) return 1;
45   if (!ipalloc_readyplus(&ipme,0)) return 0;
46   ipme.len = 0;
47   ix.pref = 0;
48  
49   if ((s = socket(AF_INET,SOCK_STREAM,0)) == -1) return -1;
50  
51   len = 256;
52   for (;;) {
53     if (!stralloc_ready(&buf,len)) { close(s); return 0; }
54     buf.len = 0;
55     ifc.ifc_buf = buf.s;
56     ifc.ifc_len = len;
57     if (ioctl(s,SIOCGIFCONF,&ifc) >= 0) /* > is for System V */
58       if (ifc.ifc_len + sizeof(*ifr) + 64 < len) { /* what a stupid interface */
59         buf.len = ifc.ifc_len;
60         break;
61       }
62     if (len > 200000) { close(s); return -1; }
63     len += 100 + (len >> 2);
64   }
65   x = buf.s;
66   while (x < buf.s + buf.len) {
67     ifr = (struct ifreq *) x;
68 #ifdef HASSALEN
69     len = sizeof(ifr->ifr_name) + ifr->ifr_addr.sa_len;
70     if (len < sizeof(*ifr))
71       len = sizeof(*ifr);
72     if (ifr->ifr_addr.sa_family == AF_INET) {
73       sin = (struct sockaddr_in *) &ifr->ifr_addr;
74       byte_copy(&ix.ip,4,&sin->sin_addr);
75       if (ioctl(s,SIOCGIFFLAGS,x) == 0)
76         if (ifr->ifr_flags & IFF_UP)
77           if (!ipalloc_append(&ipme,&ix)) { close(s); return 0; }
78     }
79 #else
80     len = sizeof(*ifr);
81     if (ioctl(s,SIOCGIFFLAGS,x) == 0)
82       if (ifr->ifr_flags & IFF_UP)
83         if (ioctl(s,SIOCGIFADDR,x) == 0)
84           if (ifr->ifr_addr.sa_family == AF_INET) {
85             sin = (struct sockaddr_in *) &ifr->ifr_addr;
86             byte_copy(&ix.ip,4,&sin->sin_addr);
87             if (!ipalloc_append(&ipme,&ix)) { close(s); return 0; }
88           }
89 #endif
90     x += len;
91   }
92   close(s);
93   ipmeok = 1;
94   return 1;
95 }