2013-08-07 88 views
1

在我們的服務器需要更新資源不足的傳感器/跟蹤設備的固件的特殊應用中,我們遇到了一個問題,即有時數據丟失遠程設備(客戶端)接收新固件的數據包。該連接是通過GPRS網絡上的TCP/IP。這些設備使用SIM900 GSM芯片作爲網絡接口。通過套接字在Linux中降低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的一些限制,但我沒有遇到任何有關這種行爲的報告。

感謝, 丹尼斯

+0

這聽起來好像每個人都在猜測這裏。設備應該正確處理標準的MTU和MSS:如果沒有,或者更有可能的話,如果它有一些其他TCP錯誤,他們應該*修復它。 – EJP

回答

0

Unless otherwise noted, optval is a pointer to an int.

,但您使用的是u_int16。我沒有看到任何說這個參數不是int的東西。

編輯:是的,here是源代碼,你可以看到:

637   if (optlen < sizeof(int)) 
638     return -EINVAL; 
+0

這就是它,謝謝!不幸的是,問題的核心仍然存在,只是以其他形式稍微存在:現在setsockopt返回0,但值不會改變。現在我正在做更多的研究! – dennis90

+0

@ dennis90您是否嘗試禁用nagle算法並使用較小的發送來調用?我想這並不能保證任何東西..但它可能會起作用。 – xaxxon

+0

禁用nagle算法是否意味着只需在由服務器的accept()調用返回的套接字上設置TCP_NODELAY?然後,我確實嘗試過,但沒有奏效。現在我會再試一次 - 也許我做錯了什麼。 – dennis90