mirror of
				https://github.com/ruby/ruby.git
				synced 2022-11-09 12:17:21 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			1250 lines
		
	
	
	
		
			29 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1250 lines
		
	
	
	
		
			29 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /**********************************************************************
 | |
| 
 | |
|   sprintf.c -
 | |
| 
 | |
|   $Author$
 | |
|   created at: Fri Oct 15 10:39:26 JST 1993
 | |
| 
 | |
|   Copyright (C) 1993-2007 Yukihiro Matsumoto
 | |
|   Copyright (C) 2000  Network Applied Communication Laboratory, Inc.
 | |
|   Copyright (C) 2000  Information-technology Promotion Agency, Japan
 | |
| 
 | |
| **********************************************************************/
 | |
| 
 | |
| #include "ruby/internal/config.h"
 | |
| 
 | |
| #include <math.h>
 | |
| #include <stdarg.h>
 | |
| 
 | |
| #ifdef HAVE_IEEEFP_H
 | |
| # include <ieeefp.h>
 | |
| #endif
 | |
| 
 | |
| #include "id.h"
 | |
| #include "internal.h"
 | |
| #include "internal/error.h"
 | |
| #include "internal/hash.h"
 | |
| #include "internal/numeric.h"
 | |
| #include "internal/object.h"
 | |
| #include "internal/sanitizers.h"
 | |
| #include "internal/symbol.h"
 | |
| #include "ruby/encoding.h"
 | |
| #include "ruby/re.h"
 | |
| #include "ruby/util.h"
 | |
| 
 | |
| #define BIT_DIGITS(N)   (((N)*146)/485 + 1)  /* log2(10) =~ 146/485 */
 | |
| 
 | |
| static char *fmt_setup(char*,size_t,int,int,int,int);
 | |
| static char *ruby_ultoa(unsigned long val, char *endp, int base, int octzero);
 | |
| 
 | |
| static char
 | |
| sign_bits(int base, const char *p)
 | |
| {
 | |
|     char c = '.';
 | |
| 
 | |
|     switch (base) {
 | |
|       case 16:
 | |
| 	if (*p == 'X') c = 'F';
 | |
| 	else c = 'f';
 | |
| 	break;
 | |
|       case 8:
 | |
| 	c = '7'; break;
 | |
|       case 2:
 | |
| 	c = '1'; break;
 | |
|     }
 | |
|     return c;
 | |
| }
 | |
| 
 | |
| #define FNONE  0
 | |
| #define FSHARP 1
 | |
| #define FMINUS 2
 | |
| #define FPLUS  4
 | |
| #define FZERO  8
 | |
| #define FSPACE 16
 | |
| #define FWIDTH 32
 | |
| #define FPREC  64
 | |
| #define FPREC0 128
 | |
| 
 | |
| #define CHECK(l) do {\
 | |
|     int cr = ENC_CODERANGE(result);\
 | |
|     while ((l) >= bsiz - blen) {\
 | |
| 	bsiz*=2;\
 | |
| 	if (bsiz<0) rb_raise(rb_eArgError, "too big specifier");\
 | |
|     }\
 | |
|     rb_str_resize(result, bsiz);\
 | |
|     ENC_CODERANGE_SET(result, cr);\
 | |
|     buf = RSTRING_PTR(result);\
 | |
| } while (0)
 | |
| 
 | |
| #define PUSH(s, l) do { \
 | |
|     CHECK(l);\
 | |
|     PUSH_(s, l);\
 | |
| } while (0)
 | |
| 
 | |
| #define PUSH_(s, l) do { \
 | |
|     memcpy(&buf[blen], (s), (l));\
 | |
|     blen += (l);\
 | |
| } while (0)
 | |
| 
 | |
| #define FILL(c, l) do { \
 | |
|     if ((l) <= 0) break;\
 | |
|     CHECK(l);\
 | |
|     FILL_(c, l);\
 | |
| } while (0)
 | |
| 
 | |
| #define FILL_(c, l) do { \
 | |
|     memset(&buf[blen], (c), (l));\
 | |
|     blen += (l);\
 | |
| } while (0)
 | |
| 
 | |
| #define GETARG() (nextvalue != Qundef ? nextvalue : \
 | |
| 		  GETNEXTARG())
 | |
| 
 | |
| #define GETNEXTARG() ( \
 | |
|     check_next_arg(posarg, nextarg), \
 | |
|     (posarg = nextarg++, GETNTHARG(posarg)))
 | |
| 
 | |
| #define GETPOSARG(n) ( \
 | |
|     check_pos_arg(posarg, (n)), \
 | |
|     (posarg = -1, GETNTHARG(n)))
 | |
| 
 | |
| #define GETNTHARG(nth) \
 | |
|     (((nth) >= argc) ? (rb_raise(rb_eArgError, "too few arguments"), 0) : argv[(nth)])
 | |
| 
 | |
| #define CHECKNAMEARG(name, len, enc) ( \
 | |
|     check_name_arg(posarg, name, len, enc), \
 | |
|     posarg = -2)
 | |
| 
 | |
| #define GETNUM(n, val) \
 | |
