CachedTranscendentalFunction.h 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. /*
  2. * Copyright (C) 2010 Apple Inc. All rights reserved.
  3. *
  4. * Redistribution and use in source and binary forms, with or without
  5. * modification, are permitted provided that the following conditions
  6. * are met:
  7. * 1. Redistributions of source code must retain the above copyright
  8. * notice, this list of conditions and the following disclaimer.
  9. * 2. Redistributions in binary form must reproduce the above copyright
  10. * notice, this list of conditions and the following disclaimer in the
  11. * documentation and/or other materials provided with the distribution.
  12. *
  13. * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
  14. * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  15. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
  16. * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
  17. * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
  18. * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
  19. * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  20. * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
  21. * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  22. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  23. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  24. */
  25. #ifndef CachedTranscendentalFunction_h
  26. #define CachedTranscendentalFunction_h
  27. #include "JSCJSValue.h"
  28. namespace JSC {
  29. typedef double (*TranscendentalFunctionPtr)(double);
  30. // CachedTranscendentalFunction provides a generic mechanism to cache results
  31. // for pure functions with the signature "double func(double)", and where NaN
  32. // maps to NaN.
  33. template<TranscendentalFunctionPtr orignalFunction>
  34. class CachedTranscendentalFunction {
  35. struct CacheEntry {
  36. double operand;
  37. double result;
  38. };
  39. public:
  40. CachedTranscendentalFunction()
  41. : m_cache(0)
  42. {
  43. }
  44. ~CachedTranscendentalFunction()
  45. {
  46. if (m_cache)
  47. fastFree(m_cache);
  48. }
  49. JSValue operator() (double operand)
  50. {
  51. if (UNLIKELY(!m_cache))
  52. initialize();
  53. CacheEntry* entry = &m_cache[hash(operand)];
  54. if (entry->operand == operand)
  55. return jsDoubleNumber(entry->result);
  56. double result = orignalFunction(operand);
  57. entry->operand = operand;
  58. entry->result = result;
  59. return jsDoubleNumber(result);
  60. }
  61. private:
  62. void initialize()
  63. {
  64. // Lazily allocate the table, populate with NaN->NaN mapping.
  65. m_cache = static_cast<CacheEntry*>(fastMalloc(s_cacheSize * sizeof(CacheEntry)));
  66. for (unsigned x = 0; x < s_cacheSize; ++x) {
  67. m_cache[x].operand = QNaN;
  68. m_cache[x].result = QNaN;
  69. }
  70. }
  71. static unsigned hash(double d)
  72. {
  73. union doubleAndUInt64 {
  74. double d;
  75. uint32_t is[2];
  76. } u;
  77. u.d = d;
  78. unsigned x = u.is[0] ^ u.is[1];
  79. x = (x >> 20) ^ (x >> 8);
  80. return x & (s_cacheSize - 1);
  81. }
  82. static const unsigned s_cacheSize = 0x1000;
  83. CacheEntry* m_cache;
  84. };
  85. }
  86. #endif // CachedTranscendentalFunction_h