Contiki-NG
Loading...
Searching...
No Matches
log.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 2017, Inria.
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. Neither the name of the Institute nor the names of its contributors
14 * may be used to endorse or promote products derived from this software
15 * without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 * SUCH DAMAGE.
28 *
29 * This file is part of the Contiki operating system.
30 *
31 */
32
33/**
34 * \file
35 * Header file for the logging system
36 * \author
37 * Simon Duquennoy <simon.duquennoy@inria.fr>
38 */
39
40/** \addtogroup sys
41 * @{ */
42
43/**
44 * \defgroup log Per-module, per-level logging
45 * @{
46 *
47 * The log module performs per-module, per-level logging
48 *
49 */
50
51#ifndef LOG_H_
52#define LOG_H_
53
54#include <stdio.h>
55#include "net/linkaddr.h"
56#include "sys/log-conf.h"
57#if NETSTACK_CONF_WITH_IPV6
58#include "net/ipv6/uip.h"
59#endif /* NETSTACK_CONF_WITH_IPV6 */
60
61/* The different log levels available */
62#define LOG_LEVEL_NONE 0 /* No log */
63#define LOG_LEVEL_ERR 1 /* Errors */
64#define LOG_LEVEL_WARN 2 /* Warnings */
65#define LOG_LEVEL_INFO 3 /* Basic info */
66#define LOG_LEVEL_DBG 4 /* Detailled debug */
67
68/* Log coloring */
69#define TC_RESET "\033[0m"
70#define TC_BLACK "\033[0;30m"
71#define TC_RED "\033[0;31m"
72#define TC_GREEN "\033[0;32m"
73#define TC_YELLOW "\033[0;33m"
74#define TC_BLUE "\033[0;34m"
75#define TC_MAGENTA "\033[0;35m"
76#define TC_CYAN "\033[0;36m"
77#define TC_WHITE "\033[0;37m"
78
79#define LOG_COLOR_RESET TC_RESET
80
81#ifndef LOG_COLOR_ERR
82#define LOG_COLOR_ERR TC_RED
83#endif
84
85#ifndef LOG_COLOR_WARN
86#define LOG_COLOR_WARN TC_YELLOW
87#endif
88
89#ifndef LOG_COLOR_INFO
90#define LOG_COLOR_INFO TC_BLUE
91#endif
92
93#ifndef LOG_COLOR_DBG
94#define LOG_COLOR_DBG TC_WHITE
95#endif
96
97#ifndef LOG_COLOR_PRI
98#define LOG_COLOR_PRI TC_CYAN
99#endif
100
101/* Per-module log level */
102
103struct log_module {
104 const char *name;
105 int *curr_log_level;
106 int max_log_level;
107};
108
109extern int curr_log_level_rpl;
110extern int curr_log_level_tcpip;
111extern int curr_log_level_ipv6;
112extern int curr_log_level_6lowpan;
113extern int curr_log_level_nullnet;
114extern int curr_log_level_mac;
115extern int curr_log_level_framer;
116extern int curr_log_level_6top;
117extern int curr_log_level_coap;
118extern int curr_log_level_dtls;
119extern int curr_log_level_snmp;
120extern int curr_log_level_lwm2m;
121extern int curr_log_level_sys;
122extern int curr_log_level_main;
123
124extern struct log_module all_modules[];
125
126#define LOG_LEVEL_RPL MIN((LOG_CONF_LEVEL_RPL), curr_log_level_rpl)
127#define LOG_LEVEL_TCPIP MIN((LOG_CONF_LEVEL_TCPIP), curr_log_level_tcpip)
128#define LOG_LEVEL_IPV6 MIN((LOG_CONF_LEVEL_IPV6), curr_log_level_ipv6)
129#define LOG_LEVEL_6LOWPAN MIN((LOG_CONF_LEVEL_6LOWPAN), curr_log_level_6lowpan)
130#define LOG_LEVEL_NULLNET MIN((LOG_CONF_LEVEL_NULLNET), curr_log_level_nullnet)
131#define LOG_LEVEL_MAC MIN((LOG_CONF_LEVEL_MAC), curr_log_level_mac)
132#define LOG_LEVEL_FRAMER MIN((LOG_CONF_LEVEL_FRAMER), curr_log_level_framer)
133#define LOG_LEVEL_6TOP MIN((LOG_CONF_LEVEL_6TOP), curr_log_level_6top)
134#define LOG_LEVEL_COAP MIN((LOG_CONF_LEVEL_COAP), curr_log_level_coap)
135#define LOG_LEVEL_DTLS MIN((LOG_CONF_LEVEL_DTLS), curr_log_level_dtls)
136#define LOG_LEVEL_SNMP MIN((LOG_CONF_LEVEL_SNMP), curr_log_level_snmp)
137#define LOG_LEVEL_LWM2M MIN((LOG_CONF_LEVEL_LWM2M), curr_log_level_lwm2m)
138#define LOG_LEVEL_SYS MIN((LOG_CONF_LEVEL_SYS), curr_log_level_sys)
139#define LOG_LEVEL_MAIN MIN((LOG_CONF_LEVEL_MAIN), curr_log_level_main)
140
141/* Main log function */
142
143#define LOG(newline, level, levelstr, levelcolor, ...) do { \
144 if(level <= (LOG_LEVEL)) { \
145 if(newline) { \
146 if(LOG_WITH_COLOR) { \
147 LOG_OUTPUT(levelcolor); \
148 } \
149 if(LOG_WITH_MODULE_PREFIX) { \
150 LOG_OUTPUT_PREFIX(level, levelstr, LOG_MODULE); \
151 } \
152 if(LOG_WITH_LOC) { \
153 LOG_OUTPUT("[%s: %d] ", __FILE__, __LINE__); \
154 } \
155 if(LOG_WITH_COLOR) { \
156 LOG_OUTPUT(LOG_COLOR_RESET); \
157 } \
158 } \
159 LOG_OUTPUT(__VA_ARGS__); \
160 } \
161 } while (0)
162
163/* For Cooja annotations */
164#define LOG_ANNOTATE(...) do { \
165 if(LOG_WITH_ANNOTATE) { \
166 LOG_OUTPUT(__VA_ARGS__); \
167 } \
168 } while (0)
169
170/* Link-layer address */
171#define LOG_LLADDR(level, lladdr) do { \
172 if(level <= (LOG_LEVEL)) { \
173 if(LOG_WITH_COMPACT_ADDR) { \
174 log_lladdr_compact(lladdr); \
175 } else { \
176 log_lladdr(lladdr); \
177 } \
178 } \
179 } while (0)
180
181/* IPv6 address */
182#define LOG_6ADDR(level, ipaddr) do { \
183 if(level <= (LOG_LEVEL)) { \
184 if(LOG_WITH_COMPACT_ADDR) { \
185 log_6addr_compact(ipaddr); \
186 } else { \
187 log_6addr(ipaddr); \
188 } \
189 } \
190 } while (0)
191
192#define LOG_BYTES(level, data, length) do { \
193 if(level <= (LOG_LEVEL)) { \
194 log_bytes(data, length); \
195 } \
196 } while (0)
197
198/* More compact versions of LOG macros */
199#define LOG_PRINT(...) LOG(1, 0, "PRI", LOG_COLOR_PRI, __VA_ARGS__)
200#define LOG_ERR(...) LOG(1, LOG_LEVEL_ERR, "ERR", LOG_COLOR_ERR, __VA_ARGS__)
201#define LOG_WARN(...) LOG(1, LOG_LEVEL_WARN, "WARN", LOG_COLOR_WARN, __VA_ARGS__)
202#define LOG_INFO(...) LOG(1, LOG_LEVEL_INFO, "INFO", LOG_COLOR_INFO, __VA_ARGS__)
203#define LOG_DBG(...) LOG(1, LOG_LEVEL_DBG, "DBG", LOG_COLOR_DBG, __VA_ARGS__)
204
205#define LOG_PRINT_(...) LOG(0, 0, "PRI", LOG_COLOR_PRI, __VA_ARGS__)
206#define LOG_ERR_(...) LOG(0, LOG_LEVEL_ERR, "ERR", LOG_COLOR_ERR, __VA_ARGS__)
207#define LOG_WARN_(...) LOG(0, LOG_LEVEL_WARN, "WARN", LOG_COLOR_WARN, __VA_ARGS__)
208#define LOG_INFO_(...) LOG(0, LOG_LEVEL_INFO, "INFO", LOG_COLOR_INFO, __VA_ARGS__)
209#define LOG_DBG_(...) LOG(0, LOG_LEVEL_DBG, "DBG", LOG_COLOR_DBG, __VA_ARGS__)
210
211#define LOG_PRINT_LLADDR(...) LOG_LLADDR(0, __VA_ARGS__)
212#define LOG_ERR_LLADDR(...) LOG_LLADDR(LOG_LEVEL_ERR, __VA_ARGS__)
213#define LOG_WARN_LLADDR(...) LOG_LLADDR(LOG_LEVEL_WARN, __VA_ARGS__)
214#define LOG_INFO_LLADDR(...) LOG_LLADDR(LOG_LEVEL_INFO, __VA_ARGS__)
215#define LOG_DBG_LLADDR(...) LOG_LLADDR(LOG_LEVEL_DBG, __VA_ARGS__)
216
217#define LOG_PRINT_6ADDR(...) LOG_6ADDR(0, __VA_ARGS__)
218#define LOG_ERR_6ADDR(...) LOG_6ADDR(LOG_LEVEL_ERR, __VA_ARGS__)
219#define LOG_WARN_6ADDR(...) LOG_6ADDR(LOG_LEVEL_WARN, __VA_ARGS__)
220#define LOG_INFO_6ADDR(...) LOG_6ADDR(LOG_LEVEL_INFO, __VA_ARGS__)
221#define LOG_DBG_6ADDR(...) LOG_6ADDR(LOG_LEVEL_DBG, __VA_ARGS__)
222
223#define LOG_PRINT_BYTES(data, length) LOG_BYTES(0, data, length)
224#define LOG_ERR_BYTES(data, length) LOG_BYTES(LOG_LEVEL_ERR, data, length)
225#define LOG_WARN_BYTES(data, length) LOG_BYTES(LOG_LEVEL_WARN, data, length)
226#define LOG_INFO_BYTES(data, length) LOG_BYTES(LOG_LEVEL_INFO, data, length)
227#define LOG_DBG_BYTES(data, length) LOG_BYTES(LOG_LEVEL_DBG, data, length)
228
229/* For checking log level.
230 As this builds on curr_log_level variables, this should not be used
231 in pre-processor macros. Use in a C 'if' statement instead, e.g.:
232 if(LOG_INFO_ENABLED) { ... }
233 Note that most compilers will still be able to strip the code out
234 for low enough log levels configurations. */
235#define LOG_ERR_ENABLED ((LOG_LEVEL) >= LOG_LEVEL_ERR)
236#define LOG_WARN_ENABLED ((LOG_LEVEL) >= LOG_LEVEL_WARN)
237#define LOG_INFO_ENABLED ((LOG_LEVEL) >= LOG_LEVEL_INFO)
238#define LOG_DBG_ENABLED ((LOG_LEVEL) >= LOG_LEVEL_DBG)
239
240#if NETSTACK_CONF_WITH_IPV6
241
242/**
243 * Logs an IPv6 address
244 * \param ipaddr The IPv6 address
245*/
246void log_6addr(const uip_ipaddr_t *ipaddr);
247
248/**
249 * Logs an IPv6 address with a compact format
250 * \param ipaddr The IPv6 address
251*/
252void log_6addr_compact(const uip_ipaddr_t *ipaddr);
253
254/**
255 * Write at most size - 1 characters of the IP address to the output string,
256 * in a compact representation. The output is always null-terminated, unless
257 * size is 0.
258 *
259 * \param buf A pointer to an output string with at least size bytes.
260 * \param size The max number of characters to write to the output string.
261 * \param ipaddr A pointer to a uip_ipaddr_t that will be printed with printf().
262 */
263int log_6addr_compact_snprint(char *buf, size_t size, const uip_ipaddr_t *ipaddr);
264
265#endif /* NETSTACK_CONF_WITH_IPV6 */
266
267/**
268 * Logs a link-layer address
269 * \param lladdr The link-layer address
270*/
271void log_lladdr(const linkaddr_t *lladdr);
272
273/**
274 * Logs a link-layer address with a compact format
275 * \param lladdr The link-layer address
276*/
277void log_lladdr_compact(const linkaddr_t *lladdr);
278
279/**
280 * Logs a byte array as hex characters
281 * \param data The byte array
282 * \param length The length of the byte array
283*/
284void log_bytes(const void *data, size_t length);
285
286/**
287 * Sets a log level at run-time. Logs are included in the firmware via
288 * the compile-time flags in log-conf.h, but this allows to force lower log
289 * levels, system-wide.
290 * \param module The target module string descriptor
291 * \param level The log level
292*/
293void log_set_level(const char *module, int level);
294
295/**
296 * Returns the current log level.
297 * \param module The target module string descriptor
298 * \return The current log level
299*/
300int log_get_level(const char *module);
301
302/**
303 * Returns a textual description of a log level
304 * \param level log level
305 * \return The textual description
306*/
307const char *log_level_to_str(int level);
308
309#endif /* LOG_H_ */
310
311/** @} */
312/** @} */
void log_set_level(const char *module, int level)
Sets a log level at run-time.
Definition log.c:176
int log_6addr_compact_snprint(char *buf, size_t size, const uip_ipaddr_t *ipaddr)
Write at most size - 1 characters of the IP address to the output string, in a compact representation...
Definition log.c:100
void log_6addr_compact(const uip_ipaddr_t *ipaddr)
Logs an IPv6 address with a compact format.
Definition log.c:122
void log_lladdr(const linkaddr_t *lladdr)
Logs a link-layer address.
Definition log.c:131
const char * log_level_to_str(int level)
Returns a textual description of a log level.
Definition log.c:207
void log_lladdr_compact(const linkaddr_t *lladdr)
Logs a link-layer address with a compact format.
Definition log.c:148
void log_6addr(const uip_ipaddr_t *ipaddr)
Logs an IPv6 address.
Definition log.c:92
int log_get_level(const char *module)
Returns the current log level.
Definition log.c:191
void log_bytes(const void *data, size_t length)
Logs a byte array as hex characters.
Definition log.c:166
Header file for the link-layer address representation.
Default log levels for a number of modules.
static uip_ipaddr_t ipaddr
Pointer to prefix information option in uip_buf.
Definition uip-nd6.c:116
Header file for the uIP TCP/IP stack.