Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
dragonwell8_hotspot
提交
22bf77f3
D
dragonwell8_hotspot
项目概览
openanolis
/
dragonwell8_hotspot
通知
2
Star
2
Fork
0
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
D
dragonwell8_hotspot
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
提交
22bf77f3
编写于
9月 23, 2014
作者:
I
iveresov
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
8058744: Crash in C1 OSRed method w/ Unsafe usage
Summary: Fix UnsafeRawOp optimizations Reviewed-by: kvn, drchase, vlivanov
上级
5c146c98
变更
2
隐藏空白更改
内联
并排
Showing
2 changed file
with
139 addition
and
82 deletion
+139
-82
src/share/vm/c1/c1_Canonicalizer.cpp
src/share/vm/c1/c1_Canonicalizer.cpp
+74
-42
src/share/vm/c1/c1_LIRGenerator.cpp
src/share/vm/c1/c1_LIRGenerator.cpp
+65
-40
未找到文件。
src/share/vm/c1/c1_Canonicalizer.cpp
浏览文件 @
22bf77f3
...
@@ -327,7 +327,7 @@ void Canonicalizer::do_ShiftOp (ShiftOp* x) {
...
@@ -327,7 +327,7 @@ void Canonicalizer::do_ShiftOp (ShiftOp* x) {
if
(
t2
->
is_constant
())
{
if
(
t2
->
is_constant
())
{
switch
(
t2
->
tag
())
{
switch
(
t2
->
tag
())
{
case
intTag
:
if
(
t2
->
as_IntConstant
()
->
value
()
==
0
)
set_canonical
(
x
->
x
());
return
;
case
intTag
:
if
(
t2
->
as_IntConstant
()
->
value
()
==
0
)
set_canonical
(
x
->
x
());
return
;
case
longTag
:
if
(
t2
->
as_
IntConstant
()
->
value
()
==
0
)
set_canonical
(
x
->
x
());
return
;
case
longTag
:
if
(
t2
->
as_
LongConstant
()
->
value
()
==
(
jlong
)
0
)
set_canonical
(
x
->
x
());
return
;
default
:
ShouldNotReachHere
();
default
:
ShouldNotReachHere
();
}
}
}
}
...
@@ -808,28 +808,41 @@ void Canonicalizer::do_ExceptionObject(ExceptionObject* x) {}
...
@@ -808,28 +808,41 @@ void Canonicalizer::do_ExceptionObject(ExceptionObject* x) {}
static
bool
match_index_and_scale
(
Instruction
*
instr
,
static
bool
match_index_and_scale
(
Instruction
*
instr
,
Instruction
**
index
,
Instruction
**
index
,
int
*
log2_scale
,
int
*
log2_scale
)
{
Instruction
**
instr_to_unpin
)
{
// Skip conversion ops. This works only on 32bit because of the implicit l2i that the
*
instr_to_unpin
=
NULL
;
// unsafe performs.
#ifndef _LP64
// Skip conversion ops
Convert
*
convert
=
instr
->
as_Convert
();
Convert
*
convert
=
instr
->
as_Convert
();
if
(
convert
!=
NULL
)
{
if
(
convert
!=
NULL
&&
convert
->
op
()
==
Bytecodes
::
_i2l
)
{
assert
(
convert
->
value
()
->
type
()
==
intType
,
"invalid input type"
);
instr
=
convert
->
value
();
instr
=
convert
->
value
();
}
}
#endif
ShiftOp
*
shift
=
instr
->
as_ShiftOp
();
ShiftOp
*
shift
=
instr
->
as_ShiftOp
();
if
(
shift
!=
NULL
)
{
if
(
shift
!=
NULL
)
{
if
(
shift
->
is_pinned
())
{
if
(
shift
->
op
()
==
Bytecodes
::
_lshl
)
{
*
instr_to_unpin
=
shift
;
assert
(
shift
->
x
()
->
type
()
==
longType
,
"invalid input type"
);
}
else
{
#ifndef _LP64
if
(
shift
->
op
()
==
Bytecodes
::
_ishl
)
{
assert
(
shift
->
x
()
->
type
()
==
intType
,
"invalid input type"
);
}
else
{
return
false
;
}
#else
return
false
;
#endif
}
}
// Constant shift value?
// Constant shift value?
Constant
*
con
=
shift
->
y
()
->
as_Constant
();
Constant
*
con
=
shift
->
y
()
->
as_Constant
();
if
(
con
==
NULL
)
return
false
;
if
(
con
==
NULL
)
return
false
;
// Well-known type and value?
// Well-known type and value?
IntConstant
*
val
=
con
->
type
()
->
as_IntConstant
();
IntConstant
*
val
=
con
->
type
()
->
as_IntConstant
();
if
(
val
==
NULL
)
return
false
;
assert
(
val
!=
NULL
,
"Should be an int constant"
)
;
if
(
shift
->
x
()
->
type
()
!=
intType
)
return
false
;
*
index
=
shift
->
x
();
*
index
=
shift
->
x
();
int
tmp_scale
=
val
->
value
();
int
tmp_scale
=
val
->
value
();
if
(
tmp_scale
>=
0
&&
tmp_scale
<
4
)
{
if
(
tmp_scale
>=
0
&&
tmp_scale
<
4
)
{
...
@@ -842,31 +855,42 @@ static bool match_index_and_scale(Instruction* instr,
...
@@ -842,31 +855,42 @@ static bool match_index_and_scale(Instruction* instr,
ArithmeticOp
*
arith
=
instr
->
as_ArithmeticOp
();
ArithmeticOp
*
arith
=
instr
->
as_ArithmeticOp
();
if
(
arith
!=
NULL
)
{
if
(
arith
!=
NULL
)
{
if
(
arith
->
is_pinned
())
{
// See if either arg is a known constant
*
instr_to_unpin
=
arith
;
Constant
*
con
=
arith
->
x
()
->
as_Constant
();
if
(
con
!=
NULL
)
{
*
index
=
arith
->
y
();
}
else
{
con
=
arith
->
y
()
->
as_Constant
();
if
(
con
==
NULL
)
return
false
;
*
index
=
arith
->
x
();
}
}
long
const_value
;
// Check for integer multiply
// Check for integer multiply
if
(
arith
->
op
()
==
Bytecodes
::
_imul
)
{
if
(
arith
->
op
()
==
Bytecodes
::
_lmul
)
{
// See if either arg is a known constant
assert
((
*
index
)
->
type
()
==
longType
,
"invalid input type"
);
Constant
*
con
=
arith
->
x
()
->
as_Constant
();
LongConstant
*
val
=
con
->
type
()
->
as_LongConstant
();
if
(
con
!=
NULL
)
{
assert
(
val
!=
NULL
,
"expecting a long constant"
);
*
index
=
arith
->
y
();
const_value
=
val
->
value
();
}
else
{
#ifndef _LP64
if
(
arith
->
op
()
==
Bytecodes
::
_imul
)
{
assert
((
*
index
)
->
type
()
==
intType
,
"invalid input type"
);
IntConstant
*
val
=
con
->
type
()
->
as_IntConstant
();
assert
(
val
!=
NULL
,
"expecting an int constant"
);
const_value
=
val
->
value
();
}
else
{
}
else
{
con
=
arith
->
y
()
->
as_Constant
();
return
false
;
if
(
con
==
NULL
)
return
false
;
*
index
=
arith
->
x
();
}
if
((
*
index
)
->
type
()
!=
intType
)
return
false
;
// Well-known type and value?
IntConstant
*
val
=
con
->
type
()
->
as_IntConstant
();
if
(
val
==
NULL
)
return
false
;
switch
(
val
->
value
())
{
case
1
:
*
log2_scale
=
0
;
return
true
;
case
2
:
*
log2_scale
=
1
;
return
true
;
case
4
:
*
log2_scale
=
2
;
return
true
;
case
8
:
*
log2_scale
=
3
;
return
true
;
default:
return
false
;
}
}
#else
return
false
;
#endif
}
switch
(
const_value
)
{
case
1
:
*
log2_scale
=
0
;
return
true
;
case
2
:
*
log2_scale
=
1
;
return
true
;
case
4
:
*
log2_scale
=
2
;
return
true
;
case
8
:
*
log2_scale
=
3
;
return
true
;
default:
return
false
;
}
}
}
}
...
@@ -879,29 +903,37 @@ static bool match(UnsafeRawOp* x,
...
@@ -879,29 +903,37 @@ static bool match(UnsafeRawOp* x,
Instruction
**
base
,
Instruction
**
base
,
Instruction
**
index
,
Instruction
**
index
,
int
*
log2_scale
)
{
int
*
log2_scale
)
{
Instruction
*
instr_to_unpin
=
NULL
;
ArithmeticOp
*
root
=
x
->
base
()
->
as_ArithmeticOp
();
ArithmeticOp
*
root
=
x
->
base
()
->
as_ArithmeticOp
();
if
(
root
==
NULL
)
return
false
;
if
(
root
==
NULL
)
return
false
;
// Limit ourselves to addition for now
// Limit ourselves to addition for now
if
(
root
->
op
()
!=
Bytecodes
::
_ladd
)
return
false
;
if
(
root
->
op
()
!=
Bytecodes
::
_ladd
)
return
false
;
bool
match_found
=
false
;
// Try to find shift or scale op
// Try to find shift or scale op
if
(
match_index_and_scale
(
root
->
y
(),
index
,
log2_scale
,
&
instr_to_unpin
))
{
if
(
match_index_and_scale
(
root
->
y
(),
index
,
log2_scale
))
{
*
base
=
root
->
x
();
*
base
=
root
->
x
();
}
else
if
(
match_index_and_scale
(
root
->
x
(),
index
,
log2_scale
,
&
instr_to_unpin
))
{
match_found
=
true
;
}
else
if
(
match_index_and_scale
(
root
->
x
(),
index
,
log2_scale
))
{
*
base
=
root
->
y
();
*
base
=
root
->
y
();
}
else
if
(
root
->
y
()
->
as_Convert
()
!=
NULL
)
{
match_found
=
true
;
}
else
if
(
NOT_LP64
(
root
->
y
()
->
as_Convert
()
!=
NULL
)
LP64_ONLY
(
false
))
{
// Skipping i2l works only on 32bit because of the implicit l2i that the unsafe performs.
// 64bit needs a real sign-extending conversion.
Convert
*
convert
=
root
->
y
()
->
as_Convert
();
Convert
*
convert
=
root
->
y
()
->
as_Convert
();
if
(
convert
->
op
()
==
Bytecodes
::
_i2l
&&
convert
->
value
()
->
type
()
==
intType
)
{
if
(
convert
->
op
()
==
Bytecodes
::
_i2l
)
{
assert
(
convert
->
value
()
->
type
()
==
intType
,
"should be an int"
);
// pick base and index, setting scale at 1
// pick base and index, setting scale at 1
*
base
=
root
->
x
();
*
base
=
root
->
x
();
*
index
=
convert
->
value
();
*
index
=
convert
->
value
();
*
log2_scale
=
0
;
*
log2_scale
=
0
;
}
else
{
match_found
=
true
;
return
false
;
}
}
}
else
{
}
// doesn't match any expected sequences
// The default solution
return
false
;
if
(
!
match_found
)
{
*
base
=
root
->
x
();
*
index
=
root
->
y
();
*
log2_scale
=
0
;
}
}
// If the value is pinned then it will be always be computed so
// If the value is pinned then it will be always be computed so
...
...
src/share/vm/c1/c1_LIRGenerator.cpp
浏览文件 @
22bf77f3
...
@@ -2042,6 +2042,8 @@ void LIRGenerator::do_RoundFP(RoundFP* x) {
...
@@ -2042,6 +2042,8 @@ void LIRGenerator::do_RoundFP(RoundFP* x) {
}
}
}
}
// Here UnsafeGetRaw may have x->base() and x->index() be int or long
// on both 64 and 32 bits. Expecting x->base() to be always long on 64bit.
void
LIRGenerator
::
do_UnsafeGetRaw
(
UnsafeGetRaw
*
x
)
{
void
LIRGenerator
::
do_UnsafeGetRaw
(
UnsafeGetRaw
*
x
)
{
LIRItem
base
(
x
->
base
(),
this
);
LIRItem
base
(
x
->
base
(),
this
);
LIRItem
idx
(
this
);
LIRItem
idx
(
this
);
...
@@ -2056,50 +2058,73 @@ void LIRGenerator::do_UnsafeGetRaw(UnsafeGetRaw* x) {
...
@@ -2056,50 +2058,73 @@ void LIRGenerator::do_UnsafeGetRaw(UnsafeGetRaw* x) {
int
log2_scale
=
0
;
int
log2_scale
=
0
;
if
(
x
->
has_index
())
{
if
(
x
->
has_index
())
{
assert
(
x
->
index
()
->
type
()
->
tag
()
==
intTag
,
"should not find non-int index"
);
log2_scale
=
x
->
log2_scale
();
log2_scale
=
x
->
log2_scale
();
}
}
assert
(
!
x
->
has_index
()
||
idx
.
value
()
==
x
->
index
(),
"should match"
);
assert
(
!
x
->
has_index
()
||
idx
.
value
()
==
x
->
index
(),
"should match"
);
LIR_Opr
base_op
=
base
.
result
();
LIR_Opr
base_op
=
base
.
result
();
LIR_Opr
index_op
=
idx
.
result
();
#ifndef _LP64
#ifndef _LP64
if
(
x
->
base
()
->
type
()
->
tag
()
==
longTag
)
{
if
(
x
->
base
()
->
type
()
->
tag
()
==
longTag
)
{
base_op
=
new_register
(
T_INT
);
base_op
=
new_register
(
T_INT
);
__
convert
(
Bytecodes
::
_l2i
,
base
.
result
(),
base_op
);
__
convert
(
Bytecodes
::
_l2i
,
base
.
result
(),
base_op
);
}
else
{
assert
(
x
->
base
()
->
type
()
->
tag
()
==
intTag
,
"must be"
);
}
}
if
(
x
->
has_index
())
{
if
(
x
->
index
()
->
type
()
->
tag
()
==
longTag
)
{
LIR_Opr
long_index_op
=
index_op
;
if
(
x
->
index
()
->
type
()
->
is_constant
())
{
long_index_op
=
new_register
(
T_LONG
);
__
move
(
index_op
,
long_index_op
);
}
index_op
=
new_register
(
T_INT
);
__
convert
(
Bytecodes
::
_l2i
,
long_index_op
,
index_op
);
}
else
{
assert
(
x
->
index
()
->
type
()
->
tag
()
==
intTag
,
"must be"
);
}
}
// At this point base and index should be all ints.
assert
(
base_op
->
type
()
==
T_INT
&&
!
base_op
->
is_constant
(),
"base should be an non-constant int"
);
assert
(
!
x
->
has_index
()
||
index_op
->
type
()
==
T_INT
,
"index should be an int"
);
#else
if
(
x
->
has_index
())
{
if
(
x
->
index
()
->
type
()
->
tag
()
==
intTag
)
{
if
(
!
x
->
index
()
->
type
()
->
is_constant
())
{
index_op
=
new_register
(
T_LONG
);
__
convert
(
Bytecodes
::
_i2l
,
idx
.
result
(),
index_op
);
}
}
else
{
assert
(
x
->
index
()
->
type
()
->
tag
()
==
longTag
,
"must be"
);
if
(
x
->
index
()
->
type
()
->
is_constant
())
{
index_op
=
new_register
(
T_LONG
);
__
move
(
idx
.
result
(),
index_op
);
}
}
}
// At this point base is a long non-constant
// Index is a long register or a int constant.
// We allow the constant to stay an int because that would allow us a more compact encoding by
// embedding an immediate offset in the address expression. If we have a long constant, we have to
// move it into a register first.
assert
(
base_op
->
type
()
==
T_LONG
&&
!
base_op
->
is_constant
(),
"base must be a long non-constant"
);
assert
(
!
x
->
has_index
()
||
(
index_op
->
type
()
==
T_INT
&&
index_op
->
is_constant
())
||
(
index_op
->
type
()
==
T_LONG
&&
!
index_op
->
is_constant
()),
"unexpected index type"
);
#endif
#endif
BasicType
dst_type
=
x
->
basic_type
();
BasicType
dst_type
=
x
->
basic_type
();
LIR_Opr
index_op
=
idx
.
result
();
LIR_Address
*
addr
;
LIR_Address
*
addr
;
if
(
index_op
->
is_constant
())
{
if
(
index_op
->
is_constant
())
{
assert
(
log2_scale
==
0
,
"must not have a scale"
);
assert
(
log2_scale
==
0
,
"must not have a scale"
);
assert
(
index_op
->
type
()
==
T_INT
,
"only int constants supported"
);
addr
=
new
LIR_Address
(
base_op
,
index_op
->
as_jint
(),
dst_type
);
addr
=
new
LIR_Address
(
base_op
,
index_op
->
as_jint
(),
dst_type
);
}
else
{
}
else
{
#ifdef X86
#ifdef X86
#ifdef _LP64
if
(
!
index_op
->
is_illegal
()
&&
index_op
->
type
()
==
T_INT
)
{
LIR_Opr
tmp
=
new_pointer_register
();
__
convert
(
Bytecodes
::
_i2l
,
index_op
,
tmp
);
index_op
=
tmp
;
}
#endif
addr
=
new
LIR_Address
(
base_op
,
index_op
,
LIR_Address
::
Scale
(
log2_scale
),
0
,
dst_type
);
addr
=
new
LIR_Address
(
base_op
,
index_op
,
LIR_Address
::
Scale
(
log2_scale
),
0
,
dst_type
);
#elif defined(ARM)
#elif defined(ARM)
addr
=
generate_address
(
base_op
,
index_op
,
log2_scale
,
0
,
dst_type
);
addr
=
generate_address
(
base_op
,
index_op
,
log2_scale
,
0
,
dst_type
);
#else
#else
if
(
index_op
->
is_illegal
()
||
log2_scale
==
0
)
{
if
(
index_op
->
is_illegal
()
||
log2_scale
==
0
)
{
#ifdef _LP64
if
(
!
index_op
->
is_illegal
()
&&
index_op
->
type
()
==
T_INT
)
{
LIR_Opr
tmp
=
new_pointer_register
();
__
convert
(
Bytecodes
::
_i2l
,
index_op
,
tmp
);
index_op
=
tmp
;
}
#endif
addr
=
new
LIR_Address
(
base_op
,
index_op
,
dst_type
);
addr
=
new
LIR_Address
(
base_op
,
index_op
,
dst_type
);
}
else
{
}
else
{
LIR_Opr
tmp
=
new_pointer_register
();
LIR_Opr
tmp
=
new_pointer_register
();
...
@@ -2126,7 +2151,6 @@ void LIRGenerator::do_UnsafePutRaw(UnsafePutRaw* x) {
...
@@ -2126,7 +2151,6 @@ void LIRGenerator::do_UnsafePutRaw(UnsafePutRaw* x) {
BasicType
type
=
x
->
basic_type
();
BasicType
type
=
x
->
basic_type
();
if
(
x
->
has_index
())
{
if
(
x
->
has_index
())
{
assert
(
x
->
index
()
->
type
()
->
tag
()
==
intTag
,
"should not find non-int index"
);
log2_scale
=
x
->
log2_scale
();
log2_scale
=
x
->
log2_scale
();
}
}
...
@@ -2149,38 +2173,39 @@ void LIRGenerator::do_UnsafePutRaw(UnsafePutRaw* x) {
...
@@ -2149,38 +2173,39 @@ void LIRGenerator::do_UnsafePutRaw(UnsafePutRaw* x) {
set_no_result
(
x
);
set_no_result
(
x
);
LIR_Opr
base_op
=
base
.
result
();
LIR_Opr
base_op
=
base
.
result
();
LIR_Opr
index_op
=
idx
.
result
();
#ifndef _LP64
#ifndef _LP64
if
(
x
->
base
()
->
type
()
->
tag
()
==
longTag
)
{
if
(
x
->
base
()
->
type
()
->
tag
()
==
longTag
)
{
base_op
=
new_register
(
T_INT
);
base_op
=
new_register
(
T_INT
);
__
convert
(
Bytecodes
::
_l2i
,
base
.
result
(),
base_op
);
__
convert
(
Bytecodes
::
_l2i
,
base
.
result
(),
base_op
);
}
else
{
assert
(
x
->
base
()
->
type
()
->
tag
()
==
intTag
,
"must be"
);
}
}
if
(
x
->
has_index
())
{
if
(
x
->
index
()
->
type
()
->
tag
()
==
longTag
)
{
index_op
=
new_register
(
T_INT
);
__
convert
(
Bytecodes
::
_l2i
,
idx
.
result
(),
index_op
);
}
}
// At this point base and index should be all ints and not constants
assert
(
base_op
->
type
()
==
T_INT
&&
!
base_op
->
is_constant
(),
"base should be an non-constant int"
);
assert
(
!
x
->
has_index
()
||
(
index_op
->
type
()
==
T_INT
&&
!
index_op
->
is_constant
()),
"index should be an non-constant int"
);
#else
if
(
x
->
has_index
())
{
if
(
x
->
index
()
->
type
()
->
tag
()
==
intTag
)
{
index_op
=
new_register
(
T_LONG
);
__
convert
(
Bytecodes
::
_i2l
,
idx
.
result
(),
index_op
);
}
}
// At this point base and index are long and non-constant
assert
(
base_op
->
type
()
==
T_LONG
&&
!
base_op
->
is_constant
(),
"base must be a non-constant long"
);
assert
(
!
x
->
has_index
()
||
(
index_op
->
type
()
==
T_LONG
&&
!
index_op
->
is_constant
()),
"index must be a non-constant long"
);
#endif
#endif
LIR_Opr
index_op
=
idx
.
result
();
if
(
log2_scale
!=
0
)
{
if
(
log2_scale
!=
0
)
{
// temporary fix (platform dependent code without shift on Intel would be better)
// temporary fix (platform dependent code without shift on Intel would be better)
index_op
=
new_pointer_register
();
// TODO: ARM also allows embedded shift in the address
#ifdef _LP64
if
(
idx
.
result
()
->
type
()
==
T_INT
)
{
__
convert
(
Bytecodes
::
_i2l
,
idx
.
result
(),
index_op
);
}
else
{
#endif
// TODO: ARM also allows embedded shift in the address
__
move
(
idx
.
result
(),
index_op
);
#ifdef _LP64
}
#endif
__
shift_left
(
index_op
,
log2_scale
,
index_op
);
__
shift_left
(
index_op
,
log2_scale
,
index_op
);
}
}
#ifdef _LP64
else
if
(
!
index_op
->
is_illegal
()
&&
index_op
->
type
()
==
T_INT
)
{
LIR_Opr
tmp
=
new_pointer_register
();
__
convert
(
Bytecodes
::
_i2l
,
index_op
,
tmp
);
index_op
=
tmp
;
}
#endif
LIR_Address
*
addr
=
new
LIR_Address
(
base_op
,
index_op
,
x
->
basic_type
());
LIR_Address
*
addr
=
new
LIR_Address
(
base_op
,
index_op
,
x
->
basic_type
());
__
move
(
value
.
result
(),
addr
);
__
move
(
value
.
result
(),
addr
);
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录