# HG changeset patch
# User Mario de Sousa <msousa@fe.up.pt>
# Date 1334583667 -3600
# Node ID 9317e04c1dde7b9a62b6c490881dd61dcc92849d
# Parent  35d391c38a30bffa89b39adbb07ffd3c3ddbc9d6
Fixing check for assignment to output variables.

diff -r 35d391c38a30 -r 9317e04c1dde stage3/lvalue_check.cc
--- a/stage3/lvalue_check.cc	Sun Apr 15 19:56:33 2012 +0100
+++ b/stage3/lvalue_check.cc	Mon Apr 16 14:41:07 2012 +0100
@@ -44,7 +44,7 @@
                                                  LAST_(symbol1,symbol2) ->last_line,  LAST_(symbol1,symbol2) ->last_column);\
     fprintf(stderr, __VA_ARGS__);                                                                                           \
     fprintf(stderr, "\n");                                                                                                  \
-    error_found = true;                                                                                                     \
+    error_count++;                                                                                                     \
   }                                                                                                                         \
 }
 
@@ -60,14 +60,14 @@
 
 
 lvalue_check_c::lvalue_check_c(symbol_c *ignore) {
-	error_found = false;
+	error_count = 0;
 }
 
 lvalue_check_c::~lvalue_check_c(void) {
 }
 
-int lvalue_check_c::get_error_found() {
-	return error_found;
+int lvalue_check_c::get_error_count() {
+	return error_count;
 }
 
 
@@ -84,18 +84,47 @@
 }
 
 
