🚀 VMUX技术分析:OPENPPP2中的虚拟多路复用技术
🌟 一、技术目标
-
🔗 连接多路复用
通过单个或多个物理链路,承载多个逻辑TCP连接。
-
🚀 高性能传输
- 支持数据包乱序重组
- 实现动态流量控制(拥塞检测/加速切换)
- 高效内存管理(缓冲池复用)
-
🌐 跨平台调优
- Windows:QoS差分服务
- Linux:网络保护
-
🔒 连接管理
- 心跳保活
- 超时自动清理
- 防火墙策略集成
🏗️ 二、核心架构图解
1. 结构总览
VMUX核心 网络层 vmux_net 连接层 vmux_skt 基础库 vmux.h 链路管理 数据包调度 连接控制 TCP连接管理 流量控制 数据转发 类型定义 协程管理 内存分配
2. 物理链路与逻辑连接关系
应用层 逻辑层 核心引擎 物理链路层 调度 调度 调度 应用1 应用2 应用N 逻辑连接1 逻辑连接2 逻辑连接N 连接1 连接2 连接N VMUX 链路1 链路2 链路N
🧩 三、关键组件详解
📡 vmux_net
(网络层)
核心结构示意图
vmux_net +ContextPtr context_ +StrandPtr strand_ -vmux_skt_map skts_ // 连接映射表 -tx_packet_ssqueue tx_queue_ // 发送队列 -rx_packet_ssqueue rx_queue_ // 接收队列 -vmux_linklayer_vector rx_links_ // 物理链路集合 +add_linklayer() +connect_yield() +packet_input_unorder() +process_tx_all_packets() +update()
多物理链路承载机制
vmux_net -vmux_linklayer_vector rx_links_ // 物理链路集合 -vmux_linklayer_list tx_links_ // 发送就绪链路 +add_linklayer() -underlyin_sent() -process_tx_all_packets() vmux_linklayer +VirtualEthernetTcpipConnectionPtr connection +server_ptr server
示例:添加链路的代码片段
cpp
bool vmux_net::add_linklayer(
const VirtualEthernetTcpipConnectionPtr& connection,
vmux_linklayer_ptr& linklayer)
{
rx_links_.emplace_back(linklayer); // 添加接收链路
tx_links_.emplace_back(linklayer); // 添加发送链路
// 启动链路转发协程
ppp::coroutines::YieldContext::Spawn(
[this, linklayer](auto& y){
forwarding(linklayer, y);
});
}
🎯 动态链路选择策略
新数据包 选择链路 |轮询| 物理链路1 |最小延迟| 物理链路2 |最大带宽| 物理链路3
轮询 延迟 带宽 新数据包 选择调度策略 调度策略1 调度策略2 调度策略3 链路1 链路2 链路3
🔗 vmux_skt
(连接层)
类结构示意
vmux_skt +uint32_t connection_id_ -packet_queue rx_queue_ // 接收队列 -std::atomic forwarding_ // 转发状态 +accept() +connect() +forward_to_rx_socket() +rx_congestions() +tx_acceleration()
连接状态机
connect()调用 收到SYN_OK 超时/失败 流量正常 拥塞检测 拥塞解除 close()调用 资源释放 DISCONNECTED CONNECTING CONNECTED ACCELERATING CONGESTED CLOSING
流量控制示例
cpp
bool vmux_skt::rx_congestions(int64_t value) {
rx_congestions_ += value;
if (rx_congestions_ >= max_congestions && status_.rx_acceleration_) {
// 触发减速
mux_->post(cmd_acceleration, &FALSE);
} else if (rx_congestions_ <= 0 && !status_.rx_acceleration_) {
// 触发加速
mux_->post(cmd_acceleration, &TRUE);
}
}
📝 工作流程
1. 连接建立流程
Client vmux_net vmux_skt Server connect_yield() 分配connection_id 发送SYN 转发SYN 返回SYN_OK 连接成功 连接就绪 Client vmux_net vmux_skt Server
2. 数据传输流程
是 否 正常 数据输入 加速模式? 直接放入发送队列 进行拥塞检测 添加序列号 通过物理链路发送 乱序重组 交付应用层
🛠️ 六、关键技术点
1. 乱序包处理
- 使用有序映射队列存储乱序包
- 依据序列号(
seq
)填补间隙
cpp
if (status_.rx_ack_ == seq) {
// 当前包连续
status_.rx_ack_++;
// 检查后续包是否连续
while (!rx_queue_.empty() && rx_queue_.begin()->first == status_.rx_ack_) {
status_.rx_ack_++;
}
}
2. 流量动态调节
cpp
bool vmux_skt::rx_congestions(int64_t value) {
rx_congestions_ += value;
if (rx_congestions_ >= max_congestions && status_.rx_acceleration_) {
// 减速
mux_->post(cmd_acceleration, &FALSE);
} else if (rx_congestions_ <= 0 && !status_.rx_acceleration_) {
// 加速
mux_->post(cmd_acceleration, &TRUE);
}
}
3. 跨平台优化
Windows QoS
cpp
#if defined(_WIN32)
bool apply_qos(SOCKET sock, const ip_address& ip) {
qoss_ = ppp::net::QoSS::New(sock);
if (qoss_) {
return qoss_->Set(ip, DSCP_AF42);
}
return false;
}
#endif
Linux 网络保护
cpp
#if defined(_LINUX)
bool protect_socket(int fd, ppp::coroutines::YieldContext& y) {
if (protector_network) {
return protector_network->Protect(fd, y);
}
return true;
}
#endif
🚀 七、性能优化策略
- 内存管理 :
- 采用**缓冲池(BufferSwapAllocator)**实现高效复用
- 固定大小块,减少内存碎片
cpp
make_byte_array(size); // 从缓冲池获取
- 零拷贝设计 :
- 使用
shared_ptr
传递数据包,避免复制
- 使用
cpp
bool underlyin_sent(const std::shared_ptr<Byte>& packet) {
transmission->Write(packet.get(), ...);
}
- 协程优化:
cpp
vmux_spawn(context, strand, [](auto y){
// 同步操作
});
🎯 八、应用场景
场景 | 描述 |
---|---|
VNP多路复用 | 多物理连接承载多个TCP会话,降低连接建立成本 |
物联网网关 | 低功耗设备连接管理,心跳维持长连接 |
游戏服务器 | 高并发连接处理,低延迟数据传输 |
📝 结论
VMUX技术 以创新的多路复用架构,结合高效的调度与管理策略,
在确保TCP兼容的前提下,极大提升连接密度和传输效率。
特别适合高并发、低延迟的网络应用,
其模块化设计也为未来协议升级和硬件加速打下坚实基础。