fixed a bug of not updating the frame to the new environment when apply a closure
This commit is contained in:
		| @ -305,8 +305,10 @@ Ada.Text_IO.Put_line ("TOO FEW ARGUMETNS FOR COMPARISON"); | |||||||
| 		-- Get_Closure_Code(Func) returns "((x y) (+ x y) (* x y))" | 		-- Get_Closure_Code(Func) returns "((x y) (+ x y) (* x y))" | ||||||
|  |  | ||||||
| 		-- Create a new environment for the closure | 		-- Create a new environment for the closure | ||||||
| 		--Interp.Environment := Make_Environment(Interp.Self, Get_Closure_Environment(Func)); |  | ||||||
| 		Envir := Make_Environment(Interp.Self, Get_Closure_Environment(Func)); | 		Envir := Make_Environment(Interp.Self, Get_Closure_Environment(Func)); | ||||||
|  | 		-- Update the environment of the frame to the one created above | ||||||
|  | 		-- so as to put the arguments into the new environment. | ||||||
|  | 		Set_Frame_Environment (Interp.Stack, Envir); | ||||||
|  |  | ||||||
| 		Fbody := Get_Closure_Code(Func); | 		Fbody := Get_Closure_Code(Func); | ||||||
| 		pragma Assert (Is_Cons(Fbody)); -- the lambda evaluator must ensure this. | 		pragma Assert (Is_Cons(Fbody)); -- the lambda evaluator must ensure this. | ||||||
| @ -325,7 +327,7 @@ Ada.Text_IO.Put_line ("TOO FEW ARGUMETNS FOR COMPARISON"); | |||||||
| 		else | 		else | ||||||
| 			while Is_Cons(Formal) loop | 			while Is_Cons(Formal) loop | ||||||
| 				if not Is_Cons(Actual) then | 				if not Is_Cons(Actual) then | ||||||
| 					Ada.Text_IO.Put_Line (">>>> Too few arguments for CLOSURE <<<<");	 | 					Ada.Text_IO.Put_Line (">>>> TOO FEW ARGUMENTS FOR CLOSURE <<<<");	 | ||||||
| 					raise Evaluation_Error; | 					raise Evaluation_Error; | ||||||
| 				end if; | 				end if; | ||||||
|  |  | ||||||
| @ -353,15 +355,10 @@ Ada.Text_IO.Put_line ("TOO FEW ARGUMETNS FOR COMPARISON"); | |||||||
| 			end if; | 			end if; | ||||||
| 		end if; | 		end if; | ||||||
| 			 | 			 | ||||||
| -- TODO: is it correct to keep the environement in the frame? |  | ||||||
| 		Set_Frame_Opcode (Interp.Stack, Opcode_Evaluate_Group); | 		Set_Frame_Opcode (Interp.Stack, Opcode_Evaluate_Group); | ||||||
| 		Set_Frame_Operand (Interp.Stack, Fbody); | 		Set_Frame_Operand (Interp.Stack, Fbody); | ||||||
| 		Clear_Frame_Result (Interp.Stack); | 		Clear_Frame_Result (Interp.Stack); | ||||||
|  |  | ||||||
| 		-- Update the environment of the frame so as to perform |  | ||||||
| 		-- body evaluation in the new environment. |  | ||||||
| 		Set_Frame_Environment (Interp.Stack, Envir); |  | ||||||
|  |  | ||||||
| 		Pop_Tops (Interp, 4); | 		Pop_Tops (Interp, 4); | ||||||
| 	end Apply_Closure; | 	end Apply_Closure; | ||||||
|  |  | ||||||
|  | |||||||
| @ -214,7 +214,6 @@ Ada.Text_IO.Put_Line ("NO ALTERNATE"); | |||||||
| 		declare | 		declare | ||||||
| 			Closure: Object_Pointer; | 			Closure: Object_Pointer; | ||||||
| 		begin | 		begin | ||||||
| 			--Closure := Make_Closure(Interp.Self, Operand, Interp.Environment); |  | ||||||
| 			Closure := Make_Closure(Interp.Self, Operand, Get_Frame_Environment(Interp.Stack)); | 			Closure := Make_Closure(Interp.Self, Operand, Get_Frame_Environment(Interp.Stack)); | ||||||
| 			Pop_Frame (Interp);  -- Done | 			Pop_Frame (Interp);  -- Done | ||||||
| 			Chain_Frame_Result (Interp, Interp.Stack, Closure); | 			Chain_Frame_Result (Interp, Interp.Stack, Closure); | ||||||
| @ -350,6 +349,27 @@ Ada.Text_IO.Put_Line ("NO ALTERNATE"); | |||||||
| 		end if; | 		end if; | ||||||
| 	end Evaluate_Letast_Syntax; | 	end Evaluate_Letast_Syntax; | ||||||
|  |  | ||||||
|  | 	procedure Evaluate_Letrec_Syntax is | ||||||
|  | 		pragma Inline (Evaluate_Letrec_Syntax); | ||||||
|  | 		Envir: Object_Pointer; | ||||||
|  | 	begin | ||||||
|  | 		Check_Let_Syntax; | ||||||
|  | 		-- Car: <bindings>, Cdr: <body> | ||||||
|  |  | ||||||
|  | ada.text_io.put_line ("XXXXX <<< LETREC IMPLEMENTATION NEEDED >>XXXXXXXXXXXXXXXXXXXXXXXXXXX"); | ||||||
|  | 		--Set_Frame_Opcode (Interp.Stack, Opcode_Let_Finish); | ||||||
|  | 		--Set_Frame_Operand (Interp.Stack, Cdr);  | ||||||
|  |  | ||||||
|  | 		-- Push a new environment to the current frame. | ||||||
|  | 		--Envir := Make_Environment(Interp.Self, Get_Frame_Environment(Interp.Stack)); | ||||||
|  | 		--Set_Frame_Environment (Interp.Stack, Envir); | ||||||
|  |  | ||||||
|  | 		--if Car /= Nil_Pointer then | ||||||
|  | 			-- <bindings> is not empty | ||||||
|  | 		--	Push_Frame (Interp, Opcode_Letrec_Binding, Car); | ||||||
|  | 		--end if; | ||||||
|  | 	end Evaluate_Letrec_Syntax; | ||||||
|  |  | ||||||
| 	procedure Evaluate_Quote_Syntax is | 	procedure Evaluate_Quote_Syntax is | ||||||
| 		pragma Inline (Evaluate_Quote_Syntax); | 		pragma Inline (Evaluate_Quote_Syntax); | ||||||
| 	begin | 	begin | ||||||
| @ -483,6 +503,9 @@ begin | |||||||
| 					when Letast_Syntax => | 					when Letast_Syntax => | ||||||
| 						Evaluate_Letast_Syntax; | 						Evaluate_Letast_Syntax; | ||||||
|  |  | ||||||
|  | 					when Letrec_Syntax => | ||||||
|  | 						Evaluate_Letrec_Syntax; | ||||||
|  |  | ||||||
| 					when Or_Syntax => | 					when Or_Syntax => | ||||||
| 						Evaluate_Or_Syntax; | 						Evaluate_Or_Syntax; | ||||||
|  |  | ||||||
|  | |||||||
| @ -2000,78 +2000,53 @@ end if; | |||||||
| 		Ada.Text_IO.New_Line; | 		Ada.Text_IO.New_Line; | ||||||
| 	end Print; | 	end Print; | ||||||
|  |  | ||||||
| 	function Pointer_To_Opcode (Pointer: in Object_Pointer) return Opcode_Type is | 	function Pointer_To_Opcode (Pointer: in Object_Pointer) return Opcode_Type renames Pointer_To_Integer; | ||||||
| 		pragma Inline (Pointer_To_Opcode); | 	function Opcode_To_Pointer (Opcode: in Opcode_Type) return Object_Pointer renames Integer_To_Pointer; | ||||||
| 	begin	 |  | ||||||
| 		return Pointer_To_Integer(Pointer); |  | ||||||
| 	end Pointer_To_Opcode; |  | ||||||
|  |  | ||||||
| 	function Opcode_To_Pointer (Opcode: in Opcode_Type) return Object_Pointer is |  | ||||||
| 		pragma Inline (Opcode_To_Pointer); |  | ||||||
| 	begin	 |  | ||||||
| 		return Integer_To_Pointer(Opcode); |  | ||||||
| 	end Opcode_To_Pointer; |  | ||||||
|  |  | ||||||
| 	procedure Push_Frame (Interp:  in out Interpreter_Record; | 	procedure Push_Frame (Interp:  in out Interpreter_Record; | ||||||
| 	                      Opcode:  in     Opcode_Type;  | 	                      Opcode:  in     Opcode_Type;  | ||||||
| 	                      Operand: in     Object_Pointer) is | 	                      Operand: in     Object_Pointer) is | ||||||
| 		pragma Inline (Push_Frame); | 		pragma Inline (Push_Frame); | ||||||
| 	begin | 	begin | ||||||
| 		--Interp.Stack := Make_Frame(Interp.Self, Interp.Stack, Opcode_To_Pointer(Opcode), Operand, Interp.Environment); |  | ||||||
| 		Interp.Stack := Make_Frame(Interp.Self, Interp.Stack, Opcode_To_Pointer(Opcode), Operand, Get_Frame_Environment(Interp.Stack)); | 		Interp.Stack := Make_Frame(Interp.Self, Interp.Stack, Opcode_To_Pointer(Opcode), Operand, Get_Frame_Environment(Interp.Stack)); | ||||||
| 	end Push_Frame; | 	end Push_Frame; | ||||||
|  |  | ||||||
| 	--procedure Pop_Frame (Interp.Stack: out Object_Pointer; |  | ||||||
| 	--                     Opcode:  out Opcode_Type; |  | ||||||
| 	--                     Operand: out Object_Pointer) is |  | ||||||
| 	--	pragma Inline (Pop_Frame); |  | ||||||
| 	--begin |  | ||||||
| 	--	pragma Assert (Interp.Stack /= Nil_Pointer); |  | ||||||
| 	--	Opcode := Pointer_To_Opcode(Interp.Stack.Pointer_Slot(Frame_Opcode_Index)); |  | ||||||
| 	--	Operand := Interp.Stack.Pointer_Slot(Frame_Operand_Index); |  | ||||||
| 	--	Interp.Stack := Interp.Stack.Pointer_Slot(Frame_Stack_Index); -- pop  |  | ||||||
| 	--end Pop_Frame; |  | ||||||
|  |  | ||||||
| 	procedure Pop_Frame (Interp: in out Interpreter_Record) is | 	procedure Pop_Frame (Interp: in out Interpreter_Record) is | ||||||
| 		pragma Inline (Pop_Frame); | 		pragma Inline (Pop_Frame); | ||||||
| 	begin | 	begin | ||||||
|  | 		pragma Assert (Interp.Stack /= Interp.Root_Frame); | ||||||
| 		pragma Assert (Interp.Stack /= Nil_Pointer); | 		pragma Assert (Interp.Stack /= Nil_Pointer); | ||||||
| 		Interp.Stack := Interp.Stack.Pointer_Slot(Frame_Stack_Index); -- pop  | 		Interp.Stack := Interp.Stack.Pointer_Slot(Frame_Stack_Index); -- pop  | ||||||
| 	end Pop_Frame; | 	end Pop_Frame; | ||||||
|  |  | ||||||
| 	procedure Execute (Interp: in out Interpreter_Record) is separate; | 	procedure Execute (Interp: in out Interpreter_Record) is separate; | ||||||
|  |  | ||||||
|  |  | ||||||
| 	procedure Evaluate (Interp: in out Interpreter_Record; | 	procedure Evaluate (Interp: in out Interpreter_Record; | ||||||
| 	                    Source: in     Object_Pointer; | 	                    Source: in     Object_Pointer; | ||||||
| 	                    Result: out    Object_Pointer) is | 	                    Result: out    Object_Pointer) is | ||||||
| 	begin | 	begin | ||||||
| 		-- Push a pseudo-frame to terminate the evaluation loop | 		Result := Nil_Pointer; | ||||||
| 		--pragma Assert (Interp.Stack = Nil_Pointer); |  | ||||||
| 		--Interp.Stack := Nil_Pointer; |  | ||||||
| 		--Push_Frame (Interp, Opcode_Exit, Nil_Pointer); |  | ||||||
| 		pragma Assert (Interp.Stack = Interp.Root_Frame); |  | ||||||
| 		pragma Assert (Get_Frame_Result(Interp.Stack) = Nil_Pointer); |  | ||||||
|  |  | ||||||
| 		-- Push the actual frame for evaluation | 		-- Perform some clean ups in case the procedure is called | ||||||
|  | 		-- again after an exception is raised | ||||||
|  | 		Clear_Tops (Interp); | ||||||
|  | 		Interp.Stack := Interp.Root_Frame; | ||||||
|  | 		Clear_Frame_Result (Interp.Stack); | ||||||
|  |  | ||||||
|  | 		-- Push an actual frame for evaluation | ||||||
| 		Push_Frame (Interp, Opcode_Evaluate_Object, Source); | 		Push_Frame (Interp, Opcode_Evaluate_Object, Source); | ||||||
|  |  | ||||||
| 		Execute (Interp); | 		Execute (Interp); | ||||||
|  |  | ||||||
|  | 		pragma Assert (Interp.Stack = Interp.Root_Frame); | ||||||
| 		pragma Assert (Get_Frame_Opcode(Interp.Stack) = Opcode_Exit); | 		pragma Assert (Get_Frame_Opcode(Interp.Stack) = Opcode_Exit); | ||||||
|  |  | ||||||
| 		Result := Get_Frame_Result (Interp.Stack); | 		Result := Get_Frame_Result(Interp.Stack); | ||||||
| 		-- There must be only 1 value chained to the top-level frame | 		-- There must be only 1 value chained to the top-level frame | ||||||
| 		-- once evaluation is over. | 		-- once evaluation is over. | ||||||
| 		pragma Assert (Get_Cdr(Result) = Nil_Pointer); | 		pragma Assert (Get_Cdr(Result) = Nil_Pointer); | ||||||
| 		-- Get the only value chained  | 		Result := Get_Car(Result); -- Get the only value chained  | ||||||
| 		Result := Get_Car(Result); | 		Clear_Frame_Result (Interp.Stack);  | ||||||
|  |  | ||||||
| 		--Pop_Frame (Interp); |  | ||||||
| 		--pragma Assert (Interp.Stack = Nil_Pointer); |  | ||||||
|  |  | ||||||
| 		pragma Assert (Interp.Stack = Interp.Root_Frame); |  | ||||||
| 		Clear_Frame_Result (Interp.Stack); |  | ||||||
| 	end Evaluate; | 	end Evaluate; | ||||||
|  |  | ||||||
| 	procedure Run_Loop (Interp: in out Interpreter_Record; | 	procedure Run_Loop (Interp: in out Interpreter_Record; | ||||||
| @ -2081,36 +2056,35 @@ end if; | |||||||
| 		pragma Assert (Interp.Base_Input.Stream /= null); | 		pragma Assert (Interp.Base_Input.Stream /= null); | ||||||
|  |  | ||||||
| --DEBUG_GC := Standard.True; | --DEBUG_GC := Standard.True; | ||||||
| 		Clear_Tops (Interp); |  | ||||||
| 		Result := Nil_Pointer; | 		Result := Nil_Pointer; | ||||||
|  |  | ||||||
|  | 		-- Perform some clean ups in case the procedure is called | ||||||
|  | 		-- again after an exception is raised | ||||||
|  | 		Clear_Tops (Interp); | ||||||
|  | 		Interp.Stack := Interp.Root_Frame; | ||||||
|  | 		Clear_Frame_Result (Interp.Stack); | ||||||
|  |  | ||||||
| 		loop | 		loop | ||||||
| 			--pragma Assert (Interp.Stack = Nil_Pointer); |  | ||||||
| 			--Interp.Stack := Nil_Pointer; |  | ||||||
| 			--Push_Frame (Interp, Opcode_Exit, Nil_Pointer); |  | ||||||
| 			pragma Assert (Interp.Stack = Interp.Root_Frame); | 			pragma Assert (Interp.Stack = Interp.Root_Frame); | ||||||
| 			pragma Assert (Get_Frame_Result(Interp.Stack) = Nil_Pointer); | 			pragma Assert (Get_Frame_Result(Interp.Stack) = Nil_Pointer); | ||||||
|  |  | ||||||
|  |  | ||||||
| 			--Push_Frame (Interp, Opcode_Print_Result, Nil_Pointer); | 			--Push_Frame (Interp, Opcode_Print_Result, Nil_Pointer); | ||||||
| 			Push_Frame (Interp, Opcode_Evaluate_Result, Nil_Pointer); | 			Push_Frame (Interp, Opcode_Evaluate_Result, Nil_Pointer); | ||||||
| 			Push_Frame (Interp, Opcode_Read_Object, Nil_Pointer); | 			Push_Frame (Interp, Opcode_Read_Object, Nil_Pointer); | ||||||
|  |  | ||||||
| 			Execute (Interp); | 			Execute (Interp); | ||||||
|  |  | ||||||
|  | 			pragma Assert (Interp.Stack = Interp.Root_Frame); | ||||||
| 			pragma Assert (Get_Frame_Opcode(Interp.Stack) = Opcode_Exit); | 			pragma Assert (Get_Frame_Opcode(Interp.Stack) = Opcode_Exit); | ||||||
|  |  | ||||||
| 			-- TODO: this result must be kept at some where that GC dowsn't sweep. | 			-- TODO: this result must be kept at some where that GC dowsn't sweep. | ||||||
| 			Result := Get_Frame_Result (Interp.Stack);  | 			Result := Get_Frame_Result(Interp.Stack);  | ||||||
| 			pragma Assert (Get_Cdr(Result) = Nil_Pointer); | 			pragma Assert (Get_Cdr(Result) = Nil_Pointer); | ||||||
| 			Result := Get_Car(Result); | 			Result := Get_Car(Result); | ||||||
|  | 			Clear_Frame_Result (Interp.Stack); | ||||||
|  |  | ||||||
| 			--Pop_Frame (Interp); |  | ||||||
| Ada.Text_IO.Put ("RESULT>>>>>"); | Ada.Text_IO.Put ("RESULT>>>>>"); | ||||||
| Print (Interp, Result); | Print (Interp, Result); | ||||||
| 			--pragma Assert (Interp.Stack = Nil_Pointer); |  | ||||||
| 			pragma Assert (Interp.Stack = Interp.Root_Frame); |  | ||||||
| 			Clear_Frame_Result (Interp.Stack); |  | ||||||
| Ada.Text_IO.Put_Line (">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> LOOP ITERATION XXXXXX CHECKPOINT");  | Ada.Text_IO.Put_Line (">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> LOOP ITERATION XXXXXX CHECKPOINT");  | ||||||
| 		end loop; | 		end loop; | ||||||
|  |  | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user