//commands.c: /* * Copyright (C) Philipp 'ph3-der-loewe' Schafft - 2008-2013 * * This file is part of roard a part of RoarAudio, * a cross-platform sound system for both, home and professional use. * See README for details. * * This file is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 3 * as published by the Free Software Foundation. * * RoarAudio 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 software; see the file COPYING. If not, write to * the Free Software Foundation, 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. * */ #include "roard.h" #if !defined(ROAR_TARGET_MICROCONTROLLER) && !defined(ROAR_MINIMAL) #define _NAME(x) (x) #else #define _NAME(x) ((char*)NULL) #endif struct roar_command g_commands[COMMAND_MAX_COMMANDS] = { {ROAR_CMD_NOOP, _NAME("NOOP"), req_on_noop, ACCLEV_NONE}, {ROAR_CMD_IDENTIFY, _NAME("IDENTIFY"), req_on_identify, ACCLEV_NONE}, {ROAR_CMD_AUTH, _NAME("AUTH"), req_on_auth, ACCLEV_IDENTED}, {ROAR_CMD_WHOAMI, _NAME("WHOAMI"), req_on_whoami, ACCLEV_NONE}, {ROAR_CMD_NEW_STREAM, _NAME("NEW_STREAM"), req_on_new_stream, ACCLEV_USER}, #ifdef ROAR_SUPPORT_META {ROAR_CMD_SET_META, _NAME("SET_META"), req_on_set_meta, ACCLEV_PWRUSER}, {ROAR_CMD_GET_META, _NAME("GET_META"), req_on_get_meta, ACCLEV_GUEST}, {ROAR_CMD_LIST_META, _NAME("LIST_META"), req_on_list_meta, ACCLEV_GUEST}, #endif {ROAR_CMD_EXEC_STREAM, _NAME("EXEC_STREAM"), req_on_exec_stream, ACCLEV_PWRUSER}, {ROAR_CMD_QUIT, _NAME("QUIT"), req_on_quit, ACCLEV_NONE}, {ROAR_CMD_CON_STREAM, _NAME("CON_STREAM"), req_on_con_stream, ACCLEV_PWRUSER}, {ROAR_CMD_PASSFH, _NAME("PASSFH"), req_on_passfh, ACCLEV_PWRUSER}, {ROAR_CMD_GETTIMEOFDAY, _NAME("GETTIMEOFDAY"), req_on_gettimeofday, ACCLEV_NONE}, {ROAR_CMD_SERVER_INFO, _NAME("SERVER_INFO"), req_on_server_info, ACCLEV_NONE}, // allow this early so the client // can decide early if this server // provieds all needed features {ROAR_CMD_SERVER_OINFO, _NAME("SERVER_OINFO"), req_on_server_oinfo, ACCLEV_IDENTED}, // same as above {ROAR_CMD_CAPS, _NAME("CAPS"), req_on_caps, ACCLEV_IDENTED}, // and again the same {ROAR_CMD_GET_STANDBY, _NAME("GET_STANDBY"), req_on_get_standby, ACCLEV_GUEST}, {ROAR_CMD_SET_STANDBY, _NAME("SET_STANDBY"), req_on_set_standby, ACCLEV_PWRUSER}, // should this be set to ACCLEV_ALL? {ROAR_CMD_EXIT, _NAME("EXIT"), req_on_exit, ACCLEV_ALL}, {ROAR_CMD_LIST_CLIENTS, _NAME("LIST_CLIENTS"), req_on_list_clients, ACCLEV_GUEST}, {ROAR_CMD_LIST_STREAMS, _NAME("LIST_STREAMS"), req_on_list_streams, ACCLEV_GUEST}, {ROAR_CMD_GET_CLIENT, _NAME("GET_CLIENT"), req_on_get_client, ACCLEV_GUEST}, {ROAR_CMD_GET_STREAM, _NAME("GET_STREAM"), req_on_get_stream, ACCLEV_GUEST}, {ROAR_CMD_KICK, _NAME("KICK"), req_on_kick, ACCLEV_PWRUSER}, {ROAR_CMD_ATTACH, _NAME("ATTACH"), req_on_attach, ACCLEV_PWRUSER}, {ROAR_CMD_SET_VOL, _NAME("SET_VOL"), req_on_set_vol, ACCLEV_PWRUSER}, {ROAR_CMD_GET_VOL, _NAME("GET_VOL"), req_on_get_vol, ACCLEV_GUEST}, {ROAR_CMD_GET_STREAM_PARA, _NAME("GET_STREAM_PARA"), req_on_get_stream_para, ACCLEV_GUEST}, {ROAR_CMD_SET_STREAM_PARA, _NAME("SET_STREAM_PARA"), req_on_set_stream_para, ACCLEV_PWRUSER}, {ROAR_CMD_ADD_DATA, _NAME("ADD_DATA"), req_on_add_data, ACCLEV_PWRUSER}, {ROAR_CMD_BEEP, _NAME("BEEP"), req_on_beep, ACCLEV_USER}, {ROAR_CMD_WAIT, _NAME("WAIT"), req_on_wait, ACCLEV_USER}, {ROAR_CMD_EOL, _NAME("END OF LIST"), NULL, ACCLEV_NONE} }; int command_get_id_by_cmd (int command) { int i; int cmd; for (i = 0; i < COMMAND_MAX_COMMANDS; i++) { if ( (cmd = g_commands[i].cmd) == ROAR_CMD_EOL ) break; if ( cmd == command ) return i; } return -1; } int command_exec (int client, struct roar_message * mes, char ** data, uint32_t flags[2]) { int cmd = command_get_id_by_cmd(mes->cmd); int (*func)(int client, struct roar_message * mes, char ** data, uint32_t flags[2]); #ifdef DEBUG int ret; #endif if ( cmd == -1 ) return -1; // NOTE: This is optimized for speed here, so we do not call clients_get(). // we maybe should do but current just don't. /client/ must be valid anyway. if ( g_commands[cmd].minacclev > g_clients[client]->acclev ) { ROAR_WARN("command_exec(client=%i, mes=%p{.cmd=%i,...}, data=%p, flags=%p) = -1 // client not allowed to use command %s", client, mes, (int)mes->cmd, data, flags, g_commands[cmd].name); return -1; } if ( (func = g_commands[cmd].handler) == NULL ) { ROAR_WARN("command_exec(client=%i, mes=%p{.cmd=%i,...}, data=%p, flags=%p) = -1 // unknown command", client, mes, (int)mes->cmd, data, flags); return -1; } ROAR_DBG("command_exec(*): Execing command %i(%s) via %p", cmd, g_commands[cmd].name, func); #ifdef DEBUG ROAR_DBG("command_exec(client=%i, mes=%p{.cmd=%i,...}, data=%p{%p}, flags=%p): func=%p", client, mes, (int)mes->cmd, data, *data, flags, func); ret = func(client, mes, data, flags); ROAR_DBG("command_exec(client=%i, mes=%p{.cmd=%i,...}, data=%p{%p}, flags=%p) = %i", client, mes, (int)mes->cmd, data, *data, flags, ret); return ret; #else return func(client, mes, data, flags); #endif } int command_get_name (int command, const char ** name) { int cmd = command_get_id_by_cmd(command); if ( cmd == -1 ) { *name = NULL; return -1; } *name = g_commands[cmd].name; return 0; } //ll