|  | @@ -1074,8 +1074,100 @@ bool check_builtin_simd_operation(CheckerContext *c, Operand *operand, Ast *call
 | 
											
												
													
														|  |  	return false;
 |  |  	return false;
 | 
											
												
													
														|  |  }
 |  |  }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | 
 |  | +LoadDirectiveResult check_load_directive(CheckerContext *c, Operand *operand, Ast *call, Type *type_hint, bool err_on_not_found) {
 | 
											
												
													
														|  | 
 |  | +	ast_node(ce, CallExpr, call);
 | 
											
												
													
														|  | 
 |  | +	ast_node(bd, BasicDirective, ce->proc);
 | 
											
												
													
														|  | 
 |  | +	String name = bd->name.string;
 | 
											
												
													
														|  | 
 |  | +	GB_ASSERT(name == "load");
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	if (ce->args.count != 1) {
 | 
											
												
													
														|  | 
 |  | +		if (ce->args.count == 0) {
 | 
											
												
													
														|  | 
 |  | +			error(ce->close, "'#load' expects 1 argument, got 0");
 | 
											
												
													
														|  | 
 |  | +		} else {
 | 
											
												
													
														|  | 
 |  | +			error(ce->args[0], "'#load' expects 1 argument, got %td", ce->args.count);
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +		return LoadDirective_Error;
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	Ast *arg = ce->args[0];
 | 
											
												
													
														|  | 
 |  | +	Operand o = {};
 | 
											
												
													
														|  | 
 |  | +	check_expr(c, &o, arg);
 | 
											
												
													
														|  | 
 |  | +	if (o.mode != Addressing_Constant) {
 | 
											
												
													
														|  | 
 |  | +		error(arg, "'#load' expected a constant string argument");
 | 
											
												
													
														|  | 
 |  | +		return LoadDirective_Error;
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	if (!is_type_string(o.type)) {
 | 
											
												
													
														|  | 
 |  | +		gbString str = type_to_string(o.type);
 | 
											
												
													
														|  | 
 |  | +		error(arg, "'#load' expected a constant string, got %s", str);
 | 
											
												
													
														|  | 
 |  | +		gb_string_free(str);
 | 
											
												
													
														|  | 
 |  | +		return LoadDirective_Error;
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	gbAllocator a = heap_allocator();
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	GB_ASSERT(o.value.kind == ExactValue_String);
 | 
											
												
													
														|  | 
 |  | +	String base_dir = dir_from_path(get_file_path_string(bd->token.pos.file_id));
 | 
											
												
													
														|  | 
 |  | +	String original_string = o.value.value_string;
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	BlockingMutex *ignore_mutex = nullptr;
 | 
											
												
													
														|  | 
 |  | +	String path = {};
 | 
											
												
													
														|  | 
 |  | +	bool ok = determine_path_from_string(ignore_mutex, call, base_dir, original_string, &path);
 | 
											
												
													
														|  | 
 |  | +	gb_unused(ok);
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	char *c_str = alloc_cstring(a, path);
 | 
											
												
													
														|  | 
 |  | +	defer (gb_free(a, c_str));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	gbFile f = {};
 | 
											
												
													
														|  | 
 |  | +	gbFileError file_err = gb_file_open(&f, c_str);
 | 
											
												
													
														|  | 
 |  | +	defer (gb_file_close(&f));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	switch (file_err) {
 | 
											
												
													
														|  | 
 |  | +	default:
 | 
											
												
													
														|  | 
 |  | +	case gbFileError_Invalid:
 | 
											
												
													
														|  | 
 |  | +		if (err_on_not_found) {
 | 
											
												
													
														|  | 
 |  | +			error(ce->proc, "Failed to `#load` file: %s; invalid file or cannot be found", c_str);
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +		call->state_flags |= StateFlag_DirectiveWasFalse;
 | 
											
												
													
														|  | 
 |  | +		return LoadDirective_NotFound;
 | 
											
												
													
														|  | 
 |  | +	case gbFileError_NotExists:
 | 
											
												
													
														|  | 
 |  | +		if (err_on_not_found) {
 | 
											
												
													
														|  | 
 |  | +			error(ce->proc, "Failed to `#load` file: %s; file cannot be found", c_str);
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +		call->state_flags |= StateFlag_DirectiveWasFalse;
 | 
											
												
													
														|  | 
 |  | +		return LoadDirective_NotFound;
 | 
											
												
													
														|  | 
 |  | +	case gbFileError_Permission:
 | 
											
												
													
														|  | 
 |  | +		if (err_on_not_found) {
 | 
											
												
													
														|  | 
 |  | +			error(ce->proc, "Failed to `#load` file: %s; file permissions problem", c_str);
 | 
											
												
													
														|  | 
 |  | +		}
 | 
											
												
													
														|  | 
 |  | +		call->state_flags |= StateFlag_DirectiveWasFalse;
 | 
											
												
													
														|  | 
 |  | +		return LoadDirective_NotFound;
 | 
											
												
													
														|  | 
 |  | +	case gbFileError_None:
 | 
											
												
													
														|  | 
 |  | +		// Okay
 | 
											
												
													
														|  | 
 |  | +		break;
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -bool check_builtin_procedure_directive(CheckerContext *c, Operand *operand, Ast *call, i32 id, Type *type_hint) {
 |  | 
 | 
											
												
													
														|  | 
 |  | +	String result = {};
 | 
											
												
													
														|  | 
 |  | +	isize file_size = cast(isize)gb_file_size(&f);
 | 
											
												
													
														|  | 
 |  | +	if (file_size > 0) {
 | 
											
												
													
														|  | 
 |  | +		u8 *data = cast(u8 *)gb_alloc(a, file_size+1);
 | 
											
												
													
														|  | 
 |  | +		gb_file_read_at(&f, data, file_size, 0);
 | 
											
												
													
														|  | 
 |  | +		data[file_size] = '\0';
 | 
											
												
													
														|  | 
 |  | +		result.text = data;
 | 
											
												
													
														|  | 
 |  | +		result.len = file_size;
 | 
											
												
													
														|  | 
 |  | +	}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +	operand->type = t_u8_slice;
 | 
											
												
													
														|  | 
 |  | +	operand->mode = Addressing_Constant;
 | 
											
												
													
														|  | 
 |  | +	operand->value = exact_value_string(result);
 | 
											
												
													
														|  | 
 |  | +	return LoadDirective_Success;
 | 
											
												
													
														|  | 
 |  | +}
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +bool check_builtin_procedure_directive(CheckerContext *c, Operand *operand, Ast *call, Type *type_hint) {
 | 
											
												
													
														|  |  	ast_node(ce, CallExpr, call);
 |  |  	ast_node(ce, CallExpr, call);
 | 
											
												
													
														|  |  	ast_node(bd, BasicDirective, ce->proc);
 |  |  	ast_node(bd, BasicDirective, ce->proc);
 | 
											
												
													
														|  |  	String name = bd->name.string;
 |  |  	String name = bd->name.string;
 | 
											
										
											
												
													
														|  | @@ -1100,81 +1192,7 @@ bool check_builtin_procedure_directive(CheckerContext *c, Operand *operand, Ast
 | 
											
												
													
														|  |  		operand->type = t_source_code_location;
 |  |  		operand->type = t_source_code_location;
 | 
											
												
													
														|  |  		operand->mode = Addressing_Value;
 |  |  		operand->mode = Addressing_Value;
 | 
											
												
													
														|  |  	} else if (name == "load") {
 |  |  	} else if (name == "load") {
 | 
											
												
													
														|  | -		if (ce->args.count != 1) {
 |  | 
 | 
											
												
													
														|  | -			if (ce->args.count == 0) {
 |  | 
 | 
											
												
													
														|  | -				error(ce->close, "'#load' expects 1 argument, got 0");
 |  | 
 | 
											
												
													
														|  | -			} else {
 |  | 
 | 
											
												
													
														|  | -				error(ce->args[0], "'#load' expects 1 argument, got %td", ce->args.count);
 |  | 
 | 
											
												
													
														|  | -			}
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -			return false;
 |  | 
 | 
											
												
													
														|  | -		}
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		Ast *arg = ce->args[0];
 |  | 
 | 
											
												
													
														|  | -		Operand o = {};
 |  | 
 | 
											
												
													
														|  | -		check_expr(c, &o, arg);
 |  | 
 | 
											
												
													
														|  | -		if (o.mode != Addressing_Constant) {
 |  | 
 | 
											
												
													
														|  | -			error(arg, "'#load' expected a constant string argument");
 |  | 
 | 
											
												
													
														|  | -			return false;
 |  | 
 | 
											
												
													
														|  | -		}
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		if (!is_type_string(o.type)) {
 |  | 
 | 
											
												
													
														|  | -			gbString str = type_to_string(o.type);
 |  | 
 | 
											
												
													
														|  | -			error(arg, "'#load' expected a constant string, got %s", str);
 |  | 
 | 
											
												
													
														|  | -			gb_string_free(str);
 |  | 
 | 
											
												
													
														|  | -			return false;
 |  | 
 | 
											
												
													
														|  | -		}
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		gbAllocator a = heap_allocator();
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		GB_ASSERT(o.value.kind == ExactValue_String);
 |  | 
 | 
											
												
													
														|  | -		String base_dir = dir_from_path(get_file_path_string(bd->token.pos.file_id));
 |  | 
 | 
											
												
													
														|  | -		String original_string = o.value.value_string;
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		BlockingMutex *ignore_mutex = nullptr;
 |  | 
 | 
											
												
													
														|  | -		String path = {};
 |  | 
 | 
											
												
													
														|  | -		bool ok = determine_path_from_string(ignore_mutex, call, base_dir, original_string, &path);
 |  | 
 | 
											
												
													
														|  | -		gb_unused(ok);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		char *c_str = alloc_cstring(a, path);
 |  | 
 | 
											
												
													
														|  | -		defer (gb_free(a, c_str));
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		gbFile f = {};
 |  | 
 | 
											
												
													
														|  | -		gbFileError file_err = gb_file_open(&f, c_str);
 |  | 
 | 
											
												
													
														|  | -		defer (gb_file_close(&f));
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		switch (file_err) {
 |  | 
 | 
											
												
													
														|  | -		default:
 |  | 
 | 
											
												
													
														|  | -		case gbFileError_Invalid:
 |  | 
 | 
											
												
													
														|  | -			error(ce->proc, "Failed to `#load` file: %s; invalid file or cannot be found", c_str);
 |  | 
 | 
											
												
													
														|  | -			return false;
 |  | 
 | 
											
												
													
														|  | -		case gbFileError_NotExists:
 |  | 
 | 
											
												
													
														|  | -			error(ce->proc, "Failed to `#load` file: %s; file cannot be found", c_str);
 |  | 
 | 
											
												
													
														|  | -			return false;
 |  | 
 | 
											
												
													
														|  | -		case gbFileError_Permission:
 |  | 
 | 
											
												
													
														|  | -			error(ce->proc, "Failed to `#load` file: %s; file permissions problem", c_str);
 |  | 
 | 
											
												
													
														|  | -			return false;
 |  | 
 | 
											
												
													
														|  | -		case gbFileError_None:
 |  | 
 | 
											
												
													
														|  | -			// Okay
 |  | 
 | 
											
												
													
														|  | -			break;
 |  | 
 | 
											
												
													
														|  | -		}
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		String result = {};
 |  | 
 | 
											
												
													
														|  | -		isize file_size = cast(isize)gb_file_size(&f);
 |  | 
 | 
											
												
													
														|  | -		if (file_size > 0) {
 |  | 
 | 
											
												
													
														|  | -			u8 *data = cast(u8 *)gb_alloc(a, file_size+1);
 |  | 
 | 
											
												
													
														|  | -			gb_file_read_at(&f, data, file_size, 0);
 |  | 
 | 
											
												
													
														|  | -			data[file_size] = '\0';
 |  | 
 | 
											
												
													
														|  | -			result.text = data;
 |  | 
 | 
											
												
													
														|  | -			result.len = file_size;
 |  | 
 | 
											
												
													
														|  | -		}
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -		operand->type = t_u8_slice;
 |  | 
 | 
											
												
													
														|  | -		operand->mode = Addressing_Constant;
 |  | 
 | 
											
												
													
														|  | -		operand->value = exact_value_string(result);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | 
 |  | +		return check_load_directive(c, operand, call, type_hint, true) == LoadDirective_Success;
 | 
											
												
													
														|  |  	} else if (name == "load_hash") {
 |  |  	} else if (name == "load_hash") {
 | 
											
												
													
														|  |  		if (ce->args.count != 2) {
 |  |  		if (ce->args.count != 2) {
 | 
											
												
													
														|  |  			if (ce->args.count == 0) {
 |  |  			if (ce->args.count == 0) {
 | 
											
										
											
												
													
														|  | @@ -1263,7 +1281,6 @@ bool check_builtin_procedure_directive(CheckerContext *c, Operand *operand, Ast
 | 
											
												
													
														|  |  		char *c_str = alloc_cstring(a, path);
 |  |  		char *c_str = alloc_cstring(a, path);
 | 
											
												
													
														|  |  		defer (gb_free(a, c_str));
 |  |  		defer (gb_free(a, c_str));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  |  		gbFile f = {};
 |  |  		gbFile f = {};
 | 
											
												
													
														|  |  		gbFileError file_err = gb_file_open(&f, c_str);
 |  |  		gbFileError file_err = gb_file_open(&f, c_str);
 | 
											
												
													
														|  |  		defer (gb_file_close(&f));
 |  |  		defer (gb_file_close(&f));
 | 
											
										
											
												
													
														|  | @@ -1321,6 +1338,8 @@ bool check_builtin_procedure_directive(CheckerContext *c, Operand *operand, Ast
 | 
											
												
													
														|  |  		operand->value = exact_value_u64(hash_value);
 |  |  		operand->value = exact_value_u64(hash_value);
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  	} else if (name == "load_or") {
 |  |  	} else if (name == "load_or") {
 | 
											
												
													
														|  | 
 |  | +		warning(call, "'#load_or' is deprecated in favour of '#load(path) or_else default'");
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  |  		if (ce->args.count != 2) {
 |  |  		if (ce->args.count != 2) {
 | 
											
												
													
														|  |  			if (ce->args.count == 0) {
 |  |  			if (ce->args.count == 0) {
 | 
											
												
													
														|  |  				error(ce->close, "'#load_or' expects 2 arguments, got 0");
 |  |  				error(ce->close, "'#load_or' expects 2 arguments, got 0");
 | 
											
										
											
												
													
														|  | @@ -1640,7 +1659,7 @@ bool check_builtin_procedure(CheckerContext *c, Operand *operand, Ast *call, i32
 | 
											
												
													
														|  |  		break;
 |  |  		break;
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  	case BuiltinProc_DIRECTIVE:
 |  |  	case BuiltinProc_DIRECTIVE:
 | 
											
												
													
														|  | -		return check_builtin_procedure_directive(c, operand, call, id, type_hint);
 |  | 
 | 
											
												
													
														|  | 
 |  | +		return check_builtin_procedure_directive(c, operand, call, type_hint);
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |  	case BuiltinProc_len:
 |  |  	case BuiltinProc_len:
 | 
											
												
													
														|  |  		check_expr_or_type(c, operand, ce->args[0]);
 |  |  		check_expr_or_type(c, operand, ce->args[0]);
 |