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 MethodHandleLongFieldAccessorImpl 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(long.class));
38 if (setter != null) {
39 setter = setter.asType(MethodType.methodType(void.class, long.class));
40 }
41 } else {
42 getter = getter.asType(MethodType.methodType(long.class, Object.class));
43 if (setter != null) {
44 setter = setter.asType(MethodType.methodType(void.class, Object.class, long.class));
45 }
46 }
47 return new MethodHandleLongFieldAccessorImpl(field, getter, setter, isReadOnly, isStatic);
48 }
49
50 MethodHandleLongFieldAccessorImpl(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 Long.valueOf(getLong(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 try {
80 if (isStatic()) {
81 return (long) getter.invokeExact();
82 } else {
83 return (long) getter.invokeExact(obj);
84 }
85 } catch (IllegalArgumentException|IllegalStateException|NullPointerException e) {
86 throw e;
87 } catch (ClassCastException e) {
88 throw newGetIllegalArgumentException(obj);
89 } catch (Throwable e) {
90 throw new InternalError(e);
91 }
92 }
93
94 public float getFloat(Object obj) throws IllegalArgumentException {
95 return getLong(obj);
96 }
97
98 public double getDouble(Object obj) throws IllegalArgumentException {
99 return getLong(obj);
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 setLong(obj, b.byteValue());
116 }
117 else if (value instanceof Short s) {
118 setLong(obj, s.shortValue());
119 }
120 else if (value instanceof Character c) {
121 setLong(obj, c.charValue());
122 }
123 else if (value instanceof Integer i) {
124 setLong(obj, i.intValue());
125 }
126 else if (value instanceof Long l) {
127 setLong(obj, l.longValue());
128 }
129 else {
130 throwSetIllegalArgumentException(value);
131 }
132 }
133
134 public void setBoolean(Object obj, boolean z)
135 throws IllegalArgumentException, IllegalAccessException
136 {
137 throwSetIllegalArgumentException(z);
138 }
139
140 public void setByte(Object obj, byte b)
141 throws IllegalArgumentException, IllegalAccessException
142 {
143 setLong(obj, b);
144 }
145
146 public void setChar(Object obj, char c)
147 throws IllegalArgumentException, IllegalAccessException
148 {
149 setLong(obj, c);
150 }
151
152 public void setShort(Object obj, short s)
153 throws IllegalArgumentException, IllegalAccessException
154 {
155 setLong(obj, s);
156 }
157
158 public void setInt(Object obj, int i)
159 throws IllegalArgumentException, IllegalAccessException
160 {
161 setLong(obj, i);
162 }
163
164 public void setLong(Object obj, long l)
165 throws IllegalArgumentException, IllegalAccessException
166 {
167 if (isReadOnly()) {
168 ensureObj(obj); // throw NPE if obj is null on instance field
169 throwFinalFieldIllegalAccessException(l);
170 }
171 try {
172 if (isStatic()) {
173 setter.invokeExact(l);
174 } else {
175 setter.invokeExact(obj, l);
176 }
177 } catch (IllegalArgumentException|IllegalStateException|NullPointerException e) {
178 throw e;
179 } catch (ClassCastException e) {
180 // receiver is of invalid type
181 throw newSetIllegalArgumentException(obj);
182 } catch (Throwable e) {
183 throw new InternalError(e);
184 }
185 }
186
187 public void setFloat(Object obj, float f)
188 throws IllegalArgumentException, IllegalAccessException
189 {
190 throwSetIllegalArgumentException(f);
191 }
192
193 public void setDouble(Object obj, double d)
194 throws IllegalArgumentException, IllegalAccessException
195 {
196 throwSetIllegalArgumentException(d);
197 }
198 }