1*a5c7a39fSPetr Mladek.. SPDX-License-Identifier: GPL-2.0 2*a5c7a39fSPetr Mladek 3*a5c7a39fSPetr Mladek============ 4*a5c7a39fSPetr MladekPrintk Index 5*a5c7a39fSPetr Mladek============ 6*a5c7a39fSPetr Mladek 7*a5c7a39fSPetr MladekThere are many ways how to monitor the state of the system. One important 8*a5c7a39fSPetr Mladeksource of information is the system log. It provides a lot of information, 9*a5c7a39fSPetr Mladekincluding more or less important warnings and error messages. 10*a5c7a39fSPetr Mladek 11*a5c7a39fSPetr MladekThere are monitoring tools that filter and take action based on messages 12*a5c7a39fSPetr Mladeklogged. 13*a5c7a39fSPetr Mladek 14*a5c7a39fSPetr MladekThe kernel messages are evolving together with the code. As a result, 15*a5c7a39fSPetr Mladekparticular kernel messages are not KABI and never will be! 16*a5c7a39fSPetr Mladek 17*a5c7a39fSPetr MladekIt is a huge challenge for maintaining the system log monitors. It requires 18*a5c7a39fSPetr Mladekknowing what messages were updated in a particular kernel version and why. 19*a5c7a39fSPetr MladekFinding these changes in the sources would require non-trivial parsers. 20*a5c7a39fSPetr MladekAlso it would require matching the sources with the binary kernel which 21*a5c7a39fSPetr Mladekis not always trivial. Various changes might be backported. Various kernel 22*a5c7a39fSPetr Mladekversions might be used on different monitored systems. 23*a5c7a39fSPetr Mladek 24*a5c7a39fSPetr MladekThis is where the printk index feature might become useful. It provides 25*a5c7a39fSPetr Mladeka dump of printk formats used all over the source code used for the kernel 26*a5c7a39fSPetr Mladekand modules on the running system. It is accessible at runtime via debugfs. 27*a5c7a39fSPetr Mladek 28*a5c7a39fSPetr MladekThe printk index helps to find changes in the message formats. Also it helps 29*a5c7a39fSPetr Mladekto track the strings back to the kernel sources and the related commit. 30*a5c7a39fSPetr Mladek 31*a5c7a39fSPetr Mladek 32*a5c7a39fSPetr MladekUser Interface 33*a5c7a39fSPetr Mladek============== 34*a5c7a39fSPetr Mladek 35*a5c7a39fSPetr MladekThe index of printk formats are split in into separate files. The files are 36*a5c7a39fSPetr Mladeknamed according to the binaries where the printk formats are built-in. There 37*a5c7a39fSPetr Mladekis always "vmlinux" and optionally also modules, for example:: 38*a5c7a39fSPetr Mladek 39*a5c7a39fSPetr Mladek /sys/kernel/debug/printk/index/vmlinux 40*a5c7a39fSPetr Mladek /sys/kernel/debug/printk/index/ext4 41*a5c7a39fSPetr Mladek /sys/kernel/debug/printk/index/scsi_mod 42*a5c7a39fSPetr Mladek 43*a5c7a39fSPetr MladekNote that only loaded modules are shown. Also printk formats from a module 44*a5c7a39fSPetr Mladekmight appear in "vmlinux" when the module is built-in. 45*a5c7a39fSPetr Mladek 46*a5c7a39fSPetr MladekThe content is inspired by the dynamic debug interface and looks like:: 47*a5c7a39fSPetr Mladek 48*a5c7a39fSPetr Mladek $> head -1 /sys/kernel/debug/printk/index/vmlinux; shuf -n 5 vmlinux 49*a5c7a39fSPetr Mladek # <level[,flags]> filename:line function "format" 50*a5c7a39fSPetr Mladek <5> block/blk-settings.c:661 disk_stack_limits "%s: Warning: Device %s is misaligned\n" 51*a5c7a39fSPetr Mladek <4> kernel/trace/trace.c:8296 trace_create_file "Could not create tracefs '%s' entry\n" 52*a5c7a39fSPetr Mladek <6> arch/x86/kernel/hpet.c:144 _hpet_print_config "hpet: %s(%d):\n" 53*a5c7a39fSPetr Mladek <6> init/do_mounts.c:605 prepare_namespace "Waiting for root device %s...\n" 54*a5c7a39fSPetr Mladek <6> drivers/acpi/osl.c:1410 acpi_no_auto_serialize_setup "ACPI: auto-serialization disabled\n" 55*a5c7a39fSPetr Mladek 56*a5c7a39fSPetr Mladek, where the meaning is: 57*a5c7a39fSPetr Mladek 58*a5c7a39fSPetr Mladek - :level: log level value: 0-7 for particular severity, -1 as default, 59*a5c7a39fSPetr Mladek 'c' as continuous line without an explicit log level 60*a5c7a39fSPetr Mladek - :flags: optional flags: currently only 'c' for KERN_CONT 61*a5c7a39fSPetr Mladek - :filename\:line: source filename and line number of the related 62*a5c7a39fSPetr Mladek printk() call. Note that there are many wrappers, for example, 63*a5c7a39fSPetr Mladek pr_warn(), pr_warn_once(), dev_warn(). 64*a5c7a39fSPetr Mladek - :function: function name where the printk() call is used. 65*a5c7a39fSPetr Mladek - :format: format string 66*a5c7a39fSPetr Mladek 67*a5c7a39fSPetr MladekThe extra information makes it a bit harder to find differences 68*a5c7a39fSPetr Mladekbetween various kernels. Especially the line number might change 69*a5c7a39fSPetr Mladekvery often. On the other hand, it helps a lot to confirm that 70*a5c7a39fSPetr Mladekit is the same string or find the commit that is responsible 71*a5c7a39fSPetr Mladekfor eventual changes. 72*a5c7a39fSPetr Mladek 73*a5c7a39fSPetr Mladek 74*a5c7a39fSPetr Mladekprintk() Is Not a Stable KABI 75*a5c7a39fSPetr Mladek============================= 76*a5c7a39fSPetr Mladek 77*a5c7a39fSPetr MladekSeveral developers are afraid that exporting all these implementation 78*a5c7a39fSPetr Mladekdetails into the user space will transform particular printk() calls 79*a5c7a39fSPetr Mladekinto KABI. 80*a5c7a39fSPetr Mladek 81*a5c7a39fSPetr MladekBut it is exactly the opposite. printk() calls must _not_ be KABI. 82*a5c7a39fSPetr MladekAnd the printk index helps user space tools to deal with this. 83*a5c7a39fSPetr Mladek 84*a5c7a39fSPetr Mladek 85*a5c7a39fSPetr MladekSubsystem specific printk wrappers 86*a5c7a39fSPetr Mladek================================== 87*a5c7a39fSPetr Mladek 88*a5c7a39fSPetr MladekThe printk index is generated using extra metadata that are stored in 89*a5c7a39fSPetr Mladeka dedicated .elf section ".printk_index". It is achieved using macro 90*a5c7a39fSPetr Mladekwrappers doing __printk_index_emit() together with the real printk() 91*a5c7a39fSPetr Mladekcall. The same technique is used also for the metadata used by 92*a5c7a39fSPetr Mladekthe dynamic debug feature. 93*a5c7a39fSPetr Mladek 94*a5c7a39fSPetr MladekThe metadata are stored for a particular message only when it is printed 95*a5c7a39fSPetr Mladekusing these special wrappers. It is implemented for the commonly 96*a5c7a39fSPetr Mladekused printk() calls, including, for example, pr_warn(), or pr_once(). 97*a5c7a39fSPetr Mladek 98*a5c7a39fSPetr MladekAdditional changes are necessary for various subsystem specific wrappers 99*a5c7a39fSPetr Mladekthat call the original printk() via a common helper function. These needs 100*a5c7a39fSPetr Mladektheir own wrappers adding __printk_index_emit(). 101*a5c7a39fSPetr Mladek 102*a5c7a39fSPetr MladekOnly few subsystem specific wrappers have been updated so far, 103*a5c7a39fSPetr Mladekfor example, dev_printk(). As a result, the printk formats from 104*a5c7a39fSPetr Mladeksome subsystes can be missing in the printk index. 105*a5c7a39fSPetr Mladek 106*a5c7a39fSPetr Mladek 107*a5c7a39fSPetr MladekSubsystem specific prefix 108*a5c7a39fSPetr Mladek========================= 109*a5c7a39fSPetr Mladek 110*a5c7a39fSPetr MladekThe macro pr_fmt() macro allows to define a prefix that is printed 111*a5c7a39fSPetr Mladekbefore the string generated by the related printk() calls. 112*a5c7a39fSPetr Mladek 113*a5c7a39fSPetr MladekSubsystem specific wrappers usually add even more complicated 114*a5c7a39fSPetr Mladekprefixes. 115*a5c7a39fSPetr Mladek 116*a5c7a39fSPetr MladekThese prefixes can be stored into the printk index metadata 117*a5c7a39fSPetr Mladekby an optional parameter of __printk_index_emit(). The debugfs 118*a5c7a39fSPetr Mladekinterface might then show the printk formats including these prefixes. 119*a5c7a39fSPetr MladekFor example, drivers/acpi/osl.c contains:: 120*a5c7a39fSPetr Mladek 121*a5c7a39fSPetr Mladek #define pr_fmt(fmt) "ACPI: OSL: " fmt 122*a5c7a39fSPetr Mladek 123*a5c7a39fSPetr Mladek static int __init acpi_no_auto_serialize_setup(char *str) 124*a5c7a39fSPetr Mladek { 125*a5c7a39fSPetr Mladek acpi_gbl_auto_serialize_methods = FALSE; 126*a5c7a39fSPetr Mladek pr_info("Auto-serialization disabled\n"); 127*a5c7a39fSPetr Mladek 128*a5c7a39fSPetr Mladek return 1; 129*a5c7a39fSPetr Mladek } 130*a5c7a39fSPetr Mladek 131*a5c7a39fSPetr MladekThis results in the following printk index entry:: 132*a5c7a39fSPetr Mladek 133*a5c7a39fSPetr Mladek <6> drivers/acpi/osl.c:1410 acpi_no_auto_serialize_setup "ACPI: auto-serialization disabled\n" 134*a5c7a39fSPetr Mladek 135*a5c7a39fSPetr MladekIt helps matching messages from the real log with printk index. 136*a5c7a39fSPetr MladekThen the source file name, line number, and function name can 137*a5c7a39fSPetr Mladekbe used to match the string with the source code. 138