[erl-frontend] Move back unary/binary operator matching into translation

Summary: There are multiple operators in Erlang that require special treatment and cannot be simply mapped to a SIL operator (for example andalso, orelse, exactly equal, xor). Therefore it is not practical to have a helper function mapping these operators to SIL operators.

Reviewed By: skcho

Differential Revision: D29814373

fbshipit-source-id: 892219fd3
master
Akos Hajdu 4 years ago committed by Facebook GitHub Bot
parent 320c82d9ad
commit 68216ac1a1

@ -91,44 +91,6 @@ let update_location line env =
{env with location}
let sil_binary_of_erlang_binary (op : Ast.binary_operator) : Binop.t option =
match op with
| Add ->
Some (PlusA None)
| And ->
Some LAnd
| AtLeast ->
Some Ge
| AtMost ->
Some Le
(* TODO: do we want to handle Equal and ExactlyEqual differently? *)
| Equal | ExactlyEqual ->
Some Eq
(* TODO: do we want to handle NotEqual and ExactlyNotEqual differently? *)
| ExactlyNotEqual | NotEqual ->
Some Ne
| Greater ->
Some Gt
| IDiv ->
Some Div
| Less ->
Some Lt
| Mul ->
Some (Mult None)
| Or ->
Some LOr
| Rem ->
Some Mod (* TODO: check semantics of Rem vs Mod *)
| Sub ->
Some (MinusA None)
| _ ->
None
let sil_unary_of_erlang_unary (op : Ast.unary_operator) : Unop.t option =
match op with UMinus -> Some Neg | UNot -> Some LNot | _ -> None
(** Groups several helpers used to create nodes. *)
module Node = struct
let make (env : (Procdesc.t present, _) environment) kind instructions =
@ -396,14 +358,43 @@ and translate_expression env {Ast.line; simple_expression} =
let id2 = Ident.create_fresh Ident.knormal in
let block1 = translate_expression {env with result= Present (Exp.Var id1)} e1 in
let block2 = translate_expression {env with result= Present (Exp.Var id2)} e2 in
let make_simple_op_block sil_op =
Block.make_load env ret_var (Exp.BinOp (sil_op, Var id1, Var id2)) any
in
let op_block =
match sil_binary_of_erlang_binary op with
| Some sil_op ->
Block.make_load env ret_var (Exp.BinOp (sil_op, Var id1, Var id2)) any
| None ->
match op with
| Add ->
make_simple_op_block (PlusA None)
| And ->
make_simple_op_block LAnd
| AtLeast ->
make_simple_op_block Ge
| AtMost ->
make_simple_op_block Le
(* TODO: proper modeling of equal vs exactly equal T95767672 *)
| Equal | ExactlyEqual ->
make_simple_op_block Eq
(* TODO: proper modeling of not equal vs exactly not equal T95767672 *)
| ExactlyNotEqual | NotEqual ->
make_simple_op_block Ne
| Greater ->
make_simple_op_block Gt
| IDiv ->
make_simple_op_block Div
| Less ->
make_simple_op_block Lt
| Mul ->
make_simple_op_block (Mult None)
| Or ->
make_simple_op_block LOr
| Rem ->
make_simple_op_block Mod (* TODO: check semantics of Rem vs Mod *)
| Sub ->
make_simple_op_block (MinusA None)
| todo ->
L.debug Capture Verbose
"@[todo ErlangTranslator.translate_expression(BinaryOperator) %s@."
(Sexp.to_string (Ast.sexp_of_binary_operator op)) ;
(Sexp.to_string (Ast.sexp_of_binary_operator todo)) ;
Block.make_success env
in
Block.all env [block1; block2; op_block]
@ -495,14 +486,19 @@ and translate_expression env {Ast.line; simple_expression} =
| UnaryOperator (op, e) ->
let id = Ident.create_fresh Ident.knormal in
let block = translate_expression {env with result= Present (Exp.Var id)} e in
let make_simple_op_block sil_op =
Block.make_load env ret_var (Exp.UnOp (sil_op, Var id, None)) any
in
let op_block =
match sil_unary_of_erlang_unary op with
| Some sil_op ->
Block.make_load env ret_var (Exp.UnOp (sil_op, Var id, None)) any
| None ->
match op with
| UMinus ->
make_simple_op_block Neg
| UNot ->
make_simple_op_block LNot
| todo ->
L.debug Capture Verbose
"@[todo ErlangTranslator.translate_expression(UnaryOperator) %s@."
(Sexp.to_string (Ast.sexp_of_unary_operator op)) ;
(Sexp.to_string (Ast.sexp_of_unary_operator todo)) ;
Block.make_success env
in
Block.all env [block; op_block]

Loading…
Cancel
Save