1 /* 2 ****************************************************************************** 3 * 4 * Copyright (C) 1997-2015, International Business Machines 5 * Corporation and others. All Rights Reserved. 6 * 7 ****************************************************************************** 8 * 9 * File DIGITLST.H 10 * 11 * Modification History: 12 * 13 * Date Name Description 14 * 02/25/97 aliu Converted from java. 15 * 03/21/97 clhuang Updated per C++ implementation. 16 * 04/15/97 aliu Changed MAX_COUNT to DBL_DIG. Changed Digit to char. 17 * 09/09/97 aliu Adapted for exponential notation support. 18 * 08/02/98 stephen Added nearest/even rounding 19 * 06/29/99 stephen Made LONG_DIGITS a macro to satisfy SUN compiler 20 * 07/09/99 stephen Removed kMaxCount (unused, for HP compiler) 21 ****************************************************************************** 22 */ 23 24 #ifndef DIGITLST_H 25 #define DIGITLST_H 26 27 #include "unicode/uobject.h" 28 29 #if !UCONFIG_NO_FORMATTING 30 #include "unicode/decimfmt.h" 31 #include <float.h> 32 #include "decContext.h" 33 #include "decNumber.h" 34 #include "cmemory.h" 35 36 // Decimal digits in a 64-bit int 37 #define INT64_DIGITS 19 38 39 typedef enum EDigitListValues { 40 MAX_DBL_DIGITS = DBL_DIG, 41 MAX_I64_DIGITS = INT64_DIGITS, 42 MAX_DIGITS = MAX_I64_DIGITS, 43 MAX_EXPONENT = DBL_DIG, 44 DIGIT_PADDING = 3, 45 DEFAULT_DIGITS = 40, // Initial storage size, will grow as needed. 46 47 // "+." + fDigits + "e" + fDecimalAt 48 MAX_DEC_DIGITS = MAX_DIGITS + DIGIT_PADDING + MAX_EXPONENT 49 } EDigitListValues; 50 51 U_NAMESPACE_BEGIN 52 53 class CharString; 54 class DigitInterval; 55 56 // Export an explicit template instantiation of the MaybeStackHeaderAndArray that 57 // is used as a data member of DigitList. 58 // 59 // MSVC requires this, even though it should not be necessary. 60 // No direct access to the MaybeStackHeaderAndArray leaks out of the i18n library. 61 // 62 // Macintosh produces duplicate definition linker errors with the explicit template 63 // instantiation. 64 // 65 #if !U_PLATFORM_IS_DARWIN_BASED 66 template class U_I18N_API MaybeStackHeaderAndArray<decNumber, char, DEFAULT_DIGITS>; 67 #endif 68 69 70 enum EStackMode { kOnStack }; 71 72 enum EFastpathBits { kFastpathOk = 1, kNoDecimal = 2 }; 73 74 /** 75 * Digit List is actually a Decimal Floating Point number. 76 * The original implementation has been replaced by a thin wrapper onto a 77 * decimal number from the decNumber library. 78 * 79 * The original DigitList API has been retained, to minimize the impact of 80 * the change on the rest of the ICU formatting code. 81 * 82 * The change to decNumber enables support for big decimal numbers, and 83 * allows rounding computations to be done directly in decimal, avoiding 84 * extra, and inaccurate, conversions to and from doubles. 85 * 86 * Original DigitList comments: 87 * 88 * Digit List utility class. Private to DecimalFormat. Handles the transcoding 89 * between numeric values and strings of characters. Only handles 90 * non-negative numbers. The division of labor between DigitList and 91 * DecimalFormat is that DigitList handles the radix 10 representation 92 * issues; DecimalFormat handles the locale-specific issues such as 93 * positive/negative, grouping, decimal point, currency, and so on. 94 * <P> 95 * A DigitList is really a representation of a floating point value. 96 * It may be an integer value; we assume that a double has sufficient 97 * precision to represent all digits of a long. 98 * <P> 99 * The DigitList representation consists of a string of characters, 100 * which are the digits radix 10, from '0' to '9'. It also has a radix 101 * 10 exponent associated with it. The value represented by a DigitList 102 * object can be computed by mulitplying the fraction f, where 0 <= f < 1, 103 * derived by placing all the digits of the list to the right of the 104 * decimal point, by 10^exponent. 105 * 106 * -------- 107 * 108 * DigitList vs. decimalNumber: 109 * 110 * DigitList stores digits with the most significant first. 111 * decNumber stores digits with the least significant first. 112 * 113 * DigitList, decimal point is before the most significant. 114 * decNumber, decimal point is after the least signficant digit. 115 * 116 * digitList: 0.ddddd * 10 ^ exp 117 * decNumber: ddddd. * 10 ^ exp 118 * 119 * digitList exponent = decNumber exponent + digit count 120 * 121 * digitList, digits are platform invariant chars, '0' - '9' 122 * decNumber, digits are binary, one per byte, 0 - 9. 123 * 124 * (decNumber library is configurable in how digits are stored, ICU has configured 125 * it this way for convenience in replacing the old DigitList implementation.) 126 */ 127 class U_I18N_API DigitList : public UMemory { // Declare external to make compiler happy 128 public: 129 130 DigitList(); 131 ~DigitList(); 132 133 /* copy constructor 134 * @param DigitList The object to be copied. 135 * @return the newly created object. 136 */ 137 DigitList(const DigitList&); // copy constructor 138 139 /* assignment operator 140 * @param DigitList The object to be copied. 141 * @return the newly created object. 142 */ 143 DigitList& operator=(const DigitList&); // assignment operator 144 145 /** 146 * Return true if another object is semantically equal to this one. 147 * @param other The DigitList to be compared for equality 148 * @return true if another object is semantically equal to this one. 149 * return false otherwise. 150 */ 151 UBool operator==(const DigitList& other) const; 152 153 int32_t compare(const DigitList& other); 154 155 156 inline UBool operator!=(const DigitList& other) const { return !operator==(other); } 157 158 /** 159 * Clears out the digits. 160 * Use before appending them. 161 * Typically, you set a series of digits with append, then at the point 162 * you hit the decimal point, you set myDigitList.fDecimalAt = myDigitList.fCount; 163 * then go on appending digits. 164 */ 165 void clear(void); 166 167 /** 168 * Remove, by rounding, any fractional part of the decimal number, 169 * leaving an integer value. 170 */ 171 void toIntegralValue(); 172 173 /** 174 * Appends digits to the list. 175 * CAUTION: this function is not recommended for new code. 176 * In the original DigitList implementation, decimal numbers were 177 * parsed by appending them to a digit list as they were encountered. 178 * With the revamped DigitList based on decNumber, append is very 179 * inefficient, and the interaction with the exponent value is confusing. 180 * Best avoided. 181 * TODO: remove this function once all use has been replaced. 182 * TODO: describe alternative to append() 183 * @param digit The digit to be appended. 184 */ 185 void append(char digit); 186 187 /** 188 * Utility routine to get the value of the digit list 189 * Returns 0.0 if zero length. 190 * @return the value of the digit list. 191 */ 192 double getDouble(void) const; 193 194 /** 195 * Utility routine to get the value of the digit list 196 * Make sure that fitsIntoLong() is called before calling this function. 197 * Returns 0 if zero length. 198 * @return the value of the digit list, return 0 if it is zero length 199 */ 200 int32_t getLong(void) /*const*/; 201 202 /** 203 * Utility routine to get the value of the digit list 204 * Make sure that fitsIntoInt64() is called before calling this function. 205 * Returns 0 if zero length. 206 * @return the value of the digit list, return 0 if it is zero length 207 */ 208 int64_t getInt64(void) /*const*/; 209 210 /** 211 * Utility routine to get the value of the digit list as a decimal string. 212 */ 213 void getDecimal(CharString &str, UErrorCode &status); 214 215 /** 216 * Return true if the number represented by this object can fit into 217 * a long. 218 * @param ignoreNegativeZero True if negative zero is ignored. 219 * @return true if the number represented by this object can fit into 220 * a long, return false otherwise. 221 */ 222 UBool fitsIntoLong(UBool ignoreNegativeZero) /*const*/; 223 224 /** 225 * Return true if the number represented by this object can fit into 226 * an int64_t. 227 * @param ignoreNegativeZero True if negative zero is ignored. 228 * @return true if the number represented by this object can fit into 229 * a long, return false otherwise. 230 */ 231 UBool fitsIntoInt64(UBool ignoreNegativeZero) /*const*/; 232 233 /** 234 * Utility routine to set the value of the digit list from a double. 235 * @param source The value to be set 236 */ 237 void set(double source); 238 239 /** 240 * Utility routine to set the value of the digit list from a long. 241 * If a non-zero maximumDigits is specified, no more than that number of 242 * significant digits will be produced. 243 * @param source The value to be set 244 */ 245 void set(int32_t source); 246 247 /** 248 * Utility routine to set the value of the digit list from an int64. 249 * If a non-zero maximumDigits is specified, no more than that number of 250 * significant digits will be produced. 251 * @param source The value to be set 252 */ 253 void set(int64_t source); 254 255 /** 256 * Utility routine to set the value of the digit list from an int64. 257 * Does not set the decnumber unless requested later 258 * If a non-zero maximumDigits is specified, no more than that number of 259 * significant digits will be produced. 260 * @param source The value to be set 261 */ 262 void setInteger(int64_t source); 263 264 /** 265 * Utility routine to set the value of the digit list from a decimal number 266 * string. 267 * @param source The value to be set. The string must be nul-terminated. 268 * @param fastpathBits special flags for fast parsing 269 */ 270 void set(const StringPiece &source, UErrorCode &status, uint32_t fastpathBits = 0); 271 272 /** 273 * Multiply this = this * arg 274 * This digitlist will be expanded if necessary to accomodate the result. 275 * @param arg the number to multiply by. 276 */ 277 void mult(const DigitList &arg, UErrorCode &status); 278 279 /** 280 * Divide this = this / arg 281 */ 282 void div(const DigitList &arg, UErrorCode &status); 283 284 // The following functions replace direct access to the original DigitList implmentation 285 // data structures. 286 287 void setRoundingMode(DecimalFormat::ERoundingMode m); 288 289 /** Test a number for zero. 290 * @return TRUE if the number is zero 291 */ 292 UBool isZero(void) const; 293 294 /** Test for a Nan 295 * @return TRUE if the number is a NaN 296 */ isNaN(void)297 UBool isNaN(void) const {return decNumberIsNaN(fDecNumber);} 298 isInfinite()299 UBool isInfinite() const {return decNumberIsInfinite(fDecNumber);} 300 301 /** Reduce, or normalize. Removes trailing zeroes, adjusts exponent appropriately. */ 302 void reduce(); 303 304 /** Remove trailing fraction zeros, adjust exponent accordingly. */ 305 void trim(); 306 307 /** Set to zero */ setToZero()308 void setToZero() {uprv_decNumberZero(fDecNumber);} 309 310 /** get the number of digits in the decimal number */ digits()311 int32_t digits() const {return fDecNumber->digits;} 312 313 /** 314 * Round the number to the given number of digits. 315 * @param maximumDigits The maximum number of digits to be shown. 316 * Upon return, count will be less than or equal to maximumDigits. 317 * result is guaranteed to be trimmed. 318 */ 319 void round(int32_t maximumDigits); 320 321 void roundFixedPoint(int32_t maximumFractionDigits); 322 323 /** Ensure capacity for digits. Grow the storage if it is currently less than 324 * the requested size. Capacity is not reduced if it is already greater 325 * than requested. 326 */ 327 void ensureCapacity(int32_t requestedSize, UErrorCode &status); 328 isPositive(void)329 UBool isPositive(void) const { return decNumberIsNegative(fDecNumber) == 0;} 330 void setPositive(UBool s); 331 332 void setDecimalAt(int32_t d); 333 int32_t getDecimalAt(); 334 335 void setCount(int32_t c); 336 int32_t getCount() const; 337 338 /** 339 * Set the digit in platform (invariant) format, from '0'..'9' 340 * @param i index of digit 341 * @param v digit value, from '0' to '9' in platform invariant format 342 */ 343 void setDigit(int32_t i, char v); 344 345 /** 346 * Get the digit in platform (invariant) format, from '0'..'9' inclusive 347 * @param i index of digit 348 * @return invariant format of the digit 349 */ 350 char getDigit(int32_t i); 351 352 353 /** 354 * Get the digit's value, as an integer from 0..9 inclusive. 355 * Note that internally this value is a decNumberUnit, but ICU configures it to be a uint8_t. 356 * @param i index of digit 357 * @return value of that digit 358 */ 359 uint8_t getDigitValue(int32_t i); 360 361 /** 362 * Gets the upper bound exponent for this value. For 987, returns 3 363 * because 10^3 is the smallest power of 10 that is just greater than 364 * 987. 365 */ 366 int32_t getUpperExponent() const; 367 368 /** 369 * Gets the lower bound exponent for this value. For 98.7, returns -1 370 * because the right most digit, is the 10^-1 place. 371 */ getLowerExponent()372 int32_t getLowerExponent() const { return fDecNumber->exponent; } 373 374 /** 375 * Sets result to the smallest DigitInterval needed to display this 376 * DigitList in fixed point form and returns result. 377 */ 378 DigitInterval& getSmallestInterval(DigitInterval &result) const; 379 380 /** 381 * Like getDigitValue, but the digit is identified by exponent. 382 * For example, getDigitByExponent(7) returns the 10^7 place of this 383 * DigitList. Unlike getDigitValue, there are no upper or lower bounds 384 * for passed parameter. Instead, getDigitByExponent returns 0 if 385 * the exponent falls outside the interval for this DigitList. 386 */ 387 uint8_t getDigitByExponent(int32_t exponent) const; 388 389 /** 390 * Appends the digits in this object to a CharString. 391 * 3 is appended as (char) 3, not '3' 392 */ 393 void appendDigitsTo(CharString &str, UErrorCode &status) const; 394 395 /** 396 * Equivalent to roundFixedPoint(-digitExponent) except unlike 397 * roundFixedPoint, this works for any digitExponent value. 398 * If maxSigDigits is set then this instance is rounded to have no more 399 * than maxSigDigits. The end result is guaranteed to be trimmed. 400 */ 401 void roundAtExponent(int32_t digitExponent, int32_t maxSigDigits=INT32_MAX); 402 403 /** 404 * Quantizes according to some amount and rounds according to the 405 * context of this instance. Quantizing 3.233 with 0.05 gives 3.25. 406 */ 407 void quantize(const DigitList &amount, UErrorCode &status); 408 409 /** 410 * Like toScientific but only returns the exponent 411 * leaving this instance unchanged. 412 */ 413 int32_t getScientificExponent( 414 int32_t minIntDigitCount, int32_t exponentMultiplier) const; 415 416 /** 417 * Converts this instance to scientific notation. This instance 418 * becomes the mantissa and the exponent is returned. 419 * @param minIntDigitCount minimum integer digits in mantissa 420 * Exponent is set so that the actual number of integer digits 421 * in mantissa is as close to the minimum as possible. 422 * @param exponentMultiplier The exponent is always a multiple of 423 * This number. Usually 1, but set to 3 for engineering notation. 424 * @return exponent 425 */ 426 int32_t toScientific( 427 int32_t minIntDigitCount, int32_t exponentMultiplier); 428 429 /** 430 * Shifts decimal to the right. 431 */ 432 void shiftDecimalRight(int32_t numPlaces); 433 434 private: 435 /* 436 * These data members are intentionally public and can be set directly. 437 *<P> 438 * The value represented is given by placing the decimal point before 439 * fDigits[fDecimalAt]. If fDecimalAt is < 0, then leading zeros between 440 * the decimal point and the first nonzero digit are implied. If fDecimalAt 441 * is > fCount, then trailing zeros between the fDigits[fCount-1] and the 442 * decimal point are implied. 443 * <P> 444 * Equivalently, the represented value is given by f * 10^fDecimalAt. Here 445 * f is a value 0.1 <= f < 1 arrived at by placing the digits in fDigits to 446 * the right of the decimal. 447 * <P> 448 * DigitList is normalized, so if it is non-zero, fDigits[0] is non-zero. We 449 * don't allow denormalized numbers because our exponent is effectively of 450 * unlimited magnitude. The fCount value contains the number of significant 451 * digits present in fDigits[]. 452 * <P> 453 * Zero is represented by any DigitList with fCount == 0 or with each fDigits[i] 454 * for all i <= fCount == '0'. 455 * 456 * int32_t fDecimalAt; 457 * int32_t fCount; 458 * UBool fIsPositive; 459 * char *fDigits; 460 * DecimalFormat::ERoundingMode fRoundingMode; 461 */ 462 463 public: 464 decContext fContext; // public access to status flags. 465 466 private: 467 decNumber *fDecNumber; 468 MaybeStackHeaderAndArray<decNumber, char, DEFAULT_DIGITS> fStorage; 469 470 /* Cached double value corresponding to this decimal number. 471 * This is an optimization for the formatting implementation, which may 472 * ask for the double value multiple times. 473 */ 474 union DoubleOrInt64 { 475 double fDouble; 476 int64_t fInt64; 477 } fUnion; 478 enum EHave { 479 kNone=0, 480 kDouble, 481 kInt64 482 } fHave; 483 484 485 486 UBool shouldRoundUp(int32_t maximumDigits) const; 487 488 public: 489 490 #if U_OVERRIDE_CXX_ALLOCATION 491 using UMemory::operator new; 492 using UMemory::operator delete; 493 #else 494 static inline void * U_EXPORT2 operator new(size_t size) U_NO_THROW { return ::operator new(size); }; 495 static inline void U_EXPORT2 operator delete(void *ptr ) U_NO_THROW { ::operator delete(ptr); }; 496 #endif 497 static char U_EXPORT2 getStrtodDecimalSeparator(); 498 499 /** 500 * Placement new for stack usage 501 * @internal 502 */ new(size_t,void * onStack,EStackMode)503 static inline void * U_EXPORT2 operator new(size_t /*size*/, void * onStack, EStackMode /*mode*/) U_NO_THROW { return onStack; } 504 505 /** 506 * Placement delete for stack usage 507 * @internal 508 */ delete(void *,void *,EStackMode)509 static inline void U_EXPORT2 operator delete(void * /*ptr*/, void * /*onStack*/, EStackMode /*mode*/) U_NO_THROW {} 510 511 private: internalSetDouble(double d)512 inline void internalSetDouble(double d) { 513 fHave = kDouble; 514 fUnion.fDouble=d; 515 } internalSetInt64(int64_t d)516 inline void internalSetInt64(int64_t d) { 517 fHave = kInt64; 518 fUnion.fInt64=d; 519 } internalClear()520 inline void internalClear() { 521 fHave = kNone; 522 } 523 }; 524 525 526 U_NAMESPACE_END 527 528 #endif // #if !UCONFIG_NO_FORMATTING 529 #endif // _DIGITLST 530 531 //eof 532