C言語-ソケットプログラミング_片方向通信_UDPサーバ

UDPによる片方向通信(サーバ側)

片方向通信とは

片方向通信とは、一方通行の通信のことです。

受信側(サーバ)は、送信側(クライアント)から送られたデータを受け取っても、送信側に応答しない仕様です。

また、UDPなので、転送中にデータが破損していても、エラーが発生したと認識しません。


プログラム説明

・ソケット生成時にUDP(User Datagram Protocol)を指定します。


socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP);

・全てのローカルインターフェイスにバインドされうることを許可します。

 接続を受けるネットワークインターフェイスがどれでもいい(どのアドレスからの接続でも待ち受ける)場合、「INADDR_ANY」を使用します。


sv_addr.sin_family = AF_INET;
sv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
sv_addr.sin_port = htons(info->sv_port);

・終了できるようにするため、クライアントから「terminate」という文字列を受け取った場合には、サーバ動作を終了する処理を追加します。


プログラム


/*
 *  UDP server
 */

#include <stdio.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>

/*
 *  サーバ情報を格納する
 */
struct server_info {
    unsigned short sv_port;
};
typedef struct server_info sv_info_t;

/*!
 * @brief      UDPメッセージを受信する
 * @param[in]  sd  ソケットディスクリプタ
 * @param[out] errmsg エラーメッセージ格納先
 * @return     成功ならば0、失敗ならば-1を返す。
 */
static int
udp_receiver(int sd, char *errmsg)
{
    int exit_flag = 1;
    int recv_msglen = 0;
    char buff[256];
    struct sockaddr_in cl_addr = {0};
    unsigned int cl_addr_len = 0;

    while(exit_flag){
        /* 入出力パラメータのサイズをセットする */
        cl_addr_len = sizeof(cl_addr);

        /* クライアントからメッセージを受信するまでブロックする */
        recv_msglen = recvfrom(sd, buff, sizeof(buff), 0,
                               (struct sockaddr *)&cl_addr, &cl_addr_len);
        if(recv_msglen < 0){
            sprintf(errmsg, "(line:%d) %s", __LINE__, strerror(errno));
            return(-1);
        }

        /* 受信メッセージを標準出力する */
        fprintf(stdout, "[client: %s]\n", inet_ntoa(cl_addr.sin_addr));
        fprintf(stdout, "message: %s\n", buff);

        /* クライアントからのサーバ終了命令を確認する */
        if(strcmp(buff, "terminate") == 0){
            exit_flag = 0;
        }
    }

    return( 0 );
}

/*!
 * @brief      UDPサーバ実行
 * @param[in]  info   クライアント接続情報
 * @param[out] errmsg エラーメッセージ格納先
 * @return     成功ならば0、失敗ならば-1を返す。
 */
static int
udp_server(sv_info_t *info, char *errmsg)
{
    struct sockaddr_in sv_addr = {0};
    int sd = 0;
    int rc = 0;

    /* ソケットの生成する */
    sd = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP);
    if(sd < 0){
        sprintf(errmsg, "(line:%d) %s", __LINE__, strerror(errno));
        return(-1);
    }

    /* ローカルのアドレス構造体を作成する */
    memset(&sv_addr, 0, sizeof(sv_addr));
    sv_addr.sin_family = AF_INET;
    sv_addr.sin_addr.s_addr = htonl(INADDR_ANY); /* ワイルドカード指定 */
    sv_addr.sin_port = htons(info->sv_port);

    /* ローカルアドレスへバインドする */
    rc = bind(sd, (struct sockaddr *)&sv_addr, sizeof(sv_addr));
    if(rc != 0){
        sprintf(errmsg, "(line:%d) %s", __LINE__, strerror(errno));
        goto close_exit;
    }

    /* 文字列を受信する */
    rc = udp_receiver(sd, errmsg);
    if(rc != 0) goto close_exit;

  close_exit:
    /* ソケットを破棄する */
    if(sd != 0) close(sd);
    return(rc);
}

/*!
 * @brief      初期化処理。待受ポート番号を設定する。
 * @param[in]  argc   コマンドライン引数の数
 * @param[in]  argv   コマンドライン引数
 * @param[out] info   サーバ情報
 * @param[out] errmsg エラーメッセージ格納先
 * @return     成功ならば0、失敗ならば-1を返す。
 */
static int
initialize(int argc, char *argv[], sv_info_t *info, char *errmsg)
{
    if(argc != 2){
        sprintf(errmsg, "Usage: %s <port>\n", argv[0]);
        return(-1);
    }

    memset(info, 0, sizeof(sv_info_t));
    info->sv_port = atoi(argv[1]);

    return(0);
}

/*!
 * @brief   main routine
 * @return  成功ならば0、失敗ならば-1を返す。
 */
int
main(int argc, char *argv[])
{
    int rc = 0;
    sv_info_t info = {0};
    char errmsg[256];

    rc = initialize(argc, argv, &info, errmsg);
    if(rc != 0){
        fprintf(stderr, "Error: %s\n", errmsg);
        return(-1);
    }

    rc = udp_server(&info, errmsg);
    if(rc != 0){
        fprintf(stderr, "Error: %s\n", errmsg);
        return(-1);
    }

    return(0);
}


関連ページ


UDP通信


TCP通信