5G 基站SCTP

如何实现SCTP多归属链路对接

文章目录

​​前言​​​​一、SCTP是什么?​​​​二、lksctp​​​​三.sctp初始化​​​​四.绑定本端两个IP​​​​ 五.与对端建链​​​​六.设置主要路径​​​​七.设置是否启用心跳​​​​八.关于防火墙的配置​

​总结​​

前言

最近完成了贝尔及华为软交换的SCTP 多归属链路对接。由于网络上对于多归属链路介绍的资料特别少。能看到的一些资料介绍,但是说明的不详细,大都是Demo性质不能完全商用。以客户端为例子,概括如下,首先绑定本端两个IP,然后绑定交换的主用Path。然后将该主用的Path的IP设置为PrimaryPath。如果对端交换不支持BEAT心跳消息,就不要发送该BEAT消息。 现在分享下我的具体的经验心得。

提示:以下是本篇文章正文内容,下面案例可供参考

一、SCTP是什么?

SCTP (Stream Control Transmission Protocol)是一种传输协议,在TCP/IP协议栈中所处的位置和TCP、UDP类似,兼有TCP/UDP两者特征。

随着网络接入技术的多样化,利用通信终端上多个网络接口实现并行多路径(Concurrent Multipath Transfer,CMT)成为研究的热点,成为提高数据传输效率重要手段。基于流传输(Stream Control Transmission Protocl,SCTP)实现的CMT是这一领域研究的重点,它通过扩展SCTP的多IP特性实现同一关联的多条端到端的路径上同时传输数据。    

SIGTRAN信令通过SCTP建链承载信令。

SCTP位于SCTP应用和无线分组网络(IP网)之间,通过两个SCTP端点之间建立关联,为两个SCTP用户提供可靠的消息传输业务。一个SCTP链路包换了一个或两个源/目的的地址。这些地址包含在每个端点的传输地址中。SCTP的关联结构如图1所示。

如何实现SCTP多归属链路对接_开发语言

SCTP发送端使用一组传输地址作为消息的目的地,会根据SCTP用户的指令和当前合法的目的地址集合的可达性状态,为每个待发送的消息选择一个目的传输地址。在关联建立后,需要为每个SCTP端点定义一条主路径,用来在正常情况下发送SCTP消息。正常情况下,SCTP只选择其中的一条可用路径作为主路径传输数据,其他路径作为备用路径。当主路径失效时,SCTP切换到其中的一条备用路径上继续传输。

二、lksctp

lksctp封装了linux内核sctp的接口函数。使用lksctp可以加速sctp的开发。代码中引入sctp.h,编译时加入-lsctp,进行动态库链接。

三.sctp初始化

from_sockfd = socket(AF_INET, SOCK_STREAM, IPPROTO_SCTP);

setsockopt(from_sockfd,SOL_SOCKET,SO_SNDBUF,(char*)&sndBufS,sndBufL);

setsockopt(from_sockfd,SOL_SOCKET,SO_RCVBUF,(char*)&rcvBufS,rcvBufL);