-/* fb_instance.var := ...  is not valid if var is output (not input ??) variable */
-void lvalue_check_c::check_assignment_to_output(symbol_c * lvalue) {
-	symbol_c *type_id = search_varfb_instance_type->get_basetype_id(lvalue);
-	if (NULL != type_id) {
-		function_block_declaration_c *fb_decl = function_block_type_symtable.find_value(type_id);
-		if (function_block_type_symtable.end_value() != fb_decl) {
-			search_var_instance_decl_c   search_var_instance_decl(fb_decl);
-			structured_variable_c * str_var = (structured_variable_c *)lvalue;
-			unsigned int vartype = search_var_instance_decl.get_vartype(str_var->field_selector);
-			if (vartype == search_var_instance_decl_c::output_vt)
-				STAGE3_ERROR(0, lvalue, lvalue, "Assignment to FB output field variable is not be allowed.");
+/* fb_instance.var := ...  is not valid if var is output variable */
+/* NOTE, if a fb_instance1.fb_instance2.fb_instance3.var is used, we must iteratively check that none of the 
+ *       FB records are declared as OUTPUT variables!!  
+ *       This is the reason why we have the while() loop in this function!
+ */
+void lvalue_check_c::check_assignment_to_output(symbol_c *lvalue) {
+	decompose_var_instance_name_c decompose_lvalue(lvalue);
+	search_base_type_c            search_base_type;
+
+	symbol_c *struct_elem = decompose_lvalue.next_part();
+	symbol_c *type_decl   = search_var_instance_decl->get_decl(struct_elem);
+	// symbol_c *type_id  = spec_init_sperator_c::get_spec(type_decl); /* this is not required! search_base_type_c can handle spec_init symbols! */
+	symbol_c *basetype_id = search_base_type.get_basetype_id(/*type_id*/ type_decl);
+	/* If we can not determine the data type of the element, then the code must have a data type semantic error.
+	 * This will have been caught by the data type semantic verifier, so we do not bother with this anymore!
+	 */
+	if (NULL == basetype_id) return;
+
+	/* Determine if the record/structure element is of a FB type. */
+	/* NOTE: If the structure element is not a FB type, then we can quit this check.
+	 *       Remember that the standard does not allow a STRUCT data type to have elements that are FB instances!
+	 *       Similarly, arrays of FB instances is also not allowed.
+	 *       So, as soon as we find one record/structure element that is not a FB, no other record/structure element
+	 *       will be of FB type, which means we can quit this check!
+	 */
+	function_block_declaration_c *fb_decl = function_block_type_symtable.find_value(basetype_id);
+	if (function_block_type_symtable.end_value() == fb_decl) return;
+
+	while (NULL != (struct_elem = decompose_lvalue.next_part())) {
+		search_var_instance_decl_c   fb_search_var_instance_decl(fb_decl);
+		if (search_var_instance_decl_c::output_vt == fb_search_var_instance_decl.get_vartype(struct_elem)) {
+			STAGE3_ERROR(0, struct_elem, struct_elem, "Assignment to FB output variable is not allowed.");
+			return; /* no need to carry on checking once the first error is found! */
 		}
+
+		/* prepare for any possible further record/structure elements */
+		type_decl   = fb_search_var_instance_decl.get_decl(struct_elem);
+		basetype_id = search_base_type.get_basetype_id(type_decl);
+		if (NULL == basetype_id) return; /* same comment as above... */
+		fb_decl = function_block_type_symtable.find_value(basetype_id);
+		if (function_block_type_symtable.end_value() == fb_decl) return; /* same comment as above... */
 	}
 }
 
@@ -113,17 +142,17 @@
 void lvalue_check_c::check_assignment_to_expression(symbol_c *lvalue) {
 	/* TODO: check whether the lvalue is an expresion! */
 	/* This may occur in function invocations, when passing values (possibly an expression) to one 
-	 * of the function's OUTPUT parameters.
+	 * of the function's OUTPUT or IN_OUT parameters.
 	 */
 }
 
 
 
 void lvalue_check_c::verify_is_lvalue(symbol_c *lvalue) {
+	check_assignment_to_expression(lvalue);
 	check_assignment_to_controlvar(lvalue);
 	check_assignment_to_output(lvalue);
 	check_assignment_to_constant(lvalue);
-	check_assignment_to_expression(lvalue);
 }
 
 
@@ -161,8 +190,7 @@
 		/* We only check if 'call_param_value' is a valid lvalue if the value is being passed
 		 * to a valid paramater of the function being called, and that parameter is either OUT or IN_OUT.
 		 */
-		if ((param_name != NULL) && 
-		    ((function_param_iterator_c::direction_out == param_direction) || (function_param_iterator_c::direction_inout == param_direction))) {
+		if ((param_name != NULL) && ((function_param_iterator_c::direction_out == param_direction) || (function_param_iterator_c::direction_inout == param_direction))) {
 			verify_is_lvalue(call_param_value);
 		}
 	}
@@ -227,10 +255,8 @@
 /***********************/
 // SYM_REF3(function_invocation_c, function_name, formal_param_list, nonformal_param_list, symbol_c *called_function_declaration; int extensible_param_count; std::vector <symbol_c *> candidate_functions;)
 void *lvalue_check_c::visit(function_invocation_c *symbol) {
-	if (NULL != symbol->formal_param_list)
-		check_formal_call(symbol, symbol->called_function_declaration);
-	if (NULL != symbol->nonformal_param_list)
-		check_nonformal_call(symbol, symbol->called_function_declaration);
+	if (NULL != symbol->formal_param_list   )  check_formal_call   (symbol, symbol->called_function_declaration);
+	if (NULL != symbol->nonformal_param_list)  check_nonformal_call(symbol, symbol->called_function_declaration);
 	return NULL;
 }
 
diff -r 35d391c38a30 -r 9317e04c1dde stage3/lvalue_check.hh
--- a/stage3/lvalue_check.hh	Sun Apr 15 19:56:33 2012 +0100
+++ b/stage3/lvalue_check.hh	Mon Apr 16 14:41:07 2012 +0100
@@ -47,7 +47,7 @@
     search_varfb_instance_type_c *search_varfb_instance_type;
     search_var_instance_decl_c *search_var_instance_decl;
     search_base_type_c search_base_type;
-    int error_found;
+    int error_count;
     int current_display_error_level;
     std::vector <symbol_c *> control_variables;
 
@@ -63,7 +63,7 @@
   public:
     lvalue_check_c(symbol_c *ignore);
     virtual ~lvalue_check_c(void);
-    int get_error_found();
+    int get_error_count();
 
     /**************************************/
     /* B 1.5 - Program organisation units */
diff -r 35d391c38a30 -r 9317e04c1dde stage3/print_datatypes_error.cc
--- a/stage3/print_datatypes_error.cc	Sun Apr 15 19:56:33 2012 +0100
+++ b/stage3/print_datatypes_error.cc	Mon Apr 16 14:41:07 2012 +0100
@@ -61,7 +61,7 @@
     fprintf(stderr, __VA_ARGS__);                                                                                           \
     fprintf(stderr, "\n");                                                                                                  \
     il_error = true;                                                                                                        \
-    error_found = true;                                                                                                     \
+    error_count++;                                                                                                     \
   }                                                                                                                         \
 }  
 
@@ -80,7 +80,7 @@
 static int debug = 0;
 
 print_datatypes_error_c::print_datatypes_error_c(symbol_c *ignore) {
-	error_found = false;
+	error_count = 0;
 	warning_found = false;
 	current_display_error_level = error_level_default;
 }
@@ -88,8 +88,8 @@
 print_datatypes_error_c::~print_datatypes_error_c(void) {
 }
 
-int print_datatypes_error_c::get_error_found() {
-	return error_found;
+int print_datatypes_error_c::get_error_count() {
+	return error_count;
 }
 
 
diff -r 35d391c38a30 -r 9317e04c1dde stage3/print_datatypes_error.hh
--- a/stage3/print_datatypes_error.hh	Sun Apr 15 19:56:33 2012 +0100
+++ b/stage3/print_datatypes_error.hh	Mon Apr 16 14:41:07 2012 +0100
@@ -73,7 +73,7 @@
      */
     int  il_parenthesis_level;
     bool il_error;
-    bool error_found;
+    int  error_count;
     bool warning_found;
 
     /* the current data type of the data stored in the IL stack, i.e. the default variable */
@@ -99,7 +99,7 @@
   public:
     print_datatypes_error_c(symbol_c *ignore);
     virtual ~print_datatypes_error_c(void);
-    int get_error_found();
+    int get_error_count();
 
 
     /*********************/
diff -r 35d391c38a30 -r 9317e04c1dde stage3/stage3.cc
--- a/stage3/stage3.cc	Sun Apr 15 19:56:33 2012 +0100
+++ b/stage3/stage3.cc	Mon Apr 16 14:41:07 2012 +0100
@@ -56,18 +56,29 @@
 	tree_root->accept(narrow_candidate_datatypes);
 	print_datatypes_error_c print_datatypes_error(tree_root);
 	tree_root->accept(print_datatypes_error);
-	if (print_datatypes_error.get_error_found())
-		return -1;
-	lvalue_check_c lvalue_check(tree_root);
-	tree_root->accept(lvalue_check);
-	if (lvalue_check.get_error_found())
-		return -1;
-	return 0;
+	return print_datatypes_error.get_error_count();
 }
 
 
+/* Type safety analysis assumes that flow control analysis has already been completed,
+ * so be sure to call flow_control_analysis() before calling this function
+ */
+static int lvalue_check(symbol_c *tree_root){
+	lvalue_check_c lvalue_check(tree_root);
+	tree_root->accept(lvalue_check);
+	return lvalue_check.get_error_count();
+}
+
 
 int stage3(symbol_c *tree_root){
-	flow_control_analysis(tree_root);
-	return type_safety(tree_root);
+	int error_count = 0;
+	error_count += flow_control_analysis(tree_root);
+	error_count += type_safety(tree_root);
+	error_count += lvalue_check(tree_root);
+	
+	if (error_count > 0) {
+		fprintf(stderr, "%d errors found. Bailing out!\n", error_count); 
+		return -1;
+	}
+	return 0;
 }