netcdf-c/libdispatch/nclog.c
Dennis Heimbigner 9380790ea8 Support MSYS2/Mingw platform
re:

The current netcdf-c release has some problems with the mingw platform
on windows. Mostly they are path issues.

Changes to support mingw+msys2:
-------------------------------
* Enable option of looking into the windows registry to find
  the mingw root path. In aid of proper path handling.
* Add mingw+msys as a specific platform in configure.ac and move testing
  of the platform to the front so it is available early.
* Handle mingw X libncpoco (dynamic loader) properly even though
  mingw does not yet support it.
* Handle mingw X plugins properly even though mingw does not yet support it.
* Alias pwd='pwd -W' to better handle paths in shell scripts.
* Plus a number of other minor compile irritations.
* Disallow the use of multiple nc_open's on the same file for windows
  (and mingw) because windows does not seem to handle these properly.
  Not sure why we did not catch this earlier.
* Add mountpoint info to dpathmgr.c to help support mingw.
* Cleanup dpathmgr conversions.

Known problems:
---------------
* I have not been able to get shared libraries to work, so
  plugins/filters must be disabled.
* There is some kind of problem with libcurl that I have not solved,
  so all uses of libcurl (currently DAP+Byterange) must be disabled.

Misc. other fixes:
------------------
* Cleanup the relationship between ENABLE_PLUGINS and various other flags
  in CMakeLists.txt and configure.ac.
* Re-arrange the TESTDIRS order in Makefile.am.
* Add pseudo-breakpoint to nclog.[ch] for debugging.
* Improve the documentation of the path manager code in ncpathmgr.h
* Add better support for relative paths in dpathmgr.c
* Default the mode args to NCfopen to include "b" (binary) for windows.
* Add optional debugging output in various places.
* Make sure that everything builds with plugins disabled.
* Fix numerous (s)printf inconsistencies betweenb the format spec
  and the arguments.
2021-12-23 22:18:56 -07:00

