Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src Update the kernhist(9) kernel history code to address issues...



details:   https://anonhg.NetBSD.org/src/rev/c5b3cd49c5c1
branches:  trunk
changeset: 357120:c5b3cd49c5c1
user:      pgoyette <pgoyette%NetBSD.org@localhost>
date:      Sat Oct 28 00:37:11 2017 +0000

description:
Update the kernhist(9) kernel history code to address issues identified
in PR kern/52639, as well as some general cleaning-up...

(As proposed on tech-kern@ with additional changes and enhancements.)

Details of changes:

* All history arguments are now stored as uintmax_t values[1], both in
  the kernel and in the structures used for exporting the history data
  to userland via sysctl(9).  This avoids problems on some architectures
  where passing a 64-bit (or larger) value to printf(3) can cause it to
  process the value as multiple arguments.  (This can be particularly
  problematic when printf()'s format string is not a literal, since in
  that case the compiler cannot know how large each argument should be.)

* Update the data structures used for exporting kernel history data to
  include a version number as well as the length of history arguments.

* All [2] existing users of kernhist(9) have had their format strings
  updated.  Each format specifier now includes an explicit length
  modifier 'j' to refer to numeric values of the size of uintmax_t.

* All [2] existing users of kernhist(9) have had their format strings
  updated to replace uses of "%p" with "%#jx", and the pointer
  arguments are now cast to (uintptr_t) before being subsequently cast
  to (uintmax_t).  This is needed to avoid compiler warnings about
  casting "pointer to integer of a different size."

* All [2] existing users of kernhist(9) have had instances of "%s" or
  "%c" format strings replaced with numeric formats; several instances
  of mis-match between format string and argument list have been fixed.

* vmstat(1) has been modified to handle the new size of arguments in the
  history data as exported by sysctl(9).

* vmstat(1) now provides a warning message if the history requested with
  the -u option does not exist (previously, this condition was silently
  ignored, with only a single blank line being printed).

* vmstat(1) now checks the version and argument length included in the
  data exported via sysctl(9) and exits if they do not match the values
  with which vmstat was built.

* The kernhist(9) man-page has been updated to note the additional
  requirements imposed on the format strings, along with several other
  minor changes and enhancements.

[1] It would have been possible to use an explicit length (for example,
    uint64_t) for the history arguments.  But that would require another
    "rototill" of all the users in the future when we add support for an
    architecture that supports a larger size.  Also, the printf(3) format
    specifiers for explicitly-sized values, such as "%"PRIu64, are much
    more verbose (and less aesthetically appealing, IMHO) than simply
    using "%ju".

[2] I've tried very hard to find "all [the] existing users of kernhist(9)"
    but it is possible that I've missed some of them.  I would be glad to
    update any stragglers that anyone identifies.

diffstat:

 share/man/man9/kernhist.9                      |  115 ++++++++-
 sys/arch/acorn26/acorn26/pmap.c                |   38 +-
 sys/arch/arm/arm32/fault.c                     |   20 +-
 sys/arch/arm/arm32/pmap.c                      |   83 +++---
 sys/arch/arm/broadcom/bcm2835_bsc.c            |   28 +-
 sys/arch/arm/omap/if_cpsw.c                    |   51 ++--
 sys/arch/arm/omap/tiotg.c                      |   11 +-
 sys/dev/ic/sl811hs.c                           |  111 ++++----
 sys/dev/usb/ehci.c                             |  293 +++++++++++++-----------
 sys/dev/usb/if_axe.c                           |   39 +-
 sys/dev/usb/motg.c                             |  141 ++++++-----
 sys/dev/usb/ohci.c                             |  273 ++++++++++++----------
 sys/dev/usb/ucom.c                             |   60 ++--
 sys/dev/usb/uhci.c                             |  193 ++++++++-------
 sys/dev/usb/uhub.c                             |   65 ++--
 sys/dev/usb/umass.c                            |  180 ++++++++------
 sys/dev/usb/umass_quirks.c                     |    8 +-
 sys/dev/usb/umass_scsipi.c                     |   52 ++--
 sys/dev/usb/usb.c                              |   24 +-
 sys/dev/usb/usb_mem.c                          |   16 +-
 sys/dev/usb/usb_subr.c                         |  153 ++++++------
 sys/dev/usb/usbdi.c                            |  141 ++++++-----
 sys/dev/usb/usbdi_util.c                       |   79 +++---
 sys/dev/usb/usbroothub.c                       |   16 +-
 sys/dev/usb/xhci.c                             |  237 ++++++++++---------
 sys/external/bsd/drm2/dist/drm/i915/i915_gem.c |    8 +-
 sys/kern/kern_history.c                        |   16 +-
 sys/kern/kern_xxx.c                            |   32 +-
 sys/kern/vfs_bio.c                             |   32 +-
 sys/miscfs/genfs/genfs_io.c                    |  104 ++++----
 sys/sys/kernhist.h                             |   58 ++--
 sys/ufs/ffs/ffs_balloc.c                       |   10 +-
 sys/ufs/lfs/lfs_vfsops.c                       |   20 +-
 sys/ufs/lfs/ulfs_inode.c                       |    8 +-
 sys/ufs/lfs/ulfs_vnops.c                       |    8 +-
 sys/ufs/ufs/ufs_inode.c                        |    8 +-
 sys/ufs/ufs/ufs_vnops.c                        |    8 +-
 sys/uvm/pmap/pmap.c                            |  178 +++++++-------
 sys/uvm/pmap/pmap_tlb.c                        |   39 +-
 sys/uvm/uvm_amap.c                             |   75 +++--
 sys/uvm/uvm_anon.c                             |   15 +-
 sys/uvm/uvm_aobj.c                             |   25 +-
 sys/uvm/uvm_bio.c                              |   34 +-
 sys/uvm/uvm_device.c                           |   27 +-
 sys/uvm/uvm_fault.c                            |   58 ++--
 sys/uvm/uvm_km.c                               |   23 +-
 sys/uvm/uvm_loan.c                             |   26 +-
 sys/uvm/uvm_map.c                              |  125 +++++-----
 sys/uvm/uvm_page.c                             |   10 +-
 sys/uvm/uvm_pager.c                            |   19 +-
 sys/uvm/uvm_pdaemon.c                          |    8 +-
 sys/uvm/uvm_swap.c                             |   61 ++--
 sys/uvm/uvm_vnode.c                            |   26 +-
 usr.bin/vmstat/vmstat.c                        |   28 +-
 54 files changed, 1899 insertions(+), 1617 deletions(-)

