162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * linux/lib/kasprintf.c 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 1991, 1992 Linus Torvalds 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#include <linux/stdarg.h> 962306a36Sopenharmony_ci#include <linux/export.h> 1062306a36Sopenharmony_ci#include <linux/slab.h> 1162306a36Sopenharmony_ci#include <linux/types.h> 1262306a36Sopenharmony_ci#include <linux/string.h> 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_ci/* Simplified asprintf. */ 1562306a36Sopenharmony_cichar *kvasprintf(gfp_t gfp, const char *fmt, va_list ap) 1662306a36Sopenharmony_ci{ 1762306a36Sopenharmony_ci unsigned int first, second; 1862306a36Sopenharmony_ci char *p; 1962306a36Sopenharmony_ci va_list aq; 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci va_copy(aq, ap); 2262306a36Sopenharmony_ci first = vsnprintf(NULL, 0, fmt, aq); 2362306a36Sopenharmony_ci va_end(aq); 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_ci p = kmalloc_track_caller(first+1, gfp); 2662306a36Sopenharmony_ci if (!p) 2762306a36Sopenharmony_ci return NULL; 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci second = vsnprintf(p, first+1, fmt, ap); 3062306a36Sopenharmony_ci WARN(first != second, "different return values (%u and %u) from vsnprintf(\"%s\", ...)", 3162306a36Sopenharmony_ci first, second, fmt); 3262306a36Sopenharmony_ci 3362306a36Sopenharmony_ci return p; 3462306a36Sopenharmony_ci} 3562306a36Sopenharmony_ciEXPORT_SYMBOL(kvasprintf); 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci/* 3862306a36Sopenharmony_ci * If fmt contains no % (or is exactly %s), use kstrdup_const. If fmt 3962306a36Sopenharmony_ci * (or the sole vararg) points to rodata, we will then save a memory 4062306a36Sopenharmony_ci * allocation and string copy. In any case, the return value should be 4162306a36Sopenharmony_ci * freed using kfree_const(). 4262306a36Sopenharmony_ci */ 4362306a36Sopenharmony_ciconst char *kvasprintf_const(gfp_t gfp, const char *fmt, va_list ap) 4462306a36Sopenharmony_ci{ 4562306a36Sopenharmony_ci if (!strchr(fmt, '%')) 4662306a36Sopenharmony_ci return kstrdup_const(fmt, gfp); 4762306a36Sopenharmony_ci if (!strcmp(fmt, "%s")) 4862306a36Sopenharmony_ci return kstrdup_const(va_arg(ap, const char*), gfp); 4962306a36Sopenharmony_ci return kvasprintf(gfp, fmt, ap); 5062306a36Sopenharmony_ci} 5162306a36Sopenharmony_ciEXPORT_SYMBOL(kvasprintf_const); 5262306a36Sopenharmony_ci 5362306a36Sopenharmony_cichar *kasprintf(gfp_t gfp, const char *fmt, ...) 5462306a36Sopenharmony_ci{ 5562306a36Sopenharmony_ci va_list ap; 5662306a36Sopenharmony_ci char *p; 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_ci va_start(ap, fmt); 5962306a36Sopenharmony_ci p = kvasprintf(gfp, fmt, ap); 6062306a36Sopenharmony_ci va_end(ap); 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_ci return p; 6362306a36Sopenharmony_ci} 6462306a36Sopenharmony_ciEXPORT_SYMBOL(kasprintf); 65