ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/gvpe/src/sockinfo.C
Revision: 1.29
Committed: Tue Dec 4 13:49:16 2012 UTC (11 years, 5 months ago) by root
Content type: text/plain
Branch: MAIN
Changes since 1.28: +16 -3 lines
Log Message:
*** empty log message ***

File Contents

# Content
1 /*
2 sockinfo.C -- socket address management
3 Copyright (C) 2003-2008 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 it
8 under the terms of the GNU General Public License as published by the
9 Free Software Foundation; either version 3 of the License, or (at your
10 option) any later version.
11
12 This program is distributed in the hope that it will be useful, but
13 WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
15 Public License for more details.
16
17 You should have received a copy of the GNU General Public License along
18 with this program; if not, see <http://www.gnu.org/licenses/>.
19
20 Additional permission under GNU GPL version 3 section 7
21
22 If you modify this Program, or any covered work, by linking or
23 combining it with the OpenSSL project's OpenSSL library (or a modified
24 version of that library), containing parts covered by the terms of the
25 OpenSSL or SSLeay licenses, the licensors of this Program grant you
26 additional permission to convey the resulting work. Corresponding
27 Source for a non-source form of such a combination shall include the
28 source code for the parts of OpenSSL used as well as that of the
29 covered work.
30 */
31
32 #include "config.h"
33
34 // for gethostbyname
35 #include <netdb.h>
36
37 // for inet_aton
38 #include <sys/socket.h>
39 #include <netinet/in.h>
40 #include <arpa/inet.h>
41
42 #include "gettext.h"
43
44 #include "sockinfo.h"
45 #include "slog.h"
46
47 #include <cstring>
48 #include <cstdio>
49
50 // all ipv4-based protocols
51 #define PROTv4 (PROT_UDPv4 | PROT_TCPv4 | PROT_ICMPv4 | PROT_IPv4 | PROT_DNSv4)
52
53 void
54 sockinfo::set (const sockaddr_in *sa, u8 prot_)
55 {
56 host = sa->sin_addr.s_addr;
57 port = prot_ & (PROT_IPv4 | PROT_ICMPv4) ? 0 : sa->sin_port;
58 prot = prot_;
59 }
60
61 void
62 sockinfo::set (const char *hostname, u16 port_, u8 prot_)
63 {
64 prot = prot_;
65 host = 0;
66 port = htons (port_);
67
68 if (prot & PROTv4
69 && hostname)
70 {
71 struct in_addr in;
72 struct hostent *he;
73
74 if (inet_aton (hostname, &in))
75 {
76 host = in.s_addr;
77 }
78 else if ((he = gethostbyname (hostname))
79 && he->h_addrtype == AF_INET
80 && he->h_length == 4
81 && he->h_addr_list[0])
82 {
83 //sa->sin_family = he->h_addrtype;
84 memcpy (&host, he->h_addr_list[0], 4);
85 }
86 else
87 slog (L_NOTICE, _("unable to resolve host '%s'"), hostname);
88 }
89 }
90
91 void
92 sockinfo::set (const conf_node *conf, u8 prot_)
93 {
94 if (prot_ == PROT_DNSv4)
95 {
96 host = htonl (conf->id); port = 0; prot = prot_;
97 }
98 else
99 set (conf->hostname,
100 prot_ == PROT_UDPv4 ? conf->udp_port
101 : prot_ == PROT_TCPv4 ? conf->tcp_port
102 : prot_ == PROT_DNSv4 ? conf->dns_port
103 : 0,
104 prot_);
105 }
106
107 const sockaddr *
108 sockinfo::sav4() const
109 {
110 static sockaddr_in sa;
111
112 sa.sin_family = AF_INET;
113 sa.sin_port = port;
114 sa.sin_addr.s_addr = host;
115
116 return (const sockaddr *)&sa;
117 }
118
119 static char hostport[10 + 15 + 1 + 5 + 1]; // proto / IPv4 : port
120
121 const char *
122 sockinfo::ntoa () const
123 {
124 in_addr ia = { host };
125
126 sprintf (hostport, "%.15s", inet_ntoa (ia));
127
128 return hostport;
129 }
130
131 bool
132 sockinfo::valid () const
133 {
134 return (prot & THISNODE->protocols) && host;
135 }
136
137 sockinfo::operator const char *() const
138 {
139 in_addr ia = { host };
140
141 sprintf (hostport, "%s/%.15s:%d", strprotocol (prot), inet_ntoa (ia), ntohs (port) & 0xffff);
142
143 return hostport;
144 }
145
146 u8
147 sockinfo::supported_protocols (conf_node *conf)
148 {
149 u8 protocols = prot;
150
151 if (prot & (PROT_IPv4 | PROT_ICMPv4 | PROT_UDPv4 | PROT_TCPv4))
152 protocols |= PROT_IPv4 | PROT_ICMPv4;
153
154 if (conf && prot & PROTv4)
155 {
156 if (conf->protocols & PROT_UDPv4 && conf->udp_port)
157 protocols |= PROT_UDPv4;
158
159 if (conf->protocols & PROT_TCPv4 && conf->tcp_port)
160 protocols |= PROT_TCPv4;
161
162 if (conf->protocols & PROT_DNSv4 && conf->dns_port)
163 protocols |= PROT_DNSv4;
164 }
165
166 return protocols;
167 }
168
169 bool
170 sockinfo::upgrade_protocol (u8 prot_, conf_node *conf)
171 {
172 if (prot_ == prot)
173 return true;
174
175 if (prot & PROTv4
176 && prot_ & PROTv4)
177 {
178 if (prot_ & (PROT_IPv4 | PROT_ICMPv4))
179 {
180 prot = prot_;
181 port = 0;
182 return true;
183 }
184
185 if (conf
186 && prot_ & PROT_UDPv4
187 && conf->protocols & PROT_UDPv4
188 && conf->udp_port)
189 {
190 prot = prot_;
191 port = htons (conf->udp_port);
192 return true;
193 }
194
195 if (conf
196 && prot_ & PROT_TCPv4
197 && conf->protocols & PROT_TCPv4
198 && conf->tcp_port)
199 {
200 prot = prot_;
201 port = htons (conf->tcp_port);
202 return true;
203 }
204 }
205
206 return false;
207 }
208
209 bool
210 operator == (const sockinfo &a, const sockinfo &b)
211 {
212 return a.host == b.host && a.port == b.port && a.prot == b.prot;
213 }
214
215 bool
216 operator < (const sockinfo &a, const sockinfo &b)
217 {
218 return a.host < b.host
219 || (a.host == b.host && (a.port < b.port
220 || (a.port == b.port && a.prot < b.prot)));
221 }
222