setsockopt(from_sockfd, SOL_SOCKET, SO_REUSEADDR, &reuse_address, sizeof(reuse_address);

setsockopt(from_sockfd , IPPROTO_SCTP , SCTP_RTOINFO , &timeout , opt_len);

setsockopt(from_sockfd , IPPROTO_SCTP , SCTP_ASSOCINFO , &assocmaxrxt , opt_len);

setsockopt(from_sockfd, IPPROTO_SCTP, SCTP_INITMSG, &initmsg, sizeof(initmsg);

四.绑定本端两个IP

struct sockaddr_in cliaddr;

int ret;

bzero( (void *)&cliaddr, sizeof(cliaddr) );

cliaddr.sin_family = AF_INET;

cliaddr.sin_addr.s_addr = pFrom_addr[0].sin_addr.s_addr;

cliaddr.sin_port = pFrom_addr[0].sin_port;

ret = bind(fd, (struct sockaddr *)&cliaddr, sizeof(cliaddr));

bzero( (void *)&cliaddr, sizeof(cliaddr) );

cliaddr.sin_family = AF_INET;

cliaddr.sin_addr.s_addr = pFrom_addr[1].sin_addr.s_addr;

cliaddr.sin_port = pFrom_addr[1].sin_port;

ret = sctp_bindx(fd, (struct sockaddr *)&cliaddr, 1, SCTP_BINDX_ADD_ADDR);

五.与对端建链

设置对端两个IP

if (sctp_peeraddr->addr_num)

{

addr = (struct sockaddr *)(cli_addr + offsetof(struct sctp_getaddrs, addrs));

for (index = 0; index < sctp_peeraddr->addr_num; index++)

{

memset(dest, 0, sizeof(dest));

if (addr->sa_family == AF_INET)

{

//Log::PrintLog(LOG_INFO, "[SCTP_GET_PEER_ADDRS] %s\n", inet_ntoa(((struct sockaddr_in *)addr)->sin_addr));

Log::PrintLog(LOG_TRACE, "Sctp.cxx", "sctp_peer_param()", "[SCTP_GET_PEER_ADDRS %s]",inet_ntoa(((struct sockaddr_in *)addr)->sin_addr));

memcpy(&paddrparams.spp_address, addr, sizeof(struct sockaddr_in));

addr = (struct sockaddr *) (((char *) addr) + sizeof(struct sockaddr_in));

}

ret = setsockopt(cfd, IPPROTO_SCTP, SCTP_PEER_ADDR_PARAMS, (const void *)&paddrparams, sizeof(paddrparams));

if (ret < 0)

{

//Log::PrintLog(LOG_INFO,"[set SCTP_PEER_ADDR_PARAMS error] %d:%s\n", errno, strerror(errno));

Log::PrintLog(LOG_TRACE, "Sctp.cxx", "sctp_peer_param()", "[SCTP_PEER_ADDR_PARAMS err] %d %s", errno, strerror(errno));

return;

}

Log::PrintLog(LOG_TRACE, "Sctp.cxx", "sctp_peer_param()", "set SCTP_PEER_ADDR_PARAMS]");

Log::PrintLog(LOG_TRACE, "Sctp.cxx", "sctp_peer_param()", "get SCTP_PEER_ADDR_PARAMS] hb = %d, sackdelay = %d, %s, %s, %s",

paddrparams.spp_hbinterval,

paddrparams.spp_sackdelay,

(paddrparams.spp_flags & SPP_HB_ENABLE)?"HB_ENABLE":"HB_DISABLE",

(paddrparams.spp_flags & SPP_PMTUD_ENABLE)?"PMTUD_ENABLE":"PMTUD_DISABLE",

(paddrparams.spp_flags & SPP_SACKDELAY_ENABLE)?"SACKDELAY_ENABLE":"SACKDELAY_DISABLE");

}

}

与一个IP建链:

sctp_connectx(from_sockfd, (struct sockaddr*)&m_primsockaddr, 1,(sctp_assoc_t *)&m_assoc[0];

切记,与对端建联时,初始时,只和一个IP进行建链。

六.设置主要路径

setsockopt(from_sockfd,IPPROTO_SCTP,SCTP_PRIMARY_ADDR,&m_primaddr,sizeof(m_primaddr)

七.设置是否启用心跳

struct sctp_paddrparams paddrparams;

if (spp_hbinterval == 0)

{

paddrparams.spp_flags |= SPP_HB_DISABLE;

}

else

{

paddrparams.spp_flags |= SPP_HB_ENABLE;

paddrparams.spp_hbinterval = spp_hbinterval;

}

ret = setsockopt(cfd, IPPROTO_SCTP, SCTP_PEER_ADDR_PARAMS, (const void *)&paddrparams, sizeof(paddrparams));

八.关于防火墙的配置

如果系统的网卡地址是通过防火墙映射的,防火墙上有的没有SCTP一个选项,那么配置TCP即可。但是防火墙一定要设置为透明模式。不然在调试多归属链路时会遇到各式各样的怪问题。

总结

SCTP偶链多归属对接,网上资料特别少,有的也是语焉不详。笔者根据最近数月的调试,将经验分享处理,希望对进行SIGTRAN信令开始的同行有帮助。

相关推荐
AI原吾1 小时前
构建灵活、高效的HTTP/1.1应用:探索h11库
网络·python·网络协议·http·ai·h11
看山还是山,看水还是。1 小时前
Nginx 的 Http 模块介绍(中)
android·运维·网络·nginx·http
Tony聊跨境1 小时前
什么是 HTTP 代理?它如何工作?
网络·http·ip
licy__1 小时前
计算机网络期末考试试卷及答案
网络
码出钞能力1 小时前
UDP组播测试
网络·网络协议·udp
苏湘涵1 小时前
socket编程---UDP
linux·开发语言·网络·php·进程通信
清酒伴风(面试准备中......)1 小时前
计算机网络——开放系统互连参考模型
网络·计算机网络·1024程序员节
不甘平凡的蜜蜂1 小时前
第三十三篇:TCP协议如何避免/减少网络拥塞,TCP系列八
运维·网络·网络协议·tcp/ip·计算机网络·智能路由器
danplus1 小时前
外贸邮件营销群发:如何避免成为垃圾邮件?
服务器·api接口·邮件营销·smtp服务器·邮件接口·营销邮件·域名邮箱
风虎云龙科研服务器2 小时前
GPU 服务器:高性能计算的核心驱动力
运维·服务器