From 85d97f05f6f8e4c1d73cd2bb096806839d16f3b0 Mon Sep 17 00:00:00 2001
From: gcontini <1121667+gcontini@users.noreply.github.com>
Date: 周六, 31 10月 2020 10:27:16 +0800
Subject: [PATCH] Pc identifier Mac address issues #108 & #107

---
 src/library/os/windows/network.cpp |  146 +++++++++++++++++++++++++++++++++---------------
 1 files changed, 100 insertions(+), 46 deletions(-)

diff --git a/src/library/os/windows/network.cpp b/src/library/os/windows/network.cpp
index f3ccd2d..5116d39 100644
--- a/src/library/os/windows/network.cpp
+++ b/src/library/os/windows/network.cpp
@@ -9,26 +9,34 @@
  * Responsibility of this module is to fill OsAdapterInfo structures, in a predictable way (skip loopback/vpn interfaces)
  */
 
+#define _CRTDBG_MAP_ALLOC
+#define NOMINMAX
+
 #ifdef _MSC_VER
 #include <Windows.h>
 #endif
 #include <iphlpapi.h>
 #include <unordered_map>
 #include <stdio.h>
-//#pragma comment(lib, "IPHLPAPI.lib")
+#include <algorithm>
+#include <cctype>
+#pragma comment(lib, "IPHLPAPI.lib")
 
 #include "../../base/string_utils.h"
 #include "../../base/logger.h"
 #include "../network.hpp"
 
