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