Implement a much more capable vuprintf()

New support as well as some buggy support fixed.

Still no floating point support if ever that would be desired.

Support (*):
* Flags: '-', '+', ' ', '#', '0'

* Width and precision: 'n', '.n', '*' and '.*'

* Length modifiers: 'hh', 'h', 'j', 'l', 'll', 't', 'z'

* Radix: 'c', 'd', 'i', 'n', 'o', 'p/P', 's', 'u', 'x/X'

(*) Provision exists to switch lesser-used stuff on or off or when
certain functionality isn't desired (bootloader?). The compulsory
radixes are everything but 'o', 'n', 'p/P' and 'x/X' with length
modifiers being optional. The default setup is 'l', 'z', 'c', 'd',
'p/P', 's', 'u', 'x/X'.

* Move fdprintf() to its own file. It was in a strange place.

* Make callers compatible and fix a couple snprintf() bugs while
at it.

Could smush it down in size but I'm gonna get over the binsize
neurosis and just the let optimizer do its thing.

Change-Id: Ibdc613a9b6775802c188b29b9dd46c568c94f7c3
This commit is contained in:
Michael Sevakis 2017-09-08 19:28:02 -04:00
parent 52af55eee8
commit 5c9688961e
8 changed files with 1129 additions and 337 deletions

View file

@ -19,19 +19,27 @@
*
****************************************************************************/
#ifndef __FORMAT_H__
#define __FORMAT_H__
#ifndef __VUPRINTF_H__
#define __VUPRINTF_H__
void format(
/* call 'push()' for each output letter */
int (*push)(void *userp, unsigned char data),
void *userp,
const char *fmt,
va_list ap);
#include <stdarg.h>
/* callback function is called for every output character (byte) with userp and
* should return 0 when ch is a char other than '\0' that should stop printing */
void vuprintf(int (*push)(void *userp, unsigned char data),
void *userp, const char *fmt, va_list ap);
/* callback function is called for every output character (byte) in the
* output with userp
*
* it must return > 0 to continue (increments counter)
* it may return 0 to stop (increments counter)
* it may return < 0 to stop (does not increment counter)
* a zero in the format string stops (does not increment counter)
*
* caller is reponsible for stopping formatting in order to keep the return
* value from overflowing (assuming they have a reason to care)
*/
typedef int (* vuprintf_push_cb)(void *userp, int c);
#endif /* __FORMAT_H__ */
/*
* returns the number of times push() was called and returned >= 0
*/
int vuprintf(vuprintf_push_cb push, void *userp, const char *fmt, va_list ap);
#endif /* __VUPRINTF_H__ */