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_snmp;
119extern int curr_log_level_lwm2m;
120extern int curr_log_level_sys;
121extern int curr_log_level_main;
122
123extern struct log_module all_modules[];
124
125#define LOG_LEVEL_RPL MIN((LOG_CONF_LEVEL_RPL), curr_log_level_rpl)
126#define LOG_LEVEL_TCPIP MIN((LOG_CONF_LEVEL_TCPIP), curr_log_level_tcpip)
127#define LOG_LEVEL_IPV6 MIN((LOG_CONF_LEVEL_IPV6), curr_log_level_ipv6)
128#define LOG_LEVEL_6LOWPAN MIN((LOG_CONF_LEVEL_6LOWPAN), curr_log_level_6lowpan)
129#define LOG_LEVEL_NULLNET MIN((LOG_CONF_LEVEL_NULLNET), curr_log_level_nullnet)
130#define LOG_LEVEL_MAC MIN((LOG_CONF_LEVEL_MAC), curr_log_level_mac)
131#define LOG_LEVEL_FRAMER MIN((LOG_CONF_LEVEL_FRAMER), curr_log_level_framer)
132#define LOG_LEVEL_6TOP MIN((LOG_CONF_LEVEL_6TOP), curr_log_level_6top)
133#define LOG_LEVEL_COAP MIN((LOG_CONF_LEVEL_COAP), curr_log_level_coap)
134#define LOG_LEVEL_SNMP MIN((LOG_CONF_LEVEL_SNMP), curr_log_level_snmp)
135#define LOG_LEVEL_LWM2M MIN((LOG_CONF_LEVEL_LWM2M), curr_log_level_lwm2m)
136#define LOG_LEVEL_SYS MIN((LOG_CONF_LEVEL_SYS), curr_log_level_sys)
137#define LOG_LEVEL_MAIN MIN((LOG_CONF_LEVEL_MAIN), curr_log_level_main)
138
139/* Main log function */
140
141#define LOG(newline, level, levelstr, levelcolor, ...) do { \
142 if(level <= (LOG_LEVEL)) { \
143 if(newline) { \
144 if(LOG_WITH_COLOR) { \
145 LOG_OUTPUT(levelcolor); \
146 } \
147 if(LOG_WITH_MODULE_PREFIX) { \
148 LOG_OUTPUT_PREFIX(level, levelstr, LOG_MODULE); \
149 } \
150 if(LOG_WITH_LOC) { \
151 LOG_OUTPUT("[%s: %d] ", __FILE__, __LINE__); \
152 } \
153 if(LOG_WITH_COLOR) { \
154 LOG_OUTPUT(LOG_COLOR_RESET); \
155 } \
156 } \
157 LOG_OUTPUT(__VA_ARGS__); \
158 } \
159 } while (0)
160
161/* For Cooja annotations */
162#define LOG_ANNOTATE(...) do { \
163 if(LOG_WITH_ANNOTATE) { \
164 LOG_OUTPUT(__VA_ARGS__); \
165 } \
166 } while (0)
167
168/* Link-layer address */
169#define LOG_LLADDR(level, lladdr) do { \
170 if(level <= (LOG_LEVEL)) { \
171 if(LOG_WITH_COMPACT_ADDR) { \
172 log_lladdr_compact(lladdr); \
173 } else { \
174 log_lladdr(lladdr); \
175 } \
176 } \
177 } while (0)
178
179/* IPv6 address */
180#define LOG_6ADDR(level, ipaddr) do { \
181 if(level <= (LOG_LEVEL)) { \
182 if(LOG_WITH_COMPACT_ADDR) { \
183 log_6addr_compact(ipaddr); \
184 } else { \
185 log_6addr(ipaddr); \
186 } \
187 } \
188 } while (0)
189
190#define LOG_BYTES(level, data, length) do { \
191 if(level <= (LOG_LEVEL)) { \
192 log_bytes(data, length); \
193 } \
194 } while (0)
195
196/* More compact versions of LOG macros */
197#define LOG_PRINT(...) LOG(1, 0, "PRI", LOG_COLOR_PRI, __VA_ARGS__)
198#define LOG_ERR(...) LOG(1, LOG_LEVEL_ERR, "ERR", LOG_COLOR_ERR, __VA_ARGS__)
199#define LOG_WARN(...) LOG(1, LOG_LEVEL_WARN, "WARN", LOG_COLOR_WARN, __VA_ARGS__)
200#define LOG_INFO(...) LOG(1, LOG_LEVEL_INFO, "INFO", LOG_COLOR_INFO, __VA_ARGS__)
201#define LOG_DBG(...) LOG(1, LOG_LEVEL_DBG, "DBG", LOG_COLOR_DBG, __VA_ARGS__)
202
203#define LOG_PRINT_(...) LOG(0, 0, "PRI", LOG_COLOR_PRI, __VA_ARGS__)
204#define LOG_ERR_(...) LOG(0, LOG_LEVEL_ERR, "ERR", LOG_COLOR_ERR, __VA_ARGS__)
205#define LOG_WARN_(...) LOG(0, LOG_LEVEL_WARN, "WARN", LOG_COLOR_WARN, __VA_ARGS__)
206#define LOG_INFO_(...) LOG(0, LOG_LEVEL_INFO, "INFO", LOG_COLOR_INFO, __VA_ARGS__)
207#define LOG_DBG_(...) LOG(0, LOG_LEVEL_DBG, "DBG", LOG_COLOR_DBG, __VA_ARGS__)
208
209#define LOG_PRINT_LLADDR(...) LOG_LLADDR(0, __VA_ARGS__)
210#define LOG_ERR_LLADDR(...) LOG_LLADDR(LOG_LEVEL_ERR, __VA_ARGS__)
211#define LOG_WARN_LLADDR(...) LOG_LLADDR(LOG_LEVEL_WARN, __VA_ARGS__)
212#define LOG_INFO_LLADDR(...) LOG_LLADDR(LOG_LEVEL_INFO, __VA_ARGS__)
213#define LOG_DBG_LLADDR(...) LOG_LLADDR(LOG_LEVEL_DBG, __VA_ARGS__)
214
215#define LOG_PRINT_6ADDR(...) LOG_6ADDR(0, __VA_ARGS__)
216#define LOG_ERR_6ADDR(...) LOG_6ADDR(LOG_LEVEL_ERR, __VA_ARGS__)
217#define LOG_WARN_6ADDR(...) LOG_6ADDR(LOG_LEVEL_WARN, __VA_ARGS__)
218#define LOG_INFO_6ADDR(...) LOG_6ADDR(LOG_LEVEL_INFO, __VA_ARGS__)
219#define LOG_DBG_6ADDR(...) LOG_6ADDR(LOG_LEVEL_DBG, __VA_ARGS__)
220
221#define LOG_PRINT_BYTES(data, length) LOG_BYTES(0, data, length)
222#define LOG_ERR_BYTES(data, length) LOG_BYTES(LOG_LEVEL_ERR, data, length)
223#define LOG_WARN_BYTES(data, length) LOG_BYTES(LOG_LEVEL_WARN, data, length)
224#define LOG_INFO_BYTES(data, length) LOG_BYTES(LOG_LEVEL_INFO, data, length)
225#define LOG_DBG_BYTES(data, length) LOG_BYTES(LOG_LEVEL_DBG, data, length)
226
227/* For checking log level.
228 As this builds on curr_log_level variables, this should not be used
229 in pre-processor macros. Use in a C 'if' statement instead, e.g.:
230 if(LOG_INFO_ENABLED) { ... }
231 Note that most compilers will still be able to strip the code out
232 for low enough log levels configurations. */
233#define LOG_ERR_ENABLED ((LOG_LEVEL) >= LOG_LEVEL_ERR)
234#define LOG_WARN_ENABLED ((LOG_LEVEL) >= LOG_LEVEL_WARN)
235#define LOG_INFO_ENABLED ((LOG_LEVEL) >= LOG_LEVEL_INFO)
236#define LOG_DBG_ENABLED ((LOG_LEVEL) >= LOG_LEVEL_DBG)
237
238#if NETSTACK_CONF_WITH_IPV6
239
240/**
241 * Logs an IPv6 address
242 * \param ipaddr The IPv6 address
243*/
244void log_6addr(const uip_ipaddr_t *ipaddr);
245
246/**
247 * Logs an IPv6 address with a compact format
248 * \param ipaddr The IPv6 address
249*/
250void log_6addr_compact(const uip_ipaddr_t *ipaddr);
251
252/**
253 * Write at most size - 1 characters of the IP address to the output string,
254 * in a compact representation. The output is always null-terminated, unless
255 * size is 0.
256 *
257 * \param buf A pointer to an output string with at least size bytes.
258 * \param size The max number of characters to write to the output string.
259 * \param ipaddr A pointer to a uip_ipaddr_t that will be printed with printf().
260 */
261int log_6addr_compact_snprint(char *buf, size_t size, const uip_ipaddr_t *ipaddr);
262
263#endif /* NETSTACK_CONF_WITH_IPV6 */
264
265/**
266 * Logs a link-layer address
267 * \param lladdr The link-layer address
268*/
269void log_lladdr(const linkaddr_t *lladdr);
270
271/**
272 * Logs a link-layer address with a compact format
273 * \param lladdr The link-layer address
274*/
275void log_lladdr_compact(const linkaddr_t *lladdr);
276
277/**
278 * Logs a byte array as hex characters
279 * \param data The byte array
280 * \param length The length of the byte array
281*/
282void log_bytes(const void *data, size_t length);
283
284/**
285 * Sets a log level at run-time. Logs are included in the firmware via
286 * the compile-time flags in log-conf.h, but this allows to force lower log
287 * levels, system-wide.
288 * \param module The target module string descriptor
289 * \param level The log level
290*/
291void log_set_level(const char *module, int level);
292
293/**
294 * Returns the current log level.
295 * \param module The target module string descriptor
296 * \return The current log level
297*/
298int log_get_level(const char *module);
299
300/**
301 * Returns a textual description of a log level
302 * \param level log level
303 * \return The textual description
304*/
305const char *log_level_to_str(int level);
306
307#endif /* LOG_H_ */
308
309/** @} */
310/** @} */
void log_set_level(const char *module, int level)
Sets a log level at run-time.
Definition log.c:175
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:99
void log_6addr_compact(const uip_ipaddr_t *ipaddr)
Logs an IPv6 address with a compact format.
Definition log.c:121
void log_lladdr(const linkaddr_t *lladdr)
Logs a link-layer address.
Definition log.c:130
const char * log_level_to_str(int level)
Returns a textual description of a log level.
Definition log.c:206
void log_lladdr_compact(const linkaddr_t *lladdr)
Logs a link-layer address with a compact format.
Definition log.c:147
void log_6addr(const uip_ipaddr_t *ipaddr)
Logs an IPv6 address.
Definition log.c:91
int log_get_level(const char *module)
Returns the current log level.
Definition log.c:190
void log_bytes(const void *data, size_t length)
Logs a byte array as hex characters.
Definition log.c:165
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.