mirror of
https://github.com/Unidata/netcdf-c.git
synced 2025-01-12 15:45:21 +08:00
6934aa2e8b
re: https://github.com/Unidata/netcdf-c/issues/1373 (partial) * Mark some global constants be const to indicate to make them easier to track. * Hide direct access to the ncrc_globalstate behind a function call. * Convert dispatch tables to constants (except the user defined ones) This has some consequences in terms of function arguments needing to be marked as const also. * Remove some no longer needed global fields * Aggregate all the globals in nclog.c * Uniformly replace nc_sizevector{0,1} with NC_coord_{zero,one} * Uniformly replace nc_ptrdffvector1 with NC_stride_one * Remove some obsolete code
234 lines
5.3 KiB
C
234 lines
5.3 KiB
C
/*********************************************************************
|
|
* Copyright 2018, UCAR/Unidata
|
|
* See netcdf/COPYRIGHT file for copying and redistribution conditions.
|
|
* $Header$
|
|
*********************************************************************/
|
|
|
|
#include "config.h"
|
|
#ifdef _MSC_VER
|
|
#include<io.h>
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <stdarg.h>
|
|
#include <string.h>
|
|
#ifdef HAVE_FCNTL_H
|
|
#include <fcntl.h>
|
|
#endif
|
|
|
|
extern FILE* fdopen(int fd, const char *mode);
|
|
|
|
#include "nclog.h"
|
|
|
|
#define PREFIXLEN 8
|
|
#define MAXTAGS 256
|
|
#define NCTAGDFALT "Log";
|
|
|
|
|
|
static int nclogginginitialized = 0;
|
|
|
|
static struct NCLOGGLOBAL {
|
|
int nclogging;
|
|
int ncsystemfile; /* 1 => we are logging to file we did not open */
|
|
char* nclogfile;
|
|
FILE* nclogstream;
|
|
} nclog_global = {0,0,NULL,NULL};
|
|
|
|
static const char* nctagset[] = {"Warning","Error","Note","Debug"};
|
|
static const int nctagsize = sizeof(nctagset)/sizeof(char*);
|
|
|
|
/* Forward */
|
|
static const char* nctagname(int tag);
|
|
|
|
/*!\defgroup NClog NClog Management
|
|
@{*/
|
|
|
|
/*!\internal
|
|
*/
|
|
|
|
void
|
|
ncloginit(void)
|
|
{
|
|
const char* file;
|
|
if(nclogginginitialized)
|
|
return;
|
|
nclogginginitialized = 1;
|
|
memset(&nclog_global,0,sizeof(nclog_global));
|
|
ncsetlogging(0);
|
|
nclog_global.nclogfile = NULL;
|
|
nclog_global.nclogstream = NULL;
|
|
/* Use environment variables to preset nclogging state*/
|
|
/* I hope this is portable*/
|
|
file = getenv(NCENVFLAG);
|
|
if(file != NULL && strlen(file) > 0) {
|
|
if(nclogopen(file)) {
|
|
ncsetlogging(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*!
|
|
Enable/Disable logging.
|
|
|
|
\param[in] tf If 1, then turn on logging, if 0, then turn off logging.
|
|
|
|
\return The previous value of the logging flag.
|
|
*/
|
|
|
|
int
|
|
ncsetlogging(int tf)
|
|
{
|
|
int was;
|
|
if(!nclogginginitialized) ncloginit();
|
|
was = nclog_global.nclogging;
|
|
nclog_global.nclogging = tf;
|
|
return was;
|
|
}
|
|
|
|
/*!
|
|
Specify a file into which to place logging output.
|
|
|
|
\param[in] file The name of the file into which to place logging output.
|
|
If the file has the value NULL, then send logging output to
|
|
stderr.
|
|
|
|
\return zero if the open failed, one otherwise.
|
|
*/
|
|
|
|
int
|
|
nclogopen(const char* file)
|
|
{
|
|
if(!nclogginginitialized) ncloginit();
|
|
nclogclose();
|
|
if(file == NULL || strlen(file) == 0) {
|
|
/* use stderr*/
|
|
nclog_global.nclogstream = stderr;
|
|
nclog_global.nclogfile = NULL;
|
|
nclog_global.ncsystemfile = 1;
|
|
} else if(strcmp(file,"stdout") == 0) {
|
|
/* use stdout*/
|
|
nclog_global.nclogstream = stdout;
|
|
nclog_global.nclogfile = NULL;
|
|
nclog_global.ncsystemfile = 1;
|
|
} else if(strcmp(file,"stderr") == 0) {
|
|
/* use stderr*/
|
|
nclog_global.nclogstream = stderr;
|
|
nclog_global.nclogfile = NULL;
|
|
nclog_global.ncsystemfile = 1;
|
|
} else {
|
|
int fd;
|
|
nclog_global.nclogfile = strdup(file);
|
|
nclog_global.nclogstream = NULL;
|
|
/* We need to deal with this file carefully
|
|
to avoid unauthorized access*/
|
|
fd = open(nclog_global.nclogfile,O_WRONLY|O_APPEND|O_CREAT,0600);
|
|
if(fd >= 0) {
|
|
nclog_global.nclogstream = fdopen(fd,"a");
|
|
} else {
|
|
free(nclog_global.nclogfile);
|
|
nclog_global.nclogfile = NULL;
|
|
nclog_global.nclogstream = NULL;
|
|
ncsetlogging(0);
|
|
return 0;
|
|
}
|
|
nclog_global.ncsystemfile = 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
void
|
|
nclogclose(void)
|
|
{
|
|
if(!nclogginginitialized) ncloginit();
|
|
if(nclog_global.nclogstream != NULL && !nclog_global.ncsystemfile) {
|
|
fclose(nclog_global.nclogstream);
|
|
}
|
|
if(nclog_global.nclogfile != NULL) free(nclog_global.nclogfile);
|
|
nclog_global.nclogstream = NULL;
|
|
nclog_global.nclogfile = NULL;
|
|
nclog_global.ncsystemfile = 0;
|
|
}
|
|
|
|
/*!
|
|
Send logging messages. This uses a variable
|
|
number of arguments and operates like the stdio
|
|
printf function.
|
|
|
|
\param[in] tag Indicate the kind of this log message.
|
|
\param[in] format Format specification as with printf.
|
|
*/
|
|
|
|
void
|
|
nclog(int tag, const char* fmt, ...)
|
|
{
|
|
va_list args;
|
|
const char* prefix;
|
|
|
|
if(!nclogginginitialized) ncloginit();
|
|
|
|
if(!nclog_global.nclogging || nclog_global.nclogstream == NULL) return;
|
|
|
|
prefix = nctagname(tag);
|
|
fprintf(nclog_global.nclogstream,"%s:",prefix);
|
|
|
|
if(fmt != NULL) {
|
|
va_start(args, fmt);
|
|
vfprintf(nclog_global.nclogstream, fmt, args);
|
|
va_end( args );
|
|
}
|
|
fprintf(nclog_global.nclogstream, "\n" );
|
|
fflush(nclog_global.nclogstream);
|
|
}
|
|
|
|
void
|
|
ncvlog(int tag, const char* fmt, va_list ap)
|
|
{
|
|
const char* prefix;
|
|
|
|
if(!nclogginginitialized) ncloginit();
|
|
|
|
if(!nclog_global.nclogging || nclog_global.nclogstream == NULL) return;
|
|
|
|
prefix = nctagname(tag);
|
|
fprintf(nclog_global.nclogstream,"%s:",prefix);
|
|
|
|
if(fmt != NULL) {
|
|
vfprintf(nclog_global.nclogstream, fmt, ap);
|
|
}
|
|
fprintf(nclog_global.nclogstream, "\n" );
|
|
fflush(nclog_global.nclogstream);
|
|
}
|
|
|
|
void
|
|
nclogtext(int tag, const char* text)
|
|
{
|
|
nclogtextn(tag,text,strlen(text));
|
|
}
|
|
|
|
/*!
|
|
Send arbitrarily long text as a logging message.
|
|
Each line will be sent using nclog with the specified tag.
|
|
\param[in] tag Indicate the kind of this log message.
|
|
\param[in] text Arbitrary text to send as a logging message.
|
|
*/
|
|
|
|
void
|
|
nclogtextn(int tag, const char* text, size_t count)
|
|
{
|
|
NC_UNUSED(tag);
|
|
if(!nclog_global.nclogging || nclog_global.nclogstream == NULL) return;
|
|
fwrite(text,1,count,nclog_global.nclogstream);
|
|
fflush(nclog_global.nclogstream);
|
|
}
|
|
|
|
static const char*
|
|
nctagname(int tag)
|
|
{
|
|
if(tag < 0 || tag >= nctagsize)
|
|
return "unknown";
|
|
return nctagset[tag];
|
|
}
|
|
|
|
/**@}*/
|