#include "cs241.c"
#define THREAD_COUNT 10
int list_s;
int connections[THREAD_COUNT];
char space[THREAD_COUNT];
int done = 0;
pthread_mutex_t muxlock = PTHREAD_MUTEX_INITIALIZER;
int *numbers;
int numbers_count;
void *listener(void *arg) {
int n = *(int *) arg;
FILE *f = fdopen(connections[n], "r");
if (f == NULL)
printf("Could not open file\n");
char *line = NULL;
size_t *len = malloc(sizeof(int));
while(getline(&line, len, f) != -1) {
printf("%s", line);
if (strcmp("END", line) == 0) {
pthread_mutex_lock(&muxlock);
done = 1;
pthread_mutex_unlock(&muxlock);
}
}
space[n] = 't';
fclose(f);
free(len);
close(connections[n]);
return NULL;
}
void initialize() {
int n;
for (n = 0; n < THREAD_COUNT; n++) {
space[n] = 't';
}
}
int check() {
int index;
for (index = 0; index < THREAD_COUNT; index++) {
if (space[index] == 't') {
space[index] = 'f';
return index;
}
}
return 0;
}
int main(int argc, char *argv[]) {
int port = 0;
int binder;
int lis;
int i = 0;
int *j = malloc(sizeof(int));
initialize();
pthread_t threads[THREAD_COUNT];
if ((argc != 2) || (sscanf(argv[1], "%d", &port) != 1)) {
fprintf(stderr, "Usage: %s [PORT]\n", argv[0]);
exit(1);
}
if (port < 1024) {
fprintf(stderr, "Port must be greater than 1024\n");
exit(1);
}
// set the initial conditions for the numbers array.
numbers = malloc(sizeof(int));
numbers_count = 0;
struct sockaddr_in servaddr; // socket address structure
// set all bytes in socket address structure to zero, and fill in the relevant data members
memset(&servaddr, 0, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
servaddr.sin_port = htons(port);
list_s = socket(AF_INET, SOCK_STREAM, 0);
if (list_s < 0) {
printf("Could not create socket\n");
exit(EXIT_FAILURE);
}
binder = bind(list_s, (struct sockaddr *) &servaddr, sizeof(servaddr));
if (binder < 0) {
printf("Could not bind socket\n");
exit(EXIT_FAILURE);
}
lis = listen(list_s, SOMAXCONN);
if (lis < 0) {
printf("Could not listen on socket\n");
exit(EXIT_FAILURE);
}
SET_NON_BLOCKING(list_s);
while (done != 1) {
connections[i] = accept(list_s, NULL, NULL);
if (connections[i] < 0) {
if ((errno == EAGAIN) || (errno == EWOULDBLOCK))
continue;
printf("Could not accept connection\n");
exit(EXIT_FAILURE);
}
i = check();
*j = i;
SET_BLOCKING(list_s);
pthread_create(&threads[i], NULL, listener, j);
}
// Verify the array.
//verify(numbers, numbers_count);
free(j);
close(list_s);
exit(EXIT_SUCCESS);
}
そのため、main() に while ループがあり、'done' = 1 のときに終了する必要があります。これは、listener() が 'END' を受け取ったときに設定されます。最初の問題は、最初の反復で 'END' を送信すると、while ループが終了せず、別の 'END' が送信された後にのみ終了することです。
ソケットのブロックを解除およびブロックするために別のファイルに 2 つのマクロ SET_NON_BLOCKING および SET_BLOCKING があるため、接続がない場合は接続を待ちます。次の問題は、これらのマクロを使用しない場合です。listener() の getline() は、ストリームから出力されるすべてを読み取ることができません。それらを使用すると、ストリームをまったく開くことができません。
問題のいくつかは、スレッドに 'j' を渡すことにあると思います。2 番目のスレッドが開始すると、最初のスレッドが読み取れる前に 'j' が上書きされます。しかし、私は数日以上そこにいて、どこにも行けません。どんな助けでも大歓迎です。ありがとう!
また、ソケットのブロックとスレッドのロックが適切な場所にあるかどうかを尋ねたいと思いますか?