518 lines
16 KiB
C++
518 lines
16 KiB
C++
#include "pch.h"
|
||
#include <strsafe.h>
|
||
#include <winsock2.h>
|
||
#include <ws2ipdef.h>
|
||
#include <ws2tcpip.h>
|
||
#include <spdlog/spdlog.h>
|
||
|
||
#include "usrerr.h"
|
||
#include "misc.h"
|
||
|
||
static const TCHAR *p2_table(unsigned pow) {
|
||
static const TCHAR *pow2[] = {
|
||
TEXT("1"),
|
||
TEXT("2"),
|
||
TEXT("4"),
|
||
TEXT("8"),
|
||
TEXT("16"),
|
||
TEXT("32"),
|
||
TEXT("64"),
|
||
TEXT("128"),
|
||
TEXT("256"),
|
||
TEXT("512"),
|
||
TEXT("1024"),
|
||
TEXT("2048"),
|
||
TEXT("4096"),
|
||
TEXT("8192"),
|
||
TEXT("16384"),
|
||
TEXT("32768"),
|
||
TEXT("65536"),
|
||
TEXT("131072"),
|
||
TEXT("262144"),
|
||
TEXT("524288"),
|
||
TEXT("1048576"),
|
||
TEXT("2097152"),
|
||
TEXT("4194304"),
|
||
TEXT("8388608"),
|
||
TEXT("16777216"),
|
||
TEXT("33554432"),
|
||
TEXT("67108864"),
|
||
TEXT("134217728"),
|
||
TEXT("268435456"),
|
||
TEXT("536870912"),
|
||
TEXT("1073741824"),
|
||
TEXT("2147483648"),
|
||
TEXT("4294967296"),
|
||
TEXT("8589934592"),
|
||
TEXT("17179869184"),
|
||
TEXT("34359738368"),
|
||
TEXT("68719476736"),
|
||
TEXT("137438953472"),
|
||
TEXT("274877906944"),
|
||
TEXT("549755813888"),
|
||
TEXT("1099511627776"),
|
||
TEXT("2199023255552"),
|
||
TEXT("4398046511104"),
|
||
TEXT("8796093022208"),
|
||
TEXT("17592186044416"),
|
||
TEXT("35184372088832"),
|
||
TEXT("70368744177664"),
|
||
TEXT("140737488355328"),
|
||
TEXT("281474976710656"),
|
||
TEXT("562949953421312"),
|
||
TEXT("1125899906842624"),
|
||
TEXT("2251799813685248"),
|
||
TEXT("4503599627370496"),
|
||
TEXT("9007199254740992"),
|
||
TEXT("18014398509481984"),
|
||
TEXT("36028797018963968"),
|
||
TEXT("72057594037927936"),
|
||
TEXT("144115188075855872"),
|
||
TEXT("288230376151711744"),
|
||
TEXT("576460752303423488"),
|
||
TEXT("1152921504606846976"),
|
||
TEXT("2305843009213693952"),
|
||
TEXT("4611686018427387904"),
|
||
TEXT("9223372036854775808"),
|
||
TEXT("18446744073709551616"),
|
||
TEXT("36893488147419103232"),
|
||
TEXT("73786976294838206464"),
|
||
TEXT("147573952589676412928"),
|
||
TEXT("295147905179352825856"),
|
||
TEXT("590295810358705651712"),
|
||
TEXT("1180591620717411303424"),
|
||
TEXT("2361183241434822606848"),
|
||
TEXT("4722366482869645213696"),
|
||
TEXT("9444732965739290427392"),
|
||
TEXT("18889465931478580854784"),
|
||
TEXT("37778931862957161709568"),
|
||
TEXT("75557863725914323419136"),
|
||
TEXT("151115727451828646838272"),
|
||
TEXT("302231454903657293676544"),
|
||
TEXT("604462909807314587353088"),
|
||
TEXT("1208925819614629174706176"),
|
||
TEXT("2417851639229258349412352"),
|
||
TEXT("4835703278458516698824704"),
|
||
TEXT("9671406556917033397649408"),
|
||
TEXT("19342813113834066795298816"),
|
||
TEXT("38685626227668133590597632"),
|
||
TEXT("77371252455336267181195264"),
|
||
TEXT("154742504910672534362390528"),
|
||
TEXT("309485009821345068724781056"),
|
||
TEXT("618970019642690137449562112"),
|
||
TEXT("1237940039285380274899124224"),
|
||
TEXT("2475880078570760549798248448"),
|
||
TEXT("4951760157141521099596496896"),
|
||
TEXT("9903520314283042199192993792"),
|
||
TEXT("19807040628566084398385987584"),
|
||
TEXT("39614081257132168796771975168"),
|
||
TEXT("79228162514264337593543950336"),
|
||
TEXT("158456325028528675187087900672"),
|
||
TEXT("316912650057057350374175801344"),
|
||
TEXT("633825300114114700748351602688"),
|
||
TEXT("1267650600228229401496703205376"),
|
||
TEXT("2535301200456458802993406410752"),
|
||
TEXT("5070602400912917605986812821504"),
|
||
TEXT("10141204801825835211973625643008"),
|
||
TEXT("20282409603651670423947251286016"),
|
||
TEXT("40564819207303340847894502572032"),
|
||
TEXT("81129638414606681695789005144064"),
|
||
TEXT("162259276829213363391578010288128"),
|
||
TEXT("324518553658426726783156020576256"),
|
||
TEXT("649037107316853453566312041152512"),
|
||
TEXT("1298074214633706907132624082305024"),
|
||
TEXT("2596148429267413814265248164610048"),
|
||
TEXT("5192296858534827628530496329220096"),
|
||
TEXT("10384593717069655257060992658440192"),
|
||
TEXT("20769187434139310514121985316880384"),
|
||
TEXT("41538374868278621028243970633760768"),
|
||
TEXT("83076749736557242056487941267521536"),
|
||
TEXT("166153499473114484112975882535043072"),
|
||
TEXT("332306998946228968225951765070086144"),
|
||
TEXT("664613997892457936451903530140172288"),
|
||
TEXT("1329227995784915872903807060280344576"),
|
||
TEXT("2658455991569831745807614120560689152"),
|
||
TEXT("5316911983139663491615228241121378304"),
|
||
TEXT("10633823966279326983230456482242756608"),
|
||
TEXT("21267647932558653966460912964485513216"),
|
||
TEXT("42535295865117307932921825928971026432"),
|
||
TEXT("85070591730234615865843651857942052864"),
|
||
TEXT("170141183460469231731687303715884105728"),
|
||
};
|
||
if (pow <= 127) {
|
||
return pow2[pow];
|
||
}
|
||
return TEXT("");
|
||
}
|
||
|
||
TCHAR *safe_strdup(const TCHAR *str) {
|
||
TCHAR *ret;
|
||
|
||
if (!str) {
|
||
return nullptr;
|
||
}
|
||
|
||
ret = _strdup(str);
|
||
if (!ret) {
|
||
SPDLOG_ERROR(TEXT("Memory allocation string failure:{0}"), str);
|
||
return nullptr;
|
||
}
|
||
|
||
return ret;
|
||
}
|
||
|
||
static int vasprintf(TCHAR **strp, const TCHAR *fmt, va_list ap) {
|
||
// _vscprintf tells you how big the buffer needs to be
|
||
const int len = _vscprintf(fmt, ap);
|
||
if (len == -1) {
|
||
return -1;
|
||
}
|
||
const size_t size = static_cast<size_t>(len) + 1;
|
||
const auto str = static_cast<TCHAR *>(malloc(size));
|
||
if (!str) {
|
||
return -1;
|
||
}
|
||
// _vsprintf_s is the "secure" version of vsprintf
|
||
const int r = vsprintf_s(str, len + 1, fmt, ap);
|
||
if (r == -1) {
|
||
free(str);
|
||
return -1;
|
||
}
|
||
*strp = str;
|
||
return r;
|
||
}
|
||
|
||
static int asprintf(TCHAR **strp, const TCHAR *fmt, ...) {
|
||
va_list ap;
|
||
va_start(ap, fmt);
|
||
const int r = vasprintf(strp, fmt, ap);
|
||
va_end(ap);
|
||
return r;
|
||
}
|
||
|
||
static int bit_count(unsigned int i) {
|
||
int c = 0;
|
||
unsigned int seen_one = 0;
|
||
|
||
while (i > 0) {
|
||
if (i & 1) {
|
||
seen_one = 1;
|
||
c++;
|
||
} else {
|
||
if (seen_one) {
|
||
return -1;
|
||
}
|
||
}
|
||
i >>= 1;
|
||
}
|
||
|
||
return c;
|
||
}
|
||
|
||
/**
|
||
* @brief creates a netmask from a specified number of bits
|
||
* This function converts a prefix length to a netmask. As CIDR (classless
|
||
* internet domain internet domain routing) has taken off, more an more IP
|
||
* addresses are being specified in the format address/prefix
|
||
* (i.e. 192.168.2.3/24, with a corresponding netmask 255.255.255.0). If you
|
||
* need to see what netmask corresponds to the prefix part of the address, this
|
||
* is the function. See also @ref mask2prefix.
|
||
* @param prefix prefix is the number of bits to create a mask for.
|
||
* @return a network mask, in network byte order.
|
||
*/
|
||
unsigned int prefix2mask(int prefix) {
|
||
if (prefix) {
|
||
return htonl(~((1 << (32 - prefix)) - 1));
|
||
} else {
|
||
return htonl(0);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief calculates the number of bits masked off by a netmask.
|
||
* This function calculates the significant bits in an IP address as specified by
|
||
* a netmask. See also @ref prefix2mask.
|
||
* @param mask is the netmask, specified as an struct in_addr in network byte order.
|
||
* @return the number of significant bits.
|
||
*/
|
||
int mask2prefix(IN_ADDR mask) {
|
||
return bit_count(ntohl(mask.s_addr));
|
||
}
|
||
|
||
static int ipv4_mask_to_int(const char *prefix) {
|
||
int ret;
|
||
IN_ADDR in;
|
||
|
||
ret = inet_pton(AF_INET, prefix, &in);
|
||
if (ret == 0) {
|
||
return -1;
|
||
}
|
||
|
||
return mask2prefix(in);
|
||
}
|
||
|
||
/**
|
||
* @brief calculate broadcast address given an IP address and a prefix length.
|
||
* @param addr an IP address in network byte order.
|
||
* @param prefix a prefix length.
|
||
* @return the calculated broadcast address for the network, in network byte order.
|
||
*/
|
||
static IN_ADDR calc_broadcast(IN_ADDR addr, int prefix) {
|
||
IN_ADDR mask;
|
||
IN_ADDR broadcast;
|
||
|
||
mask.s_addr = prefix2mask(prefix);
|
||
|
||
memset(&broadcast, 0, sizeof(broadcast));
|
||
broadcast.s_addr = (addr.s_addr & mask.s_addr) | ~mask.s_addr;
|
||
return broadcast;
|
||
}
|
||
|
||
/**
|
||
* @brief calculates the network address for a specified address and prefix.
|
||
* @param addr an IP address, in network byte order
|
||
* @param prefix the network prefix
|
||
* @return the base address of the network that addr is associated with, in
|
||
* network byte order.
|
||
*/
|
||
static IN_ADDR calc_network(IN_ADDR addr, int prefix) {
|
||
IN_ADDR mask;
|
||
IN_ADDR network;
|
||
|
||
mask.s_addr = prefix2mask(prefix);
|
||
|
||
memset(&network, 0, sizeof(network));
|
||
network.s_addr = addr.s_addr & mask.s_addr;
|
||
return network;
|
||
}
|
||
|
||
static TCHAR *ipv4_prefix_to_hosts(TCHAR *hosts, unsigned hosts_size, unsigned prefix) {
|
||
if (prefix >= 31) {
|
||
StringCbPrintf(hosts, hosts_size, TEXT("%s"), p2_table(32 - prefix));
|
||
} else {
|
||
unsigned int tmp;
|
||
tmp = (1 << (32 - prefix)) - 2;
|
||
StringCbPrintf(hosts, hosts_size, TEXT("%u"), tmp);
|
||
}
|
||
return hosts;
|
||
}
|
||
|
||
static int str_to_prefix(int *ipv6, const char *prefixStr, unsigned fix) {
|
||
int prefix;
|
||
|
||
if (!(*ipv6) && strchr(prefixStr, '.')) { /* prefix is 255.x.x.x */
|
||
prefix = ipv4_mask_to_int(prefixStr);
|
||
} else {
|
||
prefix = strtol(prefixStr, nullptr, 10);
|
||
}
|
||
|
||
if (fix && (prefix > 32 && !(*ipv6))) {
|
||
*ipv6 = 1;
|
||
}
|
||
|
||
if (prefix < 0 || (((*ipv6) && prefix > 128) || (!(*ipv6) && prefix > 32))) {
|
||
return -1;
|
||
}
|
||
return prefix;
|
||
}
|
||
|
||
static int GetIpV4Info(const TCHAR *pIpStr, int prefix, PIP_INFO pInfo, unsigned int flags) {
|
||
IN_ADDR ip, netmask, network, broadcast, minhost, maxhost;
|
||
TCHAR namebuf[INET_ADDRSTRLEN + 1];
|
||
TCHAR *ipStr = _strdup(pIpStr);
|
||
|
||
memset(pInfo, 0, sizeof(*pInfo));
|
||
|
||
if (inet_pton(AF_INET, ipStr, &ip) <= 0) {
|
||
SPDLOG_ERROR(TEXT("ipcalc: bad IPv4 address: {0}"), ipStr);
|
||
free(ipStr);
|
||
return -ERR_UN_SUPPORT;
|
||
}
|
||
|
||
/* Handle CIDR entries such as 172/8 */
|
||
if (prefix >= 0) {
|
||
auto tmp = const_cast<TCHAR *>(ipStr);
|
||
int i;
|
||
|
||
for (i = 3; i > 0; i--) {
|
||
tmp = strchr(tmp, '.');
|
||
if (!tmp) {
|
||
break;
|
||
} else {
|
||
tmp++;
|
||
}
|
||
}
|
||
|
||
tmp = nullptr;
|
||
for (; i > 0; i--) {
|
||
if (asprintf(&tmp, "%s.0", ipStr) == -1) {
|
||
SPDLOG_ERROR(TEXT("Memory allocation failure"));
|
||
free(ipStr);
|
||
return -ERR_MALLOC_MEMORY;
|
||
}
|
||
ipStr = tmp;
|
||
}
|
||
} else { // assume good old days classful Internet
|
||
prefix = 32;
|
||
}
|
||
|
||
if (prefix > 32) {
|
||
SPDLOG_ERROR(TEXT("ipcalc: bad IPv4 prefix: {0}"), prefix);
|
||
free(ipStr);
|
||
return -ERR_UN_SUPPORT;
|
||
}
|
||
|
||
if (inet_ntop(AF_INET, &ip, namebuf, sizeof(namebuf)) == 0) {
|
||
SPDLOG_ERROR(TEXT("ipcalc: error calculating the IPv4 network"));
|
||
free(ipStr);
|
||
return -ERR_UN_SUPPORT;
|
||
}
|
||
pInfo->ip = safe_strdup(namebuf);
|
||
|
||
netmask.s_addr = prefix2mask(prefix);
|
||
memset(namebuf, '\0', sizeof(namebuf));
|
||
|
||
if (inet_ntop(AF_INET, &netmask, namebuf, INET_ADDRSTRLEN) == nullptr) {
|
||
SPDLOG_ERROR(TEXT("inet_ntop error"));
|
||
free(ipStr);
|
||
return -ERR_UN_SUPPORT;
|
||
}
|
||
pInfo->netmask = safe_strdup(namebuf);
|
||
pInfo->prefix = prefix;
|
||
|
||
broadcast = calc_broadcast(ip, prefix);
|
||
|
||
memset(namebuf, '\0', sizeof(namebuf));
|
||
if (inet_ntop(AF_INET, &broadcast, namebuf, INET_ADDRSTRLEN) == nullptr) {
|
||
SPDLOG_ERROR(TEXT("inet_ntop error"));
|
||
free(ipStr);
|
||
return -ERR_UN_SUPPORT;
|
||
}
|
||
pInfo->broadcast = safe_strdup(namebuf);
|
||
|
||
network = calc_network(ip, prefix);
|
||
|
||
memset(namebuf, '\0', sizeof(namebuf));
|
||
if (inet_ntop(AF_INET, &network, namebuf, INET_ADDRSTRLEN) == nullptr) {
|
||
SPDLOG_ERROR(TEXT("inet_ntop error"));
|
||
free(ipStr);
|
||
return -ERR_UN_SUPPORT;
|
||
}
|
||
|
||
pInfo->network = safe_strdup(namebuf);
|
||
|
||
if (prefix < 32) {
|
||
memcpy(&minhost, &network, sizeof(minhost));
|
||
|
||
if (prefix <= 30) {
|
||
minhost.s_addr = htonl(ntohl(minhost.s_addr) | 1);
|
||
}
|
||
if (inet_ntop(AF_INET, &minhost, namebuf, INET_ADDRSTRLEN) == nullptr) {
|
||
SPDLOG_ERROR(TEXT("inet_ntop error"));
|
||
free(ipStr);
|
||
return -ERR_UN_SUPPORT;
|
||
}
|
||
pInfo->hostmin = safe_strdup(namebuf);
|
||
|
||
memcpy(&maxhost, &network, sizeof(minhost));
|
||
maxhost.s_addr |= ~netmask.s_addr;
|
||
if (prefix <= 30) {
|
||
maxhost.s_addr = htonl(ntohl(maxhost.s_addr) - 1);
|
||
}
|
||
if (inet_ntop(AF_INET, &maxhost, namebuf, sizeof(namebuf)) == 0) {
|
||
SPDLOG_ERROR(TEXT("ipcalc: error calculating the IPv4 network"));
|
||
free(ipStr);
|
||
return -ERR_UN_SUPPORT;
|
||
}
|
||
|
||
pInfo->hostmax = safe_strdup(namebuf);
|
||
} else {
|
||
pInfo->hostmin = pInfo->network;
|
||
pInfo->hostmax = pInfo->network;
|
||
}
|
||
|
||
ipv4_prefix_to_hosts(pInfo->hosts, sizeof(pInfo->hosts), prefix);
|
||
|
||
free(ipStr);
|
||
return ERR_SUCCESS;
|
||
}
|
||
|
||
/**
|
||
* @brief 计算 IPv4 网络信息
|
||
* @param[in] pIpStr IPv4 网络信息 '/' 分割,支持CIDR以及子网掩码 example: 192.168.1.32/24, 192.168.1.32/255.255.255.0
|
||
* @param[out] pInfo 计算结果
|
||
* @return 0: 成功, 小于0 失败 @see USER_ERRNO
|
||
* - -ERR_INPUT_PARAMS 输入参数错误
|
||
* - -ERR_UN_SUPPORT 不支持的格式转换
|
||
* - -ERR_MALLOC_MEMORY 分配内存失败
|
||
* - ERR_SUCCESS 成功
|
||
*/
|
||
int GetIpV4InfoFromCIDR(const TCHAR *pIpStr, PIP_INFO pInfo) {
|
||
int ret, prefix, familyIPv6 = 0;
|
||
TCHAR *prefixStr;
|
||
TCHAR *ipStr = _strdup(pIpStr);
|
||
|
||
if (pIpStr == nullptr || lstrlen(pIpStr) < 8) {
|
||
SPDLOG_ERROR(TEXT("Input pIpStr format error: {}."), pIpStr);
|
||
return -ERR_INPUT_PARAMS;
|
||
}
|
||
|
||
if (pInfo == nullptr) {
|
||
SPDLOG_ERROR(TEXT("Input pInfo is NULL."));
|
||
return -ERR_INPUT_PARAMS;
|
||
}
|
||
|
||
if (strchr(ipStr, '/') != nullptr) {
|
||
prefixStr = static_cast<TCHAR *>(strchr(ipStr, '/'));
|
||
*prefixStr = '\0'; /* fix up ipStr */
|
||
prefixStr++;
|
||
} else {
|
||
SPDLOG_ERROR(TEXT("Input pIpStr isn't CIDR format: {}."), pIpStr);
|
||
free(ipStr);
|
||
return -ERR_INPUT_PARAMS;
|
||
}
|
||
|
||
if (strchr(prefixStr, '.') != nullptr) {
|
||
prefix = ipv4_mask_to_int(prefixStr);
|
||
} else {
|
||
prefix = str_to_prefix(&familyIPv6, prefixStr, 0);
|
||
}
|
||
|
||
ret = GetIpV4Info(ipStr, prefix, pInfo, 0);
|
||
|
||
free(ipStr);
|
||
return ret;
|
||
}
|
||
|
||
/**
|
||
* @brief 计算 IPv4 网络信息
|
||
* @param[in] pIpStr IPv4 地址
|
||
* @param[in] pNetmask IPv4子网掩码
|
||
* @param[out] pInfo 计算结果
|
||
* @return 成功, 小于0 失败 @see USER_ERRNO
|
||
* - -ERR_INPUT_PARAMS 输入参数错误
|
||
* - -ERR_UN_SUPPORT 不支持的格式转换
|
||
* - -ERR_MALLOC_MEMORY 分配内存失败
|
||
* - ERR_SUCCESS 成功
|
||
*/
|
||
int GetIpV4InfoFromNetmask(const TCHAR *pIpStr, const TCHAR *pNetmask, PIP_INFO pInfo) {
|
||
int prefix;
|
||
|
||
if (pIpStr == nullptr || lstrlen(pIpStr) < 8) {
|
||
SPDLOG_ERROR(TEXT("Input pIpStr format error: {}."), pIpStr);
|
||
return -ERR_INPUT_PARAMS;
|
||
}
|
||
|
||
if (pNetmask == nullptr || lstrlen(pNetmask) < 8) {
|
||
SPDLOG_ERROR(TEXT("Input pNetmask format error: {}."), pNetmask);
|
||
return -ERR_INPUT_PARAMS;
|
||
}
|
||
|
||
if (pInfo == nullptr) {
|
||
SPDLOG_ERROR(TEXT("Input pInfo is NULL."));
|
||
return -ERR_INPUT_PARAMS;
|
||
}
|
||
|
||
prefix = ipv4_mask_to_int(pNetmask);
|
||
return GetIpV4Info(pIpStr, prefix, pInfo, 0);
|
||
} |