Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
dragonwell8_jdk
提交
56f42edd
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看板
提交
56f42edd
编写于
1月 22, 2020
作者:
R
rkennke
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
8227269: Slow class loading when running with JDWP
Reviewed-by: sspitsyn, cjplummer
上级
9c0990de
变更
5
显示空白变更内容
内联
并排
Showing
5 changed file
with
147 addition
and
205 deletion
+147
-205
src/share/back/classTrack.c
src/share/back/classTrack.c
+135
-204
src/share/back/classTrack.h
src/share/back/classTrack.h
+6
-0
src/share/back/eventHandler.c
src/share/back/eventHandler.c
+3
-0
src/share/back/util.c
src/share/back/util.c
+1
-1
src/share/back/util.h
src/share/back/util.h
+2
-0
未找到文件。
src/share/back/classTrack.c
浏览文件 @
56f42edd
...
...
@@ -22,236 +22,149 @@
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/*
* This module tracks classes that have been prepared, so as to
* be able to compute which have been unloaded. On VM start-up
* all prepared classes are put in a table. As class prepare
* events come in they are added to the table. After an unload
* event or series of them, the VM can be asked for the list
* of classes; this list is compared against the table keep by
* this module, any classes no longer present are known to
* have been unloaded.
*
* For efficient access, classes are keep in a hash table.
* Each slot in the hash table has a linked list of KlassNode.
*
* Comparing current set of classes is compared with previous
* set by transferring all classes in the current set into
* a new table, any that remain in the old table have been
* unloaded.
* be able to report which have been unloaded. On VM start-up
* and whenever new classes are loaded, all prepared classes'
* signatures are attached as JVMTI tag to the class object.
* Class unloading is tracked by registering
* ObjectFree callback on class objects. When this happens, we find
* the signature of the unloaded class(es) and report them back
* to the event handler to synthesize class-unload-events.
*/
#include "util.h"
#include "bag.h"
#include "classTrack.h"
/* ClassTrack hash table slot count */
#define CT_HASH_SLOT_COUNT 263
/* Prime which eauals 4k+3 for some k */
typedef
struct
KlassNode
{
jclass
klass
;
/* weak global reference */
char
*
signature
;
/* class signature */
struct
KlassNode
*
next
;
/* next node in this slot */
}
KlassNode
;
#define NOT_TAGGED 0
/*
* Hash table of prepared classes. Each entry is a pointer
* to a linked list of KlassNode.
* The JVMTI tracking env to keep track of klass tags for class-unloads
*/
static
KlassNode
**
table
;
static
jvmtiEnv
*
trackingEnv
;
/*
* Return slot in hash table to use for this class.
* A bag containing all the deleted classes' signatures. Must be accessed under
* classTrackLock.
*/
static
jint
hashKlass
(
jclass
klass
)
{
jint
hashCode
=
objectHashCode
(
klass
);
return
abs
(
hashCode
)
%
CT_HASH_SLOT_COUNT
;
}
struct
bag
*
deletedSignatures
;
/*
* Transfer a node (which represents klass) from the current
* table to the new table.
*/
static
void
transferClass
(
JNIEnv
*
env
,
jclass
klass
,
KlassNode
**
newTable
)
{
jint
slot
=
hashKlass
(
klass
);
KlassNode
**
head
=
&
table
[
slot
];
KlassNode
**
newHead
=
&
newTable
[
slot
];
KlassNode
**
nodePtr
;
KlassNode
*
node
;
/* Search the node list of the current table for klass */
for
(
nodePtr
=
head
;
node
=
*
nodePtr
,
node
!=
NULL
;
nodePtr
=
&
(
node
->
next
))
{
if
(
isSameObject
(
env
,
klass
,
node
->
klass
))
{
/* Match found transfer node */
/* unlink from old list */
*
nodePtr
=
node
->
next
;
/* insert in new list */
node
->
next
=
*
newHead
;
*
newHead
=
node
;
return
;
}
}
/* we haven't found the class, only unloads should have happenned,
* so the only reason a class should not have been found is
* that it is not prepared yet, in which case we don't want it.
* Asset that the above is true.
* Lock to keep integrity of deletedSignatures.
*/
/**** the HotSpot VM doesn't create prepare events for some internal classes ***
JDI_ASSERT_MSG((classStatus(klass) &
(JVMTI_CLASS_STATUS_PREPARED|JVMTI_CLASS_STATUS_ARRAY))==0,
classSignature(klass));
***/
}
static
jrawMonitorID
classTrackLock
;
/*
* Delete a hash table of classes.
* The signatures of classes in the table are returned.
* Invoke the callback when classes are freed, find and record the signature
* in deletedSignatures. Those are only used in addPreparedClass() by the
* same thread.
*/
static
struct
bag
*
deleteTable
(
JNIEnv
*
env
,
KlassNode
*
oldTable
[]
)
static
void
JNICALL
cbTrackingObjectFree
(
jvmtiEnv
*
jvmti_env
,
jlong
tag
)
{
struct
bag
*
signatures
=
bagCreateBag
(
sizeof
(
char
*
),
10
);
jint
slot
;
if
(
signatures
==
NULL
)
{
EXIT_ERROR
(
AGENT_ERROR_OUT_OF_MEMORY
,
"signatures"
);
}
for
(
slot
=
0
;
slot
<
CT_HASH_SLOT_COUNT
;
slot
++
)
{
KlassNode
*
node
=
oldTable
[
slot
];
while
(
node
!=
NULL
)
{
KlassNode
*
next
;
char
**
sigSpot
;
/* Add signature to the signature bag */
sigSpot
=
bagAdd
(
signatures
);
if
(
sigSpot
==
NULL
)
{
EXIT_ERROR
(
AGENT_ERROR_OUT_OF_MEMORY
,
"signature bag"
);
}
*
sigSpot
=
node
->
signature
;
/* Free weak ref and the node itself */
JNI_FUNC_PTR
(
env
,
DeleteWeakGlobalRef
)(
env
,
node
->
klass
);
next
=
node
->
next
;
jvmtiDeallocate
(
node
);
node
=
next
;
}
debugMonitorEnter
(
classTrackLock
);
if
(
deletedSignatures
==
NULL
)
{
debugMonitorExit
(
classTrackLock
);
return
;
}
jvmtiDeallocate
(
oldTable
)
;
*
(
char
**
)
bagAdd
(
deletedSignatures
)
=
(
char
*
)
tag
;
return
signatures
;
debugMonitorExit
(
classTrackLock
)
;
}
/*
* Called after class unloads have occurred. Creates a new hash table
* of currently loaded prepared classes.
* The signatures of classes which were unloaded (not present in the
* new table) are returned.
* Called after class unloads have occurred.
* The signatures of classes which were unloaded are returned.
*/
struct
bag
*
classTrack_processUnloads
(
JNIEnv
*
env
)
{
KlassNode
**
newTable
;
struct
bag
*
unloadedSignatures
;
unloadedSignatures
=
NULL
;
newTable
=
jvmtiAllocate
(
CT_HASH_SLOT_COUNT
*
sizeof
(
KlassNode
*
));
if
(
newTable
==
NULL
)
{
EXIT_ERROR
(
AGENT_ERROR_OUT_OF_MEMORY
,
"classTrack table"
);
}
else
{
(
void
)
memset
(
newTable
,
0
,
CT_HASH_SLOT_COUNT
*
sizeof
(
KlassNode
*
));
WITH_LOCAL_REFS
(
env
,
1
)
{
jint
classCount
;
jclass
*
classes
;
jvmtiError
error
;
int
i
;
error
=
allLoadedClasses
(
&
classes
,
&
classCount
);
if
(
error
!=
JVMTI_ERROR_NONE
)
{
jvmtiDeallocate
(
newTable
);
EXIT_ERROR
(
error
,
"loaded classes"
);
}
else
{
/* Transfer each current class into the new table */
for
(
i
=
0
;
i
<
classCount
;
i
++
)
{
jclass
klass
=
classes
[
i
];
transferClass
(
env
,
klass
,
newTable
);
}
jvmtiDeallocate
(
classes
);
/* Delete old table, install new one */
unloadedSignatures
=
deleteTable
(
env
,
table
);
table
=
newTable
;
}
}
END_WITH_LOCAL_REFS
(
env
)
}
return
unloadedSignatures
;
debugMonitorEnter
(
classTrackLock
);
if
(
deletedSignatures
==
NULL
)
{
// Class tracking not initialized, nobody's interested.
debugMonitorExit
(
classTrackLock
);
return
NULL
;
}
struct
bag
*
deleted
=
deletedSignatures
;
deletedSignatures
=
bagCreateBag
(
sizeof
(
char
*
),
10
);
debugMonitorExit
(
classTrackLock
);
return
deleted
;
}
/*
* Add a class to the prepared class hash table.
* Assumes no duplicates.
* Add a class to the prepared class table.
*/
void
classTrack_addPreparedClass
(
JNIEnv
*
env
,
jclass
klass
)
classTrack_addPreparedClass
(
JNIEnv
*
env
_unused
,
jclass
klass
)
{
jint
slot
=
hashKlass
(
klass
);
KlassNode
**
head
=
&
table
[
slot
];
KlassNode
*
node
;
jvmtiError
error
;
jvmtiEnv
*
env
=
trackingEnv
;
if
(
gdata
->
assertOn
)
{
/* Check this is not a duplicate */
for
(
node
=
*
head
;
node
!=
NULL
;
node
=
node
->
next
)
{
if
(
isSameObject
(
env
,
klass
,
node
->
klass
))
{
JDI_ASSERT_FAILED
(
"Attempting to insert duplicate class"
);
break
;
}
if
(
gdata
&&
gdata
->
assertOn
)
{
// Check this is not already tagged.
jlong
tag
;
error
=
JVMTI_FUNC_PTR
(
trackingEnv
,
GetTag
)(
env
,
klass
,
&
tag
);
if
(
error
!=
JVMTI_ERROR_NONE
)
{
EXIT_ERROR
(
error
,
"Unable to GetTag with class trackingEnv"
);
}
JDI_ASSERT
(
tag
==
NOT_TAGGED
);
}
node
=
jvmtiAllocate
(
sizeof
(
KlassNode
));
if
(
node
==
NULL
)
{
EXIT_ERROR
(
AGENT_ERROR_OUT_OF_MEMORY
,
"KlassNode"
);
}
error
=
classSignature
(
klass
,
&
(
node
->
signature
),
NULL
);
char
*
signature
;
error
=
classSignature
(
klass
,
&
signature
,
NULL
);
if
(
error
!=
JVMTI_ERROR_NONE
)
{
jvmtiDeallocate
(
node
);
EXIT_ERROR
(
error
,
"signature"
);
}
if
((
node
->
klass
=
JNI_FUNC_PTR
(
env
,
NewWeakGlobalRef
)(
env
,
klass
))
==
NULL
)
{
jvmtiDeallocate
(
node
->
signature
);
jvmtiDeallocate
(
nod
e
);
EXIT_ERROR
(
AGENT_ERROR_NULL_POINTER
,
"NewWeakGlobalRef
"
);
error
=
JVMTI_FUNC_PTR
(
trackingEnv
,
SetTag
)(
env
,
klass
,
(
jlong
)
signature
);
if
(
error
!=
JVMTI_ERROR_NONE
)
{
jvmtiDeallocate
(
signatur
e
);
EXIT_ERROR
(
error
,
"SetTag
"
);
}
}
/* Insert the new node */
node
->
next
=
*
head
;
*
head
=
node
;
static
jboolean
setupEvents
()
{
jvmtiCapabilities
caps
;
memset
(
&
caps
,
0
,
sizeof
(
caps
));
caps
.
can_generate_object_free_events
=
1
;
jvmtiError
error
=
JVMTI_FUNC_PTR
(
trackingEnv
,
AddCapabilities
)(
trackingEnv
,
&
caps
);
if
(
error
!=
JVMTI_ERROR_NONE
)
{
return
JNI_FALSE
;
}
jvmtiEventCallbacks
cb
;
memset
(
&
cb
,
0
,
sizeof
(
cb
));
cb
.
ObjectFree
=
cbTrackingObjectFree
;
error
=
JVMTI_FUNC_PTR
(
trackingEnv
,
SetEventCallbacks
)(
trackingEnv
,
&
cb
,
sizeof
(
cb
));
if
(
error
!=
JVMTI_ERROR_NONE
)
{
return
JNI_FALSE
;
}
error
=
JVMTI_FUNC_PTR
(
trackingEnv
,
SetEventNotificationMode
)(
trackingEnv
,
JVMTI_ENABLE
,
JVMTI_EVENT_OBJECT_FREE
,
NULL
);
if
(
error
!=
JVMTI_ERROR_NONE
)
{
return
JNI_FALSE
;
}
return
JNI_TRUE
;
}
/*
* Called once to
build the initial prepared class hash table
.
* Called once to
initialize class-tracking
.
*/
void
classTrack_initialize
(
JNIEnv
*
env
)
{
WITH_LOCAL_REFS
(
env
,
1
)
{
deletedSignatures
=
NULL
;
classTrackLock
=
debugMonitorCreate
(
"Deleted class tag lock"
);
trackingEnv
=
getSpecialJvmti
();
if
(
trackingEnv
==
NULL
)
{
EXIT_ERROR
(
AGENT_ERROR_INTERNAL
,
"Failed to allocate tag-tracking jvmtiEnv"
);
}
if
(
!
setupEvents
())
{
EXIT_ERROR
(
AGENT_ERROR_INTERNAL
,
"Unable to setup ObjectFree tracking"
);
}
jint
classCount
;
jclass
*
classes
;
...
...
@@ -260,35 +173,53 @@ classTrack_initialize(JNIEnv *env)
error
=
allLoadedClasses
(
&
classes
,
&
classCount
);
if
(
error
==
JVMTI_ERROR_NONE
)
{
table
=
jvmtiAllocate
(
CT_HASH_SLOT_COUNT
*
sizeof
(
KlassNode
*
));
if
(
table
!=
NULL
)
{
(
void
)
memset
(
table
,
0
,
CT_HASH_SLOT_COUNT
*
sizeof
(
KlassNode
*
));
for
(
i
=
0
;
i
<
classCount
;
i
++
)
{
for
(
i
=
0
;
i
<
classCount
;
i
++
)
{
jclass
klass
=
classes
[
i
];
jint
status
;
jint
wanted
=
(
JVMTI_CLASS_STATUS_PREPARED
|
JVMTI_CLASS_STATUS_ARRAY
);
/* We only want prepared classes and arrays */
jint
wanted
=
JVMTI_CLASS_STATUS_PREPARED
|
JVMTI_CLASS_STATUS_ARRAY
;
status
=
classStatus
(
klass
);
if
(
(
status
&
wanted
)
!=
0
)
{
if
((
status
&
wanted
)
!=
0
)
{
classTrack_addPreparedClass
(
env
,
klass
);
}
}
}
else
{
jvmtiDeallocate
(
classes
);
EXIT_ERROR
(
AGENT_ERROR_OUT_OF_MEMORY
,
"KlassNode"
);
}
jvmtiDeallocate
(
classes
);
}
else
{
EXIT_ERROR
(
error
,
"loaded classes array"
);
}
}
}
END_WITH_LOCAL_REFS
(
env
)
/*
* Called to activate class-tracking when a listener registers for EI_GC_FINISH.
*/
void
classTrack_activate
(
JNIEnv
*
env
)
{
debugMonitorEnter
(
classTrackLock
);
deletedSignatures
=
bagCreateBag
(
sizeof
(
char
*
),
1000
);
debugMonitorExit
(
classTrackLock
);
}
static
jboolean
cleanDeleted
(
void
*
signatureVoid
,
void
*
arg
)
{
char
*
sig
=
*
(
char
**
)
signatureVoid
;
jvmtiDeallocate
(
sig
);
return
JNI_TRUE
;
}
/*
* Called when agent detaches.
*/
void
classTrack_reset
(
void
)
{
debugMonitorEnter
(
classTrackLock
);
if
(
deletedSignatures
!=
NULL
)
{
bagEnumerateOver
(
deletedSignatures
,
cleanDeleted
,
NULL
);
bagDestroyBag
(
deletedSignatures
);
deletedSignatures
=
NULL
;
}
debugMonitorExit
(
classTrackLock
);
}
src/share/back/classTrack.h
浏览文件 @
56f42edd
...
...
@@ -45,6 +45,12 @@ classTrack_addPreparedClass(JNIEnv *env, jclass klass);
void
classTrack_initialize
(
JNIEnv
*
env
);
/*
* Activates class tracking.
*/
void
classTrack_activate
(
JNIEnv
*
env
);
/*
* Reset class tracking.
*/
...
...
src/share/back/eventHandler.c
浏览文件 @
56f42edd
...
...
@@ -1628,6 +1628,9 @@ installHandler(HandlerNode *node,
node
->
handlerID
=
external
?
++
requestIdCounter
:
0
;
error
=
eventFilterRestricted_install
(
node
);
if
(
node
->
ei
==
EI_GC_FINISH
)
{
classTrack_activate
(
getEnv
());
}
if
(
error
==
JVMTI_ERROR_NONE
)
{
insert
(
getHandlerChain
(
node
->
ei
),
node
);
}
...
...
src/share/back/util.c
浏览文件 @
56f42edd
...
...
@@ -1744,7 +1744,7 @@ isMethodObsolete(jmethodID method)
}
/* Get the jvmti environment to be used with tags */
static
jvmtiEnv
*
jvmtiEnv
*
getSpecialJvmti
(
void
)
{
jvmtiEnv
*
jvmti
;
...
...
src/share/back/util.h
浏览文件 @
56f42edd
...
...
@@ -429,4 +429,6 @@ void createLocalRefSpace(JNIEnv *env, jint capacity);
void
saveGlobalRef
(
JNIEnv
*
env
,
jobject
obj
,
jobject
*
pobj
);
void
tossGlobalRef
(
JNIEnv
*
env
,
jobject
*
pobj
);
jvmtiEnv
*
getSpecialJvmti
(
void
);
#endif
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录