1  
     2  
     3  
     4  
     5  
     6  
     7  
     8  
     9  
    10  
    11  package strconv
    12  
    13  const (
    14  	lowerhex = "0123456789abcdef"
    15  	upperhex = "0123456789ABCDEF"
    16  )
    17  
    18  type floatInfo struct {
    19  	mantbits uint
    20  	expbits  uint
    21  	bias     int
    22  }
    23  
    24  const (
    25  	float32MantBits = 23
    26  	float32ExpBits  = 8
    27  	float32Bias     = -127
    28  	float64MantBits = 52
    29  	float64ExpBits  = 11
    30  	float64Bias     = -1023
    31  )
    32  
    33  var (
    34  	float32info = floatInfo{float32MantBits, float32ExpBits, float32Bias}
    35  	float64info = floatInfo{float64MantBits, float64ExpBits, float64Bias}
    36  )
    37  
    38  
    39  
    40  
    41  
    42  
    43  
    44  
    45  
    46  
    47  
    48  
    49  
    50  
    51  
    52  
    53  
    54  
    55  
    56  
    57  
    58  
    59  
    60  
    61  
    62  func FormatFloat(f float64, fmt byte, prec, bitSize int) string {
    63  	return string(genericFtoa(make([]byte, 0, max(prec+4, 24)), f, fmt, prec, bitSize))
    64  }
    65  
    66  
    67  
    68  func AppendFloat(dst []byte, f float64, fmt byte, prec, bitSize int) []byte {
    69  	return genericFtoa(dst, f, fmt, prec, bitSize)
    70  }
    71  
    72  func genericFtoa(dst []byte, val float64, fmt byte, prec, bitSize int) []byte {
    73  	var bits uint64
    74  	var flt *floatInfo
    75  	switch bitSize {
    76  	case 32:
    77  		bits = uint64(float32bits(float32(val)))
    78  		flt = &float32info
    79  	case 64:
    80  		bits = float64bits(val)
    81  		flt = &float64info
    82  	default:
    83  		panic("strconv: illegal AppendFloat/FormatFloat bitSize")
    84  	}
    85  
    86  	neg := bits>>(flt.expbits+flt.mantbits) != 0
    87  	exp := int(bits>>flt.mantbits) & (1<<flt.expbits - 1)
    88  	mant := bits & (uint64(1)<<flt.mantbits - 1)
    89  
    90  	switch exp {
    91  	case 1<<flt.expbits - 1:
    92  		
    93  		var s string
    94  		switch {
    95  		case mant != 0:
    96  			s = "NaN"
    97  		case neg:
    98  			s = "-Inf"
    99  		default:
   100  			s = "+Inf"
   101  		}
   102  		return append(dst, s...)
   103  
   104  	case 0:
   105  		
   106  		exp++
   107  
   108  	default:
   109  		
   110  		mant |= uint64(1) << flt.mantbits
   111  	}
   112  	exp += flt.bias
   113  
   114  	
   115  	if fmt == 'b' {
   116  		return fmtB(dst, neg, mant, exp, flt)
   117  	}
   118  	if fmt == 'x' || fmt == 'X' {
   119  		return fmtX(dst, prec, fmt, neg, mant, exp, flt)
   120  	}
   121  
   122  	if !optimize {
   123  		return bigFtoa(dst, prec, fmt, neg, mant, exp, flt)
   124  	}
   125  
   126  	var digs decimalSlice
   127  	ok := false
   128  	
   129  	shortest := prec < 0
   130  	if shortest {
   131  		
   132  		var buf [32]byte
   133  		digs.d = buf[:]
   134  		ryuFtoaShortest(&digs, mant, exp-int(flt.mantbits), flt)
   135  		ok = true
   136  		
   137  		switch fmt {
   138  		case 'e', 'E':
   139  			prec = max(digs.nd-1, 0)
   140  		case 'f':
   141  			prec = max(digs.nd-digs.dp, 0)
   142  		case 'g', 'G':
   143  			prec = digs.nd
   144  		}
   145  	} else if fmt != 'f' {
   146  		
   147  		digits := prec
   148  		switch fmt {
   149  		case 'e', 'E':
   150  			digits++
   151  		case 'g', 'G':
   152  			if prec == 0 {
   153  				prec = 1
   154  			}
   155  			digits = prec
   156  		default:
   157  			
   158  			digits = 1
   159  		}
   160  		var buf [24]byte
   161  		if bitSize == 32 && digits <= 9 {
   162  			digs.d = buf[:]
   163  			ryuFtoaFixed32(&digs, uint32(mant), exp-int(flt.mantbits), digits)
   164  			ok = true
   165  		} else if digits <= 18 {
   166  			digs.d = buf[:]
   167  			ryuFtoaFixed64(&digs, mant, exp-int(flt.mantbits), digits)
   168  			ok = true
   169  		}
   170  	}
   171  	if !ok {
   172  		return bigFtoa(dst, prec, fmt, neg, mant, exp, flt)
   173  	}
   174  	return formatDigits(dst, shortest, neg, digs, prec, fmt)
   175  }
   176  
   177  
   178  func bigFtoa(dst []byte, prec int, fmt byte, neg bool, mant uint64, exp int, flt *floatInfo) []byte {
   179  	d := new(decimal)
   180  	d.Assign(mant)
   181  	d.Shift(exp - int(flt.mantbits))
   182  	var digs decimalSlice
   183  	shortest := prec < 0
   184  	if shortest {
   185  		roundShortest(d, mant, exp, flt)
   186  		digs = decimalSlice{d: d.d[:], nd: d.nd, dp: d.dp}
   187  		
   188  		switch fmt {
   189  		case 'e', 'E':
   190  			prec = digs.nd - 1
   191  		case 'f':
   192  			prec = max(digs.nd-digs.dp, 0)
   193  		case 'g', 'G':
   194  			prec = digs.nd
   195  		}
   196  	} else {
   197  		
   198  		switch fmt {
   199  		case 'e', 'E':
   200  			d.Round(prec + 1)
   201  		case 'f':
   202  			d.Round(d.dp + prec)
   203  		case 'g', 'G':
   204  			if prec == 0 {
   205  				prec = 1
   206  			}
   207  			d.Round(prec)
   208  		}
   209  		digs = decimalSlice{d: d.d[:], nd: d.nd, dp: d.dp}
   210  	}
   211  	return formatDigits(dst, shortest, neg, digs, prec, fmt)
   212  }
   213  
   214  func formatDigits(dst []byte, shortest bool, neg bool, digs decimalSlice, prec int, fmt byte) []byte {
   215  	switch fmt {
   216  	case 'e', 'E':
   217  		return fmtE(dst, neg, digs, prec, fmt)
   218  	case 'f':
   219  		return fmtF(dst, neg, digs, prec)
   220  	case 'g', 'G':
   221  		
   222  		eprec := prec
   223  		if eprec > digs.nd && digs.nd >= digs.dp {
   224  			eprec = digs.nd
   225  		}
   226  		
   227  		
   228  		
   229  		if shortest {
   230  			eprec = 6
   231  		}
   232  		exp := digs.dp - 1
   233  		if exp < -4 || exp >= eprec {
   234  			if prec > digs.nd {
   235  				prec = digs.nd
   236  			}
   237  			return fmtE(dst, neg, digs, prec-1, fmt+'e'-'g')
   238  		}
   239  		if prec > digs.dp {
   240  			prec = digs.nd
   241  		}
   242  		return fmtF(dst, neg, digs, max(prec-digs.dp, 0))
   243  	}
   244  
   245  	
   246  	return append(dst, '%', fmt)
   247  }
   248  
   249  
   250  
   251  func roundShortest(d *decimal, mant uint64, exp int, flt *floatInfo) {
   252  	
   253  	if mant == 0 {
   254  		d.nd = 0
   255  		return
   256  	}
   257  
   258  	
   259  	
   260  	
   261  
   262  	
   263  	
   264  	
   265  	
   266  	
   267  	
   268  	
   269  	
   270  	
   271  	
   272  	minexp := flt.bias + 1 
   273  	if exp > minexp && 332*(d.dp-d.nd) >= 100*(exp-int(flt.mantbits)) {
   274  		
   275  		return
   276  	}
   277  
   278  	
   279  	
   280  	
   281  	upper := new(decimal)
   282  	upper.Assign(mant*2 + 1)
   283  	upper.Shift(exp - int(flt.mantbits) - 1)
   284  
   285  	
   286  	
   287  	
   288  	
   289  	
   290  	
   291  	var mantlo uint64
   292  	var explo int
   293  	if mant > 1<<flt.mantbits || exp == minexp {
   294  		mantlo = mant - 1
   295  		explo = exp
   296  	} else {
   297  		mantlo = mant*2 - 1
   298  		explo = exp - 1
   299  	}
   300  	lower := new(decimal)
   301  	lower.Assign(mantlo*2 + 1)
   302  	lower.Shift(explo - int(flt.mantbits) - 1)
   303  
   304  	
   305  	
   306  	
   307  	inclusive := mant%2 == 0
   308  
   309  	
   310  	
   311  	
   312  	
   313  	
   314  	
   315  	
   316  	
   317  	
   318  	
   319  	
   320  	var upperdelta uint8
   321  
   322  	
   323  	
   324  	for ui := 0; ; ui++ {
   325  		
   326  		
   327  		
   328  		mi := ui - upper.dp + d.dp
   329  		if mi >= d.nd {
   330  			break
   331  		}
   332  		li := ui - upper.dp + lower.dp
   333  		l := byte('0') 
   334  		if li >= 0 && li < lower.nd {
   335  			l = lower.d[li]
   336  		}
   337  		m := byte('0') 
   338  		if mi >= 0 {
   339  			m = d.d[mi]
   340  		}
   341  		u := byte('0') 
   342  		if ui < upper.nd {
   343  			u = upper.d[ui]
   344  		}
   345  
   346  		
   347  		
   348  		
   349  		okdown := l != m || inclusive && li+1 == lower.nd
   350  
   351  		switch {
   352  		case upperdelta == 0 && m+1 < u:
   353  			
   354  			
   355  			
   356  			upperdelta = 2
   357  		case upperdelta == 0 && m != u:
   358  			
   359  			
   360  			
   361  			upperdelta = 1
   362  		case upperdelta == 1 && (m != '9' || u != '0'):
   363  			
   364  			
   365  			
   366  			upperdelta = 2
   367  		}
   368  		
   369  		
   370  		okup := upperdelta > 0 && (inclusive || upperdelta > 1 || ui+1 < upper.nd)
   371  
   372  		
   373  		
   374  		switch {
   375  		case okdown && okup:
   376  			d.Round(mi + 1)
   377  			return
   378  		case okdown:
   379  			d.RoundDown(mi + 1)
   380  			return
   381  		case okup:
   382  			d.RoundUp(mi + 1)
   383  			return
   384  		}
   385  	}
   386  }
   387  
   388  type decimalSlice struct {
   389  	d      []byte
   390  	nd, dp int
   391  }
   392  
   393  
   394  func fmtE(dst []byte, neg bool, d decimalSlice, prec int, fmt byte) []byte {
   395  	
   396  	if neg {
   397  		dst = append(dst, '-')
   398  	}
   399  
   400  	
   401  	ch := byte('0')
   402  	if d.nd != 0 {
   403  		ch = d.d[0]
   404  	}
   405  	dst = append(dst, ch)
   406  
   407  	
   408  	if prec > 0 {
   409  		dst = append(dst, '.')
   410  		i := 1
   411  		m := min(d.nd, prec+1)
   412  		if i < m {
   413  			dst = append(dst, d.d[i:m]...)
   414  			i = m
   415  		}
   416  		for ; i <= prec; i++ {
   417  			dst = append(dst, '0')
   418  		}
   419  	}
   420  
   421  	
   422  	dst = append(dst, fmt)
   423  	exp := d.dp - 1
   424  	if d.nd == 0 { 
   425  		exp = 0
   426  	}
   427  	if exp < 0 {
   428  		ch = '-'
   429  		exp = -exp
   430  	} else {
   431  		ch = '+'
   432  	}
   433  	dst = append(dst, ch)
   434  
   435  	
   436  	switch {
   437  	case exp < 10:
   438  		dst = append(dst, '0', byte(exp)+'0')
   439  	case exp < 100:
   440  		dst = append(dst, byte(exp/10)+'0', byte(exp%10)+'0')
   441  	default:
   442  		dst = append(dst, byte(exp/100)+'0', byte(exp/10)%10+'0', byte(exp%10)+'0')
   443  	}
   444  
   445  	return dst
   446  }
   447  
   448  
   449  func fmtF(dst []byte, neg bool, d decimalSlice, prec int) []byte {
   450  	
   451  	if neg {
   452  		dst = append(dst, '-')
   453  	}
   454  
   455  	
   456  	if d.dp > 0 {
   457  		m := min(d.nd, d.dp)
   458  		dst = append(dst, d.d[:m]...)
   459  		for ; m < d.dp; m++ {
   460  			dst = append(dst, '0')
   461  		}
   462  	} else {
   463  		dst = append(dst, '0')
   464  	}
   465  
   466  	
   467  	if prec > 0 {
   468  		dst = append(dst, '.')
   469  		for i := 0; i < prec; i++ {
   470  			ch := byte('0')
   471  			if j := d.dp + i; 0 <= j && j < d.nd {
   472  				ch = d.d[j]
   473  			}
   474  			dst = append(dst, ch)
   475  		}
   476  	}
   477  
   478  	return dst
   479  }
   480  
   481  
   482  func fmtB(dst []byte, neg bool, mant uint64, exp int, flt *floatInfo) []byte {
   483  	
   484  	if neg {
   485  		dst = append(dst, '-')
   486  	}
   487  
   488  	
   489  	dst = AppendUint(dst, mant, 10)
   490  
   491  	
   492  	dst = append(dst, 'p')
   493  
   494  	
   495  	exp -= int(flt.mantbits)
   496  	if exp >= 0 {
   497  		dst = append(dst, '+')
   498  	}
   499  	dst = AppendInt(dst, int64(exp), 10)
   500  
   501  	return dst
   502  }
   503  
   504  
   505  func fmtX(dst []byte, prec int, fmt byte, neg bool, mant uint64, exp int, flt *floatInfo) []byte {
   506  	if mant == 0 {
   507  		exp = 0
   508  	}
   509  
   510  	
   511  	mant <<= 60 - flt.mantbits
   512  	for mant != 0 && mant&(1<<60) == 0 {
   513  		mant <<= 1
   514  		exp--
   515  	}
   516  
   517  	
   518  	if prec >= 0 && prec < 15 {
   519  		shift := uint(prec * 4)
   520  		extra := (mant << shift) & (1<<60 - 1)
   521  		mant >>= 60 - shift
   522  		if extra|(mant&1) > 1<<59 {
   523  			mant++
   524  		}
   525  		mant <<= 60 - shift
   526  		if mant&(1<<61) != 0 {
   527  			
   528  			mant >>= 1
   529  			exp++
   530  		}
   531  	}
   532  
   533  	hex := lowerhex
   534  	if fmt == 'X' {
   535  		hex = upperhex
   536  	}
   537  
   538  	
   539  	if neg {
   540  		dst = append(dst, '-')
   541  	}
   542  	dst = append(dst, '0', fmt, '0'+byte((mant>>60)&1))
   543  
   544  	
   545  	mant <<= 4 
   546  	if prec < 0 && mant != 0 {
   547  		dst = append(dst, '.')
   548  		for mant != 0 {
   549  			dst = append(dst, hex[(mant>>60)&15])
   550  			mant <<= 4
   551  		}
   552  	} else if prec > 0 {
   553  		dst = append(dst, '.')
   554  		for i := 0; i < prec; i++ {
   555  			dst = append(dst, hex[(mant>>60)&15])
   556  			mant <<= 4
   557  		}
   558  	}
   559  
   560  	
   561  	ch := byte('P')
   562  	if fmt == lower(fmt) {
   563  		ch = 'p'
   564  	}
   565  	dst = append(dst, ch)
   566  	if exp < 0 {
   567  		ch = '-'
   568  		exp = -exp
   569  	} else {
   570  		ch = '+'
   571  	}
   572  	dst = append(dst, ch)
   573  
   574  	
   575  	switch {
   576  	case exp < 100:
   577  		dst = append(dst, byte(exp/10)+'0', byte(exp%10)+'0')
   578  	case exp < 1000:
   579  		dst = append(dst, byte(exp/100)+'0', byte((exp/10)%10)+'0', byte(exp%10)+'0')
   580  	default:
   581  		dst = append(dst, byte(exp/1000)+'0', byte(exp/100)%10+'0', byte((exp/10)%10)+'0', byte(exp%10)+'0')
   582  	}
   583  
   584  	return dst
   585  }
   586  
View as plain text