84 lines
3.2 KiB
C
84 lines
3.2 KiB
C
#ifndef COSMOPOLITAN_LIBC_FMT_PFLINK_H_
|
|
#define COSMOPOLITAN_LIBC_FMT_PFLINK_H_
|
|
#include "libc/dce.h"
|
|
#if !(__ASSEMBLER__ + __LINKER__ + 0)
|
|
#ifndef __STRICT_ANSI__
|
|
|
|
/**
|
|
* @fileoverview builtin+preprocessor+linker tricks for printf/scanf.
|
|
*
|
|
* Your printf() function only requires that you pay for what you use.
|
|
* These macros ensure that its code size starts at under 4kb, growing
|
|
* to about 40kb for a fully-loaded implementation. This works best when
|
|
* format strings are constexprs that only contain directives.
|
|
*/
|
|
|
|
#define PFLINK(FMT) \
|
|
({ \
|
|
if (___PFLINK(FMT, strpbrk, "bxdinupo")) STATIC_YOINK("ntoa"); \
|
|
if (___PFLINK(FMT, strpbrk, "fFgGaA")) STATIC_YOINK("ftoa"); \
|
|
if (___PFLINK(FMT, strpbrk, "cmrqs")) { \
|
|
STATIC_YOINK("stoa"); \
|
|
if (___PFLINK(FMT, strchr, '#')) STATIC_YOINK("kCp437"); \
|
|
if (___PFLINK(FMT, strstr, "%m")) STATIC_YOINK("strerror"); \
|
|
if (!IsTiny() && (___PFLINK(FMT, strstr, "%*") || \
|
|
___PFLINK(FMT, strpbrk, "0123456789"))) { \
|
|
STATIC_YOINK("strnwidth"); \
|
|
STATIC_YOINK("strnwidth16"); \
|
|
STATIC_YOINK("wcsnwidth"); \
|
|
} \
|
|
} \
|
|
FMT; \
|
|
})
|
|
|
|
#define SFLINK(FMT) \
|
|
({ \
|
|
if (___PFLINK(FMT, strchr, 'm')) { \
|
|
STATIC_YOINK("malloc"); \
|
|
STATIC_YOINK("calloc"); \
|
|
STATIC_YOINK("free_s"); \
|
|
STATIC_YOINK("__grow"); \
|
|
} \
|
|
FMT; \
|
|
})
|
|
|
|
#if __GNUC__ + 0 < 4 || defined(__llvm__)
|
|
#define ___PFLINK(FMT, FN, C) 1
|
|
#else
|
|
#define ___PFLINK(FMT, FN, C) \
|
|
!isconstant(FMT) || ((FMT) && __builtin_##FN(FMT, C) != NULL)
|
|
#endif
|
|
|
|
#if defined(__GNUC__) && __GNUC__ < 6
|
|
/*
|
|
* Compilers don't understand the features we've added to the format
|
|
* string DSL, such as c string escaping, therefore we can't use it.
|
|
* Ideally compilers should grant us more flexibility to define DSLs
|
|
*
|
|
* The recommended approach to turning this back on is `CFLAGS=-std=c11`
|
|
* which puts the compiler in __STRICT_ANSI__ mode, which Cosmopolitan
|
|
* respects by disabling all the esoteric tuning in headers like this.
|
|
*/
|
|
#pragma GCC diagnostic ignored "-Wformat-security"
|
|
#endif /* __GNUC__ + 0 < 6 */
|
|
|
|
#else
|
|
#define PFLINK(FMT) FMT
|
|
#define SFLINK(FMT) FMT
|
|
asm(".pushsection .yoink\n\t"
|
|
"nop\tntoa(%rip)\n\t"
|
|
"nop\tftoa(%rip)\n\t"
|
|
"nop\tkCp437(%rip)\n\t"
|
|
"nop\tstrerror(%rip)\n\t"
|
|
"nop\tstrnwidth(%rip)\n\t"
|
|
"nop\tstrnwidth16(%rip)\n\t"
|
|
"nop\twcsnwidth(%rip)\n\t"
|
|
"nop\tmalloc(%rip)\n\t"
|
|
"nop\tcalloc(%rip)\n\t"
|
|
"nop\tfree_s(%rip)\n\t"
|
|
"nop\t__grow(%rip)\n\t"
|
|
".popsection");
|
|
#endif /* __STRICT_ANSI__ */
|
|
#endif /* !(__ASSEMBLER__ + __LINKER__ + 0) */
|
|
#endif /* COSMOPOLITAN_LIBC_FMT_PFLINK_H_ */
|