mirror of https://github.com/nirenjan/libx52.git
Merge commit '50dc946c3177f5006e4bd63eef4bf67331f59dea' as 'lib/pinelog'
commit
fd79166a89
|
@ -0,0 +1,38 @@
|
|||
# Compiled object files
|
||||
*.o
|
||||
|
||||
# Generated objects (source, executables, tarballs, etc.)
|
||||
|
||||
# Vim swap files
|
||||
.*.swp
|
||||
|
||||
# Autotools objects
|
||||
.deps
|
||||
.dirstamp
|
||||
.libs
|
||||
ar-lib
|
||||
autom4te.cache
|
||||
m4
|
||||
compile
|
||||
config.*
|
||||
!config.h.in
|
||||
configure
|
||||
depcomp
|
||||
install-sh
|
||||
libtool
|
||||
ltmain.sh
|
||||
missing
|
||||
Makefile
|
||||
Makefile.in
|
||||
*.la
|
||||
*.lo
|
||||
*.m4
|
||||
stamp-h1
|
||||
tap-driver.sh
|
||||
test-driver
|
||||
*.log
|
||||
*.trs
|
||||
*.pc
|
||||
|
||||
# Build directory
|
||||
/build/
|
|
@ -0,0 +1,21 @@
|
|||
MIT License
|
||||
|
||||
Copyright (c) 2021 Nirenjan Krishnan
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
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.
|
|
@ -0,0 +1,84 @@
|
|||
# Top level Automake for pinelog
|
||||
#
|
||||
# Copyright (C) 2012-2018 Nirenjan Krishnan (nirenjan@nirenjan.org)
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
|
||||
ACLOCAL_AMFLAGS = -I m4
|
||||
|
||||
# Extra files that need to be in the distribution
|
||||
EXTRA_DIST = \
|
||||
LICENSE \
|
||||
README.md \
|
||||
pinelog.h
|
||||
|
||||
noinst_LTLIBRARIES = libpinelog.la
|
||||
|
||||
# pinelog logging library
|
||||
libpinelog_la_SOURCES = pinelog.c
|
||||
libpinelog_la_CFLAGS = @PINELOG_CFLAGS@ $(WARN_CFLAGS) -I $(top_builddir)
|
||||
libpinelog_la_LDFLAGS = $(WARN_LDFLAGS)
|
||||
|
||||
test_SRCFILES = test_pinelog.c $(libpinelog_la_SOURCES)
|
||||
test_CFLAGS = \
|
||||
-DPINELOG_FATAL_STR='"F"' \
|
||||
-DPINELOG_ERROR_STR='"E"' \
|
||||
-DPINELOG_WARNING_STR='"W"' \
|
||||
-DPINELOG_INFO_STR='"I"' \
|
||||
-DPINELOG_DEBUG_STR='"D"' \
|
||||
-DPINELOG_TRACE_STR='"T"' \
|
||||
-DPINELOG_DEFAULT_LEVEL=PINELOG_LVL_TRACE \
|
||||
-DPINELOG_DEFAULT_STREAM=stderr \
|
||||
-DPINELOG_TEST -I $(top_builddir)
|
||||
|
||||
LOG_DRIVER = env AM_TAP_AWK='$(AWK)' $(SHELL) $(top_srcdir)/tap-driver.sh
|
||||
TESTS = \
|
||||
test_ts_lvl_tr \
|
||||
test_ts_lvl_notr \
|
||||
test_ts_nolvl_tr \
|
||||
test_ts_nolvl_notr \
|
||||
test_nots_lvl_tr \
|
||||
test_nots_lvl_notr \
|
||||
test_nots_nolvl_tr \
|
||||
test_nots_nolvl_notr
|
||||
|
||||
check_PROGRAMS = $(TESTS)
|
||||
test_ts_lvl_tr_SOURCES = $(test_SRCFILES)
|
||||
test_ts_lvl_tr_CFLAGS = $(WARN_CFLAGS) $(test_CFLAGS) \
|
||||
-DPINELOG_SHOW_DATE=1 -DPINELOG_SHOW_LEVEL=1 -DPINELOG_SHOW_BACKTRACE=1
|
||||
test_ts_lvl_tr_LDFLAGS = $(WARN_LDFLAGS)
|
||||
|
||||
test_ts_lvl_notr_SOURCES = $(test_SRCFILES)
|
||||
test_ts_lvl_notr_CFLAGS = $(WARN_CFLAGS) $(test_CFLAGS) \
|
||||
-DPINELOG_SHOW_DATE=1 -DPINELOG_SHOW_LEVEL=1 -DPINELOG_SHOW_BACKTRACE=0
|
||||
test_ts_lvl_notr_LDFLAGS = $(WARN_LDFLAGS)
|
||||
|
||||
test_ts_nolvl_tr_SOURCES = $(test_SRCFILES)
|
||||
test_ts_nolvl_tr_CFLAGS = $(WARN_CFLAGS) $(test_CFLAGS) \
|
||||
-DPINELOG_SHOW_DATE=1 -DPINELOG_SHOW_LEVEL=0 -DPINELOG_SHOW_BACKTRACE=1
|
||||
test_ts_nolvl_tr_LDFLAGS = $(WARN_LDFLAGS)
|
||||
|
||||
test_ts_nolvl_notr_SOURCES = $(test_SRCFILES)
|
||||
test_ts_nolvl_notr_CFLAGS = $(WARN_CFLAGS) $(test_CFLAGS) \
|
||||
-DPINELOG_SHOW_DATE=1 -DPINELOG_SHOW_LEVEL=0 -DPINELOG_SHOW_BACKTRACE=0
|
||||
test_ts_nolvl_notr_LDFLAGS = $(WARN_LDFLAGS)
|
||||
|
||||
test_nots_lvl_tr_SOURCES = $(test_SRCFILES)
|
||||
test_nots_lvl_tr_CFLAGS = $(WARN_CFLAGS) $(test_CFLAGS) \
|
||||
-DPINELOG_SHOW_DATE=0 -DPINELOG_SHOW_LEVEL=1 -DPINELOG_SHOW_BACKTRACE=1
|
||||
test_nots_lvl_tr_LDFLAGS = $(WARN_LDFLAGS)
|
||||
|
||||
test_nots_lvl_notr_SOURCES = $(test_SRCFILES)
|
||||
test_nots_lvl_notr_CFLAGS = $(WARN_CFLAGS) $(test_CFLAGS) \
|
||||
-DPINELOG_SHOW_DATE=0 -DPINELOG_SHOW_LEVEL=1 -DPINELOG_SHOW_BACKTRACE=0
|
||||
test_nots_lvl_notr_LDFLAGS = $(WARN_LDFLAGS)
|
||||
|
||||
test_nots_nolvl_tr_SOURCES = $(test_SRCFILES)
|
||||
test_nots_nolvl_tr_CFLAGS = $(WARN_CFLAGS) $(test_CFLAGS) \
|
||||
-DPINELOG_SHOW_DATE=0 -DPINELOG_SHOW_LEVEL=0 -DPINELOG_SHOW_BACKTRACE=1
|
||||
test_nots_nolvl_tr_LDFLAGS = $(WARN_LDFLAGS)
|
||||
|
||||
test_nots_nolvl_notr_SOURCES = $(test_SRCFILES)
|
||||
test_nots_nolvl_notr_CFLAGS = $(WARN_CFLAGS) $(test_CFLAGS) \
|
||||
-DPINELOG_SHOW_DATE=0 -DPINELOG_SHOW_LEVEL=0 -DPINELOG_SHOW_BACKTRACE=0
|
||||
test_nots_nolvl_notr_LDFLAGS = $(WARN_LDFLAGS)
|
|
@ -0,0 +1,134 @@
|
|||
Pinelog - a lightweight logging API
|
||||
===================================
|
||||
|
||||
Pinelog is a lightweight logging API for C programs that's designed to be
|
||||
included in your program source code. Parameters for Pinelog are configured at
|
||||
build time by means of preprocessor flags.
|
||||
|
||||
# Usage
|
||||
## Logging macros
|
||||
|
||||
Pinelog uses `printf` style formatting, using the following list of macros. The
|
||||
macro indicates the level at which the message is logged.
|
||||
|
||||
* `PINELOG_FATAL`
|
||||
* `PINELOG_ERROR`
|
||||
* `PINELOG_WARN`
|
||||
* `PINELOG_INFO`
|
||||
* `PINELOG_DEBUG`
|
||||
* `PINELOG_TRACE`
|
||||
|
||||
**Note:** `PINELOG_FATAL` is used when the program encounters a fatal condition
|
||||
and needs to abort. This will log the fatal message and terminate the program
|
||||
with an exit code of 1.
|
||||
|
||||
### Example
|
||||
|
||||
```C
|
||||
PINELOG_INFO("configuration file %s not found, using defaults", config_file);
|
||||
```
|
||||
|
||||
## Logging levels
|
||||
|
||||
The default logging level is `ERROR`, and this can be controlled by the
|
||||
preprocessor flag `PINELOG_DEFAULT_LEVEL`.
|
||||
|
||||
The program can control the level at which messages can be logged at runtime,
|
||||
by using the `pinelog_set_level` function. This function takes in the level
|
||||
definition, which is one of the following, in increasing order of priority.
|
||||
|
||||
* `PINELOG_LVL_TRACE`
|
||||
* `PINELOG_LVL_DEBUG`
|
||||
* `PINELOG_LVL_INFO`
|
||||
* `PINELOG_LVL_WARNING`
|
||||
* `PINELOG_LVL_ERROR`
|
||||
* `PINELOG_LVL_FATAL`
|
||||
* `PINELOG_LVL_NONE`
|
||||
|
||||
Setting the level to a given priority suppresses all log messages of lower
|
||||
priority, i.e., if the level is set to `PINELOG_LVL_ERROR`, messages at
|
||||
`WARNING` level and below will be suppressed, but `ERROR` and `FATAL` messages
|
||||
will be logged.
|
||||
|
||||
**Note:** `PINELOG_LVL_NONE` suppresses all log messages, but `PINELOG_FATAL`
|
||||
will still terminate the program, even though nothing is logged.
|
||||
|
||||
### Example
|
||||
|
||||
```C
|
||||
pinelog_set_level(PINELOG_LVL_WARNING);
|
||||
```
|
||||
|
||||
```
|
||||
-DPINELOG_DEFAULT_LEVEL=PINELOG_LVL_WARNING
|
||||
```
|
||||
|
||||
## Output redirection
|
||||
|
||||
Pinelog defaults to writing the log messages to standard output, and this can
|
||||
be controlled by the preprocessor flag `PINELOG_DEFAULT_STREAM`.
|
||||
|
||||
However, the application can redirect log messages at runtime to a different
|
||||
`FILE *` stream, or to a file by using one of the following two methods:
|
||||
|
||||
```C
|
||||
FILE *out = fopen("/run/app.fifo", "w");
|
||||
pinelog_set_output_stream(out);
|
||||
pinelog_set_output_file("/var/log/app.log");
|
||||
```
|
||||
|
||||
```
|
||||
-DPINELOG_DEFAULT_STREAM=stderr
|
||||
```
|
||||
|
||||
## Logging format
|
||||
|
||||
Pinelog uses an opinionated logging format that is fixed as follows. Fields
|
||||
within `[]` are optional and controlled by build time flags.
|
||||
|
||||
[2021-07-14 11:08:04 ][ERROR: ][./test_pinelog.c:108 ]formatted message.
|
||||
|
||||
The program can be controlled by the following preprocessor flags, all of which
|
||||
default to `0` (disabled). Set the flag to `1` to enable it.
|
||||
|
||||
* `PINELOG_SHOW_DATE` - Display the ISO 8601 date and time when the message is
|
||||
logged.
|
||||
* `PINELOG_SHOW_LEVEL` - Display the level at which the message is logged.
|
||||
* `PINELOG_SHOW_BACKTRACE` - Display the file and line where the message is
|
||||
logged.
|
||||
|
||||
Set these flags by using the `-D` compiler argument, .e.g.
|
||||
`-DPINELOG_SHOW_LEVEL=1 -DPINELOG_SHOW_DATE=1`
|
||||
|
||||
### Level strings
|
||||
|
||||
The application can control the level strings displayed by means of preprocessor
|
||||
flags, if the application wishes to display the log messages in a language other
|
||||
than English. This can be achieved by means of the following preprocessor
|
||||
definitions.
|
||||
|
||||
* `PINELOG_FATAL_STR`
|
||||
* `PINELOG_ERROR_STR`
|
||||
* `PINELOG_WARNING_STR`
|
||||
* `PINELOG_INFO_STR`
|
||||
* `PINELOG_DEBUG_STR`
|
||||
* `PINELOG_TRACE_STR`
|
||||
|
||||
### Example
|
||||
|
||||
```
|
||||
-DPINELOG_ERROR_STR=\"E\" -DPINELOG_FATAL_STR=\"F\"
|
||||
```
|
||||
|
||||
# Integrating Pinelog
|
||||
|
||||
Pinelog is intended to be integrated into your application source tree, either
|
||||
by means of including the sources directly, or by including the repository as
|
||||
a Git submodule or subtree.
|
||||
|
||||
The default build of Pinelog uses an autotools generated `config.h` file, which
|
||||
includes checks for the following GCC attributes. If you don't care about these,
|
||||
then either create a dummy config.h which includes the macros
|
||||
`HAVE_FUNC_ATTRIBUTE_CONSTRUCTOR` and `HAVE_FUNC_ATTRIBUTE_FORMAT`, or use the
|
||||
`AX_GCC_FUNC_ATTRIBUTE` macro to check for the `constructor` and `format`
|
||||
attributes in your application's `configure.ac` file.
|
|
@ -0,0 +1,3 @@
|
|||
#!/bin/sh -x
|
||||
|
||||
autoreconf --install
|
|
@ -0,0 +1,67 @@
|
|||
/* config.h.in. Generated from configure.ac by autoheader. */
|
||||
|
||||
/* Define to 1 if you have the <dlfcn.h> header file. */
|
||||
#undef HAVE_DLFCN_H
|
||||
|
||||
/* Define to 1 if the system has the `constructor' function attribute */
|
||||
#undef HAVE_FUNC_ATTRIBUTE_CONSTRUCTOR
|
||||
|
||||
/* Define to 1 if the system has the `format' function attribute */
|
||||
#undef HAVE_FUNC_ATTRIBUTE_FORMAT
|
||||
|
||||
/* Define to 1 if you have the <inttypes.h> header file. */
|
||||
#undef HAVE_INTTYPES_H
|
||||
|
||||
/* Define to 1 if you have the <memory.h> header file. */
|
||||
#undef HAVE_MEMORY_H
|
||||
|
||||
/* Define to 1 if you have the <stdint.h> header file. */
|
||||
#undef HAVE_STDINT_H
|
||||
|
||||
/* Define to 1 if you have the <stdlib.h> header file. */
|
||||
#undef HAVE_STDLIB_H
|
||||
|
||||
/* Define to 1 if you have the <strings.h> header file. */
|
||||
#undef HAVE_STRINGS_H
|
||||
|
||||
/* Define to 1 if you have the <string.h> header file. */
|
||||
#undef HAVE_STRING_H
|
||||
|
||||
/* Define to 1 if you have the <sys/stat.h> header file. */
|
||||
#undef HAVE_SYS_STAT_H
|
||||
|
||||
/* Define to 1 if you have the <sys/types.h> header file. */
|
||||
#undef HAVE_SYS_TYPES_H
|
||||
|
||||
/* Define to 1 if you have the <unistd.h> header file. */
|
||||
#undef HAVE_UNISTD_H
|
||||
|
||||
/* Define to the sub-directory where libtool stores uninstalled libraries. */
|
||||
#undef LT_OBJDIR
|
||||
|
||||
/* Name of package */
|
||||
#undef PACKAGE
|
||||
|
||||
/* Define to the address where bug reports for this package should be sent. */
|
||||
#undef PACKAGE_BUGREPORT
|
||||
|
||||
/* Define to the full name of this package. */
|
||||
#undef PACKAGE_NAME
|
||||
|
||||
/* Define to the full name and version of this package. */
|
||||
#undef PACKAGE_STRING
|
||||
|
||||
/* Define to the one symbol short name of this package. */
|
||||
#undef PACKAGE_TARNAME
|
||||
|
||||
/* Define to the home page for this package. */
|
||||
#undef PACKAGE_URL
|
||||
|
||||
/* Define to the version of this package. */
|
||||
#undef PACKAGE_VERSION
|
||||
|
||||
/* Define to 1 if you have the ANSI C header files. */
|
||||
#undef STDC_HEADERS
|
||||
|
||||
/* Version number of package */
|
||||
#undef VERSION
|
|
@ -0,0 +1,28 @@
|
|||
# Autoconf settings for pinelog
|
||||
#
|
||||
# Copyright (C) 2021 Nirenjan Krishnan (nirenjan@nirenjan.org)
|
||||
#
|
||||
# SPDX-License-Identifier: MIT
|
||||
|
||||
AC_INIT([pinelog], [1.0.0], [nirenjan@nirenjan.org])
|
||||
AC_CONFIG_MACRO_DIR([m4])
|
||||
AM_INIT_AUTOMAKE([-Wall foreign subdir-objects])
|
||||
AC_REQUIRE_AUX_FILE([tap-driver.sh])
|
||||
AC_PROG_CC
|
||||
AC_PROG_CC_STDC
|
||||
AC_PROG_AWK
|
||||
AM_PROG_AR
|
||||
LT_INIT
|
||||
PKG_PROG_PKG_CONFIG
|
||||
PKG_INSTALLDIR
|
||||
AX_COMPILER_FLAGS
|
||||
AX_GCC_FUNC_ATTRIBUTE([constructor])
|
||||
AX_GCC_FUNC_ATTRIBUTE([format])
|
||||
|
||||
AC_SUBST([PINELOG_CFLAGS])
|
||||
|
||||
# Configuration headers
|
||||
AC_CONFIG_HEADERS([config.h])
|
||||
|
||||
AC_CONFIG_FILES([Makefile])
|
||||
AC_OUTPUT
|
|
@ -0,0 +1,46 @@
|
|||
# ============================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_append_compile_flags.html
|
||||
# ============================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_APPEND_COMPILE_FLAGS([FLAG1 FLAG2 ...], [FLAGS-VARIABLE], [EXTRA-FLAGS], [INPUT])
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# For every FLAG1, FLAG2 it is checked whether the compiler works with the
|
||||
# flag. If it does, the flag is added FLAGS-VARIABLE
|
||||
#
|
||||
# If FLAGS-VARIABLE is not specified, the current language's flags (e.g.
|
||||
# CFLAGS) is used. During the check the flag is always added to the
|
||||
# current language's flags.
|
||||
#
|
||||
# If EXTRA-FLAGS is defined, it is added to the current language's default
|
||||
# flags (e.g. CFLAGS) when the check is done. The check is thus made with
|
||||
# the flags: "CFLAGS EXTRA-FLAGS FLAG". This can for example be used to
|
||||
# force the compiler to issue an error when a bad flag is given.
|
||||
#
|
||||
# INPUT gives an alternative input source to AC_COMPILE_IFELSE.
|
||||
#
|
||||
# NOTE: This macro depends on the AX_APPEND_FLAG and
|
||||
# AX_CHECK_COMPILE_FLAG. Please keep this macro in sync with
|
||||
# AX_APPEND_LINK_FLAGS.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2011 Maarten Bosmans <mkbosmans@gmail.com>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 7
|
||||
|
||||
AC_DEFUN([AX_APPEND_COMPILE_FLAGS],
|
||||
[AX_REQUIRE_DEFINED([AX_CHECK_COMPILE_FLAG])
|
||||
AX_REQUIRE_DEFINED([AX_APPEND_FLAG])
|
||||
for flag in $1; do
|
||||
AX_CHECK_COMPILE_FLAG([$flag], [AX_APPEND_FLAG([$flag], [$2])], [], [$3], [$4])
|
||||
done
|
||||
])dnl AX_APPEND_COMPILE_FLAGS
|
|
@ -0,0 +1,50 @@
|
|||
# ===========================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_append_flag.html
|
||||
# ===========================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_APPEND_FLAG(FLAG, [FLAGS-VARIABLE])
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# FLAG is appended to the FLAGS-VARIABLE shell variable, with a space
|
||||
# added in between.
|
||||
#
|
||||
# If FLAGS-VARIABLE is not specified, the current language's flags (e.g.
|
||||
# CFLAGS) is used. FLAGS-VARIABLE is not changed if it already contains
|
||||
# FLAG. If FLAGS-VARIABLE is unset in the shell, it is set to exactly
|
||||
# FLAG.
|
||||
#
|
||||
# NOTE: Implementation based on AX_CFLAGS_GCC_OPTION.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2008 Guido U. Draheim <guidod@gmx.de>
|
||||
# Copyright (c) 2011 Maarten Bosmans <mkbosmans@gmail.com>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 8
|
||||
|
||||
AC_DEFUN([AX_APPEND_FLAG],
|
||||
[dnl
|
||||
AC_PREREQ(2.64)dnl for _AC_LANG_PREFIX and AS_VAR_SET_IF
|
||||
AS_VAR_PUSHDEF([FLAGS], [m4_default($2,_AC_LANG_PREFIX[FLAGS])])
|
||||
AS_VAR_SET_IF(FLAGS,[
|
||||
AS_CASE([" AS_VAR_GET(FLAGS) "],
|
||||
[*" $1 "*], [AC_RUN_LOG([: FLAGS already contains $1])],
|
||||
[
|
||||
AS_VAR_APPEND(FLAGS,[" $1"])
|
||||
AC_RUN_LOG([: FLAGS="$FLAGS"])
|
||||
])
|
||||
],
|
||||
[
|
||||
AS_VAR_SET(FLAGS,[$1])
|
||||
AC_RUN_LOG([: FLAGS="$FLAGS"])
|
||||
])
|
||||
AS_VAR_POPDEF([FLAGS])dnl
|
||||
])dnl AX_APPEND_FLAG
|
|
@ -0,0 +1,44 @@
|
|||
# ===========================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_append_link_flags.html
|
||||
# ===========================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_APPEND_LINK_FLAGS([FLAG1 FLAG2 ...], [FLAGS-VARIABLE], [EXTRA-FLAGS], [INPUT])
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# For every FLAG1, FLAG2 it is checked whether the linker works with the
|
||||
# flag. If it does, the flag is added FLAGS-VARIABLE
|
||||
#
|
||||
# If FLAGS-VARIABLE is not specified, the linker's flags (LDFLAGS) is
|
||||
# used. During the check the flag is always added to the linker's flags.
|
||||
#
|
||||
# If EXTRA-FLAGS is defined, it is added to the linker's default flags
|
||||
# when the check is done. The check is thus made with the flags: "LDFLAGS
|
||||
# EXTRA-FLAGS FLAG". This can for example be used to force the linker to
|
||||
# issue an error when a bad flag is given.
|
||||
#
|
||||
# INPUT gives an alternative input source to AC_COMPILE_IFELSE.
|
||||
#
|
||||
# NOTE: This macro depends on the AX_APPEND_FLAG and AX_CHECK_LINK_FLAG.
|
||||
# Please keep this macro in sync with AX_APPEND_COMPILE_FLAGS.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2011 Maarten Bosmans <mkbosmans@gmail.com>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 7
|
||||
|
||||
AC_DEFUN([AX_APPEND_LINK_FLAGS],
|
||||
[AX_REQUIRE_DEFINED([AX_CHECK_LINK_FLAG])
|
||||
AX_REQUIRE_DEFINED([AX_APPEND_FLAG])
|
||||
for flag in $1; do
|
||||
AX_CHECK_LINK_FLAG([$flag], [AX_APPEND_FLAG([$flag], [m4_default([$2], [LDFLAGS])])], [], [$3], [$4])
|
||||
done
|
||||
])dnl AX_APPEND_LINK_FLAGS
|
|
@ -0,0 +1,53 @@
|
|||
# ===========================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_check_compile_flag.html
|
||||
# ===========================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_CHECK_COMPILE_FLAG(FLAG, [ACTION-SUCCESS], [ACTION-FAILURE], [EXTRA-FLAGS], [INPUT])
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# Check whether the given FLAG works with the current language's compiler
|
||||
# or gives an error. (Warnings, however, are ignored)
|
||||
#
|
||||
# ACTION-SUCCESS/ACTION-FAILURE are shell commands to execute on
|
||||
# success/failure.
|
||||
#
|
||||
# If EXTRA-FLAGS is defined, it is added to the current language's default
|
||||
# flags (e.g. CFLAGS) when the check is done. The check is thus made with
|
||||
# the flags: "CFLAGS EXTRA-FLAGS FLAG". This can for example be used to
|
||||
# force the compiler to issue an error when a bad flag is given.
|
||||
#
|
||||
# INPUT gives an alternative input source to AC_COMPILE_IFELSE.
|
||||
#
|
||||
# NOTE: Implementation based on AX_CFLAGS_GCC_OPTION. Please keep this
|
||||
# macro in sync with AX_CHECK_{PREPROC,LINK}_FLAG.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2008 Guido U. Draheim <guidod@gmx.de>
|
||||
# Copyright (c) 2011 Maarten Bosmans <mkbosmans@gmail.com>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 6
|
||||
|
||||
AC_DEFUN([AX_CHECK_COMPILE_FLAG],
|
||||
[AC_PREREQ(2.64)dnl for _AC_LANG_PREFIX and AS_VAR_IF
|
||||
AS_VAR_PUSHDEF([CACHEVAR],[ax_cv_check_[]_AC_LANG_ABBREV[]flags_$4_$1])dnl
|
||||
AC_CACHE_CHECK([whether _AC_LANG compiler accepts $1], CACHEVAR, [
|
||||
ax_check_save_flags=$[]_AC_LANG_PREFIX[]FLAGS
|
||||
_AC_LANG_PREFIX[]FLAGS="$[]_AC_LANG_PREFIX[]FLAGS $4 $1"
|
||||
AC_COMPILE_IFELSE([m4_default([$5],[AC_LANG_PROGRAM()])],
|
||||
[AS_VAR_SET(CACHEVAR,[yes])],
|
||||
[AS_VAR_SET(CACHEVAR,[no])])
|
||||
_AC_LANG_PREFIX[]FLAGS=$ax_check_save_flags])
|
||||
AS_VAR_IF(CACHEVAR,yes,
|
||||
[m4_default([$2], :)],
|
||||
[m4_default([$3], :)])
|
||||
AS_VAR_POPDEF([CACHEVAR])dnl
|
||||
])dnl AX_CHECK_COMPILE_FLAGS
|
|
@ -0,0 +1,53 @@
|
|||
# ===========================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_check_link_flag.html
|
||||
# ===========================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_CHECK_LINK_FLAG(FLAG, [ACTION-SUCCESS], [ACTION-FAILURE], [EXTRA-FLAGS], [INPUT])
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# Check whether the given FLAG works with the linker or gives an error.
|
||||
# (Warnings, however, are ignored)
|
||||
#
|
||||
# ACTION-SUCCESS/ACTION-FAILURE are shell commands to execute on
|
||||
# success/failure.
|
||||
#
|
||||
# If EXTRA-FLAGS is defined, it is added to the linker's default flags
|
||||
# when the check is done. The check is thus made with the flags: "LDFLAGS
|
||||
# EXTRA-FLAGS FLAG". This can for example be used to force the linker to
|
||||
# issue an error when a bad flag is given.
|
||||
#
|
||||
# INPUT gives an alternative input source to AC_LINK_IFELSE.
|
||||
#
|
||||
# NOTE: Implementation based on AX_CFLAGS_GCC_OPTION. Please keep this
|
||||
# macro in sync with AX_CHECK_{PREPROC,COMPILE}_FLAG.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2008 Guido U. Draheim <guidod@gmx.de>
|
||||
# Copyright (c) 2011 Maarten Bosmans <mkbosmans@gmail.com>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 6
|
||||
|
||||
AC_DEFUN([AX_CHECK_LINK_FLAG],
|
||||
[AC_PREREQ(2.64)dnl for _AC_LANG_PREFIX and AS_VAR_IF
|
||||
AS_VAR_PUSHDEF([CACHEVAR],[ax_cv_check_ldflags_$4_$1])dnl
|
||||
AC_CACHE_CHECK([whether the linker accepts $1], CACHEVAR, [
|
||||
ax_check_save_flags=$LDFLAGS
|
||||
LDFLAGS="$LDFLAGS $4 $1"
|
||||
AC_LINK_IFELSE([m4_default([$5],[AC_LANG_PROGRAM()])],
|
||||
[AS_VAR_SET(CACHEVAR,[yes])],
|
||||
[AS_VAR_SET(CACHEVAR,[no])])
|
||||
LDFLAGS=$ax_check_save_flags])
|
||||
AS_VAR_IF(CACHEVAR,yes,
|
||||
[m4_default([$2], :)],
|
||||
[m4_default([$3], :)])
|
||||
AS_VAR_POPDEF([CACHEVAR])dnl
|
||||
])dnl AX_CHECK_LINK_FLAGS
|
|
@ -0,0 +1,158 @@
|
|||
# ===========================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_compiler_flags.html
|
||||
# ===========================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_COMPILER_FLAGS([CFLAGS-VARIABLE], [LDFLAGS-VARIABLE], [IS-RELEASE], [EXTRA-BASE-CFLAGS], [EXTRA-YES-CFLAGS], [UNUSED], [UNUSED], [UNUSED], [EXTRA-BASE-LDFLAGS], [EXTRA-YES-LDFLAGS], [UNUSED], [UNUSED], [UNUSED])
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# Check for the presence of an --enable-compile-warnings option to
|
||||
# configure, defaulting to "error" in normal operation, or "yes" if
|
||||
# IS-RELEASE is equal to "yes". Return the value in the variable
|
||||
# $ax_enable_compile_warnings.
|
||||
#
|
||||
# Depending on the value of --enable-compile-warnings, different compiler
|
||||
# warnings are checked to see if they work with the current compiler and,
|
||||
# if so, are appended to CFLAGS-VARIABLE and LDFLAGS-VARIABLE. This
|
||||
# allows a consistent set of baseline compiler warnings to be used across
|
||||
# a code base, irrespective of any warnings enabled locally by individual
|
||||
# developers. By standardising the warnings used by all developers of a
|
||||
# project, the project can commit to a zero-warnings policy, using -Werror
|
||||
# to prevent compilation if new warnings are introduced. This makes
|
||||
# catching bugs which are flagged by warnings a lot easier.
|
||||
#
|
||||
# By providing a consistent --enable-compile-warnings argument across all
|
||||
# projects using this macro, continuous integration systems can easily be
|
||||
# configured the same for all projects. Automated systems or build
|
||||
# systems aimed at beginners may want to pass the --disable-Werror
|
||||
# argument to unconditionally prevent warnings being fatal.
|
||||
#
|
||||
# --enable-compile-warnings can take the values:
|
||||
#
|
||||
# * no: Base compiler warnings only; not even -Wall.
|
||||
# * yes: The above, plus a broad range of useful warnings.
|
||||
# * error: The above, plus -Werror so that all warnings are fatal.
|
||||
# Use --disable-Werror to override this and disable fatal
|
||||
# warnings.
|
||||
#
|
||||
# The set of base and enabled flags can be augmented using the
|
||||
# EXTRA-*-CFLAGS and EXTRA-*-LDFLAGS variables, which are tested and
|
||||
# appended to the output variable if --enable-compile-warnings is not
|
||||
# "no". Flags should not be disabled using these arguments, as the entire
|
||||
# point of AX_COMPILER_FLAGS is to enforce a consistent set of useful
|
||||
# compiler warnings on code, using warnings which have been chosen for low
|
||||
# false positive rates. If a compiler emits false positives for a
|
||||
# warning, a #pragma should be used in the code to disable the warning
|
||||
# locally. See:
|
||||
#
|
||||
# https://gcc.gnu.org/onlinedocs/gcc-4.9.2/gcc/Diagnostic-Pragmas.html#Diagnostic-Pragmas
|
||||
#
|
||||
# The EXTRA-* variables should only be used to supply extra warning flags,
|
||||
# and not general purpose compiler flags, as they are controlled by
|
||||
# configure options such as --disable-Werror.
|
||||
#
|
||||
# IS-RELEASE can be used to disable -Werror when making a release, which
|
||||
# is useful for those hairy moments when you just want to get the release
|
||||
# done as quickly as possible. Set it to "yes" to disable -Werror. By
|
||||
# default, it uses the value of $ax_is_release, so if you are using the
|
||||
# AX_IS_RELEASE macro, there is no need to pass this parameter. For
|
||||
# example:
|
||||
#
|
||||
# AX_IS_RELEASE([git-directory])
|
||||
# AX_COMPILER_FLAGS()
|
||||
#
|
||||
# CFLAGS-VARIABLE defaults to WARN_CFLAGS, and LDFLAGS-VARIABLE defaults
|
||||
# to WARN_LDFLAGS. Both variables are AC_SUBST-ed by this macro, but must
|
||||
# be manually added to the CFLAGS and LDFLAGS variables for each target in
|
||||
# the code base.
|
||||
#
|
||||
# If C++ language support is enabled with AC_PROG_CXX, which must occur
|
||||
# before this macro in configure.ac, warning flags for the C++ compiler
|
||||
# are AC_SUBST-ed as WARN_CXXFLAGS, and must be manually added to the
|
||||
# CXXFLAGS variables for each target in the code base. EXTRA-*-CFLAGS can
|
||||
# be used to augment the base and enabled flags.
|
||||
#
|
||||
# Warning flags for g-ir-scanner (from GObject Introspection) are
|
||||
# AC_SUBST-ed as WARN_SCANNERFLAGS. This variable must be manually added
|
||||
# to the SCANNERFLAGS variable for each GIR target in the code base. If
|
||||
# extra g-ir-scanner flags need to be enabled, the AX_COMPILER_FLAGS_GIR
|
||||
# macro must be invoked manually.
|
||||
#
|
||||
# AX_COMPILER_FLAGS may add support for other tools in future, in addition
|
||||
# to the compiler and linker. No extra EXTRA-* variables will be added
|
||||
# for those tools, and all extra support will still use the single
|
||||
# --enable-compile-warnings configure option. For finer grained control
|
||||
# over the flags for individual tools, use AX_COMPILER_FLAGS_CFLAGS,
|
||||
# AX_COMPILER_FLAGS_LDFLAGS and AX_COMPILER_FLAGS_* for new tools.
|
||||
#
|
||||
# The UNUSED variables date from a previous version of this macro, and are
|
||||
# automatically appended to the preceding non-UNUSED variable. They should
|
||||
# be left empty in new uses of the macro.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2014, 2015 Philip Withnall <philip@tecnocode.co.uk>
|
||||
# Copyright (c) 2015 David King <amigadave@amigadave.com>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 14
|
||||
|
||||
# _AX_COMPILER_FLAGS_LANG([LANGNAME])
|
||||
m4_defun([_AX_COMPILER_FLAGS_LANG],
|
||||
[m4_ifdef([_AX_COMPILER_FLAGS_LANG_]$1[_enabled], [],
|
||||
[m4_define([_AX_COMPILER_FLAGS_LANG_]$1[_enabled], [])dnl
|
||||
AX_REQUIRE_DEFINED([AX_COMPILER_FLAGS_]$1[FLAGS])])dnl
|
||||
])
|
||||
|
||||
AC_DEFUN([AX_COMPILER_FLAGS],[
|
||||
# C support is enabled by default.
|
||||
_AX_COMPILER_FLAGS_LANG([C])
|
||||
# Only enable C++ support if AC_PROG_CXX is called. The redefinition of
|
||||
# AC_PROG_CXX is so that a fatal error is emitted if this macro is called
|
||||
# before AC_PROG_CXX, which would otherwise cause no C++ warnings to be
|
||||
# checked.
|
||||
AC_PROVIDE_IFELSE([AC_PROG_CXX],
|
||||
[_AX_COMPILER_FLAGS_LANG([CXX])],
|
||||
[m4_define([AC_PROG_CXX], defn([AC_PROG_CXX])[_AX_COMPILER_FLAGS_LANG([CXX])])])
|
||||
AX_REQUIRE_DEFINED([AX_COMPILER_FLAGS_LDFLAGS])
|
||||
|
||||
# Default value for IS-RELEASE is $ax_is_release
|
||||
ax_compiler_flags_is_release=m4_tolower(m4_normalize(ifelse([$3],,
|
||||
[$ax_is_release],
|
||||
[$3])))
|
||||
|
||||
AC_ARG_ENABLE([compile-warnings],
|
||||
AS_HELP_STRING([--enable-compile-warnings=@<:@no/yes/error@:>@],
|
||||
[Enable compiler warnings and errors]),,
|
||||
[AS_IF([test "$ax_compiler_flags_is_release" = "yes"],
|
||||
[enable_compile_warnings="yes"],
|
||||
[enable_compile_warnings="error"])])
|
||||
AC_ARG_ENABLE([Werror],
|
||||
AS_HELP_STRING([--disable-Werror],
|
||||
[Unconditionally make all compiler warnings non-fatal]),,
|
||||
[enable_Werror=maybe])
|
||||
|
||||
# Return the user's chosen warning level
|
||||
AS_IF([test "$enable_Werror" = "no" -a \
|
||||
"$enable_compile_warnings" = "error"],[
|
||||
enable_compile_warnings="yes"
|
||||
])
|
||||
|
||||
ax_enable_compile_warnings=$enable_compile_warnings
|
||||
|
||||
AX_COMPILER_FLAGS_CFLAGS([$1],[$ax_compiler_flags_is_release],
|
||||
[$4],[$5 $6 $7 $8])
|
||||
m4_ifdef([_AX_COMPILER_FLAGS_LANG_CXX_enabled],
|
||||
[AX_COMPILER_FLAGS_CXXFLAGS([WARN_CXXFLAGS],
|
||||
[$ax_compiler_flags_is_release],
|
||||
[$4],[$5 $6 $7 $8])])
|
||||
AX_COMPILER_FLAGS_LDFLAGS([$2],[$ax_compiler_flags_is_release],
|
||||
[$9],[$10 $11 $12 $13])
|
||||
AX_COMPILER_FLAGS_GIR([WARN_SCANNERFLAGS],[$ax_compiler_flags_is_release])
|
||||
])dnl AX_COMPILER_FLAGS
|
|
@ -0,0 +1,161 @@
|
|||
# =============================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_compiler_flags_cflags.html
|
||||
# =============================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_COMPILER_FLAGS_CFLAGS([VARIABLE], [IS-RELEASE], [EXTRA-BASE-FLAGS], [EXTRA-YES-FLAGS])
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# Add warning flags for the C compiler to VARIABLE, which defaults to
|
||||
# WARN_CFLAGS. VARIABLE is AC_SUBST-ed by this macro, but must be
|
||||
# manually added to the CFLAGS variable for each target in the code base.
|
||||
#
|
||||
# This macro depends on the environment set up by AX_COMPILER_FLAGS.
|
||||
# Specifically, it uses the value of $ax_enable_compile_warnings to decide
|
||||
# which flags to enable.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2014, 2015 Philip Withnall <philip@tecnocode.co.uk>
|
||||
# Copyright (c) 2017, 2018 Reini Urban <rurban@cpan.org>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 17
|
||||
|
||||
AC_DEFUN([AX_COMPILER_FLAGS_CFLAGS],[
|
||||
AC_REQUIRE([AC_PROG_SED])
|
||||
AX_REQUIRE_DEFINED([AX_APPEND_COMPILE_FLAGS])
|
||||
AX_REQUIRE_DEFINED([AX_APPEND_FLAG])
|
||||
AX_REQUIRE_DEFINED([AX_CHECK_COMPILE_FLAG])
|
||||
|
||||
# Variable names
|
||||
m4_define([ax_warn_cflags_variable],
|
||||
[m4_normalize(ifelse([$1],,[WARN_CFLAGS],[$1]))])
|
||||
|
||||
AC_LANG_PUSH([C])
|
||||
|
||||
AC_COMPILE_IFELSE([AC_LANG_PROGRAM([
|
||||
[#ifndef __cplusplus
|
||||
#error "no C++"
|
||||
#endif]])],
|
||||
[ax_compiler_cxx=yes;],
|
||||
[ax_compiler_cxx=no;])
|
||||
|
||||
# Always pass -Werror=unknown-warning-option to get Clang to fail on bad
|
||||
# flags, otherwise they are always appended to the warn_cflags variable, and
|
||||
# Clang warns on them for every compilation unit.
|
||||
# If this is passed to GCC, it will explode, so the flag must be enabled
|
||||
# conditionally.
|
||||
AX_CHECK_COMPILE_FLAG([-Werror=unknown-warning-option],[
|
||||
ax_compiler_flags_test="-Werror=unknown-warning-option"
|
||||
],[
|
||||
ax_compiler_flags_test=""
|
||||
])
|
||||
|
||||
# Check that -Wno-suggest-attribute=format is supported
|
||||
AX_CHECK_COMPILE_FLAG([-Wno-suggest-attribute=format],[
|
||||
ax_compiler_no_suggest_attribute_flags="-Wno-suggest-attribute=format"
|
||||
],[
|
||||
ax_compiler_no_suggest_attribute_flags=""
|
||||
])
|
||||
|
||||
# Base flags
|
||||
AX_APPEND_COMPILE_FLAGS([ dnl
|
||||
-fno-strict-aliasing dnl
|
||||
$3 dnl
|
||||
],ax_warn_cflags_variable,[$ax_compiler_flags_test])
|
||||
|
||||
AS_IF([test "$ax_enable_compile_warnings" != "no"],[
|
||||
if test "$ax_compiler_cxx" = "no" ; then
|
||||
# C-only flags. Warn in C++
|
||||
AX_APPEND_COMPILE_FLAGS([ dnl
|
||||
-Wnested-externs dnl
|
||||
-Wmissing-prototypes dnl
|
||||
-Wstrict-prototypes dnl
|
||||
-Wdeclaration-after-statement dnl
|
||||
-Wimplicit-function-declaration dnl
|
||||
-Wold-style-definition dnl
|
||||
-Wjump-misses-init dnl
|
||||
],ax_warn_cflags_variable,[$ax_compiler_flags_test])
|
||||
fi
|
||||
|
||||
# "yes" flags
|
||||
AX_APPEND_COMPILE_FLAGS([ dnl
|
||||
-Wall dnl
|
||||
-Wextra dnl
|
||||
-Wundef dnl
|
||||
-Wwrite-strings dnl
|
||||
-Wpointer-arith dnl
|
||||
-Wmissing-declarations dnl
|
||||
-Wredundant-decls dnl
|
||||
-Wno-unused-parameter dnl
|
||||
-Wno-missing-field-initializers dnl
|
||||
-Wformat=2 dnl
|
||||
-Wcast-align dnl
|
||||
-Wformat-nonliteral dnl
|
||||
-Wformat-security dnl
|
||||
-Wsign-compare dnl
|
||||
-Wstrict-aliasing dnl
|
||||
-Wshadow dnl
|
||||
-Winline dnl
|
||||
-Wpacked dnl
|
||||
-Wmissing-format-attribute dnl
|
||||
-Wmissing-noreturn dnl
|
||||
-Winit-self dnl
|
||||
-Wredundant-decls dnl
|
||||
-Wmissing-include-dirs dnl
|
||||
-Wunused-but-set-variable dnl
|
||||
-Warray-bounds dnl
|
||||
-Wreturn-type dnl
|
||||
-Wswitch-enum dnl
|
||||
-Wswitch-default dnl
|
||||
-Wduplicated-cond dnl
|
||||
-Wduplicated-branches dnl
|
||||
-Wlogical-op dnl
|
||||
-Wrestrict dnl
|
||||
-Wnull-dereference dnl
|
||||
-Wdouble-promotion dnl
|
||||
$4 dnl
|
||||
$5 dnl
|
||||
$6 dnl
|
||||
$7 dnl
|
||||
],ax_warn_cflags_variable,[$ax_compiler_flags_test])
|
||||
])
|
||||
AS_IF([test "$ax_enable_compile_warnings" = "error"],[
|
||||
# "error" flags; -Werror has to be appended unconditionally because
|
||||
# it's not possible to test for
|
||||
#
|
||||
# suggest-attribute=format is disabled because it gives too many false
|
||||
# positives
|
||||
AX_APPEND_FLAG([-Werror],ax_warn_cflags_variable)
|
||||
|
||||
AX_APPEND_COMPILE_FLAGS([ dnl
|
||||
[$ax_compiler_no_suggest_attribute_flags] dnl
|
||||
],ax_warn_cflags_variable,[$ax_compiler_flags_test])
|
||||
])
|
||||
|
||||
# In the flags below, when disabling specific flags, always add *both*
|
||||
# -Wno-foo and -Wno-error=foo. This fixes the situation where (for example)
|
||||
# we enable -Werror, disable a flag, and a build bot passes CFLAGS=-Wall,
|
||||
# which effectively turns that flag back on again as an error.
|
||||
for flag in $ax_warn_cflags_variable; do
|
||||
AS_CASE([$flag],
|
||||
[-Wno-*=*],[],
|
||||
[-Wno-*],[
|
||||
AX_APPEND_COMPILE_FLAGS([-Wno-error=$(AS_ECHO([$flag]) | $SED 's/^-Wno-//')],
|
||||
ax_warn_cflags_variable,
|
||||
[$ax_compiler_flags_test])
|
||||
])
|
||||
done
|
||||
|
||||
AC_LANG_POP([C])
|
||||
|
||||
# Substitute the variables
|
||||
AC_SUBST(ax_warn_cflags_variable)
|
||||
])dnl AX_COMPILER_FLAGS
|
|
@ -0,0 +1,60 @@
|
|||
# ===========================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_compiler_flags_gir.html
|
||||
# ===========================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_COMPILER_FLAGS_GIR([VARIABLE], [IS-RELEASE], [EXTRA-BASE-FLAGS], [EXTRA-YES-FLAGS])
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# Add warning flags for the g-ir-scanner (from GObject Introspection) to
|
||||
# VARIABLE, which defaults to WARN_SCANNERFLAGS. VARIABLE is AC_SUBST-ed
|
||||
# by this macro, but must be manually added to the SCANNERFLAGS variable
|
||||
# for each GIR target in the code base.
|
||||
#
|
||||
# This macro depends on the environment set up by AX_COMPILER_FLAGS.
|
||||
# Specifically, it uses the value of $ax_enable_compile_warnings to decide
|
||||
# which flags to enable.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2015 Philip Withnall <philip@tecnocode.co.uk>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 6
|
||||
|
||||
AC_DEFUN([AX_COMPILER_FLAGS_GIR],[
|
||||
AX_REQUIRE_DEFINED([AX_APPEND_FLAG])
|
||||
|
||||
# Variable names
|
||||
m4_define([ax_warn_scannerflags_variable],
|
||||
[m4_normalize(ifelse([$1],,[WARN_SCANNERFLAGS],[$1]))])
|
||||
|
||||
# Base flags
|
||||
AX_APPEND_FLAG([$3],ax_warn_scannerflags_variable)
|
||||
|
||||
AS_IF([test "$ax_enable_compile_warnings" != "no"],[
|
||||
# "yes" flags
|
||||
AX_APPEND_FLAG([ dnl
|
||||
--warn-all dnl
|
||||
$4 dnl
|
||||
$5 dnl
|
||||
$6 dnl
|
||||
$7 dnl
|
||||
],ax_warn_scannerflags_variable)
|
||||
])
|
||||
AS_IF([test "$ax_enable_compile_warnings" = "error"],[
|
||||
# "error" flags
|
||||
AX_APPEND_FLAG([ dnl
|
||||
--warn-error dnl
|
||||
],ax_warn_scannerflags_variable)
|
||||
])
|
||||
|
||||
# Substitute the variables
|
||||
AC_SUBST(ax_warn_scannerflags_variable)
|
||||
])dnl AX_COMPILER_FLAGS
|
|
@ -0,0 +1,111 @@
|
|||
# ==============================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_compiler_flags_ldflags.html
|
||||
# ==============================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_COMPILER_FLAGS_LDFLAGS([VARIABLE], [IS-RELEASE], [EXTRA-BASE-FLAGS], [EXTRA-YES-FLAGS])
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# Add warning flags for the linker to VARIABLE, which defaults to
|
||||
# WARN_LDFLAGS. VARIABLE is AC_SUBST-ed by this macro, but must be
|
||||
# manually added to the LDFLAGS variable for each target in the code base.
|
||||
#
|
||||
# This macro depends on the environment set up by AX_COMPILER_FLAGS.
|
||||
# Specifically, it uses the value of $ax_enable_compile_warnings to decide
|
||||
# which flags to enable.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2014, 2015 Philip Withnall <philip@tecnocode.co.uk>
|
||||
# Copyright (c) 2017, 2018 Reini Urban <rurban@cpan.org>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 9
|
||||
|
||||
AC_DEFUN([AX_COMPILER_FLAGS_LDFLAGS],[
|
||||
AX_REQUIRE_DEFINED([AX_APPEND_LINK_FLAGS])
|
||||
AX_REQUIRE_DEFINED([AX_APPEND_FLAG])
|
||||
AX_REQUIRE_DEFINED([AX_CHECK_COMPILE_FLAG])
|
||||
AX_REQUIRE_DEFINED([AX_CHECK_LINK_FLAG])
|
||||
|
||||
# Variable names
|
||||
m4_define([ax_warn_ldflags_variable],
|
||||
[m4_normalize(ifelse([$1],,[WARN_LDFLAGS],[$1]))])
|
||||
|
||||
# Always pass -Werror=unknown-warning-option to get Clang to fail on bad
|
||||
# flags, otherwise they are always appended to the warn_ldflags variable,
|
||||
# and Clang warns on them for every compilation unit.
|
||||
# If this is passed to GCC, it will explode, so the flag must be enabled
|
||||
# conditionally.
|
||||
AX_CHECK_COMPILE_FLAG([-Werror=unknown-warning-option],[
|
||||
ax_compiler_flags_test="-Werror=unknown-warning-option"
|
||||
],[
|
||||
ax_compiler_flags_test=""
|
||||
])
|
||||
|
||||
AX_CHECK_LINK_FLAG([-Wl,--as-needed], [
|
||||
AX_APPEND_LINK_FLAGS([-Wl,--as-needed],
|
||||
[AM_LDFLAGS],[$ax_compiler_flags_test])
|
||||
])
|
||||
AX_CHECK_LINK_FLAG([-Wl,-z,relro], [
|
||||
AX_APPEND_LINK_FLAGS([-Wl,-z,relro],
|
||||
[AM_LDFLAGS],[$ax_compiler_flags_test])
|
||||
])
|
||||
AX_CHECK_LINK_FLAG([-Wl,-z,now], [
|
||||
AX_APPEND_LINK_FLAGS([-Wl,-z,now],
|
||||
[AM_LDFLAGS],[$ax_compiler_flags_test])
|
||||
])
|
||||
AX_CHECK_LINK_FLAG([-Wl,-z,noexecstack], [
|
||||
AX_APPEND_LINK_FLAGS([-Wl,-z,noexecstack],
|
||||
[AM_LDFLAGS],[$ax_compiler_flags_test])
|
||||
])
|
||||
# textonly, retpolineplt not yet
|
||||
|
||||
# macOS and cygwin linker do not have --as-needed
|
||||
AX_CHECK_LINK_FLAG([-Wl,--no-as-needed], [
|
||||
ax_compiler_flags_as_needed_option="-Wl,--no-as-needed"
|
||||
], [
|
||||
ax_compiler_flags_as_needed_option=""
|
||||
])
|
||||
|
||||
# macOS linker speaks with a different accent
|
||||
ax_compiler_flags_fatal_warnings_option=""
|
||||
AX_CHECK_LINK_FLAG([-Wl,--fatal-warnings], [
|
||||
ax_compiler_flags_fatal_warnings_option="-Wl,--fatal-warnings"
|
||||
])
|
||||
AX_CHECK_LINK_FLAG([-Wl,-fatal_warnings], [
|
||||
ax_compiler_flags_fatal_warnings_option="-Wl,-fatal_warnings"
|
||||
])
|
||||
|
||||
# Base flags
|
||||
AX_APPEND_LINK_FLAGS([ dnl
|
||||
$ax_compiler_flags_as_needed_option dnl
|
||||
$3 dnl
|
||||
],ax_warn_ldflags_variable,[$ax_compiler_flags_test])
|
||||
|
||||
AS_IF([test "$ax_enable_compile_warnings" != "no"],[
|
||||
# "yes" flags
|
||||
AX_APPEND_LINK_FLAGS([$4 $5 $6 $7],
|
||||
ax_warn_ldflags_variable,
|
||||
[$ax_compiler_flags_test])
|
||||
])
|
||||
AS_IF([test "$ax_enable_compile_warnings" = "error"],[
|
||||
# "error" flags; -Werror has to be appended unconditionally because
|
||||
# it's not possible to test for
|
||||
#
|
||||
# suggest-attribute=format is disabled because it gives too many false
|
||||
# positives
|
||||
AX_APPEND_LINK_FLAGS([ dnl
|
||||
$ax_compiler_flags_fatal_warnings_option dnl
|
||||
],ax_warn_ldflags_variable,[$ax_compiler_flags_test])
|
||||
])
|
||||
|
||||
# Substitute the variables
|
||||
AC_SUBST(ax_warn_ldflags_variable)
|
||||
])dnl AX_COMPILER_FLAGS
|
|
@ -0,0 +1,242 @@
|
|||
# ===========================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_gcc_func_attribute.html
|
||||
# ===========================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_GCC_FUNC_ATTRIBUTE(ATTRIBUTE)
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# This macro checks if the compiler supports one of GCC's function
|
||||
# attributes; many other compilers also provide function attributes with
|
||||
# the same syntax. Compiler warnings are used to detect supported
|
||||
# attributes as unsupported ones are ignored by default so quieting
|
||||
# warnings when using this macro will yield false positives.
|
||||
#
|
||||
# The ATTRIBUTE parameter holds the name of the attribute to be checked.
|
||||
#
|
||||
# If ATTRIBUTE is supported define HAVE_FUNC_ATTRIBUTE_<ATTRIBUTE>.
|
||||
#
|
||||
# The macro caches its result in the ax_cv_have_func_attribute_<attribute>
|
||||
# variable.
|
||||
#
|
||||
# The macro currently supports the following function attributes:
|
||||
#
|
||||
# alias
|
||||
# aligned
|
||||
# alloc_size
|
||||
# always_inline
|
||||
# artificial
|
||||
# cold
|
||||
# const
|
||||
# constructor
|
||||
# constructor_priority for constructor attribute with priority
|
||||
# deprecated
|
||||
# destructor
|
||||
# dllexport
|
||||
# dllimport
|
||||
# error
|
||||
# externally_visible
|
||||
# fallthrough
|
||||
# flatten
|
||||
# format
|
||||
# format_arg
|
||||
# gnu_format
|
||||
# gnu_inline
|
||||
# hot
|
||||
# ifunc
|
||||
# leaf
|
||||
# malloc
|
||||
# noclone
|
||||
# noinline
|
||||
# nonnull
|
||||
# noreturn
|
||||
# nothrow
|
||||
# optimize
|
||||
# pure
|
||||
# sentinel
|
||||
# sentinel_position
|
||||
# unused
|
||||
# used
|
||||
# visibility
|
||||
# warning
|
||||
# warn_unused_result
|
||||
# weak
|
||||
# weakref
|
||||
#
|
||||
# Unsupported function attributes will be tested with a prototype
|
||||
# returning an int and not accepting any arguments and the result of the
|
||||
# check might be wrong or meaningless so use with care.
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2013 Gabriele Svelto <gabriele.svelto@gmail.com>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 13
|
||||
|
||||
AC_DEFUN([AX_GCC_FUNC_ATTRIBUTE], [
|
||||
AS_VAR_PUSHDEF([ac_var], [ax_cv_have_func_attribute_$1])
|
||||
|
||||
AC_CACHE_CHECK([for __attribute__(($1))], [ac_var], [
|
||||
AC_LINK_IFELSE([AC_LANG_PROGRAM([
|
||||
m4_case([$1],
|
||||
[alias], [
|
||||
int foo( void ) { return 0; }
|
||||
int bar( void ) __attribute__(($1("foo")));
|
||||
],
|
||||
[aligned], [
|
||||
int foo( void ) __attribute__(($1(32)));
|
||||
],
|
||||
[alloc_size], [
|
||||
void *foo(int a) __attribute__(($1(1)));
|
||||
],
|
||||
[always_inline], [
|
||||
inline __attribute__(($1)) int foo( void ) { return 0; }
|
||||
],
|
||||
[artificial], [
|
||||
inline __attribute__(($1)) int foo( void ) { return 0; }
|
||||
],
|
||||
[cold], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[const], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[constructor_priority], [
|
||||
int foo( void ) __attribute__((__constructor__(65535/2)));
|
||||
],
|
||||
[constructor], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[deprecated], [
|
||||
int foo( void ) __attribute__(($1("")));
|
||||
],
|
||||
[destructor], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[dllexport], [
|
||||
__attribute__(($1)) int foo( void ) { return 0; }
|
||||
],
|
||||
[dllimport], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[error], [
|
||||
int foo( void ) __attribute__(($1("")));
|
||||
],
|
||||
[externally_visible], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[fallthrough], [
|
||||
void foo( int x ) {switch (x) { case 1: __attribute__(($1)); case 2: break ; }};
|
||||
],
|
||||
[flatten], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[format], [
|
||||
int foo(const char *p, ...) __attribute__(($1(printf, 1, 2)));
|
||||
],
|
||||
[gnu_format], [
|
||||
int foo(const char *p, ...) __attribute__((format(gnu_printf, 1, 2)));
|
||||
],
|
||||
[format_arg], [
|
||||
char *foo(const char *p) __attribute__(($1(1)));
|
||||
],
|
||||
[gnu_inline], [
|
||||
inline __attribute__(($1)) int foo( void ) { return 0; }
|
||||
],
|
||||
[hot], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[ifunc], [
|
||||
int my_foo( void ) { return 0; }
|
||||
static int (*resolve_foo(void))(void) { return my_foo; }
|
||||
int foo( void ) __attribute__(($1("resolve_foo")));
|
||||
],
|
||||
[leaf], [
|
||||
__attribute__(($1)) int foo( void ) { return 0; }
|
||||
],
|
||||
[malloc], [
|
||||
void *foo( void ) __attribute__(($1));
|
||||
],
|
||||
[noclone], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[noinline], [
|
||||
__attribute__(($1)) int foo( void ) { return 0; }
|
||||
],
|
||||
[nonnull], [
|
||||
int foo(char *p) __attribute__(($1(1)));
|
||||
],
|
||||
[noreturn], [
|
||||
void foo( void ) __attribute__(($1));
|
||||
],
|
||||
[nothrow], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[optimize], [
|
||||
__attribute__(($1(3))) int foo( void ) { return 0; }
|
||||
],
|
||||
[pure], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[sentinel], [
|
||||
int foo(void *p, ...) __attribute__(($1));
|
||||
],
|
||||
[sentinel_position], [
|
||||
int foo(void *p, ...) __attribute__(($1(1)));
|
||||
],
|
||||
[returns_nonnull], [
|
||||
void *foo( void ) __attribute__(($1));
|
||||
],
|
||||
[unused], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[used], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[visibility], [
|
||||
int foo_def( void ) __attribute__(($1("default")));
|
||||
int foo_hid( void ) __attribute__(($1("hidden")));
|
||||
int foo_int( void ) __attribute__(($1("internal")));
|
||||
int foo_pro( void ) __attribute__(($1("protected")));
|
||||
],
|
||||
[warning], [
|
||||
int foo( void ) __attribute__(($1("")));
|
||||
],
|
||||
[warn_unused_result], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[weak], [
|
||||
int foo( void ) __attribute__(($1));
|
||||
],
|
||||
[weakref], [
|
||||
static int foo( void ) { return 0; }
|
||||
static int bar( void ) __attribute__(($1("foo")));
|
||||
],
|
||||
[
|
||||
m4_warn([syntax], [Unsupported attribute $1, the test may fail])
|
||||
int foo( void ) __attribute__(($1));
|
||||
]
|
||||
)], [])
|
||||
],
|
||||
dnl GCC doesn't exit with an error if an unknown attribute is
|
||||
dnl provided but only outputs a warning, so accept the attribute
|
||||
dnl only if no warning were issued.
|
||||
[AS_IF([grep -- -Wattributes conftest.err],
|
||||
[AS_VAR_SET([ac_var], [no])],
|
||||
[AS_VAR_SET([ac_var], [yes])])],
|
||||
[AS_VAR_SET([ac_var], [no])])
|
||||
])
|
||||
|
||||
AS_IF([test yes = AS_VAR_GET([ac_var])],
|
||||
[AC_DEFINE_UNQUOTED(AS_TR_CPP(HAVE_FUNC_ATTRIBUTE_$1), 1,
|
||||
[Define to 1 if the system has the `$1' function attribute])], [])
|
||||
|
||||
AS_VAR_POPDEF([ac_var])
|
||||
])
|
|
@ -0,0 +1,37 @@
|
|||
# ===========================================================================
|
||||
# https://www.gnu.org/software/autoconf-archive/ax_require_defined.html
|
||||
# ===========================================================================
|
||||
#
|
||||
# SYNOPSIS
|
||||
#
|
||||
# AX_REQUIRE_DEFINED(MACRO)
|
||||
#
|
||||
# DESCRIPTION
|
||||
#
|
||||
# AX_REQUIRE_DEFINED is a simple helper for making sure other macros have
|
||||
# been defined and thus are available for use. This avoids random issues
|
||||
# where a macro isn't expanded. Instead the configure script emits a
|
||||
# non-fatal:
|
||||
#
|
||||
# ./configure: line 1673: AX_CFLAGS_WARN_ALL: command not found
|
||||
#
|
||||
# It's like AC_REQUIRE except it doesn't expand the required macro.
|
||||
#
|
||||
# Here's an example:
|
||||
#
|
||||
# AX_REQUIRE_DEFINED([AX_CHECK_LINK_FLAG])
|
||||
#
|
||||
# LICENSE
|
||||
#
|
||||
# Copyright (c) 2014 Mike Frysinger <vapier@gentoo.org>
|
||||
#
|
||||
# Copying and distribution of this file, with or without modification, are
|
||||
# permitted in any medium without royalty provided the copyright notice
|
||||
# and this notice are preserved. This file is offered as-is, without any
|
||||
# warranty.
|
||||
|
||||
#serial 2
|
||||
|
||||
AC_DEFUN([AX_REQUIRE_DEFINED], [dnl
|
||||
m4_ifndef([$1], [m4_fatal([macro ]$1[ is not defined; is a m4 file missing?])])
|
||||
])dnl AX_REQUIRE_DEFINED
|
|
@ -0,0 +1,208 @@
|
|||
/*
|
||||
* Pinelog lightweight logging library
|
||||
*
|
||||
* Copyright (C) 2021 Nirenjan Krishnan (nirenjan@nirenjan.org)
|
||||
*
|
||||
* SPDX-License-Identifier: MIT
|
||||
*/
|
||||
|
||||
#include "config.h"
|
||||
#include <stdio.h>
|
||||
#include <stdarg.h>
|
||||
#include <time.h>
|
||||
#include <errno.h>
|
||||
|
||||
#include "pinelog.h"
|
||||
|
||||
/**********************************************************************
|
||||
* Configure defaults
|
||||
*********************************************************************/
|
||||
#ifndef PINELOG_DEFAULT_STREAM
|
||||
#define PINELOG_DEFAULT_STREAM stdout
|
||||
#endif
|
||||
|
||||
#ifndef PINELOG_DEFAULT_LEVEL
|
||||
#define PINELOG_DEFAULT_LEVEL PINELOG_LVL_ERROR
|
||||
#endif
|
||||
|
||||
/**********************************************************************
|
||||
* Configure logging parameters
|
||||
*********************************************************************/
|
||||
#ifndef PINELOG_SHOW_DATE
|
||||
#define PINELOG_SHOW_DATE 0
|
||||
#endif
|
||||
|
||||
#ifndef PINELOG_SHOW_LEVEL
|
||||
#define PINELOG_SHOW_LEVEL 0
|
||||
#endif
|
||||
|
||||
#ifndef PINELOG_SHOW_BACKTRACE
|
||||
#define PINELOG_SHOW_BACKTRACE 0
|
||||
#endif
|
||||
|
||||
/**********************************************************************
|
||||
* Configure level strings
|
||||
*********************************************************************/
|
||||
#ifndef PINELOG_FATAL_STR
|
||||
#define PINELOG_FATAL_STR "FATAL"
|
||||
#endif
|
||||
|
||||
#ifndef PINELOG_ERROR_STR
|
||||
#define PINELOG_ERROR_STR "ERROR"
|
||||
#endif
|
||||
|
||||
#ifndef PINELOG_WARNING_STR
|
||||
#define PINELOG_WARNING_STR "WARNING"
|
||||
#endif
|
||||
|
||||
#ifndef PINELOG_INFO_STR
|
||||
#define PINELOG_INFO_STR "INFO"
|
||||
#endif
|
||||
|
||||
#ifndef PINELOG_DEBUG_STR
|
||||
#define PINELOG_DEBUG_STR "DEBUG"
|
||||
#endif
|
||||
|
||||
#ifndef PINELOG_TRACE_STR
|
||||
#define PINELOG_TRACE_STR "TRACE"
|
||||
#endif
|
||||
|
||||
/**********************************************************************
|
||||
* Global variables
|
||||
*********************************************************************/
|
||||
|
||||
/** Stream buffer */
|
||||
static FILE *output_stream = NULL;
|
||||
|
||||
/** Default logging level */
|
||||
static int log_level = PINELOG_DEFAULT_LEVEL;
|
||||
|
||||
/* Initialize defaults */
|
||||
#if HAVE_FUNC_ATTRIBUTE_CONSTRUCTOR
|
||||
__attribute__((constructor))
|
||||
#endif
|
||||
void pinelog_set_defaults(void)
|
||||
{
|
||||
output_stream = PINELOG_DEFAULT_STREAM;
|
||||
log_level = PINELOG_DEFAULT_LEVEL;
|
||||
}
|
||||
|
||||
int pinelog_set_output_stream(FILE *stream)
|
||||
{
|
||||
if (stream == NULL) {
|
||||
return EINVAL;
|
||||
}
|
||||
|
||||
/* If current output stream is not stdout or stderr, then close it */
|
||||
if (output_stream != stdout && output_stream != stderr) {
|
||||
fclose(output_stream);
|
||||
}
|
||||
|
||||
setlinebuf(stream);
|
||||
output_stream = stream;
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef PINELOG_TEST
|
||||
FILE * pinelog_get_output_stream(void)
|
||||
{
|
||||
return output_stream;
|
||||
}
|
||||
#endif
|
||||
|
||||
int pinelog_set_output_file(const char *file)
|
||||
{
|
||||
FILE *stream;
|
||||
if (file == NULL) {
|
||||
return EINVAL;
|
||||
}
|
||||
|
||||
errno = 0;
|
||||
stream = fopen(file, "w");
|
||||
if (stream == NULL) {
|
||||
return errno;
|
||||
}
|
||||
|
||||
return pinelog_set_output_stream(stream);
|
||||
}
|
||||
|
||||
int pinelog_get_level(void)
|
||||
{
|
||||
return log_level;
|
||||
}
|
||||
|
||||
int pinelog_set_level(int level)
|
||||
{
|
||||
if (level < PINELOG_LVL_NONE || level > PINELOG_LVL_TRACE) {
|
||||
return EINVAL;
|
||||
}
|
||||
|
||||
log_level = level;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**********************************************************************
|
||||
* Log the message to the output stream
|
||||
*********************************************************************/
|
||||
void pinelog_log_message(int level, const char *file, int line, const char *fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
|
||||
/* Don't log anything if the level is not severe enough */
|
||||
if (level > log_level || level < 0) {
|
||||
return;
|
||||
}
|
||||
|
||||
/* Cap the log level */
|
||||
if (level > PINELOG_LVL_TRACE) {
|
||||
level = PINELOG_LVL_TRACE;
|
||||
}
|
||||
|
||||
#if !HAVE_FUNC_ATTRIBUTE_CONSTRUCTOR
|
||||
/*
|
||||
* Validate and set output stream. Only necessary if the compiler doesn't
|
||||
* support the constructor attribute
|
||||
*/
|
||||
if (output_stream == NULL) {
|
||||
output_stream = PINELOG_DEFAULT_STREAM;
|
||||
}
|
||||
#endif
|
||||
|
||||
#if PINELOG_SHOW_DATE
|
||||
do {
|
||||
time_t t;
|
||||
struct tm *tmp;
|
||||
char date_string[30];
|
||||
t = time(NULL);
|
||||
tmp = localtime(&t);
|
||||
strftime(date_string, sizeof(date_string), "%F %T ", tmp);
|
||||
fputs(date_string, output_stream);
|
||||
} while (0);
|
||||
#endif
|
||||
|
||||
#if PINELOG_SHOW_LEVEL
|
||||
do {
|
||||
static const char *level_strings[] = {
|
||||
PINELOG_FATAL_STR,
|
||||
PINELOG_ERROR_STR,
|
||||
PINELOG_WARNING_STR,
|
||||
PINELOG_INFO_STR,
|
||||
PINELOG_DEBUG_STR,
|
||||
PINELOG_TRACE_STR,
|
||||
};
|
||||
|
||||
fputs(level_strings[level], output_stream);
|
||||
fputs(": ", output_stream);
|
||||
} while (0);
|
||||
#endif
|
||||
|
||||
#if PINELOG_SHOW_BACKTRACE
|
||||
fprintf(output_stream, "%s:%d ", file, line);
|
||||
#endif
|
||||
|
||||
va_start(ap, fmt);
|
||||
vfprintf(output_stream, fmt, ap);
|
||||
va_end(ap);
|
||||
// Append a trailing newline to flush the log message
|
||||
fputs("\n", output_stream);
|
||||
}
|
|
@ -0,0 +1,173 @@
|
|||
/*
|
||||
* Pinelog lightweight logging library
|
||||
*
|
||||
* Copyright (C) 2021 Nirenjan Krishnan (nirenjan@nirenjan.org)
|
||||
*
|
||||
* SPDX-License-Identifier: MIT
|
||||
*/
|
||||
|
||||
/**
|
||||
* @file logging.h
|
||||
* @brief Logging utility library
|
||||
*
|
||||
* This file contains the prototypes for the pinelog logging library
|
||||
* used by any programs that need to log messages.
|
||||
*
|
||||
* @author Nirenjan Krishnan (nirenjan@nirenjan.org)
|
||||
*/
|
||||
#ifndef LOGGING_H
|
||||
#define LOGGING_H
|
||||
|
||||
#include "config.h"
|
||||
#include <stdio.h>
|
||||
#ifndef PINELOG_TEST
|
||||
#include <stdlib.h>
|
||||
#endif
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
/**
|
||||
* @brief Logging levels
|
||||
*
|
||||
* The log levels indicate the lowest severity level that will actually be
|
||||
* logged to the logging framework.
|
||||
*/
|
||||
enum {
|
||||
/** No messages will be logged */
|
||||
PINELOG_LVL_NONE = -1,
|
||||
|
||||
/** Only fatal messages will be logged */
|
||||
PINELOG_LVL_FATAL,
|
||||
|
||||
/** Error messages. This is the default log level */
|
||||
PINELOG_LVL_ERROR,
|
||||
|
||||
/** Warning messages */
|
||||
PINELOG_LVL_WARNING,
|
||||
|
||||
/** Informational messages */
|
||||
PINELOG_LVL_INFO,
|
||||
|
||||
/** Debug messages */
|
||||
PINELOG_LVL_DEBUG,
|
||||
|
||||
/** Trace messages */
|
||||
PINELOG_LVL_TRACE,
|
||||
};
|
||||
|
||||
/**
|
||||
* @brief Set the default log level and output stream
|
||||
*/
|
||||
void pinelog_set_defaults(void);
|
||||
|
||||
#ifdef PINELOG_TEST
|
||||
/**
|
||||
* @brief Get the pointer to the output stream. Only used in test harness.
|
||||
*
|
||||
* @returns FILE pointer to output stream
|
||||
*/
|
||||
FILE * pinelog_get_output_stream(void);
|
||||
#endif
|
||||
|
||||
/**
|
||||
* @brief Set the output stream. Must be a FILE pointer.
|
||||
*
|
||||
* @param[in] stream Pointer to the output stream
|
||||
*
|
||||
* @returns 0 on success, EINVAL if the pointer is not valid.
|
||||
*/
|
||||
int pinelog_set_output_stream(FILE *stream);
|
||||
|
||||
/**
|
||||
* @brief Set the output file.
|
||||
*
|
||||
* @param[in] file Filename to write to
|
||||
*
|
||||
* @returns 0 on success, EINVAL if the filename pointer is not valid, other
|
||||
* error if the file could not be opened for writing.
|
||||
*/
|
||||
int pinelog_set_output_file(const char *file);
|
||||
|
||||
/**
|
||||
* @brief Set the logging level
|
||||
*
|
||||
* @param[in] level Level to filter
|
||||
*
|
||||
* @returns 0 on success, EINVAL if the level is not valid
|
||||
*/
|
||||
int pinelog_set_level(int level);
|
||||
|
||||
/**
|
||||
* @brief Get the logging level
|
||||
*
|
||||
* @returns the configured logging level
|
||||
*/
|
||||
int pinelog_get_level(void);
|
||||
|
||||
/**
|
||||
* @brief Log a message to the logger
|
||||
*
|
||||
* This is the actual function that logs the message. The application should
|
||||
* never need to call this directly, but instead, should always use the
|
||||
* \code PINELOG_* macros.
|
||||
*
|
||||
* @param[in] level Level to log the message at
|
||||
* @param[in] fmt Format string
|
||||
*
|
||||
* @returns None
|
||||
*/
|
||||
#if HAVE_FUNC_ATTRIBUTE_FORMAT
|
||||
__attribute__((format(printf, 4, 5)))
|
||||
#endif
|
||||
|
||||
void pinelog_log_message(int level, const char *file, int line, const char *fmt, ...);
|
||||
|
||||
// Test harness will redefine pinelog_exit
|
||||
#ifndef PINELOG_TEST
|
||||
#define pinelog_exit exit
|
||||
#endif
|
||||
|
||||
#define PINELOG_FATAL(fmt, ...) do { \
|
||||
if (PINELOG_LVL_FATAL <= pinelog_get_level()) { \
|
||||
pinelog_log_message(PINELOG_LVL_FATAL, __FILE__, __LINE__, fmt, ##__VA_ARGS__); \
|
||||
} \
|
||||
pinelog_exit(1); \
|
||||
} while (0)
|
||||
|
||||
#define PINELOG_ERROR(fmt, ...) do { \
|
||||
if (PINELOG_LVL_ERROR <= pinelog_get_level()) { \
|
||||
pinelog_log_message(PINELOG_LVL_ERROR, __FILE__, __LINE__, fmt, ##__VA_ARGS__); \
|
||||
} \
|
||||
} while (0)
|
||||
|
||||
#define PINELOG_WARN(fmt, ...) do { \
|
||||
if (PINELOG_LVL_WARNING <= pinelog_get_level()) { \
|
||||
pinelog_log_message(PINELOG_LVL_WARNING, __FILE__, __LINE__, fmt, ##__VA_ARGS__); \
|
||||
} \
|
||||
} while(0)
|
||||
|
||||
#define PINELOG_INFO(fmt, ...) do { \
|
||||
if (PINELOG_LVL_INFO <= pinelog_get_level()) { \
|
||||
pinelog_log_message(PINELOG_LVL_INFO, __FILE__, __LINE__, fmt, ##__VA_ARGS__); \
|
||||
} \
|
||||
} while(0)
|
||||
|
||||
#define PINELOG_DEBUG(fmt, ...) do { \
|
||||
if (PINELOG_LVL_DEBUG <= pinelog_get_level()) { \
|
||||
pinelog_log_message(PINELOG_LVL_DEBUG, __FILE__, __LINE__, fmt, ##__VA_ARGS__); \
|
||||
} \
|
||||
} while(0)
|
||||
|
||||
#define PINELOG_TRACE(fmt, ...) do { \
|
||||
if (PINELOG_LVL_TRACE <= pinelog_get_level()) { \
|
||||
pinelog_log_message(PINELOG_LVL_TRACE, __FILE__, __LINE__, fmt, ##__VA_ARGS__); \
|
||||
} \
|
||||
} while(0)
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif // !defined LOGGING_H
|
|
@ -0,0 +1,194 @@
|
|||
/*
|
||||
* Pinelog lightweight logging library - test harness
|
||||
*
|
||||
* Copyright (C) 2021 Nirenjan Krishnan (nirenjan@nirenjan.org)
|
||||
*
|
||||
* SPDX-License-Identifier: MIT
|
||||
*/
|
||||
|
||||
#include "pinelog.h"
|
||||
#include <stdio.h>
|
||||
#include <stdarg.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
#include <limits.h>
|
||||
#include <time.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/stat.h>
|
||||
#include <fcntl.h>
|
||||
#include <unistd.h>
|
||||
|
||||
/**********************************************************************
|
||||
* Global variables
|
||||
*********************************************************************/
|
||||
// Test ID (of current test case)
|
||||
static unsigned int test_id;
|
||||
|
||||
// Observed output stream
|
||||
static FILE *observed_stream_w;
|
||||
static FILE *observed_stream_r;
|
||||
|
||||
// Temporary pipe for observed data
|
||||
static char observed_fifo[NAME_MAX];
|
||||
|
||||
// Buffer for expected output
|
||||
static char expected_output[1024];
|
||||
static size_t expected_len;
|
||||
|
||||
static void test_case(const char *desc, bool test)
|
||||
{
|
||||
test_id++;
|
||||
if (test) {
|
||||
printf("ok %u %s\n", test_id, desc);
|
||||
} else {
|
||||
printf("not ok %u %s\n", test_id, desc);
|
||||
}
|
||||
}
|
||||
|
||||
static void pinelog_exit(int status)
|
||||
{
|
||||
fprintf(observed_stream_w, "EXIT(%d)\n", status);
|
||||
expected_len += snprintf(&expected_output[expected_len],
|
||||
sizeof(expected_output) - expected_len,
|
||||
"EXIT(%d)\n", status);
|
||||
}
|
||||
|
||||
static void dump_data(const char *type, size_t len, char *data)
|
||||
{
|
||||
char *line;
|
||||
printf("# %s (%lu bytes):\n", type, len);
|
||||
line = strtok(data, "\n");
|
||||
while (line != NULL) {
|
||||
printf("#\t%s\n", line);
|
||||
line = strtok(NULL, "\n");
|
||||
}
|
||||
printf("\n");
|
||||
}
|
||||
|
||||
static void test_setup(int level, int filter, const char *file, int line, const char *fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
|
||||
expected_len = 0;
|
||||
memset(expected_output, 0, sizeof(expected_output));
|
||||
|
||||
if (level <= filter) {
|
||||
if (PINELOG_SHOW_DATE) {
|
||||
time_t t;
|
||||
struct tm *tmp;
|
||||
|
||||
t = time(NULL);
|
||||
tmp = localtime(&t);
|
||||
expected_len += strftime(&expected_output[expected_len],
|
||||
sizeof(expected_output) - expected_len,
|
||||
"%F %T ", tmp);
|
||||
}
|
||||
|
||||
if (PINELOG_SHOW_LEVEL) {
|
||||
const char * level_string[] = {
|
||||
PINELOG_FATAL_STR,
|
||||
PINELOG_ERROR_STR,
|
||||
PINELOG_WARNING_STR,
|
||||
PINELOG_INFO_STR,
|
||||
PINELOG_DEBUG_STR,
|
||||
PINELOG_TRACE_STR,
|
||||
};
|
||||
expected_len += snprintf(&expected_output[expected_len],
|
||||
sizeof(expected_output) - expected_len,
|
||||
"%s: ", level_string[level]);
|
||||
}
|
||||
|
||||
if (PINELOG_SHOW_BACKTRACE) {
|
||||
expected_len += snprintf(&expected_output[expected_len],
|
||||
sizeof(expected_output) - expected_len,
|
||||
"%s:%d ", file, line);
|
||||
}
|
||||
|
||||
va_start(ap, fmt);
|
||||
expected_len += vsnprintf(&expected_output[expected_len],
|
||||
sizeof(expected_output) - expected_len,
|
||||
fmt, ap);
|
||||
va_end(ap);
|
||||
expected_output[expected_len] = '\n';
|
||||
expected_len++;
|
||||
}
|
||||
}
|
||||
|
||||
static void test_teardown(const char *desc)
|
||||
{
|
||||
// Compare the output
|
||||
static char observed[1024];
|
||||
size_t observed_len;
|
||||
int result;
|
||||
|
||||
observed_len = fread(observed, 1, sizeof(observed), observed_stream_r);
|
||||
|
||||
result = ((expected_len == observed_len) &&
|
||||
(memcmp(expected_output, observed, expected_len) == 0));
|
||||
test_case(desc, result);
|
||||
if (!result) {
|
||||
dump_data("expected", expected_len, expected_output);
|
||||
dump_data("observed", observed_len, observed);
|
||||
}
|
||||
}
|
||||
|
||||
static void verify_defaults(void)
|
||||
{
|
||||
test_case("Get default output stream",
|
||||
pinelog_get_output_stream() == PINELOG_DEFAULT_STREAM);
|
||||
test_case("Get default logging level",
|
||||
pinelog_get_level() == PINELOG_DEFAULT_LEVEL);
|
||||
}
|
||||
|
||||
#define PINELOG_WARNING PINELOG_WARN
|
||||
|
||||
#define TEST_LOG(lvl, filter, fmt, ...) do { \
|
||||
test_setup(PINELOG_LVL_ ## lvl, PINELOG_LVL_ ## filter, \
|
||||
__FILE__, __LINE__, fmt, ##__VA_ARGS__); \
|
||||
PINELOG_ ## lvl (fmt, ##__VA_ARGS__); \
|
||||
test_teardown("Log " #lvl " filter " #filter); \
|
||||
} while(0)
|
||||
|
||||
#define TEST(filter, fmt, ...) do { \
|
||||
pinelog_set_level(PINELOG_LVL_ ## filter); \
|
||||
TEST_LOG(TRACE, filter, fmt, ##__VA_ARGS__); \
|
||||
TEST_LOG(DEBUG, filter, fmt, ##__VA_ARGS__); \
|
||||
TEST_LOG(INFO, filter, fmt, ##__VA_ARGS__); \
|
||||
TEST_LOG(WARNING, filter, fmt, ##__VA_ARGS__); \
|
||||
TEST_LOG(ERROR, filter, fmt, ##__VA_ARGS__); \
|
||||
TEST_LOG(FATAL, filter, fmt, ##__VA_ARGS__); \
|
||||
} while (0)
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
int fifo_fd_r, fifo_fd_w;
|
||||
snprintf(observed_fifo, sizeof(observed_fifo), "%s.fifo", argv[0]);
|
||||
mkfifo(observed_fifo, 0777);
|
||||
|
||||
fifo_fd_r = open(observed_fifo, O_RDONLY | O_NONBLOCK);
|
||||
fifo_fd_w = open(observed_fifo, O_WRONLY | O_NONBLOCK);
|
||||
observed_stream_r = fdopen(fifo_fd_r, "r");
|
||||
observed_stream_w = fdopen(fifo_fd_w, "w");
|
||||
|
||||
verify_defaults();
|
||||
|
||||
pinelog_set_output_stream(observed_stream_w);
|
||||
TEST(TRACE, "testing %s... %d, %f, %u", "testing", -1, 0.0, 1);
|
||||
TEST(DEBUG, "testing %s... %d, %f, %u", "testing", -1, 0.0, 1);
|
||||
TEST(INFO, "testing %s... %d, %f, %u", "testing", -1, 0.0, 1);
|
||||
TEST(WARNING, "testing %s... %d, %f, %u", "testing", -1, 0.0, 1);
|
||||
TEST(ERROR, "testing %s... %d, %f, %u", "testing", -1, 0.0, 1);
|
||||
TEST(FATAL, "testing %s... %d, %f, %u", "testing", -1, 0.0, 1);
|
||||
TEST(NONE, "testing %s... %d, %f, %u", "testing", -1, 0.0, 1);
|
||||
|
||||
printf("1..%u\n", test_id);
|
||||
|
||||
fclose(observed_stream_w);
|
||||
fclose(observed_stream_r);
|
||||
close(fifo_fd_w);
|
||||
close(fifo_fd_r);
|
||||
unlink(observed_fifo);
|
||||
|
||||
return 0;
|
||||
}
|
Loading…
Reference in New Issue