Printk changes for 5.1

-----BEGIN PGP SIGNATURE-----
 Version: GnuPG v2
 
 iQIcBAABAgAGBQJcf9djAAoJEFKgDEdIgJTyVGYP/REwDqjU7HrHCWdB0XJqens5
 OLkCnsnqRyhGUBJX6tc9dnY507FjEFS6DEAZDAPK14T/a50/57xWyXRNObzs7qud
 sP0otc/Gm7rWOP96ucEbMcPgzhixbC8fO3kvdQhc2HtxMQs4sotXqxDxVDHrbwNt
 7hvboTPLbiD7IFmqsQhtNWpGAXGhGzE8RwzltcWP+PcdI4oZVLW5lc4CUPG89AoX
 Fwd+7TAyH1X0/1QH/RQdOoUadiGIPucMC2HPbk+rDWF6+7zA83W9r28bYvuQemtZ
 myhRTj6Y46HfngJm5UFCZ56F3mLhs7V7Nc+bGNh6sU2xepCnzcAy5aBwl8xqnftZ
 hCxzyCdd8JrlcyMKRS4nq8dV0gsJKARRg0S3ZOdKElU4s7ep5dDJ0Ca9lq3K012I
 nPXrFgC6ZWp1i1aAWV2g5F9i6vB6rvbzqg+MGiuGijEOe0Lz80Humm/Rc4+0KQd2
 kT0AOZnpknYMXVs+vbkhntx1Pv1ubzER3B9aysB0hM1NLmZWs2lQgU6sLZnJie3i
 pEaFeCW+t52p+fkMJjYuirdjK4+zRYyeSqbSECdPCCjSjQcC2SX97hh/qyShtVxc
 EFjkEKCtHrquFmeGYzCyMk4/rlGtmOJRYchqJi9J/lULW9wHA4zpAyRa1FZkmBzl
 Zus2Kbmy2oilqsPdpAqt
 =dXuJ
 -----END PGP SIGNATURE-----

Merge tag 'printk-for-5.1' of git://git.kernel.org/pub/scm/linux/kernel/git/pmladek/printk

Pull printk updates from Petr Mladek:

 - Allow to sort mixed lines by an extra information about the caller

 - Remove no longer used LOG_PREFIX.

 - Some clean up and documentation update.

* tag 'printk-for-5.1' of git://git.kernel.org/pub/scm/linux/kernel/git/pmladek/printk:
  printk/docs: Add extra integer types to printk-formats
  printk: Remove no longer used LOG_PREFIX.
  lib/vsprintf: Remove %pCr remnant in comment
  printk: Pass caller information to log_store().
  printk: Add caller information to printk() output.
This commit is contained in:
Linus Torvalds 2019-03-09 09:22:42 -08:00
commit c4703acd6d
6 changed files with 96 additions and 23 deletions

View File

@ -13,6 +13,10 @@ Integer types
If variable is of Type, use printk format specifier: If variable is of Type, use printk format specifier:
------------------------------------------------------------ ------------------------------------------------------------
char %hhd or %hhx
unsigned char %hhu or %hhx
short int %hd or %hx
unsigned short int %hu or %hx
int %d or %x int %d or %x
unsigned int %u or %x unsigned int %u or %x
long %ld or %lx long %ld or %lx
@ -21,6 +25,10 @@ Integer types
unsigned long long %llu or %llx unsigned long long %llu or %llx
size_t %zu or %zx size_t %zu or %zx
ssize_t %zd or %zx ssize_t %zd or %zx
s8 %hhd or %hhx
u8 %hhu or %hhx
s16 %hd or %hx
u16 %hu or %hx
s32 %d or %x s32 %d or %x
u32 %u or %x u32 %u or %x
s64 %lld or %llx s64 %lld or %llx

View File

@ -14,7 +14,7 @@
#define KERN_INFO KERN_SOH "6" /* informational */ #define KERN_INFO KERN_SOH "6" /* informational */
#define KERN_DEBUG KERN_SOH "7" /* debug-level messages */ #define KERN_DEBUG KERN_SOH "7" /* debug-level messages */
#define KERN_DEFAULT KERN_SOH "d" /* the default kernel loglevel */ #define KERN_DEFAULT "" /* the default kernel loglevel */
/* /*
* Annotation for a "continued" line of log printout (only done after a * Annotation for a "continued" line of log printout (only done after a

View File

@ -18,7 +18,6 @@ static inline int printk_get_level(const char *buffer)
if (buffer[0] == KERN_SOH_ASCII && buffer[1]) { if (buffer[0] == KERN_SOH_ASCII && buffer[1]) {
switch (buffer[1]) { switch (buffer[1]) {
case '0' ... '7': case '0' ... '7':
case 'd': /* KERN_DEFAULT */
case 'c': /* KERN_CONT */ case 'c': /* KERN_CONT */
return buffer[1]; return buffer[1];
} }

