/* NOTE: Under Graphene, this test must be run only in fork mode. * This is due to Graphene restricting communication via Unix * domain sockets only for processes in same Graphene instance * (i.e. only between parent and its child in this test). */ #include #include #include #include #include #include #include #include #include #include enum { SINGLE, PARALLEL } mode = PARALLEL; int do_fork = 0; int pipefds[2]; int server_dummy_socket(void) { int create_socket; struct sockaddr_un address; if ((create_socket = socket(AF_UNIX, SOCK_STREAM, 0)) > 0) printf("Dummy socket was created\n"); address.sun_family = AF_UNIX; strncpy(address.sun_path, "dummy", sizeof(address.sun_path)); if (bind(create_socket, (struct sockaddr*)&address, sizeof(address)) < 0) { perror("bind"); close(create_socket); exit(1); } if (listen(create_socket, 3) < 0) { perror("listen"); close(create_socket); exit(1); } /* do not close this socket to test two sockets in parallel */ return 0; } int server(void) { int create_socket, new_socket; socklen_t addrlen; int bufsize = 1024; char* buffer = malloc(bufsize); struct sockaddr_un address; if ((create_socket = socket(AF_UNIX, SOCK_STREAM, 0)) > 0) printf("The socket was created\n"); address.sun_family = AF_UNIX; strncpy(address.sun_path, "u", sizeof(address.sun_path)); if (bind(create_socket, (struct sockaddr*)&address, sizeof(address)) < 0) { perror("bind"); close(create_socket); exit(1); } if (listen(create_socket, 3) < 0) { perror("listen"); close(create_socket); exit(1); } if (mode == PARALLEL) { close(pipefds[0]); char byte = 0; if (write(pipefds[1], &byte, 1) != 1) { perror("write error"); exit(1); } } addrlen = sizeof(address); new_socket = accept(create_socket, (struct sockaddr*)&address, &addrlen); if (new_socket < 0) { perror("accept"); close(create_socket); exit(1); } close(create_socket); printf("The client is connected...\n"); if (do_fork) { if (fork() > 0) { __asm__ volatile ("int $3"); close(new_socket); wait(NULL); return 0; } } for (int i = 0; i < 10; i++) { sprintf(buffer, "Data: This is packet %d\n", i); if (sendto(new_socket, buffer, strlen(buffer), 0, 0, 0) == -1) { fprintf(stderr, "sendto() failed\n"); exit(1); } } close(new_socket); if (do_fork) exit(0); return 0; } int client(void) { int count, create_socket; int bufsize = 1024; char* buffer = malloc(bufsize); struct sockaddr_un address; if (mode == PARALLEL) { close(pipefds[1]); char byte = 0; if (read(pipefds[0], &byte, 1) != 1) { perror("read error"); return 1; } } if ((create_socket = socket(AF_UNIX, SOCK_STREAM, 0)) >= 0) printf("The socket was created\n"); address.sun_family = AF_UNIX; strncpy(address.sun_path, "u", sizeof(address.sun_path)); if (connect(create_socket, (struct sockaddr*)&address, sizeof(address)) == 0) printf("The connection was accepted with the server\n"); else { printf("The connection was not accepted with the server\n"); exit(0); } if (do_fork) { if (fork() > 0) { close(create_socket); wait(NULL); return 0; } } puts("Receiving:"); while ((count = recv(create_socket, buffer, bufsize, 0)) > 0) { fwrite(buffer, count, 1, stdout); } puts("Done"); close(create_socket); if (do_fork) exit(0); return 0; } int main(int argc, char** argv) { if (argc > 1) { if (strcmp(argv[1], "client") == 0) { mode = SINGLE; return client(); } else if (strcmp(argv[1], "server") == 0) { mode = SINGLE; server_dummy_socket(); return server(); } else if (strcmp(argv[1], "fork") == 0) { do_fork = 1; } else { printf("Invalid argument\n"); return 1; } } if (pipe(pipefds) < 0) { perror("pipe error"); return 1; } int pid = fork(); if (pid < 0) { perror("fork error"); return 1; } else if (pid == 0) { return client(); } else { server_dummy_socket(); return server(); } }