vsprintf: Recursive vsnprintf: Add "%pV", struct va_format
authorJoe Perches <joe@perches.com>
Sun, 27 Jun 2010 01:02:33 +0000 (01:02 +0000)
committerDavid S. Miller <davem@davemloft.net>
Sun, 4 Jul 2010 17:40:17 +0000 (10:40 -0700)
Add the ability to print a format and va_list from a structure pointer

Allows __dev_printk to be implemented as a single printk while
minimizing string space duplication.

%pV should not be used without some mechanism to verify the
format and argument use ala __attribute__(format (printf(...))).

Signed-off-by: Joe Perches <joe@perches.com>
Acked-by: Greg Kroah-Hartman <gregkh@suse.de>
Signed-off-by: David S. Miller <davem@davemloft.net>
include/linux/kernel.h
lib/vsprintf.c

index 8317ec4b9f3b4bbe615109073e7bd8ad9b946c1b..01dfc05ef4ac83eeb13eb0c7e4181c4e75fa32e7 100644 (file)
@@ -171,6 +171,11 @@ static inline void might_fault(void)
 }
 #endif
 
+struct va_format {
+       const char *fmt;
+       va_list *va;
+};
+
 extern struct atomic_notifier_head panic_notifier_list;
 extern long (*panic_blink)(long time);
 NORET_TYPE void panic(const char * fmt, ...)
index b8a2f549ab0ef5db70ecae07de8ea3be4d67a06d..4ee19d0d3910b36ad15f88688db46d5d1249fa61 100644 (file)
@@ -980,6 +980,11 @@ char *uuid_string(char *buf, char *end, const u8 *addr,
  *             [0][1][2][3]-[4][5]-[6][7]-[8][9]-[10][11][12][13][14][15]
  *           little endian output byte order is:
  *             [3][2][1][0]-[5][4]-[7][6]-[8][9]-[10][11][12][13][14][15]
+ * - 'V' For a struct va_format which contains a format string * and va_list *,
+ *       call vsnprintf(->format, *->va_list).
+ *       Implements a "recursive vsnprintf".
+ *       Do not use this feature without some mechanism to verify the
+ *       correctness of the format string and va_list arguments.
  *
  * Note: The difference between 'S' and 'F' is that on ia64 and ppc64
  * function pointers are really function descriptors, which contain a
@@ -1025,6 +1030,10 @@ char *pointer(const char *fmt, char *buf, char *end, void *ptr,
                break;
        case 'U':
                return uuid_string(buf, end, ptr, spec, fmt);
+       case 'V':
+               return buf + vsnprintf(buf, end - buf,
+                                      ((struct va_format *)ptr)->fmt,
+                                      *(((struct va_format *)ptr)->va));
        }
        spec.flags |= SMALL;
        if (spec.field_width == -1) {