diffs (truncated from 11295 to 300 lines):

diff -r 33c9f54999bd -r c5b3cd49c5c1 share/man/man9/kernhist.9
--- a/share/man/man9/kernhist.9 Fri Oct 27 23:22:01 2017 +0000
+++ b/share/man/man9/kernhist.9 Sat Oct 28 00:37:11 2017 +0000
@@ -1,4 +1,4 @@
-.\"    $NetBSD: kernhist.9,v 1.4 2015/10/29 00:32:55 mrg Exp $
+.\"    $NetBSD: kernhist.9,v 1.5 2017/10/28 00:37:11 pgoyette Exp $
 .\"
 .\" Copyright (c) 2015 Matthew R. Green
 .\" All rights reserved.
@@ -26,7 +26,7 @@
 .\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 .\" SUCH DAMAGE.
 .\"
-.Dd October 28, 2015
+.Dd October 25, 2017
 .Dt KERNHIST 9
 .Os
 .Sh NAME
@@ -63,29 +63,61 @@
 extremely early in the kernel initialisation.
 It provides a simple restricted
 .Xr printf 3
-format syntax with a maximum of 4 arguments.
-The format must be a literal string that can be referenced later as it
-is not stored with the event, only a pointer to it.
+format syntax with a maximum of 4 arguments, each of type
+.Vt uintmax_t.
 .Pp
 .Cd options KERNHIST
 must be present in the kernel configuration to enable these functions and
 macros.
 .Pp
-Arguments that require additional dereferences, such as
+A kernel history is a fixed-size buffer of an either statically or dynamically
+allocated buffer that is used and read in a cycled basis.
+It includes the time an entry was made, the CPU from which the entry was
+recorded, the
+.Xr printf 3
+like format and length, the function name and length, the unique call count
+for this function, and the 4 arguments.
+.Pp
+The history data can be viewed using the
+.Fl U
+and
+.Fl u
+.Ar histname
+options to
+.Xr vmstat 1 ,
+or by the
+.Ic show kernhist
+command in
+.Xr ddb 4 .
+User-written programs can retrieve history data from the kernel using the
+.Xr sysctl 9
+variable kern.hist.histname.
+.Pp
+The format string must be a literal string that can be referenced later as it
+is not stored with the event, only a pointer to it.
+It should only contain conversion specifiers suitable for
+.Vt uintmax_t
+values, such as
+.Dq %jx ,
+.Dq %ju ,
+and
+.Dq %jo ,
+and address (pointer) arguments should be cast to
+.Vt uintptr_t
+to avoid compiler errors on architectures where pointers are smaller than
+.Vt uintmax_t
+integers.
+Format specifiers without a length modifier, and specifiers with length
+modifiers other than j, should not be used.
+.Pp
+Format specifiers that require additional dereferences of their
+corresponding arguments, such as
 .Dq %s ,
 will not work in
 .Xr vmstat 1 ,
