You are viewing a plain text version of this content. The canonical link for it is here.
Posted to commits@activemq.apache.org by ta...@apache.org on 2007/08/06 21:01:46 UTC

svn commit: r563223 - in /activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util: NumberConverter.cpp NumberConverter.h

Author: tabish
Date: Mon Aug  6 12:01:45 2007
New Revision: 563223

URL: http://svn.apache.org/viewvc?view=rev&rev=563223
Log:
http://issues.apache.org/activemq/browse/AMQCPP-103

Implementing the Primitive Wrappers fully

Added:
    activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.cpp
    activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.h

Added: activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.cpp
URL: http://svn.apache.org/viewvc/activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.cpp?view=auto&rev=563223
==============================================================================
--- activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.cpp (added)
+++ activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.cpp Mon Aug  6 12:01:45 2007
@@ -0,0 +1,306 @@
+/*
+ *  Licensed to the Apache Software Foundation (ASF) under one or more
+ *  contributor license agreements.  See the NOTICE file distributed with
+ *  this work for additional information regarding copyright ownership.
+ *  The ASF licenses this file to You under the Apache License, Version 2.0
+ *  (the "License"); you may not use this file except in compliance with
+ *  the License.  You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software
+ *  distributed under the License is distributed on an "AS IS" BASIS,
+ *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ *  See the License for the specific language governing permissions and
+ *  limitations under the License.
+ */
+
+#include "NumberConverter.h"
+
+using namespace decaf;
+using namespace decaf::internal;
+using namespace decaf::internal::util;
+
+////////////////////////////////////////////////////////////////////////////////
+NumberConverter::NumberConverter() {
+}
+
+
+public final class NumberConverter {
+
+    private int setCount; // number of times u and k have been gotten
+
+    private int getCount; // number of times u and k have been set
+
+    private int[] uArray = new int[64];
+
+    private int firstK;
+
+    private final static double invLogOfTenBaseTwo = Math.log(2.0)
+            / Math.log(10.0);
+
+    private final static long[] TEN_TO_THE = new long[20];
+
+    static {
+        TEN_TO_THE[0] = 1L;
+        for (int i = 1; i < TEN_TO_THE.length; ++i) {
+            long previous = TEN_TO_THE[i - 1];
+            TEN_TO_THE[i] = (previous << 1) + (previous << 3);
+        }
+    }
+
+    private static NumberConverter getConverter() {
+        return new NumberConverter();
+    }
+
+    public static String convert(double input) {
+        return getConverter().convertD(input);
+    }
+
+    public static String convert(float input) {
+        return getConverter().convertF(input);
+    }
+
+    public String convertD(double inputNumber) {
+        int p = 1023 + 52; // the power offset (precision)
+        long signMask = 0x8000000000000000L; // the mask to get the sign of
+        // the number
+        long eMask = 0x7FF0000000000000L; // the mask to get the power bits
+        long fMask = 0x000FFFFFFFFFFFFFL; // the mask to get the significand
+        // bits
+
+        long inputNumberBits = Double.doubleToLongBits(inputNumber);
+        // the value of the sign... 0 is positive, ~0 is negative
+        String signString = (inputNumberBits & signMask) == 0 ? "" : "-";
+        // the value of the 'power bits' of the inputNumber
+        int e = (int) ((inputNumberBits & eMask) >> 52);
+        // the value of the 'significand bits' of the inputNumber
+        long f = inputNumberBits & fMask;
+        boolean mantissaIsZero = f == 0;
+        int pow = 0, numBits = 52;
+
+        if (e == 2047)
+            return mantissaIsZero ? signString + "Infinity" : "NaN";
+        if (e == 0) {
+            if (mantissaIsZero)
+                return signString + "0.0";
+            if (f == 1)
+                // special case to increase precision even though 2 *
+                // Double.MIN_VALUE is 1.0e-323
+                return signString + "4.9E-324";
+            pow = 1 - p; // a denormalized number
+            long ff = f;
+            while ((ff & 0x0010000000000000L) == 0) {
+                ff = ff << 1;
+                numBits--;
+            }
+        } else {
+            // 0 < e < 2047
+            // a "normalized" number
+            f = f | 0x0010000000000000L;
+            pow = e - p;
+        }
+
+        if (-59 < pow && pow < 6 || (pow == -59 && !mantissaIsZero))
+            longDigitGenerator(f, pow, e == 0, mantissaIsZero, numBits);
+        else
+            bigIntDigitGeneratorInstImpl(f, pow, e == 0, mantissaIsZero,
+                    numBits);
+
+        if (inputNumber >= 1e7D || inputNumber <= -1e7D
+                || (inputNumber > -1e-3D && inputNumber < 1e-3D))
+            return signString + freeFormatExponential();
+
+        return signString + freeFormat();
+    }
+
+    public String convertF(float inputNumber) {
+        int p = 127 + 23; // the power offset (precision)
+        int signMask = 0x80000000; // the mask to get the sign of the number
+        int eMask = 0x7F800000; // the mask to get the power bits
+        int fMask = 0x007FFFFF; // the mask to get the significand bits
+
+        int inputNumberBits = Float.floatToIntBits(inputNumber);
+        // the value of the sign... 0 is positive, ~0 is negative
+        String signString = (inputNumberBits & signMask) == 0 ? "" : "-";
+        // the value of the 'power bits' of the inputNumber
+        int e = (inputNumberBits & eMask) >> 23;
+        // the value of the 'significand bits' of the inputNumber
+        int f = inputNumberBits & fMask;
+        boolean mantissaIsZero = f == 0;
+        int pow = 0, numBits = 23;
+
+        if (e == 255)
+            return mantissaIsZero ? signString + "Infinity" : "NaN";
+        if (e == 0) {
+            if (mantissaIsZero)
+                return signString + "0.0";
+            pow = 1 - p; // a denormalized number
+            if (f < 8) { // want more precision with smallest values
+                f = f << 2;
+                pow -= 2;
+            }
+            int ff = f;
+            while ((ff & 0x00800000) == 0) {
+                ff = ff << 1;
+                numBits--;
+            }
+        } else {
+            // 0 < e < 255
+            // a "normalized" number
+            f = f | 0x00800000;
+            pow = e - p;
+        }
+
+        if (-59 < pow && pow < 35 || (pow == -59 && !mantissaIsZero))
+            longDigitGenerator(f, pow, e == 0, mantissaIsZero, numBits);
+        else
+            bigIntDigitGeneratorInstImpl(f, pow, e == 0, mantissaIsZero,
+                    numBits);
+        if (inputNumber >= 1e7f || inputNumber <= -1e7f
+                || (inputNumber > -1e-3f && inputNumber < 1e-3f))
+            return signString + freeFormatExponential();
+
+        return signString + freeFormat();
+    }
+
+    private String freeFormatExponential() {
+        // corresponds to process "Free-Format Exponential"
+        char[] formattedDecimal = new char[25];
+        formattedDecimal[0] = (char) ('0' + uArray[getCount++]);
+        formattedDecimal[1] = '.';
+        // the position the next character is to be inserted into
+        // formattedDecimal
+        int charPos = 2;
+
+        int k = firstK;
+        int expt = k;
+        while (true) {
+            k--;
+            if (getCount >= setCount)
+                break;
+
+            formattedDecimal[charPos++] = (char) ('0' + uArray[getCount++]);
+        }
+
+        if (k == expt - 1)
+            formattedDecimal[charPos++] = '0';
+        formattedDecimal[charPos++] = 'E';
+        return new String(formattedDecimal, 0, charPos)
+                + Integer.toString(expt);
+    }
+
+    private String freeFormat() {
+        // corresponds to process "Free-Format"
+        char[] formattedDecimal = new char[25];
+        // the position the next character is to be inserted into
+        // formattedDecimal
+        int charPos = 0;
+        int k = firstK;
+        if (k < 0) {
+            formattedDecimal[0] = '0';
+            formattedDecimal[1] = '.';
+            charPos += 2;
+            for (int i = k + 1; i < 0; i++)
+                formattedDecimal[charPos++] = '0';
+        }
+
+        int U = uArray[getCount++];
+        do {
+            if (U != -1)
+                formattedDecimal[charPos++] = (char) ('0' + U);
+            else if (k >= -1)
+                formattedDecimal[charPos++] = '0';
+
+            if (k == 0)
+                formattedDecimal[charPos++] = '.';
+
+            k--;
+            U = getCount < setCount ? uArray[getCount++] : -1;
+        } while (U != -1 || k >= -1);
+        return new String(formattedDecimal, 0, charPos);
+    }
+
+    private native void bigIntDigitGeneratorInstImpl(long f, int e,
+            boolean isDenormalized, boolean mantissaIsZero, int p);
+
+    private void longDigitGenerator(long f, int e, boolean isDenormalized,
+            boolean mantissaIsZero, int p) {
+        long R, S, M;
+        if (e >= 0) {
+            M = 1l << e;
+            if (!mantissaIsZero) {
+                R = f << (e + 1);
+                S = 2;
+            } else {
+                R = f << (e + 2);
+                S = 4;
+            }
+        } else {
+            M = 1;
+            if (isDenormalized || !mantissaIsZero) {
+                R = f << 1;
+                S = 1l << (1 - e);
+            } else {
+                R = f << 2;
+                S = 1l << (2 - e);
+            }
+        }
+
+        int k = (int) Math.ceil((e + p - 1) * invLogOfTenBaseTwo - 1e-10);
+
+        if (k > 0) {
+            S = S * TEN_TO_THE[k];
+        } else if (k < 0) {
+            long scale = TEN_TO_THE[-k];
+            R = R * scale;
+            M = M == 1 ? scale : M * scale;
+        }
+
+        if (R + M > S) { // was M_plus
+            firstK = k;
+        } else {
+            firstK = k - 1;
+            R = R * 10;
+            M = M * 10;
+        }
+
+        getCount = setCount = 0; // reset indices
+        boolean low, high;
+        int U;
+        long[] Si = new long[] { S, S << 1, S << 2, S << 3 };
+        while (true) {
+            // set U to be floor (R / S) and R to be the remainder
+            // using a kind of "binary search" to find the answer.
+            // It's a lot quicker than actually dividing since we know
+            // the answer will be between 0 and 10
+            U = 0;
+            long remainder;
+            for (int i = 3; i >= 0; i--) {
+                remainder = R - Si[i];
+                if (remainder >= 0) {
+                    R = remainder;
+                    U += 1 << i;
+                }
+            }
+
+            low = R < M; // was M_minus
+            high = R + M > S; // was M_plus
+
+            if (low || high)
+                break;
+
+            R = R * 10;
+            M = M * 10;
+            uArray[setCount++] = U;
+        }
+        if (low && !high)
+            uArray[setCount++] = U;
+        else if (high && !low)
+            uArray[setCount++] = U + 1;
+        else if ((R << 1) < S)
+            uArray[setCount++] = U;
+        else
+            uArray[setCount++] = U + 1;
+    }
+}