|     (!(p = get_num(p, end, enc, &(n))) ? \
 | |
|      rb_raise(rb_eArgError, #val " too big") : (void)0)
 | |
| 
 | |
| #define GETASTER(val) do { \
 | |
|     t = p++; \
 | |
|     n = 0; \
 | |
|     GETNUM(n, val); \
 | |
|     if (*p == '$') { \
 | |
| 	tmp = GETPOSARG(n); \
 | |
|     } \
 | |
|     else { \
 | |
| 	tmp = GETNEXTARG(); \
 | |
| 	p = t; \
 | |
|     } \
 | |
|     (val) = NUM2INT(tmp); \
 | |
| } while (0)
 | |
| 
 | |
| static const char *
 | |
| get_num(const char *p, const char *end, rb_encoding *enc, int *valp)
 | |
| {
 | |
|     int next_n = *valp;
 | |
|     for (; p < end && rb_enc_isdigit(*p, enc); p++) {
 | |
| 	if (MUL_OVERFLOW_INT_P(10, next_n))
 | |
| 	    return NULL;
 | |
| 	next_n *= 10;
 | |
| 	if (INT_MAX - (*p - '0') < next_n)
 | |
| 	    return NULL;
 | |
| 	next_n += *p - '0';
 | |
|     }
 | |
|     if (p >= end) {
 | |
| 	rb_raise(rb_eArgError, "malformed format string - %%*[0-9]");
 | |
|     }
 | |
|     *valp = next_n;
 | |
|     return p;
 | |
| }
 | |
| 
 | |
| static void
 | |
| check_next_arg(int posarg, int nextarg)
 | |
| {
 | |
|     switch (posarg) {
 | |
|       case -1:
 | |
| 	rb_raise(rb_eArgError, "unnumbered(%d) mixed with numbered", nextarg);
 | |
|       case -2:
 | |
| 	rb_raise(rb_eArgError, "unnumbered(%d) mixed with named", nextarg);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| check_pos_arg(int posarg, int n)
 | |
| {
 | |
|     if (posarg > 0) {
 | |
| 	rb_raise(rb_eArgError, "numbered(%d) after unnumbered(%d)", n, posarg);
 | |
|     }
 | |
|     if (posarg == -2) {
 | |
| 	rb_raise(rb_eArgError, "numbered(%d) after named", n);
 | |
|     }
 | |
|     if (n < 1) {
 | |
| 	rb_raise(rb_eArgError, "invalid index - %d$", n);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| check_name_arg(int posarg, const char *name, int len, rb_encoding *enc)
 | |
| {
 | |
|     if (posarg > 0) {
 | |
| 	rb_enc_raise(enc, rb_eArgError, "named%.*s after unnumbered(%d)", len, name, posarg);
 | |
|     }
 | |
|     if (posarg == -1) {
 | |
| 	rb_enc_raise(enc, rb_eArgError, "named%.*s after numbered", len, name);
 | |
|     }
 | |
| }
 | |
| 
 | |
| static VALUE
 | |
| get_hash(volatile VALUE *hash, int argc, const VALUE *argv)
 | |
| {
 | |
|     VALUE tmp;
 | |
| 
 | |
|     if (*hash != Qundef) return *hash;
 | |
|     if (argc != 2) {
 | |
| 	rb_raise(rb_eArgError, "one hash required");
 | |
|     }
 | |
|     tmp = rb_check_hash_type(argv[1]);
 | |
|     if (NIL_P(tmp)) {
 | |
| 	rb_raise(rb_eArgError, "one hash required");
 | |
|     }
 | |
|     return (*hash = tmp);
 | |
| }
 | |
| 
 | |
| VALUE
 | |
| rb_f_sprintf(int argc, const VALUE *argv)
 | |
| {
 | |
|     return rb_str_format(argc - 1, argv + 1, GETNTHARG(0));
 | |
| }
 | |
| 
 | |
| VALUE
 | |
| rb_str_format(int argc, const VALUE *argv, VALUE fmt)
 | |
| {
 | |
|     enum {default_float_precision = 6};
 | |
|     rb_encoding *enc;
 | |
|     const char *p, *end;
 | |
|     char *buf;
 | |
|     long blen, bsiz;
 | |
|     VALUE result;
 | |
| 
 | |
|     long scanned = 0;
 | |
|     int coderange = ENC_CODERANGE_7BIT;
 | |
|     int width, prec, flags = FNONE;
 | |
|     int nextarg = 1;
 | |
|     int posarg = 0;
 | |
|     VALUE nextvalue;
 | |
|     VALUE tmp;
 | |
|     VALUE orig;
 | |
|     VALUE str;
 | |
|     volatile VALUE hash = Qundef;
 | |
| 
 | |
| #define CHECK_FOR_WIDTH(f)				 \
 | |
|     if ((f) & FWIDTH) {					 \
 | |
| 	rb_raise(rb_eArgError, "width given twice");	 \
 | |
|     }							 \
 | |
|     if ((f) & FPREC0) {					 \
 | |
| 	rb_raise(rb_eArgError, "width after precision"); \
 | |
|     }
 | |
| #define CHECK_FOR_FLAGS(f)				 \
 | |
|     if ((f) & FWIDTH) {					 \
 | |
| 	rb_raise(rb_eArgError, "flag after width");	 \
 | |
|     }							 \
 | |
|     if ((f) & FPREC0) {					 \
 | |
| 	rb_raise(rb_eArgError, "flag after precision"); \
 | |
|     }
 | |
| 
 | |
|     ++argc;
 | |
|     --argv;
 | |
|     StringValue(fmt);
 | |
|     enc = rb_enc_get(fmt);
 | |
|     orig = fmt;
 | |
|     fmt = rb_str_tmp_frozen_acquire(fmt);
 | |
|     p = RSTRING_PTR(fmt);
 | |
|     end = p + RSTRING_LEN(fmt);
 | |
|     blen = 0;
 | |
|     bsiz = 120;
 | |
|     result = rb_str_buf_new(bsiz);
 | |
|     rb_enc_associate(result, enc);
 | |
|     buf = RSTRING_PTR(result);
 | |
|     memset(buf, 0, bsiz);
 | |
|     ENC_CODERANGE_SET(result, coderange);
 | |
| 
 | |
|     for (; p < end; p++) {
 | |
| 	const char *t;
 | |
| 	int n;
 | |
| 	VALUE sym = Qnil;
 | |
| 
 | |
| 	for (t = p; t < end && *t != '%'; t++) ;
 | |
| 	if (t + 1 == end) {
 | |
| 	    rb_raise(rb_eArgError, "incomplete format specifier; use %%%% (double %%) instead");
 | |
| 	}
 | |
| 	PUSH(p, t - p);
 | |
| 	if (coderange != ENC_CODERANGE_BROKEN && scanned < blen) {
 | |
| 	    scanned += rb_str_coderange_scan_restartable(buf+scanned, buf+blen, enc, &coderange);
 | |
| 	    ENC_CODERANGE_SET(result, coderange);
 | |
| 	}
 | |
| 	if (t >= end) {
 | |
| 	    /* end of fmt string */
 | |
| 	    goto sprint_exit;
 | |
| 	}
 | |
| 	p = t + 1;		/* skip `%' */
 | |
| 
 | |
| 	width = prec = -1;
 | |
| 	nextvalue = Qundef;
 | |
|       retry:
 | |
| 	switch (*p) {
 | |
| 	  default:
 | |
| 	    if (rb_enc_isprint(*p, enc))
 | |
| 		rb_raise(rb_eArgError, "malformed format string - %%%c", *p);
 | |
| 	    else
 | |
| 		rb_raise(rb_eArgError, "malformed format string");
 | |
| 	    break;
 | |
| 
 | |
| 	  case ' ':
 | |
| 	    CHECK_FOR_FLAGS(flags);
 | |
| 	    flags |= FSPACE;
 | |
| 	    p++;
 | |
| 	    goto retry;
 | |
| 
 | |
| 	  case '#':
 | |
| 	    CHECK_FOR_FLAGS(flags);
 | |
| 	    flags |= FSHARP;
 | |
| 	    p++;
 | |
| 	    goto retry;
 | |
| 
 | |
| 	  case '+':
 | |
| 	    CHECK_FOR_FLAGS(flags);
 | |
| 	    flags |= FPLUS;
 | |
| 	    p++;
 | |
| 	    goto retry;
 | |
| 
 | |
| 	  case '-':
 | |
| 	    CHECK_FOR_FLAGS(flags);
 | |
| 	    flags |= FMINUS;
 | |
| 	    p++;
 | |
| 	    goto retry;
 | |
| 
 | |
| 	  case '0':
 | |
| 	    CHECK_FOR_FLAGS(flags);
 | |
| 	    flags |= FZERO;
 | |
| 	    p++;
 | |
| 	    goto retry;
 | |
| 
 | |
| 	  case '1': case '2': case '3': case '4':
 | |
| 	  case '5': case '6': case '7': case '8': case '9':
 | |
| 	    n = 0;
 | |
| 	    GETNUM(n, width);
 | |
| 	    if (*p == '$') {
 | |
| 		if (nextvalue != Qundef) {
 | |
| 		    rb_raise(rb_eArgError, "value given twice - %d$", n);
 | |
| 		}
 | |
| 		nextvalue = GETPOSARG(n);
 | |
| 		p++;
 | |
| 		goto retry;
 | |
| 	    }
 | |
| 	    CHECK_FOR_WIDTH(flags);
 | |
| 	    width = n;
 | |
| 	    flags |= FWIDTH;
 | |
| 	    goto retry;
 | |
| 
 | |
| 	  case '<':
 | |
| 	  case '{':
 | |
| 	    {
 | |
| 		const char *start = p;
 | |
| 		char term = (*p == '<') ? '>' : '}';
 | |
| 		int len;
 | |
| 
 | |
| 		for (; p < end && *p != term; ) {
 | |
| 		    p += rb_enc_mbclen(p, end, enc);
 | |
| 		}
 | |
| 		if (p >= end) {
 | |
| 		    rb_raise(rb_eArgError, "malformed name - unmatched parenthesis");
 | |
| 		}
 | |
| #if SIZEOF_INT < SIZEOF_SIZE_T
 | |
| 		if ((size_t)(p - start) >= INT_MAX) {
 | |
| 		    const int message_limit = 20;
 | |
| 		    len = (int)(rb_enc_right_char_head(start, start + message_limit, p, enc) - start);
 | |
| 		    rb_enc_raise(enc, rb_eArgError,
 | |
| 				 "too long name (%"PRIuSIZE" bytes) - %.*s...%c",
 | |
| 				 (size_t)(p - start - 2), len, start, term);
 | |
| 		}
 | |
| #endif
 | |
| 		len = (int)(p - start + 1); /* including parenthesis */
 | |
| 		if (sym != Qnil) {
 | |
| 		    rb_enc_raise(enc, rb_eArgError, "named%.*s after <%"PRIsVALUE">",
 | |
| 				 len, start, rb_sym2str(sym));
 | |
| 		}
 | |
| 		CHECKNAMEARG(start, len, enc);
 | |
| 		get_hash(&hash, argc, argv);
 | |
| 		sym = rb_check_symbol_cstr(start + 1,
 | |
| 					   len - 2 /* without parenthesis */,
 | |
| 					   enc);
 | |
| 		if (!NIL_P(sym)) nextvalue = rb_hash_lookup2(hash, sym, Qundef);
 | |
| 		if (nextvalue == Qundef) {
 | |
| 		    if (NIL_P(sym)) {
 | |
| 			sym = rb_sym_intern(start + 1,
 | |
| 					    len - 2 /* without parenthesis */,
 | |
| 					    enc);
 | |
| 		    }
 | |
| 		    nextvalue = rb_hash_default_value(hash, sym);
 | |
| 		    if (NIL_P(nextvalue)) {
 | |
| 			rb_key_err_raise(rb_enc_sprintf(enc, "key%.*s not found", len, start), hash, sym);
 | |
| 		    }
 | |
| 		}
 | |
| 		if (term == '}') goto format_s;
 | |
| 		p++;
 | |
| 		goto retry;
 | |
| 	    }
 | |
| 
 | |
| 	  case '*':
 | |
| 	    CHECK_FOR_WIDTH(flags);
 | |
| 	    flags |= FWIDTH;
 | |
| 	    GETASTER(width);
 | |
| 	    if (width < 0) {
 | |
| 		flags |= FMINUS;
 | |
| 		width = -width;
 | |
| 		if (width < 0) rb_raise(rb_eArgError, "width too big");
 | |
| 	    }
 | |
| 	    p++;
 | |
| 	    goto retry;
 | |
| 
 | |
| 	  case '.':
 | |
| 	    if (flags & FPREC0) {
 | |
| 		rb_raise(rb_eArgError, "precision given twice");
 | |
| 	    }
 | |
| 	    flags |= FPREC|FPREC0;
 | |
| 
 | |
| 	    prec = 0;
 | |
| 	    p++;
 | |
| 	    if (*p == '*') {
 | |
| 		GETASTER(prec);
 | |
| 		if (prec < 0) {	/* ignore negative precision */
 | |
| 		    flags &= ~FPREC;
 | |
| 		}
 | |
| 		p++;
 | |
| 		goto retry;
 | |
| 	    }
 | |
| 
 | |
| 	    GETNUM(prec, precision);
 | |
| 	    goto retry;
 | |
| 
 | |
| 	  case '\n':
 | |
| 	  case '\0':
 | |
| 	    p--;
 | |
|             /* fall through */
 | |
| 	  case '%':
 | |
| 	    if (flags != FNONE) {
 | |
| 		rb_raise(rb_eArgError, "invalid format character - %%");
 | |
| 	    }
 | |
| 	    PUSH("%", 1);
 | |
| 	    break;
 | |
| 
 | |
| 	  case 'c':
 | |
| 	    {
 | |
| 		VALUE val = GETARG();
 | |
| 		VALUE tmp;
 | |
| 		unsigned int c;
 | |
| 		int n;
 | |
| 
 | |
| 		tmp = rb_check_string_type(val);
 | |
| 		if (!NIL_P(tmp)) {
 | |
| 		    if (rb_enc_strlen(RSTRING_PTR(tmp),RSTRING_END(tmp),enc) != 1) {
 | |
| 			rb_raise(rb_eArgError, "%%c requires a character");
 | |
| 		    }
 | |
| 		    c = rb_enc_codepoint_len(RSTRING_PTR(tmp), RSTRING_END(tmp), &n, enc);
 | |
| 		    RB_GC_GUARD(tmp);
 | |
| 		}
 | |
| 		else {
 | |
| 		    c = NUM2INT(val);
 | |
| 		    n = rb_enc_codelen(c, enc);
 | |
| 		}
 | |
| 		if (n <= 0) {
 | |
| 		    rb_raise(rb_eArgError, "invalid character");
 | |
| 		}
 | |
| 		if (!(flags & FWIDTH)) {
 | |
| 		    CHECK(n);
 | |
| 		    rb_enc_mbcput(c, &buf[blen], enc);
 | |
| 		    blen += n;
 | |
| 		}
 | |
| 		else if ((flags & FMINUS)) {
 | |
| 		    CHECK(n);
 | |
| 		    rb_enc_mbcput(c, &buf[blen], enc);
 | |
| 		    blen += n;
 | |
| 		    if (width > 1) FILL(' ', width-1);
 | |
| 		}
 | |
| 		else {
 | |
| 		    if (width > 1) FILL(' ', width-1);
 | |
| 		    CHECK(n);
 | |
| 		    rb_enc_mbcput(c, &buf[blen], enc);
 | |
| 		    blen += n;
 | |
| 		}
 | |
| 	    }
 | |
| 	    break;
 | |
| 
 | |
| 	  case 's':
 | |
| 	  case 'p':
 | |
| 	  format_s:
 | |
| 	    {
 | |
| 		VALUE arg = GETARG();
 | |
| 		long len, slen;
 | |
| 
 | |
| 		if (*p == 'p') {
 | |
| 		    str = rb_inspect(arg);
 | |
| 		}
 | |
| 		else {
 | |
| 		    str = rb_obj_as_string(arg);
 | |
| 		}
 | |
| 		len = RSTRING_LEN(str);
 | |
| 		rb_str_set_len(result, blen);
 | |
| 		if (coderange != ENC_CODERANGE_BROKEN && scanned < blen) {
 | |
| 		    int cr = coderange;
 | |
| 		    scanned += rb_str_coderange_scan_restartable(buf+scanned, buf+blen, enc, &cr);
 | |
| 		    ENC_CODERANGE_SET(result,
 | |
| 				      (cr == ENC_CODERANGE_UNKNOWN ?
 | |
| 				       ENC_CODERANGE_BROKEN : (coderange = cr)));
 | |
| 		}
 | |
| 		enc = rb_enc_check(result, str);
 | |
| 		if (flags&(FPREC|FWIDTH)) {
 | |
| 		    slen = rb_enc_strlen(RSTRING_PTR(str),RSTRING_END(str),enc);
 | |
| 		    if (slen < 0) {
 | |
| 			rb_raise(rb_eArgError, "invalid mbstring sequence");
 | |
| 		    }
 | |
| 		    if ((flags&FPREC) && (prec < slen)) {
 | |
| 			char *p = rb_enc_nth(RSTRING_PTR(str), RSTRING_END(str),
 | |
| 					     prec, enc);
 | |
| 			slen = prec;
 | |
| 			len = p - RSTRING_PTR(str);
 | |
| 		    }
 | |
| 		    /* need to adjust multi-byte string pos */
 | |
| 		    if ((flags&FWIDTH) && (width > slen)) {
 | |
| 			width -= (int)slen;
 | |
| 			if (!(flags&FMINUS)) {
 | |
| 			    FILL(' ', width);
 | |
| 			    width = 0;
 | |
| 			}
 | |
| 			CHECK(len);
 | |
| 			memcpy(&buf[blen], RSTRING_PTR(str), len);
 | |
| 			RB_GC_GUARD(str);
 | |
| 			blen += len;
 | |
| 			if (flags&FMINUS) {
 | |
| 			    FILL(' ', width);
 | |
| 			}
 | |
| 			rb_enc_associate(result, enc);
 | |
| 			break;
 | |
| 		    }
 | |
| 		}
 | |
| 		PUSH(RSTRING_PTR(str), len);
 | |
| 		RB_GC_GUARD(str);
 | |
| 		rb_enc_associate(result, enc);
 | |
| 	    }
 | |
| 	    break;
 | |
| 
 | |
| 	  case 'd':
 | |
| 	  case 'i':
 | |
| 	  case 'o':
 | |
| 	  case 'x':
 | |
| 	  case 'X':
 | |
| 	  case 'b':
 | |
| 	  case 'B':
 | |
| 	  case 'u':
 | |
| 	    {
 | |
| 		volatile VALUE val = GETARG();
 | |
|                 int valsign;
 | |
| 		char nbuf[BIT_DIGITS(SIZEOF_LONG*CHAR_BIT)+2], *s;
 | |
| 		const char *prefix = 0;
 | |
| 		int sign = 0, dots = 0;
 | |
| 		char sc = 0;
 | |
| 		long v = 0;
 | |
| 		int base, bignum = 0;
 | |
| 		int len;
 | |
| 
 | |
| 		switch (*p) {
 | |
| 		  case 'd':
 | |
| 		  case 'i':
 | |
| 		  case 'u':
 | |
| 		    sign = 1; break;
 | |
| 		  case 'o':
 | |
| 		  case 'x':
 | |
| 		  case 'X':
 | |
| 		  case 'b':
 | |
| 		  case 'B':
 | |
| 		    if (flags&(FPLUS|FSPACE)) sign = 1;
 | |
| 		    break;
 | |
| 		}
 | |
| 		if (flags & FSHARP) {
 | |
| 		    switch (*p) {
 | |
| 		      case 'o':
 | |
| 			prefix = "0"; break;
 | |
| 		      case 'x':
 | |
| 			prefix = "0x"; break;
 | |
| 		      case 'X':
 | |
| 			prefix = "0X"; break;
 | |
| 		      case 'b':
 | |
| 			prefix = "0b"; break;
 | |
| 		      case 'B':
 | |
| 			prefix = "0B"; break;
 | |
| 		    }
 | |
| 		}
 | |
| 
 | |
| 	      bin_retry:
 | |
| 		switch (TYPE(val)) {
 | |
| 		  case T_FLOAT:
 | |
| 		    if (FIXABLE(RFLOAT_VALUE(val))) {
 | |
| 			val = LONG2FIX((long)RFLOAT_VALUE(val));
 | |
| 			goto bin_retry;
 | |
| 		    }
 | |
| 		    val = rb_dbl2big(RFLOAT_VALUE(val));
 | |
| 		    if (FIXNUM_P(val)) goto bin_retry;
 | |
| 		    bignum = 1;
 | |
| 		    break;
 | |
| 		  case T_STRING:
 | |
| 		    val = rb_str_to_inum(val, 0, TRUE);
 | |
| 		    goto bin_retry;
 | |
| 		  case T_BIGNUM:
 | |
| 		    bignum = 1;
 | |
| 		    break;
 | |
| 		  case T_FIXNUM:
 | |
| 		    v = FIX2LONG(val);
 | |
| 		    break;
 | |
| 		  default:
 | |
| 		    val = rb_Integer(val);
 | |
| 		    goto bin_retry;
 | |
| 		}
 | |
| 
 | |
| 		switch (*p) {
 | |
| 		  case 'o':
 | |
| 		    base = 8; break;
 | |
| 		  case 'x':
 | |
| 		  case 'X':
 | |
| 		    base = 16; break;
 | |
| 		  case 'b':
 | |
| 		  case 'B':
 | |
| 		    base = 2; break;
 | |
| 		  case 'u':
 | |
| 		  case 'd':
 | |
| 		  case 'i':
 | |
| 		  default:
 | |
| 		    base = 10; break;
 | |
| 		}
 | |
| 
 | |
|                 if (base != 10) {
 | |
|                     int numbits = ffs(base)-1;
 | |
|                     size_t abs_nlz_bits;
 | |
|                     size_t numdigits = rb_absint_numwords(val, numbits, &abs_nlz_bits);
 | |
|                     long i;
 | |
|                     if (INT_MAX-1 < numdigits) /* INT_MAX is used because rb_long2int is used later. */
 | |
|                         rb_raise(rb_eArgError, "size too big");
 | |
|                     if (sign) {
 | |
|                         if (numdigits == 0)
 | |
|                             numdigits = 1;
 | |
|                         tmp = rb_str_new(NULL, numdigits);
 | |
|                         valsign = rb_integer_pack(val, RSTRING_PTR(tmp), RSTRING_LEN(tmp),
 | |
|                                 1, CHAR_BIT-numbits, INTEGER_PACK_BIG_ENDIAN);
 | |
|                         for (i = 0; i < RSTRING_LEN(tmp); i++)
 | |
|                             RSTRING_PTR(tmp)[i] = ruby_digitmap[((unsigned char *)RSTRING_PTR(tmp))[i]];
 | |
|                         s = RSTRING_PTR(tmp);
 | |
|                         if (valsign < 0) {
 | |
|                             sc = '-';
 | |
|                             width--;
 | |
|                         }
 | |
|                         else if (flags & FPLUS) {
 | |
|                             sc = '+';
 | |
|                             width--;
 | |
|                         }
 | |
|                         else if (flags & FSPACE) {
 | |
|                             sc = ' ';
 | |
|                             width--;
 | |
|                         }
 | |
|                     }
 | |
|                     else {
 | |
|                         /* Following conditional "numdigits++" guarantees the
 | |
|                          * most significant digit as
 | |
|                          * - '1'(bin), '7'(oct) or 'f'(hex) for negative numbers
 | |
|                          * - '0' for zero
 | |
|                          * - not '0' for positive numbers.
 | |
|                          *
 | |
|                          * It also guarantees the most significant two
 | |
|                          * digits will not be '11'(bin), '77'(oct), 'ff'(hex)
 | |
|                          * or '00'.  */
 | |
|                         if (numdigits == 0 ||
 | |
|                                 ((abs_nlz_bits != (size_t)(numbits-1) ||
 | |
|                                   !rb_absint_singlebit_p(val)) &&
 | |
|                                  (!bignum ? v < 0 : BIGNUM_NEGATIVE_P(val))))
 | |
|                             numdigits++;
 | |
|                         tmp = rb_str_new(NULL, numdigits);
 | |
|                         valsign = rb_integer_pack(val, RSTRING_PTR(tmp), RSTRING_LEN(tmp),
 | |
|                                 1, CHAR_BIT-numbits, INTEGER_PACK_2COMP | INTEGER_PACK_BIG_ENDIAN);
 | |
|                         for (i = 0; i < RSTRING_LEN(tmp); i++)
 | |
|                             RSTRING_PTR(tmp)[i] = ruby_digitmap[((unsigned char *)RSTRING_PTR(tmp))[i]];
 | |
|                         s = RSTRING_PTR(tmp);
 | |
|                         dots = valsign < 0;
 | |
|                     }
 | |
|                     len = rb_long2int(RSTRING_END(tmp) - s);
 | |
|                 }
 | |
|                 else if (!bignum) {
 | |
|                     valsign = 1;
 | |
|                     if (v < 0) {
 | |
|                         v = -v;
 | |
|                         sc = '-';
 | |
|                         width--;
 | |
|                         valsign = -1;
 | |
|                     }
 | |
|                     else if (flags & FPLUS) {
 | |
|                         sc = '+';
 | |
|                         width--;
 | |
|                     }
 | |
|                     else if (flags & FSPACE) {
 | |
|                         sc = ' ';
 | |
|                         width--;
 | |
|                     }
 | |
| 		    s = ruby_ultoa((unsigned long)v, nbuf + sizeof(nbuf), 10, 0);
 | |
| 		    len = (int)(nbuf + sizeof(nbuf) - s);
 | |
| 		}
 | |
| 		else {
 | |
|                     tmp = rb_big2str(val, 10);
 | |
|                     s = RSTRING_PTR(tmp);
 | |
|                     valsign = 1;
 | |
|                     if (s[0] == '-') {
 | |
|                         s++;
 | |
|                         sc = '-';
 | |
|                         width--;
 | |
|                         valsign = -1;
 | |
|                     }
 | |
|                     else if (flags & FPLUS) {
 | |
|                         sc = '+';
 | |
|                         width--;
 | |
|                     }
 | |
|                     else if (flags & FSPACE) {
 | |
|                         sc = ' ';
 | |
|                         width--;
 | |
|                     }
 | |
| 		    len = rb_long2int(RSTRING_END(tmp) - s);
 | |
| 		}
 | |
| 
 | |
| 		if (dots) {
 | |
| 		    prec -= 2;
 | |
| 		    width -= 2;
 | |
| 		}
 | |
| 
 | |
| 		if (*p == 'X') {
 | |
| 		    char *pp = s;
 | |
| 		    int c;
 | |
| 		    while ((c = (int)(unsigned char)*pp) != 0) {
 | |
| 			*pp = rb_enc_toupper(c, enc);
 | |
| 			pp++;
 | |
| 		    }
 | |
| 		}
 | |
| 		if (prefix && !prefix[1]) { /* octal */
 | |
| 		    if (dots) {
 | |
| 			prefix = 0;
 | |
| 		    }
 | |
| 		    else if (len == 1 && *s == '0') {
 | |
| 			len = 0;
 | |
| 			if (flags & FPREC) prec--;
 | |
| 		    }
 | |
| 		    else if ((flags & FPREC) && (prec > len)) {
 | |
| 			prefix = 0;
 | |
| 		    }
 | |
| 		}
 | |
| 		else if (len == 1 && *s == '0') {
 | |
| 		    prefix = 0;
 | |
| 		}
 | |
| 		if (prefix) {
 | |
| 		    width -= (int)strlen(prefix);
 | |
| 		}
 | |
| 		if ((flags & (FZERO|FMINUS|FPREC)) == FZERO) {
 | |
| 		    prec = width;
 | |
| 		    width = 0;
 | |
| 		}
 | |
| 		else {
 | |
| 		    if (prec < len) {
 | |
| 			if (!prefix && prec == 0 && len == 1 && *s == '0') len = 0;
 | |
| 			prec = len;
 | |
| 		    }
 | |
| 		    width -= prec;
 | |
| 		}
 | |
| 		if (!(flags&FMINUS)) {
 | |
| 		    FILL(' ', width);
 | |
| 		    width = 0;
 | |
| 		}
 | |
| 		if (sc) PUSH(&sc, 1);
 | |
| 		if (prefix) {
 | |
| 		    int plen = (int)strlen(prefix);
 | |
| 		    PUSH(prefix, plen);
 | |
| 		}
 | |
| 		if (dots) PUSH("..", 2);
 | |
| 		if (prec > len) {
 | |
| 		    CHECK(prec - len);
 | |
| 		    if (!sign && valsign < 0) {
 | |
| 			char c = sign_bits(base, p);
 | |
| 			FILL_(c, prec - len);
 | |
| 		    }
 | |
| 		    else if ((flags & (FMINUS|FPREC)) != FMINUS) {
 | |
| 			FILL_('0', prec - len);
 | |
| 		    }
 | |
| 		}
 | |
| 		PUSH(s, len);
 | |
| 		RB_GC_GUARD(tmp);
 | |
| 		FILL(' ', width);
 | |
| 	    }
 | |
| 	    break;
 | |
| 
 | |
| 	  case 'f':
 | |
| 	    {
 | |
| 		VALUE val = GETARG(), num, den;
 | |
| 		int sign = (flags&FPLUS) ? 1 : 0, zero = 0;
 | |
| 		long len, fill;
 | |
| 		if (RB_INTEGER_TYPE_P(val)) {
 | |
| 		    den = INT2FIX(1);
 | |
| 		    num = val;
 | |
| 		}
 | |
| 		else if (RB_TYPE_P(val, T_RATIONAL)) {
 | |
| 		    den = rb_rational_den(val);
 | |
| 		    num = rb_rational_num(val);
 | |
| 		}
 | |
| 		else {
 | |
| 		    nextvalue = val;
 | |
| 		    goto float_value;
 | |
| 		}
 | |
| 		if (!(flags&FPREC)) prec = default_float_precision;
 | |
| 		if (FIXNUM_P(num)) {
 | |
| 		    if ((SIGNED_VALUE)num < 0) {
 | |
| 			long n = -FIX2LONG(num);
 | |
| 			num = LONG2FIX(n);
 | |
| 			sign = -1;
 | |
| 		    }
 | |
| 		}
 | |
| 		else if (BIGNUM_NEGATIVE_P(num)) {
 | |
| 		    sign = -1;
 | |
| 		    num = rb_big_uminus(num);
 | |
| 		}
 | |
| 		if (den != INT2FIX(1)) {
 | |
| 		    num = rb_int_mul(num, rb_int_positive_pow(10, prec));
 | |
| 		    num = rb_int_plus(num, rb_int_idiv(den, INT2FIX(2)));
 | |
| 		    num = rb_int_idiv(num, den);
 | |
| 		}
 | |
| 		else if (prec >= 0) {
 | |
| 		    zero = prec;
 | |
| 		}
 | |
| 		val = rb_int2str(num, 10);
 | |
| 		len = RSTRING_LEN(val) + zero;
 | |
| 		if (prec >= len) len = prec + 1; /* integer part 0 */
 | |
| 		if (sign || (flags&FSPACE)) ++len;
 | |
| 		if (prec > 0) ++len; /* period */
 | |
| 		fill = width > len ? width - len : 0;
 | |
| 		CHECK(fill + len);
 | |
| 		if (fill && !(flags&(FMINUS|FZERO))) {
 | |
| 		    FILL_(' ', fill);
 | |
| 		}
 | |
| 		if (sign || (flags&FSPACE)) {
 | |
| 		    buf[blen++] = sign > 0 ? '+' : sign < 0 ? '-' : ' ';
 | |
| 		}
 | |
| 		if (fill && (flags&(FMINUS|FZERO)) == FZERO) {
 | |
| 		    FILL_('0', fill);
 | |
| 		}
 | |
| 		len = RSTRING_LEN(val) + zero;
 | |
| 		t = RSTRING_PTR(val);
 | |
| 		if (len > prec) {
 | |
| 		    PUSH_(t, len - prec);
 | |
| 		}
 | |
| 		else {
 | |
| 		    buf[blen++] = '0';
 | |
| 		}
 | |
| 		if (prec > 0) {
 | |
| 		    buf[blen++] = '.';
 | |
| 		}
 | |
| 		if (zero) {
 | |
| 		    FILL_('0', zero);
 | |
| 		}
 | |
| 		else if (prec > len) {
 | |
| 		    FILL_('0', prec - len);
 | |
| 		    PUSH_(t, len);
 | |
| 		}
 | |
| 		else if (prec > 0) {
 | |
| 		    PUSH_(t + len - prec, prec);
 | |
| 		}
 | |
| 		if (fill && (flags&FMINUS)) {
 | |
| 		    FILL_(' ', fill);
 | |
| 		}
 | |
| 		RB_GC_GUARD(val);
 | |
| 		break;
 | |
| 	    }
 | |
| 	  case 'g':
 | |
| 	  case 'G':
 | |
| 	  case 'e':
 | |
| 	  case 'E':
 | |
| 	    /* TODO: rational support */
 | |
| 	  case 'a':
 | |
| 	  case 'A':
 | |
| 	  float_value:
 | |
| 	    {
 | |
| 		VALUE val = GETARG();
 | |
| 		double fval;
 | |
| 
 | |
| 		fval = RFLOAT_VALUE(rb_Float(val));
 | |
| 		if (isnan(fval) || isinf(fval)) {
 | |
| 		    const char *expr;
 | |
| 		    int need;
 | |
| 		    int elen;
 | |
| 		    char sign = '\0';
 | |
| 
 | |
| 		    if (isnan(fval)) {
 | |
| 			expr = "NaN";
 | |
| 		    }
 | |
| 		    else {
 | |
| 			expr = "Inf";
 | |
| 		    }
 | |
| 		    need = (int)strlen(expr);
 | |
| 		    elen = need;
 | |
| 		    if (!isnan(fval) && fval < 0.0)
 | |
| 			sign = '-';
 | |
| 		    else if (flags & (FPLUS|FSPACE))
 | |
| 			sign = (flags & FPLUS) ? '+' : ' ';
 | |
| 		    if (sign)
 | |
| 			++need;
 | |
| 		    if ((flags & FWIDTH) && need < width)
 | |
| 			need = width;
 | |
| 
 | |
| 		    FILL(' ', need);
 | |
| 		    if (flags & FMINUS) {
 | |
| 			if (sign)
 | |
| 			    buf[blen - need--] = sign;
 | |
| 			memcpy(&buf[blen - need], expr, elen);
 | |
| 		    }
 | |
| 		    else {
 | |
| 			if (sign)
 | |
| 			    buf[blen - elen - 1] = sign;
 | |
| 			memcpy(&buf[blen - elen], expr, elen);
 | |
| 		    }
 | |
| 		    break;
 | |
| 		}
 | |
| 		else {
 | |
| 		    int cr = ENC_CODERANGE(result);
 | |
| 		    char fbuf[2*BIT_DIGITS(SIZEOF_INT*CHAR_BIT)+10];
 | |
| 		    char *fmt = fmt_setup(fbuf, sizeof(fbuf), *p, flags, width, prec);
 | |
| 		    rb_str_set_len(result, blen);
 | |
| 		    rb_str_catf(result, fmt, fval);
 | |
| 		    ENC_CODERANGE_SET(result, cr);
 | |
| 		    bsiz = rb_str_capacity(result);
 | |
| 		    RSTRING_GETMEM(result, buf, blen);
 | |
| 		}
 | |
| 	    }
 | |
| 	    break;
 | |
| 	}
 | |
| 	flags = FNONE;
 | |
|     }
 | |
| 
 | |
|   sprint_exit:
 | |
|     rb_str_tmp_frozen_release(orig, fmt);
 | |
|     /* XXX - We cannot validate the number of arguments if (digit)$ style used.
 | |
|      */
 | |
|     if (posarg >= 0 && nextarg < argc) {
 | |
| 	const char *mesg = "too many arguments for format string";
 | |
| 	if (RTEST(ruby_debug)) rb_raise(rb_eArgError, "%s", mesg);
 | |
| 	if (RTEST(ruby_verbose)) rb_warn("%s", mesg);
 | |
|     }
 | |
|     rb_str_resize(result, blen);
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| static char *
 | |
| fmt_setup(char *buf, size_t size, int c, int flags, int width, int prec)
 | |
| {
 | |
|     buf += size;
 | |
|     *--buf = '\0';
 | |
|     *--buf = c;
 | |
| 
 | |
|     if (flags & FPREC) {
 | |
| 	buf = ruby_ultoa(prec, buf, 10, 0);
 | |
| 	*--buf = '.';
 | |
|     }
 | |
| 
 | |
|     if (flags & FWIDTH) {
 | |
| 	buf = ruby_ultoa(width, buf, 10, 0);
 | |
|     }
 | |
| 
 | |
|     if (flags & FSPACE) *--buf = ' ';
 | |
|     if (flags & FZERO)  *--buf = '0';
 | |
|     if (flags & FMINUS) *--buf = '-';
 | |
|     if (flags & FPLUS)  *--buf = '+';
 | |
|     if (flags & FSHARP) *--buf = '#';
 | |
|     *--buf = '%';
 | |
|     return buf;
 | |
| }
 | |
| 
 | |
| #undef FILE
 | |
| #define FILE rb_printf_buffer
 | |
| #define __sbuf rb_printf_sbuf
 | |
| #define __sFILE rb_printf_sfile
 | |
| #undef feof
 | |
| #undef ferror
 | |
| #undef clearerr
 | |
| #undef fileno
 | |
| #if SIZEOF_LONG < SIZEOF_LONG_LONG
 | |
| # if SIZEOF_LONG_LONG == SIZEOF_VOIDP
 | |
| /* actually this doesn't mean a pointer is strictly 64bit, but just
 | |
|  * quad_t size */
 | |
| #   define _HAVE_LLP64_
 | |
| # endif
 | |
| # define _HAVE_SANE_QUAD_
 | |
| # define quad_t LONG_LONG
 | |
| # define u_quad_t unsigned LONG_LONG
 | |
| #endif
 | |
| #define FLOATING_POINT 1
 | |
| #define BSD__dtoa ruby_dtoa
 | |
| #define BSD__hdtoa ruby_hdtoa
 | |
| #ifdef RUBY_PRI_VALUE_MARK
 | |
| # define PRI_EXTRA_MARK RUBY_PRI_VALUE_MARK
 | |
| #endif
 | |
| #define lower_hexdigits (ruby_hexdigits+0)
 | |
| #define upper_hexdigits (ruby_hexdigits+16)
 | |
| #include "vsnprintf.c"
 | |
| 
 | |
| static char *
 | |
| ruby_ultoa(unsigned long val, char *endp, int base, int flags)
 | |
| {
 | |
|     const char *xdigs = lower_hexdigits;
 | |
|     int octzero = flags & FSHARP;
 | |
|     return BSD__ultoa(val, endp, base, octzero, xdigs);
 | |
| }
 | |
| 
 | |
| static int ruby_do_vsnprintf(char *str, size_t n, const char *fmt, va_list ap);
 | |
| 
 | |
| int
 | |
| ruby_vsnprintf(char *str, size_t n, const char *fmt, va_list ap)
 | |
| {
 | |
|     if (str && (ssize_t)n < 1)
 | |
| 	return (EOF);
 | |
|     return ruby_do_vsnprintf(str, n, fmt, ap);
 | |
| }
 | |
| 
 | |
| static int
 | |
| ruby_do_vsnprintf(char *str, size_t n, const char *fmt, va_list ap)
 | |
| {
 | |
|     ssize_t ret;
 | |
|     rb_printf_buffer f;
 | |
| 
 | |
|     f._flags = __SWR | __SSTR;
 | |
|     f._bf._base = f._p = (unsigned char *)str;
 | |
|     f._bf._size = f._w = str ? (n - 1) : 0;
 | |
|     f.vwrite = BSD__sfvwrite;
 | |
|     f.vextra = 0;
 | |
|     ret = BSD_vfprintf(&f, fmt, ap);
 | |
|     if (str) *f._p = 0;
 | |
| #if SIZEOF_SIZE_T > SIZEOF_INT
 | |
|     if (n > INT_MAX) return INT_MAX;
 | |
| #endif
 | |
|     return (int)ret;
 | |
| }
 | |
| 
 | |
| int
 | |
| ruby_snprintf(char *str, size_t n, char const *fmt, ...)
 | |
| {
 | |
|     int ret;
 | |
|     va_list ap;
 | |
| 
 | |
|     if (str && (ssize_t)n < 1)
 | |
| 	return (EOF);
 | |
| 
 | |
|     va_start(ap, fmt);
 | |
|     ret = ruby_do_vsnprintf(str, n, fmt, ap);
 | |
|     va_end(ap);
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| typedef struct {
 | |
|     rb_printf_buffer base;
 | |
|     volatile VALUE value;
 | |
| } rb_printf_buffer_extra;
 | |
| 
 | |
| static int
 | |
| ruby__sfvwrite(register rb_printf_buffer *fp, register struct __suio *uio)
 | |
| {
 | |
|     struct __siov *iov;
 | |
|     VALUE result = (VALUE)fp->_bf._base;
 | |
|     char *buf = (char*)fp->_p;
 | |
|     long len, n;
 | |
|     long blen = buf - RSTRING_PTR(result), bsiz = fp->_w;
 | |
| 
 | |
|     if (RBASIC(result)->klass) {
 | |
| 	rb_raise(rb_eRuntimeError, "rb_vsprintf reentered");
 | |
|     }
 | |
|     if (uio->uio_resid == 0)
 | |
| 	return 0;
 | |
| #if SIZE_MAX > LONG_MAX
 | |
|     if (uio->uio_resid >= LONG_MAX)
 | |
| 	rb_raise(rb_eRuntimeError, "too big string");
 | |
| #endif
 | |
|     len = (long)uio->uio_resid;
 | |
|     CHECK(len);
 | |
|     buf += blen;
 | |
|     fp->_w = bsiz;
 | |
|     for (iov = uio->uio_iov; len > 0; ++iov) {
 | |
| 	MEMCPY(buf, iov->iov_base, char, n = iov->iov_len);
 | |
| 	buf += n;
 | |
| 	len -= n;
 | |
|     }
 | |
|     fp->_p = (unsigned char *)buf;
 | |
|     rb_str_set_len(result, buf - RSTRING_PTR(result));
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static const char *
 | |
| ruby__sfvextra(rb_printf_buffer *fp, size_t valsize, void *valp, long *sz, int sign)
 | |
| {
 | |
|     VALUE value, result = (VALUE)fp->_bf._base;
 | |
|     rb_encoding *enc;
 | |
|     char *cp;
 | |
| 
 | |
|     if (valsize != sizeof(VALUE)) return 0;
 | |
|     value = *(VALUE *)valp;
 | |
|     if (RBASIC(result)->klass) {
 | |
| 	rb_raise(rb_eRuntimeError, "rb_vsprintf reentered");
 | |
|     }
 | |
|     if (sign == '+') {
 | |
| 	if (RB_TYPE_P(value, T_CLASS)) {
 | |
| # define LITERAL(str) (*sz = rb_strlen_lit(str), str)
 | |
| 
 | |
| 	    if (value == rb_cNilClass) {
 | |
| 		return LITERAL("nil");
 | |
| 	    }
 | |
| 	    else if (value == rb_cInteger) {
 | |
| 		return LITERAL("Integer");
 | |
| 	    }
 | |
| 	    else if (value == rb_cSymbol) {
 | |
| 		return LITERAL("Symbol");
 | |
| 	    }
 | |
| 	    else if (value == rb_cTrueClass) {
 | |
| 		return LITERAL("true");
 | |
| 	    }
 | |
| 	    else if (value == rb_cFalseClass) {
 | |
| 		return LITERAL("false");
 | |
| 	    }
 | |
| # undef LITERAL
 | |
| 	}
 | |
| 	value = rb_inspect(value);
 | |
|     }
 | |
|     else if (SYMBOL_P(value)) {
 | |
| 	value = rb_sym2str(value);
 | |
| 	if (sign == ' ' && !rb_str_symname_p(value)) {
 | |
| 	    value = rb_str_escape(value);
 | |
| 	}
 | |
|     }
 | |
|     else {
 | |
| 	value = rb_obj_as_string(value);
 | |
| 	if (sign == ' ') value = QUOTE(value);
 | |
|     }
 | |
|     enc = rb_enc_compatible(result, value);
 | |
|     if (enc) {
 | |
| 	rb_enc_associate(result, enc);
 | |
|     }
 | |
|     else {
 | |
| 	enc = rb_enc_get(result);
 | |
| 	value = rb_str_conv_enc_opts(value, rb_enc_get(value), enc,
 | |
| 				     ECONV_UNDEF_REPLACE|ECONV_INVALID_REPLACE,
 | |
| 				     Qnil);
 | |
| 	*(volatile VALUE *)valp = value;
 | |
|     }
 | |
|     StringValueCStr(value);
 | |
|     RSTRING_GETMEM(value, cp, *sz);
 | |
|     ((rb_printf_buffer_extra *)fp)->value = value;
 | |
|     return cp;
 | |
| }
 | |
| 
 | |
| VALUE
 | |
| rb_enc_vsprintf(rb_encoding *enc, const char *fmt, va_list ap)
 | |
| {
 | |
|     rb_printf_buffer_extra buffer;
 | |
| #define f buffer.base
 | |
|     VALUE result;
 | |
| 
 | |
|     f._flags = __SWR | __SSTR;
 | |
|     f._bf._size = 0;
 | |
|     f._w = 120;
 | |
|     result = rb_str_buf_new(f._w);
 | |
|     if (enc) {
 | |
| 	if (rb_enc_mbminlen(enc) > 1) {
 | |
| 	    /* the implementation deeply depends on plain char */
 | |
| 	    rb_raise(rb_eArgError, "cannot construct wchar_t based encoding string: %s",
 | |
| 		     rb_enc_name(enc));
 | |
| 	}
 | |
| 	rb_enc_associate(result, enc);
 | |
|     }
 | |
|     f._bf._base = (unsigned char *)result;
 | |
|     f._p = (unsigned char *)RSTRING_PTR(result);
 | |
|     RBASIC_CLEAR_CLASS(result);
 | |
|     f.vwrite = ruby__sfvwrite;
 | |
|     f.vextra = ruby__sfvextra;
 | |
|     buffer.value = 0;
 | |
|     BSD_vfprintf(&f, fmt, ap);
 | |
|     RBASIC_SET_CLASS_RAW(result, rb_cString);
 | |
|     rb_str_resize(result, (char *)f._p - RSTRING_PTR(result));
 | |
| #undef f
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| VALUE
 | |
| rb_enc_sprintf(rb_encoding *enc, const char *format, ...)
 | |
| {
 | |
|     VALUE result;
 | |
|     va_list ap;
 | |
| 
 | |
|     va_start(ap, format);
 | |
|     result = rb_enc_vsprintf(enc, format, ap);
 | |
|     va_end(ap);
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| VALUE
 | |
| rb_vsprintf(const char *fmt, va_list ap)
 | |
| {
 | |
|     return rb_enc_vsprintf(NULL, fmt, ap);
 | |
| }
 | |
| 
 | |
| VALUE
 | |
| rb_sprintf(const char *format, ...)
 | |
| {
 | |
|     VALUE result;
 | |
|     va_list ap;
 | |
| 
 | |
|     va_start(ap, format);
 | |
|     result = rb_vsprintf(format, ap);
 | |
|     va_end(ap);
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| VALUE
 | |
| rb_str_vcatf(VALUE str, const char *fmt, va_list ap)
 | |
| {
 | |
|     rb_printf_buffer_extra buffer;
 | |
| #define f buffer.base
 | |
|     VALUE klass;
 | |
| 
 | |
|     StringValue(str);
 | |
|     rb_str_modify(str);
 | |
|     f._flags = __SWR | __SSTR;
 | |
|     f._bf._size = 0;
 | |
|     f._w = rb_str_capacity(str);
 | |
|     f._bf._base = (unsigned char *)str;
 | |
|     f._p = (unsigned char *)RSTRING_END(str);
 | |
|     klass = RBASIC(str)->klass;
 | |
|     RBASIC_CLEAR_CLASS(str);
 | |
|     f.vwrite = ruby__sfvwrite;
 | |
|     f.vextra = ruby__sfvextra;
 | |
|     buffer.value = 0;
 | |
|     BSD_vfprintf(&f, fmt, ap);
 | |
|     RBASIC_SET_CLASS_RAW(str, klass);
 | |
|     rb_str_resize(str, (char *)f._p - RSTRING_PTR(str));
 | |
| #undef f
 | |
| 
 | |
|     return str;
 | |
| }
 | |
| 
 | |
| VALUE
 | |
| rb_str_catf(VALUE str, const char *format, ...)
 | |
| {
 | |
|     va_list ap;
 | |
| 
 | |
|     va_start(ap, format);
 | |
|     str = rb_str_vcatf(str, format, ap);
 | |
|     va_end(ap);
 | |
| 
 | |
|     return str;
 | |
| }
 | 
