お世話になります。
C言語でソケット通信でサーバ側のaccept処理のブロック状態を
alarm関数で事前にタイムアウトする秒数を設定しておき、割り込みをおこさせて割り込み処理後、再開した際にaccept関数がエラーを返しerrnoにEINTRが返ってきているかを判断してタイムアウト処理を行うという目的で以下のソースを作成しました。
しかし、実際に動作させてみるとalarmが呼び出されシグナルハンドラとして設定しているsigcatch関数が呼び出され、標準出力にsigcatch関数中で出力しているメッセージが出力されますが、そのままブロック状態から変化しませんでした。
ソケットのクライアントプログラムを接続したところ、acceptが成功して接続が確立されました。
この方式による実現が可能かどうか、不可能であれば代替方法を教えていただきたいと考えております。
お手数おかけしますが、ご回答のほどよろしくお願いします。
コンパイルおよび実行環境は以下となっています。
Red Hat Enterprise Linux ES release 4 (Nahant Update 3)
Kernel 2.6.9-34.ELsmp on an i686
glibc-2.3.4-2.19
###################以下、C言語のソースです#####################
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <signal.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>
#define BUFFER_SIZE 256
void sigcatch(int);
int
main() {
unsigned short port = 9876;
int srcSocket;
int dstSocket;
struct sigaction sa_sigint;
struct sockaddr_in srcAddr;
struct sockaddr_in dstAddr;
int dstAddrSize = sizeof(dstAddr);
int numrcv;
char buffer[BUFFER_SIZE];
int status = 0;
int select_cnt = 0;
fd_set readfds;
int n;
struct timeval timeout;
int re;
char *toSendText = "This is a test";
memset(&srcAddr, 0, sizeof(srcAddr));
srcAddr.sin_port = htons(port);
srcAddr.sin_family = AF_INET;
srcAddr.sin_addr.s_addr = htonl(INADDR_ANY);
srcSocket = socket(AF_INET, SOCK_STREAM, 0);
bind(srcSocket, (struct sockaddr *) &srcAddr, sizeof(srcAddr));
listen(srcSocket, 1);
printf("Waiting for connection ...\n");
memset(&sa_sigint, 0, sizeof(sa_sigint));
sa_sigint.sa_handler = sigcatch;
sa_sigint.sa_flags = SA_RESTART;
if (sigaction(SIGALRM, &sa_sigint, NULL) < 0) {
perror("sigaction");
exit(1);
}
alarm(10);
dstSocket = accept(srcSocket, (struct sockaddr *) &dstAddr, &dstAddrSize);
if ( dstSocket == -1 ){
if ( errno == EINTR ){
printf("accept timeout!\n");
}else{
printf("accept error![%d]\n",errno);
}
exit(-1);
}
signal( SIGALRM , SIG_IGN );
printf("Connected from %s\n", inet_ntoa(dstAddr.sin_addr));
while(1) {
FD_ZERO( &readfds );
FD_SET( dstSocket , &readfds );
n = dstSocket + 1;
timeout.tv_sec = 0;
timeout.tv_usec = 0;
re = select( n , &readfds , NULL , NULL , &timeout );
select_cnt++;
if( re > 0 ){
printf("select count %d\n",select_cnt);
if ( FD_ISSET( dstSocket , &readfds ) ){
numrcv = recv(dstSocket, buffer, BUFFER_SIZE, 0);
if(numrcv == 0 || numrcv == -1) {
status = close(dstSocket); break;
}
printf("received: %s\n", buffer);
send(dstSocket, toSendText, strlen(toSendText)+1, 0);
}
}
}
}
void sigcatch(int sig) {
printf("catch signal %d\n", sig);
if (sig == SIGALRM) {
printf("catch SIGALRM and exit.\n");
}
}
No.1
- 回答日時:
シグナルをキャッチしていてデフォルト動作でない。
IamMasakiさんのOSは、シグナルをキャッチして復帰したときに再実行という処理をするようになっているのではないですか?^^シグナルをキャッチしなければ、意図通り動くのでは?
No.2
- 回答日時:
書き間違ってました^^『シグナルをキャッチしなければ』じゃなくて、キャッチするのはよくて、『sa_sigint.sa_flags = SA_RESTART;』を指定してるのが、リスタート指定なので、コメントアウトしたほうがいいと思います。
『sa_sigint.sa_flags = SA_RESTART;』を無視するOSもあると思いますが、IamMasakiさんのOSはきちんとリスタートするんでしょう^^No.3ベストアンサー
- 回答日時:
要はacceptするときに、タイムアウトの監視もしたいということでしょうか。
その場合は、alarmによる方法は採りません。通常は、以下の方法を採ります。
acceptの前に、そのソケットに対し、selectを行います。その時、タイムアウト時間も指定します。selectから返るのは、acceptが出来る状態になったときか、selectがタイムアウトした場合の何れかです。
行いたかったことはaccept時にタイムアウト監視をしたいということでした。
さっそくtatsu99さんの方法にて確認したところ無事タイムアウト監視ができるようになりました。ありがとうございます。
以下に修正ソースを記載します。
###################以下、C言語のソースです#####################
<宣言部元ソースと同じなので省略>
//void sigcatch(int);
int
main() {
unsigned short port = 9876;
int srcSocket;
int dstSocket;
struct sigaction sa_sigint;
struct sockaddr_in srcAddr;
struct sockaddr_in dstAddr;
int dstAddrSize = sizeof(dstAddr);
int numrcv;
char buffer[BUFFER_SIZE];
int status = 0;
int select_cnt = 0;
fd_set readfds;
int n;
struct timeval timeout;
int re;
char *toSendText = "This is a test";
memset(&srcAddr, 0, sizeof(srcAddr));
srcAddr.sin_port = htons(port);
srcAddr.sin_family = AF_INET;
srcAddr.sin_addr.s_addr = htonl(INADDR_ANY);
srcSocket = socket(AF_INET, SOCK_STREAM, 0);
bind(srcSocket, (struct sockaddr *) &srcAddr, sizeof(srcAddr));
listen(srcSocket, 1);
printf("Waiting for connection ...\n");
FD_ZERO( &readfds );
FD_SET( srcSocket , &readfds );
n = srcSocket + 1;
timeout.tv_sec = 10;
timeout.tv_usec = 0;
re = select( n , &readfds , NULL , NULL , &timeout );
if ( FD_ISSET( srcSocket , &readfds ) ){
dstSocket = accept(srcSocket, (struct sockaddr *) &dstAddr, &dstAddrSize);
}else{
printf("accept timeout!\n");
exit(-1);
}
if ( dstSocket == -1 ){
printf("accept error![%d]\n",errno);
exit(-1);
}
printf("Connected from %s\n", inet_ntoa(dstAddr.sin_addr));
<accept以後の処理元ソースと同じなので省略>
<シグナルハンドラの関数は削除>
//void sigcatch(int sig) {
//printf("catch signal %d\n", sig);
//if (sig == SIGALRM) {
//printf("catch SIGALRM and exit.\n");
//}
//}
No.4
- 回答日時:
シグナル動作は実行環境に依存するでしょうし動作も安全・確実とは限らないので、シグナル(システムの内部でシグナルを使って実装している
かどうか知らないが^^)を使わないで、やりたいことができるなら、シグナルを使わない select() や poll() を使う、tatsu99 さんの方法のほうがいいでしょうね^^お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
このQ&Aを見た人はこんなQ&Aも見ています
-
ゆるやかでぃべーと すべての高校生はアルバイトをするべきだ。
高校生はアルバイトするべきだろうか?賛成だったら「賛」、否定だったら「否」を文頭においてあなたの意見を教えてください。
-
フォロワー20万人のアカウントであなたのあるあるを披露してみませんか?
あなたが普段思っている「これまだ誰も言ってなかったけど共感されるだろうな」というあるあるを教えてください
-
映画のエンドロール観る派?観ない派?
映画が終わった後、すぐに席を立って帰る方もちらほら見かけます。皆さんはエンドロールの最後まで観ていきますか?
-
海外旅行から帰ってきたら、まず何を食べる?
帰国して1番食べたくなるもの、食べたくなるだろうなと思うもの、皆さんはありますか?
-
天使と悪魔選手権
悪魔がこんなささやきをしていたら、天使のあなたはなんと言って止めますか?
-
Winsockで接続待ちタイムアウトを設定する方法
C言語・C++・C#
-
read関数をノンブロッキングで実行する(c言語)
C言語・C++・C#
-
ネットワーク切断を検出するには?
C言語・C++・C#
-
-
4
スレッドの監視方法について
C言語・C++・C#
-
5
setsockoptのタイムアウト確認について
C言語・C++・C#
-
6
絶対パスの取得について
C言語・C++・C#
-
7
<unistd.h>をVisualStudioでつかえるようにする
C言語・C++・C#
-
8
C言語での引数の省略方法
C言語・C++・C#
-
9
alarmの使用について
C言語・C++・C#
-
10
fgetsなどのときのstdinのバッファを消すには?
C言語・C++・C#
-
11
ソケット通信で同時受信の制限?
C言語・C++・C#
-
12
sys/types.hの必要性について
C言語・C++・C#
-
13
[c++] <pthread.h>がinclude できない
C言語・C++・C#
-
14
TCP/IP のパケットの分断と結合について
UNIX・Linux
関連するカテゴリからQ&Aを探す
おすすめ情報
- ・漫画をレンタルでお得に読める!
- ・人生のプチ美学を教えてください!!
- ・10秒目をつむったら…
- ・あなたの習慣について教えてください!!
- ・牛、豚、鶏、どれか一つ食べられなくなるとしたら?
- ・【大喜利】【投稿~9/18】 おとぎ話『桃太郎』の知られざるエピソード
- ・街中で見かけて「グッときた人」の思い出
- ・「一気に最後まで読んだ」本、教えて下さい!
- ・幼稚園時代「何組」でしたか?
- ・激凹みから立ち直る方法
- ・1つだけ過去を変えられるとしたら?
- ・【あるあるbot連動企画】あるあるbotに投稿したけど採用されなかったあるある募集
- ・【あるあるbot連動企画】フォロワー20万人のアカウントであなたのあるあるを披露してみませんか?
- ・映画のエンドロール観る派?観ない派?
- ・海外旅行から帰ってきたら、まず何を食べる?
- ・誕生日にもらった意外なもの
- ・天使と悪魔選手権
- ・ちょっと先の未来クイズ第2問
- ・【大喜利】【投稿~9/7】 ロボットの住む世界で流行ってる罰ゲームとは?
- ・推しミネラルウォーターはありますか?
- ・都道府県穴埋めゲーム
- ・この人頭いいなと思ったエピソード
- ・準・究極の選択
このQ&Aを見た人がよく見るQ&A
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
std::set<int> で、ある値が何...
-
C#のコンパイルエラーCS0120に...
-
Cでボリュームコントロールを制...
-
int16_t の _t は何?
-
C言語の関数で戻り値を返す必要...
-
fprintfでの文字化け
-
C言語での引数の省略方法
-
因数分解を行うプログラムについて
-
引数 戻り値 return文について
-
演算子オーバーロードのプログ...
-
atoi関数の自作
-
C言語での奇数の和
-
c言語 〇×ゲーム
-
プログラムのバグについて
-
C言語で分からないところがあり...
-
ColorをRGBで指定する方法
-
C言語の基礎 . 2乗値の差につ...
-
剰余演算を論理演算と加減算に...
-
クイックソートでの整列
-
c/c++でのファイルの上書き保存...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
「指定されたキャストは有効で...
-
C言語での引数の省略方法
-
#define _CRT_SECURE_NO_WARNIN...
-
AtCoderABC135の問題Cについて
-
C言語 エラーの原因がわからな...
-
複数桁10進数の*桁目だけを抽出...
-
【C++】関数ポインタの使い方
-
実数の整数部,小数部の取得
-
ラップ関数とはどんなものですか?
-
if と配列の組み合わせ
-
return 1L
-
read関数をノンブロッキングで...
-
(int *)の意味
-
std::set<int> で、ある値が何...
-
Win32APIで作るコンボボックス...
-
C++でvectorにテキストファイル...
-
「{ } で囲むだけ」は正しい?
-
足して100になるような乱数のア...
-
Arduinoのプログラムにエラーが...
-
課題でつまってます・・・
おすすめ情報