View File

@ -344,7 +344,6 @@ static int console_msg_format = MSG_FORMAT_DEFAULT;
enum log_flags { enum log_flags {
LOG_NEWLINE = 2, /* text ended with a newline */ LOG_NEWLINE = 2, /* text ended with a newline */
LOG_PREFIX = 4, /* text started with a prefix */
LOG_CONT = 8, /* text is a fragment of a continuation line */ LOG_CONT = 8, /* text is a fragment of a continuation line */
}; };
@ -356,6 +355,9 @@ struct printk_log {
u8 facility; /* syslog facility */ u8 facility; /* syslog facility */
u8 flags:5; /* internal record flags */ u8 flags:5; /* internal record flags */
u8 level:3; /* syslog level */ u8 level:3; /* syslog level */
#ifdef CONFIG_PRINTK_CALLER
u32 caller_id; /* thread id or processor id */
#endif
} }
#ifdef CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS #ifdef CONFIG_HAVE_EFFICIENT_UNALIGNED_ACCESS
__packed __aligned(4) __packed __aligned(4)
@ -422,7 +424,11 @@ static u64 exclusive_console_stop_seq;
static u64 clear_seq; static u64 clear_seq;
static u32 clear_idx; static u32 clear_idx;
#ifdef CONFIG_PRINTK_CALLER
#define PREFIX_MAX 48
#else
#define PREFIX_MAX 32 #define PREFIX_MAX 32
#endif
#define LOG_LINE_MAX (1024 - PREFIX_MAX) #define LOG_LINE_MAX (1024 - PREFIX_MAX)
#define LOG_LEVEL(v) ((v) & 0x07) #define LOG_LEVEL(v) ((v) & 0x07)
@ -577,7 +583,7 @@ static u32 truncate_msg(u16 *text_len, u16 *trunc_msg_len,
} }
/* insert record into the buffer, discard old ones, update heads */ /* insert record into the buffer, discard old ones, update heads */
static int log_store(int facility, int level, static int log_store(u32 caller_id, int facility, int level,
enum log_flags flags, u64 ts_nsec, enum log_flags flags, u64 ts_nsec,
const char *dict, u16 dict_len, const char *dict, u16 dict_len,
const char *text, u16 text_len) const char *text, u16 text_len)
@ -625,6 +631,9 @@ static int log_store(int facility, int level,
msg->ts_nsec = ts_nsec; msg->ts_nsec = ts_nsec;
else else
msg->ts_nsec = local_clock(); msg->ts_nsec = local_clock();
#ifdef CONFIG_PRINTK_CALLER
msg->caller_id = caller_id;
#endif
memset(log_dict(msg) + dict_len, 0, pad_len); memset(log_dict(msg) + dict_len, 0, pad_len);
msg->len = size; msg->len = size;
@ -688,12 +697,21 @@ static ssize_t msg_print_ext_header(char *buf, size_t size,
struct printk_log *msg, u64 seq) struct printk_log *msg, u64 seq)
{ {
u64 ts_usec = msg->ts_nsec; u64 ts_usec = msg->ts_nsec;
char caller[20];
#ifdef CONFIG_PRINTK_CALLER
u32 id = msg->caller_id;
snprintf(caller, sizeof(caller), ",caller=%c%u",
id & 0x80000000 ? 'C' : 'T', id & ~0x80000000);
#else
caller[0] = '\0';
#endif
do_div(ts_usec, 1000); do_div(ts_usec, 1000);
return scnprintf(buf, size, "%u,%llu,%llu,%c;", return scnprintf(buf, size, "%u,%llu,%llu,%c%s;",
(msg->facility << 3) | msg->level, seq, ts_usec, (msg->facility << 3) | msg->level, seq, ts_usec,
msg->flags & LOG_CONT ? 'c' : '-'); msg->flags & LOG_CONT ? 'c' : '-', caller);
} }
static ssize_t msg_print_ext_body(char *buf, size_t size, static ssize_t msg_print_ext_body(char *buf, size_t size,
@ -1038,6 +1056,9 @@ void log_buf_vmcoreinfo_setup(void)
VMCOREINFO_OFFSET(printk_log, len); VMCOREINFO_OFFSET(printk_log, len);
VMCOREINFO_OFFSET(printk_log, text_len); VMCOREINFO_OFFSET(printk_log, text_len);
VMCOREINFO_OFFSET(printk_log, dict_len); VMCOREINFO_OFFSET(printk_log, dict_len);
#ifdef CONFIG_PRINTK_CALLER
VMCOREINFO_OFFSET(printk_log, caller_id);
#endif
} }
#endif #endif
@ -1236,10 +1257,23 @@ static size_t print_time(u64 ts, char *buf)
{ {
unsigned long rem_nsec = do_div(ts, 1000000000); unsigned long rem_nsec = do_div(ts, 1000000000);
return sprintf(buf, "[%5lu.%06lu] ", return sprintf(buf, "[%5lu.%06lu]",
(unsigned long)ts, rem_nsec / 1000); (unsigned long)ts, rem_nsec / 1000);
} }
#ifdef CONFIG_PRINTK_CALLER
static size_t print_caller(u32 id, char *buf)
{
char caller[12];
snprintf(caller, sizeof(caller), "%c%u",
id & 0x80000000 ? 'C' : 'T', id & ~0x80000000);
return sprintf(buf, "[%6s]", caller);
}
#else
#define print_caller(id, buf) 0
#endif
static size_t print_prefix(const struct printk_log *msg, bool syslog, static size_t print_prefix(const struct printk_log *msg, bool syslog,
bool time, char *buf) bool time, char *buf)
{ {
@ -1247,8 +1281,17 @@ static size_t print_prefix(const struct printk_log *msg, bool syslog,
if (syslog) if (syslog)
len = print_syslog((msg->facility << 3) | msg->level, buf); len = print_syslog((msg->facility << 3) | msg->level, buf);
if (time) if (time)
len += print_time(msg->ts_nsec, buf + len); len += print_time(msg->ts_nsec, buf + len);
len += print_caller(msg->caller_id, buf + len);
if (IS_ENABLED(CONFIG_PRINTK_CALLER) || time) {
buf[len++] = ' ';
buf[len] = '\0';
}
return len; return len;
} }
@ -1752,6 +1795,12 @@ static inline void printk_delay(void)
} }
} }
static inline u32 printk_caller_id(void)
{
return in_task() ? task_pid_nr(current) :
0x80000000 + raw_smp_processor_id();
}
/* /*
* Continuation lines are buffered, and not committed to the record buffer * Continuation lines are buffered, and not committed to the record buffer
* until the line is complete, or a race forces it. The line fragments * until the line is complete, or a race forces it. The line fragments
@ -1761,7 +1810,7 @@ static inline void printk_delay(void)
static struct cont { static struct cont {
char buf[LOG_LINE_MAX]; char buf[LOG_LINE_MAX];
size_t len; /* length == 0 means unused buffer */ size_t len; /* length == 0 means unused buffer */
struct task_struct *owner; /* task of first print*/ u32 caller_id; /* printk_caller_id() of first print */
u64 ts_nsec; /* time of first print */ u64 ts_nsec; /* time of first print */
u8 level; /* log level of first message */ u8 level; /* log level of first message */
u8 facility; /* log facility of first message */ u8 facility; /* log facility of first message */
@ -1773,12 +1822,13 @@ static void cont_flush(void)
if (cont.len == 0) if (cont.len == 0)
return; return;
log_store(cont.facility, cont.level, cont.flags, cont.ts_nsec, log_store(cont.caller_id, cont.facility, cont.level, cont.flags,
NULL, 0, cont.buf, cont.len); cont.ts_nsec, NULL, 0, cont.buf, cont.len);
cont.len = 0; cont.len = 0;
} }
static bool cont_add(int facility, int level, enum log_flags flags, const char *text, size_t len) static bool cont_add(u32 caller_id, int facility, int level,
enum log_flags flags, const char *text, size_t len)
{ {
/* If the line gets too long, split it up in separate records. */ /* If the line gets too long, split it up in separate records. */
if (cont.len + len > sizeof(cont.buf)) { if (cont.len + len > sizeof(cont.buf)) {
@ -1789,7 +1839,7 @@ static bool cont_add(int facility, int level, enum log_flags flags, const char *
if (!cont.len) { if (!cont.len) {
cont.facility = facility; cont.facility = facility;
cont.level = level; cont.level = level;
cont.owner = current; cont.caller_id = caller_id;
cont.ts_nsec = local_clock(); cont.ts_nsec = local_clock();
cont.flags = flags; cont.flags = flags;
} }
@ -1809,13 +1859,15 @@ static bool cont_add(int facility, int level, enum log_flags flags, const char *
static size_t log_output(int facility, int level, enum log_flags lflags, const char *dict, size_t dictlen, char *text, size_t text_len) static size_t log_output(int facility, int level, enum log_flags lflags, const char *dict, size_t dictlen, char *text, size_t text_len)
{ {
const u32 caller_id = printk_caller_id();
/* /*
* If an earlier line was buffered, and we're a continuation * If an earlier line was buffered, and we're a continuation
* write from the same process, try to add it to the buffer. * write from the same context, try to add it to the buffer.
*/ */
if (cont.len) { if (cont.len) {
if (cont.owner == current && (lflags & LOG_CONT)) { if (cont.caller_id == caller_id && (lflags & LOG_CONT)) {
if (cont_add(facility, level, lflags, text, text_len)) if (cont_add(caller_id, facility, level, lflags, text, text_len))
return text_len; return text_len;
} }
/* Otherwise, make sure it's flushed */ /* Otherwise, make sure it's flushed */
@ -1828,12 +1880,13 @@ static size_t log_output(int facility, int level, enum log_flags lflags, const c
/* If it doesn't end in a newline, try to buffer the current line */ /* If it doesn't end in a newline, try to buffer the current line */
if (!(lflags & LOG_NEWLINE)) { if (!(lflags & LOG_NEWLINE)) {
if (cont_add(facility, level, lflags, text, text_len)) if (cont_add(caller_id, facility, level, lflags, text, text_len))
return text_len; return text_len;
} }
/* Store it in the record log */ /* Store it in the record log */
return log_store(facility, level, lflags, 0, dict, dictlen, text, text_len); return log_store(caller_id, facility, level, lflags, 0,
dict, dictlen, text, text_len);
} }
/* Must be called under logbuf_lock. */ /* Must be called under logbuf_lock. */
@ -1867,9 +1920,6 @@ int vprintk_store(int facility, int level,
case '0' ... '7': case '0' ... '7':
if (level == LOGLEVEL_DEFAULT) if (level == LOGLEVEL_DEFAULT)
level = kern_level - '0'; level = kern_level - '0';
/* fallthrough */
case 'd': /* KERN_DEFAULT */
lflags |= LOG_PREFIX;
break; break;
case 'c': /* KERN_CONT */ case 'c': /* KERN_CONT */
lflags |= LOG_CONT; lflags |= LOG_CONT;
@ -1884,7 +1934,7 @@ int vprintk_store(int facility, int level,
level = default_message_loglevel; level = default_message_loglevel;
if (dict) if (dict)
lflags |= LOG_PREFIX|LOG_NEWLINE; lflags |= LOG_NEWLINE;
return log_output(facility, level, lflags, return log_output(facility, level, lflags,
dict, dictlen, text, text_len); dict, dictlen, text, text_len);

View File

@ -17,6 +17,23 @@ config PRINTK_TIME
The behavior is also controlled by the kernel command line The behavior is also controlled by the kernel command line
parameter printk.time=1. See Documentation/admin-guide/kernel-parameters.rst parameter printk.time=1. See Documentation/admin-guide/kernel-parameters.rst
config PRINTK_CALLER
bool "Show caller information on printks"
depends on PRINTK
help
Selecting this option causes printk() to add a caller "thread id" (if
in task context) or a caller "processor id" (if not in task context)
to every message.
This option is intended for environments where multiple threads
concurrently call printk() for many times, for it is difficult to
interpret without knowing where these lines (or sometimes individual
line which was divided into multiple lines due to race) came from.
Since toggling after boot makes the code racy, currently there is
no option to enable/disable at the kernel command line parameter or
sysfs interface.
config CONSOLE_LOGLEVEL_DEFAULT config CONSOLE_LOGLEVEL_DEFAULT
int "Default console loglevel (1-15)" int "Default console loglevel (1-15)"
range 1 15 range 1 15

View File

@ -1931,7 +1931,6 @@ char *device_node_string(char *buf, char *end, struct device_node *dn,
* (legacy clock framework) of the clock * (legacy clock framework) of the clock
* - 'Cn' For a clock, it prints the name (Common Clock Framework) or address * - 'Cn' For a clock, it prints the name (Common Clock Framework) or address
* (legacy clock framework) of the clock * (legacy clock framework) of the clock
* - 'Cr' For a clock, it prints the current rate of the clock
* - 'G' For flags to be printed as a collection of symbolic strings that would * - 'G' For flags to be printed as a collection of symbolic strings that would
* construct the specific value. Supported flags given by option: * construct the specific value. Supported flags given by option:
* p page flags (see struct page) given as pointer to unsigned long * p page flags (see struct page) given as pointer to unsigned long