1 /*
  2  * Copyright (c) 2021, 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 
 26 package jdk.internal.reflect;
 27 
 28 import java.lang.invoke.MethodHandle;
 29 import java.lang.invoke.MethodType;
 30 import java.lang.reflect.Field;
 31 import java.lang.reflect.Modifier;
 32 
 33 class MethodHandleDoubleFieldAccessorImpl extends MethodHandleFieldAccessorImpl {
 34     static FieldAccessorImpl fieldAccessor(Field field, MethodHandle getter, MethodHandle setter, boolean isReadOnly) {
 35         boolean isStatic = Modifier.isStatic(field.getModifiers());
 36         if (isStatic) {
 37             getter = getter.asType(MethodType.methodType(double.class));
 38             if (setter != null) {
 39                 setter = setter.asType(MethodType.methodType(void.class, double.class));
 40             }
 41         } else {
 42             getter = getter.asType(MethodType.methodType(double.class, Object.class));
 43             if (setter != null) {
 44                 setter = setter.asType(MethodType.methodType(void.class, Object.class, double.class));
 45             }
 46         }
 47         return new MethodHandleDoubleFieldAccessorImpl(field, getter, setter, isReadOnly, isStatic);
 48     }
 49 
 50     MethodHandleDoubleFieldAccessorImpl(Field field, MethodHandle getter, MethodHandle setter, boolean isReadOnly, boolean isStatic) {
 51         super(field, getter, setter, isReadOnly, isStatic);
 52     }
 53 
 54     public Object get(Object obj) throws IllegalArgumentException {
 55         return Double.valueOf(getDouble(obj));
 56     }
 57 
 58     public boolean getBoolean(Object obj) throws IllegalArgumentException {
 59         throw newGetBooleanIllegalArgumentException();
 60     }
 61 
 62     public byte getByte(Object obj) throws IllegalArgumentException {
 63         throw newGetByteIllegalArgumentException();
 64     }
 65 
 66     public char getChar(Object obj) throws IllegalArgumentException {
 67         throw newGetCharIllegalArgumentException();
 68     }
 69 
 70     public short getShort(Object obj) throws IllegalArgumentException {
 71         throw newGetShortIllegalArgumentException();
 72     }
 73 
 74     public int getInt(Object obj) throws IllegalArgumentException {
 75         throw newGetIntIllegalArgumentException();
 76     }
 77 
 78     public long getLong(Object obj) throws IllegalArgumentException {
 79         throw newGetLongIllegalArgumentException();
 80     }
 81 
 82     public float getFloat(Object obj) throws IllegalArgumentException {
 83         throw newGetFloatIllegalArgumentException();
 84     }
 85 
 86     public double getDouble(Object obj) throws IllegalArgumentException {
 87         try {
 88             if (isStatic()) {
 89                 return (double) getter.invokeExact();
 90             } else {
 91                 return (double) getter.invokeExact(obj);
 92             }
 93         } catch (IllegalArgumentException|IllegalStateException|NullPointerException e) {
 94             throw e;
 95         } catch (ClassCastException e) {
 96             throw newGetIllegalArgumentException(obj);
 97         } catch (Throwable e) {
 98             throw new InternalError(e);
 99         }
100     }
101 
102     public void set(Object obj, Object value)
103             throws IllegalArgumentException, IllegalAccessException
104     {
105         ensureObj(obj);
106         if (isReadOnly()) {
107             throwFinalFieldIllegalAccessException(value);
108         }
109 
110         if (value == null) {
111             throwSetIllegalArgumentException(value);
112         }
113 
114         if (value instanceof Byte b) {
115             setDouble(obj, b.byteValue());
116         }
117         else if (value instanceof Short s) {
118             setDouble(obj, s.shortValue());
119         }
120         else if (value instanceof Character c) {
121             setDouble(obj, c.charValue());
122         }
123         else if (value instanceof Integer i) {
124             setDouble(obj, i.intValue());
125         }
126         else if (value instanceof Long l) {
127             setDouble(obj, l.longValue());
128         }
129         else if (value instanceof Float f) {
130             setDouble(obj, f.floatValue());
131         }
132         else if (value instanceof Double d) {
133             setDouble(obj, d.doubleValue());
134         }
135         else {
136             throwSetIllegalArgumentException(value);
137         }
138     }
139 
140     public void setBoolean(Object obj, boolean z)
141         throws IllegalArgumentException, IllegalAccessException
142     {
143         throwSetIllegalArgumentException(z);
144     }
145 
146     public void setByte(Object obj, byte b)
147         throws IllegalArgumentException, IllegalAccessException
148     {
149         setDouble(obj, b);
150     }
151 
152     public void setChar(Object obj, char c)
153         throws IllegalArgumentException, IllegalAccessException
154     {
155         setDouble(obj, c);
156     }
157 
158     public void setShort(Object obj, short s)
159         throws IllegalArgumentException, IllegalAccessException
160     {
161         setDouble(obj, s);
162     }
163 
164     public void setInt(Object obj, int i)
165         throws IllegalArgumentException, IllegalAccessException
166     {
167         setDouble(obj, i);
168     }
169 
170     public void setLong(Object obj, long l)
171         throws IllegalArgumentException, IllegalAccessException
172     {
173         setDouble(obj, l);
174     }
175 
176     public void setFloat(Object obj, float f)
177         throws IllegalArgumentException, IllegalAccessException
178     {
179         setDouble(obj, f);
180     }
181 
182     public void setDouble(Object obj, double d)
183         throws IllegalArgumentException, IllegalAccessException
184     {
185         if (isReadOnly()) {
186             ensureObj(obj);     // throw NPE if obj is null on instance field
187             throwFinalFieldIllegalAccessException(d);
188         }
189         try {
190             if (isStatic()) {
191                 setter.invokeExact(d);
192             } else {
193                 setter.invokeExact(obj, d);
194             }
195         } catch (IllegalArgumentException|IllegalStateException|NullPointerException e) {
196             throw e;
197         } catch (ClassCastException e) {
198             // receiver is of invalid type
199             throw newSetIllegalArgumentException(obj);
200         } catch (Throwable e) {
201             throw new InternalError(e);
202         }
203     }
204 }