DynFixedImpl.java revision 608:7e06bf1dcb09
1/*
2 * Copyright (c) 2000, 2003, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation.  Oracle designates this
8 * particular file as subject to the "Classpath" exception as provided
9 * by Oracle in the LICENSE file that accompanied this code.
10 *
11 * This code is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
14 * version 2 for more details (a copy is included in the LICENSE file that
15 * accompanied this code).
16 *
17 * You should have received a copy of the GNU General Public License version
18 * 2 along with this work; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20 *
21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22 * or visit www.oracle.com if you need additional information or have any
23 * questions.
24 */
25
26package com.sun.corba.se.impl.dynamicany;
27
28import org.omg.CORBA.TypeCode;
29import org.omg.CORBA.Any;
30import org.omg.CORBA.NO_IMPLEMENT;
31import org.omg.DynamicAny.*;
32import org.omg.DynamicAny.DynAnyPackage.*;
33import java.math.BigDecimal;
34import java.math.BigInteger;
35import org.omg.CORBA.TypeCodePackage.BadKind;
36
37import com.sun.corba.se.spi.orb.ORB ;
38import com.sun.corba.se.spi.logging.CORBALogDomains ;
39import com.sun.corba.se.impl.logging.ORBUtilSystemException ;
40
41public class DynFixedImpl extends DynAnyBasicImpl implements DynFixed
42{
43    //
44    // Constructors
45    //
46
47    private DynFixedImpl() {
48        this(null, (Any)null, false);
49    }
50
51    protected DynFixedImpl(ORB orb, Any any, boolean copyValue) {
52        super(orb, any, copyValue);
53    }
54
55    // Sets the current position to -1 and the value to zero.
56    protected DynFixedImpl(ORB orb, TypeCode typeCode) {
57        super(orb, typeCode);
58        index = NO_INDEX;
59    }
60
61    //
62    // DynAny interface methods
63    //
64/*
65    public int component_count() {
66        return 0;
67    }
68*/
69    //
70    // DynFixed interface methods
71    //
72
73    public String get_value () {
74        if (status == STATUS_DESTROYED) {
75            throw wrapper.dynAnyDestroyed() ;
76        }
77        return any.extract_fixed().toString();
78    }
79
80    // Initializes the value of the DynFixed.
81    // The val string must contain a fixed string constant in the same format
82    // as used for IDL fixed-point literals.
83    //
84    // It may consist of an integer part, an optional decimal point,
85    // a fraction part and an optional letter d or D.
86    // The integer and fraction parts both must be sequences of decimal (base 10) digits.
87    // Either the integer part or the fraction part, but not both, may be missing.
88    //
89    // If val contains a value whose scale exceeds that of the DynFixed or is not initialized,
90    // the operation raises InvalidValue.
91    // The return value is true if val can be represented as the DynFixed without loss of precision.
92    // If val has more fractional digits than can be represented in the DynFixed,
93    // fractional digits are truncated and the return value is false.
94    // If val does not contain a valid fixed-point literal or contains extraneous characters
95    // other than leading or trailing white space, the operation raises TypeMismatch.
96    //
97    public boolean set_value (String val)
98        throws org.omg.DynamicAny.DynAnyPackage.TypeMismatch,
99               org.omg.DynamicAny.DynAnyPackage.InvalidValue
100    {
101        if (status == STATUS_DESTROYED) {
102            throw wrapper.dynAnyDestroyed() ;
103        }
104        int digits = 0;
105        int scale = 0;
106        boolean preservedPrecision = true;
107        try {
108            digits = any.type().fixed_digits();
109            scale = any.type().fixed_scale();
110        } catch (BadKind ex) { // impossible
111        }
112        // First get rid of leading or trailing whitespace which is allowed
113        String string = val.trim();
114        if (string.length() == 0)
115            throw new TypeMismatch();
116        // Now scan for the sign
117        String sign = "";
118        if (string.charAt(0) == '-') {
119            sign = "-";
120            string = string.substring(1);
121        } else if (string.charAt(0) == '+') {
122            sign = "+";
123            string = string.substring(1);
124        }
125        // Now get rid of the letter d or D.
126        int dIndex = string.indexOf('d');
127        if (dIndex == -1) {
128            dIndex = string.indexOf('D');
129        }
130        if (dIndex != -1) {
131            string = string.substring(0, dIndex);
132        }
133        // Just to be sure
134        if (string.length() == 0)
135            throw new TypeMismatch();
136        // Now look for the dot to determine the integer part
137        String integerPart;
138        String fractionPart;
139        int currentScale;
140        int currentDigits;
141        int dotIndex = string.indexOf('.');
142        if (dotIndex == -1) {
143            integerPart = string;
144            fractionPart = null;
145            currentScale = 0;
146            currentDigits = integerPart.length();
147        } else if (dotIndex == 0 ) {
148            integerPart = null;
149            fractionPart = string;
150            currentScale = fractionPart.length();
151            currentDigits = currentScale;
152        } else {
153            integerPart = string.substring(0, dotIndex);
154            fractionPart = string.substring(dotIndex + 1);
155            currentScale = fractionPart.length();
156            currentDigits = integerPart.length() + currentScale;
157        }
158        // Let's see if we have to drop some precision
159        if (currentDigits > digits) {
160            preservedPrecision = false;
161            // truncate the fraction part
162            if (integerPart.length() < digits) {
163                fractionPart = fractionPart.substring(0, digits - integerPart.length());
164            } else if (integerPart.length() == digits) {
165                // currentScale > 0
166                // drop the fraction completely
167                fractionPart = null;
168            } else {
169                // integerPart.length() > digits
170                // unable to truncate fraction part
171                throw new InvalidValue();
172            }
173        }
174        // If val contains a value whose scale exceeds that of the DynFixed or is not initialized,
175        // the operation raises InvalidValue.
176        // Reinterpreted to mean raise InvalidValue only if the integer part exceeds precision,
177        // which is handled above (integerPart.length() > digits)
178/*
179        if (currentScale > scale) {
180            throw new InvalidValue("Scale exceeds " + scale);
181        }
182*/
183        // Now check whether both parts are valid numbers
184        BigDecimal result;
185        try {
186            new BigInteger(integerPart);
187            if (fractionPart == null) {
188                result = new BigDecimal(sign + integerPart);
189            } else {
190                new BigInteger(fractionPart);
191                result = new BigDecimal(sign + integerPart + "." + fractionPart);
192            }
193        } catch (NumberFormatException nfe) {
194            throw new TypeMismatch();
195        }
196        any.insert_fixed(result, any.type());
197        return preservedPrecision;
198    }
199
200    public String toString() {
201        int digits = 0;
202        int scale = 0;
203        try {
204            digits = any.type().fixed_digits();
205            scale = any.type().fixed_scale();
206        } catch (BadKind ex) { // impossible
207        }
208        return "DynFixed with value=" + this.get_value() + ", digits=" + digits + ", scale=" + scale;
209    }
210}
211