-but will when called from
+but will work when called from
 .Xr ddb 4 .
 .Pp
-A kernel history is a fixed-size buffer of an either statically or dynamically
-allocated buffer that is used and read in a cycled basis.
-It includes the time an entry was made, the CPU that the entry was recorded
-from, the
-.Xr printf 3
-like format and length, the function name and length, the unique call count
-for this function, and the 4 argumnts.
-.Pp
 These macros provide access to most kernel history functionality:
 .Bl -tag -width 4n
 .It Fn KERNHIST_DECL name
@@ -118,12 +150,13 @@
 .It Fn KERNHIST_LOG name fmt arg0 arg1 arg2 arg3
 For the given kernel history
 .Fa name ,
-log the format and arguments in the history as a unique event.
+log the format string and arguments in the history as a unique event.
 .It Fn KERNHIST_CALLED name
 Declare a function as being called.
 Either this or
 .Fn KERNHIST_CALLARGS
-must be used near the function entry point.
+must be used near the function entry point to maintain the number of
+times the function has been called.
 .It Fn KERNHIST_CALLARGS name fmt arg0 arg1 arg2 arg3
 A frontend to
 .Fn KERNHIST_LOG
@@ -160,6 +193,9 @@
 .It KERNHIST_SCDEBUGHIST
 Include events from
 .Dq scdebughist .
+.It KERNHIST_BIOHIST
+Include events from
+.Dq biohist .
 .El
 .It Fn kernhist_print pr
 Print all the kernel histories to the kernel message buffer.
@@ -167,6 +203,19 @@
 .Fn pr
 argument is currently ignored.
 .El
+.Sh CODE REFERENCES
+The
+.Nm
+functionality is implemented within the files
+.Pa sys/sys/kernhist.h
+and
+.Pa sys/kern/kern_history.c .
+The former file contains the definitions of data structures used to export
+the
+.Nm data
+via the
+.Xr sysctl 9
+mechanism.
 .Sh SEE ALSO
 .Xr vmstat 1 ,
 .Xr usbdi 9 ,
@@ -176,6 +225,22 @@
 .\" add example here of code usage
 .\"
 .Sh HISTORY
+The uvm-specific version of the
+.Nm
+facility first appeared in
+.Nx 1.4 .
+The generalized version of
+.Nm
+appeared in
+.Nm 6.0 .
+The
+.Xr sysctl 9
+interface to
+.Nm
+was introduced in
+.Nx 8.0 .
+.Sh AUTHORS
+.An -nosplit
 .Nm
 was originally written by
 .An Charles D. Cranor
@@ -183,15 +248,19 @@
 .Xr uvm 9
 framework, under the name UVMHIST.
 .An Matthew R. Green
-generalised it into its current form to be available to non
+generalized it into its current form to be available to non
 .Xr uvm 9
 frameworks.
+.An Paul Goyette Aq Mt pgoyette%NetBSD.org@localhost
+provided the
+.Xr 9 sysctl
+interface.
 .Sh BUGS
-The restriction about using
+The restriction against using
 .Dq %s
 .Xr printf 3
-format strings could be reduced to literal strings (such as the table of
-system call names) if
+specifier in format strings could be reduced to literal strings (such as
+the table of system call names) if
 .Xr vmstat 1
 was extended to convert
 .Dq %s
@@ -205,6 +274,10 @@
 list of masks could be properly published and made available, and as such
 this function may be removed in a future release.
 .Pp
+In addition to a statically-defined set of kernel histories, it would be
+possible to allow modular code to register and unregister their own
+histories dynamically, when a module is loaded or unloaded.
+.Pp
 The
 .Fn kernhist_print
 function currently ignores its
diff -r 33c9f54999bd -r c5b3cd49c5c1 sys/arch/acorn26/acorn26/pmap.c
--- a/sys/arch/acorn26/acorn26/pmap.c   Fri Oct 27 23:22:01 2017 +0000
+++ b/sys/arch/acorn26/acorn26/pmap.c   Sat Oct 28 00:37:11 2017 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: pmap.c,v 1.38 2016/12/24 17:11:31 cherry Exp $ */
+/* $NetBSD: pmap.c,v 1.39 2017/10/28 00:37:12 pgoyette Exp $ */
 /*-
  * Copyright (c) 1997, 1998, 2000 Ben Harris
  * All rights reserved.
@@ -102,7 +102,7 @@
 
 #include <sys/param.h>
 
-__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.38 2016/12/24 17:11:31 cherry Exp $");
+__KERNEL_RCSID(0, "$NetBSD: pmap.c,v 1.39 2017/10/28 00:37:12 pgoyette Exp $");
 
 #include <sys/kernel.h> /* for cold */
 #include <sys/kmem.h>
