File: log.c

package info (click to toggle)
drgn 0.1.0-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 7,852 kB
  • sloc: python: 74,992; ansic: 54,589; awk: 423; makefile: 351; sh: 99
file content (119 lines) | stat: -rw-r--r-- 2,865 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
// Copyright (c) Meta Platforms, Inc. and affiliates.
// SPDX-License-Identifier: LGPL-2.1-or-later

#include <stdarg.h>
#include <stdio.h>
#include <sys/ioctl.h>

#include "log.h"
#include "program.h"
#include "util.h"

LIBDRGN_PUBLIC void drgn_program_set_log_level(struct drgn_program *prog,
					       int level)
{
	prog->log_level = level;
}

LIBDRGN_PUBLIC int drgn_program_get_log_level(struct drgn_program *prog)
{
	return prog->log_level;
}

static void drgn_file_log_fn(struct drgn_program *prog, void *arg,
			     enum drgn_log_level level, const char *format,
			     va_list ap, struct drgn_error *err)
{
	FILE *file = arg;
	flockfile(file);

	static const char * const prefix[] = {
		[DRGN_LOG_DEBUG] = "debug: ",
		[DRGN_LOG_INFO] = "info: ",
		[DRGN_LOG_WARNING] = "warning: ",
		[DRGN_LOG_ERROR] = "error: ",
		[DRGN_LOG_CRITICAL] = "critical: ",
	};
	fputs(prefix[level], file);
	vfprintf(file, format, ap);
	if (err)
		drgn_error_fwrite(file, err);
	else
		putc('\n', file);

	funlockfile(file);
}

LIBDRGN_PUBLIC void drgn_program_set_log_file(struct drgn_program *prog,
					      FILE *file)
{
	drgn_program_set_log_callback(prog, drgn_file_log_fn, file);
}

LIBDRGN_PUBLIC void drgn_program_set_log_callback(struct drgn_program *prog,
						  drgn_log_fn *callback,
						  void *callback_arg)
{
	prog->log_fn = callback;
	prog->log_arg = callback_arg;
}

LIBDRGN_PUBLIC void drgn_program_get_log_callback(struct drgn_program *prog,
						  drgn_log_fn **callback_ret,
						  void **callback_arg_ret)
{
	*callback_ret = prog->log_fn;
	*callback_arg_ret = prog->log_arg;
}

bool drgn_log_is_enabled(struct drgn_program *prog, enum drgn_log_level level)
{
	return level >= prog->log_level;
}

void drgn_error_log(enum drgn_log_level level, struct drgn_program *prog,
		    struct drgn_error *err, const char *format, ...)
{
	if (!drgn_log_is_enabled(prog, level))
		return;

	va_list ap;
	va_start(ap, format);
	prog->log_fn(prog, prog->log_arg, level, format, ap, err);
	va_end(ap);
}

LIBDRGN_PUBLIC void drgn_program_set_progress_file(struct drgn_program *prog,
						   FILE *file)
{
	prog->progress_file = file;
	prog->default_progress_file = false;
}

FILE *drgn_program_get_progress_file(struct drgn_program *prog,
				     int *columns_ret)
{
	*columns_ret = -1;

	if (!prog->default_progress_file) {
		if (prog->progress_file) {
			int fd = fileno(prog->progress_file);
			struct winsize winsize;
			if (fd >= 0 && ioctl(fd, TIOCGWINSZ, &winsize) == 0)
				*columns_ret = winsize.ws_col;
		}
		return prog->progress_file;
	}

	if (drgn_log_is_enabled(prog, DRGN_LOG_WARNING)
	    && prog->log_fn == drgn_file_log_fn && prog->log_arg == stderr) {
		int fd = fileno(stderr);
		struct winsize winsize;
		if (fd >= 0 && ioctl(fd, TIOCGWINSZ, &winsize) == 0) {
			*columns_ret = winsize.ws_col;
			return stderr;
		}
	}

	return NULL;
}