2013-01-16 05:57:27 +01:00
|
|
|
//
|
|
|
|
// IPAddress.cpp
|
|
|
|
//
|
|
|
|
// $Id: //poco/1.4/Net/src/IPAddress.cpp#5 $
|
|
|
|
//
|
|
|
|
// Library: Net
|
|
|
|
// Package: NetCore
|
|
|
|
// Module: IPAddress
|
|
|
|
//
|
|
|
|
// Copyright (c) 2005-2011, Applied Informatics Software Engineering GmbH.
|
|
|
|
// and Contributors.
|
|
|
|
//
|
2014-05-04 21:02:42 +02:00
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
2013-01-16 05:57:27 +01:00
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
#include "Poco/Net/IPAddressImpl.h"
|
|
|
|
#include "Poco/Net/NetException.h"
|
|
|
|
#include "Poco/RefCountedObject.h"
|
|
|
|
#include "Poco/NumberFormatter.h"
|
2014-11-06 11:12:14 +01:00
|
|
|
#include "Poco/ByteOrder.h"
|
2013-01-16 05:57:27 +01:00
|
|
|
#include "Poco/String.h"
|
|
|
|
#include "Poco/Types.h"
|
|
|
|
|
|
|
|
|
|
|
|
using Poco::RefCountedObject;
|
|
|
|
using Poco::NumberFormatter;
|
|
|
|
using Poco::toLower;
|
|
|
|
using Poco::UInt8;
|
|
|
|
using Poco::UInt16;
|
|
|
|
using Poco::UInt32;
|
|
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2014-11-06 11:12:14 +01:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
template <typename T>
|
|
|
|
unsigned maskBits(T val, unsigned size)
|
|
|
|
/// Returns the length of the mask (number of bits set in val).
|
|
|
|
/// The val should be either all zeros or two contiguos areas of 1s and 0s.
|
|
|
|
/// The algorithm ignores invalid non-contiguous series of 1s and treats val
|
|
|
|
/// as if all bits between MSb and last non-zero bit are set to 1.
|
|
|
|
{
|
|
|
|
unsigned count = 0;
|
|
|
|
if (val)
|
|
|
|
{
|
|
|
|
val = (val ^ (val - 1)) >> 1;
|
|
|
|
for (count = 0; val; ++count) val >>= 1;
|
|
|
|
}
|
|
|
|
else count = size;
|
|
|
|
return size - count;
|
|
|
|
}
|
|
|
|
|
2014-11-06 11:12:14 +01:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
|
|
namespace Poco {
|
|
|
|
namespace Net {
|
|
|
|
namespace Impl {
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
//
|
|
|
|
// IPAddressImpl
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
IPAddressImpl::IPAddressImpl()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPAddressImpl::~IPAddressImpl()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// IPv4AddressImpl
|
|
|
|
//
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPv4AddressImpl::IPv4AddressImpl()
|
|
|
|
{
|
|
|
|
std::memset(&_addr, 0, sizeof(_addr));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv4AddressImpl::IPv4AddressImpl(const void* addr)
|
|
|
|
{
|
|
|
|
std::memcpy(&_addr, addr, sizeof(_addr));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv4AddressImpl::IPv4AddressImpl(unsigned prefix)
|
|
|
|
{
|
|
|
|
UInt32 addr = (prefix == 32) ? 0xffffffff : ~(0xffffffff >> prefix);
|
2014-11-06 11:12:14 +01:00
|
|
|
_addr.s_addr = ByteOrder::toNetwork(addr);
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv4AddressImpl::IPv4AddressImpl(const IPv4AddressImpl& addr)
|
|
|
|
{
|
|
|
|
std::memcpy(&_addr, &addr._addr, sizeof(_addr));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv4AddressImpl& IPv4AddressImpl::operator = (const IPv4AddressImpl& addr)
|
|
|
|
{
|
2014-10-17 15:15:07 +02:00
|
|
|
if (this == &addr)
|
|
|
|
return *this;
|
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
std::memcpy(&_addr, &addr._addr, sizeof(_addr));
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::string IPv4AddressImpl::toString() const
|
|
|
|
{
|
|
|
|
const UInt8* bytes = reinterpret_cast<const UInt8*>(&_addr);
|
|
|
|
std::string result;
|
|
|
|
result.reserve(16);
|
|
|
|
NumberFormatter::append(result, bytes[0]);
|
|
|
|
result.append(".");
|
|
|
|
NumberFormatter::append(result, bytes[1]);
|
|
|
|
result.append(".");
|
|
|
|
NumberFormatter::append(result, bytes[2]);
|
|
|
|
result.append(".");
|
|
|
|
NumberFormatter::append(result, bytes[3]);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
poco_socklen_t IPv4AddressImpl::length() const
|
|
|
|
{
|
|
|
|
return sizeof(_addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const void* IPv4AddressImpl::addr() const
|
|
|
|
{
|
|
|
|
return &_addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPAddressImpl::Family IPv4AddressImpl::family() const
|
|
|
|
{
|
|
|
|
return IPAddressImpl::IPv4;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int IPv4AddressImpl::af() const
|
|
|
|
{
|
|
|
|
return AF_INET;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
unsigned IPv4AddressImpl::prefixLength() const
|
|
|
|
{
|
|
|
|
return maskBits(ntohl(_addr.s_addr), 32);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Poco::UInt32 IPv4AddressImpl::scope() const
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isWildcard() const
|
|
|
|
{
|
|
|
|
return _addr.s_addr == INADDR_ANY;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isBroadcast() const
|
|
|
|
{
|
|
|
|
return _addr.s_addr == INADDR_NONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isLoopback() const
|
|
|
|
{
|
|
|
|
return (ntohl(_addr.s_addr) & 0xFF000000) == 0x7F000000; // 127.0.0.1 to 127.255.255.255
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isMulticast() const
|
|
|
|
{
|
|
|
|
return (ntohl(_addr.s_addr) & 0xF0000000) == 0xE0000000; // 224.0.0.0/24 to 239.0.0.0/24
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isLinkLocal() const
|
|
|
|
{
|
|
|
|
return (ntohl(_addr.s_addr) & 0xFFFF0000) == 0xA9FE0000; // 169.254.0.0/16
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isSiteLocal() const
|
|
|
|
{
|
|
|
|
UInt32 addr = ntohl(_addr.s_addr);
|
|
|
|
return (addr & 0xFF000000) == 0x0A000000 || // 10.0.0.0/24
|
|
|
|
(addr & 0xFFFF0000) == 0xC0A80000 || // 192.68.0.0/16
|
|
|
|
(addr >= 0xAC100000 && addr <= 0xAC1FFFFF); // 172.16.0.0 to 172.31.255.255
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isIPv4Compatible() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isIPv4Mapped() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isWellKnownMC() const
|
|
|
|
{
|
|
|
|
return (ntohl(_addr.s_addr) & 0xFFFFFF00) == 0xE0000000; // 224.0.0.0/8
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isNodeLocalMC() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isLinkLocalMC() const
|
|
|
|
{
|
|
|
|
return (ntohl(_addr.s_addr) & 0xFF000000) == 0xE0000000; // 244.0.0.0/24
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isSiteLocalMC() const
|
|
|
|
{
|
|
|
|
return (ntohl(_addr.s_addr) & 0xFFFF0000) == 0xEFFF0000; // 239.255.0.0/16
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isOrgLocalMC() const
|
|
|
|
{
|
|
|
|
return (ntohl(_addr.s_addr) & 0xFFFF0000) == 0xEFC00000; // 239.192.0.0/16
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::isGlobalMC() const
|
|
|
|
{
|
|
|
|
UInt32 addr = ntohl(_addr.s_addr);
|
|
|
|
return addr >= 0xE0000100 && addr <= 0xEE000000; // 224.0.1.0 to 238.255.255.255
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv4AddressImpl IPv4AddressImpl::parse(const std::string& addr)
|
|
|
|
{
|
|
|
|
if (addr.empty()) return IPv4AddressImpl();
|
|
|
|
#if defined(_WIN32)
|
|
|
|
struct in_addr ia;
|
|
|
|
ia.s_addr = inet_addr(addr.c_str());
|
|
|
|
if (ia.s_addr == INADDR_NONE && addr != "255.255.255.255")
|
|
|
|
return IPv4AddressImpl();
|
|
|
|
else
|
|
|
|
return IPv4AddressImpl(&ia);
|
|
|
|
#else
|
|
|
|
#if __GNUC__ < 3 || defined(POCO_VXWORKS)
|
|
|
|
struct in_addr ia;
|
|
|
|
ia.s_addr = inet_addr(const_cast<char*>(addr.c_str()));
|
|
|
|
if (ia.s_addr == INADDR_NONE && addr != "255.255.255.255")
|
|
|
|
return IPv4AddressImpl();
|
|
|
|
else
|
|
|
|
return IPv4AddressImpl(&ia);
|
|
|
|
#else
|
|
|
|
struct in_addr ia;
|
|
|
|
if (inet_aton(addr.c_str(), &ia))
|
|
|
|
return IPv4AddressImpl(&ia);
|
|
|
|
else
|
|
|
|
return IPv4AddressImpl();
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void IPv4AddressImpl::mask(const IPAddressImpl* pMask, const IPAddressImpl* pSet)
|
|
|
|
{
|
|
|
|
poco_assert (pMask->af() == AF_INET && pSet->af() == AF_INET);
|
|
|
|
|
|
|
|
_addr.s_addr &= static_cast<const IPv4AddressImpl*>(pMask)->_addr.s_addr;
|
|
|
|
_addr.s_addr |= static_cast<const IPv4AddressImpl*>(pSet)->_addr.s_addr & ~static_cast<const IPv4AddressImpl*>(pMask)->_addr.s_addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPAddressImpl* IPv4AddressImpl::clone() const
|
|
|
|
{
|
|
|
|
return new IPv4AddressImpl(&_addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv4AddressImpl IPv4AddressImpl::operator & (const IPv4AddressImpl& addr) const
|
|
|
|
{
|
|
|
|
IPv4AddressImpl result(&_addr);
|
|
|
|
result._addr.s_addr &= addr._addr.s_addr;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv4AddressImpl IPv4AddressImpl::operator | (const IPv4AddressImpl& addr) const
|
|
|
|
{
|
|
|
|
IPv4AddressImpl result(&_addr);
|
|
|
|
result._addr.s_addr |= addr._addr.s_addr;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv4AddressImpl IPv4AddressImpl::operator ^ (const IPv4AddressImpl& addr) const
|
|
|
|
{
|
|
|
|
IPv4AddressImpl result(&_addr);
|
|
|
|
result._addr.s_addr ^= addr._addr.s_addr;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv4AddressImpl IPv4AddressImpl::operator ~ () const
|
|
|
|
{
|
|
|
|
IPv4AddressImpl result(&_addr);
|
|
|
|
result._addr.s_addr ^= 0xffffffff;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::operator == (const IPv4AddressImpl& addr) const
|
|
|
|
{
|
|
|
|
return 0 == std::memcmp(&addr._addr, &_addr, sizeof(_addr));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv4AddressImpl::operator != (const IPv4AddressImpl& addr) const
|
|
|
|
{
|
|
|
|
return !(*this == addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if defined(POCO_HAVE_IPv6)
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// IPv6AddressImpl
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
IPv6AddressImpl::IPv6AddressImpl(): _scope(0)
|
|
|
|
{
|
|
|
|
std::memset(&_addr, 0, sizeof(_addr));
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPv6AddressImpl::IPv6AddressImpl(const void* addr): _scope(0)
|
|
|
|
{
|
|
|
|
std::memcpy(&_addr, addr, sizeof(_addr));
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPv6AddressImpl::IPv6AddressImpl(const void* addr, Poco::UInt32 scope): _scope(scope)
|
|
|
|
{
|
|
|
|
std::memcpy(&_addr, addr, sizeof(_addr));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv6AddressImpl::IPv6AddressImpl(const IPv6AddressImpl& addr): _scope(addr._scope)
|
|
|
|
{
|
|
|
|
std::memcpy((void*) &_addr, (void*) &addr._addr, sizeof(_addr));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv6AddressImpl& IPv6AddressImpl::operator = (const IPv6AddressImpl& addr)
|
|
|
|
{
|
2014-10-17 15:15:07 +02:00
|
|
|
if (this == &addr)
|
|
|
|
return *this;
|
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
_scope = addr._scope;
|
|
|
|
std::memcpy(&_addr, &addr._addr, sizeof(_addr));
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IPv6AddressImpl::IPv6AddressImpl(unsigned prefix):
|
|
|
|
_scope(0)
|
|
|
|
{
|
|
|
|
unsigned i = 0;
|
|
|
|
#ifdef POCO_OS_FAMILY_WINDOWS
|
2014-11-06 11:12:14 +01:00
|
|
|
for (; prefix >= 16; ++i, prefix -= 16)
|
|
|
|
{
|
2013-01-16 05:57:27 +01:00
|
|
|
_addr.s6_addr16[i] = 0xffff;
|
|
|
|
}
|
|
|
|
if (prefix > 0)
|
2014-11-06 11:12:14 +01:00
|
|
|
{
|
2014-11-07 08:33:30 +01:00
|
|
|
_addr.s6_addr16[i++] = ByteOrder::toNetwork(static_cast<Poco::UInt16>(~(0xffff >> prefix)));
|
2014-11-06 11:12:14 +01:00
|
|
|
}
|
2013-01-16 05:57:27 +01:00
|
|
|
while (i < 8)
|
2014-11-06 11:12:14 +01:00
|
|
|
{
|
2013-01-16 05:57:27 +01:00
|
|
|
_addr.s6_addr16[i++] = 0;
|
2014-11-06 11:12:14 +01:00
|
|
|
}
|
2013-01-16 05:57:27 +01:00
|
|
|
#else
|
2014-11-06 11:12:14 +01:00
|
|
|
for (; prefix >= 32; ++i, prefix -= 32)
|
|
|
|
{
|
2013-01-16 05:57:27 +01:00
|
|
|
_addr.s6_addr32[i] = 0xffffffff;
|
|
|
|
}
|
|
|
|
if (prefix > 0)
|
2014-11-06 11:12:14 +01:00
|
|
|
{
|
|
|
|
_addr.s6_addr32[i++] = ByteOrder::toNetwork(~(0xffffffffU >> prefix));
|
|
|
|
}
|
2013-01-16 05:57:27 +01:00
|
|
|
while (i < 4)
|
2014-11-06 11:12:14 +01:00
|
|
|
{
|
2013-01-16 05:57:27 +01:00
|
|
|
_addr.s6_addr32[i++] = 0;
|
2014-11-06 11:12:14 +01:00
|
|
|
}
|
2013-01-16 05:57:27 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
std::string IPv6AddressImpl::toString() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-09-15 19:17:09 +02:00
|
|
|
if ((isIPv4Compatible() && !isLoopback()) || isIPv4Mapped())
|
2013-01-16 05:57:27 +01:00
|
|
|
{
|
|
|
|
std::string result;
|
|
|
|
result.reserve(24);
|
|
|
|
if (words[5] == 0)
|
|
|
|
result.append("::");
|
|
|
|
else
|
|
|
|
result.append("::ffff:");
|
|
|
|
const UInt8* bytes = reinterpret_cast<const UInt8*>(&_addr);
|
|
|
|
NumberFormatter::append(result, bytes[12]);
|
|
|
|
result.append(".");
|
|
|
|
NumberFormatter::append(result, bytes[13]);
|
|
|
|
result.append(".");
|
|
|
|
NumberFormatter::append(result, bytes[14]);
|
|
|
|
result.append(".");
|
|
|
|
NumberFormatter::append(result, bytes[15]);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::string result;
|
|
|
|
result.reserve(64);
|
|
|
|
bool zeroSequence = false;
|
|
|
|
int i = 0;
|
|
|
|
while (i < 8)
|
|
|
|
{
|
|
|
|
if (!zeroSequence && words[i] == 0)
|
|
|
|
{
|
|
|
|
int zi = i;
|
|
|
|
while (zi < 8 && words[zi] == 0) ++zi;
|
|
|
|
if (zi > i + 1)
|
|
|
|
{
|
|
|
|
i = zi;
|
|
|
|
result.append(":");
|
|
|
|
zeroSequence = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (i > 0) result.append(":");
|
2014-11-06 11:12:14 +01:00
|
|
|
if (i < 8) NumberFormatter::appendHex(result, ByteOrder::fromNetwork(words[i++]));
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
if (_scope > 0)
|
|
|
|
{
|
|
|
|
result.append("%");
|
|
|
|
#if defined(_WIN32)
|
|
|
|
NumberFormatter::append(result, _scope);
|
|
|
|
#else
|
|
|
|
char buffer[IFNAMSIZ];
|
|
|
|
if (if_indextoname(_scope, buffer))
|
|
|
|
{
|
|
|
|
result.append(buffer);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NumberFormatter::append(result, _scope);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
return toLower(result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
poco_socklen_t IPv6AddressImpl::length() const
|
|
|
|
{
|
|
|
|
return sizeof(_addr);
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
const void* IPv6AddressImpl::addr() const
|
|
|
|
{
|
|
|
|
return &_addr;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPAddressImpl::Family IPv6AddressImpl::family() const
|
|
|
|
{
|
|
|
|
return IPAddressImpl::IPv6;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
int IPv6AddressImpl::af() const
|
|
|
|
{
|
|
|
|
return AF_INET6;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
unsigned IPv6AddressImpl::prefixLength() const
|
|
|
|
{
|
|
|
|
unsigned bits = 0;
|
|
|
|
unsigned bitPos = 128;
|
|
|
|
#if defined(POCO_OS_FAMILY_UNIX)
|
|
|
|
for (int i = 3; i >= 0; --i)
|
|
|
|
{
|
|
|
|
unsigned addr = ntohl(_addr.s6_addr32[i]);
|
|
|
|
if ((bits = maskBits(addr, 32))) return (bitPos - (32 - bits));
|
|
|
|
bitPos -= 32;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
#elif defined(POCO_OS_FAMILY_WINDOWS)
|
|
|
|
for (int i = 7; i >= 0; --i)
|
|
|
|
{
|
2014-11-06 11:12:14 +01:00
|
|
|
unsigned short addr = ByteOrder::fromNetwork(_addr.s6_addr16[i]);
|
2013-01-16 05:57:27 +01:00
|
|
|
if ((bits = maskBits(addr, 16))) return (bitPos - (16 - bits));
|
|
|
|
bitPos -= 16;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
#warning prefixLength() not implemented
|
|
|
|
throw NotImplementedException("prefixLength() not implemented");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
Poco::UInt32 IPv6AddressImpl::scope() const
|
|
|
|
{
|
|
|
|
return _scope;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isWildcard() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
|
|
|
return words[0] == 0 && words[1] == 0 && words[2] == 0 && words[3] == 0 &&
|
|
|
|
words[4] == 0 && words[5] == 0 && words[6] == 0 && words[7] == 0;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isBroadcast() const
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isLoopback() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
|
|
|
return words[0] == 0 && words[1] == 0 && words[2] == 0 && words[3] == 0 &&
|
2014-11-06 11:12:14 +01:00
|
|
|
words[4] == 0 && words[5] == 0 && words[6] == 0 && ByteOrder::fromNetwork(words[7]) == 0x0001;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isMulticast() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return (ByteOrder::fromNetwork(words[0]) & 0xFFE0) == 0xFF00;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
2014-09-15 19:17:09 +02:00
|
|
|
|
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isLinkLocal() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return (ByteOrder::fromNetwork(words[0]) & 0xFFE0) == 0xFE80;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isSiteLocal() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return ((ByteOrder::fromNetwork(words[0]) & 0xFFE0) == 0xFEC0) || ((ByteOrder::fromNetwork(words[0]) & 0xFF00) == 0xFC00);
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isIPv4Compatible() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
|
|
|
return words[0] == 0 && words[1] == 0 && words[2] == 0 && words[3] == 0 && words[4] == 0 && words[5] == 0;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isIPv4Mapped() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return words[0] == 0 && words[1] == 0 && words[2] == 0 && words[3] == 0 && words[4] == 0 && ByteOrder::fromNetwork(words[5]) == 0xFFFF;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isWellKnownMC() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return (ByteOrder::fromNetwork(words[0]) & 0xFFF0) == 0xFF00;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isNodeLocalMC() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return (ByteOrder::fromNetwork(words[0]) & 0xFFEF) == 0xFF01;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isLinkLocalMC() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return (ByteOrder::fromNetwork(words[0]) & 0xFFEF) == 0xFF02;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isSiteLocalMC() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return (ByteOrder::fromNetwork(words[0]) & 0xFFEF) == 0xFF05;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isOrgLocalMC() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return (ByteOrder::fromNetwork(words[0]) & 0xFFEF) == 0xFF08;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
bool IPv6AddressImpl::isGlobalMC() const
|
|
|
|
{
|
|
|
|
const UInt16* words = reinterpret_cast<const UInt16*>(&_addr);
|
2014-11-06 11:12:14 +01:00
|
|
|
return (ByteOrder::fromNetwork(words[0]) & 0xFFEF) == 0xFF0F;
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPv6AddressImpl IPv6AddressImpl::parse(const std::string& addr)
|
|
|
|
{
|
|
|
|
if (addr.empty()) return IPv6AddressImpl();
|
|
|
|
#if defined(_WIN32)
|
|
|
|
struct addrinfo* pAI;
|
|
|
|
struct addrinfo hints;
|
|
|
|
std::memset(&hints, 0, sizeof(hints));
|
|
|
|
hints.ai_flags = AI_NUMERICHOST;
|
|
|
|
int rc = getaddrinfo(addr.c_str(), NULL, &hints, &pAI);
|
|
|
|
if (rc == 0)
|
|
|
|
{
|
|
|
|
IPv6AddressImpl result = IPv6AddressImpl(&reinterpret_cast<struct sockaddr_in6*>(pAI->ai_addr)->sin6_addr, static_cast<int>(reinterpret_cast<struct sockaddr_in6*>(pAI->ai_addr)->sin6_scope_id));
|
|
|
|
freeaddrinfo(pAI);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
else return IPv6AddressImpl();
|
|
|
|
#else
|
|
|
|
struct in6_addr ia;
|
|
|
|
std::string::size_type pos = addr.find('%');
|
|
|
|
if (std::string::npos != pos)
|
|
|
|
{
|
|
|
|
std::string::size_type start = ('[' == addr[0]) ? 1 : 0;
|
|
|
|
std::string unscopedAddr(addr, start, pos - start);
|
|
|
|
std::string scope(addr, pos + 1, addr.size() - start - pos);
|
|
|
|
Poco::UInt32 scopeId(0);
|
|
|
|
if (!(scopeId = if_nametoindex(scope.c_str())))
|
2013-02-04 03:19:14 +01:00
|
|
|
return IPv6AddressImpl();
|
2013-01-16 05:57:27 +01:00
|
|
|
if (inet_pton(AF_INET6, unscopedAddr.c_str(), &ia) == 1)
|
|
|
|
return IPv6AddressImpl(&ia, scopeId);
|
|
|
|
else
|
2013-02-04 03:19:14 +01:00
|
|
|
return IPv6AddressImpl();
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (inet_pton(AF_INET6, addr.c_str(), &ia) == 1)
|
2013-02-04 03:19:14 +01:00
|
|
|
return IPv6AddressImpl(&ia);
|
2013-01-16 05:57:27 +01:00
|
|
|
else
|
2013-02-04 03:19:14 +01:00
|
|
|
return IPv6AddressImpl();
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
void IPv6AddressImpl::mask(const IPAddressImpl* pMask, const IPAddressImpl* pSet)
|
|
|
|
{
|
|
|
|
throw Poco::NotImplementedException("mask() is only supported for IPv4 addresses");
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPAddressImpl* IPv6AddressImpl::clone() const
|
|
|
|
{
|
2014-10-17 15:15:07 +02:00
|
|
|
return new IPv6AddressImpl(*this);
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPv6AddressImpl IPv6AddressImpl::operator & (const IPv6AddressImpl& addr) const
|
|
|
|
{
|
2014-11-06 11:12:14 +01:00
|
|
|
if (_scope != addr._scope)
|
|
|
|
throw Poco::InvalidArgumentException("Scope ID of passed IPv6 address does not match with the source one.");
|
2014-10-17 15:15:07 +02:00
|
|
|
|
|
|
|
IPv6AddressImpl result(*this);
|
2013-01-16 05:57:27 +01:00
|
|
|
#ifdef POCO_OS_FAMILY_WINDOWS
|
|
|
|
result._addr.s6_addr16[0] &= addr._addr.s6_addr16[0];
|
|
|
|
result._addr.s6_addr16[1] &= addr._addr.s6_addr16[1];
|
|
|
|
result._addr.s6_addr16[2] &= addr._addr.s6_addr16[2];
|
|
|
|
result._addr.s6_addr16[3] &= addr._addr.s6_addr16[3];
|
|
|
|
result._addr.s6_addr16[4] &= addr._addr.s6_addr16[4];
|
|
|
|
result._addr.s6_addr16[5] &= addr._addr.s6_addr16[5];
|
|
|
|
result._addr.s6_addr16[6] &= addr._addr.s6_addr16[6];
|
|
|
|
result._addr.s6_addr16[7] &= addr._addr.s6_addr16[7];
|
|
|
|
#else
|
|
|
|
result._addr.s6_addr32[0] &= addr._addr.s6_addr32[0];
|
|
|
|
result._addr.s6_addr32[1] &= addr._addr.s6_addr32[1];
|
|
|
|
result._addr.s6_addr32[2] &= addr._addr.s6_addr32[2];
|
|
|
|
result._addr.s6_addr32[3] &= addr._addr.s6_addr32[3];
|
|
|
|
#endif
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPv6AddressImpl IPv6AddressImpl::operator | (const IPv6AddressImpl& addr) const
|
|
|
|
{
|
2014-11-06 11:12:14 +01:00
|
|
|
if (_scope != addr._scope)
|
|
|
|
throw Poco::InvalidArgumentException("Scope ID of passed IPv6 address does not match with the source one.");
|
2014-10-17 15:15:07 +02:00
|
|
|
|
|
|
|
IPv6AddressImpl result(*this);
|
2013-01-16 05:57:27 +01:00
|
|
|
#ifdef POCO_OS_FAMILY_WINDOWS
|
|
|
|
result._addr.s6_addr16[0] |= addr._addr.s6_addr16[0];
|
|
|
|
result._addr.s6_addr16[1] |= addr._addr.s6_addr16[1];
|
|
|
|
result._addr.s6_addr16[2] |= addr._addr.s6_addr16[2];
|
|
|
|
result._addr.s6_addr16[3] |= addr._addr.s6_addr16[3];
|
|
|
|
result._addr.s6_addr16[4] |= addr._addr.s6_addr16[4];
|
|
|
|
result._addr.s6_addr16[5] |= addr._addr.s6_addr16[5];
|
|
|
|
result._addr.s6_addr16[6] |= addr._addr.s6_addr16[6];
|
|
|
|
result._addr.s6_addr16[7] |= addr._addr.s6_addr16[7];
|
|
|
|
#else
|
|
|
|
result._addr.s6_addr32[0] |= addr._addr.s6_addr32[0];
|
|
|
|
result._addr.s6_addr32[1] |= addr._addr.s6_addr32[1];
|
|
|
|
result._addr.s6_addr32[2] |= addr._addr.s6_addr32[2];
|
|
|
|
result._addr.s6_addr32[3] |= addr._addr.s6_addr32[3];
|
|
|
|
#endif
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPv6AddressImpl IPv6AddressImpl::operator ^ (const IPv6AddressImpl& addr) const
|
|
|
|
{
|
2014-11-06 11:12:14 +01:00
|
|
|
if (_scope != addr._scope)
|
|
|
|
throw Poco::InvalidArgumentException("Scope ID of passed IPv6 address does not match with the source one.");
|
2014-10-17 15:15:07 +02:00
|
|
|
|
|
|
|
IPv6AddressImpl result(*this);
|
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
#ifdef POCO_OS_FAMILY_WINDOWS
|
|
|
|
result._addr.s6_addr16[0] ^= addr._addr.s6_addr16[0];
|
|
|
|
result._addr.s6_addr16[1] ^= addr._addr.s6_addr16[1];
|
|
|
|
result._addr.s6_addr16[2] ^= addr._addr.s6_addr16[2];
|
|
|
|
result._addr.s6_addr16[3] ^= addr._addr.s6_addr16[3];
|
|
|
|
result._addr.s6_addr16[4] ^= addr._addr.s6_addr16[4];
|
|
|
|
result._addr.s6_addr16[5] ^= addr._addr.s6_addr16[5];
|
|
|
|
result._addr.s6_addr16[6] ^= addr._addr.s6_addr16[6];
|
|
|
|
result._addr.s6_addr16[7] ^= addr._addr.s6_addr16[7];
|
|
|
|
#else
|
|
|
|
result._addr.s6_addr32[0] ^= addr._addr.s6_addr32[0];
|
|
|
|
result._addr.s6_addr32[1] ^= addr._addr.s6_addr32[1];
|
|
|
|
result._addr.s6_addr32[2] ^= addr._addr.s6_addr32[2];
|
|
|
|
result._addr.s6_addr32[3] ^= addr._addr.s6_addr32[3];
|
|
|
|
#endif
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-09-15 19:17:09 +02:00
|
|
|
|
2013-01-16 05:57:27 +01:00
|
|
|
IPv6AddressImpl IPv6AddressImpl::operator ~ () const
|
|
|
|
{
|
2014-10-17 15:15:07 +02:00
|
|
|
IPv6AddressImpl result(*this);
|
2013-01-16 05:57:27 +01:00
|
|
|
#ifdef POCO_OS_FAMILY_WINDOWS
|
|
|
|
result._addr.s6_addr16[0] ^= 0xffff;
|
|
|
|
result._addr.s6_addr16[1] ^= 0xffff;
|
|
|
|
result._addr.s6_addr16[2] ^= 0xffff;
|
|
|
|
result._addr.s6_addr16[3] ^= 0xffff;
|
|
|
|
result._addr.s6_addr16[4] ^= 0xffff;
|
|
|
|
result._addr.s6_addr16[5] ^= 0xffff;
|
|
|
|
result._addr.s6_addr16[6] ^= 0xffff;
|
|
|
|
result._addr.s6_addr16[7] ^= 0xffff;
|
|
|
|
#else
|
|
|
|
result._addr.s6_addr32[0] ^= 0xffffffff;
|
|
|
|
result._addr.s6_addr32[1] ^= 0xffffffff;
|
|
|
|
result._addr.s6_addr32[2] ^= 0xffffffff;
|
|
|
|
result._addr.s6_addr32[3] ^= 0xffffffff;
|
|
|
|
#endif
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv6AddressImpl::operator == (const IPv6AddressImpl& addr) const
|
|
|
|
{
|
2014-10-17 15:15:07 +02:00
|
|
|
return _scope == addr._scope && 0 == std::memcmp(&addr._addr, &_addr, sizeof(_addr));
|
2013-01-16 05:57:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IPv6AddressImpl::operator != (const IPv6AddressImpl& addr) const
|
|
|
|
{
|
|
|
|
return !(*this == addr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#endif // POCO_HAVE_IPv6
|
|
|
|
|
|
|
|
|
|
|
|
} } } // namespace Poco::Net::Impl
|