aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorshyouhei <shyouhei@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2018-09-13 03:46:46 +0000
committershyouhei <shyouhei@b2dd03c8-39d4-4d8f-98ff-823fe69b080e>2018-09-13 03:46:46 +0000
commit6925c9c6f5e435fc88e683d7b9df475dd46c28ce (patch)
treea9c1c737ae7f45f8db587888debf990b2ef12613
parent63130b2348440c731d20f9488f922d53abe16b6e (diff)
downloadruby-6925c9c6f5e435fc88e683d7b9df475dd46c28ce.tar.gz
move canary-related statements into macros
This is mostly cosmetic. Should generate a slightly readable vm.inc output. git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@64709 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
-rwxr-xr-xtool/ruby_vm/models/bare_instructions.rb16
-rw-r--r--tool/ruby_vm/views/_insn_entry.erb24
-rw-r--r--vm_insnhelper.h21
3 files changed, 38 insertions, 23 deletions
diff --git a/tool/ruby_vm/models/bare_instructions.rb b/tool/ruby_vm/models/bare_instructions.rb
index 30a6e0ff88..ed3d886014 100755
--- a/tool/ruby_vm/models/bare_instructions.rb
+++ b/tool/ruby_vm/models/bare_instructions.rb
@@ -109,8 +109,20 @@ class RubyVM::BareInstructions
@attrs.fetch('leaf').expr.expr == 'true;'
end
- def complicated_return_values?
- @sig[:ret].any? {|i| i == '...' }
+ def handle_canary stmt
+ # Stack canary is basically a good thing that we want to add, however:
+ #
+ # - When the instruction returns variadic number of return values,
+ # it is not easy to tell where is the stack top. We can't but
+ # skip it.
+ #
+ # - When the instruction body is empty (like putobject), we can
+ # say for 100% sure that canary is a waste of time.
+ #
+ # So we skip canary for those cases.
+ return '' if @sig[:ret].any? {|i| i == '...' }
+ return '' if @expr.blank?
+ return " #{stmt};\n"
end
def inspect
diff --git a/tool/ruby_vm/views/_insn_entry.erb b/tool/ruby_vm/views/_insn_entry.erb
index 90a489f624..d3c7b63582 100644
--- a/tool/ruby_vm/views/_insn_entry.erb
+++ b/tool/ruby_vm/views/_insn_entry.erb
@@ -13,16 +13,11 @@ INSN_ENTRY(<%= insn.name %>)
%# NAME_OF_CURRENT_INSN is used in vm_exec.h
# define NAME_OF_CURRENT_INSN <%= insn.name %>
# define INSN_ATTR(x) <%= insn.call_attribute(' ## x ## ') %>
-% unless insn.complicated_return_values?
-# if VM_CHECK_MODE > 0
- bool leaf;
- VALUE *canary;
-# endif
-% end
% unless insn.declarations.empty?
<%= insn.declarations.join(";\n ") %>;
% end
+<%= insn.handle_canary "DECLARE_CANARY" -%>
START_OF_ORIGINAL_INSN(<%= insn.name %>);
% insn.preamble.each do |konst|
<%= render_c_expr konst -%>
@@ -39,27 +34,14 @@ INSN_ENTRY(<%= insn.name %>)
% if insn.handles_sp?
POPN(INSN_ATTR(popn));
% end
-% unless insn.complicated_return_values?
-#if VM_CHECK_MODE > 0
- if ((leaf = INSN_ATTR(leaf))) {
- canary = STACK_ADDR_FROM_TOP(-1);
- *canary = vm_stack_canary;
- }
-#endif
-% end
+<%= insn.handle_canary "SETUP_CANARY()" -%>
COLLECT_USAGE_INSN(INSN_ATTR(bin));
% insn.opes.each_with_index do |ope, i|
COLLECT_USAGE_OPERAND(INSN_ATTR(bin), <%= i %>, <%= ope[:name] %>);
% end
<%= render_c_expr insn.expr -%>
CHECK_VM_STACK_OVERFLOW_FOR_INSN(VM_REG_CFP, INSN_ATTR(retn));
-% unless insn.complicated_return_values?
-#if VM_CHECK_MODE > 0
- if (leaf && (*canary != vm_stack_canary)) {
- vm_canary_is_found_dead(INSN_ATTR(bin), *canary);
- }
-#endif
-% end
+<%= insn.handle_canary "CHECK_CANARY()" -%>
% if insn.handles_sp?
% insn.rets.reverse_each do |ret|
PUSH(<%= insn.cast_to_VALUE ret %>);
diff --git a/vm_insnhelper.h b/vm_insnhelper.h
index e65e889898..b5f35df2bd 100644
--- a/vm_insnhelper.h
+++ b/vm_insnhelper.h
@@ -201,6 +201,27 @@ enum vm_regan_acttype {
/**********************************************************/
+/* deal with stack canary */
+/**********************************************************/
+
+#if VM_CHECK_MODE > 0
+#define DECLARE_CANARY bool leaf; VALUE *canary
+#define SETUP_CANARY() \
+ if ((leaf = INSN_ATTR(leaf))) { \
+ canary = GET_SP(); \
+ SET_SV(vm_stack_canary); \
+ }
+#define CHECK_CANARY() \
+ if (leaf && (*canary != vm_stack_canary)) { \
+ vm_canary_is_found_dead(INSN_ATTR(bin), *canary); \
+ }
+#else
+#define DECLARE_CANARY /* void */
+#define SETUP_CANARY() /* void */
+#define CHECK_CANARY() /* void */
+#endif
+
+/**********************************************************/
/* others */
/**********************************************************/