/** * mds — A micro-display server * Copyright © 2014, 2015, 2016, 2017 Mattias Andrée (maandree@kth.se) * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "slavery.h" #include "globals.h" #include "client.h" #include #include #include #include #include #include /** * Master function for slave threads * * @param data Input data * @return Outout data */ void *slave_loop(void *); /** * Receive a full message and update open status if the client closes * * @param client The client * @return Zero on success, -2 on failure, otherwise -1 */ int fetch_message(client_t *client) { int r = mds_message_read(&(client->message), client->socket_fd); if (!r) { return 0; } else if (r == -2) { eprint("corrupt message received."); fail_if (1); } else if (errno == ECONNRESET) { r = mds_message_read(&(client->message), client->socket_fd); client->open = 0; /* Connection closed. */ } else if (errno != EINTR) { xperror(*argv); fail_if (1); } fail_if (r == -2); return r; fail: return -2; } /** * Create, start and detache a slave thread * * @param thread The address at where to store the thread * @param slave_fd The file descriptor of the slave's socket * @return Zero on success, -1 on error, error message will have been printed */ int create_slave(pthread_t *thread_slot, int slave_fd) { if ((errno = pthread_create(thread_slot, NULL, slave_loop, (void *)(intptr_t)slave_fd))) { xperror(*argv); with_mutex (slave_mutex, running_slaves--;); fail_if (1); } if ((errno = pthread_detach(*thread_slot))) { xperror(*argv); fail_if (1); } return 0; fail: return -1; } /** * Initialise a client, except for threading * * @param client_fd The file descriptor of the client's socket * @return The client information, `NULL` on error */ client_t * initialise_client(int client_fd) { ssize_t entry = LINKED_LIST_UNUSED; client_t *information; int locked = 0, saved_errno; size_t tmp; /* Create information table. */ fail_if (xmalloc(information, 1, client_t)); client_initialise(information); /* Add to list of clients. */ fail_if ((errno = pthread_mutex_lock(&slave_mutex))); locked = 1; entry = linked_list_insert_end(&client_list, (size_t)(void *)information); fail_if (entry == LINKED_LIST_UNUSED); /* Add client to table. */ tmp = fd_table_put(&client_map, client_fd, (size_t)(void *)information); fail_if (!tmp && errno); pthread_mutex_unlock(&slave_mutex); locked = 0; /* Fill information table. */ information->list_entry = entry; information->socket_fd = client_fd; information->open = 1; fail_if (mds_message_initialise(&(information->message))); return information; fail: saved_errno = errno; if (locked) pthread_mutex_unlock(&slave_mutex); free(information); if (entry != LINKED_LIST_UNUSED) with_mutex (slave_mutex, linked_list_remove(&client_list, entry);); return errno = saved_errno, NULL; }