Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openanolis
dragonwell8_hotspot
提交
0fd06556
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看板
提交
0fd06556
编写于
4月 25, 2014
作者:
A
adlertz
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
8029302: Performance regression in Math.pow intrinsic
Summary: Added special case for x^y where y == 2 Reviewed-by: kvn
上级
58699dca
变更
2
隐藏空白更改
内联
并排
Showing
2 changed file
with
52 addition
and
19 deletion
+52
-19
src/cpu/x86/vm/macroAssembler_x86.cpp
src/cpu/x86/vm/macroAssembler_x86.cpp
+2
-0
src/share/vm/opto/library_call.cpp
src/share/vm/opto/library_call.cpp
+50
-19
未找到文件。
src/cpu/x86/vm/macroAssembler_x86.cpp
浏览文件 @
0fd06556
...
@@ -3152,10 +3152,12 @@ void MacroAssembler::fast_pow() {
...
@@ -3152,10 +3152,12 @@ void MacroAssembler::fast_pow() {
// if fast computation is not possible, result is NaN. Requires
// if fast computation is not possible, result is NaN. Requires
// fallback from user of this macro.
// fallback from user of this macro.
// increase precision for intermediate steps of the computation
// increase precision for intermediate steps of the computation
BLOCK_COMMENT
(
"fast_pow {"
);
increase_precision
();
increase_precision
();
fyl2x
();
// Stack: (Y*log2(X)) ...
fyl2x
();
// Stack: (Y*log2(X)) ...
pow_exp_core_encoding
();
// Stack: exp(X) ...
pow_exp_core_encoding
();
// Stack: exp(X) ...
restore_precision
();
restore_precision
();
BLOCK_COMMENT
(
"} fast_pow"
);
}
}
void
MacroAssembler
::
fast_exp
()
{
void
MacroAssembler
::
fast_exp
()
{
...
...
src/share/vm/opto/library_call.cpp
浏览文件 @
0fd06556
...
@@ -216,7 +216,7 @@ class LibraryCallKit : public GraphKit {
...
@@ -216,7 +216,7 @@ class LibraryCallKit : public GraphKit {
bool
inline_math_subtractExactL
(
bool
is_decrement
);
bool
inline_math_subtractExactL
(
bool
is_decrement
);
bool
inline_exp
();
bool
inline_exp
();
bool
inline_pow
();
bool
inline_pow
();
void
finish_pow_exp
(
Node
*
result
,
Node
*
x
,
Node
*
y
,
const
TypeFunc
*
call_type
,
address
funcAddr
,
const
char
*
funcName
);
Node
*
finish_pow_exp
(
Node
*
result
,
Node
*
x
,
Node
*
y
,
const
TypeFunc
*
call_type
,
address
funcAddr
,
const
char
*
funcName
);
bool
inline_min_max
(
vmIntrinsics
::
ID
id
);
bool
inline_min_max
(
vmIntrinsics
::
ID
id
);
Node
*
generate_min_max
(
vmIntrinsics
::
ID
id
,
Node
*
x
,
Node
*
y
);
Node
*
generate_min_max
(
vmIntrinsics
::
ID
id
,
Node
*
x
,
Node
*
y
);
// This returns Type::AnyPtr, RawPtr, or OopPtr.
// This returns Type::AnyPtr, RawPtr, or OopPtr.
...
@@ -1678,7 +1678,7 @@ bool LibraryCallKit::inline_trig(vmIntrinsics::ID id) {
...
@@ -1678,7 +1678,7 @@ bool LibraryCallKit::inline_trig(vmIntrinsics::ID id) {
return
true
;
return
true
;
}
}
void
LibraryCallKit
::
finish_pow_exp
(
Node
*
result
,
Node
*
x
,
Node
*
y
,
const
TypeFunc
*
call_type
,
address
funcAddr
,
const
char
*
funcName
)
{
Node
*
LibraryCallKit
::
finish_pow_exp
(
Node
*
result
,
Node
*
x
,
Node
*
y
,
const
TypeFunc
*
call_type
,
address
funcAddr
,
const
char
*
funcName
)
{
//-------------------
//-------------------
//result=(result.isNaN())? funcAddr():result;
//result=(result.isNaN())? funcAddr():result;
// Check: If isNaN() by checking result!=result? then either trap
// Check: If isNaN() by checking result!=result? then either trap
...
@@ -1694,7 +1694,7 @@ void LibraryCallKit::finish_pow_exp(Node* result, Node* x, Node* y, const TypeFu
...
@@ -1694,7 +1694,7 @@ void LibraryCallKit::finish_pow_exp(Node* result, Node* x, Node* y, const TypeFu
uncommon_trap
(
Deoptimization
::
Reason_intrinsic
,
uncommon_trap
(
Deoptimization
::
Reason_intrinsic
,
Deoptimization
::
Action_make_not_entrant
);
Deoptimization
::
Action_make_not_entrant
);
}
}
set_result
(
result
)
;
return
result
;
}
else
{
}
else
{
// If this inlining ever returned NaN in the past, we compile a call
// If this inlining ever returned NaN in the past, we compile a call
// to the runtime to properly handle corner cases
// to the runtime to properly handle corner cases
...
@@ -1724,9 +1724,10 @@ void LibraryCallKit::finish_pow_exp(Node* result, Node* x, Node* y, const TypeFu
...
@@ -1724,9 +1724,10 @@ void LibraryCallKit::finish_pow_exp(Node* result, Node* x, Node* y, const TypeFu
result_region
->
init_req
(
2
,
control
());
result_region
->
init_req
(
2
,
control
());
result_val
->
init_req
(
2
,
value
);
result_val
->
init_req
(
2
,
value
);
set_result
(
result_region
,
result_val
);
set_control
(
_gvn
.
transform
(
result_region
));
return
result_val
;
}
else
{
}
else
{
set_result
(
result
)
;
return
result
;
}
}
}
}
}
}
...
@@ -1738,7 +1739,8 @@ bool LibraryCallKit::inline_exp() {
...
@@ -1738,7 +1739,8 @@ bool LibraryCallKit::inline_exp() {
Node
*
arg
=
round_double_node
(
argument
(
0
));
Node
*
arg
=
round_double_node
(
argument
(
0
));
Node
*
n
=
_gvn
.
transform
(
new
(
C
)
ExpDNode
(
C
,
control
(),
arg
));
Node
*
n
=
_gvn
.
transform
(
new
(
C
)
ExpDNode
(
C
,
control
(),
arg
));
finish_pow_exp
(
n
,
arg
,
NULL
,
OptoRuntime
::
Math_D_D_Type
(),
CAST_FROM_FN_PTR
(
address
,
SharedRuntime
::
dexp
),
"EXP"
);
n
=
finish_pow_exp
(
n
,
arg
,
NULL
,
OptoRuntime
::
Math_D_D_Type
(),
CAST_FROM_FN_PTR
(
address
,
SharedRuntime
::
dexp
),
"EXP"
);
set_result
(
n
);
C
->
set_has_split_ifs
(
true
);
// Has chance for split-if optimization
C
->
set_has_split_ifs
(
true
);
// Has chance for split-if optimization
return
true
;
return
true
;
...
@@ -1748,27 +1750,48 @@ bool LibraryCallKit::inline_exp() {
...
@@ -1748,27 +1750,48 @@ bool LibraryCallKit::inline_exp() {
// Inline power instructions, if possible.
// Inline power instructions, if possible.
bool
LibraryCallKit
::
inline_pow
()
{
bool
LibraryCallKit
::
inline_pow
()
{
// Pseudocode for pow
// Pseudocode for pow
// if (x <= 0.0) {
// if (y == 2) {
// long longy = (long)y;
// return x * x;
// if ((double)longy == y) { // if y is long
// } else {
// if (y + 1 == y) longy = 0; // huge number: even
// if (x <= 0.0) {
// result = ((1&longy) == 0)?-DPow(abs(x), y):DPow(abs(x), y);
// long longy = (long)y;
// if ((double)longy == y) { // if y is long
// if (y + 1 == y) longy = 0; // huge number: even
// result = ((1&longy) == 0)?-DPow(abs(x), y):DPow(abs(x), y);
// } else {
// result = NaN;
// }
// } else {
// } else {
// result =
NaN
;
// result =
DPow(x,y)
;
// }
// }
// } else {
// if (result != result)? {
// result = DPow(x,y);
// result = uncommon_trap() or runtime_call();
// }
// }
// if (result != result)? {
// return result;
// result = uncommon_trap() or runtime_call();
// }
// }
// return result;
Node
*
x
=
round_double_node
(
argument
(
0
));
Node
*
x
=
round_double_node
(
argument
(
0
));
Node
*
y
=
round_double_node
(
argument
(
2
));
Node
*
y
=
round_double_node
(
argument
(
2
));
Node
*
result
=
NULL
;
Node
*
result
=
NULL
;
Node
*
const_two_node
=
makecon
(
TypeD
::
make
(
2.0
));
Node
*
cmp_node
=
_gvn
.
transform
(
new
(
C
)
CmpDNode
(
y
,
const_two_node
));
Node
*
bool_node
=
_gvn
.
transform
(
new
(
C
)
BoolNode
(
cmp_node
,
BoolTest
::
eq
));
IfNode
*
if_node
=
create_and_xform_if
(
control
(),
bool_node
,
PROB_STATIC_INFREQUENT
,
COUNT_UNKNOWN
);
Node
*
if_true
=
_gvn
.
transform
(
new
(
C
)
IfTrueNode
(
if_node
));
Node
*
if_false
=
_gvn
.
transform
(
new
(
C
)
IfFalseNode
(
if_node
));
RegionNode
*
region_node
=
new
(
C
)
RegionNode
(
3
);
region_node
->
init_req
(
1
,
if_true
);
Node
*
phi_node
=
new
(
C
)
PhiNode
(
region_node
,
Type
::
DOUBLE
);
// special case for x^y where y == 2, we can convert it to x * x
phi_node
->
init_req
(
1
,
_gvn
.
transform
(
new
(
C
)
MulDNode
(
x
,
x
)));
// set control to if_false since we will now process the false branch
set_control
(
if_false
);
if
(
!
too_many_traps
(
Deoptimization
::
Reason_intrinsic
))
{
if
(
!
too_many_traps
(
Deoptimization
::
Reason_intrinsic
))
{
// Short form: skip the fancy tests and just check for NaN result.
// Short form: skip the fancy tests and just check for NaN result.
result
=
_gvn
.
transform
(
new
(
C
)
PowDNode
(
C
,
control
(),
x
,
y
));
result
=
_gvn
.
transform
(
new
(
C
)
PowDNode
(
C
,
control
(),
x
,
y
));
...
@@ -1892,7 +1915,15 @@ bool LibraryCallKit::inline_pow() {
...
@@ -1892,7 +1915,15 @@ bool LibraryCallKit::inline_pow() {
result
=
_gvn
.
transform
(
phi
);
result
=
_gvn
.
transform
(
phi
);
}
}
finish_pow_exp
(
result
,
x
,
y
,
OptoRuntime
::
Math_DD_D_Type
(),
CAST_FROM_FN_PTR
(
address
,
SharedRuntime
::
dpow
),
"POW"
);
result
=
finish_pow_exp
(
result
,
x
,
y
,
OptoRuntime
::
Math_DD_D_Type
(),
CAST_FROM_FN_PTR
(
address
,
SharedRuntime
::
dpow
),
"POW"
);
// control from finish_pow_exp is now input to the region node
region_node
->
set_req
(
2
,
control
());
// the result from finish_pow_exp is now input to the phi node
phi_node
->
init_req
(
2
,
_gvn
.
transform
(
result
));
set_control
(
_gvn
.
transform
(
region_node
));
record_for_igvn
(
region_node
);
set_result
(
_gvn
.
transform
(
phi_node
));
C
->
set_has_split_ifs
(
true
);
// Has chance for split-if optimization
C
->
set_has_split_ifs
(
true
);
// Has chance for split-if optimization
return
true
;
return
true
;
...
...
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录