diff options
author | Tristan Gingold <tgingold@free.fr> | 2014-01-27 20:24:45 +0100 |
---|---|---|
committer | Tristan Gingold <tgingold@free.fr> | 2014-01-27 20:24:45 +0100 |
commit | 51d115c72b13507fa3e182f387651dc4aff98b5f (patch) | |
tree | 1e3eae1b98eb12767dea00eee6d066fcd8f47f59 /ortho/oread | |
parent | f74185d729e80fb2073375a6c4c39081209e914f (diff) | |
download | ghdl-51d115c72b13507fa3e182f387651dc4aff98b5f.tar.gz ghdl-51d115c72b13507fa3e182f387651dc4aff98b5f.tar.bz2 ghdl-51d115c72b13507fa3e182f387651dc4aff98b5f.zip |
oread: add a little bit of type inference to simplify .on files.
Diffstat (limited to 'ortho/oread')
-rw-r--r-- | ortho/oread/ortho_front.adb | 238 |
1 files changed, 124 insertions, 114 deletions
diff --git a/ortho/oread/ortho_front.adb b/ortho/oread/ortho_front.adb index 2b82fd801..626241a70 100644 --- a/ortho/oread/ortho_front.adb +++ b/ortho/oread/ortho_front.adb @@ -1201,8 +1201,9 @@ package body Ortho_Front is -- procedure Parse_Declaration; - function Parse_Unary_Expression (Atype : Node_Acc) return O_Enode; - function Parse_Expression (Expr_Type : Node_Acc) return O_Enode; + procedure Parse_Expression (Expr_Type : Node_Acc; + Expr : out O_Enode; + Res_Type : out Node_Acc); procedure Parse_Name (Prefix : Node_Acc; Name : out O_Lnode; N_Type : out Node_Acc); procedure Parse_Lvalue (N : in out O_Lnode; N_Type : in out Node_Acc); @@ -1282,6 +1283,7 @@ package body Ortho_Front is return Res; end Parse_Alignof; + -- Parse a literal whose type is ATYPE. function Parse_Typed_Literal (Atype : Node_Acc) return O_Cnode is Res : O_Cnode; @@ -1395,12 +1397,12 @@ package body Ortho_Front is end Parse_Typed_Literal; -- expect: next token - function Parse_Named_Expression - (Atype : Node_Acc; Name : Node_Acc; Stop_At_All : Boolean) - return O_Enode + -- Parse an expression starting with NAME. + procedure Parse_Named_Expression + (Atype : Node_Acc; Name : Node_Acc; Stop_At_All : Boolean; + Res : out O_Enode; + Res_Type : out Node_Acc) is - Res : O_Enode; - R_Type : Node_Acc; begin if Tok = Tok_Tick then Next_Token; @@ -1408,55 +1410,59 @@ package body Ortho_Front is -- Typed literal. Next_Token; Res := New_Lit (Parse_Typed_Literal (Name.Decl_Dtype)); + Res_Type := Name.Decl_Dtype; Expect (Tok_Right_Brack); Next_Token; - return Res; elsif Tok = Tok_Left_Paren then - -- Typed expression. + -- Typed expression (used for comparaison operators) Next_Token; - Res := Parse_Expression (Name.Decl_Dtype); + Parse_Expression (Name.Decl_Dtype, Res, Res_Type); Expect (Tok_Right_Paren); Next_Token; - return Res; elsif Tok = Tok_Ident then -- Attribute. if Token_Sym = Id_Conv then Next_Expect (Tok_Left_Paren); Next_Token; - Res := Parse_Expression (null); + Parse_Expression (null, Res, Res_Type); + -- Discard Res_Type. Expect (Tok_Right_Paren); Next_Token; - R_Type := Name.Decl_Dtype; - Res := New_Convert_Ov (Res, R_Type.Type_Onode); + Res_Type := Name.Decl_Dtype; + Res := New_Convert_Ov (Res, Res_Type.Type_Onode); -- Fall-through. elsif Token_Sym = Id_Address or Token_Sym = Id_Unchecked_Address or Token_Sym = Id_Subprg_Addr then - R_Type := Name.Decl_Dtype; + Res_Type := Name.Decl_Dtype; Res := Parse_Address (Name); -- Fall-through. elsif Token_Sym = Id_Sizeof then - Res := New_Lit (Parse_Sizeof (Name.Decl_Dtype)); + Res_Type := Name.Decl_Dtype; + Res := New_Lit (Parse_Sizeof (Res_Type)); Next_Token; - return Res; + return; elsif Token_Sym = Id_Alignof then - Res := New_Lit (Parse_Alignof (Name.Decl_Dtype)); + Res_Type := Name.Decl_Dtype; + Res := New_Lit (Parse_Alignof (Res_Type)); Next_Token; - return Res; + return; elsif Token_Sym = Id_Alloca then Next_Expect (Tok_Left_Paren); Next_Token; - Res := New_Alloca - (Name.Decl_Dtype.Type_Onode, - Parse_Expression (null)); + Parse_Expression (null, Res, Res_Type); + -- Discard Res_Type. + Res_Type := Name.Decl_Dtype; + Res := New_Alloca (Res_Type.Type_Onode, Res); Expect (Tok_Right_Paren); Next_Token; - return Res; + return; elsif Token_Sym = Id_Offsetof then - Res := New_Lit (Parse_Offsetof (Atype)); + Res_Type := Atype; + Res := New_Lit (Parse_Offsetof (Res_Type)); Next_Token; - return Res; + return; else Parse_Error ("unknown attribute name"); end if; @@ -1473,7 +1479,7 @@ package body Ortho_Front is begin Parse_Association (Constr, Name); Res := New_Function_Call (Constr); - R_Type := Name.Decl_Dtype; + Res_Type := Name.Decl_Dtype; -- Fall-through. end; elsif Name.Kind = Node_Object @@ -1482,10 +1488,9 @@ package body Ortho_Front is -- Name. declare Lval : O_Lnode; - L_Type : Node_Acc; begin - Parse_Name (Name, Lval, L_Type); - return New_Value (Lval); + Parse_Name (Name, Lval, Res_Type); + Res := New_Value (Lval); end; else Parse_Error ("bad ident expression: " @@ -1496,11 +1501,11 @@ package body Ortho_Front is -- R_TYPE and RES must be set. if Tok = Tok_Dot then if Stop_At_All then - return Res; + return; end if; Next_Token; if Tok = Tok_All then - if R_Type.Kind /= Type_Access then + if Res_Type.Kind /= Type_Access then Parse_Error ("type of prefix is not an access"); end if; declare @@ -1508,101 +1513,69 @@ package body Ortho_Front is begin Next_Token; N := New_Access_Element (Res); - R_Type := R_Type.Access_Dtype; - Parse_Lvalue (N, R_Type); + Res_Type := Res_Type.Access_Dtype; + Parse_Lvalue (N, Res_Type); Res := New_Value (N); end; - return Res; + return; else Parse_Error ("'.all' expected"); end if; - else - return Res; end if; end Parse_Named_Expression; - function Parse_Primary_Expression (Atype : Node_Acc) return O_Enode + procedure Parse_Primary_Expression (Atype : Node_Acc; + Res : out O_Enode; + Res_Type : out Node_Acc) is - Res : O_Enode; begin case Tok is when Tok_Num | Tok_Float_Num => - return New_Lit (Parse_Typed_Literal (Atype)); + if Atype = null then + Parse_Error ("numeric literal without type context"); + end if; + Res_Type := Atype; + Res := New_Lit (Parse_Typed_Literal (Atype)); when Tok_Ident => declare N : Node_Acc; begin N := Get_Decl (Token_Sym); Next_Token; - return Parse_Named_Expression (Atype, N, False); + Parse_Named_Expression (Atype, N, False, Res, Res_Type); end; when Tok_Left_Paren => Next_Token; - Res := Parse_Expression (Atype); + Parse_Expression (Atype, Res, Res_Type); Expect (Tok_Right_Paren); Next_Token; - return Res; --- when Tok_Ident => --- declare --- Inter : Node_Acc; --- begin --- Inter := Token_Sym.Inter; --- while Inter /= null loop --- case Inter.Kind is --- when Inter_Var --- | Inter_Param => --- Res := New_Value (Inter.Object_Node); --- Next_Token; --- return Res; --- when Inter_Subprg => --- return Parse_Function_Call (Inter); --- when Inter_Keyword => --- raise Program_Error; --- end case; --- Inter := Inter.Next; --- end loop; --- Parse_Error ("undefined name " & Get_String (Token_Sym.Ident)); --- return O_Enode_Null; --- end; when others => Parse_Error ("bad primary expression: " & Token_Type'Image (Tok)); - return O_Enode_Null; end case; end Parse_Primary_Expression; - function Parse_Unary_Expression (Atype : Node_Acc) return O_Enode + -- Parse '-' EXPR, 'not' EXPR, 'abs' EXPR or EXPR. + procedure Parse_Unary_Expression (Atype : Node_Acc; + Res : out O_Enode; + Res_Type : out Node_Acc) is - Operand : O_Enode; begin case Tok is when Tok_Minus => Next_Token; - case Tok is --- when Tok_Float_Num => --- Operand := New_Float_Literal (Atype.Type_Onode, --- -Token_Float); --- Next_Token; --- return Operand; --- when Tok_Num => --- Operand := New_Signed_Literal (Atype.Type_Onode, --- -Integer_64 (Token_Number)); --- Next_Token; --- return Operand; - when others => - Operand := Parse_Primary_Expression (Atype); - return New_Monadic_Op (ON_Neg_Ov, Operand); - end case; + Parse_Primary_Expression (Atype, Res, Res_Type); + Res := New_Monadic_Op (ON_Neg_Ov, Res); when Tok_Not => Next_Token; - Operand := Parse_Unary_Expression (Atype); - return New_Monadic_Op (ON_Not, Operand); + Parse_Unary_Expression (Atype, Res, Res_Type); + Res := New_Monadic_Op (ON_Not, Res); when Tok_Abs => Next_Token; - Operand := Parse_Unary_Expression (Atype); - return New_Monadic_Op (ON_Abs_Ov, Operand); + Parse_Unary_Expression (Atype, Res, Res_Type); + Res := New_Monadic_Op (ON_Abs_Ov, Res); when others => - return Parse_Primary_Expression (Atype); + Parse_Primary_Expression (Atype, Res, Res_Type); end case; end Parse_Unary_Expression; @@ -1613,13 +1586,23 @@ package body Ortho_Front is return Op_Ov; end Check_Sharp; - function Parse_Expression (Expr_Type : Node_Acc) return O_Enode + procedure Parse_Expression (Expr_Type : Node_Acc; + Expr : out O_Enode; + Res_Type : out Node_Acc) is + Op_Type : Node_Acc; L : O_Enode; R : O_Enode; Op : ON_Op_Kind; begin - L := Parse_Unary_Expression (Expr_Type); + if Expr_Type = null or else Expr_Type.Kind = Type_Boolean then + -- The type of the expression isn't known, as this can be a + -- comparaison operator. + Op_Type := null; + else + Op_Type := Expr_Type; + end if; + Parse_Unary_Expression (Op_Type, L, Res_Type); case Tok is when Tok_Div => Op := Check_Sharp (ON_Div_Ov); @@ -1658,18 +1641,23 @@ package body Ortho_Front is Next_Token; when others => - return L; + Expr := L; + return; end case; if Op in ON_Compare_Op_Kind then Next_Token; end if; - R := Parse_Unary_Expression (Expr_Type); + Parse_Unary_Expression (Res_Type, R, Res_Type); case Op is when ON_Dyadic_Op_Kind => - return New_Dyadic_Op (Op, L, R); + Expr := New_Dyadic_Op (Op, L, R); when ON_Compare_Op_Kind => - return New_Compare_Op (Op, L, R, Expr_Type.Type_Onode); + if Expr_Type = null then + Parse_Error ("comparaison operator requires a type"); + end if; + Expr := New_Compare_Op (Op, L, R, Expr_Type.Type_Onode); + Res_Type := Expr_Type; when others => raise Program_Error; end case; @@ -1712,6 +1700,7 @@ package body Ortho_Front is declare V : O_Enode; Bt : Node_Acc; + Res_Type : Node_Acc; begin Next_Token; if N_Type.Kind = Type_Subarray then @@ -1722,7 +1711,7 @@ package body Ortho_Front is if Bt.Kind /= Type_Array then Parse_Error ("type of prefix is not an array"); end if; - V := Parse_Expression (Bt.Array_Index); + Parse_Expression (Bt.Array_Index, V, Res_Type); if Tok = Tok_Elipsis then N := New_Slice (N, Bt.Type_Onode, V); Next_Token; @@ -1754,8 +1743,10 @@ package body Ortho_Front is declare Val : O_Enode; begin - Val := Parse_Named_Expression (null, Prefix, True); - N_Type := Prefix.Decl_Dtype; + Parse_Named_Expression (null, Prefix, True, Val, N_Type); + if N_Type /= Prefix.Decl_Dtype then + Parse_Error ("type doesn't match"); + end if; if Tok = Tok_Dot then Next_Token; if Tok = Tok_All then @@ -1783,6 +1774,8 @@ package body Ortho_Front is procedure Parse_Association (Constr : in out O_Assoc_List; Decl : Node_Acc) is Param : Node_Acc; + Expr : O_Enode; + Expr_Type : Node_Acc; begin Start_Association (Constr, Decl.Subprg_Node); if Tok /= Tok_Left_Paren then @@ -1794,7 +1787,8 @@ package body Ortho_Front is if Param = null then Parse_Error ("too many parameters"); end if; - New_Association (Constr, Parse_Expression (Param.Decl_Dtype)); + Parse_Expression (Param.Decl_Dtype, Expr, Expr_Type); + New_Association (Constr, Expr); Param := Param.Param_Next; exit when Tok /= Tok_Comma; Next_Token; @@ -1880,9 +1874,12 @@ package body Ortho_Front is when Tok_If => declare If_Blk : O_If_Block; + Cond : O_Enode; + Cond_Type : Node_Acc; begin Next_Token; - Start_If_Stmt (If_Blk, Parse_Expression (null)); + Parse_Expression (null, Cond, Cond_Type); + Start_If_Stmt (If_Blk, Cond); Expect (Tok_Then); Next_Token; Parse_Statements; @@ -1947,16 +1944,22 @@ package body Ortho_Front is end; when Tok_Return => - Next_Token; - if Tok /= Tok_Semicolon then - New_Return_Stmt (Parse_Expression (Current_Subprg.Decl_Dtype)); + declare + Res : O_Enode; + Res_Type : Node_Acc; + begin + Next_Token; if Tok /= Tok_Semicolon then - Parse_Error ("';' expected at end of return statement"); + Parse_Expression (Current_Subprg.Decl_Dtype, Res, Res_Type); + New_Return_Stmt (Res); + if Tok /= Tok_Semicolon then + Parse_Error ("';' expected at end of return statement"); + end if; + else + New_Return_Stmt; end if; - else - New_Return_Stmt; - end if; - Next_Token; + Next_Token; + end; when Tok_Ident => -- This is either a procedure call or an assignment. @@ -1982,6 +1985,8 @@ package body Ortho_Front is -- An assignment. declare Name : O_Lnode; + Expr : O_Enode; + Expr_Type : Node_Acc; N_Type : Node_Acc; begin Parse_Name (Inter, Name, N_Type); @@ -1989,7 +1994,8 @@ package body Ortho_Front is Parse_Error ("`:=' expected after a variable"); end if; Next_Token; - New_Assign_Stmt (Name, Parse_Expression (N_Type)); + Parse_Expression (N_Type, Expr, Expr_Type); + New_Assign_Stmt (Name, Expr); if Tok /= Tok_Semicolon then Parse_Error ("';' expected at end of assignment"); end if; @@ -2003,9 +2009,12 @@ package body Ortho_Front is declare Case_Blk : O_Case_Block; L : O_Cnode; + Choice : O_Enode; + Choice_Type : Node_Acc; begin Next_Token; - Start_Case_Stmt (Case_Blk, Parse_Expression (null)); + Parse_Expression (null, Choice, Choice_Type); + Start_Case_Stmt (Case_Blk, Choice); Expect (Tok_Is); Next_Token; loop @@ -2367,13 +2376,14 @@ package body Ortho_Front is Start_Record_Aggr (Constr, Atype.Type_Onode); Field := Atype.Record_Union_Fields; while Field /= null loop - Expect (Tok_Dot); - Next_Expect (Tok_Ident); - if Token_Sym /= Field.Field_Ident then - Parse_Error ("bad field name"); + if Tok = Tok_Dot then + Next_Expect (Tok_Ident); + if Token_Sym /= Field.Field_Ident then + Parse_Error ("bad field name"); + end if; + Next_Expect (Tok_Equal); + Next_Token; end if; - Next_Expect (Tok_Equal); - Next_Token; New_Record_Aggr_El (Constr, Parse_Constant_Value (Field.Field_Type)); Field := Field.Field_Next; |