blob: af3d40c322fbfcd7ab2e08c52540264f69c7ef91 [file] [log] [blame]
Benny Prijono5dcb38d2005-11-21 01:55:47 +00001/* $Id$ */
2/*
3 * Copyright (C)2003-2006 Benny Prijono <benny@prijono.org>
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18 */
19#include <pj/addr_resolv.h>
20#include <pj/assert.h>
21#include <pj/string.h>
Benny Prijono5dcb38d2005-11-21 01:55:47 +000022#include <pj/errno.h>
Benny Prijono594e4c52006-09-14 18:51:01 +000023#include <pj/compat/socket.h>
Benny Prijono5dcb38d2005-11-21 01:55:47 +000024
25
26PJ_DEF(pj_status_t) pj_gethostbyname(const pj_str_t *hostname, pj_hostent *phe)
27{
28 struct hostent *he;
29 char copy[PJ_MAX_HOSTNAME];
30
31 pj_assert(hostname && hostname ->slen < PJ_MAX_HOSTNAME);
32
33 if (hostname->slen >= PJ_MAX_HOSTNAME)
34 return PJ_ENAMETOOLONG;
35
36 pj_memcpy(copy, hostname->ptr, hostname->slen);
37 copy[ hostname->slen ] = '\0';
38
39 he = gethostbyname(copy);
40 if (!he)
Benny Prijonocca32812006-09-14 21:16:11 +000041 return PJ_ERESOLVE;
Benny Prijono5dcb38d2005-11-21 01:55:47 +000042
43 phe->h_name = he->h_name;
44 phe->h_aliases = he->h_aliases;
45 phe->h_addrtype = he->h_addrtype;
46 phe->h_length = he->h_length;
47 phe->h_addr_list = he->h_addr_list;
48
49 return PJ_SUCCESS;
50}
51
Benny Prijono594e4c52006-09-14 18:51:01 +000052/* Resolve the IP address of local machine */
53pj_status_t pj_gethostip(pj_in_addr *addr)
54{
55 const pj_str_t *hostname = pj_gethostname();
56 struct pj_hostent he;
57 pj_str_t cp;
58 pj_in_addr loopip;
59 pj_status_t status;
60
61 cp = pj_str("127.0.0.1");
62 loopip = pj_inet_addr(&cp);
63
64 /* Try with resolving local hostname first */
65 status = pj_gethostbyname(hostname, &he);
66 if (status == PJ_SUCCESS) {
67 *addr = *(pj_in_addr*)he.h_addr;
68 }
69
70
71 /* If we end up with 127.0.0.1, resolve the IP by getting the default
72 * interface to connect to some public host.
73 */
74 if (status != PJ_SUCCESS || addr->s_addr == loopip.s_addr) {
75 pj_sock_t fd;
76 pj_sockaddr_in a;
77 int len;
78
79 status = pj_sock_socket(PJ_AF_INET, PJ_SOCK_DGRAM, 0, &fd);
80 if (status != PJ_SUCCESS) {
81 return status;
82 }
83
84 cp = pj_str("1.1.1.1");
85 pj_sockaddr_in_init(&a, &cp, 53);
86
87 status = pj_sock_connect(fd, &a, sizeof(a));
88 if (status != PJ_SUCCESS) {
89 pj_sock_close(fd);
90 return status;
91 }
92
93 len = sizeof(a);
94 status = pj_sock_getsockname(fd, &a, &len);
95 if (status != PJ_SUCCESS) {
96 pj_sock_close(fd);
97 return status;
98 }
99
100 pj_sock_close(fd);
101
102 *addr = a.sin_addr;
103 }
104
105 return status;
106}
107
108