Socket/Protocols/INet: Implemented INet4Network
[senf.git] / Socket / Protocols / INet / INet4Address.cc
1 // $Id$
2 //
3 // Copyright (C) 2007 
4 // Fraunhofer Institut fuer offene Kommunikationssysteme (FOKUS)
5 // Kompetenzzentrum fuer Satelitenkommunikation (SatCom)
6 //     Stefan Bund <g0dil@berlios.de>
7 //
8 // This program is free software; you can redistribute it and/or modify
9 // it under the terms of the GNU General Public License as published by
10 // the Free Software Foundation; either version 2 of the License, or
11 // (at your option) any later version.
12 //
13 // This program is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 // GNU General Public License for more details.
17 //
18 // You should have received a copy of the GNU General Public License
19 // along with this program; if not, write to the
20 // Free Software Foundation, Inc.,
21 // 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22
23 /** \file
24     \brief INet4Address non-inline non-template implementation */
25
26 #include "INet4Address.hh"
27 //#include "INet4Address.ih"
28
29 // Custom includes
30 #include <arpa/inet.h>
31 #include <netdb.h>
32 #include <sys/socket.h>
33 #include <boost/lexical_cast.hpp>
34 #ifdef _REENTRANT
35 #include <boost/thread/mutex.hpp>
36 #endif
37
38 //#include "INet4Address.mpp"
39 #define prefix_
40 ///////////////////////////////cc.p////////////////////////////////////////
41
42 ///////////////////////////////////////////////////////////////////////////
43 // senf::INet4Address::INet4Address
44
45 prefix_ senf::INet4Address::INet4Address(address_type value)
46 {
47     iref() = htonl(value);
48 }
49
50 prefix_ senf::INet4Address senf::INet4Address::from_string(std::string const & s)
51 {
52     struct in_addr ina;
53     if (::inet_pton(AF_INET,s.c_str(),&ina) > 0)
54         return senf::INet4Address::from_inaddr(ina.s_addr);
55     int herr (0);
56
57     // If available, we use the reentrant GNU variant. This has the additional advantage, that we
58     // can explicitly ask for IpV4 addresses
59
60 #   ifdef __GLIBC__
61
62     struct hostent entbuf;
63     char buffer[4096];
64     struct hostent * ent (0);
65     ::gethostbyname2_r(s.c_str(), AF_INET, &entbuf, buffer, sizeof(buffer), &ent, &herr);
66
67 #   else // ! __GLIBC__
68
69 #   ifdef _REENTRANT
70     static boost::mutex mutex;
71     boost::mutex::scoped_lock lock(mutex);
72 #   endif
73     struct hostent * ent (::gethostbyname(s.c_str()));
74     herr = h_errno;
75
76 #   endif // __GLIBC__
77
78     if (!ent)
79         ///\fixme Need to give better exception here
80         throw SyntaxException(); 
81     if (ent->h_addrtype != AF_INET)
82         throw SyntaxException();    
83
84     // We are only interested in the first address ...
85     return senf::INet4Address::from_inaddr(
86         reinterpret_cast<in_addr*>(*(ent->h_addr_list))->s_addr);
87 }
88
89 prefix_ bool senf::INet4Address::local()
90     const
91 {
92     address_type l (address());
93     return 
94         (l & 0xFF000000u) == 0x0A000000u ||
95         (l & 0xFFF00000u) == 0xAC100000u ||
96         (l & 0xFFFF0000u) == 0xA9FE0000u ||
97         (l & 0xFFFF0000u) == 0xC0A80000u;
98 }
99
100 prefix_ bool senf::INet4Address::loopback()
101     const
102 {
103     return (address() & 0xFF000000u) == 0x7F000000u;
104 }
105
106 prefix_ bool senf::INet4Address::multicast()
107     const
108 {
109     return (address() & 0xF0000000u) == 0xE0000000u;
110 }
111
112 prefix_ senf::INet4Address::address_type senf::INet4Address::address()
113     const
114 {
115     return ntohl(iref());
116 }
117
118 senf::INet4Address const senf::INet4Address::None;
119 senf::INet4Address const senf::INet4Address::Loopback (0x7F000001u);
120 senf::INet4Address const senf::INet4Address::Broadcast (0xFFFFFFFFu);
121
122 ///////////////////////////////////////////////////////////////////////////
123 // senf::INet4Network
124
125 prefix_ senf::INet4Network::INet4Network(std::string s)
126 {
127     std::string::size_type i (s.find('/'));
128     if (i == std::string::npos)
129         throw INet4Address::SyntaxException();
130     try {
131         prefix_len_ = boost::lexical_cast<unsigned>(std::string(s,i+1));
132     } catch (boost::bad_lexical_cast const &) {
133         throw INet4Address::SyntaxException();
134     }
135     address_ = INet4Address(INet4Address::from_string(std::string(s, 0, i)).address() & mask());
136 }
137
138 ///////////////////////////////////////////////////////////////////////////
139 // namespace members
140
141 prefix_ std::ostream & senf::operator<<(std::ostream & os, INet4Address const & addr)
142 {
143     ::in_addr ina;
144     char buffer[16];
145     ina.s_addr = addr.inaddr();
146     ::inet_ntop(AF_INET,&ina,buffer,16);
147     buffer[15] = 0; 
148     os << buffer;
149     return os;
150 }
151
152 ///////////////////////////////cc.e////////////////////////////////////////
153 #undef prefix_
154 //#include "INet4Address.mpp"
155
156 \f
157 // Local Variables:
158 // mode: c++
159 // fill-column: 100
160 // comment-column: 40
161 // c-file-style: "senf"
162 // indent-tabs-mode: nil
163 // ispell-local-dictionary: "american"
164 // compile-command: "scons -u test"
165 // End: