1 // natField.cc - Implementation of java.lang.reflect.Field native methods.
3 /* Copyright (C) 1998, 1999, 2000, 2001, 2003 Free Software Foundation
5 This file is part of libgcj.
7 This software is copyrighted work licensed under the terms of the
8 Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
16 #include <java/lang/reflect/Field.h>
17 #include <java/lang/reflect/Modifier.h>
18 #include <java/lang/ArrayIndexOutOfBoundsException.h>
19 #include <java/lang/IllegalArgumentException.h>
20 #include <java/lang/IllegalAccessException.h>
21 #include <java/lang/NullPointerException.h>
22 #include <java/lang/Byte.h>
23 #include <java/lang/Short.h>
24 #include <java/lang/Integer.h>
25 #include <java/lang/Long.h>
26 #include <java/lang/Float.h>
27 #include <java/lang/Double.h>
28 #include <java/lang/Boolean.h>
29 #include <java/lang/Character.h>
32 java::lang::reflect::Field::getModifiers ()
34 return _Jv_FromReflectedField (this)->getModifiers ();
38 java::lang::reflect::Field::getName ()
41 name
= _Jv_NewStringUtf8Const (_Jv_FromReflectedField (this)->name
);
46 java::lang::reflect::Field::getType ()
48 jfieldID fld
= _Jv_FromReflectedField (this);
49 JvSynchronize
sync (declaringClass
);
50 _Jv_ResolveField (fld
, declaringClass
->getClassLoaderInternal ());
55 getAddr (java::lang::reflect::Field
* field
, jclass caller
, jobject obj
)
57 // FIXME: we know CALLER is NULL here. At one point we planned to
58 // have the compiler insert the caller as a hidden argument in some
59 // calls. However, we never implemented that, so we have to find
60 // the caller by hand instead.
62 using namespace java::lang::reflect
;
64 jfieldID fld
= _Jv_FromReflectedField (field
);
65 _Jv_ushort flags
= fld
->getModifiers();
67 // Check accessibility, if required.
68 if (! (Modifier::isPublic (flags
) || field
->isAccessible()))
70 gnu::gcj::runtime::StackTrace
*t
71 = new gnu::gcj::runtime::StackTrace(7);
74 // We want to skip all the frames on the stack from this class.
75 for (int i
= 1; !caller
|| caller
== &Field::class$
; i
++)
76 caller
= t
->classAt (i
);
78 catch (::java::lang::ArrayIndexOutOfBoundsException
*e
)
82 if (! _Jv_CheckAccess (caller
, field
->getDeclaringClass(), flags
))
83 throw new java::lang::IllegalAccessException
;
86 if (flags
& Modifier::STATIC
)
88 jclass fldClass
= field
->getDeclaringClass ();
89 JvInitClass(fldClass
);
95 throw new java::lang::NullPointerException
;
96 if (! _Jv_IsInstanceOf (obj
, field
->getDeclaringClass()))
97 throw new java::lang::IllegalArgumentException
;
98 return (void*) ((char*) obj
+ fld
->getOffset ());
103 getBoolean (jclass cls
, void* addr
)
105 if (cls
== JvPrimClass (boolean
))
106 return * (jboolean
*) addr
;
107 throw new java::lang::IllegalArgumentException
;
111 getChar (jclass cls
, void* addr
)
113 if (cls
== JvPrimClass (char))
114 return * (jchar
*) addr
;
115 throw new java::lang::IllegalArgumentException
;
119 getByte (jclass cls
, void* addr
)
121 if (cls
== JvPrimClass (byte
))
122 return * (jbyte
*) addr
;
123 throw new java::lang::IllegalArgumentException
;
127 getShort (jclass cls
, void* addr
)
129 if (cls
== JvPrimClass (short))
130 return * (jshort
*) addr
;
131 if (cls
== JvPrimClass (byte
))
132 return * (jbyte
*) addr
;
133 throw new java::lang::IllegalArgumentException
;
137 getInt (jclass cls
, void* addr
)
139 if (cls
== JvPrimClass (int))
140 return * (jint
*) addr
;
141 if (cls
== JvPrimClass (short))
142 return * (jshort
*) addr
;
143 if (cls
== JvPrimClass (char))
144 return * (jchar
*) addr
;
145 if (cls
== JvPrimClass (byte
))
146 return * (jbyte
*) addr
;
147 throw new java::lang::IllegalArgumentException
;
151 getLong (jclass cls
, void* addr
)
153 if (cls
== JvPrimClass (long))
154 return * (jlong
*) addr
;
155 return ::getInt(cls
, addr
);
159 getFloat (jclass cls
, void* addr
)
161 if (cls
== JvPrimClass (float))
162 return * (jfloat
*) addr
;
163 if (cls
== JvPrimClass (long))
164 return * (jlong
*) addr
;
165 return ::getInt(cls
, addr
);
169 getDouble (jclass cls
, void* addr
)
171 if (cls
== JvPrimClass (double))
172 return * (jdouble
*) addr
;
173 if (cls
== JvPrimClass (float))
174 return * (jfloat
*) addr
;
175 if (cls
== JvPrimClass (long))
176 return * (jlong
*) addr
;
177 return ::getInt(cls
, addr
);
181 java::lang::reflect::Field::getBoolean (jclass caller
, jobject obj
)
183 return ::getBoolean (this->getType(), getAddr (this, caller
, obj
));
187 java::lang::reflect::Field::getChar (jclass caller
, jobject obj
)
189 return ::getChar (this->getType(), getAddr (this, caller
, obj
));
193 java::lang::reflect::Field::getByte (jclass caller
, jobject obj
)
195 return ::getByte (this->getType(), getAddr (this, caller
, obj
));
199 java::lang::reflect::Field::getShort (jclass caller
, jobject obj
)
201 return ::getShort (this->getType(), getAddr (this, caller
, obj
));
205 java::lang::reflect::Field::getInt (jclass caller
, jobject obj
)
207 return ::getInt (this->getType(), getAddr (this, caller
, obj
));
211 java::lang::reflect::Field::getLong (jclass caller
, jobject obj
)
213 return ::getLong (this->getType(), getAddr (this, caller
, obj
));
217 java::lang::reflect::Field::getFloat (jclass caller
, jobject obj
)
219 return ::getFloat (this->getType(), getAddr (this, caller
, obj
));
223 java::lang::reflect::Field::getDouble (jclass caller
, jobject obj
)
225 return ::getDouble (this->getType(), getAddr (this, caller
, obj
));
229 java::lang::reflect::Field::get (jclass caller
, jobject obj
)
231 jclass type
= this->getType();
232 void* addr
= getAddr (this, caller
, obj
);
233 if (! type
->isPrimitive ())
234 return * (jobject
*) addr
;
235 if (type
== JvPrimClass (double))
236 return new java::lang::Double (* (jdouble
*) addr
);
237 if (type
== JvPrimClass (float))
238 return new java::lang::Float (* (jfloat
*) addr
);
239 if (type
== JvPrimClass (long))
240 return new java::lang::Long (* (jlong
*) addr
);
241 if (type
== JvPrimClass (int))
242 return new java::lang::Integer (* (jint
*) addr
);
243 if (type
== JvPrimClass (short))
244 return new java::lang::Short (* (jshort
*) addr
);
245 if (type
== JvPrimClass (byte
))
246 return new java::lang::Byte (* (jbyte
*) addr
);
247 if (type
== JvPrimClass (char))
248 return new java::lang::Character (* (jchar
*) addr
);
249 if (type
== JvPrimClass (boolean
))
251 _Jv_InitClass (&java::lang::Boolean::class$
);
252 if (* (jboolean
*) addr
)
253 return java::lang::Boolean::TRUE
;
255 return java::lang::Boolean::FALSE
;
257 throw new java::lang::IllegalArgumentException
;
261 setAddr (java::lang::reflect::Field
* field
, jclass caller
, jobject obj
)
263 void *addr
= getAddr(field
, caller
, obj
);
264 if (!field
->isAccessible()
265 && field
->getModifiers() & java::lang::reflect::Modifier::FINAL
)
266 throw new java::lang::IllegalAccessException();
271 setBoolean (jclass type
, void *addr
, jboolean value
)
273 if (type
== JvPrimClass (boolean
))
274 * (jboolean
*) addr
= value
;
276 throw new java::lang::IllegalArgumentException
;
280 setChar (jclass type
, void *addr
, jchar value
)
282 if (type
== JvPrimClass (char))
283 * (jchar
*) addr
= value
;
284 else if (type
== JvPrimClass (int))
285 * (jint
*) addr
= value
;
286 else if (type
== JvPrimClass (long))
287 * (jlong
*) addr
= value
;
288 else if (type
== JvPrimClass (float))
289 * (jfloat
*) addr
= value
;
290 else if (type
== JvPrimClass (double))
291 * (jdouble
*) addr
= value
;
293 throw new java::lang::IllegalArgumentException
;
297 setByte (jclass type
, void *addr
, jbyte value
)
299 if (type
== JvPrimClass (byte
))
300 * (jbyte
*) addr
= value
;
301 else if (type
== JvPrimClass (short))
302 * (jshort
*) addr
= value
;
303 else if (type
== JvPrimClass (int))
304 * (jint
*) addr
= value
;
305 else if (type
== JvPrimClass (long))
306 * (jlong
*) addr
= value
;
307 else if (type
== JvPrimClass (float))
308 * (jfloat
*) addr
= value
;
309 else if (type
== JvPrimClass (double))
310 * (jdouble
*) addr
= value
;
312 throw new java::lang::IllegalArgumentException
;
316 setShort (jclass type
, void *addr
, jshort value
)
318 if (type
== JvPrimClass (short))
319 * (jshort
*) addr
= value
;
320 else if (type
== JvPrimClass (int))
321 * (jint
*) addr
= value
;
322 else if (type
== JvPrimClass (long))
323 * (jlong
*) addr
= value
;
324 else if (type
== JvPrimClass (float))
325 * (jfloat
*) addr
= value
;
326 else if (type
== JvPrimClass (double))
327 * (jdouble
*) addr
= value
;
329 throw new java::lang::IllegalArgumentException
;
333 setInt (jclass type
, void *addr
, jint value
)
335 if (type
== JvPrimClass (int))
336 * (jint
*) addr
= value
;
337 else if (type
== JvPrimClass (long))
338 * (jlong
*) addr
= value
;
339 else if (type
== JvPrimClass (float))
340 * (jfloat
*) addr
= value
;
341 else if (type
== JvPrimClass (double))
342 * (jdouble
*) addr
= value
;
344 throw new java::lang::IllegalArgumentException
;
348 setLong (jclass type
, void *addr
, jlong value
)
350 if (type
== JvPrimClass (long))
351 * (jlong
*) addr
= value
;
352 else if (type
== JvPrimClass (float))
353 * (jfloat
*) addr
= value
;
354 else if (type
== JvPrimClass (double))
355 * (jdouble
*) addr
= value
;
357 throw new java::lang::IllegalArgumentException
;
361 setFloat (jclass type
, void *addr
, jfloat value
)
363 if (type
== JvPrimClass (float))
364 * (jfloat
*) addr
= value
;
365 else if (type
== JvPrimClass (double))
366 * (jdouble
*) addr
= value
;
368 throw new java::lang::IllegalArgumentException
;
372 setDouble (jclass type
, void *addr
, jdouble value
)
374 if (type
== JvPrimClass (double))
375 * (jdouble
*) addr
= value
;
377 throw new java::lang::IllegalArgumentException
;
381 java::lang::reflect::Field::setBoolean (jclass caller
, jobject obj
, jboolean b
)
383 ::setBoolean (this->getType(), setAddr (this, caller
, obj
), b
);
387 java::lang::reflect::Field::setChar (jclass caller
, jobject obj
, jchar c
)
389 ::setChar (this->getType(), setAddr (this, caller
, obj
), c
);
393 java::lang::reflect::Field::setByte (jclass caller
, jobject obj
, jbyte b
)
395 ::setByte (this->getType(), setAddr (this, caller
, obj
), b
);
399 java::lang::reflect::Field::setShort (jclass caller
, jobject obj
, jshort s
)
401 ::setShort (this->getType(), setAddr (this, caller
, obj
), s
);
405 java::lang::reflect::Field::setInt (jclass caller
, jobject obj
, jint i
)
407 ::setInt (this->getType(), setAddr (this, caller
, obj
), i
);
411 java::lang::reflect::Field::setLong (jclass caller
, jobject obj
, jlong l
)
413 ::setLong (this->getType(), setAddr (this, caller
, obj
), l
);
416 java::lang::reflect::Field::setFloat (jclass caller
, jobject obj
, jfloat f
)
418 ::setFloat (this->getType(), setAddr (this, caller
, obj
), f
);
422 java::lang::reflect::Field::setDouble (jclass caller
, jobject obj
, jdouble d
)
424 ::setDouble (this->getType(), setAddr (this, caller
, obj
), d
);
428 java::lang::reflect::Field::set (jclass caller
, jobject object
, jobject value
,
431 void* addr
= setAddr (this, caller
, object
);
432 if (value
!= NULL
&& ! _Jv_IsInstanceOf (value
, type
))
433 throw new java::lang::IllegalArgumentException
;
434 * (jobject
*) addr
= value
;