timeout for connect
[radsecproxy.git] / util.c
1 /*
2  * Copyright (C) 2006-2008 Stig Venaas <venaas@uninett.no>
3  *
4  * Permission to use, copy, modify, and distribute this software for any
5  * purpose with or without fee is hereby granted, provided that the above
6  * copyright notice and this permission notice appear in all copies.
7  */
8
9 #include <sys/socket.h>
10 #include <netinet/in.h>
11 #include <netdb.h>
12 #include <stdio.h>
13 #include <stdlib.h>
14 #include <string.h>
15 #include <unistd.h>
16 #include <fcntl.h>
17 #include <errno.h>
18 #include <sys/select.h>
19 #include <stdarg.h>
20 #include "debug.h"
21 #include "util.h"
22
23 char *stringcopy(const char *s, int len) {
24     char *r;
25     if (!s)
26         return NULL;
27     if (!len)
28         len = strlen(s);
29     r = malloc(len + 1);
30     if (!r)
31         debug(DBG_ERR, "stringcopy: malloc failed");
32     memcpy(r, s, len);
33     r[len] = '\0';
34     return r;
35 }
36
37 void printfchars(char *prefixfmt, char *prefix, char *charfmt, char *chars, int len) {
38     int i;
39     unsigned char *s = (unsigned char *)chars;
40     if (prefix)
41         printf(prefixfmt ? prefixfmt : "%s: ", prefix);
42     for (i = 0; i < len; i++)
43         printf(charfmt ? charfmt : "%c", s[i]);
44     printf("\n");
45 }
46
47 void port_set(struct sockaddr *sa, uint16_t port) {
48     switch (sa->sa_family) {
49     case AF_INET:
50         ((struct sockaddr_in *)sa)->sin_port = htons(port);
51         break;
52     case AF_INET6:
53         ((struct sockaddr_in6 *)sa)->sin6_port = htons(port);
54         break;
55     }
56 }
57
58 struct sockaddr *addr_copy(struct sockaddr *in) {
59     struct sockaddr *out = NULL;
60     
61     switch (in->sa_family) {
62     case AF_INET:
63         out = malloc(sizeof(struct sockaddr_in));
64         if (out) {
65             memset(out, 0, sizeof(struct sockaddr_in));
66             ((struct sockaddr_in *)out)->sin_addr = ((struct sockaddr_in *)in)->sin_addr;
67         }
68         break;
69     case AF_INET6:
70         out = malloc(sizeof(struct sockaddr_in6));
71         if (out) {
72             memset(out, 0, sizeof(struct sockaddr_in6));
73             ((struct sockaddr_in6 *)out)->sin6_addr = ((struct sockaddr_in6 *)in)->sin6_addr;
74         }
75         break;
76     }
77     out->sa_family = in->sa_family;
78     return out;
79 }
80
81 char *addr2string(struct sockaddr *addr) {
82     struct sockaddr_in6 *sa6;
83     struct sockaddr_in sa4;
84     static char addr_buf[2][INET6_ADDRSTRLEN];
85     static int i = 0;
86     i = !i;
87     if (addr->sa_family == AF_INET6) {
88         sa6 = (struct sockaddr_in6 *)addr;
89         if (IN6_IS_ADDR_V4MAPPED(&sa6->sin6_addr)) {
90             memset(&sa4, 0, sizeof(sa4));
91             sa4.sin_family = AF_INET;
92             sa4.sin_port = sa6->sin6_port;
93             memcpy(&sa4.sin_addr, &sa6->sin6_addr.s6_addr[12], 4);
94             addr = (struct sockaddr *)&sa4;
95         }
96     }
97     if (getnameinfo(addr, SOCKADDRP_SIZE(addr), addr_buf[i], sizeof(addr_buf[i]),
98                     NULL, 0, NI_NUMERICHOST)) {
99         debug(DBG_WARN, "getnameinfo failed");
100         return "getnameinfo_failed";
101     }
102     return addr_buf[i];
103 }
104
105 #if 0
106 /* not in use */
107 int connectport(int type, char *host, char *port) {
108     struct addrinfo hints, *res0, *res;
109     int s = -1;
110     
111     memset(&hints, 0, sizeof(hints));
112     hints.ai_socktype = type;
113     hints.ai_family = AF_UNSPEC;
114
115     if (getaddrinfo(host, port, &hints, &res0) != 0) {
116         debug(DBG_ERR, "connectport: can't resolve host %s port %s", host, port);
117         return -1;
118     }
119
120     for (res = res0; res; res = res->ai_next) {
121         s = socket(res->ai_family, res->ai_socktype, res->ai_protocol);
122         if (s < 0) {
123             debug(DBG_WARN, "connectport: socket failed");
124             continue;
125         }
126         if (connect(s, res->ai_addr, res->ai_addrlen) == 0)
127             break;
128         debug(DBG_WARN, "connectport: connect failed");
129         close(s);
130         s = -1;
131     }
132     freeaddrinfo(res0);
133     return s;
134 }
135 #endif
136
137 int bindtoaddr(struct addrinfo *addrinfo, int family, int reuse, int v6only) {
138     int s, on = 1;
139     struct addrinfo *res;
140
141     for (res = addrinfo; res; res = res->ai_next) {
142         if (family != AF_UNSPEC && family != res->ai_family)
143             continue;
144         s = socket(res->ai_family, res->ai_socktype, res->ai_protocol);
145         if (s < 0) {
146             debug(DBG_WARN, "bindtoaddr: socket failed");
147             continue;
148         }
149         if (reuse)
150             setsockopt(s, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
151         #ifdef IPV6_V6ONLY
152         if (v6only)
153             setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY, &on, sizeof(on));
154         #endif
155         if (!bind(s, res->ai_addr, res->ai_addrlen))
156             return s;
157         debug(DBG_WARN, "bindtoaddr: bind failed");
158         close(s);
159     }
160     return -1;
161 }
162
163 int connectnonblocking(int s, const struct sockaddr *addr, socklen_t addrlen, struct timeval *timeout) {
164     int origflags, error = 0, r = -1;
165     fd_set writefds;
166     socklen_t len;
167     
168     origflags = fcntl(s, F_GETFL, 0);
169     fcntl(s, F_SETFL, origflags | O_NONBLOCK);
170     if (!connect(s, addr, addrlen)) {
171         r = 0;
172         goto exit;
173     }
174     if (errno != EINPROGRESS)
175         goto exit;
176
177     FD_ZERO(&writefds);
178     FD_SET(s, &writefds);
179     if (select(s + 1, NULL, &writefds, NULL, timeout) < 1)
180         goto exit;
181
182     len = sizeof(error);
183     if (!getsockopt(s, SOL_SOCKET, SO_ERROR, (char*)&error, &len) && !error)
184         r = 0;
185
186  exit:    
187     fcntl(s, F_SETFL, origflags);
188     return r;
189 }
190
191 int connecttcp(struct addrinfo *addrinfo, struct addrinfo *src, uint16_t timeout) {
192     int s;
193     struct addrinfo *res;
194     struct timeval to;
195
196     s = -1;
197     if (timeout) {
198         if (res && res->ai_next && timeout > 5)
199             timeout = 5;
200         to.tv_sec = timeout;
201         to.tv_usec = 0;
202     }
203     
204     for (res = addrinfo; res; res = res->ai_next) {
205         s = bindtoaddr(src, res->ai_family, 1, 1);
206         if (s < 0) {
207             debug(DBG_WARN, "connecttoserver: socket failed");
208             continue;
209         }
210         if ((timeout
211              ? connectnonblocking(s, res->ai_addr, res->ai_addrlen, &to)
212              : connect(s, res->ai_addr, res->ai_addrlen)) == 0)
213             break;
214         debug(DBG_WARN, "connecttoserver: connect failed");
215         close(s);
216         s = -1;
217     }
218     return s;
219 }