| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-18 08:08:12 +11:00
										 |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup clog | 
					
						
							| 
									
										
										
										
											2018-03-31 12:26:37 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | #include <stdarg.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <stdint.h>
 | 
					
						
							|  |  |  | #include <assert.h>
 | 
					
						
							| 
									
										
										
										
											2019-02-21 11:31:00 +11:00
										 |  |  | #include <pthread.h>
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* For 'isatty' to check for color. */ | 
					
						
							| 
									
										
										
										
											2019-01-04 17:04:04 +11:00
										 |  |  | #if defined(__unix__) || defined(__APPLE__) || defined(__HAIKU__)
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | #  include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | #  include <sys/time.h>
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 16:24:06 +02:00
										 |  |  | #if defined(_MSC_VER)
 | 
					
						
							|  |  |  | #  include <io.h>
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | #  include <windows.h>
 | 
					
						
							| 
									
										
										
										
											2018-04-14 16:24:06 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* For printing timestamp. */ | 
					
						
							|  |  |  | #define __STDC_FORMAT_MACROS
 | 
					
						
							|  |  |  | #include <inttypes.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-21 11:31:00 +11:00
										 |  |  | #include "atomic_ops.h"
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 11:25:49 +02:00
										 |  |  | /* Only other dependency (could use regular malloc too). */ | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | #include "MEM_guardedalloc.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* own include. */ | 
					
						
							|  |  |  | #include "CLG_log.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Local utility defines */ | 
					
						
							|  |  |  | #define STREQ(a, b) (strcmp(a, b) == 0)
 | 
					
						
							|  |  |  | #define STREQLEN(a, b, n) (strncmp(a, b, n) == 0)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | #  define PATHSEP_CHAR '\\'
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #  define PATHSEP_CHAR '/'
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Internal Types
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct CLG_IDFilter { | 
					
						
							|  |  |  | 	struct CLG_IDFilter *next; | 
					
						
							|  |  |  | 	/** Over alloc. */ | 
					
						
							|  |  |  | 	char match[0]; | 
					
						
							|  |  |  | } CLG_IDFilter; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct CLogContext { | 
					
						
							|  |  |  | 	/** Single linked list of types.  */ | 
					
						
							|  |  |  | 	CLG_LogType *types; | 
					
						
							| 
									
										
										
										
											2019-02-21 11:31:00 +11:00
										 |  |  | 	pthread_mutex_t types_lock; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | 	/* exclude, include filters.  */ | 
					
						
							|  |  |  | 	CLG_IDFilter *filters[2]; | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	bool use_color; | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | 	bool use_basename; | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | 	bool use_timestamp; | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/** Borrowed, not owned. */ | 
					
						
							| 
									
										
										
										
											2018-04-14 13:59:33 +02:00
										 |  |  | 	int output; | 
					
						
							|  |  |  | 	FILE *output_file; | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | 	/** For timer (use_timestamp). */ | 
					
						
							|  |  |  | 	uint64_t timestamp_tick_start; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	/** For new types. */ | 
					
						
							|  |  |  | 	struct { | 
					
						
							|  |  |  | 		int level; | 
					
						
							|  |  |  | 	} default_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct { | 
					
						
							|  |  |  | 		void (*fatal_fn)(void *file_handle); | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | 		void (*backtrace_fn)(void *file_handle); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	} callbacks; | 
					
						
							|  |  |  | } CLogContext; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Mini Buffer Functionality
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Use so we can do a single call to write. | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define CLOG_BUF_LEN_INIT 512
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct CLogStringBuf { | 
					
						
							|  |  |  | 	char *data; | 
					
						
							|  |  |  | 	uint  len; | 
					
						
							|  |  |  | 	uint  len_alloc; | 
					
						
							|  |  |  | 	bool is_alloc; | 
					
						
							|  |  |  | } CLogStringBuf; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void clg_str_init(CLogStringBuf *cstr, char *buf_stack, uint buf_stack_len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	cstr->data = buf_stack; | 
					
						
							|  |  |  | 	cstr->len_alloc = buf_stack_len; | 
					
						
							|  |  |  | 	cstr->len = 0; | 
					
						
							|  |  |  | 	cstr->is_alloc = false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void clg_str_free(CLogStringBuf *cstr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (cstr->is_alloc) { | 
					
						
							|  |  |  | 		MEM_freeN(cstr->data); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void clg_str_reserve(CLogStringBuf *cstr, const uint len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (len > cstr->len_alloc) { | 
					
						
							|  |  |  | 		cstr->len_alloc *= 2; | 
					
						
							|  |  |  | 		if (len > cstr->len_alloc) { | 
					
						
							|  |  |  | 			cstr->len_alloc = len; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-03-31 11:25:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (cstr->is_alloc) { | 
					
						
							|  |  |  | 			cstr->data = MEM_reallocN(cstr->data, cstr->len_alloc); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			/* Copy the static buffer. */ | 
					
						
							|  |  |  | 			char *data = MEM_mallocN(cstr->len_alloc, __func__); | 
					
						
							|  |  |  | 			memcpy(data, cstr->data, cstr->len); | 
					
						
							|  |  |  | 			cstr->data = data; | 
					
						
							|  |  |  | 			cstr->is_alloc = true; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 		cstr->len_alloc = len; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void clg_str_append_with_len(CLogStringBuf *cstr, const char *str, const uint len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	uint len_next = cstr->len + len; | 
					
						
							|  |  |  | 	clg_str_reserve(cstr, len_next); | 
					
						
							|  |  |  | 	char *str_dst = cstr->data + cstr->len; | 
					
						
							|  |  |  | 	memcpy(str_dst, str, len); | 
					
						
							|  |  |  | #if 0 /* no need. */
 | 
					
						
							|  |  |  | 	str_dst[len] = '\0'; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	cstr->len = len_next; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void clg_str_append(CLogStringBuf *cstr, const char *str) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	clg_str_append_with_len(cstr, str, strlen(str)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void clg_str_vappendf(CLogStringBuf *cstr, const char *fmt, va_list args) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Use limit because windows may use '-1' for a formatting error. */ | 
					
						
							|  |  |  | 	const uint len_max = 65535; | 
					
						
							|  |  |  | 	uint len_avail = (cstr->len_alloc - cstr->len); | 
					
						
							|  |  |  | 	if (len_avail == 0) { | 
					
						
							|  |  |  | 		len_avail = CLOG_BUF_LEN_INIT; | 
					
						
							|  |  |  | 		clg_str_reserve(cstr, len_avail); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	while (true) { | 
					
						
							|  |  |  | 		va_list args_cpy; | 
					
						
							|  |  |  | 		va_copy(args_cpy, args); | 
					
						
							|  |  |  | 		int retval = vsnprintf(cstr->data + cstr->len, len_avail, fmt, args_cpy); | 
					
						
							|  |  |  | 		va_end(args_cpy); | 
					
						
							|  |  |  | 		if (retval != -1) { | 
					
						
							| 
									
										
										
										
											2018-03-31 15:43:47 +02:00
										 |  |  | 			cstr->len += retval; | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			len_avail *= 2; | 
					
						
							|  |  |  | 			if (len_avail >= len_max) { | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			clg_str_reserve(cstr, len_avail); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Internal Utilities
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | enum eCLogColor { | 
					
						
							|  |  |  | 	COLOR_DEFAULT, | 
					
						
							|  |  |  | 	COLOR_RED, | 
					
						
							|  |  |  | 	COLOR_GREEN, | 
					
						
							|  |  |  | 	COLOR_YELLOW, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	COLOR_RESET, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #define COLOR_LEN (COLOR_RESET + 1)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *clg_color_table[COLOR_LEN] = {NULL}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void clg_color_table_init(bool use_color) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	for (int i = 0; i < COLOR_LEN; i++) { | 
					
						
							|  |  |  | 		clg_color_table[i] = ""; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (use_color) { | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | 		/* TODO */ | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		clg_color_table[COLOR_DEFAULT]      = "\033[1;37m"; | 
					
						
							|  |  |  | 		clg_color_table[COLOR_RED]          = "\033[1;31m"; | 
					
						
							|  |  |  | 		clg_color_table[COLOR_GREEN]        = "\033[1;32m"; | 
					
						
							|  |  |  | 		clg_color_table[COLOR_YELLOW]       = "\033[1;33m"; | 
					
						
							|  |  |  | 		clg_color_table[COLOR_RESET]        = "\033[0m"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *clg_severity_str[CLG_SEVERITY_LEN] = { | 
					
						
							|  |  |  | 	[CLG_SEVERITY_INFO] =       "INFO", | 
					
						
							|  |  |  | 	[CLG_SEVERITY_WARN] =       "WARN", | 
					
						
							|  |  |  | 	[CLG_SEVERITY_ERROR] =      "ERROR", | 
					
						
							|  |  |  | 	[CLG_SEVERITY_FATAL] =      "FATAL", | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *clg_severity_as_text(enum CLG_Severity severity) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	bool ok = (unsigned int)severity < CLG_SEVERITY_LEN; | 
					
						
							|  |  |  | 	assert(ok); | 
					
						
							|  |  |  | 	if (ok) { | 
					
						
							|  |  |  | 		return clg_severity_str[severity]; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		return "INVALID_SEVERITY"; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static enum eCLogColor clg_severity_to_color(enum CLG_Severity severity) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-31 23:51:38 +02:00
										 |  |  | 	assert((unsigned int)severity < CLG_SEVERITY_LEN); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	enum eCLogColor color = COLOR_DEFAULT; | 
					
						
							|  |  |  | 	switch (severity) { | 
					
						
							|  |  |  | 		case CLG_SEVERITY_INFO: | 
					
						
							|  |  |  | 			color = COLOR_DEFAULT; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case CLG_SEVERITY_WARN: | 
					
						
							|  |  |  | 			color = COLOR_YELLOW; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		case CLG_SEVERITY_ERROR: | 
					
						
							|  |  |  | 		case CLG_SEVERITY_FATAL: | 
					
						
							|  |  |  | 			color = COLOR_RED; | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 		default: | 
					
						
							|  |  |  | 			/* should never get here. */ | 
					
						
							|  |  |  | 			assert(false); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return color; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Context Type Access
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Filter the indentifier based on very basic globbing. | 
					
						
							|  |  |  |  * - `foo` exact match of `foo`. | 
					
						
							|  |  |  |  * - `foo.bar` exact match for `foo.bar` | 
					
						
							|  |  |  |  * - `foo.*` match for `foo` & `foo.bar` & `foo.bar.baz` | 
					
						
							|  |  |  |  * - `*` matches everything. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static bool clg_ctx_filter_check(CLogContext *ctx, const char *identifier) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	const int identifier_len = strlen(identifier); | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | 	for (uint i = 0; i < 2; i++) { | 
					
						
							|  |  |  | 		const CLG_IDFilter *flt = ctx->filters[i]; | 
					
						
							|  |  |  | 		while (flt != NULL) { | 
					
						
							|  |  |  | 			const int len = strlen(flt->match); | 
					
						
							|  |  |  | 			if (STREQ(flt->match, "*") || | 
					
						
							|  |  |  | 				((len == identifier_len) && (STREQ(identifier, flt->match)))) | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 			{ | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | 				return (bool)i; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			if ((len >= 2) && (STREQLEN(".*", &flt->match[len - 2], 2))) { | 
					
						
							|  |  |  | 				if (((identifier_len == len - 2) && STREQLEN(identifier, flt->match, len - 2)) || | 
					
						
							|  |  |  | 					((identifier_len >= len - 1) && STREQLEN(identifier, flt->match, len - 1))) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					return (bool)i; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | 			flt = flt->next; | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * \note This should never be called per logging call. | 
					
						
							|  |  |  |  * Searching is only to get an initial handle. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static CLG_LogType *clg_ctx_type_find_by_name(CLogContext *ctx, const char *identifier) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	for (CLG_LogType *ty = ctx->types; ty; ty = ty->next) { | 
					
						
							|  |  |  | 		if (STREQ(identifier, ty->identifier)) { | 
					
						
							|  |  |  | 			return ty; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static CLG_LogType *clg_ctx_type_register(CLogContext *ctx, const char *identifier) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	assert(clg_ctx_type_find_by_name(ctx, identifier) == NULL); | 
					
						
							|  |  |  | 	CLG_LogType *ty = MEM_callocN(sizeof(*ty), __func__); | 
					
						
							|  |  |  | 	ty->next = ctx->types; | 
					
						
							|  |  |  | 	ctx->types = ty; | 
					
						
							|  |  |  | 	strncpy(ty->identifier, identifier, sizeof(ty->identifier) - 1); | 
					
						
							|  |  |  | 	ty->ctx = ctx; | 
					
						
							|  |  |  | 	ty->level = ctx->default_type.level; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (clg_ctx_filter_check(ctx, ty->identifier)) { | 
					
						
							|  |  |  | 		ty->flag |= CLG_FLAG_USE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return ty; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | static void clg_ctx_fatal_action(CLogContext *ctx) | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	if (ctx->callbacks.fatal_fn != NULL) { | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | 		ctx->callbacks.fatal_fn(ctx->output_file); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | 	fflush(ctx->output_file); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	abort(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | static void clg_ctx_backtrace(CLogContext *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	/* Note: we avoid writing fo 'FILE', for backtrace we make an exception,
 | 
					
						
							|  |  |  | 	 * if necessary we could have a version of the callback that writes to file descriptor all at once. */ | 
					
						
							|  |  |  | 	ctx->callbacks.backtrace_fn(ctx->output_file); | 
					
						
							|  |  |  | 	fflush(ctx->output_file); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | static uint64_t clg_timestamp_ticks_get(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	uint64_t tick; | 
					
						
							|  |  |  | #if defined(_MSC_VER)
 | 
					
						
							|  |  |  | 	tick = GetTickCount64(); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	struct timeval tv; | 
					
						
							|  |  |  | 	gettimeofday(&tv, NULL); | 
					
						
							|  |  |  | 	tick = tv.tv_sec * 1000 + tv.tv_usec / 1000; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	return tick; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Logging API
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | static void write_timestamp(CLogStringBuf *cstr, const uint64_t timestamp_tick_start) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	char timestamp_str[64]; | 
					
						
							|  |  |  | 	const uint64_t timestamp = clg_timestamp_ticks_get() - timestamp_tick_start; | 
					
						
							|  |  |  | 	const uint timestamp_len = snprintf( | 
					
						
							|  |  |  | 	        timestamp_str, sizeof(timestamp_str), "%" PRIu64 ".%03u ", | 
					
						
							|  |  |  | 	        timestamp / 1000, (uint)(timestamp % 1000)); | 
					
						
							|  |  |  | 	clg_str_append_with_len(cstr, timestamp_str, timestamp_len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | static void write_severity(CLogStringBuf *cstr, enum CLG_Severity severity, bool use_color) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	assert((unsigned int)severity < CLG_SEVERITY_LEN); | 
					
						
							|  |  |  | 	if (use_color) { | 
					
						
							|  |  |  | 		enum eCLogColor color = clg_severity_to_color(severity); | 
					
						
							|  |  |  | 		clg_str_append(cstr, clg_color_table[color]); | 
					
						
							|  |  |  | 		clg_str_append(cstr, clg_severity_as_text(severity)); | 
					
						
							|  |  |  | 		clg_str_append(cstr, clg_color_table[COLOR_RESET]); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							|  |  |  | 		clg_str_append(cstr, clg_severity_as_text(severity)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void write_type(CLogStringBuf *cstr, CLG_LogType *lg) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	clg_str_append(cstr, " ("); | 
					
						
							|  |  |  | 	clg_str_append(cstr, lg->identifier); | 
					
						
							|  |  |  | 	clg_str_append(cstr, "): "); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | static void write_file_line_fn(CLogStringBuf *cstr, const char *file_line, const char *fn, const bool use_basename) | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | 	uint file_line_len = strlen(file_line); | 
					
						
							|  |  |  | 	if (use_basename) { | 
					
						
							|  |  |  | 		uint file_line_offset = file_line_len; | 
					
						
							|  |  |  | 		while (file_line_offset-- > 0) { | 
					
						
							|  |  |  | 			if (file_line[file_line_offset] == PATHSEP_CHAR) { | 
					
						
							|  |  |  | 				file_line_offset++; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		file_line += file_line_offset; | 
					
						
							|  |  |  | 		file_line_len -= file_line_offset; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	clg_str_append_with_len(cstr, file_line, file_line_len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	clg_str_append(cstr, " "); | 
					
						
							|  |  |  | 	clg_str_append(cstr, fn); | 
					
						
							|  |  |  | 	clg_str_append(cstr, ": "); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CLG_log_str( | 
					
						
							|  |  |  |         CLG_LogType *lg, enum CLG_Severity severity, const char *file_line, const char *fn, | 
					
						
							|  |  |  |         const char *message) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLogStringBuf cstr; | 
					
						
							|  |  |  | 	char cstr_stack_buf[CLOG_BUF_LEN_INIT]; | 
					
						
							|  |  |  | 	clg_str_init(&cstr, cstr_stack_buf, sizeof(cstr_stack_buf)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | 	if (lg->ctx->use_timestamp) { | 
					
						
							|  |  |  | 		write_timestamp(&cstr, lg->ctx->timestamp_tick_start); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	write_severity(&cstr, severity, lg->ctx->use_color); | 
					
						
							|  |  |  | 	write_type(&cstr, lg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | 		write_file_line_fn(&cstr, file_line, fn, lg->ctx->use_basename); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 		clg_str_append(&cstr, message); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	clg_str_append(&cstr, "\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* could be optional */ | 
					
						
							| 
									
										
										
										
											2018-05-13 15:48:55 +02:00
										 |  |  | 	int bytes_written = write(lg->ctx->output, cstr.data, cstr.len); | 
					
						
							| 
									
										
										
										
											2018-05-13 14:10:05 +02:00
										 |  |  | 	(void)bytes_written; | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	clg_str_free(&cstr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | 	if (lg->ctx->callbacks.backtrace_fn) { | 
					
						
							|  |  |  | 		clg_ctx_backtrace(lg->ctx); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	if (severity == CLG_SEVERITY_FATAL) { | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | 		clg_ctx_fatal_action(lg->ctx); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CLG_logf( | 
					
						
							|  |  |  |         CLG_LogType *lg, enum CLG_Severity severity, const char *file_line, const char *fn, | 
					
						
							|  |  |  |         const char *fmt, ...) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLogStringBuf cstr; | 
					
						
							|  |  |  | 	char cstr_stack_buf[CLOG_BUF_LEN_INIT]; | 
					
						
							|  |  |  | 	clg_str_init(&cstr, cstr_stack_buf, sizeof(cstr_stack_buf)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | 	if (lg->ctx->use_timestamp) { | 
					
						
							|  |  |  | 		write_timestamp(&cstr, lg->ctx->timestamp_tick_start); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	write_severity(&cstr, severity, lg->ctx->use_color); | 
					
						
							|  |  |  | 	write_type(&cstr, lg); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | 		write_file_line_fn(&cstr, file_line, fn, lg->ctx->use_basename); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		va_list ap; | 
					
						
							|  |  |  | 		va_start(ap, fmt); | 
					
						
							|  |  |  | 		clg_str_vappendf(&cstr, fmt, ap); | 
					
						
							|  |  |  | 		va_end(ap); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	clg_str_append(&cstr, "\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* could be optional */ | 
					
						
							| 
									
										
										
										
											2018-05-13 15:48:55 +02:00
										 |  |  | 	int bytes_written = write(lg->ctx->output, cstr.data, cstr.len); | 
					
						
							| 
									
										
										
										
											2018-05-13 14:10:05 +02:00
										 |  |  | 	(void)bytes_written; | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 11:25:49 +02:00
										 |  |  | 	clg_str_free(&cstr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | 	if (lg->ctx->callbacks.backtrace_fn) { | 
					
						
							|  |  |  | 		clg_ctx_backtrace(lg->ctx); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	if (severity == CLG_SEVERITY_FATAL) { | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | 		clg_ctx_fatal_action(lg->ctx); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Logging Context API
 | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void CLG_ctx_output_set(CLogContext *ctx, void *file_handle) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2018-04-14 13:59:33 +02:00
										 |  |  | 	ctx->output_file = file_handle; | 
					
						
							| 
									
										
										
										
											2018-07-31 20:52:36 +10:00
										 |  |  | 	ctx->output = fileno(ctx->output_file); | 
					
						
							| 
									
										
										
										
											2018-04-16 07:38:11 +02:00
										 |  |  | #if defined(__unix__) || defined(__APPLE__)
 | 
					
						
							| 
									
										
										
										
											2018-04-14 13:59:33 +02:00
										 |  |  | 	ctx->use_color = isatty(ctx->output); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 15:33:15 +02:00
										 |  |  | static void CLG_ctx_output_use_basename_set(CLogContext *ctx, int value) | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-31 15:33:15 +02:00
										 |  |  | 	ctx->use_basename = (bool)value; | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | static void CLG_ctx_output_use_timestamp_set(CLogContext *ctx, int value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ctx->use_timestamp = (bool)value; | 
					
						
							|  |  |  | 	if (ctx->use_timestamp) { | 
					
						
							|  |  |  | 		ctx->timestamp_tick_start = clg_timestamp_ticks_get(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | /** Action on fatal severity. */ | 
					
						
							|  |  |  | static void CLG_ctx_fatal_fn_set(CLogContext *ctx, void (*fatal_fn)(void *file_handle)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ctx->callbacks.fatal_fn = fatal_fn; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | static void CLG_ctx_backtrace_fn_set(CLogContext *ctx, void (*backtrace_fn)(void *file_handle)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ctx->callbacks.backtrace_fn = backtrace_fn; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | static void clg_ctx_type_filter_append(CLG_IDFilter **flt_list, const char *type_match, int type_match_len) | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | 	if (type_match_len == 0) { | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	CLG_IDFilter *flt = MEM_callocN(sizeof(*flt) + (type_match_len + 1), __func__); | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | 	flt->next = *flt_list; | 
					
						
							|  |  |  | 	*flt_list = flt; | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	memcpy(flt->match, type_match, type_match_len); | 
					
						
							|  |  |  | 	/* no need to null terminate since we calloc'd */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | static void CLG_ctx_type_filter_exclude(CLogContext *ctx, const char *type_match, int type_match_len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	clg_ctx_type_filter_append(&ctx->filters[0], type_match, type_match_len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void CLG_ctx_type_filter_include(CLogContext *ctx, const char *type_match, int type_match_len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	clg_ctx_type_filter_append(&ctx->filters[1], type_match, type_match_len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-04 08:00:22 +02:00
										 |  |  | static void CLG_ctx_level_set(CLogContext *ctx, int level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ctx->default_type.level = level; | 
					
						
							|  |  |  | 	for (CLG_LogType *ty = ctx->types; ty; ty = ty->next) { | 
					
						
							|  |  |  | 		ty->level = level; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | static CLogContext *CLG_ctx_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLogContext *ctx = MEM_callocN(sizeof(*ctx), __func__); | 
					
						
							| 
									
										
										
										
											2019-02-21 11:31:00 +11:00
										 |  |  | 	pthread_mutex_init(&ctx->types_lock, NULL); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	ctx->use_color = true; | 
					
						
							|  |  |  | 	ctx->default_type.level = 1; | 
					
						
							|  |  |  | 	CLG_ctx_output_set(ctx, stdout); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ctx; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void CLG_ctx_free(CLogContext *ctx) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	while (ctx->types != NULL) { | 
					
						
							|  |  |  | 		CLG_LogType *item = ctx->types; | 
					
						
							|  |  |  | 		ctx->types = item->next; | 
					
						
							|  |  |  | 		MEM_freeN(item); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for (uint i = 0; i < 2; i++) { | 
					
						
							|  |  |  | 		while (ctx->filters[i] != NULL) { | 
					
						
							|  |  |  | 			CLG_IDFilter *item = ctx->filters[i]; | 
					
						
							|  |  |  | 			ctx->filters[i] = item->next; | 
					
						
							|  |  |  | 			MEM_freeN(item); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-02-21 11:31:00 +11:00
										 |  |  | 	pthread_mutex_destroy(&ctx->types_lock); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 	MEM_freeN(ctx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Public Logging API
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Currently uses global context. | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* We could support multiple at once, for now this seems not needed. */ | 
					
						
							| 
									
										
										
										
											2018-08-08 11:00:57 +10:00
										 |  |  | static struct CLogContext *g_ctx = NULL; | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | void CLG_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	g_ctx = CLG_ctx_init(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	clg_color_table_init(g_ctx->use_color); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CLG_exit(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLG_ctx_free(g_ctx); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CLG_output_set(void *file_handle) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLG_ctx_output_set(g_ctx, file_handle); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 15:33:15 +02:00
										 |  |  | void CLG_output_use_basename_set(int value) | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 	CLG_ctx_output_use_basename_set(g_ctx, value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-16 16:29:40 +11:00
										 |  |  | void CLG_output_use_timestamp_set(int value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLG_ctx_output_use_timestamp_set(g_ctx, value); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-03-31 15:27:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | void CLG_fatal_fn_set(void (*fatal_fn)(void *file_handle)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLG_ctx_fatal_fn_set(g_ctx, fatal_fn); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-18 11:00:47 +02:00
										 |  |  | void CLG_backtrace_fn_set(void (*fatal_fn)(void *file_handle)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLG_ctx_backtrace_fn_set(g_ctx, fatal_fn); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | void CLG_type_filter_exclude(const char *type_match, int type_match_len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLG_ctx_type_filter_exclude(g_ctx, type_match, type_match_len); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CLG_type_filter_include(const char *type_match, int type_match_len) | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-03-31 12:52:47 +02:00
										 |  |  | 	CLG_ctx_type_filter_include(g_ctx, type_match, type_match_len); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-04 08:00:22 +02:00
										 |  |  | void CLG_level_set(int level) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CLG_ctx_level_set(g_ctx, level); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | /** \} */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* -------------------------------------------------------------------- */ | 
					
						
							|  |  |  | /** \name Logging Reference API
 | 
					
						
							|  |  |  |  * Use to avoid lookups each time. | 
					
						
							|  |  |  |  * \{ */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CLG_logref_init(CLG_LogRef *clg_ref) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-02-21 11:31:00 +11:00
										 |  |  | 	/* Only runs once when initializing a static type in most cases. */ | 
					
						
							|  |  |  | 	pthread_mutex_lock(&g_ctx->types_lock); | 
					
						
							|  |  |  | 	if (clg_ref->type == NULL) { | 
					
						
							|  |  |  | 		CLG_LogType *clg_ty = clg_ctx_type_find_by_name(g_ctx, clg_ref->identifier); | 
					
						
							|  |  |  | 		if (clg_ty == NULL) { | 
					
						
							|  |  |  | 			clg_ty = clg_ctx_type_register(g_ctx, clg_ref->identifier); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		atomic_cas_ptr((void **)&clg_ref->type, clg_ref->type, clg_ty); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	pthread_mutex_unlock(&g_ctx->types_lock); | 
					
						
							| 
									
										
										
										
											2018-03-29 20:38:32 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \} */ |