1 /* Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
2  * Use of this source code is governed by a BSD-style license that can be
3  * found in the LICENSE file.
4  */
5 
6 #include <stdarg.h>
7 #include <stdlib.h>
8 #include <string.h>
9 #include <syslog.h>
10 
11 #include "dumper.h"
12 
dumpf(struct dumper * dumper,const char * format,...)13 void dumpf(struct dumper *dumper, const char *format, ...)
14 {
15 	va_list ap;
16 	va_start(ap, format);
17 	dumper->vprintf(dumper, format, ap);
18 	va_end(ap);
19 }
20 
21 /* dumper which outputs to syslog */
22 
23 struct syslog_data {
24 	int priority;
25 	struct dumper *mem_dumper;
26 };
27 
syslog_vprintf(struct dumper * dumper,const char * fmt,va_list ap)28 static void syslog_vprintf(struct dumper *dumper, const char *fmt, va_list ap)
29 {
30 	char *buf;
31 	int size, i;
32 	struct syslog_data *data = (struct syslog_data *)dumper->data;
33 	struct dumper *mem_dumper = data->mem_dumper;
34 
35 	/* We cannot use syslog() directly each time we are called,
36 	 * because syslog() will always append a newline to the
37 	 * output, so the user will not be able to construct a line
38 	 * incrementally using multiple calls. What we do here is to
39 	 * collect the output in a buffer until a newline is given by
40 	 * the user. */
41 
42 	mem_dumper->vprintf(mem_dumper, fmt, ap);
43 again:
44 	mem_dumper_get(mem_dumper, &buf, &size);
45 	for (i = 0; i < size; i++) {
46 		if (buf[i] == '\n') {
47 			syslog(data->priority, "%.*s", i + 1, buf);
48 			mem_dumper_consume(mem_dumper, i + 1);
49 			goto again;
50 		}
51 	}
52 }
53 
syslog_dumper_create(int priority)54 struct dumper *syslog_dumper_create(int priority)
55 {
56 	struct dumper *dumper = calloc(1, sizeof(struct dumper));
57 	struct syslog_data *data = calloc(1, sizeof(struct syslog_data));
58 	data->priority = priority;
59 	data->mem_dumper = mem_dumper_create();
60 	dumper->data = data;
61 	dumper->vprintf = &syslog_vprintf;
62 	return dumper;
63 }
64 
syslog_dumper_free(struct dumper * dumper)65 void syslog_dumper_free(struct dumper *dumper)
66 {
67 	mem_dumper_free(((struct syslog_data *)dumper->data)->mem_dumper);
68 	free(dumper->data);
69 	free(dumper);
70 }
71 
72 /* dumper which outputs to a memory buffer */
73 
74 struct mem_data {
75 	char *buf;
76 	int size;
77 	int capacity;
78 };
79 
mem_vprintf(struct dumper * dumper,const char * format,va_list ap)80 static void mem_vprintf(struct dumper *dumper, const char *format, va_list ap)
81 {
82 	int n;
83 	char *tmp;
84 	struct mem_data *data = (struct mem_data *)dumper->data;
85 
86 	while (1) {
87 		/* try to use the remaining space */
88 		int remaining = data->capacity - data->size;
89 		n = vsnprintf(data->buf + data->size, remaining, format, ap);
90 
91 		/* enough space? */
92 		if (n > -1 && n < remaining) {
93 			data->size += n;
94 			return;
95 		}
96 
97 		/* allocate more space and try again */
98 		tmp = realloc(data->buf, data->capacity * 2);
99 		if (tmp == NULL)
100 			return;
101 		data->buf = tmp;
102 		data->capacity *= 2;
103 	}
104 }
105 
mem_dumper_create()106 struct dumper *mem_dumper_create()
107 {
108 	struct dumper *dumper = calloc(1, sizeof(struct dumper));
109 	struct mem_data *data = calloc(1, sizeof(struct mem_data));
110 	if (!dumper || !data)
111 		goto error;
112 	data->size = 0;
113 	data->capacity = 80;
114 	data->buf = malloc(data->capacity);
115 	if (!data->buf)
116 		goto error;
117 	data->buf[0] = '\0';
118 	dumper->data = data;
119 	dumper->vprintf = &mem_vprintf;
120 	return dumper;
121 
122 error:
123 	if (dumper)
124 		free(dumper);
125 	if (data)
126 		free(data);
127 	return NULL;
128 }
129 
mem_dumper_free(struct dumper * dumper)130 void mem_dumper_free(struct dumper *dumper)
131 {
132 	struct mem_data *data = (struct mem_data *)dumper->data;
133 	free(data->buf);
134 	free(data);
135 	free(dumper);
136 }
137 
mem_dumper_clear(struct dumper * dumper)138 void mem_dumper_clear(struct dumper *dumper)
139 {
140 	struct mem_data *data = (struct mem_data *)dumper->data;
141 	data->buf[0] = '\0';
142 	data->size = 0;
143 }
144 
mem_dumper_consume(struct dumper * dumper,int n)145 void mem_dumper_consume(struct dumper *dumper, int n)
146 {
147 	struct mem_data *data = (struct mem_data *)dumper->data;
148 	if (n > data->size)
149 		n = data->size;
150 	memmove(data->buf, data->buf + n, data->size - n + 1);
151 	data->size -= n;
152 }
153 
mem_dumper_get(struct dumper * dumper,char ** buf,int * size)154 void mem_dumper_get(struct dumper *dumper, char **buf, int *size)
155 {
156 	struct mem_data *data = (struct mem_data *)dumper->data;
157 	*buf = data->buf;
158 	*size = data->size;
159 }
160