OLD | NEW |
(Empty) | |
| 1 /* |
| 2 ******************************************************************************* |
| 3 * Copyright (C) 2004 - 2008, International Business Machines Corporation and |
| 4 * others. All Rights Reserved. |
| 5 ******************************************************************************* |
| 6 */ |
| 7 |
| 8 #ifndef UTMSCALE_H |
| 9 #define UTMSCALE_H |
| 10 |
| 11 #include "unicode/utypes.h" |
| 12 |
| 13 #if !UCONFIG_NO_FORMATTING |
| 14 |
| 15 /** |
| 16 * \file |
| 17 * \brief C API: Universal Time Scale |
| 18 * |
| 19 * There are quite a few different conventions for binary datetime, depending on
different |
| 20 * platforms and protocols. Some of these have severe drawbacks. For example, pe
ople using |
| 21 * Unix time (seconds since Jan 1, 1970) think that they are safe until near the
year 2038. |
| 22 * But cases can and do arise where arithmetic manipulations causes serious prob
lems. Consider |
| 23 * the computation of the average of two datetimes, for example: if one calculat
es them with |
| 24 * <code>averageTime = (time1 + time2)/2</code>, there will be overflow even wit
h dates |
| 25 * around the present. Moreover, even if these problems don't occur, there is th
e issue of |
| 26 * conversion back and forth between different systems. |
| 27 * |
| 28 * <p> |
| 29 * Binary datetimes differ in a number of ways: the datatype, the unit, |
| 30 * and the epoch (origin). We'll refer to these as time scales. For example: |
| 31 * |
| 32 * <table border="1" cellspacing="0" cellpadding="4"> |
| 33 * <caption>Table 1: Binary Time Scales</caption> |
| 34 * <tr> |
| 35 * <th align="left">Source</th> |
| 36 * <th align="left">Datatype</th> |
| 37 * <th align="left">Unit</th> |
| 38 * <th align="left">Epoch</th> |
| 39 * </tr> |
| 40 * |
| 41 * <tr> |
| 42 * <td>UDTS_JAVA_TIME</td> |
| 43 * <td>int64_t</td> |
| 44 * <td>milliseconds</td> |
| 45 * <td>Jan 1, 1970</td> |
| 46 * </tr> |
| 47 * <tr> |
| 48 * |
| 49 * <td>UDTS_UNIX_TIME</td> |
| 50 * <td>int32_t or int64_t</td> |
| 51 * <td>seconds</td> |
| 52 * <td>Jan 1, 1970</td> |
| 53 * </tr> |
| 54 * <tr> |
| 55 * <td>UDTS_ICU4C_TIME</td> |
| 56 * |
| 57 * <td>double</td> |
| 58 * <td>milliseconds</td> |
| 59 * <td>Jan 1, 1970</td> |
| 60 * </tr> |
| 61 * <tr> |
| 62 * <td>UDTS_WINDOWS_FILE_TIME</td> |
| 63 * <td>int64_t</td> |
| 64 * |
| 65 * <td>ticks (100 nanoseconds)</td> |
| 66 * <td>Jan 1, 1601</td> |
| 67 * </tr> |
| 68 * <tr> |
| 69 * <td>UDTS_DOTNET_DATE_TIME</td> |
| 70 * <td>int64_t</td> |
| 71 * <td>ticks (100 nanoseconds)</td> |
| 72 * |
| 73 * <td>Jan 1, 0001</td> |
| 74 * </tr> |
| 75 * <tr> |
| 76 * <td>UDTS_MAC_OLD_TIME</td> |
| 77 * <td>int32_t or int64_t</td> |
| 78 * <td>seconds</td> |
| 79 * <td>Jan 1, 1904</td> |
| 80 * |
| 81 * </tr> |
| 82 * <tr> |
| 83 * <td>UDTS_MAC_TIME</td> |
| 84 * <td>double</td> |
| 85 * <td>seconds</td> |
| 86 * <td>Jan 1, 2001</td> |
| 87 * </tr> |
| 88 * |
| 89 * <tr> |
| 90 * <td>UDTS_EXCEL_TIME</td> |
| 91 * <td>?</td> |
| 92 * <td>days</td> |
| 93 * <td>Dec 31, 1899</td> |
| 94 * </tr> |
| 95 * <tr> |
| 96 * |
| 97 * <td>UDTS_DB2_TIME</td> |
| 98 * <td>?</td> |
| 99 * <td>days</td> |
| 100 * <td>Dec 31, 1899</td> |
| 101 * </tr> |
| 102 * |
| 103 * <tr> |
| 104 * <td>UDTS_UNIX_MICROSECONDS_TIME</td> |
| 105 * <td>int64_t</td> |
| 106 * <td>microseconds</td> |
| 107 * <td>Jan 1, 1970</td> |
| 108 * </tr> |
| 109 * </table> |
| 110 * |
| 111 * <p> |
| 112 * All of the epochs start at 00:00 am (the earliest possible time on the day in
question), |
| 113 * and are assumed to be UTC. |
| 114 * |
| 115 * <p> |
| 116 * The ranges for different datatypes are given in the following table (all valu
es in years). |
| 117 * The range of years includes the entire range expressible with positive and ne
gative |
| 118 * values of the datatype. The range of years for double is the range that would
be allowed |
| 119 * without losing precision to the corresponding unit. |
| 120 * |
| 121 * <table border="1" cellspacing="0" cellpadding="4"> |
| 122 * <tr> |
| 123 * <th align="left">Units</th> |
| 124 * <th align="left">int64_t</th> |
| 125 * <th align="left">double</th> |
| 126 * <th align="left">int32_t</th> |
| 127 * </tr> |
| 128 * |
| 129 * <tr> |
| 130 * <td>1 sec</td> |
| 131 * <td align="right">5.84542x10<sup>11</sup></td> |
| 132 * <td align="right">285,420,920.94</td> |
| 133 * <td align="right">136.10</td> |
| 134 * </tr> |
| 135 * <tr> |
| 136 * |
| 137 * <td>1 millisecond</td> |
| 138 * <td align="right">584,542,046.09</td> |
| 139 * <td align="right">285,420.92</td> |
| 140 * <td align="right">0.14</td> |
| 141 * </tr> |
| 142 * <tr> |
| 143 * <td>1 microsecond</td> |
| 144 * |
| 145 * <td align="right">584,542.05</td> |
| 146 * <td align="right">285.42</td> |
| 147 * <td align="right">0.00</td> |
| 148 * </tr> |
| 149 * <tr> |
| 150 * <td>100 nanoseconds (tick)</td> |
| 151 * <td align="right">58,454.20</td> |
| 152 * <td align="right">28.54</td> |
| 153 * <td align="right">0.00</td> |
| 154 * </tr> |
| 155 * <tr> |
| 156 * <td>1 nanosecond</td> |
| 157 * <td align="right">584.5420461</td> |
| 158 * <td align="right">0.2854</td> |
| 159 * <td align="right">0.00</td> |
| 160 * </tr> |
| 161 * </table> |
| 162 * |
| 163 * <p> |
| 164 * These functions implement a universal time scale which can be used as a 'pivo
t', |
| 165 * and provide conversion functions to and from all other major time scales. |
| 166 * This datetimes to be converted to the pivot time, safely manipulated, |
| 167 * and converted back to any other datetime time scale. |
| 168 * |
| 169 *<p> |
| 170 * So what to use for this pivot? Java time has plenty of range, but cannot repr
esent |
| 171 * .NET <code>System.DateTime</code> values without severe loss of precision. IC
U4C time addresses this by using a |
| 172 * <code>double</code> that is otherwise equivalent to the Java time. However, t
here are disadvantages |
| 173 * with <code>doubles</code>. They provide for much more graceful degradation in
arithmetic operations. |
| 174 * But they only have 53 bits of accuracy, which means that they will lose preci
sion when |
| 175 * converting back and forth to ticks. What would really be nice would be a |
| 176 * <code>long double</code> (80 bits -- 64 bit mantissa), but that is not suppor
ted on most systems. |
| 177 * |
| 178 *<p> |
| 179 * The Unix extended time uses a structure with two components: time in seconds
and a |
| 180 * fractional field (microseconds). However, this is clumsy, slow, and |
| 181 * prone to error (you always have to keep track of overflow and underflow in th
e |
| 182 * fractional field). <code>BigDecimal</code> would allow for arbitrary precisio
n and arbitrary range, |
| 183 * but we do not want to use this as the normal type, because it is slow and doe
s not |
| 184 * have a fixed size. |
| 185 * |
| 186 *<p> |
| 187 * Because of these issues, we ended up concluding that the .NET framework's |
| 188 * <code>System.DateTime</code> would be the best pivot. However, we use the ful
l range |
| 189 * allowed by the datatype, allowing for datetimes back to 29,000 BC and up to 2
9,000 AD. |
| 190 * This time scale is very fine grained, does not lose precision, and covers a r
ange that |
| 191 * will meet almost all requirements. It will not handle the range that Java tim
es do, |
| 192 * but frankly, being able to handle dates before 29,000 BC or after 29,000 AD i
s of very limited interest. |
| 193 * |
| 194 */ |
| 195 |
| 196 /** |
| 197 * <code>UDateTimeScale</code> values are used to specify the time scale used fo
r |
| 198 * conversion into or out if the universal time scale. |
| 199 * |
| 200 * @stable ICU 3.2 |
| 201 */ |
| 202 typedef enum UDateTimeScale { |
| 203 /** |
| 204 * Used in the JDK. Data is a Java <code>long</code> (<code>int64_t</code>).
Value |
| 205 * is milliseconds since January 1, 1970. |
| 206 * |
| 207 * @stable ICU 3.2 |
| 208 */ |
| 209 UDTS_JAVA_TIME = 0, |
| 210 |
| 211 /** |
| 212 * Used on Unix systems. Data is <code>int32_t</code> or <code>int64_t</code
>. Value |
| 213 * is seconds since January 1, 1970. |
| 214 * |
| 215 * @stable ICU 3.2 |
| 216 */ |
| 217 UDTS_UNIX_TIME, |
| 218 |
| 219 /** |
| 220 * Used in IUC4C. Data is a <code>double</code>. Value |
| 221 * is milliseconds since January 1, 1970. |
| 222 * |
| 223 * @stable ICU 3.2 |
| 224 */ |
| 225 UDTS_ICU4C_TIME, |
| 226 |
| 227 /** |
| 228 * Used in Windows for file times. Data is an <code>int64_t</code>. Value |
| 229 * is ticks (1 tick == 100 nanoseconds) since January 1, 1601. |
| 230 * |
| 231 * @stable ICU 3.2 |
| 232 */ |
| 233 UDTS_WINDOWS_FILE_TIME, |
| 234 |
| 235 /** |
| 236 * Used in the .NET framework's <code>System.DateTime</code> structure. Data
is an <code>int64_t</code>. Value |
| 237 * is ticks (1 tick == 100 nanoseconds) since January 1, 0001. |
| 238 * |
| 239 * @stable ICU 3.2 |
| 240 */ |
| 241 UDTS_DOTNET_DATE_TIME, |
| 242 |
| 243 /** |
| 244 * Used in older Macintosh systems. Data is <code>int32_t</code> or <code>in
t64_t</code>. Value |
| 245 * is seconds since January 1, 1904. |
| 246 * |
| 247 * @stable ICU 3.2 |
| 248 */ |
| 249 UDTS_MAC_OLD_TIME, |
| 250 |
| 251 /** |
| 252 * Used in newer Macintosh systems. Data is a <code>double</code>. Value |
| 253 * is seconds since January 1, 2001. |
| 254 * |
| 255 * @stable ICU 3.2 |
| 256 */ |
| 257 UDTS_MAC_TIME, |
| 258 |
| 259 /** |
| 260 * Used in Excel. Data is an <code>?unknown?</code>. Value |
| 261 * is days since December 31, 1899. |
| 262 * |
| 263 * @stable ICU 3.2 |
| 264 */ |
| 265 UDTS_EXCEL_TIME, |
| 266 |
| 267 /** |
| 268 * Used in DB2. Data is an <code>?unknown?</code>. Value |
| 269 * is days since December 31, 1899. |
| 270 * |
| 271 * @stable ICU 3.2 |
| 272 */ |
| 273 UDTS_DB2_TIME, |
| 274 |
| 275 /** |
| 276 * Data is a <code>long</code>. Value is microseconds since January 1, 1970. |
| 277 * Similar to Unix time (linear value from 1970) and struct timeval |
| 278 * (microseconds resolution). |
| 279 * |
| 280 * @stable ICU 3.8 |
| 281 */ |
| 282 UDTS_UNIX_MICROSECONDS_TIME, |
| 283 |
| 284 /** |
| 285 * The first unused time scale value. The limit of this enum |
| 286 */ |
| 287 UDTS_MAX_SCALE |
| 288 } UDateTimeScale; |
| 289 |
| 290 /** |
| 291 * <code>UTimeScaleValue</code> values are used to specify the time scale values |
| 292 * to <code>utmscale_getTimeScaleValue</code>. |
| 293 * |
| 294 * @see utmscale_getTimeScaleValue |
| 295 * |
| 296 * @stable ICU 3.2 |
| 297 */ |
| 298 typedef enum UTimeScaleValue { |
| 299 /** |
| 300 * The constant used to select the units vale |
| 301 * for a time scale. |
| 302 * |
| 303 * @see utmscale_getTimeScaleValue |
| 304 * |
| 305 * @stable ICU 3.2 |
| 306 */ |
| 307 UTSV_UNITS_VALUE = 0, |
| 308 |
| 309 /** |
| 310 * The constant used to select the epoch offset value |
| 311 * for a time scale. |
| 312 * |
| 313 * @see utmscale_getTimeScaleValue |
| 314 * |
| 315 * @stable ICU 3.2 |
| 316 */ |
| 317 UTSV_EPOCH_OFFSET_VALUE=1, |
| 318 |
| 319 /** |
| 320 * The constant used to select the minimum from value |
| 321 * for a time scale. |
| 322 * |
| 323 * @see utmscale_getTimeScaleValue |
| 324 * |
| 325 * @stable ICU 3.2 |
| 326 */ |
| 327 UTSV_FROM_MIN_VALUE=2, |
| 328 |
| 329 /** |
| 330 * The constant used to select the maximum from value |
| 331 * for a time scale. |
| 332 * |
| 333 * @see utmscale_getTimeScaleValue |
| 334 * |
| 335 * @stable ICU 3.2 |
| 336 */ |
| 337 UTSV_FROM_MAX_VALUE=3, |
| 338 |
| 339 /** |
| 340 * The constant used to select the minimum to value |
| 341 * for a time scale. |
| 342 * |
| 343 * @see utmscale_getTimeScaleValue |
| 344 * |
| 345 * @stable ICU 3.2 |
| 346 */ |
| 347 UTSV_TO_MIN_VALUE=4, |
| 348 |
| 349 /** |
| 350 * The constant used to select the maximum to value |
| 351 * for a time scale. |
| 352 * |
| 353 * @see utmscale_getTimeScaleValue |
| 354 * |
| 355 * @stable ICU 3.2 |
| 356 */ |
| 357 UTSV_TO_MAX_VALUE=5, |
| 358 |
| 359 #ifndef U_HIDE_INTERNAL_API |
| 360 /** |
| 361 * The constant used to select the epoch plus one value |
| 362 * for a time scale. |
| 363 * |
| 364 * NOTE: This is an internal value. DO NOT USE IT. May not |
| 365 * actually be equal to the epoch offset value plus one. |
| 366 * |
| 367 * @see utmscale_getTimeScaleValue |
| 368 * |
| 369 * @internal ICU 3.2 |
| 370 */ |
| 371 UTSV_EPOCH_OFFSET_PLUS_1_VALUE=6, |
| 372 |
| 373 /** |
| 374 * The constant used to select the epoch plus one value |
| 375 * for a time scale. |
| 376 * |
| 377 * NOTE: This is an internal value. DO NOT USE IT. May not |
| 378 * actually be equal to the epoch offset value plus one. |
| 379 * |
| 380 * @see utmscale_getTimeScaleValue |
| 381 * |
| 382 * @internal ICU 3.2 |
| 383 */ |
| 384 UTSV_EPOCH_OFFSET_MINUS_1_VALUE=7, |
| 385 |
| 386 /** |
| 387 * The constant used to select the units round value |
| 388 * for a time scale. |
| 389 * |
| 390 * NOTE: This is an internal value. DO NOT USE IT. |
| 391 * |
| 392 * @see utmscale_getTimeScaleValue |
| 393 * |
| 394 * @internal ICU 3.2 |
| 395 */ |
| 396 UTSV_UNITS_ROUND_VALUE=8, |
| 397 |
| 398 /** |
| 399 * The constant used to select the minimum safe rounding value |
| 400 * for a time scale. |
| 401 * |
| 402 * NOTE: This is an internal value. DO NOT USE IT. |
| 403 * |
| 404 * @see utmscale_getTimeScaleValue |
| 405 * |
| 406 * @internal ICU 3.2 |
| 407 */ |
| 408 UTSV_MIN_ROUND_VALUE=9, |
| 409 |
| 410 /** |
| 411 * The constant used to select the maximum safe rounding value |
| 412 * for a time scale. |
| 413 * |
| 414 * NOTE: This is an internal value. DO NOT USE IT. |
| 415 * |
| 416 * @see utmscale_getTimeScaleValue |
| 417 * |
| 418 * @internal ICU 3.2 |
| 419 */ |
| 420 UTSV_MAX_ROUND_VALUE=10, |
| 421 |
| 422 #endif /* U_HIDE_INTERNAL_API */ |
| 423 |
| 424 /** |
| 425 * The number of time scale values, in other words limit of this enum. |
| 426 * |
| 427 * @see utmscale_getTimeScaleValue |
| 428 */ |
| 429 UTSV_MAX_SCALE_VALUE=11 |
| 430 |
| 431 } UTimeScaleValue; |
| 432 |
| 433 /** |
| 434 * Get a value associated with a particular time scale. |
| 435 * |
| 436 * @param timeScale The time scale |
| 437 * @param value A constant representing the value to get |
| 438 * @param status The status code. Set to <code>U_ILLEGAL_ARGUMENT_ERROR</code> i
f arguments are invalid. |
| 439 * @return - the value. |
| 440 * |
| 441 * @stable ICU 3.2 |
| 442 */ |
| 443 U_STABLE int64_t U_EXPORT2 |
| 444 utmscale_getTimeScaleValue(UDateTimeScale timeScale, UTimeScaleValue value,
UErrorCode *status); |
| 445 |
| 446 /* Conversion to 'universal time scale' */ |
| 447 |
| 448 /** |
| 449 * Convert a <code>int64_t</code> datetime from the given time scale to the univ
ersal time scale. |
| 450 * |
| 451 * @param otherTime The <code>int64_t</code> datetime |
| 452 * @param timeScale The time scale to convert from |
| 453 * @param status The status code. Set to <code>U_ILLEGAL_ARGUMENT_ERROR</code> i
f the conversion is out of range. |
| 454 * |
| 455 * @return The datetime converted to the universal time scale |
| 456 * |
| 457 * @stable ICU 3.2 |
| 458 */ |
| 459 U_STABLE int64_t U_EXPORT2 |
| 460 utmscale_fromInt64(int64_t otherTime, UDateTimeScale timeScale, UErrorCode *
status); |
| 461 |
| 462 /* Conversion from 'universal time scale' */ |
| 463 |
| 464 /** |
| 465 * Convert a datetime from the universal time scale to a <code>int64_t</code> in
the given time scale. |
| 466 * |
| 467 * @param universalTime The datetime in the universal time scale |
| 468 * @param timeScale The time scale to convert to |
| 469 * @param status The status code. Set to <code>U_ILLEGAL_ARGUMENT_ERROR</code> i
f the conversion is out of range. |
| 470 * |
| 471 * @return The datetime converted to the given time scale |
| 472 * |
| 473 * @stable ICU 3.2 |
| 474 */ |
| 475 U_STABLE int64_t U_EXPORT2 |
| 476 utmscale_toInt64(int64_t universalTime, UDateTimeScale timeScale, UErrorCode
*status); |
| 477 |
| 478 #endif /* #if !UCONFIG_NO_FORMATTING */ |
| 479 |
| 480 #endif |
| 481 |
OLD | NEW |