|  | @@ -897,6 +897,34 @@ void check_assignment(CheckerContext *c, Operand *operand, Type *type, String co
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +bool polymorphic_assign_index(Type **gt_, i64 *dst_count, i64 source_count) {
 | 
	
		
			
				|  |  | +	Type *gt = *gt_;
 | 
	
		
			
				|  |  | +	
 | 
	
		
			
				|  |  | +	GB_ASSERT(gt->kind == Type_Generic);
 | 
	
		
			
				|  |  | +	Entity *e = scope_lookup(gt->Generic.scope, gt->Generic.name);
 | 
	
		
			
				|  |  | +	GB_ASSERT(e != nullptr);
 | 
	
		
			
				|  |  | +	if (e->kind == Entity_TypeName) {
 | 
	
		
			
				|  |  | +		*gt_ = nullptr;
 | 
	
		
			
				|  |  | +		*dst_count = source_count;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +		e->kind = Entity_Constant;
 | 
	
		
			
				|  |  | +		e->Constant.value = exact_value_i64(source_count);
 | 
	
		
			
				|  |  | +		e->type = t_untyped_integer;
 | 
	
		
			
				|  |  | +		return true;
 | 
	
		
			
				|  |  | +	} else if (e->kind == Entity_Constant) {
 | 
	
		
			
				|  |  | +		*gt_ = nullptr;
 | 
	
		
			
				|  |  | +		if (e->Constant.value.kind != ExactValue_Integer) {
 | 
	
		
			
				|  |  | +			return false;
 | 
	
		
			
				|  |  | +		}
 | 
	
		
			
				|  |  | +		i64 count = big_int_to_i64(&e->Constant.value.value_integer);
 | 
	
		
			
				|  |  | +		if (count != source_count) {
 | 
	
		
			
				|  |  | +			return false;
 | 
	
		
			
				|  |  | +		}
 | 
	
		
			
				|  |  | +		*dst_count = source_count;
 | 
	
		
			
				|  |  | +		return true;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +	return false;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  bool is_polymorphic_type_assignable(CheckerContext *c, Type *poly, Type *source, bool compound, bool modify_type) {
 | 
	
		
			
				|  |  |  	Operand o = {Addressing_Value};
 | 
	
	
		
			
				|  | @@ -951,28 +979,7 @@ bool is_polymorphic_type_assignable(CheckerContext *c, Type *poly, Type *source,
 | 
	
		
			
				|  |  |  	case Type_Array:
 | 
	
		
			
				|  |  |  		if (source->kind == Type_Array) {
 | 
	
		
			
				|  |  |  			if (poly->Array.generic_count != nullptr) {
 | 
	
		
			
				|  |  | -				Type *gt = poly->Array.generic_count;
 | 
	
		
			
				|  |  | -				GB_ASSERT(gt->kind == Type_Generic);
 | 
	
		
			
				|  |  | -				Entity *e = scope_lookup(gt->Generic.scope, gt->Generic.name);
 | 
	
		
			
				|  |  | -				GB_ASSERT(e != nullptr);
 | 
	
		
			
				|  |  | -				if (e->kind == Entity_TypeName) {
 | 
	
		
			
				|  |  | -					poly->Array.generic_count = nullptr;
 | 
	
		
			
				|  |  | -					poly->Array.count = source->Array.count;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -					e->kind = Entity_Constant;
 | 
	
		
			
				|  |  | -					e->Constant.value = exact_value_i64(source->Array.count);
 | 
	
		
			
				|  |  | -					e->type = t_untyped_integer;
 | 
	
		
			
				|  |  | -				} else if (e->kind == Entity_Constant) {
 | 
	
		
			
				|  |  | -					poly->Array.generic_count = nullptr;
 | 
	
		
			
				|  |  | -					if (e->Constant.value.kind != ExactValue_Integer) {
 | 
	
		
			
				|  |  | -						return false;
 | 
	
		
			
				|  |  | -					}
 | 
	
		
			
				|  |  | -					i64 count = big_int_to_i64(&e->Constant.value.value_integer);
 | 
	
		
			
				|  |  | -					if (count != source->Array.count) {
 | 
	
		
			
				|  |  | -						return false;
 | 
	
		
			
				|  |  | -					}
 | 
	
		
			
				|  |  | -					poly->Array.count = source->Array.count;
 | 
	
		
			
				|  |  | -				} else {
 | 
	
		
			
				|  |  | +				if (!polymorphic_assign_index(&poly->Array.generic_count, &poly->Array.count, source->Array.count)) {
 | 
	
		
			
				|  |  |  					return false;
 | 
	
		
			
				|  |  |  				}
 | 
	
		
			
				|  |  |  			}
 | 
	
	
		
			
				|  | @@ -1169,54 +1176,14 @@ bool is_polymorphic_type_assignable(CheckerContext *c, Type *poly, Type *source,
 | 
	
		
			
				|  |  |  	case Type_Matrix:
 | 
	
		
			
				|  |  |  		if (source->kind == Type_Matrix) {
 | 
	
		
			
				|  |  |  			if (poly->Matrix.generic_row_count != nullptr) {
 | 
	
		
			
				|  |  | -				Type *gt = poly->Matrix.generic_row_count;
 | 
	
		
			
				|  |  | -				GB_ASSERT(gt->kind == Type_Generic);
 | 
	
		
			
				|  |  | -				Entity *e = scope_lookup(gt->Generic.scope, gt->Generic.name);
 | 
	
		
			
				|  |  | -				GB_ASSERT(e != nullptr);
 | 
	
		
			
				|  |  | -				if (e->kind == Entity_TypeName) {
 | 
	
		
			
				|  |  | -					poly->Matrix.generic_row_count = nullptr;
 | 
	
		
			
				|  |  | -					poly->Matrix.row_count = source->Matrix.row_count;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -					e->kind = Entity_Constant;
 | 
	
		
			
				|  |  | -					e->Constant.value = exact_value_i64(source->Matrix.row_count);
 | 
	
		
			
				|  |  | -					e->type = t_untyped_integer;
 | 
	
		
			
				|  |  | -				} else if (e->kind == Entity_Constant) {
 | 
	
		
			
				|  |  | -					poly->Matrix.generic_row_count = nullptr;
 | 
	
		
			
				|  |  | -					if (e->Constant.value.kind != ExactValue_Integer) {
 | 
	
		
			
				|  |  | -						return false;
 | 
	
		
			
				|  |  | -					}
 | 
	
		
			
				|  |  | -					i64 count = big_int_to_i64(&e->Constant.value.value_integer);
 | 
	
		
			
				|  |  | -					if (count != source->Matrix.row_count) {
 | 
	
		
			
				|  |  | -						return false;
 | 
	
		
			
				|  |  | -					}
 | 
	
		
			
				|  |  | -					poly->Matrix.row_count = source->Matrix.row_count;
 | 
	
		
			
				|  |  | -				} else {
 | 
	
		
			
				|  |  | +				poly->Matrix.stride_in_bytes = 0;
 | 
	
		
			
				|  |  | +				if (!polymorphic_assign_index(&poly->Matrix.generic_row_count, &poly->Matrix.row_count, source->Matrix.row_count)) {
 | 
	
		
			
				|  |  |  					return false;
 | 
	
		
			
				|  |  |  				}
 | 
	
		
			
				|  |  |  			}
 | 
	
		
			
				|  |  |  			if (poly->Matrix.generic_column_count != nullptr) {
 | 
	
		
			
				|  |  | -				Type *gt = poly->Matrix.generic_column_count;
 | 
	
		
			
				|  |  | -				GB_ASSERT(gt->kind == Type_Generic);
 | 
	
		
			
				|  |  | -				Entity *e = scope_lookup(gt->Generic.scope, gt->Generic.name);
 | 
	
		
			
				|  |  | -				GB_ASSERT(e != nullptr);
 | 
	
		
			
				|  |  | -				if (e->kind == Entity_TypeName) {
 | 
	
		
			
				|  |  | -					poly->Matrix.generic_column_count = nullptr;
 | 
	
		
			
				|  |  | -					poly->Matrix.column_count = source->Matrix.column_count;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -					e->kind = Entity_Constant;
 | 
	
		
			
				|  |  | -					e->Constant.value = exact_value_i64(source->Matrix.column_count);
 | 
	
		
			
				|  |  | -					e->type = t_untyped_integer;
 | 
	
		
			
				|  |  | -				} else if (e->kind == Entity_Constant) {
 | 
	
		
			
				|  |  | -					poly->Matrix.generic_column_count = nullptr;
 | 
	
		
			
				|  |  | -					if (e->Constant.value.kind != ExactValue_Integer) {
 | 
	
		
			
				|  |  | -						return false;
 | 
	
		
			
				|  |  | -					}
 | 
	
		
			
				|  |  | -					i64 count = big_int_to_i64(&e->Constant.value.value_integer);
 | 
	
		
			
				|  |  | -					if (count != source->Matrix.column_count) {
 | 
	
		
			
				|  |  | -						return false;
 | 
	
		
			
				|  |  | -					}
 | 
	
		
			
				|  |  | -					poly->Matrix.column_count = source->Matrix.column_count;
 | 
	
		
			
				|  |  | -				} else {
 | 
	
		
			
				|  |  | +				poly->Matrix.stride_in_bytes = 0;
 | 
	
		
			
				|  |  | +				if (!polymorphic_assign_index(&poly->Matrix.generic_column_count, &poly->Matrix.column_count, source->Matrix.column_count)) {
 | 
	
		
			
				|  |  |  					return false;
 | 
	
		
			
				|  |  |  				}
 | 
	
		
			
				|  |  |  			}
 |