1
1
openmpi/opal/util/show_help.c
Gilles Gouaillardet ca3a275823 opal/util: fix misc memory leaks reported by Coverity
fixes CID 996174, 996920, 1196735, 1196769 and 1196770
2015-02-13 14:28:59 +09:00

370 строки
9.8 KiB
C

/*
* Copyright (c) 2004-2005 The Trustees of Indiana University and Indiana
* University Research and Technology
* Corporation. All rights reserved.
* Copyright (c) 2004-2006 The University of Tennessee and The University
* of Tennessee Research Foundation. All rights
* reserved.
* Copyright (c) 2004-2005 High Performance Computing Center Stuttgart,
* University of Stuttgart. All rights reserved.
* Copyright (c) 2004-2005 The Regents of the University of California.
* All rights reserved.
* Copyright (c) 2008 Cisco Systems, Inc. All rights reserved.
* Copyright (c) 2015 Research Organization for Information Science
* and Technology (RIST). All rights reserved.
* $COPYRIGHT$
*
* Additional copyrights may follow
*
* $HEADER$
*/
#include "opal_config.h"
#include <stdio.h>
#include <string.h>
#include <locale.h>
#include <errno.h>
#include "opal/mca/installdirs/installdirs.h"
#include "opal/util/show_help.h"
#include "opal/util/show_help_lex.h"
#include "opal/util/printf.h"
#include "opal/util/argv.h"
#include "opal/util/os_path.h"
#include "opal/util/output.h"
#include "opal/constants.h"
/*
* Private variables
*/
static const char *default_filename = "help-messages";
static const char *dash_line = "--------------------------------------------------------------------------\n";
static int output_stream = -1;
static char **search_dirs = NULL;
/*
* Local functions
*/
static int opal_show_vhelp_internal(const char *filename, const char *topic,
bool want_error_header, va_list arglist);
static int opal_show_help_internal(const char *filename, const char *topic,
bool want_error_header, ...);
opal_show_help_fn_t opal_show_help = opal_show_help_internal;
opal_show_vhelp_fn_t opal_show_vhelp = opal_show_vhelp_internal;
int opal_show_help_init(void)
{
opal_output_stream_t lds;
OBJ_CONSTRUCT(&lds, opal_output_stream_t);
lds.lds_want_stderr = true;
output_stream = opal_output_open(&lds);
opal_argv_append_nosize(&search_dirs, opal_install_dirs.opaldatadir);
return OPAL_SUCCESS;
}
int opal_show_help_finalize(void)
{
opal_output_close(output_stream);
output_stream = -1;
/* destruct the search list */
if (NULL != search_dirs) {
opal_argv_free(search_dirs);
search_dirs = NULL;
};
return OPAL_SUCCESS;
}
/*
* Make one big string with all the lines. This isn't the most
* efficient method in the world, but we're going for clarity here --
* not optimization. :-)
*/
static int array2string(char **outstring,
bool want_error_header, char **lines)
{
int i, count;
size_t len;
/* See how much space we need */
len = want_error_header ? 2 * strlen(dash_line) : 0;
count = opal_argv_count(lines);
for (i = 0; i < count; ++i) {
if (NULL == lines[i]) {
break;
}
len += strlen(lines[i]) + 1;
}
/* Malloc it out */
(*outstring) = (char*) malloc(len + 1);
if (NULL == *outstring) {
return OPAL_ERR_OUT_OF_RESOURCE;
}
/* Fill the big string */
*(*outstring) = '\0';
if (want_error_header) {
strcat(*outstring, dash_line);
}
for (i = 0; i < count; ++i) {
if (NULL == lines[i]) {
break;
}
strcat(*outstring, lines[i]);
strcat(*outstring, "\n");
}
if (want_error_header) {
strcat(*outstring, dash_line);
}
return OPAL_SUCCESS;
}
/*
* Find the right file to open
*/
static int open_file(const char *base, const char *topic)
{
char *filename;
char *err_msg = NULL;
size_t base_len;
int i;
/* If no filename was supplied, use the default */
if (NULL == base) {
base = default_filename;
}
/* if this is called prior to someone initializing the system,
* then don't try to look
*/
if (NULL != search_dirs) {
/* Try to open the file. If we can't find it, try it with a .txt
* extension.
*/
for (i=0; NULL != search_dirs[i]; i++) {
filename = opal_os_path( false, search_dirs[i], base, NULL );
opal_show_help_yyin = fopen(filename, "r");
if (NULL == opal_show_help_yyin) {
asprintf(&err_msg, "%s: %s", filename, strerror(errno));
base_len = strlen(base);
if (4 > base_len || 0 != strcmp(base + base_len - 4, ".txt")) {
free(filename);
asprintf(&filename, "%s%s%s.txt", search_dirs[i], OPAL_PATH_SEP, base);
opal_show_help_yyin = fopen(filename, "r");
}
}
free(filename);
if (NULL != opal_show_help_yyin) {
break;
}
}
}
/* If we still couldn't open it, then something is wrong */
if (NULL == opal_show_help_yyin) {
opal_output(output_stream, "%sSorry! You were supposed to get help about:\n %s\nBut I couldn't open the help file:\n %s. Sorry!\n%s", dash_line, topic, err_msg, dash_line);
free(err_msg);
return OPAL_ERR_NOT_FOUND;
}
if (NULL != err_msg) {
free(err_msg);
}
/* Set the buffer */
opal_show_help_init_buffer(opal_show_help_yyin);
/* Happiness */
return OPAL_SUCCESS;
}
/*
* In the file that has already been opened, find the topic that we're
* supposed to output
*/
static int find_topic(const char *base, const char *topic)
{
int token, ret;
char *tmp;
/* Examine every topic */
while (1) {
token = opal_show_help_yylex();
switch (token) {
case OPAL_SHOW_HELP_PARSE_TOPIC:
tmp = strdup(opal_show_help_yytext);
if (NULL == tmp) {
return OPAL_ERR_OUT_OF_RESOURCE;
}
tmp[strlen(tmp) - 1] = '\0';
ret = strcmp(tmp + 1, topic);
free(tmp);
if (0 == ret) {
return OPAL_SUCCESS;
}
break;
case OPAL_SHOW_HELP_PARSE_MESSAGE:
break;
case OPAL_SHOW_HELP_PARSE_DONE:
opal_output(output_stream, "%sSorry! You were supposed to get help about:\n %s\nfrom the file:\n %s\nBut I couldn't find that topic in the file. Sorry!\n%s", dash_line, topic, base, dash_line);
return OPAL_ERR_NOT_FOUND;
break;
default:
break;
}
}
/* Never get here */
}
/*
* We have an open file, and we're pointed at the right topic. So
* read in all the lines in the topic and make a list of them.
*/
static int read_topic(char ***array)
{
int token, rc;
while (1) {
token = opal_show_help_yylex();
switch (token) {
case OPAL_SHOW_HELP_PARSE_MESSAGE:
/* opal_argv_append_nosize does strdup(opal_show_help_yytext) */
rc = opal_argv_append_nosize(array, opal_show_help_yytext);
if (rc != OPAL_SUCCESS) {
return rc;
}
break;
default:
return OPAL_SUCCESS;
break;
}
}
/* Never get here */
}
static int load_array(char ***array, const char *filename, const char *topic)
{
int ret;
if (OPAL_SUCCESS != (ret = open_file(filename, topic))) {
return ret;
}
ret = find_topic(filename, topic);
if (OPAL_SUCCESS == ret) {
ret = read_topic(array);
}
fclose(opal_show_help_yyin);
opal_show_help_yylex_destroy ();
if (OPAL_SUCCESS != ret) {
opal_argv_free(*array);
}
return ret;
}
char *opal_show_help_vstring(const char *filename, const char *topic,
bool want_error_header, va_list arglist)
{
int rc;
char *single_string, *output, **array = NULL;
/* Load the message */
if (OPAL_SUCCESS != (rc = load_array(&array, filename, topic))) {
return NULL;
}
/* Convert it to a single raw string */
rc = array2string(&single_string, want_error_header, array);
if (OPAL_SUCCESS == rc) {
/* Apply the formatting to make the final output string */
vasprintf(&output, single_string, arglist);
free(single_string);
}
opal_argv_free(array);
return (OPAL_SUCCESS == rc) ? output : NULL;
}
char *opal_show_help_string(const char *filename, const char *topic,
bool want_error_handler, ...)
{
char *output;
va_list arglist;
va_start(arglist, want_error_handler);
output = opal_show_help_vstring(filename, topic, want_error_handler,
arglist);
va_end(arglist);
return output;
}
static int opal_show_vhelp_internal(const char *filename, const char *topic,
bool want_error_header, va_list arglist)
{
char *output;
/* Convert it to a single string */
output = opal_show_help_vstring(filename, topic, want_error_header,
arglist);
/* If we got a single string, output it with formatting */
if (NULL != output) {
opal_output(output_stream, "%s", output);
free(output);
}
return (NULL == output) ? OPAL_ERROR : OPAL_SUCCESS;
}
static int opal_show_help_internal(const char *filename, const char *topic,
bool want_error_header, ...)
{
va_list arglist;
int rc;
/* Convert it to a single string */
va_start(arglist, want_error_header);
rc = opal_show_vhelp(filename, topic, want_error_header, arglist);
va_end(arglist);
return rc;
}
int opal_show_help_add_dir(const char *directory)
{
opal_argv_append_nosize(&search_dirs, directory);
return OPAL_SUCCESS;
}