322 lines
7.1 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>
#include <errno.h>
#ifdef HAVE_FCNTL_H
#include <fcntl.h>
#endif
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#ifdef HAVE_EXECINFO_H
#include <execinfo.h>
#endif
#include "netcdf.h"
#include "nclog.h"
#define PREFIXLEN 8
#define MAXTAGS 256
#define NCTAGDFALT "Log";
#define NC_MAX_FRAMES 256
static int nclogginginitialized = 0;
static struct NCLOGGLOBAL {
int nclogging;
int tracelevel;
FILE* nclogstream;
int depth;
struct Frame {
const char* fcn;
int level;
int depth;
} frames[NC_MAX_FRAMES];
} nclog_global = {0,-1,NULL};
static const char* nctagset[] = {"Note","Warning","Error","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* envv = NULL;
if(nclogginginitialized)
return;
nclogginginitialized = 1;
memset(&nclog_global,0,sizeof(nclog_global));
nclog_global.tracelevel = -1;
ncsetlogging(0);
nclog_global.nclogstream = stderr;
/* Use environment variables to preset nclogging state*/
/* I hope this is portable*/
envv = getenv(NCENVLOGGING);
if(envv != NULL) {
ncsetlogging(1);
}
envv = getenv(NCENVTRACING);
if(envv != NULL) {
nctracelevel(atoi(envv));
}
}
/*!
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;
if(nclog_global.nclogstream == NULL) nclogopen(NULL);
return was;
}
int
nclogopen(FILE* stream)
{
if(!nclogginginitialized) ncloginit();
if(stream == NULL) stream = stderr;
nclog_global.nclogstream = stream;
return 1;
}
/*!
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, ...)
{
if(fmt != NULL) {
va_list args;
va_start(args, fmt);
ncvlog(tag,fmt,args);
va_end(args);
}
}
int
ncvlog(int tag, const char* fmt, va_list ap)
{
const char* prefix;
int was = -1;
if(!nclogginginitialized) ncloginit();
if(tag == NCLOGERR) was = ncsetlogging(1);
if(!nclog_global.nclogging || nclog_global.nclogstream == NULL) return was;
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);
return was;
}
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];
}
/*!
Send trace messages.
\param[in] level Indicate the level of trace
\param[in] format Format specification as with printf.
*/
int
nctracelevel(int level)
{
int oldlevel;
if(!nclogginginitialized) ncloginit();
oldlevel = nclog_global.tracelevel;
if(level < 0) {
nclog_global.tracelevel = level;
ncsetlogging(0);
} else { /*(level >= 0)*/
nclog_global.tracelevel = level;
ncsetlogging(1);
nclogopen(NULL); /* use stderr */
}
return oldlevel;
}
void
nctrace(int level, const char* fcn, const char* fmt, ...)
{
va_list args;
va_start(args, fmt);
ncvtrace(level,fcn,fmt,args);
va_end(args);
}
void
nctracemore(int level, const char* fmt, ...)
{
va_list args;
va_start(args, fmt);
ncvtrace(level,NULL,fmt,args);
va_end(args);
}
void
ncvtrace(int level, const char* fcn, const char* fmt, va_list ap)
{
struct Frame* frame;
if(!nclogginginitialized) ncloginit();
if(nclog_global.tracelevel < 0) ncsetlogging(0);
if(fcn != NULL) {
frame = &nclog_global.frames[nclog_global.depth];
frame->fcn = fcn;
frame->level = level;
frame->depth = nclog_global.depth;
}
if(level <= nclog_global.tracelevel) {
if(fcn != NULL)
fprintf(nclog_global.nclogstream,"%s: (%d): %s:","Enter",level,fcn);
if(fmt != NULL)
vfprintf(nclog_global.nclogstream, fmt, ap);
fprintf(nclog_global.nclogstream, "\n" );
fflush(nclog_global.nclogstream);
}
if(fcn != NULL) nclog_global.depth++;
}
int
ncuntrace(const char* fcn, int err, const char* fmt, ...)
{
va_list args;
struct Frame* frame;
va_start(args, fmt);
if(nclog_global.depth == 0) {
fprintf(nclog_global.nclogstream,"*** Unmatched untrace: %s: depth==0\n",fcn);
goto done;
}
nclog_global.depth--;
frame = &nclog_global.frames[nclog_global.depth];
if(frame->depth != nclog_global.depth || strcmp(frame->fcn,fcn) != 0) {
fprintf(nclog_global.nclogstream,"*** Unmatched untrace: fcn=%s expected=%s\n",frame->fcn,fcn);
goto done;
}
if(frame->level <= nclog_global.tracelevel) {
fprintf(nclog_global.nclogstream,"%s: (%d): %s: ","Exit",frame->level,frame->fcn);
if(err)
fprintf(nclog_global.nclogstream,"err=(%d) '%s':",err,nc_strerror(err));
if(fmt != NULL)
vfprintf(nclog_global.nclogstream, fmt, args);
fprintf(nclog_global.nclogstream, "\n" );
fflush(nclog_global.nclogstream);
#ifdef HAVE_EXECINFO_H
if(err != 0)
ncbacktrace();
#endif
}
done:
va_end(args);
if(err != 0)
return ncbreakpoint(err);
else
return err;
}
int
ncthrow(int err,const char* file,int line)
{
if(err == 0) return err;
return ncbreakpoint(err);
}
int
ncbreakpoint(int err)
{
return err;
}
#ifdef HAVE_EXECINFO_H
#define MAXSTACKDEPTH 100
void
ncbacktrace(void)
{
int j, nptrs;
void* buffer[MAXSTACKDEPTH];
char **strings;
if(getenv("NCBACKTRACE") == NULL) return;
nptrs = backtrace(buffer, MAXSTACKDEPTH);
strings = backtrace_symbols(buffer, nptrs);
if (strings == NULL) {
perror("backtrace_symbols");
errno = 0;
return;
}
fprintf(stderr,"Backtrace:\n");
for(j = 0; j < nptrs; j++)
fprintf(stderr,"%s\n", strings[j]);
free(strings);
}
#endif
/**@}*/