diff options
Diffstat (limited to 'lib/vsprintf.c')
| -rw-r--r-- | lib/vsprintf.c | 58 | 
1 files changed, 52 insertions, 6 deletions
diff --git a/lib/vsprintf.c b/lib/vsprintf.c index 7c488a1ce318..259e55895933 100644 --- a/lib/vsprintf.c +++ b/lib/vsprintf.c @@ -34,6 +34,7 @@  #include <linux/dcache.h>  #include <linux/cred.h>  #include <linux/rtc.h> +#include <linux/time.h>  #include <linux/uuid.h>  #include <linux/of.h>  #include <net/addrconf.h> @@ -58,7 +59,7 @@   * @endp: A pointer to the end of the parsed string will be placed here   * @base: The number base to use   * - * This function is obsolete. Please use kstrtoull instead. + * This function has caveats. Please use kstrtoull instead.   */  unsigned long long simple_strtoull(const char *cp, char **endp, unsigned int base)  { @@ -83,7 +84,7 @@ EXPORT_SYMBOL(simple_strtoull);   * @endp: A pointer to the end of the parsed string will be placed here   * @base: The number base to use   * - * This function is obsolete. Please use kstrtoul instead. + * This function has caveats. Please use kstrtoul instead.   */  unsigned long simple_strtoul(const char *cp, char **endp, unsigned int base)  { @@ -97,7 +98,7 @@ EXPORT_SYMBOL(simple_strtoul);   * @endp: A pointer to the end of the parsed string will be placed here   * @base: The number base to use   * - * This function is obsolete. Please use kstrtol instead. + * This function has caveats. Please use kstrtol instead.   */  long simple_strtol(const char *cp, char **endp, unsigned int base)  { @@ -114,7 +115,7 @@ EXPORT_SYMBOL(simple_strtol);   * @endp: A pointer to the end of the parsed string will be placed here   * @base: The number base to use   * - * This function is obsolete. Please use kstrtoll instead. + * This function has caveats. Please use kstrtoll instead.   */  long long simple_strtoll(const char *cp, char **endp, unsigned int base)  { @@ -794,6 +795,13 @@ static char *ptr_to_id(char *buf, char *end, const void *ptr,  	unsigned long hashval;  	int ret; +	/* +	 * Print the real pointer value for NULL and error pointers, +	 * as they are not actual addresses. +	 */ +	if (IS_ERR_OR_NULL(ptr)) +		return pointer_string(buf, end, ptr, spec); +  	/* When debugging early boot use non-cryptographically secure hash. */  	if (unlikely(debug_boot_weak_hash)) {  		hashval = hash_long((unsigned long)ptr, 32); @@ -1820,14 +1828,39 @@ char *rtc_str(char *buf, char *end, const struct rtc_time *tm,  }  static noinline_for_stack +char *time64_str(char *buf, char *end, const time64_t time, +		 struct printf_spec spec, const char *fmt) +{ +	struct rtc_time rtc_time; +	struct tm tm; + +	time64_to_tm(time, 0, &tm); + +	rtc_time.tm_sec = tm.tm_sec; +	rtc_time.tm_min = tm.tm_min; +	rtc_time.tm_hour = tm.tm_hour; +	rtc_time.tm_mday = tm.tm_mday; +	rtc_time.tm_mon = tm.tm_mon; +	rtc_time.tm_year = tm.tm_year; +	rtc_time.tm_wday = tm.tm_wday; +	rtc_time.tm_yday = tm.tm_yday; + +	rtc_time.tm_isdst = 0; + +	return rtc_str(buf, end, &rtc_time, spec, fmt); +} + +static noinline_for_stack  char *time_and_date(char *buf, char *end, void *ptr, struct printf_spec spec,  		    const char *fmt)  {  	switch (fmt[1]) {  	case 'R':  		return rtc_str(buf, end, (const struct rtc_time *)ptr, spec, fmt); +	case 'T': +		return time64_str(buf, end, *(const time64_t *)ptr, spec, fmt);  	default: -		return error_string(buf, end, "(%ptR?)", spec); +		return error_string(buf, end, "(%pt?)", spec);  	}  } @@ -2143,8 +2176,9 @@ char *fwnode_string(char *buf, char *end, struct fwnode_handle *fwnode,   * - 'd[234]' For a dentry name (optionally 2-4 last components)   * - 'D[234]' Same as 'd' but for a struct file   * - 'g' For block_device name (gendisk + partition number) - * - 't[R][dt][r]' For time and date as represented: + * - 't[RT][dt][r]' For time and date as represented by:   *      R    struct rtc_time + *      T    time64_t   * - 'C' For a clock, it prints the name (Common Clock Framework) or address   *       (legacy clock framework) of the clock   * - 'Cn' For a clock, it prints the name (Common Clock Framework) or address @@ -2168,6 +2202,10 @@ char *fwnode_string(char *buf, char *end, struct fwnode_handle *fwnode,   *		f full name   *		P node name, including a possible unit address   * - 'x' For printing the address. Equivalent to "%lx". + * - '[ku]s' For a BPF/tracing related format specifier, e.g. used out of + *           bpf_trace_printk() where [ku] prefix specifies either kernel (k) + *           or user (u) memory to probe, and: + *              s a string, equivalent to "%s" on direct vsnprintf() use   *   * ** When making changes please also update:   *	Documentation/core-api/printk-formats.rst @@ -2251,6 +2289,14 @@ char *pointer(const char *fmt, char *buf, char *end, void *ptr,  		if (!IS_ERR(ptr))  			break;  		return err_ptr(buf, end, ptr, spec); +	case 'u': +	case 'k': +		switch (fmt[1]) { +		case 's': +			return string(buf, end, ptr, spec); +		default: +			return error_string(buf, end, "(einval)", spec); +		}  	}  	/* default is to _not_ leak addresses, hash before printing */  | 
