/* 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; write(pipefds[1], &byte, 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; read(pipefds[0], &byte, 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; client(); return 0; } if (strcmp(argv[1], "server") == 0) { mode = SINGLE; server_dummy_socket(); server(); return 0; } if (strcmp(argv[1], "fork") == 0) { do_fork = 1; goto old; } } else { old: pipe(pipefds); int pid = fork(); if (pid == 0) { client(); } else { server_dummy_socket(); server(); } } return 0; }