@@ -556,7 +556,8 @@
        UVMHIST_FUNC("pv_get");
 
        UVMHIST_CALLED(pmaphist);
-       UVMHIST_LOG(pmaphist, "(pmap=%p, ppn=%d, lpn=%d)", pmap, ppn, lpn, 0);
+       UVMHIST_LOG(pmaphist, "(pmap=%#jx, ppn=%jd, lpn=%jd)",
+           (uintptr_t)pmap, ppn, lpn, 0);
        /* If the head entry's free use that. */
        pv = &pv_table[ppn];
        if (pv->pv_pmap == NULL) {
@@ -567,8 +568,8 @@
        /* If this mapping exists already, use that. */
        for (pv = pv; pv != NULL; pv = pv->pv_next)
                if (pv->pv_pmap == pmap && pv->pv_lpn == lpn) {
-                       UVMHIST_LOG(pmaphist, "<-- existing (pv=%p)",
-                           pv, 0, 0, 0);
+                       UVMHIST_LOG(pmaphist, "<-- existing (pv=%#jx)",
+                           (uintptr_t)pv, 0, 0, 0);
                        return pv;
                }
        /* Otherwise, allocate a new entry and link it in after the head. */
@@ -578,7 +579,7 @@
        pv->pv_next = pv_table[ppn].pv_next;
        pv_table[ppn].pv_next = pv;
        pmap->pm_stats.resident_count++;
-       UVMHIST_LOG(pmaphist, "<-- new (pv=%p)", pv, 0, 0, 0);
+       UVMHIST_LOG(pmaphist, "<-- new (pv=%#jx)", (uintptr_t)pv, 0, 0, 0);
        return pv;
 }
 
@@ -592,7 +593,8 @@
        UVMHIST_FUNC("pv_release");
 
        UVMHIST_CALLED(pmaphist);
-       UVMHIST_LOG(pmaphist, "(pmap=%p, ppn=%d, lpn=%d)", pmap, ppn, lpn, 0);
+       UVMHIST_LOG(pmaphist, "(pmap=%#jx, ppn=%jd, lpn=%jd)",
+           (uintptr_t)pmap, ppn, lpn, 0);
        pv = &pv_table[ppn];
        /*
         * If it is the first entry on the list, it is actually
@@ -603,13 +605,15 @@
        if (pmap == pv->pv_pmap && lpn == pv->pv_lpn) {
                npv = pv->pv_next;
                if (npv) {
-                       UVMHIST_LOG(pmaphist, "pv=%p; pull-up", pv, 0, 0, 0);
+                       UVMHIST_LOG(pmaphist, "pv=%#jx; pull-up",
+                           (uintptr_t)pv, 0, 0, 0);
                        /* Pull up first entry from chain. */
                        memcpy(pv, npv, offsetof(struct pv_entry, pv_pflags));
                        pv->pv_pmap->pm_entries[pv->pv_lpn] = pv;
                        pv_free(npv);
                } else {
-                       UVMHIST_LOG(pmaphist, "pv=%p; empty", pv, 0, 0, 0);
+                       UVMHIST_LOG(pmaphist, "pv=%#jx; empty",
+                           (uintptr_t)pv, 0, 0, 0);
                        memset(pv, 0, offsetof(struct pv_entry, pv_pflags));
                }
        } else {
@@ -619,7 +623,7 @@
                        pv = npv;
                }
                KASSERT(npv != NULL);
-               UVMHIST_LOG(pmaphist, "pv=%p; tail", pv, 0, 0, 0);
+               UVMHIST_LOG(pmaphist, "pv=%#jx; tail", (uintptr_t)pv, 0, 0, 0);
                pv->pv_next = npv->pv_next;
                pv_free(npv);
        }
@@ -658,8 +662,8 @@
        UVMHIST_CALLED(pmaphist);
        ppn = atop(pa); lpn = atop(va);
 
-       UVMHIST_LOG(pmaphist, "mapping ppn %d at lpn %d in pmap %p",
-              ppn, lpn, pmap, 0);
+       UVMHIST_LOG(pmaphist, "mapping ppn %jd at lpn %jd in pmap %#jx",
+              ppn, lpn, (uintptr_t)pmap, 0);
        s = splvm();
 
        /* Remove any existing mapping at this lpn */
@@ -714,8 +718,8 @@
 
        UVMHIST_CALLED(pmaphist);
        slpn = atop(sva); elpn = atop(eva);
-       UVMHIST_LOG(pmaphist, "clearing from lpn %d to lpn %d in pmap %p",
-              slpn, elpn - 1, pmap, 0);
+       UVMHIST_LOG(pmaphist, "clearing from lpn %jd to lpn %jd in pmap %#jx",



Home | Main Index | Thread Index | Old Index