Столкнулся со странным (непонятным) поведением fgets при чтении данных с сокета посредством fdopen. При запросе с десктопной версии браузера всё хорошо, но когда запрос идет с мобильного браузера chrome, то fgets виснет.
С чем это связано? fgets не находит символа новой строки/EOF? Так fgets все равно должен завершиться при достижении предела num.
char *fgets(char *str, int num, FILE *stream);
Тогда, получается, сам мобильный chrome шлет непрерывные данные в каком-то запросе? Я правильно понимаю (баг chrome? почему fgets при достижении предела num не завершился?)? Или в чем причина?
Да, при использовании read такое поведение не замечено.
ssize_t read(int fd, void *buf, size_t count);
Минималку из своего pet-проекта делать не стал, вот отсюда (https://ycpcs.github.io/cs365-spring2017/lectures/lecture15.html) тупо (для теста, вдруг я где накосячил) взял пример - то же самое поведение.
// This is like server.c, but uses fdopen to allow communication
// via stdio functions rather than Unix system calls
#include <unistd.h>
#include <stdio.h> // for perror()
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>
#include <fcntl.h>
#include <netinet/in.h> // for struct sockaddr_in
#include <arpa/inet.h> // for inet_pton
#include <string.h>
#define PORT 40002
#define BUFSIZE 2000
int main(void)
{
int server_sock_fd;
int rc;
char buf[BUFSIZE];
int done = 0;
server_sock_fd = socket(PF_INET, SOCK_STREAM, 0);
if (server_sock_fd < 0) {
perror("Couldn't create socket");
exit(1);
}
struct sockaddr_in server_addr;
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(PORT);
inet_pton(AF_INET, "0.0.0.0", &server_addr.sin_addr);
rc = bind(
server_sock_fd,
(struct sockaddr *) &server_addr,
sizeof(server_addr));
if (rc < 0) {
perror("Couldn't bind server socket");
exit(1);
}
rc = listen(server_sock_fd, 5);
if (rc < 0) {
perror("Couldn't listen on server socket");
exit(1);
}
while (!done) {
int client_socket_fd;
struct sockaddr_in client_addr;
socklen_t client_addr_size = sizeof(client_addr);
client_socket_fd = accept(
server_sock_fd,
(struct sockaddr*) &client_addr,
&client_addr_size);
if (client_socket_fd < 0) {
perror("Couldn't accept connection");
exit(1);
}
// Wrap the socket file descriptor using a FILE*,
// for reading only.
FILE *client_socket_fh = fdopen(client_socket_fd, "r");
printf("before fgets\n");
// Read one line from the client
fgets(buf, BUFSIZE, client_socket_fh);
printf("after fgets\n");
// Close the client socket
fclose(client_socket_fh);
// Write the received message to stdout
printf("%s", buf);
// Отсутствовало в примере, добавил
close(client_socket_fd);
if (strcmp(buf, "quit\r\n") == 0) {
done = 1;
}
}
close(server_sock_fd);
return 0;
}
Т.е. c десктопного chrome вывод в консоль: before fgets after fgets
C мобильного под android: before fgets <висим до закрытия вкладки/браузера> after fgets