1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
|
// Copyright (c) 2011 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "net/base/ip_endpoint.h"
#include "base/logging.h"
#if defined(OS_WIN)
#include <winsock2.h>
#elif defined(OS_POSIX)
#include <netinet/in.h>
#endif
namespace net {
const size_t kIPv4AddressSize = 4;
const size_t kIPv6AddressSize = 16;
IPEndPoint::IPEndPoint() : port_(0) {}
IPEndPoint::~IPEndPoint() {}
IPEndPoint::IPEndPoint(const IPAddressNumber& address, int port)
: address_(address),
port_(port) {}
IPEndPoint::IPEndPoint(const IPEndPoint& endpoint) {
address_ = endpoint.address_;
port_ = endpoint.port_;
}
int IPEndPoint::GetFamily() const {
switch (address_.size()) {
case kIPv4AddressSize: {
return AF_INET;
}
case kIPv6AddressSize: {
return AF_INET6;
}
default: {
NOTREACHED() << "Bad IP address";
return AF_INET;
}
}
}
bool IPEndPoint::ToSockAddr(struct sockaddr* address,
size_t* address_length) const {
DCHECK(address);
DCHECK(address_length);
switch (address_.size()) {
case kIPv4AddressSize: {
if (*address_length < sizeof(struct sockaddr_in))
return false;
*address_length = sizeof(struct sockaddr_in);
struct sockaddr_in* addr = reinterpret_cast<struct sockaddr_in*>(address);
memset(addr, 0, sizeof(struct sockaddr_in));
addr->sin_family = AF_INET;
addr->sin_port = htons(port_);
memcpy(&addr->sin_addr, &address_[0], kIPv4AddressSize);
break;
}
case kIPv6AddressSize: {
if (*address_length < sizeof(struct sockaddr_in6))
return false;
*address_length = sizeof(struct sockaddr_in6);
struct sockaddr_in6* addr6 =
reinterpret_cast<struct sockaddr_in6*>(address);
memset(addr6, 0, sizeof(struct sockaddr_in6));
addr6->sin6_family = AF_INET6;
addr6->sin6_port = htons(port_);
memcpy(&addr6->sin6_addr, &address_[0], kIPv6AddressSize);
break;
}
default: {
NOTREACHED() << "Bad IP address";
break;
}
}
return true;
}
bool IPEndPoint::FromSockAddr(const struct sockaddr* address,
size_t address_length) {
DCHECK(address);
switch (address->sa_family) {
case AF_INET: {
const struct sockaddr_in* addr =
reinterpret_cast<const struct sockaddr_in*>(address);
port_ = ntohs(addr->sin_port);
const char* bytes = reinterpret_cast<const char*>(&addr->sin_addr);
address_.assign(&bytes[0], &bytes[kIPv4AddressSize]);
break;
}
case AF_INET6: {
const struct sockaddr_in6* addr =
reinterpret_cast<const struct sockaddr_in6*>(address);
port_ = ntohs(addr->sin6_port);
const char* bytes = reinterpret_cast<const char*>(&addr->sin6_addr);
address_.assign(&bytes[0], &bytes[kIPv6AddressSize]);
break;
}
default: {
NOTREACHED() << "Bad IP address";
break;
}
}
return true;
}
bool IPEndPoint::operator<(const IPEndPoint& that) const {
return address_ < that.address_ || port_ < that.port_;
}
bool IPEndPoint::operator==(const IPEndPoint& that) const {
return address_ == that.address_ && port_ == that.port_;
}
} // namespace net
|