在我们的服务器需要更新低资源传感器/跟踪设备固件的特殊应用中,我们遇到了一个问题,即在接收新固件数据包的远程设备(客户端)中有时会丢失数据。连接是通过GPRS网络的TCP/IP连接。该设备使用SIM900 GSM芯片作为网络接口。
这些问题可能是由于设备接收太多数据引起的。我们尝试通过更少发送数据包来减少流量,但有时仍然出现错误。
我们联系了SIM900芯片的当地零售商,他们负责提供技术支持并可能联系芯片的中国制造商(simcom)。他们说,首先我们应该尝试减少连接的TCP MSS(最大分段大小)。
在我们的服务器上,我做了以下操作:
运行上述代码会得到以下结果:
这些问题可能是由于设备接收太多数据引起的。我们尝试通过更少发送数据包来减少流量,但有时仍然出现错误。
我们联系了SIM900芯片的当地零售商,他们负责提供技术支持并可能联系芯片的中国制造商(simcom)。他们说,首先我们应该尝试减少连接的TCP MSS(最大分段大小)。
在我们的服务器上,我做了以下操作:
static int
create_master_socket(unsigned short master_port) {
static struct sockaddr_in master_address;
int master_socket = socket(AF_INET,SOCK_STREAM,0);
if(!master_socket) {
perror("socket");
throw runtime_error("Failed to create master socket.");
}
int tr=1;
if(setsockopt(master_socket,SOL_SOCKET,SO_REUSEADDR,&tr,sizeof(int))==-1) {
perror("setsockopt");
throw runtime_error("Failed to set SO_REUSEADDR on master socket");
}
master_address.sin_family = AF_INET;
master_address.sin_addr.s_addr = INADDR_ANY;
master_address.sin_port = htons(master_port);
uint16_t tcp_maxseg;
socklen_t tcp_maxseg_len = sizeof(tcp_maxseg);
if(getsockopt(master_socket, IPPROTO_TCP, TCP_MAXSEG, &tcp_maxseg, &tcp_maxseg_len)) {
log_error << "Failed to get TCP_MAXSEG for master socket. Reason: " << errno;
perror("getsockopt");
} else {
log_info << "TCP_MAXSEG: " << tcp_maxseg;
}
tcp_maxseg = 256;
if(setsockopt(master_socket, IPPROTO_TCP, TCP_MAXSEG, &tcp_maxseg, tcp_maxseg_len)) {
log_error << "Failed to set TCP_MAXSEG for master socket. Reason: " << errno;
perror("setsockopt");
} else {
log_info << "TCP_MAXSEG: " << tcp_maxseg;
}
if(getsockopt(master_socket, IPPROTO_TCP, TCP_MAXSEG, &tcp_maxseg, &tcp_maxseg_len)) {
log_error << "Failed to get TCP_MAXSEG for master socket. Reason: " << errno;
perror("getsockopt");
} else {
log_info << "TCP_MAXSEG: " << tcp_maxseg;
}
if(bind(master_socket, (struct sockaddr*)&master_address,
sizeof(master_address))) {
perror("bind");
close(master_socket);
throw runtime_error("Failed to bind master_socket to port");
}
return master_socket;
}
运行上述代码会得到以下结果:
I0807 ... main.cpp:267] TCP_MAXSEG: 536
E0807 ... main.cpp:271] Failed to set TCP_MAXSEG for master socket. Reason: 22 setsockopt: Invalid argument
I0807 ... main.cpp:280] TCP_MAXSEG: 536
正如您所看到的,输出的第二行中存在问题:setsockopt返回“无效参数”。
为什么会这样?我了解到在设置TCP_MAXSEG时存在一些限制,但我没有遇到过这种行为的报告。
谢谢, Dennis