Compliance: Server-side anti-proxy for trail authorization
This commit is contained in:
@@ -820,6 +820,8 @@ BEGIN_MESSAGE_MAP(CMy2015RemoteDlg, CDialogEx)
|
||||
ON_MESSAGE(WM_SHOWMESSAGE, OnShowMessage)
|
||||
ON_MESSAGE(WM_SHOWNOTIFY, OnShowNotify)
|
||||
ON_MESSAGE(WM_SHOWERRORMSG, OnShowErrMessage)
|
||||
ON_MESSAGE(WM_TRIAL_RTT_ABUSE, OnTrialRttAbuse)
|
||||
ON_MESSAGE(WM_TRIAL_WAN_IP_ABUSE, OnTrialWanIpAbuse)
|
||||
ON_MESSAGE(WM_INJECT_SHELLCODE, InjectShellcode)
|
||||
ON_MESSAGE(WM_ANTI_BLACKSCREEN, AntiBlackScreen)
|
||||
ON_MESSAGE(WM_SHARE_CLIENT, ShareClient)
|
||||
@@ -1574,6 +1576,52 @@ VOID CMy2015RemoteDlg::ShowMessage(CString strType, CString strMsg)
|
||||
m_StatusBar.SetPaneText(0,strStatusMsg); //在状态条上显示文字
|
||||
}
|
||||
|
||||
// 试用版 IP 段触发:OnAccept 发现入站连接对端是公网 IP(已透过 Proxy Protocol v2 解出真实 IP)。
|
||||
// 与 OnTrialRttAbuse 共用 IOCPServer::s_TrialAbuseWarned latch,本函数每进程最多调一次。
|
||||
LRESULT CMy2015RemoteDlg::OnTrialWanIpAbuse(WPARAM wParam, LPARAM lParam)
|
||||
{
|
||||
CString* ip = (CString*)wParam;
|
||||
CString detail;
|
||||
detail.FormatL("入站公网 IP=%s (Proxy Protocol 真实 IP 或 raw TCP 对端)",
|
||||
ip ? (LPCTSTR)*ip : _T("?"));
|
||||
ShowMessage(_TR("入站告警"), detail);
|
||||
|
||||
CString msg;
|
||||
msg.FormatL(
|
||||
"检测到入站连接来自公网 IP:%s\r\n\r\n"
|
||||
"试用版仅供 LAN 内自用,跨网使用属于违反授权条款。\r\n"
|
||||
"如需跨网远控,请向发行方申请正式授权。\r\n\r\n"
|
||||
"详细记录见消息列表与运行日志。",
|
||||
ip ? (LPCTSTR)*ip : _T("?"));
|
||||
THIS_APP->MessageBox(msg, _TR("试用版 LAN-only 限制"), MB_OK | MB_ICONWARNING | MB_TOPMOST);
|
||||
if (ip) delete ip;
|
||||
return S_OK;
|
||||
}
|
||||
|
||||
// 试用版反代理触发后的主窗口处理:写日志列表 + 弹一次模态框(前面已 latch,本函数每进程只会被调一次)。
|
||||
// 不在 IOCPServer 的 RTT 轮询线程里直接弹框,避免阻塞后续采样。
|
||||
LRESULT CMy2015RemoteDlg::OnTrialRttAbuse(WPARAM wParam, LPARAM lParam)
|
||||
{
|
||||
uint32_t clientIdLow = (uint32_t)wParam;
|
||||
int medianMs = (int)lParam;
|
||||
CString detail;
|
||||
// 纯英文格式串,不进翻译表
|
||||
detail.Format(_T("ClientID(low32)=%u median RTT=%d ms threshold=%d ms"),
|
||||
clientIdLow, medianMs, (int)TcpRttBreachDetector::RTT_THRESHOLD_MS);
|
||||
ShowMessage(_TR("反代理告警"), detail);
|
||||
|
||||
CString msg;
|
||||
msg.FormatL(
|
||||
"检测到可疑连接:内核 RTT 中位数 %d ms,超出阈值 %d ms。\r\n\r\n"
|
||||
"持续偏高的 RTT 提示该连接可能经由代理 / VPN / 隧道中转。\r\n"
|
||||
"试用版仅供 LAN 内自用,跨网使用属于违反授权条款。\r\n\r\n"
|
||||
"如需跨网远控,请向发行方申请正式授权。\r\n"
|
||||
"详细记录见消息列表与运行日志。",
|
||||
medianMs, (int)TcpRttBreachDetector::RTT_THRESHOLD_MS);
|
||||
THIS_APP->MessageBox(msg, _TR("试用版 LAN-only 限制"), MB_OK | MB_ICONWARNING | MB_TOPMOST);
|
||||
return S_OK;
|
||||
}
|
||||
|
||||
LRESULT CMy2015RemoteDlg::OnShowErrMessage(WPARAM wParam, LPARAM lParam)
|
||||
{
|
||||
CString* text = (CString*)wParam;
|
||||
|
||||
@@ -536,6 +536,8 @@ public:
|
||||
afx_msg void OnNMCustomdrawOnline(NMHDR* pNMHDR, LRESULT* pResult);
|
||||
afx_msg void OnOnlineRunAsAdmin();
|
||||
afx_msg LRESULT OnShowErrMessage(WPARAM wParam, LPARAM lParam);
|
||||
afx_msg LRESULT OnTrialRttAbuse(WPARAM wParam, LPARAM lParam);
|
||||
afx_msg LRESULT OnTrialWanIpAbuse(WPARAM wParam, LPARAM lParam);
|
||||
afx_msg void OnMainWallet();
|
||||
afx_msg void OnMainNetwork();
|
||||
afx_msg void OnToolRcedit();
|
||||
|
||||
@@ -7,6 +7,68 @@
|
||||
#include <iostream>
|
||||
#include <ws2tcpip.h>
|
||||
|
||||
// 服务端 RTT 反代理(试用版执法)。声明在主对话框 cpp 中,无单独头文件。
|
||||
BOOL IsTrail(const std::string& passcode);
|
||||
|
||||
// ============================================================================
|
||||
// SIO_TCP_INFO 兼容性 shim
|
||||
//
|
||||
// SIO_TCP_INFO 自 Win10 1703 / Server 2016 起提供,对应的 SDK 头声明只在
|
||||
// NTDDI_VERSION >= NTDDI_WIN10_RS2 (0x0A000003) 时才可见。本项目当前
|
||||
// _WIN32_WINNT=0x0602 / NTDDI_VERSION=0x06020000(Win8),整体上调宏会
|
||||
// 波及其他模块,且会排除 Win8/8.1 用户。因此在此处本地声明常量与结构,
|
||||
// 运行时若 OS 不支持,WSAIoctl 会返回 WSAEOPNOTSUPP,由探测代码静默降级。
|
||||
//
|
||||
// 结构体字段顺序严格遵循 MS 公开的 TCP_INFO_v0 定义,不要随意调整。
|
||||
// ============================================================================
|
||||
#ifndef SIO_TCP_INFO
|
||||
#define SIO_TCP_INFO _WSAIORW(IOC_VENDOR, 39)
|
||||
#endif
|
||||
|
||||
typedef struct _TCP_INFO_v0_local {
|
||||
ULONG State; // TCPSTATE(枚举,按 4 字节读)
|
||||
ULONG Mss;
|
||||
ULONG64 ConnectionTimeMs;
|
||||
UCHAR TimestampsEnabled;
|
||||
UCHAR Pad_[3]; // 显式 padding,让 RttUs 落在 4 字节边界
|
||||
ULONG RttUs; // <-- 本文件唯一关心的字段
|
||||
ULONG MinRttUs;
|
||||
ULONG BytesInFlight;
|
||||
ULONG Cwnd;
|
||||
ULONG SndWnd;
|
||||
ULONG RcvWnd;
|
||||
ULONG RcvBuf;
|
||||
ULONG64 BytesOut;
|
||||
ULONG64 BytesIn;
|
||||
ULONG BytesReordered;
|
||||
ULONG BytesRetrans;
|
||||
ULONG FastRetrans;
|
||||
ULONG DupAcksIn;
|
||||
ULONG TimeoutEpisodes;
|
||||
UCHAR SynRetrans;
|
||||
} TCP_INFO_v0_local;
|
||||
|
||||
// 读取 socket 的内核测得 RTT。成功返回 0 并写入 *rttUs;失败返回 WSAGetLastError()。
|
||||
static int QuerySocketTcpRttUs(SOCKET s, uint32_t* rttUs)
|
||||
{
|
||||
TCP_INFO_v0_local info; ZeroMemory(&info, sizeof(info));
|
||||
DWORD ver = 0; // request v0
|
||||
DWORD bytesReturned = 0;
|
||||
int ret = WSAIoctl(s, SIO_TCP_INFO,
|
||||
&ver, sizeof(ver),
|
||||
&info, sizeof(info),
|
||||
&bytesReturned, NULL, NULL);
|
||||
if (ret == 0) {
|
||||
if (rttUs) *rttUs = info.RttUs;
|
||||
return 0;
|
||||
}
|
||||
return WSAGetLastError();
|
||||
}
|
||||
|
||||
// 全 server 进程级 latch,IP 段触发与 RTT 触发共用。多 server 实例(多端口监听)共享一份,
|
||||
// 任一先触发后其余 server 与其它触发路径不再重复弹框。
|
||||
std::atomic<bool> IOCPServer::s_TrialAbuseWarned{false};
|
||||
|
||||
// Proxy Protocol v2 签名 (12 字节)
|
||||
static const unsigned char PROXY_PROTOCOL_V2_SIGNATURE[12] = {
|
||||
0x0D, 0x0A, 0x0D, 0x0A, 0x00, 0x0D, 0x0A, 0x51, 0x55, 0x49, 0x54, 0x0A
|
||||
@@ -353,6 +415,13 @@ void IOCPServer::Destroy()
|
||||
|
||||
if (m_hKillEvent != NULL) {
|
||||
SetEvent(m_hKillEvent);
|
||||
// RTT 轮询线程要等它退出后再关 m_hKillEvent,否则线程仍在 WaitForSingleObject 上时
|
||||
// 关句柄是 UB。监听 / 工作线程是用 m_bTimeToKill 兜底的,原有时序不动。
|
||||
if (m_hRttThread != NULL) {
|
||||
WaitForSingleObject(m_hRttThread, 5000);
|
||||
SAFE_CLOSE_HANDLE(m_hRttThread);
|
||||
m_hRttThread = NULL;
|
||||
}
|
||||
SAFE_CLOSE_HANDLE(m_hKillEvent);
|
||||
m_hKillEvent = NULL;
|
||||
}
|
||||
@@ -414,7 +483,10 @@ UINT IOCPServer::StartServer(pfnNotifyProc NotifyProc, pfnOfflineProc OffProc, U
|
||||
m_nPort = uPort;
|
||||
m_NotifyProc = NotifyProc;
|
||||
m_OfflineProc = OffProc;
|
||||
m_hKillEvent = CreateEvent(NULL,FALSE,FALSE,NULL);
|
||||
// manual-reset:本进程内可能有多个等待者(ListenThread / RttPollThreadProc)。
|
||||
// 自动重置会让 SetEvent 只唤醒一个等待者,另一个要等自身 timeout(≤1s)。
|
||||
// 改 manual-reset 后所有等待者一次性醒来;本工程从无 ResetEvent 调用,无副作用。
|
||||
m_hKillEvent = CreateEvent(NULL,TRUE,FALSE,NULL);
|
||||
|
||||
if (m_hKillEvent==NULL) {
|
||||
return 1;
|
||||
@@ -507,6 +579,20 @@ UINT IOCPServer::StartServer(pfnNotifyProc NotifyProc, pfnOfflineProc OffProc, U
|
||||
|
||||
//启动工作线程 1 2
|
||||
InitializeIOCP();
|
||||
|
||||
// 试用版反代理 RTT 轮询(仅在主控自身为试用模式时启动)。
|
||||
// 检测信号来自内核 SIO_TCP_INFO,详见 IOCPServer.h 头部 / RttPollThreadProc 注释。
|
||||
{
|
||||
std::string pwd = THIS_CFG.GetStr("settings", "Password", "");
|
||||
m_bTrialMode = (IsTrail(pwd) == TRUE);
|
||||
}
|
||||
if (m_bTrialMode) {
|
||||
m_hRttThread = CreateThread(NULL, 0, RttPollThreadProc, (void*)this, 0, NULL);
|
||||
if (m_hRttThread == NULL) {
|
||||
Mprintf("[Compliance] RTT poll thread spawn failed (err=%lu); LANRttChecker (client-side) remains as fallback.\n",
|
||||
GetLastError());
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
@@ -930,6 +1016,97 @@ BOOL IOCPServer::OnClientPostSending(CONTEXT_OBJECT* ContextObject,ULONG ulCompl
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
// ============================================================================
|
||||
// 试用版反代理 —— 服务端 RTT 轮询线程
|
||||
//
|
||||
// 仅在主控自身处于试用模式(IsTrail(passcode) == TRUE)时由 StartServer 启动。
|
||||
// 1 Hz 遍历 m_ContextConnectionList,对每个活跃连接调 WSAIoctl(SIO_TCP_INFO) 取
|
||||
// 内核测得的纯网络 RTT,喂给 ctx->m_RttDetector。任一 detector 首次触发 →
|
||||
// 通过 s_TrialAbuseWarned latch 抢一次 PostMessage 给主窗口弹框;其余 detector
|
||||
// 仍照常运转(继续记日志),但不再重复弹框。
|
||||
//
|
||||
// 并发模型:对齐既有 IoRefCount / IsRemoved 模式 —— 持 m_cs snapshot 指针并
|
||||
// 引用计数 ++,锁外做 WSAIoctl + 写 atomic,最后引用计数 --。RemoveStaleContext
|
||||
// 会等 IoRefCount==0 才回收,无悬空指针。
|
||||
//
|
||||
// 不支持 SIO_TCP_INFO 的 OS(Win8 / Server 2012 等):首次探测命中
|
||||
// WSAEOPNOTSUPP 时打日志后线程自行退出;客户端 LANRttChecker 仍作为兜底。
|
||||
// ============================================================================
|
||||
DWORD IOCPServer::RttPollThreadProc(LPVOID lParam)
|
||||
{
|
||||
IOCPServer* This = (IOCPServer*)lParam;
|
||||
|
||||
while (!This->m_bTimeToKill) {
|
||||
DWORD waitRet = WaitForSingleObject(This->m_hKillEvent, 1000);
|
||||
if (waitRet == WAIT_OBJECT_0 || waitRet == WAIT_FAILED) break;
|
||||
if (This->m_bTimeToKill) break;
|
||||
|
||||
// —— 步骤 1:持锁快照 + 占引用 —— 锁外才做 WSAIoctl,避免阻塞其他 I/O
|
||||
std::vector<PCONTEXT_OBJECT> snap;
|
||||
EnterCriticalSection(&This->m_cs);
|
||||
for (POSITION pos = This->m_ContextConnectionList.GetHeadPosition(); pos != NULL; ) {
|
||||
PCONTEXT_OBJECT ctx = This->m_ContextConnectionList.GetNext(pos);
|
||||
if (!ctx) continue;
|
||||
if (ctx->IsRemoved.load(std::memory_order_acquire)) continue;
|
||||
ctx->IoRefCount.fetch_add(1, std::memory_order_acq_rel);
|
||||
snap.push_back(ctx);
|
||||
}
|
||||
LeaveCriticalSection(&This->m_cs);
|
||||
|
||||
// —— 步骤 2:OS 兼容性探测(一次性,借第一个真实连接做) —— 探测失败的 OS
|
||||
// 上整个线程不必再活,本次循环把已占的引用还掉就退出。
|
||||
if (!This->m_bSioTcpInfoProbed.load(std::memory_order_acquire) && !snap.empty()) {
|
||||
uint32_t probeRtt = 0;
|
||||
int err = QuerySocketTcpRttUs(snap[0]->sClientSocket, &probeRtt);
|
||||
if (err == WSAEOPNOTSUPP) {
|
||||
Mprintf("[Compliance] SIO_TCP_INFO not supported by OS (WSAEOPNOTSUPP); "
|
||||
"server-side RTT monitoring disabled. Client-side LANRttChecker remains active.\n");
|
||||
This->m_bSioTcpInfoSupported.store(false, std::memory_order_release);
|
||||
This->m_bSioTcpInfoProbed.store(true, std::memory_order_release);
|
||||
for (auto* c : snap) c->IoRefCount.fetch_sub(1, std::memory_order_acq_rel);
|
||||
break;
|
||||
}
|
||||
// 其它错误(如 WSAENOTCONN 短连接刚断)不视为 OS 问题,下一轮再试
|
||||
if (err == 0) {
|
||||
This->m_bSioTcpInfoSupported.store(true, std::memory_order_release);
|
||||
This->m_bSioTcpInfoProbed.store(true, std::memory_order_release);
|
||||
Mprintf("[Compliance] SIO_TCP_INFO probe OK; server-side anti-proxy RTT monitor armed "
|
||||
"(threshold=%d ms, trigger after >=%d consecutive median breaches @1Hz).\n",
|
||||
TcpRttBreachDetector::RTT_THRESHOLD_MS, TcpRttBreachDetector::BREACH_PERSIST_COUNT);
|
||||
}
|
||||
}
|
||||
|
||||
// —— 步骤 3:逐 ctx 取 RTT + 喂检测器 —— 同步释放引用
|
||||
for (auto* ctx : snap) {
|
||||
uint32_t rttUs = 0;
|
||||
int err = QuerySocketTcpRttUs(ctx->sClientSocket, &rttUs);
|
||||
if (err == 0 && rttUs > 0) {
|
||||
ctx->SetRttUs(rttUs);
|
||||
// RttUs 单位是微秒,转毫秒喂检测器
|
||||
int rttMs = (int)((rttUs + 500) / 1000);
|
||||
if (ctx->m_RttDetector.Feed(rttMs)) {
|
||||
// 本 ctx 首次触发:记日志(每个 ctx 都记,便于排查 abusive 来源);
|
||||
// 全 server 一次性 latch 决定要不要弹框
|
||||
Mprintf("[Compliance] !!! Trial-mode anti-proxy triggered: client=%llu IP=%s "
|
||||
"median RTT=%d ms (threshold=%d ms).\n",
|
||||
ctx->ID, ctx->GetPeerName().c_str(),
|
||||
ctx->m_RttDetector.TriggerMedianMs(),
|
||||
TcpRttBreachDetector::RTT_THRESHOLD_MS);
|
||||
bool expected = false;
|
||||
if (s_TrialAbuseWarned.compare_exchange_strong(expected, true) && This->m_hMainWnd) {
|
||||
// WPARAM 携带 abusive ctx 的 ClientID 低 32 位(仅用于展示),LPARAM 携带 medianMs
|
||||
PostMessageA(This->m_hMainWnd, WM_TRIAL_RTT_ABUSE,
|
||||
(WPARAM)(ctx->ID & 0xFFFFFFFF),
|
||||
(LPARAM)ctx->m_RttDetector.TriggerMedianMs());
|
||||
}
|
||||
}
|
||||
}
|
||||
ctx->IoRefCount.fetch_sub(1, std::memory_order_acq_rel);
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
DWORD IOCPServer::ListenThreadProc(LPVOID lParam) //监听线程
|
||||
{
|
||||
IOCPServer* This = (IOCPServer*)(lParam);
|
||||
@@ -1012,6 +1189,29 @@ void IOCPServer::OnAccept()
|
||||
}
|
||||
RecordConnection(clientIP);
|
||||
|
||||
// 试用版反代理 —— 入站 IP 段检测(即时触发,对合作型代理透明)
|
||||
//
|
||||
// 与 RttPollThreadProc 的 SIO_TCP_INFO 检测互补:RTT 测的是"我↔直接 TCP 对端",
|
||||
// 任何 TCP 终结型代理都能欺骗它;本检测用 Proxy Protocol v2 解出的真实 IP(若有)
|
||||
// 或 getpeername 的 raw IP 直接判私网段。
|
||||
// - 覆盖:直连 WAN、PP2 透出真实 IP 是公网
|
||||
// - 不覆盖:socat / 不发 PP2 的中转 —— 那种场景仍由客户端 LANRttChecker 兜底
|
||||
//
|
||||
// 性能:每个新连接走一次 IsPrivateIPv4Str(几个位运算),不放心跳路径,可忽略。
|
||||
// 不主动断开连接(与 RTT 路径一致仅告警),由运营商看到弹框后自行处置。
|
||||
// 详见 docs/Compliance_TechnicalMeasures.md(口径文档,可能比这里更新)。
|
||||
if (m_bTrialMode && !LANChecker::IsPrivateIPv4Str(clientIP)) {
|
||||
Mprintf("[Compliance] !!! Trial-mode WAN inbound: IP=%s (resolved via %s).\n",
|
||||
clientIP.c_str(),
|
||||
ContextObject->GetPeerName().empty() ? "getpeername" : "Proxy Protocol v2 or getpeername");
|
||||
bool expected = false;
|
||||
if (s_TrialAbuseWarned.compare_exchange_strong(expected, true) && m_hMainWnd) {
|
||||
// CString* 由 OnTrialWanIpAbuse handler 负责 delete,与 OnShowErrMessage 一致
|
||||
PostMessageA(m_hMainWnd, WM_TRIAL_WAN_IP_ABUSE,
|
||||
(WPARAM)new CString(clientIP.c_str()), 0);
|
||||
}
|
||||
}
|
||||
|
||||
ContextObject->wsaInBuf.buf = (char*)ContextObject->szBuffer;
|
||||
ContextObject->wsaInBuf.len = sizeof(ContextObject->szBuffer);
|
||||
|
||||
|
||||
@@ -78,9 +78,19 @@ protected:
|
||||
void LoadIPWhitelist();
|
||||
void LoadIPBlacklist();
|
||||
|
||||
// RTT 反代理(试用版执法)相关。详见 IOCPServer.cpp 中的实现注释。
|
||||
HANDLE m_hRttThread = NULL;
|
||||
bool m_bTrialMode = false; // StartServer 时根据 IsTrail(passcode) 缓存
|
||||
std::atomic<bool> m_bSioTcpInfoProbed{false}; // 是否已完成 OS 兼容性探测
|
||||
std::atomic<bool> m_bSioTcpInfoSupported{false}; // 探测结果(不支持则 RTT 线程会自行退出)
|
||||
// 全 server 进程一次性 latch,IP 段触发与 RTT 触发共用。任一先触发后另一者只记日志不再弹框,
|
||||
// 避免对运营商反复打扰;不影响每条 abusive 连接的独立日志。
|
||||
static std::atomic<bool> s_TrialAbuseWarned;
|
||||
|
||||
private:
|
||||
static DWORD WINAPI ListenThreadProc(LPVOID lParam);
|
||||
static DWORD WINAPI WorkThreadProc(LPVOID lParam);
|
||||
static DWORD WINAPI RttPollThreadProc(LPVOID lParam);
|
||||
|
||||
BOOL InitializeIOCP(VOID);
|
||||
VOID OnAccept();
|
||||
|
||||
@@ -8,6 +8,7 @@
|
||||
#include "Buffer.h"
|
||||
#define XXH_INLINE_ALL
|
||||
#include "common/xxhash.h"
|
||||
#include "common/LANChecker.h"
|
||||
#include <WS2tcpip.h>
|
||||
#include <common/ikcp.h>
|
||||
#include <atomic>
|
||||
@@ -378,6 +379,14 @@ public:
|
||||
std::atomic<int> IoRefCount{0}; // I/O 处理引用计数
|
||||
std::atomic<bool> IsRemoved{false}; // 标记是否已被标记为移除
|
||||
|
||||
// 内核测得的纯网络 RTT(μs),由 IOCPServer 的 RTT 轮询线程通过
|
||||
// WSAIoctl(SIO_TCP_INFO) 周期性写入;任何线程可通过 GetRttUs() 安全读取。
|
||||
// m_LastRttSampleMs == 0 表示从未成功采样过(OS 不支持或连接太短未轮询到)。
|
||||
std::atomic<uint32_t> m_RttUs{0};
|
||||
std::atomic<uint64_t> m_LastRttSampleMs{0};
|
||||
// 试用版反代理逐连接检测器。仅由 IOCPServer 的 RTT 轮询线程访问,免锁。
|
||||
TcpRttBreachDetector m_RttDetector;
|
||||
|
||||
// 子连接身份校验:客户端发 TOKEN_CONN_AUTH 通过验证后置位。
|
||||
// 主连接(走 TOKEN_LOGIN 流程)不参与此机制。当前阶段宽容(未通过也接受),
|
||||
// 仅作为标记供后续命令处理 / 未来收紧策略使用。
|
||||
@@ -517,9 +526,28 @@ public:
|
||||
IoRefCount.store(0, std::memory_order_release);
|
||||
// 复用对象池时清空校验状态
|
||||
m_bAuthenticated.store(false, std::memory_order_release);
|
||||
// 复用对象池时重置 RTT 相关状态,避免上一个连接的数据污染
|
||||
m_RttUs.store(0, std::memory_order_release);
|
||||
m_LastRttSampleMs.store(0, std::memory_order_release);
|
||||
m_RttDetector.Reset();
|
||||
}
|
||||
void SetAuthenticated(bool v) { m_bAuthenticated.store(v, std::memory_order_release); }
|
||||
bool IsAuthenticated() const { return m_bAuthenticated.load(std::memory_order_acquire); }
|
||||
|
||||
// 由 RTT 轮询线程写入。rttUs 为内核测得的纯网络 RTT(微秒)。
|
||||
void SetRttUs(uint32_t rttUs)
|
||||
{
|
||||
m_RttUs.store(rttUs, std::memory_order_release);
|
||||
m_LastRttSampleMs.store((uint64_t)GetTickCount64(), std::memory_order_release);
|
||||
}
|
||||
uint32_t GetRttUs() const { return m_RttUs.load(std::memory_order_acquire); }
|
||||
// 供 UI 展示用:μs → ms;样本超过 10 秒未更新(连接刚断/未轮询到)视为不可用,返回 -1。
|
||||
int GetRttMsForDisplay() const
|
||||
{
|
||||
uint64_t last = m_LastRttSampleMs.load(std::memory_order_acquire);
|
||||
if (last == 0 || (uint64_t)GetTickCount64() - last > 10000) return -1;
|
||||
return (int)((m_RttUs.load(std::memory_order_acquire) + 500) / 1000);
|
||||
}
|
||||
uint64_t GetAliveTime()const
|
||||
{
|
||||
return time(0) - OnlineTime;
|
||||
|
||||
@@ -1847,3 +1847,10 @@ IOCP
|
||||
快照=Snapshot
|
||||
预览=Preview
|
||||
主机列表预览图=Host List Thumbnails
|
||||
|
||||
入站告警=Inbound Alert
|
||||
反代理告警=Anti-Proxy Alert
|
||||
试用版 LAN-only 限制=Trial Version - LAN Only Restriction
|
||||
入站公网 IP=%s (Proxy Protocol 真实 IP 或 raw TCP 对端)=Inbound public IP=%s (resolved via Proxy Protocol v2 real IP or raw TCP peer)
|
||||
检测到入站连接来自公网 IP:%s\r\n\r\n试用版仅供 LAN 内自用,跨网使用属于违反授权条款。\r\n如需跨网远控,请向发行方申请正式授权。\r\n\r\n详细记录见消息列表与运行日志。=Inbound connection from public IP: %s\r\n\r\nTrial version is restricted to LAN-only usage; cross-network use violates the license terms.\r\nFor cross-network remote control, please obtain a commercial license from the publisher.\r\n\r\nSee the message list and runtime log for full details.
|
||||
检测到可疑连接:内核 RTT 中位数 %d ms,超出阈值 %d ms。\r\n\r\n持续偏高的 RTT 提示该连接可能经由代理 / VPN / 隧道中转。\r\n试用版仅供 LAN 内自用,跨网使用属于违反授权条款。\r\n\r\n如需跨网远控,请向发行方申请正式授权。\r\n详细记录见消息列表与运行日志。=Suspicious connection detected: kernel-measured RTT median %d ms exceeds the threshold of %d ms.\r\n\r\nA persistently elevated RTT suggests the connection is being relayed through a proxy / VPN / tunnel.\r\nTrial version is restricted to LAN-only usage; cross-network use violates the license terms.\r\n\r\nFor cross-network remote control, please obtain a commercial license from the publisher.\r\nSee the message list and runtime log for full details.
|
||||
|
||||
@@ -1838,3 +1838,10 @@ IOCP
|
||||
快照=快照
|
||||
预览=預覽
|
||||
主机列表预览图=主機列表預覽圖
|
||||
|
||||
入站告警=入站告警
|
||||
反代理告警=反代理告警
|
||||
试用版 LAN-only 限制=試用版 LAN-only 限制
|
||||
入站公网 IP=%s (Proxy Protocol 真实 IP 或 raw TCP 对端)=入站公網 IP=%s (Proxy Protocol 真實 IP 或 raw TCP 對端)
|
||||
检测到入站连接来自公网 IP:%s\r\n\r\n试用版仅供 LAN 内自用,跨网使用属于违反授权条款。\r\n如需跨网远控,请向发行方申请正式授权。\r\n\r\n详细记录见消息列表与运行日志。=檢測到入站連線來自公網 IP:%s\r\n\r\n試用版僅供 LAN 內自用,跨網使用屬於違反授權條款。\r\n如需跨網遠控,請向發行方申請正式授權。\r\n\r\n詳細記錄請見訊息列表與執行日誌。
|
||||
检测到可疑连接:内核 RTT 中位数 %d ms,超出阈值 %d ms。\r\n\r\n持续偏高的 RTT 提示该连接可能经由代理 / VPN / 隧道中转。\r\n试用版仅供 LAN 内自用,跨网使用属于违反授权条款。\r\n\r\n如需跨网远控,请向发行方申请正式授权。\r\n详细记录见消息列表与运行日志。=檢測到可疑連線:核心 RTT 中位數 %d ms,超出閾值 %d ms。\r\n\r\n持續偏高的 RTT 提示該連線可能經由代理 / VPN / 隧道中轉。\r\n試用版僅供 LAN 內自用,跨網使用屬於違反授權條款。\r\n\r\n如需跨網遠控,請向發行方申請正式授權。\r\n詳細記錄請見訊息列表與執行日誌。
|
||||
|
||||
@@ -103,6 +103,8 @@
|
||||
#define WM_OPENTERMINALDIALOG WM_USER+3033
|
||||
#define WM_PREVIEW_RESPONSE WM_USER+3034
|
||||
#define WM_PREVIEW_LOOP_CLOSED WM_USER+3035
|
||||
#define WM_TRIAL_RTT_ABUSE WM_USER+3036 // 试用版 RTT 反代理:服务端检测到滥用,通知主窗口弹框
|
||||
#define WM_TRIAL_WAN_IP_ABUSE WM_USER+3037 // 试用版 IP 段检测:OnAccept 发现入站为公网 IP,通知主窗口弹框
|
||||
|
||||
#ifdef _UNICODE
|
||||
#if defined _M_IX86
|
||||
|
||||
Reference in New Issue
Block a user