1

サーバーがリソースの少ないセンサー/追跡デバイスのファームウェアを更新する必要がある特別なアプリケーションでは、新しいファームウェアのパケットを受信するリモート デバイス (クライアント) でデータが失われることがあるという問題が発生しました。接続は GPRS ネットワーク上の TCP/IP です。デバイスは、ネットワーク インターフェイスとして SIM900 GSM チップを使用します。

デバイスが受信するデータが多すぎるために、問題が発生する可能性があります。パッケージを送信する頻度を減らしてトラフィックを減らしてみましたが、それでもエラーが発生する場合がありました。

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

ご覧のとおり、出力の 2 行目の問題: setsockopt は「無効な引数」を返します。

なぜこれが起こるのですか?TCP_MAXSEG を設定する際のいくつかの制約について読みましたが、このような動作に関する報告はありませんでした。

ありがとう、デニス

4

2 に答える 2

1

特に明記しない限り、optval は int へのポインタです。

しかし、u_int16 を使用しています。このパラメーターが int ではないと言っているものは何もありません。

編集:ええ、ここにソースコードがあり、あなたは見ることができます:

637         if (optlen < sizeof(int))
638                 return -EINVAL;
于 2013-08-07T22:23:25.717 に答える