+#define MALLOC(x) HeapAlloc(GetProcessHeap(), 0, (x))
+#define FREE(x) HeapFree(GetProcessHeap(), 0, (x))
+
 namespace license {
 namespace os {
 using namespace std;
 
-static int translate(char ipStringIn[16], unsigned char ipv4[4]) {
+static int translate(const char ipStringIn[16], unsigned char ipv4[4]) {
 	size_t index = 0;
 
-	char *str2 = ipStringIn; /* save the pointer */
+	const char *str2 = ipStringIn; /* save the pointer */
 	while (*str2) {
 		if (isdigit((unsigned char)*str2)) {
 			ipv4[index] *= 10;
@@ -40,71 +48,117 @@
 	}
 	return 0;
 }
-/**
+
+int score(const OsAdapterInfo &a) {
+	int score = 0;
+	bool allzero = true;
+	const char *bads[] = {"virtual", "ppp", "tunnel", "vpn"};
+	const char *goods[] = {"realtek", "intel", "wireless"};
+
+	for (int i = 0; i < sizeof(a.description) && allzero; i++) {
+		allzero = allzero && (a.description[i] == 0);
+	}
+	if (!allzero) {
+		score++;
+	}
+
+	string descr=string(a.description);
+	std::transform(descr.begin(), descr.end(), descr.begin(), [](unsigned char c) { return std::tolower(c); });
+	for (auto bad: bads) {
+		score += descr.find(bad) == std::string::npos ? 1 : -1;
+	}
+	for (auto good : goods) {
+		score += descr.find(good) == std::string::npos ? -1 : 1;
+	}
+	return score;
+}
+
+bool cmp(const OsAdapterInfo &a, const OsAdapterInfo &b) { return score(a) >= score(b); }
+	/**
  *
  * @param adapterInfos
  * @param adapter_info_size
  * @return
  */
 FUNCTION_RETURN getAdapterInfos(vector<OsAdapterInfo> &adapterInfos) {
-	unordered_map<string, OsAdapterInfo> adapterByName;
+	vector<OsAdapterInfo> tmpAdapters;
 	FUNCTION_RETURN f_return = FUNC_RET_OK;
 	DWORD dwStatus;
-	int adapter_info_size;
-	PIP_ADAPTER_INFO pAdapterInfo;
-	DWORD dwBufLen = sizeof(IP_ADAPTER_INFO);
 
-	unsigned int i = 3;
-	do {
-		pAdapterInfo = (PIP_ADAPTER_INFO)malloc(dwBufLen);
-		dwStatus = GetAdaptersInfo(	 // Call GetAdapterInfo
-			pAdapterInfo,  // [out] buffer to receive data
-			&dwBufLen  // [in] size of receive data buffer
-		);
-		if (dwStatus != NO_ERROR && pAdapterInfo != nullptr) {
-			free(pAdapterInfo);
-			pAdapterInfo = nullptr;
-		}
-	} while (dwStatus == ERROR_BUFFER_OVERFLOW && i-- > 0);
+	ULONG ulOutBufLen = sizeof(IP_ADAPTER_INFO) *10;
+	IP_ADAPTER_INFO *pAdapterInfo = (IP_ADAPTER_INFO *)MALLOC(sizeof(IP_ADAPTER_INFO) * 10);
 
-	if (dwStatus == ERROR_BUFFER_OVERFLOW) {
+	if (pAdapterInfo == nullptr) {
 		return FUNC_RET_ERROR;
 	}
 
-	adapter_info_size = dwBufLen / sizeof(IP_ADAPTER_INFO);
-	if (adapter_info_size == 0) {
-		return FUNC_RET_NOT_AVAIL;
+	dwStatus = GetAdaptersInfo(
+		pAdapterInfo,  // [out] buffer to receive data
+		&ulOutBufLen  // [in] size of receive data buffer
+	);
+
+	// Incase the buffer was too small, reallocate with the returned dwBufLen
+	if (dwStatus == ERROR_BUFFER_OVERFLOW) {
+		FREE(pAdapterInfo);
+		pAdapterInfo = (IP_ADAPTER_INFO *)MALLOC(ulOutBufLen);
+
+		// Will only fail if buffer cannot be allocated (out of memory)
+		if (pAdapterInfo == nullptr) {
+			return FUNC_RET_BUFFER_TOO_SMALL;
+		}
+
+		dwStatus = GetAdaptersInfo(	 // Call GetAdapterInfo
+			pAdapterInfo,  // [out] buffer to receive data
+			&ulOutBufLen  // [in] size of receive data buffer
+		);
+
+		switch (dwStatus) {
+			case NO_ERROR:
+				break;
+
+			case ERROR_BUFFER_OVERFLOW:
+				FREE(pAdapterInfo);
+				return FUNC_RET_BUFFER_TOO_SMALL;
+
+			default:
+				FREE(pAdapterInfo);
+				return FUNC_RET_ERROR;
+		}
 	}
 
-	PIP_ADAPTER_INFO pAdapter = pAdapterInfo;
-	i = 0;
+	IP_ADAPTER_INFO* pAdapter = pAdapterInfo;
 	FUNCTION_RETURN result = FUNC_RET_OK;
 	while (pAdapter) {
-		OsAdapterInfo ai = {};
-		strncpy(ai.description, pAdapter->Description,
-				min((int)sizeof(ai.description), MAX_ADAPTER_DESCRIPTION_LENGTH));
-		memcpy(ai.mac_address, pAdapter->Address, 8);
-		translate(pAdapter->IpAddressList.IpAddress.String, ai.ipv4_address);
-		ai.type = IFACE_TYPE_ETHERNET;
-		i++;
-		pAdapter = pAdapter->Next;
-		if (i == adapter_info_size) {
-			result = FUNC_RET_BUFFER_TOO_SMALL;
-			break;
+		if (pAdapter->Type == MIB_IF_TYPE_ETHERNET) {
+			OsAdapterInfo ai = {};
+			LOG_DEBUG("Ethernet found %s, %s, mac_l: %d", pAdapter->AdapterName, pAdapter->Description, pAdapter->AddressLength);
+			if (pAdapter->AddressLength > 0) {
+				bool allzero = true;
+				const size_t size_to_be_copied = std::min(sizeof(ai.mac_address), (size_t)pAdapter->AddressLength);
+				for (int i = 0; i < size_to_be_copied && allzero; i++) {
+					allzero = allzero && (pAdapter->Address[i] == 0);
+				}
+				if (!allzero) {
+					strncpy(ai.description, pAdapter->Description,
+						min(sizeof(ai.description) - 1, (size_t)MAX_ADAPTER_DESCRIPTION_LENGTH));
+					memcpy(ai.mac_address, pAdapter->Address, size_to_be_copied);
+					translate(pAdapter->IpAddressList.IpAddress.String, ai.ipv4_address);
+					ai.type = IFACE_TYPE_ETHERNET;
+					tmpAdapters.push_back(ai);
+				}
+			}
 		}
-		adapterByName[string(ai.description)] = ai;
+		pAdapter = pAdapter->Next;
 	}
-	free(pAdapterInfo);
+	if (pAdapterInfo!=nullptr) {
+		FREE(pAdapterInfo);
+	}
 
-	// FIXME sort by eth , enps, wlan
-	if (adapterByName.size() == 0) {
+	if (tmpAdapters.size() == 0) {
 		f_return = FUNC_RET_NOT_AVAIL;
 	} else {
-		f_return = FUNC_RET_OK;
-		adapterInfos.reserve(adapterByName.size());
-		for (auto &it : adapterByName) {
-			adapterInfos.push_back(it.second);
-		}
+		std::sort(tmpAdapters.begin(), tmpAdapters.end(), cmp);
+		adapterInfos = std::move(tmpAdapters);
 	}
 	return f_return;
 }

--
Gitblit v1.9.1