/* * Copyright (c) 2005-2009 Voltaire, Inc. All rights reserved. * Copyright (c) 2008 HNR Consulting. All rights reserved. * Copyright (c) 2013 Oracle and/or its affiliates. All rights reserved. * * This software is available to you under a choice of one of two * licenses. You may choose to be licensed under the terms of the GNU * General Public License (GPL) Version 2, available from the file * COPYING in the main directory of this source tree, or the * OpenIB.org BSD license below: * * Redistribution and use in source and binary forms, with or * without modification, are permitted provided that the following * conditions are met: * * - Redistributions of source code must retain the above * copyright notice, this list of conditions and the following * disclaimer. * * - Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials * provided with the distribution. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. * */ /* * Abstract: * Provide a framework for the Console which decouples the connection * or I/O from the functionality, or commands. * * Extensible - allows a variety of connection methods independent of * the console commands. */ #if HAVE_CONFIG_H # include #endif /* HAVE_CONFIG_H */ #ifdef ENABLE_OSM_CONSOLE_LOOPBACK #include #include #include #include #endif #include #include #include #include #define FILE_ID OSM_FILE_CONSOLE_IO_C #include #include static int is_local(char *str) { /* convenience - checks if just stdin/stdout */ if (str) return (strcmp(str, OSM_LOCAL_CONSOLE) == 0); return 0; } #ifdef ENABLE_OSM_CONSOLE_LOOPBACK static int is_loopback(char *str) { /* convenience - checks if socket based connection */ if (str) return (strcmp(str, OSM_LOOPBACK_CONSOLE) == 0); return 0; } #else #define is_loopback is_local #endif #ifdef ENABLE_OSM_CONSOLE_SOCKET static int is_remote(char *str) { /* convenience - checks if socket based connection */ if (str) return strcmp(str, OSM_REMOTE_CONSOLE) == 0 || is_loopback(str); return 0; } #else #define is_remote is_loopback #endif int is_console_enabled(osm_subn_opt_t * p_opt) { /* checks for a variety of types of consoles - default is off or 0 */ if (p_opt) return is_local(p_opt->console) || is_loopback(p_opt->console) || is_remote(p_opt->console); return 0; } #ifdef ENABLE_OSM_CONSOLE_LOOPBACK int cio_close(osm_console_t * p_oct, osm_log_t * p_log) { int rtnval = -1; if (p_oct && p_oct->in_fd > 0) { OSM_LOG(p_log, OSM_LOG_VERBOSE, "Console connection closed: %s (%s)\n", p_oct->client_hn, p_oct->client_ip); rtnval = fclose(p_oct->in); p_oct->in_fd = -1; p_oct->out_fd = -1; p_oct->in = NULL; p_oct->out = NULL; } return rtnval; } int cio_open(osm_console_t * p_oct, int new_fd, osm_log_t * p_log) { /* returns zero if opened fine, -1 otherwise */ char *p_line; size_t len; ssize_t n; if (p_oct->in_fd >= 0) { FILE *file = fdopen(new_fd, "w+"); fprintf(file, "OpenSM Console connection already in use\n" " kill other session (y/n)? "); fflush(file); p_line = NULL; n = getline(&p_line, &len, file); if (n > 0 && (p_line[0] == 'y' || p_line[0] == 'Y')) cio_close(p_oct, p_log); else { OSM_LOG(p_log, OSM_LOG_INFO, "Console connection aborted: %s (%s) - " "already in use\n", p_oct->client_hn, p_oct->client_ip); fclose(file); free(p_line); return -1; } free(p_line); } p_oct->in_fd = new_fd; p_oct->out_fd = p_oct->in_fd; p_oct->in = fdopen(p_oct->in_fd, "w+"); p_oct->out = p_oct->in; osm_console_prompt(p_oct->out); OSM_LOG(p_log, OSM_LOG_VERBOSE, "Console connection accepted: %s (%s)\n", p_oct->client_hn, p_oct->client_ip); return (p_oct->in == NULL) ? -1 : 0; } /********************************************************************** * Do authentication & authorization check **********************************************************************/ int is_authorized(osm_console_t * p_oct) { /* allowed to use the console? */ p_oct->authorized = !is_remote(p_oct->client_type) || hosts_ctl((char *)OSM_DAEMON_NAME, p_oct->client_hn, p_oct->client_ip, (char *)STRING_UNKNOWN); return p_oct->authorized; } #endif void osm_console_prompt(FILE * out) { if (out) { fprintf(out, "OpenSM %s", OSM_COMMAND_PROMPT); fflush(out); } } int osm_console_init(osm_subn_opt_t * opt, osm_console_t * p_oct, osm_log_t * p_log) { p_oct->socket = -1; strncpy(p_oct->client_type, opt->console, sizeof(p_oct->client_type) - 1); p_oct->client_type[sizeof(p_oct->client_type) - 1] = '\0'; /* set up the file descriptors for the console */ if (strcmp(opt->console, OSM_LOCAL_CONSOLE) == 0) { p_oct->in = stdin; p_oct->out = stdout; p_oct->in_fd = fileno(stdin); p_oct->out_fd = fileno(stdout); osm_console_prompt(p_oct->out); #ifdef ENABLE_OSM_CONSOLE_LOOPBACK } else if (strcmp(opt->console, OSM_LOOPBACK_CONSOLE) == 0 #ifdef ENABLE_OSM_CONSOLE_SOCKET || strcmp(opt->console, OSM_REMOTE_CONSOLE) == 0 #endif ) { struct sockaddr_in sin; int optval = 1; if ((p_oct->socket = socket(AF_INET, SOCK_STREAM, 0)) < 0) { OSM_LOG(p_log, OSM_LOG_ERROR, "ERR 4B01: Failed to open console socket: %s\n", strerror(errno)); return -1; } if (setsockopt(p_oct->socket, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval))) { OSM_LOG(p_log, OSM_LOG_ERROR, "ERR 4B06: Failed to set socket option: %s\n", strerror(errno)); return -1; } sin.sin_family = AF_INET; sin.sin_port = htons(opt->console_port); #ifdef ENABLE_OSM_CONSOLE_SOCKET if (strcmp(opt->console, OSM_REMOTE_CONSOLE) == 0) sin.sin_addr.s_addr = htonl(INADDR_ANY); else #endif sin.sin_addr.s_addr = htonl(INADDR_LOOPBACK); if (bind(p_oct->socket, &sin, sizeof(sin)) < 0) { OSM_LOG(p_log, OSM_LOG_ERROR, "ERR 4B02: Failed to bind console socket: %s\n", strerror(errno)); return -1; } if (listen(p_oct->socket, 1) < 0) { OSM_LOG(p_log, OSM_LOG_ERROR, "ERR 4B03: Failed to listen on console socket: %s\n", strerror(errno)); return -1; } signal(SIGPIPE, SIG_IGN); /* protect ourselves from closed pipes */ p_oct->in = NULL; p_oct->out = NULL; p_oct->in_fd = -1; p_oct->out_fd = -1; OSM_LOG(p_log, OSM_LOG_INFO, "Console listening on port %d\n", opt->console_port); #endif } return 0; } /* clean up and release resources */ void osm_console_exit(osm_console_t * p_oct, osm_log_t * p_log) { #ifdef ENABLE_OSM_CONSOLE_LOOPBACK cio_close(p_oct, p_log); if (p_oct->socket > 0) { OSM_LOG(p_log, OSM_LOG_INFO, "Closing console socket\n"); close(p_oct->socket); p_oct->socket = -1; } #endif }