ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/gvpe/src/sockinfo.C
Revision: 1.15
Committed: Thu Mar 3 16:54:34 2005 UTC (19 years, 2 months ago) by pcg
Content type: text/plain
Branch: MAIN
Changes since 1.14: +5 -3 lines
Log Message:
*** empty log message ***

File Contents

# Content
1 /*
2 sockinfo.C -- socket address management
3 Copyright (C) 2003 Marc Lehmann <gvpe@schmorp.de>
4
5 This file is part of GVPE.
6
7 GVPE is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 2 of the License, or
10 (at your option) any later version.
11
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with gvpe; if not, write to the Free Software
19 Foundation, Inc. 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 */
21
22 #include "config.h"
23
24 #include <netdb.h>
25
26 #include "gettext.h"
27
28 #include "sockinfo.h"
29 #include "slog.h"
30
31 // all ipv4-based protocols
32 #define PROTv4 (PROT_UDPv4 | PROT_TCPv4 | PROT_ICMPv4 | PROT_IPv4 | PROT_DNSv4)
33
34 void sockinfo::set (const sockaddr_in *sa, u8 prot_)
35 {
36 host = sa->sin_addr.s_addr;
37 port = prot_ & (PROT_IPv4 | PROT_ICMPv4) ? 0 : sa->sin_port;
38 prot = prot_;
39 }
40
41 void sockinfo::set (const char *hostname, u16 port_, u8 prot_)
42 {
43 prot = prot_;
44 host = 0;
45 port = htons (port_);
46
47 if (prot & PROTv4
48 && hostname)
49 {
50 struct hostent *he = gethostbyname (hostname);
51
52 if (he
53 && he->h_addrtype == AF_INET && he->h_length == 4 && he->h_addr_list[0])
54 {
55 //sa->sin_family = he->h_addrtype;
56 memcpy (&host, he->h_addr_list[0], 4);
57 }
58 else
59 slog (L_NOTICE, _("unable to resolve host '%s'"), hostname);
60 }
61 }
62
63 void
64 sockinfo::set (const conf_node *conf, u8 prot_)
65 {
66 if (prot_ == PROT_DNSv4)
67 set (conf->dns_hostname, conf->dns_hostname ? conf->dns_port : 0, prot_);
68 else
69 set (conf->hostname,
70 prot_ == PROT_UDPv4 ? conf->udp_port
71 : prot_ == PROT_TCPv4 ? conf->tcp_port
72 : prot_ == PROT_DNSv4 ? conf->dns_port
73 : 0,
74 prot_);
75 }
76
77 const sockaddr *
78 sockinfo::sav4() const
79 {
80 static sockaddr_in sa;
81
82 sa.sin_family = AF_INET;
83 sa.sin_port = port;
84 sa.sin_addr.s_addr = host;
85
86 return (const sockaddr *)&sa;
87 }
88
89 static char hostport[10 + 15 + 1 + 5 + 1]; // proto / IPv4 : port
90
91 const char *
92 sockinfo::ntoa () const
93 {
94 in_addr ia = { host };
95
96 sprintf (hostport, "%.15s", inet_ntoa (ia));
97
98 return hostport;
99 }
100
101 sockinfo::operator const char *() const
102 {
103 in_addr ia = { host };
104
105 sprintf (hostport, "%s/%.15s:%d", strprotocol (prot), inet_ntoa (ia), ntohs (port) & 0xffff);
106
107 return hostport;
108 }
109
110 u8
111 sockinfo::supported_protocols (conf_node *conf)
112 {
113 u8 protocols = prot;
114
115 if (prot & (PROT_UDPv4 | PROT_TCPv4))
116 protocols |= PROT_IPv4 | PROT_ICMPv4;
117
118 if (conf
119 && prot & PROTv4
120 && conf->protocols & PROT_UDPv4
121 && conf->udp_port)
122 protocols |= PROT_UDPv4;
123
124 if (conf
125 && prot & PROTv4
126 && conf->protocols & PROT_TCPv4
127 && conf->tcp_port)
128 protocols |= PROT_TCPv4;
129
130 if (conf
131 && prot & PROTv4
132 && conf->protocols & PROT_DNSv4
133 && conf->dns_port)
134 protocols |= PROT_DNSv4;
135
136 return protocols;
137 }
138
139 bool
140 sockinfo::upgrade_protocol (u8 prot_, conf_node *conf)
141 {
142 if (prot_ == prot)
143 return true;
144
145 if (prot & PROTv4
146 && prot_ & PROTv4)
147 {
148 if (prot_ & (PROT_IPv4 | PROT_ICMPv4))
149 {
150 prot = prot_;
151 port = 0;
152 return true;
153 }
154
155 if (conf
156 && prot_ & PROT_UDPv4
157 && conf->protocols & PROT_UDPv4
158 && conf->udp_port)
159 {
160 prot = prot_;
161 port = htons (conf->udp_port);
162 return true;
163 }
164
165 if (conf
166 && prot_ & PROT_TCPv4
167 && conf->protocols & PROT_TCPv4
168 && conf->tcp_port)
169 {
170 prot = prot_;
171 port = htons (conf->tcp_port);
172 return true;
173 }
174
175 #if ENABLE_DNS
176 if (conf
177 && prot_ & PROT_DNSv4
178 && conf->protocols & PROT_DNSv4
179 && conf->dns_port)
180 {
181 set (::conf.dns_forw_host, ::conf.dns_forw_port, prot_);
182 return true;
183 }
184 #endif
185 }
186
187 return false;
188 }
189
190 bool
191 operator == (const sockinfo &a, const sockinfo &b)
192 {
193 return a.host == b.host && a.port == b.port && a.prot == b.prot;
194 }
195
196 bool
197 operator < (const sockinfo &a, const sockinfo &b)
198 {
199 return a.host < b.host
200 || (a.host == b.host && (a.port < b.port
201 || (a.port == b.port && a.prot < b.prot)));
202 }
203