Added: activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.h
URL: http://svn.apache.org/viewvc/activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.h?view=auto&rev=563223
==============================================================================
--- activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.h (added)
+++ activemq/activemq-cpp/trunk/src/decaf/src/main/decaf/internal/util/NumberConverter.h Mon Aug  6 12:01:45 2007
@@ -0,0 +1,39 @@
+/*
+ *  Licensed to the Apache Software Foundation (ASF) under one or more
+ *  contributor license agreements.  See the NOTICE file distributed with
+ *  this work for additional information regarding copyright ownership.
+ *  The ASF licenses this file to You under the Apache License, Version 2.0
+ *  (the "License"); you may not use this file except in compliance with
+ *  the License.  You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ *  Unless required by applicable law or agreed to in writing, software
+ *  distributed under the License is distributed on an "AS IS" BASIS,
+ *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ *  See the License for the specific language governing permissions and
+ *  limitations under the License.
+ */
+
+#ifndef _DECAF_INTERNAL_UTIL_NUMBERCONVERTER_H_
+#define _DECAF_INTERNAL_UTIL_NUMBERCONVERTER_H_
+
+namespace decaf{
+namespace internal{
+namespace util{
+
+    class NumberConverter {
+    private:
+
+    public:
+
+        NumberConverter();
+        virtual ~NumberConverter() {}
+
+    public:   // Statics
+
+    };
+
+}}}
+
+#endif /*_DECAF_INTERNAL_UTIL_NUMBERCONVERTER_H_*/