Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
dragonwell8_jdk
提交
5c9dc93c
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看板
提交
5c9dc93c
编写于
10月 07, 2016
作者:
B
bchristi
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
8151486: Class.forName causes memory leak
Summary: Remove ClassLoader.domains Reviewed-by: coleenp, dholmes, mchung, naoto
上级
9d83225b
变更
4
隐藏空白更改
内联
并排
Showing
4 changed file
with
174 addition
and
7 deletion
+174
-7
src/share/classes/java/lang/ClassLoader.java
src/share/classes/java/lang/ClassLoader.java
+0
-7
test/java/lang/ClassLoader/forNameLeak/ClassForName.java
test/java/lang/ClassLoader/forNameLeak/ClassForName.java
+45
-0
test/java/lang/ClassLoader/forNameLeak/ClassForNameLeak.java
test/java/lang/ClassLoader/forNameLeak/ClassForNameLeak.java
+123
-0
test/java/lang/ClassLoader/forNameLeak/test.policy
test/java/lang/ClassLoader/forNameLeak/test.policy
+6
-0
未找到文件。
src/share/classes/java/lang/ClassLoader.java
浏览文件 @
5c9dc93c
...
@@ -255,9 +255,6 @@ public abstract class ClassLoader {
...
@@ -255,9 +255,6 @@ public abstract class ClassLoader {
new
ProtectionDomain
(
new
CodeSource
(
null
,
(
Certificate
[])
null
),
new
ProtectionDomain
(
new
CodeSource
(
null
,
(
Certificate
[])
null
),
null
,
this
,
null
);
null
,
this
,
null
);
// The initiating protection domains for all classes loaded by this loader
private
final
Set
<
ProtectionDomain
>
domains
;
// Invoked by the VM to record every loaded class with this loader.
// Invoked by the VM to record every loaded class with this loader.
void
addClass
(
Class
<?>
c
)
{
void
addClass
(
Class
<?>
c
)
{
classes
.
addElement
(
c
);
classes
.
addElement
(
c
);
...
@@ -281,14 +278,11 @@ public abstract class ClassLoader {
...
@@ -281,14 +278,11 @@ public abstract class ClassLoader {
if
(
ParallelLoaders
.
isRegistered
(
this
.
getClass
()))
{
if
(
ParallelLoaders
.
isRegistered
(
this
.
getClass
()))
{
parallelLockMap
=
new
ConcurrentHashMap
<>();
parallelLockMap
=
new
ConcurrentHashMap
<>();
package2certs
=
new
ConcurrentHashMap
<>();
package2certs
=
new
ConcurrentHashMap
<>();
domains
=
Collections
.
synchronizedSet
(
new
HashSet
<
ProtectionDomain
>());
assertionLock
=
new
Object
();
assertionLock
=
new
Object
();
}
else
{
}
else
{
// no finer-grained lock; lock on the classloader instance
// no finer-grained lock; lock on the classloader instance
parallelLockMap
=
null
;
parallelLockMap
=
null
;
package2certs
=
new
Hashtable
<>();
package2certs
=
new
Hashtable
<>();
domains
=
new
HashSet
<>();
assertionLock
=
this
;
assertionLock
=
this
;
}
}
}
}
...
@@ -505,7 +499,6 @@ public abstract class ClassLoader {
...
@@ -505,7 +499,6 @@ public abstract class ClassLoader {
},
new
AccessControlContext
(
new
ProtectionDomain
[]
{
pd
}));
},
new
AccessControlContext
(
new
ProtectionDomain
[]
{
pd
}));
}
}
}
}
domains
.
add
(
pd
);
}
}
/**
/**
...
...
test/java/lang/ClassLoader/forNameLeak/ClassForName.java
0 → 100644
浏览文件 @
5c9dc93c
/*
* Copyright (c) 2016, 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.
*/
import
java.net.URLClassLoader
;
/*
* This class is loaded by the custom URLClassLoader, and then calls
* Class.forName();
*/
public
class
ClassForName
implements
Runnable
{
static
{
if
(!(
ClassForName
.
class
.
getClassLoader
()
instanceof
URLClassLoader
))
{
throw
new
RuntimeException
(
"Supposed to be loaded by URLClassLoader"
);
}
}
public
void
run
()
{
try
{
Class
.
forName
(
java
.
util
.
List
.
class
.
getName
(),
false
,
ClassLoader
.
getSystemClassLoader
());
}
catch
(
Throwable
e
)
{
e
.
printStackTrace
();
}
}
}
test/java/lang/ClassLoader/forNameLeak/ClassForNameLeak.java
0 → 100644
浏览文件 @
5c9dc93c
/*
* Copyright (c) 2016, 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 8151486
* @summary Call Class.forName() on the system classloader from a class loaded
* from a custom classloader.
* @library /lib/testlibrary
* @build jdk.testlibrary.Utils jdk.testlibrary.JarUtils
* @build ClassForName ClassForNameLeak
* @run main/othervm/policy=test.policy -Djava.security.manager ClassForNameLeak
*/
import
java.lang.ref.PhantomReference
;
import
java.lang.ref.Reference
;
import
java.lang.ref.ReferenceQueue
;
import
java.net.URL
;
import
java.net.URLClassLoader
;
import
java.nio.file.FileSystems
;
import
java.nio.file.Files
;
import
java.nio.file.Path
;
import
java.nio.file.Paths
;
import
java.util.List
;
import
java.util.concurrent.Callable
;
import
java.util.concurrent.ExecutorService
;
import
java.util.concurrent.Executors
;
import
java.util.concurrent.Future
;
import
java.util.stream.Collectors
;
import
java.util.stream.Stream
;
import
jdk.testlibrary.Utils
;
import
jdk.testlibrary.JarUtils
;
/*
* Create .jar, load ClassForName from .jar using a URLClassLoader
*/
public
class
ClassForNameLeak
{
private
static
final
long
TIMEOUT
=
(
long
)(
5000.0
*
Utils
.
TIMEOUT_FACTOR
);
private
static
final
String
TESTCLASSES
=
System
.
getProperty
(
"test.classes"
,
"."
);
private
static
final
String
CLASSFILENAME
=
"ClassForName.class"
;
private
static
final
int
THREADS
=
10
;
private
static
final
ReferenceQueue
<
ClassLoader
>
rq
=
new
ReferenceQueue
<>();
// Use a new classloader to load the ClassForName class, then run its
// Runnable.
public
static
PhantomReference
<
ClassLoader
>
loadAndRun
(
Path
jarFilePath
)
throws
Exception
{
ClassLoader
classLoader
=
new
URLClassLoader
(
new
URL
[]{
jarFilePath
.
toUri
().
toURL
()})
{
@Override
public
String
toString
()
{
return
"LeakedClassLoader"
;
}
};
Class
<?>
loadClass
=
Class
.
forName
(
"ClassForName"
,
true
,
classLoader
);
((
Runnable
)
loadClass
.
newInstance
()).
run
();
PhantomReference
<
ClassLoader
>
ref
=
new
PhantomReference
<>(
classLoader
,
rq
);
System
.
out
.
println
(
"returning phantom ref: "
+
ref
+
" to "
+
classLoader
);
return
ref
;
}
public
static
void
main
(
final
String
[]
args
)
throws
Exception
{
Path
classFile
=
FileSystems
.
getDefault
().
getPath
(
TESTCLASSES
,
CLASSFILENAME
);
// Create a temporary .jar file containing ClassForName.class
Path
testClassesDir
=
Paths
.
get
(
TESTCLASSES
);
Path
jarFilePath
=
Files
.
createTempFile
(
"cfn"
,
".jar"
);
JarUtils
.
createJarFile
(
jarFilePath
,
testClassesDir
,
classFile
);
jarFilePath
.
toFile
().
deleteOnExit
();
// Remove the ClassForName.class file that jtreg built, to make sure
// we're loading from the tmp .jar
Files
.
delete
(
classFile
);
// Make simultaneous calls to the test method, to stress things a bit
ExecutorService
es
=
Executors
.
newFixedThreadPool
(
THREADS
);
List
<
Callable
<
PhantomReference
<
ClassLoader
>>>
callables
=
Stream
.
generate
(()
->
{
Callable
<
PhantomReference
<
ClassLoader
>>
cprcl
=
()
->
{
return
loadAndRun
(
jarFilePath
);
};
return
cprcl
;
}).
limit
(
THREADS
).
collect
(
Collectors
.
toList
());
List
<
Future
<
PhantomReference
<
ClassLoader
>>>
refs
=
es
.
invokeAll
(
callables
);
// Give the GC a chance to enqueue the PhantomReferences
for
(
int
i
=
0
;
i
<
10
;
i
++)
{
System
.
gc
();
}
// Make sure all PhantomReferences to the leaked classloader are enqueued
for
(
int
j
=
0
;
j
<
THREADS
;
j
++)
{
Reference
rmRef
=
rq
.
remove
(
TIMEOUT
);
if
(
rmRef
==
null
)
{
throw
new
RuntimeException
(
"ClassLoader was never enqueued!"
);
}
else
{
System
.
out
.
println
(
"Enqueued "
+
rmRef
);
}
}
System
.
out
.
println
(
"All Classloaders successfully enqued"
);
}
}
test/java/lang/ClassLoader/forNameLeak/test.policy
0 → 100644
浏览文件 @
5c9dc93c
grant {
permission java.io.FilePermission "<<ALL FILES>>", "read, write, delete";
permission java.lang.RuntimePermission "createClassLoader";
permission java.lang.RuntimePermission "getClassLoader";
permission java.util.PropertyPermission "*", "read"; /* for Utils */
};
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录