Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
dragonwell8_jdk
提交
fdbaccc9
D
dragonwell8_jdk
项目概览
openanolis
/
dragonwell8_jdk
通知
4
Star
2
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
D
dragonwell8_jdk
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
fdbaccc9
编写于
7月 15, 2013
作者:
P
plevart
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
7122142: (ann) Race condition between isAnnotationPresent and getAnnotations
Reviewed-by: dholmes, jfranck
上级
6f3712d0
变更
7
显示空白变更内容
内联
并排
Showing
7 changed file
with
433 addition
and
57 deletion
+433
-57
src/share/classes/java/lang/Class.java
src/share/classes/java/lang/Class.java
+45
-28
src/share/classes/java/lang/System.java
src/share/classes/java/lang/System.java
+5
-2
src/share/classes/sun/misc/JavaLangAccess.java
src/share/classes/sun/misc/JavaLangAccess.java
+8
-2
src/share/classes/sun/reflect/annotation/AnnotationParser.java
...hare/classes/sun/reflect/annotation/AnnotationParser.java
+59
-7
src/share/classes/sun/reflect/annotation/AnnotationType.java
src/share/classes/sun/reflect/annotation/AnnotationType.java
+35
-18
test/java/lang/annotation/AnnotationType/AnnotationTypeDeadlockTest.java
...annotation/AnnotationType/AnnotationTypeDeadlockTest.java
+105
-0
test/java/lang/annotation/AnnotationType/AnnotationTypeRuntimeAssumptionTest.java
...n/AnnotationType/AnnotationTypeRuntimeAssumptionTest.java
+176
-0
未找到文件。
src/share/classes/java/lang/Class.java
浏览文件 @
fdbaccc9
...
...
@@ -2307,6 +2307,45 @@ public final class Class<T> implements java.io.Serializable,
return
name
;
}
/**
* Atomic operations support.
*/
private
static
class
Atomic
{
// initialize Unsafe machinery here, since we need to call Class.class instance method
// and have to avoid calling it in the static initializer of the Class class...
private
static
final
Unsafe
unsafe
=
Unsafe
.
getUnsafe
();
// offset of Class.reflectionData instance field
private
static
final
long
reflectionDataOffset
;
// offset of Class.annotationType instance field
private
static
final
long
annotationTypeOffset
;
static
{
Field
[]
fields
=
Class
.
class
.
getDeclaredFields0
(
false
);
// bypass caches
reflectionDataOffset
=
objectFieldOffset
(
fields
,
"reflectionData"
);
annotationTypeOffset
=
objectFieldOffset
(
fields
,
"annotationType"
);
}
private
static
long
objectFieldOffset
(
Field
[]
fields
,
String
fieldName
)
{
Field
field
=
searchFields
(
fields
,
fieldName
);
if
(
field
==
null
)
{
throw
new
Error
(
"No "
+
fieldName
+
" field found in java.lang.Class"
);
}
return
unsafe
.
objectFieldOffset
(
field
);
}
static
<
T
>
boolean
casReflectionData
(
Class
<?>
clazz
,
SoftReference
<
ReflectionData
<
T
>>
oldData
,
SoftReference
<
ReflectionData
<
T
>>
newData
)
{
return
unsafe
.
compareAndSwapObject
(
clazz
,
reflectionDataOffset
,
oldData
,
newData
);
}
static
<
T
>
boolean
casAnnotationType
(
Class
<?>
clazz
,
AnnotationType
oldType
,
AnnotationType
newType
)
{
return
unsafe
.
compareAndSwapObject
(
clazz
,
annotationTypeOffset
,
oldType
,
newType
);
}
}
/**
* Reflection support.
*/
...
...
@@ -2333,29 +2372,6 @@ public final class Class<T> implements java.io.Serializable,
ReflectionData
(
int
redefinedCount
)
{
this
.
redefinedCount
=
redefinedCount
;
}
// initialize Unsafe machinery here, since we need to call Class.class instance method
// and have to avoid calling it in the static initializer of the Class class...
private
static
final
Unsafe
unsafe
;
// offset of Class.reflectionData instance field
private
static
final
long
reflectionDataOffset
;
static
{
unsafe
=
Unsafe
.
getUnsafe
();
// bypass caches
Field
reflectionDataField
=
searchFields
(
Class
.
class
.
getDeclaredFields0
(
false
),
"reflectionData"
);
if
(
reflectionDataField
==
null
)
{
throw
new
Error
(
"No reflectionData field found in java.lang.Class"
);
}
reflectionDataOffset
=
unsafe
.
objectFieldOffset
(
reflectionDataField
);
}
static
<
T
>
boolean
compareAndSwap
(
Class
<?>
clazz
,
SoftReference
<
ReflectionData
<
T
>>
oldData
,
SoftReference
<
ReflectionData
<
T
>>
newData
)
{
return
unsafe
.
compareAndSwapObject
(
clazz
,
reflectionDataOffset
,
oldData
,
newData
);
}
}
private
volatile
transient
SoftReference
<
ReflectionData
<
T
>>
reflectionData
;
...
...
@@ -2387,7 +2403,7 @@ public final class Class<T> implements java.io.Serializable,
while
(
true
)
{
ReflectionData
<
T
>
rd
=
new
ReflectionData
<>(
classRedefinedCount
);
// try to CAS it...
if
(
ReflectionData
.
compareAndSwap
(
this
,
oldReflectionData
,
new
SoftReference
<>(
rd
)))
{
if
(
Atomic
.
casReflectionData
(
this
,
oldReflectionData
,
new
SoftReference
<>(
rd
)))
{
return
rd
;
}
// else retry
...
...
@@ -2430,7 +2446,7 @@ public final class Class<T> implements java.io.Serializable,
}
// Annotations handling
private
native
byte
[]
getRawAnnotations
();
native
byte
[]
getRawAnnotations
();
// Since 1.8
native
byte
[]
getRawTypeAnnotations
();
static
byte
[]
getExecutableTypeAnnotationBytes
(
Executable
ex
)
{
...
...
@@ -3290,10 +3306,11 @@ public final class Class<T> implements java.io.Serializable,
// Annotation types cache their internal (AnnotationType) form
private
AnnotationType
annotationType
;
@SuppressWarnings
(
"UnusedDeclaration"
)
private
volatile
transient
AnnotationType
annotationType
;
void
setAnnotationType
(
AnnotationType
t
ype
)
{
annotationType
=
type
;
boolean
casAnnotationType
(
AnnotationType
oldType
,
AnnotationType
newT
ype
)
{
return
Atomic
.
casAnnotationType
(
this
,
oldType
,
newType
)
;
}
AnnotationType
getAnnotationType
()
{
...
...
src/share/classes/java/lang/System.java
浏览文件 @
fdbaccc9
...
...
@@ -1220,12 +1220,15 @@ public final class System {
public
sun
.
reflect
.
ConstantPool
getConstantPool
(
Class
<?>
klass
)
{
return
klass
.
getConstantPool
();
}
public
void
setAnnotationType
(
Class
<?>
klass
,
AnnotationType
t
ype
)
{
klass
.
setAnnotationType
(
t
ype
);
public
boolean
casAnnotationType
(
Class
<?>
klass
,
AnnotationType
oldType
,
AnnotationType
newT
ype
)
{
return
klass
.
casAnnotationType
(
oldType
,
newT
ype
);
}
public
AnnotationType
getAnnotationType
(
Class
<?>
klass
)
{
return
klass
.
getAnnotationType
();
}
public
byte
[]
getRawClassAnnotations
(
Class
<?>
klass
)
{
return
klass
.
getRawAnnotations
();
}
public
byte
[]
getRawClassTypeAnnotations
(
Class
<?>
klass
)
{
return
klass
.
getRawTypeAnnotations
();
}
...
...
src/share/classes/sun/misc/JavaLangAccess.java
浏览文件 @
fdbaccc9
...
...
@@ -36,10 +36,10 @@ public interface JavaLangAccess {
ConstantPool
getConstantPool
(
Class
<?>
klass
);
/**
*
Set
the AnnotationType instance corresponding to this class.
*
Compare-And-Swap
the AnnotationType instance corresponding to this class.
* (This method only applies to annotation types.)
*/
void
setAnnotationType
(
Class
<?>
klass
,
AnnotationType
annotation
Type
);
boolean
casAnnotationType
(
Class
<?>
klass
,
AnnotationType
oldType
,
AnnotationType
new
Type
);
/**
* Get the AnnotationType instance corresponding to this class.
...
...
@@ -47,6 +47,12 @@ public interface JavaLangAccess {
*/
AnnotationType
getAnnotationType
(
Class
<?>
klass
);
/**
* Get the array of bytes that is the class-file representation
* of this Class' annotations.
*/
byte
[]
getRawClassAnnotations
(
Class
<?>
klass
);
/**
* Get the array of bytes that is the class-file representation
* of this Class' type annotations.
...
...
src/share/classes/sun/reflect/annotation/AnnotationParser.java
浏览文件 @
fdbaccc9
...
...
@@ -69,7 +69,35 @@ public class AnnotationParser {
return
Collections
.
emptyMap
();
try
{
return
parseAnnotations2
(
rawAnnotations
,
constPool
,
container
);
return
parseAnnotations2
(
rawAnnotations
,
constPool
,
container
,
null
);
}
catch
(
BufferUnderflowException
e
)
{
throw
new
AnnotationFormatError
(
"Unexpected end of annotations."
);
}
catch
(
IllegalArgumentException
e
)
{
// Type mismatch in constant pool
throw
new
AnnotationFormatError
(
e
);
}
}
/**
* Like {@link #parseAnnotations(byte[], sun.reflect.ConstantPool, Class)}
* with an additional parameter {@code selectAnnotationClasses} which selects the
* annotation types to parse (other than selected are quickly skipped).<p>
* This method is only used to parse select meta annotations in the construction
* phase of {@link AnnotationType} instances to prevent infinite recursion.
*
* @param selectAnnotationClasses an array of annotation types to select when parsing
*/
@SafeVarargs
static
Map
<
Class
<?
extends
Annotation
>,
Annotation
>
parseSelectAnnotations
(
byte
[]
rawAnnotations
,
ConstantPool
constPool
,
Class
<?>
container
,
Class
<?
extends
Annotation
>
...
selectAnnotationClasses
)
{
if
(
rawAnnotations
==
null
)
return
Collections
.
emptyMap
();
try
{
return
parseAnnotations2
(
rawAnnotations
,
constPool
,
container
,
selectAnnotationClasses
);
}
catch
(
BufferUnderflowException
e
)
{
throw
new
AnnotationFormatError
(
"Unexpected end of annotations."
);
}
catch
(
IllegalArgumentException
e
)
{
...
...
@@ -81,22 +109,23 @@ public class AnnotationParser {
private
static
Map
<
Class
<?
extends
Annotation
>,
Annotation
>
parseAnnotations2
(
byte
[]
rawAnnotations
,
ConstantPool
constPool
,
Class
<?>
container
)
{
Class
<?>
container
,
Class
<?
extends
Annotation
>[]
selectAnnotationClasses
)
{
Map
<
Class
<?
extends
Annotation
>,
Annotation
>
result
=
new
LinkedHashMap
<
Class
<?
extends
Annotation
>,
Annotation
>();
ByteBuffer
buf
=
ByteBuffer
.
wrap
(
rawAnnotations
);
int
numAnnotations
=
buf
.
getShort
()
&
0xFFFF
;
for
(
int
i
=
0
;
i
<
numAnnotations
;
i
++)
{
Annotation
a
=
parseAnnotation
(
buf
,
constPool
,
container
,
false
);
Annotation
a
=
parseAnnotation
2
(
buf
,
constPool
,
container
,
false
,
selectAnnotationClasses
);
if
(
a
!=
null
)
{
Class
<?
extends
Annotation
>
klass
=
a
.
annotationType
();
AnnotationType
type
=
AnnotationType
.
getInstance
(
klass
);
if
(
type
.
retention
()
==
RetentionPolicy
.
RUNTIME
)
if
(
result
.
put
(
klass
,
a
)
!=
null
)
if
(
AnnotationType
.
getInstance
(
klass
).
retention
()
==
RetentionPolicy
.
RUNTIME
&&
result
.
put
(
klass
,
a
)
!=
null
)
{
throw
new
AnnotationFormatError
(
"Duplicate annotation for class: "
+
klass
+
": "
+
a
);
}
}
}
return
result
;
}
...
...
@@ -189,11 +218,19 @@ public class AnnotationParser {
* TypeNotPresentException if a referenced annotation type is not
* available at runtime
*/
@SuppressWarnings
(
"unchecked"
)
static
Annotation
parseAnnotation
(
ByteBuffer
buf
,
ConstantPool
constPool
,
Class
<?>
container
,
boolean
exceptionOnMissingAnnotationClass
)
{
return
parseAnnotation2
(
buf
,
constPool
,
container
,
exceptionOnMissingAnnotationClass
,
null
);
}
@SuppressWarnings
(
"unchecked"
)
private
static
Annotation
parseAnnotation2
(
ByteBuffer
buf
,
ConstantPool
constPool
,
Class
<?>
container
,
boolean
exceptionOnMissingAnnotationClass
,
Class
<?
extends
Annotation
>[]
selectAnnotationClasses
)
{
int
typeIndex
=
buf
.
getShort
()
&
0xFFFF
;
Class
<?
extends
Annotation
>
annotationClass
=
null
;
String
sig
=
"[unknown]"
;
...
...
@@ -219,6 +256,10 @@ public class AnnotationParser {
skipAnnotation
(
buf
,
false
);
return
null
;
}
if
(
selectAnnotationClasses
!=
null
&&
!
contains
(
selectAnnotationClasses
,
annotationClass
))
{
skipAnnotation
(
buf
,
false
);
return
null
;
}
AnnotationType
type
=
null
;
try
{
type
=
AnnotationType
.
getInstance
(
annotationClass
);
...
...
@@ -800,6 +841,17 @@ public class AnnotationParser {
skipMemberValue
(
buf
);
}
/**
* Searches for given {@code element} in given {@code array} by identity.
* Returns {@code true} if found {@code false} if not.
*/
private
static
boolean
contains
(
Object
[]
array
,
Object
element
)
{
for
(
Object
e
:
array
)
if
(
e
==
element
)
return
true
;
return
false
;
}
/*
* This method converts the annotation map returned by the parseAnnotations()
* method to an array. It is called by Field.getDeclaredAnnotations(),
...
...
src/share/classes/sun/reflect/annotation/AnnotationType.java
浏览文件 @
fdbaccc9
...
...
@@ -25,6 +25,8 @@
package
sun.reflect.annotation
;
import
sun.misc.JavaLangAccess
;
import
java.lang.annotation.*
;
import
java.lang.reflect.*
;
import
java.util.*
;
...
...
@@ -61,12 +63,12 @@ public class AnnotationType {
/**
* The retention policy for this annotation type.
*/
private
RetentionPolicy
retention
=
RetentionPolicy
.
RUNTIME
;
;
private
final
RetentionPolicy
retention
;
/**
* Whether this annotation type is inherited.
*/
private
boolean
inherited
=
false
;
private
final
boolean
inherited
;
/**
* Returns an AnnotationType instance for the specified annotation type.
...
...
@@ -74,13 +76,20 @@ public class AnnotationType {
* @throw IllegalArgumentException if the specified class object for
* does not represent a valid annotation type
*/
public
static
synchronized
AnnotationType
getInstance
(
public
static
AnnotationType
getInstance
(
Class
<?
extends
Annotation
>
annotationClass
)
{
AnnotationType
result
=
sun
.
misc
.
SharedSecrets
.
getJavaLangAccess
().
getAnnotationType
(
annotationClass
);
if
(
result
==
null
)
result
=
new
AnnotationType
((
Class
<?
extends
Annotation
>)
annotationClass
);
JavaLangAccess
jla
=
sun
.
misc
.
SharedSecrets
.
getJavaLangAccess
();
AnnotationType
result
=
jla
.
getAnnotationType
(
annotationClass
);
// volatile read
if
(
result
==
null
)
{
result
=
new
AnnotationType
(
annotationClass
);
// try to CAS the AnnotationType: null -> result
if
(!
jla
.
casAnnotationType
(
annotationClass
,
null
,
result
))
{
// somebody was quicker -> read it's result
result
=
jla
.
getAnnotationType
(
annotationClass
);
assert
result
!=
null
;
}
}
return
result
;
}
...
...
@@ -121,16 +130,25 @@ public class AnnotationType {
memberDefaults
.
put
(
name
,
defaultValue
);
}
sun
.
misc
.
SharedSecrets
.
getJavaLangAccess
().
setAnnotationType
(
annotationClass
,
this
);
// Initialize retention, & inherited fields. Special treatment
// of the corresponding annotation types breaks infinite recursion.
if
(
annotationClass
!=
Retention
.
class
&&
annotationClass
!=
Inherited
.
class
)
{
Retention
ret
=
annotationClass
.
getAnnotation
(
Retention
.
class
);
JavaLangAccess
jla
=
sun
.
misc
.
SharedSecrets
.
getJavaLangAccess
();
Map
<
Class
<?
extends
Annotation
>,
Annotation
>
metaAnnotations
=
AnnotationParser
.
parseSelectAnnotations
(
jla
.
getRawClassAnnotations
(
annotationClass
),
jla
.
getConstantPool
(
annotationClass
),
annotationClass
,
Retention
.
class
,
Inherited
.
class
);
Retention
ret
=
(
Retention
)
metaAnnotations
.
get
(
Retention
.
class
);
retention
=
(
ret
==
null
?
RetentionPolicy
.
CLASS
:
ret
.
value
());
inherited
=
annotationClass
.
isAnnotationPresent
(
Inherited
.
class
);
inherited
=
metaAnnotations
.
containsKey
(
Inherited
.
class
);
}
else
{
retention
=
RetentionPolicy
.
RUNTIME
;
inherited
=
false
;
}
}
...
...
@@ -205,11 +223,10 @@ public class AnnotationType {
* For debugging.
*/
public
String
toString
()
{
StringBuffer
s
=
new
StringBuffer
(
"Annotation Type:"
+
"\n"
);
s
.
append
(
" Member types: "
+
memberTypes
+
"\n"
);
s
.
append
(
" Member defaults: "
+
memberDefaults
+
"\n"
);
s
.
append
(
" Retention policy: "
+
retention
+
"\n"
);
s
.
append
(
" Inherited: "
+
inherited
);
return
s
.
toString
();
return
"Annotation Type:\n"
+
" Member types: "
+
memberTypes
+
"\n"
+
" Member defaults: "
+
memberDefaults
+
"\n"
+
" Retention policy: "
+
retention
+
"\n"
+
" Inherited: "
+
inherited
;
}
}
test/java/lang/annotation/AnnotationType/AnnotationTypeDeadlockTest.java
0 → 100644
浏览文件 @
fdbaccc9
/*
* Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* @test
* @bug 7122142
* @summary Test deadlock situation when recursive annotations are parsed
*/
import
java.lang.annotation.Retention
;
import
java.util.concurrent.CountDownLatch
;
import
java.util.concurrent.atomic.AtomicInteger
;
import
static
java
.
lang
.
annotation
.
RetentionPolicy
.
RUNTIME
;
public
class
AnnotationTypeDeadlockTest
{
@Retention
(
RUNTIME
)
@AnnB
public
@interface
AnnA
{
}
@Retention
(
RUNTIME
)
@AnnA
public
@interface
AnnB
{
}
static
class
Task
extends
Thread
{
final
CountDownLatch
prepareLatch
;
final
AtomicInteger
goLatch
;
final
Class
<?>
clazz
;
Task
(
CountDownLatch
prepareLatch
,
AtomicInteger
goLatch
,
Class
<?>
clazz
)
{
super
(
clazz
.
getSimpleName
());
setDaemon
(
true
);
// in case it deadlocks
this
.
prepareLatch
=
prepareLatch
;
this
.
goLatch
=
goLatch
;
this
.
clazz
=
clazz
;
}
@Override
public
void
run
()
{
prepareLatch
.
countDown
();
// notify we are prepared
while
(
goLatch
.
get
()
>
0
);
// spin-wait before go
clazz
.
getDeclaredAnnotations
();
}
}
static
void
dumpState
(
Task
task
)
{
System
.
err
.
println
(
"Task["
+
task
.
getName
()
+
"].state: "
+
task
.
getState
()
+
" ..."
);
for
(
StackTraceElement
ste
:
task
.
getStackTrace
())
{
System
.
err
.
println
(
"\tat "
+
ste
);
}
System
.
err
.
println
();
}
public
static
void
main
(
String
[]
args
)
throws
Exception
{
CountDownLatch
prepareLatch
=
new
CountDownLatch
(
2
);
AtomicInteger
goLatch
=
new
AtomicInteger
(
1
);
Task
taskA
=
new
Task
(
prepareLatch
,
goLatch
,
AnnA
.
class
);
Task
taskB
=
new
Task
(
prepareLatch
,
goLatch
,
AnnB
.
class
);
taskA
.
start
();
taskB
.
start
();
// wait until both threads start-up
prepareLatch
.
await
();
// let them go
goLatch
.
set
(
0
);
// attempt to join them
taskA
.
join
(
5000L
);
taskB
.
join
(
5000L
);
if
(
taskA
.
isAlive
()
||
taskB
.
isAlive
())
{
dumpState
(
taskA
);
dumpState
(
taskB
);
throw
new
IllegalStateException
(
taskA
.
getState
()
==
Thread
.
State
.
BLOCKED
&&
taskB
.
getState
()
==
Thread
.
State
.
BLOCKED
?
"deadlock detected"
:
"unexpected condition"
);
}
}
}
test/java/lang/annotation/AnnotationType/AnnotationTypeRuntimeAssumptionTest.java
0 → 100644
浏览文件 @
fdbaccc9
/*
* Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* @test
* @summary Test consistent parsing of ex-RUNTIME annotations that
* were changed and separately compiled to have CLASS retention
*/
import
sun.misc.IOUtils
;
import
java.io.IOException
;
import
java.io.InputStream
;
import
java.lang.annotation.Retention
;
import
java.lang.annotation.RetentionPolicy
;
import
static
java
.
lang
.
annotation
.
RetentionPolicy
.
CLASS
;
import
static
java
.
lang
.
annotation
.
RetentionPolicy
.
RUNTIME
;
/**
* This test simulates a situation where there are two mutually recursive
* {@link RetentionPolicy#RUNTIME RUNTIME} annotations {@link AnnA_v1 AnnA_v1}
* and {@link AnnB AnnB} and then the first is changed to have
* {@link RetentionPolicy#CLASS CLASS} retention and separately compiled.
* When {@link AnnA_v1 AnnA_v1} annotation is looked-up on {@link AnnB AnnB}
* it still appears to have {@link RetentionPolicy#RUNTIME RUNTIME} retention.
*/
public
class
AnnotationTypeRuntimeAssumptionTest
{
@Retention
(
RUNTIME
)
@AnnB
public
@interface
AnnA_v1
{
}
// An alternative version of AnnA_v1 with CLASS retention instead.
// Used to simulate separate compilation (see AltClassLoader below).
@Retention
(
CLASS
)
@AnnB
public
@interface
AnnA_v2
{
}
@Retention
(
RUNTIME
)
@AnnA_v1
public
@interface
AnnB
{
}
@AnnA_v1
public
static
class
TestTask
implements
Runnable
{
@Override
public
void
run
()
{
AnnA_v1
ann1
=
TestTask
.
class
.
getDeclaredAnnotation
(
AnnA_v1
.
class
);
if
(
ann1
!=
null
)
{
throw
new
IllegalStateException
(
"@"
+
ann1
.
annotationType
().
getSimpleName
()
+
" found on: "
+
TestTask
.
class
.
getName
()
+
" should not be visible at runtime"
);
}
AnnA_v1
ann2
=
AnnB
.
class
.
getDeclaredAnnotation
(
AnnA_v1
.
class
);
if
(
ann2
!=
null
)
{
throw
new
IllegalStateException
(
"@"
+
ann2
.
annotationType
().
getSimpleName
()
+
" found on: "
+
AnnB
.
class
.
getName
()
+
" should not be visible at runtime"
);
}
}
}
public
static
void
main
(
String
[]
args
)
throws
Exception
{
ClassLoader
altLoader
=
new
AltClassLoader
(
AnnotationTypeRuntimeAssumptionTest
.
class
.
getClassLoader
());
Runnable
altTask
=
(
Runnable
)
Class
.
forName
(
TestTask
.
class
.
getName
(),
true
,
altLoader
).
newInstance
();
altTask
.
run
();
}
/**
* A ClassLoader implementation that loads alternative implementations of
* classes. If class name ends with "_v1" it locates instead a class with
* name ending with "_v2" and loads that class instead.
*/
static
class
AltClassLoader
extends
ClassLoader
{
AltClassLoader
(
ClassLoader
parent
)
{
super
(
parent
);
}
@Override
protected
Class
<?>
loadClass
(
String
name
,
boolean
resolve
)
throws
ClassNotFoundException
{
if
(
name
.
indexOf
(
'.'
)
<
0
)
{
// root package is our class
synchronized
(
getClassLoadingLock
(
name
))
{
// First, check if the class has already been loaded
Class
<?>
c
=
findLoadedClass
(
name
);
if
(
c
==
null
)
{
c
=
findClass
(
name
);
}
if
(
resolve
)
{
resolveClass
(
c
);
}
return
c
;
}
}
else
{
// not our class
return
super
.
loadClass
(
name
,
resolve
);
}
}
@Override
protected
Class
<?>
findClass
(
String
name
)
throws
ClassNotFoundException
{
// special class name -> replace it with alternative name
if
(
name
.
endsWith
(
"_v1"
))
{
String
altName
=
name
.
substring
(
0
,
name
.
length
()
-
3
)
+
"_v2"
;
String
altPath
=
altName
.
replace
(
'.'
,
'/'
).
concat
(
".class"
);
try
(
InputStream
is
=
getResourceAsStream
(
altPath
))
{
if
(
is
!=
null
)
{
byte
[]
bytes
=
IOUtils
.
readFully
(
is
,
-
1
,
true
);
// patch class bytes to contain original name
for
(
int
i
=
0
;
i
<
bytes
.
length
-
2
;
i
++)
{
if
(
bytes
[
i
]
==
'_'
&&
bytes
[
i
+
1
]
==
'v'
&&
bytes
[
i
+
2
]
==
'2'
)
{
bytes
[
i
+
2
]
=
'1'
;
}
}
return
defineClass
(
name
,
bytes
,
0
,
bytes
.
length
);
}
else
{
throw
new
ClassNotFoundException
(
name
);
}
}
catch
(
IOException
e
)
{
throw
new
ClassNotFoundException
(
name
,
e
);
}
}
else
{
// not special class name -> just load the class
String
path
=
name
.
replace
(
'.'
,
'/'
).
concat
(
".class"
);
try
(
InputStream
is
=
getResourceAsStream
(
path
))
{
if
(
is
!=
null
)
{
byte
[]
bytes
=
IOUtils
.
readFully
(
is
,
-
1
,
true
);
return
defineClass
(
name
,
bytes
,
0
,
bytes
.
length
);
}
else
{
throw
new
ClassNotFoundException
(
name
);
}
}
catch
(
IOException
e
)
{
throw
new
ClassNotFoundException
(
name
,
e
);
}
}
}
}
}
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录