private meth use_if_true_variable_ctx(e: PExpr)
do
var ctx = e.if_true_variable_ctx
- if ctx != null then
- variable_ctx = ctx
- end
+ if ctx != null then variable_ctx = ctx
+ end
+
+ # Make the if_false_variable_ctx of the expression effective
+ private meth use_if_false_variable_ctx(e: PExpr)
+ do
+ var ctx = e.if_false_variable_ctx
+ if ctx != null then variable_ctx = ctx
end
# Number of nested once
# The variable type information if current boolean expression is true
readable private attr _if_true_variable_ctx: VariableContext
+
+ # The variable type information if current boolean expression is false
+ readable private attr _if_false_variable_ctx: VariableContext
end
redef class AVardeclExpr
v.visit(n_expr)
v.check_conform_expr(n_expr, v.type_bool)
+ # Prepare 'then' context
v.use_if_true_variable_ctx(n_expr)
- v.variable_ctx = v.variable_ctx.sub(n_then)
- v.visit(n_then)
+ # Process the 'then'
+ if n_then != null then
+ v.variable_ctx = v.variable_ctx.sub(n_then)
+ v.visit(n_then)
+ end
- if n_else == null then
- # Restore variable ctx since the 'then' block is optional
- v.variable_ctx = old_var_ctx
- else
- # Remember what appened in the 'then'
- var then_var_ctx = v.variable_ctx
- # Reset to process the 'else'
- v.variable_ctx = old_var_ctx.sub(n_else)
+ # Remember what appened in the 'then'
+ var then_var_ctx = v.variable_ctx
+
+ # Prepare 'else' context
+ v.variable_ctx = old_var_ctx
+ v.use_if_false_variable_ctx(n_expr)
+
+ # Process the 'else'
+ if n_else != null then
+ v.variable_ctx = v.variable_ctx.sub(n_else)
v.visit(n_else)
- # Merge then and else in the old control_flow
- old_var_ctx.merge2(then_var_ctx, v.variable_ctx)
- v.variable_ctx = old_var_ctx
end
+
+ # Merge 'then' and 'else' contexts
+ old_var_ctx.merge2(then_var_ctx, v.variable_ctx)
+ v.variable_ctx = old_var_ctx
_is_typed = true
end
end
v.use_if_true_variable_ctx(n_expr)
v.visit(n_then)
v.variable_ctx = old_var_ctx
+ v.use_if_false_variable_ctx(n_expr)
v.visit(n_else)
v.check_conform_expr(n_expr, v.type_bool)
redef class AOrExpr
redef meth after_typing(v)
do
+ var old_var_ctx = v.variable_ctx
+
+ v.visit(n_expr)
+ v.use_if_false_variable_ctx(n_expr)
+
+ v.visit(n_expr2)
+ if n_expr2.if_false_variable_ctx != null then
+ _if_false_variable_ctx = n_expr2.if_false_variable_ctx
+ else
+ _if_false_variable_ctx = v.variable_ctx
+ end
+
+ v.variable_ctx = old_var_ctx
+
v.check_conform_expr(n_expr, v.type_bool)
v.check_conform_expr(n_expr2, v.type_bool)
_stype = v.type_bool
redef meth after_typing(v)
do
v.check_conform_expr(n_expr, v.type_bool)
+
+ # Invert if_true/if_false information
+ _if_false_variable_ctx = n_expr._if_true_variable_ctx
+ _if_true_variable_ctx = n_expr._if_false_variable_ctx
+
_stype = v.type_bool
_is_typed = true
end
--- /dev/null
+# This file is part of NIT ( http://www.nitlanguage.org ).
+#
+# Copyright 2009 Jean Privat <jean@pryen.org>
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+import kernel
+
+class A
+ init do end
+end
+
+class B
+special A
+ meth foo(i: Int) do i.output
+ init do end
+end
+
+var a: A = new B
+
+if a isa B then
+ a.foo(1)
+else
+ #alt1#a.foo(-1)
+end
+#alt2#a.foo(-2)
+
+if not a isa B then
+ #alt3#a.foo(-3)
+else
+ a.foo(2)
+end
+#alt4#a.foo(-4)
+
+if a isa B and false then
+ a.foo(3)
+else
+ #alt5#a.foo(-5)
+end
+#alt6#a.foo(-6)
+
+if not a isa B or true then
+ #alt7#a.foo(-7)
+else
+ a.foo(3)
+end
+#alt8#a.foo(-8)
+
+
+