/* * Copyright (C) 2016 Christian Meffert * * 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 2 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, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "commands.h" #include #include #include #include #include #include #include "logger.h" #include "misc.h" struct command { pthread_mutex_t lck; pthread_cond_t cond; command_function func; command_function func_bh; void *arg; int nonblock; int ret; int pending; }; struct commands_base { struct event_base *evbase; command_exit_cb exit_cb; int command_pipe[2]; struct event *command_event; struct command *current_cmd; }; /* * Asynchronous execution of the command function */ static void command_cb_async(struct commands_base *cmdbase, struct command *cmd) { enum command_state cmdstate; // Command is executed asynchronously cmdstate = cmd->func(cmd->arg, &cmd->ret); // Only free arg if there are no pending events (used in worker.c) if (cmdstate != COMMAND_PENDING && cmd->arg) free(cmd->arg); free(cmd); event_add(cmdbase->command_event, NULL); } /* * Synchronous execution of the command function */ static void command_cb_sync(struct commands_base *cmdbase, struct command *cmd) { enum command_state cmdstate; CHECK_ERR(L_MAIN, pthread_mutex_lock(&cmd->lck)); cmdstate = cmd->func(cmd->arg, &cmd->ret); if (cmdstate == COMMAND_PENDING) { // Command execution is waiting for pending events before returning to the caller cmdbase->current_cmd = cmd; cmd->pending = cmd->ret; } else { // Command execution finished, execute the bottom half function if (cmd->ret == 0 && cmd->func_bh) cmd->func_bh(cmd->arg, &cmd->ret); event_add(cmdbase->command_event, NULL); // Signal the calling thread that the command execution finished CHECK_ERR(L_MAIN, pthread_cond_signal(&cmd->cond)); CHECK_ERR(L_MAIN, pthread_mutex_unlock(&cmd->lck)); // Note if cmd->func was cmdloop_exit then cmdbase may be invalid now, // because commands_base_destroy() may have freed it } } /* * Event callback function * * Function is triggered by libevent if there is data to read on the command pipe (writing to the command pipe happens through * the send_command function). */ static void command_cb(int fd, short what, void *arg) { struct commands_base *cmdbase; struct command *cmd; int ret; cmdbase = arg; // Get the command to execute from the pipe ret = read(cmdbase->command_pipe[0], &cmd, sizeof(cmd)); if (ret != sizeof(cmd)) { DPRINTF(E_LOG, L_MAIN, "Error reading command from command pipe: expected %zu bytes, read %d bytes\n", sizeof(cmd), ret); event_add(cmdbase->command_event, NULL); return; } // Execute the command function if (cmd->nonblock) { // Command is executed asynchronously command_cb_async(cmdbase, cmd); } else { // Command is executed synchronously, caller is waiting until signaled that the execution finished command_cb_sync(cmdbase, cmd); } } /* * Writes the given command to the command pipe */ static int send_command(struct commands_base *cmdbase, struct command *cmd) { int ret; if (!cmd->func) { DPRINTF(E_LOG, L_MAIN, "Programming error: send_command called with command->func NULL!\n"); return -1; } ret = write(cmdbase->command_pipe[1], &cmd, sizeof(cmd)); if (ret != sizeof(cmd)) { DPRINTF(E_LOG, L_MAIN, "Bad write to command pipe (write incomplete)\n"); return -1; } return 0; } /* * Creates a new command base, needs to be freed by commands_base_destroy or commands_base_free. * * @param evbase The libevent base to use for command handling * @param exit_cb Optional callback function to be called during commands_base_destroy */ struct commands_base * commands_base_new(struct event_base *evbase, command_exit_cb exit_cb) { struct commands_base *cmdbase; int ret; cmdbase = calloc(1, sizeof(struct commands_base)); if (!cmdbase) { DPRINTF(E_LOG, L_MAIN, "Out of memory for cmdbase\n"); return NULL; } #ifdef HAVE_PIPE2 ret = pipe2(cmdbase->command_pipe, O_CLOEXEC); #else ret = pipe(cmdbase->command_pipe); #endif if (ret < 0) { DPRINTF(E_LOG, L_MAIN, "Could not create command pipe: %s\n", strerror(errno)); free(cmdbase); return NULL; } cmdbase->command_event = event_new(evbase, cmdbase->command_pipe[0], EV_READ, command_cb, cmdbase); if (!cmdbase->command_event) { DPRINTF(E_LOG, L_MAIN, "Could not create cmd event\n"); close(cmdbase->command_pipe[0]); close(cmdbase->command_pipe[1]); free(cmdbase); return NULL; } ret = event_add(cmdbase->command_event, NULL); if (ret != 0) { DPRINTF(E_LOG, L_MAIN, "Could not add cmd event\n"); close(cmdbase->command_pipe[0]); close(cmdbase->command_pipe[1]); free(cmdbase); return NULL; } cmdbase->evbase = evbase; cmdbase->exit_cb = exit_cb; return cmdbase; } /* * Frees the command base and closes the (internally used) pipes */ int commands_base_free(struct commands_base *cmdbase) { event_free(cmdbase->command_event); close(cmdbase->command_pipe[0]); close(cmdbase->command_pipe[1]); free(cmdbase); return 0; } /* * Gets the current return value for the current pending command. * * If a command has more than one pending event, each event can access the previous set return value * if it depends on it. * * @param cmdbase The command base * @return The current return value */ int commands_exec_returnvalue(struct commands_base *cmdbase) { if (cmdbase->current_cmd == NULL) return 0; return cmdbase->current_cmd->ret; } /* * If a command function returned COMMAND_PENDING, each event triggered by this command needs to * call command_exec_end, passing it the return value of the event execution. * * If a command function is waiting for multiple events, each event needs to call command_exec_end. * The command base keeps track of the number of still pending events and only returns to the caller * if there are no pending events left. * * @param cmdbase The command base (holds the current pending command) * @param retvalue The return value for the calling thread */ void commands_exec_end(struct commands_base *cmdbase, int retvalue) { struct command *current_cmd = cmdbase->current_cmd; if (!current_cmd) return; // A pending event finished, decrease the number of pending events and update the return value current_cmd->pending--; current_cmd->ret = retvalue; DPRINTF(E_DBG, L_MAIN, "Command has %d pending events\n", current_cmd->pending); // If there are still pending events return if (current_cmd->pending > 0) return; // All pending events have finished, execute the bottom half and signal the caller that the command execution finished if (current_cmd->func_bh) current_cmd->func_bh(current_cmd->arg, ¤t_cmd->ret); cmdbase->current_cmd = NULL; /* Process commands again */ event_add(cmdbase->command_event, NULL); CHECK_ERR(L_MAIN, pthread_cond_signal(¤t_cmd->cond)); CHECK_ERR(L_MAIN, pthread_mutex_unlock(¤t_cmd->lck)); } /* * Execute the function 'func' with the given argument 'arg' in the event loop thread. * Blocks the caller (thread) until the function returned. * * If a function 'func_bh' ("bottom half") is given, it is executed after 'func' has successfully * finished. * * @param cmdbase The command base * @param func The function to be executed * @param func_bh The bottom half function to be executed after all pending events from func are processed * @param arg Argument passed to func (and func_bh) * @return Return value of func (or func_bh if func_bh is not NULL) */ int commands_exec_sync(struct commands_base *cmdbase, command_function func, command_function func_bh, void *arg) { struct command cmd; int ret; memset(&cmd, 0, sizeof(struct command)); cmd.func = func; cmd.func_bh = func_bh; cmd.arg = arg; cmd.nonblock = 0; CHECK_ERR(L_MAIN, mutex_init(&cmd.lck)); CHECK_ERR(L_MAIN, pthread_cond_init(&cmd.cond, NULL)); CHECK_ERR(L_MAIN, pthread_mutex_lock(&cmd.lck)); ret = send_command(cmdbase, &cmd); if (ret < 0) { DPRINTF(E_LOG, L_MAIN, "Error sending command\n"); cmd.ret = -1; } else { CHECK_ERR(L_MAIN, pthread_cond_wait(&cmd.cond, &cmd.lck)); } CHECK_ERR(L_MAIN, pthread_mutex_unlock(&cmd.lck)); CHECK_ERR(L_MAIN, pthread_cond_destroy(&cmd.cond)); CHECK_ERR(L_MAIN, pthread_mutex_destroy(&cmd.lck)); return cmd.ret; } /* * Execute the function 'func' with the given argument 'arg' in the event loop thread. * Triggers the function execution and immediately returns (does not wait for func to finish). * * The pointer passed as argument is freed in the event loop thread after func returned. * * @param cmdbase The command base * @param func The function to be executed * @param arg Argument passed to func * @return 0 if triggering the function execution succeeded, -1 on failure. */ int commands_exec_async(struct commands_base *cmdbase, command_function func, void *arg) { struct command *cmd; int ret; cmd = calloc(1, sizeof(struct command)); cmd->func = func; cmd->func_bh = NULL; cmd->arg = arg; cmd->nonblock = 1; ret = send_command(cmdbase, cmd); if (ret < 0) { free(cmd); return -1; } return 0; } /* * Command to break the libevent loop * * If the command base was created with an exit_cb function, exit_cb is called before breaking the * libevent loop. * * @param arg The command base * @param retval Always set to COMMAND_END */ static enum command_state cmdloop_exit(void *arg, int *retval) { struct commands_base *cmdbase = arg; *retval = 0; if (cmdbase->exit_cb) cmdbase->exit_cb(); event_base_loopbreak(cmdbase->evbase); return COMMAND_END; } /* * Break the libevent loop for the given command base, closes the internally used pipes * and frees the command base. * * @param cmdbase The command base */ void commands_base_destroy(struct commands_base *cmdbase) { commands_exec_sync(cmdbase, cmdloop_exit, NULL, cmdbase); commands_base_free(cmdbase); }