X-Git-Url: https://pd.if.org/git/?a=blobdiff_plain;f=draft.c;h=570f305353b9352e07b51be7265c5ccb4e981ef7;hb=9b1e6c0a721f9358d9bbc76410e719a878e27edb;hp=5282fc1aed23e74cde4a40542ca1977e30371d81;hpb=9c937c3ef4747ec3f2f46a49a7ccf268112dab6a;p=pdclib diff --git a/draft.c b/draft.c index 5282fc1..570f305 100644 --- a/draft.c +++ b/draft.c @@ -25,9 +25,10 @@ #define E_intmax 1<<10 #define E_size 1<<11 #define E_ptrdiff 1<<12 -#define E_double 1<<13 -#define E_lower 1<<14 -#define E_unsigned 1<<15 +#define E_intptr 1<<13 +#define E_double 1<<14 +#define E_lower 1<<15 +#define E_unsigned 1<<16 struct status_t { @@ -45,7 +46,8 @@ struct status_t const char * parse_out( const char * spec, struct status_t * status ); inline void test( size_t n, const char * expect, ... ); -int _PDCLIB_sprintf( char * buffer, size_t n, const char * format, va_list ap ); +int _PDCLIB_vsnprintf( char * buffer, size_t n, const char * format, va_list ap ); +int _PDCLIB_snprintf( char * s, size_t n, const char * format, ... ); /* The following only for testing. */ #include @@ -84,6 +86,7 @@ int main( void ) test( SIZE_MAX, "%#x", -1u ); test( SIZE_MAX, "%o", UINT_MAX ); test( SIZE_MAX, "%#o", -1u ); + test( SIZE_MAX, "%.0#o", 0 ); test( SIZE_MAX, "%+d", INT_MIN ); test( SIZE_MAX, "%+d", INT_MAX ); test( SIZE_MAX, "%+d", 0 ); @@ -171,6 +174,19 @@ int main( void ) test( SIZE_MAX, "%+03.6d", INT_MAX ); test( SIZE_MAX, "- %d", INT_MAX ); test( SIZE_MAX, "- %d %% %d", INT_MAX, INT_MIN ); + test( SIZE_MAX, "%c", 'x' ); + test( SIZE_MAX, "%s", "abcdef" ); + test( SIZE_MAX, "%p", 0xdeadbeef ); + { + char buffer[50]; + int val1, val2, val3, val4; + snprintf( buffer, SIZE_MAX, "123456%n789%n", &val1, &val2 ); + _PDCLIB_snprintf( buffer, SIZE_MAX, "123456%n789%n", &val3, &val4 ); + if ( ( val1 != val3 ) || ( val2 != val4 ) ) + { + printf( "Output %d/%d\nExpect %d/%d\n\n", val1, val2, val3, val4 ); + } + } return 0; } @@ -451,6 +467,7 @@ const char * parse_out( const char * spec, struct status_t * status ) switch ( *spec ) { case 'd': + /* FALLTHROUGH */ case 'i': status->base = 10; break; @@ -481,13 +498,30 @@ const char * parse_out( const char * spec, struct status_t * status ) case 'A': break; case 'c': - break; + /* TODO: Flags, wide chars. */ + DELIVER( va_arg( status->ap, int ) ); + return ++spec; case 's': - break; + /* TODO: Flags, wide chars. */ + { + char * s = va_arg( status->ap, char * ); + while ( *s != '\0' ) + { + DELIVER( *(s++) ); + } + return ++spec; + } case 'p': - /* uint2base( 16, (intptr_t)value, true ) */ - case 'n': + /* TODO: E_long -> E_intptr */ + status->base = 16; + status->flags |= ( E_lower | E_unsigned | E_alt | E_long ); break; + case 'n': + { + int * val = va_arg( status->ap, int * ); + *val = status->i; + return ++spec; + } default: /* No conversion specifier. Bad conversion. */ return orig_spec; @@ -592,7 +626,7 @@ inline void test( size_t n, const char * expect, ... ) int rc; va_list ap; va_start( ap, expect ); - myrc = _PDCLIB_sprintf( buffer1, n, expect, ap ); + myrc = _PDCLIB_vsnprintf( buffer1, n, expect, ap ); rc = vsnprintf( buffer2, n, expect, ap ); if ( ( strcmp( buffer1, buffer2 ) != 0 ) || ( myrc != rc ) ) { @@ -602,7 +636,7 @@ inline void test( size_t n, const char * expect, ... ) free( buffer2 ); } -int _PDCLIB_sprintf( char * buffer, size_t n, const char * format, va_list ap ) +int _PDCLIB_vsnprintf( char * buffer, size_t n, const char * format, va_list ap ) { struct status_t status = { 0, 0, n, 0, 0, buffer, 0, 0, NULL, ap }; while ( *format != '\0' ) @@ -623,6 +657,13 @@ int _PDCLIB_sprintf( char * buffer, size_t n, const char * format, va_list ap ) return status.i; } +int _PDCLIB_snprintf( char * s, size_t n, const char * format, ... ) +{ + va_list ap; + va_start( ap, format ); + return _PDCLIB_vsnprintf( s, n, format, ap ); +} + #if 0 int _PDCLIB_fprintf( FILE * stream, const char * format, va_list ap ) {