diff options
Diffstat (limited to 'main/main.cpp')
| -rw-r--r-- | main/main.cpp | 379 | 
1 files changed, 255 insertions, 124 deletions
| diff --git a/main/main.cpp b/main/main.cpp index fc9ec3b2d9..cdaee06135 100644 --- a/main/main.cpp +++ b/main/main.cpp @@ -30,6 +30,7 @@  #include "main.h" +#include "core/crypto/crypto.h"  #include "core/input_map.h"  #include "core/io/file_access_network.h"  #include "core/io/file_access_pack.h" @@ -37,15 +38,12 @@  #include "core/io/image_loader.h"  #include "core/io/ip.h"  #include "core/io/resource_loader.h" -#include "core/io/stream_peer_ssl.h" -#include "core/io/stream_peer_tcp.h"  #include "core/message_queue.h"  #include "core/os/dir_access.h"  #include "core/os/os.h"  #include "core/project_settings.h"  #include "core/register_core_types.h"  #include "core/script_debugger_local.h" -#include "core/script_debugger_remote.h"  #include "core/script_language.h"  #include "core/translation.h"  #include "core/version.h" @@ -60,12 +58,14 @@  #include "main/tests/test_main.h"  #include "modules/register_module_types.h"  #include "platform/register_platform_apis.h" +#include "scene/debugger/script_debugger_remote.h"  #include "scene/main/scene_tree.h"  #include "scene/main/viewport.h"  #include "scene/register_scene_types.h"  #include "scene/resources/packed_scene.h"  #include "servers/arvr_server.h"  #include "servers/audio_server.h" +#include "servers/camera_server.h"  #include "servers/physics_2d_server.h"  #include "servers/physics_server.h"  #include "servers/register_server_types.h" @@ -98,6 +98,7 @@ static MessageQueue *message_queue = NULL;  // Initialized in setup2()  static AudioServer *audio_server = NULL; +static CameraServer *camera_server = NULL;  static ARVRServer *arvr_server = NULL;  static PhysicsServer *physics_server = NULL;  static Physics2DServer *physics_2d_server = NULL; @@ -161,7 +162,7 @@ static String unescape_cmdline(const String &p_str) {  static String get_full_version_string() {  	String hash = String(VERSION_HASH);  	if (hash.length() != 0) -		hash = "." + hash.left(7); +		hash = "." + hash.left(9);  	return String(VERSION_FULL_BUILD) + hash;  } @@ -204,7 +205,8 @@ void finalize_physics() {  void Main::print_help(const char *p_binary) { -	print_line(String(VERSION_NAME) + " v" + get_full_version_string() + " - https://godotengine.org"); +	print_line(String(VERSION_NAME) + " v" + get_full_version_string() + " - " + String(VERSION_WEBSITE)); +	OS::get_singleton()->print("Free and open source software under the terms of the MIT license.\n");  	OS::get_singleton()->print("(c) 2007-2019 Juan Linietsky, Ariel Manzur.\n");  	OS::get_singleton()->print("(c) 2014-2019 Godot Engine contributors.\n");  	OS::get_singleton()->print("\n"); @@ -247,6 +249,7 @@ void Main::print_help(const char *p_binary) {  	OS::get_singleton()->print(").\n");  	OS::get_singleton()->print("\n"); +#ifndef SERVER_ENABLED  	OS::get_singleton()->print("Display options:\n");  	OS::get_singleton()->print("  -f, --fullscreen                 Request fullscreen mode.\n");  	OS::get_singleton()->print("  -m, --maximized                  Request a maximized window.\n"); @@ -256,15 +259,18 @@ void Main::print_help(const char *p_binary) {  	OS::get_singleton()->print("  --position <X>,<Y>               Request window position.\n");  	OS::get_singleton()->print("  --low-dpi                        Force low-DPI mode (macOS and Windows only).\n");  	OS::get_singleton()->print("  --no-window                      Disable window creation (Windows only). Useful together with --script.\n"); +	OS::get_singleton()->print("  --enable-vsync-via-compositor    When vsync is enabled, vsync via the OS' window compositor (Windows only).\n"); +	OS::get_singleton()->print("  --disable-vsync-via-compositor   Disable vsync via the OS' window compositor (Windows only).\n");  	OS::get_singleton()->print("\n"); +#endif  	OS::get_singleton()->print("Debug options:\n");  	OS::get_singleton()->print("  -d, --debug                      Debug (local stdout debugger).\n");  	OS::get_singleton()->print("  -b, --breakpoints                Breakpoint list as source::line comma-separated pairs, no spaces (use %%20 instead).\n");  	OS::get_singleton()->print("  --profiling                      Enable profiling in the script debugger.\n");  	OS::get_singleton()->print("  --remote-debug <address>         Remote debug (<host/IP>:<port> address).\n"); -#ifdef DEBUG_ENABLED -	OS::get_singleton()->print("  --debug-collisions               Show collisions shapes when running the scene.\n"); +#if defined(DEBUG_ENABLED) && !defined(SERVER_ENABLED) +	OS::get_singleton()->print("  --debug-collisions               Show collision shapes when running the scene.\n");  	OS::get_singleton()->print("  --debug-navigation               Show navigation polygons when running the scene.\n");  #endif  	OS::get_singleton()->print("  --frame-delay <ms>               Simulate high CPU load (delay each frame by <ms> milliseconds).\n"); @@ -372,7 +378,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  	while (I) { -		I->get() = unescape_cmdline(I->get().strip_escapes()); +		I->get() = unescape_cmdline(I->get().strip_edges());  		I = I->next();  	} @@ -384,6 +390,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  	bool upwards = false;  	String debug_mode;  	String debug_host; +	bool skip_breakpoints = false;  	String main_pack;  	bool quiet_stdout = false;  	int rtm = -1; @@ -394,6 +401,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  	Vector<String> breakpoints;  	bool use_custom_res = true;  	bool force_res = false; +	bool saw_vsync_via_compositor_override = false;  #ifdef TOOLS_ENABLED  	bool found_project = false;  #endif @@ -429,6 +437,101 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  			print_line(get_full_version_string());  			goto error; +		} else if (I->get() == "-v" || I->get() == "--verbose") { // verbose output + +			OS::get_singleton()->_verbose_stdout = true; +		} else if (I->get() == "--quiet") { // quieter output + +			quiet_stdout = true; + +		} else if (I->get() == "--audio-driver") { // audio driver + +			if (I->next()) { + +				audio_driver = I->next()->get(); + +				bool found = false; +				for (int i = 0; i < OS::get_singleton()->get_audio_driver_count(); i++) { +					if (audio_driver == OS::get_singleton()->get_audio_driver_name(i)) { +						found = true; +					} +				} + +				if (!found) { +					OS::get_singleton()->print("Unknown audio driver '%s', aborting.\nValid options are ", audio_driver.utf8().get_data()); + +					for (int i = 0; i < OS::get_singleton()->get_audio_driver_count(); i++) { +						if (i == OS::get_singleton()->get_audio_driver_count() - 1) { +							OS::get_singleton()->print(" and "); +						} else if (i != 0) { +							OS::get_singleton()->print(", "); +						} + +						OS::get_singleton()->print("'%s'", OS::get_singleton()->get_audio_driver_name(i)); +					} + +					OS::get_singleton()->print(".\n"); + +					goto error; +				} + +				N = I->next()->next(); +			} else { +				OS::get_singleton()->print("Missing audio driver argument, aborting.\n"); +				goto error; +			} + +		} else if (I->get() == "--video-driver") { // force video driver + +			if (I->next()) { + +				video_driver = I->next()->get(); + +				bool found = false; +				for (int i = 0; i < OS::get_singleton()->get_video_driver_count(); i++) { +					if (video_driver == OS::get_singleton()->get_video_driver_name(i)) { +						found = true; +					} +				} + +				if (!found) { +					OS::get_singleton()->print("Unknown video driver '%s', aborting.\nValid options are ", video_driver.utf8().get_data()); + +					for (int i = 0; i < OS::get_singleton()->get_video_driver_count(); i++) { +						if (i == OS::get_singleton()->get_video_driver_count() - 1) { +							OS::get_singleton()->print(" and "); +						} else if (i != 0) { +							OS::get_singleton()->print(", "); +						} + +						OS::get_singleton()->print("'%s'", OS::get_singleton()->get_video_driver_name(i)); +					} + +					OS::get_singleton()->print(".\n"); + +					goto error; +				} + +				N = I->next()->next(); +			} else { +				OS::get_singleton()->print("Missing video driver argument, aborting.\n"); +				goto error; +			} +#ifndef SERVER_ENABLED +		} else if (I->get() == "-f" || I->get() == "--fullscreen") { // force fullscreen + +			init_fullscreen = true; +		} else if (I->get() == "-m" || I->get() == "--maximized") { // force maximized window + +			init_maximized = true; +			video_mode.maximized = true; + +		} else if (I->get() == "-w" || I->get() == "--windowed") { // force windowed window + +			init_windowed = true; +		} else if (I->get() == "-t" || I->get() == "--always-on-top") { // force always-on-top window + +			init_always_on_top = true;  		} else if (I->get() == "--resolution") { // force resolution  			if (I->next()) { @@ -459,6 +562,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  				OS::get_singleton()->print("Missing resolution argument, aborting.\n");  				goto error;  			} +  		} else if (I->get() == "--position") { // set window position  			if (I->next()) { @@ -483,29 +587,25 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  				goto error;  			} -		} else if (I->get() == "-m" || I->get() == "--maximized") { // force maximized window +		} else if (I->get() == "--low-dpi") { // force low DPI (macOS only) -			init_maximized = true; -			video_mode.maximized = true; -		} else if (I->get() == "-w" || I->get() == "--windowed") { // force windowed window +			force_lowdpi = true; +		} else if (I->get() == "--no-window") { // disable window creation (Windows only) -			init_windowed = true; -		} else if (I->get() == "-t" || I->get() == "--always-on-top") { // force always-on-top window +			OS::get_singleton()->set_no_window_mode(true); +		} else if (I->get() == "--enable-vsync-via-compositor") { -			init_always_on_top = true; +			video_mode.vsync_via_compositor = true; +			saw_vsync_via_compositor_override = true; +		} else if (I->get() == "--disable-vsync-via-compositor") { + +			video_mode.vsync_via_compositor = false; +			saw_vsync_via_compositor_override = true; +#endif  		} else if (I->get() == "--profiling") { // enable profiling  			use_debug_profiler = true; -		} else if (I->get() == "--video-driver") { // force video driver - -			if (I->next()) { -				video_driver = I->next()->get(); -				N = I->next()->next(); -			} else { -				OS::get_singleton()->print("Missing video driver argument, aborting.\n"); -				goto error; -			}  		} else if (I->get() == "-l" || I->get() == "--language") { // language  			if (I->next()) { @@ -516,9 +616,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  				OS::get_singleton()->print("Missing language argument, aborting.\n");  				goto error;  			} -		} else if (I->get() == "--low-dpi") { // force low DPI (macOS only) -			force_lowdpi = true;  		} else if (I->get() == "--remote-fs") { // remote filesystem  			if (I->next()) { @@ -555,22 +653,6 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  				OS::get_singleton()->print("Missing render thread mode argument, aborting.\n");  				goto error;  			} - -		} else if (I->get() == "--audio-driver") { // audio driver - -			if (I->next()) { - -				audio_driver = I->next()->get(); -				N = I->next()->next(); -			} else { -				OS::get_singleton()->print("Missing audio driver argument, aborting.\n"); -				goto error; -			} - -		} else if (I->get() == "-f" || I->get() == "--fullscreen") { // force fullscreen - -			//video_mode.fullscreen=false; -			init_fullscreen = true;  #ifdef TOOLS_ENABLED  		} else if (I->get() == "-e" || I->get() == "--editor") { // starts editor @@ -582,15 +664,19 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  			auto_build_solutions = true;  			editor = true; -#endif -		} else if (I->get() == "--no-window") { // disable window creation, Windows only +#ifdef DEBUG_METHODS_ENABLED +		} else if (I->get() == "--gdnative-generate-json-api") { +			// Register as an editor instance to use the GLES2 fallback automatically on hardware that doesn't support the GLES3 backend +			editor = true; -			OS::get_singleton()->set_no_window_mode(true); -		} else if (I->get() == "--quiet") { // quieter output +			// We still pass it to the main arguments since the argument handling itself is not done in this function +			main_args.push_back(I->get()); +#endif +		} else if (I->get() == "--export" || I->get() == "--export-debug") { // Export project -			quiet_stdout = true; -		} else if (I->get() == "-v" || I->get() == "--verbose") { // verbose output -			OS::get_singleton()->_verbose_stdout = true; +			editor = true; +			main_args.push_back(I->get()); +#endif  		} else if (I->get() == "--path") { // set path of project to start or edit  			if (I->next()) { @@ -674,7 +760,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  		} else if (I->get() == "-d" || I->get() == "--debug") {  			debug_mode = "local"; -#ifdef DEBUG_ENABLED +#if defined(DEBUG_ENABLED) && !defined(SERVER_ENABLED)  		} else if (I->get() == "--debug-collisions") {  			debug_collisions = true;  		} else if (I->get() == "--debug-navigation") { @@ -717,6 +803,8 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  			print_fps = true;  		} else if (I->get() == "--disable-crash-handler") {  			OS::get_singleton()->disable_crash_handler(); +		} else if (I->get() == "--skip-breakpoints") { +			skip_breakpoints = true;  		} else {  			main_args.push_back(I->get());  		} @@ -770,8 +858,10 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  	ProjectSettings::get_singleton()->set_custom_property_info("network/limits/debugger_stdout/max_chars_per_second", PropertyInfo(Variant::INT, "network/limits/debugger_stdout/max_chars_per_second", PROPERTY_HINT_RANGE, "0, 4096, 1, or_greater"));  	GLOBAL_DEF("network/limits/debugger_stdout/max_messages_per_frame", 10);  	ProjectSettings::get_singleton()->set_custom_property_info("network/limits/debugger_stdout/max_messages_per_frame", PropertyInfo(Variant::INT, "network/limits/debugger_stdout/max_messages_per_frame", PROPERTY_HINT_RANGE, "0, 20, 1, or_greater")); -	GLOBAL_DEF("network/limits/debugger_stdout/max_errors_per_frame", 10); -	ProjectSettings::get_singleton()->set_custom_property_info("network/limits/debugger_stdout/max_errors_per_frame", PropertyInfo(Variant::INT, "network/limits/debugger_stdout/max_errors_per_frame", PROPERTY_HINT_RANGE, "0, 20, 1, or_greater")); +	GLOBAL_DEF("network/limits/debugger_stdout/max_errors_per_second", 100); +	ProjectSettings::get_singleton()->set_custom_property_info("network/limits/debugger_stdout/max_errors_per_second", PropertyInfo(Variant::INT, "network/limits/debugger_stdout/max_errors_per_second", PROPERTY_HINT_RANGE, "0, 200, 1, or_greater")); +	GLOBAL_DEF("network/limits/debugger_stdout/max_warnings_per_second", 100); +	ProjectSettings::get_singleton()->set_custom_property_info("network/limits/debugger_stdout/max_warnings_per_second", PropertyInfo(Variant::INT, "network/limits/debugger_stdout/max_warnings_per_second", PROPERTY_HINT_RANGE, "0, 200, 1, or_greater"));  	if (debug_mode == "remote") { @@ -784,6 +874,8 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  		}  		Error derr = sdr->connect_to_host(debug_host, debug_port); +		sdr->set_skip_breakpoints(skip_breakpoints); +  		if (derr != OK) {  			memdelete(sdr);  		} else { @@ -801,10 +893,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  			String bp = breakpoints[i];  			int sp = bp.find_last(":"); -			if (sp == -1) { -				ERR_EXPLAIN("Invalid breakpoint: '" + bp + "', expected file:line format."); -				ERR_CONTINUE(sp == -1); -			} +			ERR_CONTINUE_MSG(sp == -1, "Invalid breakpoint: '" + bp + "', expected file:line format.");  			script_debugger->insert_breakpoint(bp.substr(sp + 1, bp.length()).to_int(), bp.substr(0, sp));  		} @@ -906,10 +995,13 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  			video_mode.height = GLOBAL_GET("display/window/size/height");  			if (globals->has_setting("display/window/size/test_width") && globals->has_setting("display/window/size/test_height")) { +  				int tw = globals->get("display/window/size/test_width"); -				int th = globals->get("display/window/size/test_height"); -				if (tw > 0 && th > 0) { +				if (tw > 0) {  					video_mode.width = tw; +				} +				int th = globals->get("display/window/size/test_height"); +				if (th > 0) {  					video_mode.height = th;  				}  			} @@ -925,12 +1017,21 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  		OS::get_singleton()->_allow_hidpi = GLOBAL_DEF("display/window/dpi/allow_hidpi", false);  	} -	video_mode.use_vsync = GLOBAL_DEF("display/window/vsync/use_vsync", true); +	video_mode.use_vsync = GLOBAL_DEF_RST("display/window/vsync/use_vsync", true);  	OS::get_singleton()->_use_vsync = video_mode.use_vsync; +	if (!saw_vsync_via_compositor_override) { +		// If one of the command line options to enable/disable vsync via the +		// window compositor ("--enable-vsync-via-compositor" or +		// "--disable-vsync-via-compositor") was present then it overrides the +		// project setting. +		video_mode.vsync_via_compositor = GLOBAL_DEF("display/window/vsync/vsync_via_compositor", false); +	} + +	OS::get_singleton()->_vsync_via_compositor = video_mode.vsync_via_compositor; +  	OS::get_singleton()->_allow_layered = GLOBAL_DEF("display/window/per_pixel_transparency/allowed", false);  	video_mode.layered = GLOBAL_DEF("display/window/per_pixel_transparency/enabled", false); -	video_mode.layered_splash = GLOBAL_DEF("display/window/per_pixel_transparency/splash", false);  	GLOBAL_DEF("rendering/quality/intended_usage/framebuffer_allocation", 2);  	GLOBAL_DEF("rendering/quality/intended_usage/framebuffer_allocation.mobile", 3); @@ -966,10 +1067,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  	}  	if (video_driver_idx < 0) { - -		//OS::get_singleton()->alert("Invalid Video Driver: " + video_driver);  		video_driver_idx = 0; -		//goto error;  	}  	if (audio_driver == "") { // specified in project.godot @@ -986,10 +1084,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  	}  	if (audio_driver_idx < 0) { - -		OS::get_singleton()->alert("Invalid Audio Driver: " + audio_driver);  		audio_driver_idx = 0; -		//goto error;  	}  	{ @@ -1012,6 +1107,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  	}  	Engine::get_singleton()->set_iterations_per_second(GLOBAL_DEF("physics/common/physics_fps", 60)); +	ProjectSettings::get_singleton()->set_custom_property_info("physics/common/physics_fps", PropertyInfo(Variant::INT, "physics/common/physics_fps", PROPERTY_HINT_RANGE, "1,120,1,or_greater"));  	Engine::get_singleton()->set_physics_jitter_fix(GLOBAL_DEF("physics/common/physics_jitter_fix", 0.5));  	Engine::get_singleton()->set_target_fps(GLOBAL_DEF("debug/settings/fps/force_fps", 0));  	ProjectSettings::get_singleton()->set_custom_property_info("debug/settings/fps/force_fps", PropertyInfo(Variant::INT, "debug/settings/fps/force_fps", PROPERTY_HINT_RANGE, "0,120,1,or_greater")); @@ -1027,7 +1123,7 @@ Error Main::setup(const char *execpath, int argc, char *argv[], bool p_second_ph  	}  	OS::get_singleton()->set_low_processor_usage_mode(GLOBAL_DEF("application/run/low_processor_mode", false)); -	OS::get_singleton()->set_low_processor_usage_mode_sleep_usec(GLOBAL_DEF("application/run/low_processor_mode_sleep_usec", 8000)); +	OS::get_singleton()->set_low_processor_usage_mode_sleep_usec(GLOBAL_DEF("application/run/low_processor_mode_sleep_usec", 6900)); // Roughly 144 FPS  	ProjectSettings::get_singleton()->set_custom_property_info("application/run/low_processor_mode_sleep_usec", PropertyInfo(Variant::INT, "application/run/low_processor_mode_sleep_usec", PROPERTY_HINT_RANGE, "0,33200,1,or_greater")); // No negative numbers  	Engine::get_singleton()->set_frame_delay(frame_delay); @@ -1083,6 +1179,9 @@ error:  Error Main::setup2(Thread::ID p_main_tid_override) { +	// Print engine name and version +	print_line(String(VERSION_NAME) + " v" + get_full_version_string() + " - " + String(VERSION_WEBSITE)); +  	if (p_main_tid_override) {  		Thread::_main_thread_id = p_main_tid_override;  	} @@ -1092,6 +1191,8 @@ Error Main::setup2(Thread::ID p_main_tid_override) {  		return err;  	} +	print_line(" "); //add a blank line for readability +  	if (init_use_custom_pos) {  		OS::get_singleton()->set_window_position(init_custom_pos);  	} @@ -1142,6 +1243,7 @@ Error Main::setup2(Thread::ID p_main_tid_override) {  	if (show_logo) { //boot logo!  		String boot_logo_path = GLOBAL_DEF("application/boot_splash/image", String());  		bool boot_logo_scale = GLOBAL_DEF("application/boot_splash/fullsize", true); +		bool boot_logo_filter = GLOBAL_DEF("application/boot_splash/use_filter", true);  		ProjectSettings::get_singleton()->set_custom_property_info("application/boot_splash/image", PropertyInfo(Variant::STRING, "application/boot_splash/image", PROPERTY_HINT_FILE, "*.png"));  		Ref<Image> boot_logo; @@ -1152,13 +1254,13 @@ Error Main::setup2(Thread::ID p_main_tid_override) {  			boot_logo.instance();  			Error load_err = ImageLoader::load_image(boot_logo_path, boot_logo);  			if (load_err) -				ERR_PRINTS("Non-existing or invalid boot splash at: " + boot_logo_path + ". Loading default splash."); +				ERR_PRINTS("Non-existing or invalid boot splash at '" + boot_logo_path + "'. Loading default splash.");  		}  		Color boot_bg_color = GLOBAL_DEF("application/boot_splash/bg_color", boot_splash_bg_color);  		if (boot_logo.is_valid()) {  			OS::get_singleton()->_msec_splash = OS::get_singleton()->get_ticks_msec(); -			VisualServer::get_singleton()->set_boot_image(boot_logo, boot_bg_color, boot_logo_scale); +			VisualServer::get_singleton()->set_boot_image(boot_logo, boot_bg_color, boot_logo_scale, boot_logo_filter);  		} else {  #ifndef NO_DEFAULT_BOOT_LOGO @@ -1189,6 +1291,12 @@ Error Main::setup2(Thread::ID p_main_tid_override) {  	GLOBAL_DEF("application/config/icon", String());  	ProjectSettings::get_singleton()->set_custom_property_info("application/config/icon", PropertyInfo(Variant::STRING, "application/config/icon", PROPERTY_HINT_FILE, "*.png,*.webp")); +	GLOBAL_DEF("application/config/macos_native_icon", String()); +	ProjectSettings::get_singleton()->set_custom_property_info("application/config/macos_native_icon", PropertyInfo(Variant::STRING, "application/config/macos_native_icon", PROPERTY_HINT_FILE, "*.icns")); + +	GLOBAL_DEF("application/config/windows_native_icon", String()); +	ProjectSettings::get_singleton()->set_custom_property_info("application/config/windows_native_icon", PropertyInfo(Variant::STRING, "application/config/windows_native_icon", PROPERTY_HINT_FILE, "*.ico")); +  	InputDefault *id = Object::cast_to<InputDefault>(Input::get_singleton());  	if (id) {  		if (bool(GLOBAL_DEF("input_devices/pointing/emulate_touch_from_mouse", false)) && !(editor || project_manager)) { @@ -1233,6 +1341,8 @@ Error Main::setup2(Thread::ID p_main_tid_override) {  	register_platform_apis();  	register_module_types(); +	camera_server = CameraServer::create(); +  	initialize_physics();  	register_server_singletons(); @@ -1263,8 +1373,8 @@ Error Main::setup2(Thread::ID p_main_tid_override) {  	ClassDB::set_current_api(ClassDB::API_NONE); //no more api is registered at this point -	print_verbose("CORE API HASH: " + itos(ClassDB::get_api_hash(ClassDB::API_CORE))); -	print_verbose("EDITOR API HASH: " + itos(ClassDB::get_api_hash(ClassDB::API_EDITOR))); +	print_verbose("CORE API HASH: " + uitos(ClassDB::get_api_hash(ClassDB::API_CORE))); +	print_verbose("EDITOR API HASH: " + uitos(ClassDB::get_api_hash(ClassDB::API_EDITOR)));  	MAIN_PRINT("Main: Done");  	return OK; @@ -1322,20 +1432,10 @@ bool Main::start() {  					removal_docs.push_back(args[j]);  			} else if (args[i] == "--export") {  				editor = true; //needs editor -				if (i + 1 < args.size()) { -					_export_preset = args[i + 1]; -				} else { -					ERR_PRINT("Export preset name not specified"); -					return false; -				} +				_export_preset = args[i + 1];  			} else if (args[i] == "--export-debug") {  				editor = true; //needs editor -				if (i + 1 < args.size()) { -					_export_preset = args[i + 1]; -				} else { -					ERR_PRINT("Export preset name not specified"); -					return false; -				} +				_export_preset = args[i + 1];  				export_debug = true;  #endif  			} else { @@ -1348,18 +1448,15 @@ bool Main::start() {  		}  	} -	GLOBAL_DEF("editor/active", editor); -  	String main_loop_type;  #ifdef TOOLS_ENABLED  	if (doc_tool != "") { +		Engine::get_singleton()->set_editor_hint(true); // Needed to instance editor-only classes for their default values +  		{  			DirAccessRef da = DirAccess::open(doc_tool); -			if (!da) { -				ERR_EXPLAIN("Argument supplied to --doctool must be a base godot build directory"); -				ERR_FAIL_V(false); -			} +			ERR_FAIL_COND_V_MSG(!da, false, "Argument supplied to --doctool must be a base Godot build directory.");  		}  		DocData doc;  		doc.generate(doc_base); @@ -1375,12 +1472,23 @@ bool Main::start() {  			doc_data_classes[name] = path;  			if (!checked_paths.has(path)) {  				checked_paths.insert(path); + +				// Create the module documentation directory if it doesn't exist +				DirAccess *da = DirAccess::create_for_path(path); +				da->make_dir_recursive(path); +				memdelete(da); +  				docsrc.load_classes(path);  				print_line("Loading docs from: " + path);  			}  		}  		String index_path = doc_tool.plus_file("doc/classes"); +		// Create the main documentation directory if it doesn't exist +		DirAccess *da = DirAccess::create_for_path(index_path); +		da->make_dir_recursive(index_path); +		memdelete(da); +  		docsrc.load_classes(index_path);  		checked_paths.insert(index_path);  		print_line("Loading docs from: " + index_path); @@ -1421,21 +1529,22 @@ bool Main::start() {  	};  	if (test != "") { -#ifdef DEBUG_ENABLED +#ifdef TOOLS_ENABLED  		main_loop = test_main(test, args);  		if (!main_loop)  			return false; -  #endif  	} else if (script != "") {  		Ref<Script> script_res = ResourceLoader::load(script); -		ERR_EXPLAIN("Can't load script: " + script); -		ERR_FAIL_COND_V(script_res.is_null(), false); +		ERR_FAIL_COND_V_MSG(script_res.is_null(), false, "Can't load script: " + script);  		if (check_only) { +			if (!script_res->is_valid()) { +				OS::get_singleton()->set_exit_code(1); +			}  			return false;  		} @@ -1447,8 +1556,7 @@ bool Main::start() {  			if (!script_loop) {  				if (obj)  					memdelete(obj); -				ERR_EXPLAIN("Can't load script '" + script + "', it does not inherit from a MainLoop type"); -				ERR_FAIL_COND_V(!script_loop, false); +				ERR_FAIL_V_MSG(false, "Can't load script '" + script + "', it does not inherit from a MainLoop type.");  			}  			script_loop->set_init_script(script_res); @@ -1472,17 +1580,13 @@ bool Main::start() {  		} else {  			Object *ml = ClassDB::instance(main_loop_type); -			if (!ml) { -				ERR_EXPLAIN("Can't instance MainLoop type"); -				ERR_FAIL_V(false); -			} +			ERR_FAIL_COND_V_MSG(!ml, false, "Can't instance MainLoop type.");  			main_loop = Object::cast_to<MainLoop>(ml);  			if (!main_loop) {  				memdelete(ml); -				ERR_EXPLAIN("Invalid MainLoop type"); -				ERR_FAIL_V(false); +				ERR_FAIL_V_MSG(false, "Invalid MainLoop type.");  			}  		}  	} @@ -1505,6 +1609,12 @@ bool Main::start() {  		if (!project_manager && !editor) { // game  			if (game_path != "" || script != "") { +				if (script_debugger && script_debugger->is_remote()) { +					ScriptDebuggerRemote *remote_debugger = static_cast<ScriptDebuggerRemote *>(script_debugger); + +					remote_debugger->set_scene_tree(sml); +				} +  				//autoload  				List<PropertyInfo> props;  				ProjectSettings::get_singleton()->get_property_list(&props); @@ -1545,8 +1655,7 @@ bool Main::start() {  					}  					RES res = ResourceLoader::load(path); -					ERR_EXPLAIN("Can't autoload: " + path); -					ERR_CONTINUE(res.is_null()); +					ERR_CONTINUE_MSG(res.is_null(), "Can't autoload: " + path);  					Node *n = NULL;  					if (res->is_class("PackedScene")) {  						Ref<PackedScene> ps = res; @@ -1555,20 +1664,17 @@ bool Main::start() {  						Ref<Script> script_res = res;  						StringName ibt = script_res->get_instance_base_type();  						bool valid_type = ClassDB::is_parent_class(ibt, "Node"); -						ERR_EXPLAIN("Script does not inherit a Node: " + path); -						ERR_CONTINUE(!valid_type); +						ERR_CONTINUE_MSG(!valid_type, "Script does not inherit a Node: " + path);  						Object *obj = ClassDB::instance(ibt); -						ERR_EXPLAIN("Cannot instance script for autoload, expected 'Node' inheritance, got: " + String(ibt)); -						ERR_CONTINUE(obj == NULL); +						ERR_CONTINUE_MSG(obj == NULL, "Cannot instance script for autoload, expected 'Node' inheritance, got: " + String(ibt));  						n = Object::cast_to<Node>(obj);  						n->set_script(script_res.get_ref_ptr());  					} -					ERR_EXPLAIN("Path in autoload not a node or script: " + path); -					ERR_CONTINUE(!n); +					ERR_CONTINUE_MSG(!n, "Path in autoload not a node or script: " + path);  					n->set_name(name);  					//defer so references are all valid on _ready() @@ -1692,13 +1798,13 @@ bool Main::start() {  						if (sep == -1) {  							DirAccess *da = DirAccess::create(DirAccess::ACCESS_FILESYSTEM); -							local_game_path = da->get_current_dir() + "/" + local_game_path; +							local_game_path = da->get_current_dir().plus_file(local_game_path);  							memdelete(da);  						} else {  							DirAccess *da = DirAccess::open(local_game_path.substr(0, sep));  							if (da) { -								local_game_path = da->get_current_dir() + "/" + local_game_path.substr(sep + 1, local_game_path.length()); +								local_game_path = da->get_current_dir().plus_file(local_game_path.substr(sep + 1, local_game_path.length()));  								memdelete(da);  							}  						} @@ -1726,8 +1832,8 @@ bool Main::start() {  		if (!project_manager && !editor) { // game -			// Load SSL Certificates from Project Settings (or builtin) -			StreamPeerSSL::load_certs_from_memory(StreamPeerSSL::get_project_cert_array()); +			// Load SSL Certificates from Project Settings (or builtin). +			Crypto::load_default_certificates(GLOBAL_DEF("network/ssl/certificates", ""));  			if (game_path != "") {  				Node *scene = NULL; @@ -1735,12 +1841,27 @@ bool Main::start() {  				if (scenedata.is_valid())  					scene = scenedata->instance(); -				ERR_EXPLAIN("Failed loading scene: " + local_game_path); -				ERR_FAIL_COND_V(!scene, false) +				ERR_FAIL_COND_V_MSG(!scene, false, "Failed loading scene: " + local_game_path);  				sml->add_current_scene(scene); +#ifdef OSX_ENABLED +				String mac_iconpath = GLOBAL_DEF("application/config/macos_native_icon", "Variant()"); +				if (mac_iconpath != "") { +					OS::get_singleton()->set_native_icon(mac_iconpath); +					hasicon = true; +				} +#endif + +#ifdef WINDOWS_ENABLED +				String win_iconpath = GLOBAL_DEF("application/config/windows_native_icon", "Variant()"); +				if (win_iconpath != "") { +					OS::get_singleton()->set_native_icon(win_iconpath); +					hasicon = true; +				} +#endif +  				String iconpath = GLOBAL_DEF("application/config/icon", "Variant()"); -				if (iconpath != "") { +				if ((iconpath != "") && (!hasicon)) {  					Ref<Image> icon;  					icon.instance();  					if (ImageLoader::load_image(iconpath, icon) == OK) { @@ -1760,15 +1881,16 @@ bool Main::start() {  			pmanager->add_child(progress_dialog);  			sml->get_root()->add_child(pmanager);  			OS::get_singleton()->set_context(OS::CONTEXT_PROJECTMAN); +			project_manager = true;  		}  		if (project_manager || editor) { +			// Hide console window if requested (Windows-only). +			bool hide_console = EditorSettings::get_singleton()->get_setting("interface/editor/hide_console_window"); +			OS::get_singleton()->set_console_visible(!hide_console); +  			// Load SSL Certificates from Editor Settings (or builtin) -			String certs = EditorSettings::get_singleton()->get_setting("network/ssl/editor_ssl_certificates").operator String(); -			if (certs != "") -				StreamPeerSSL::load_certs_from_file(certs); -			else -				StreamPeerSSL::load_certs_from_memory(StreamPeerSSL::get_project_cert_array()); +			Crypto::load_default_certificates(EditorSettings::get_singleton()->get_setting("network/ssl/editor_ssl_certificates").operator String());  		}  #endif  	} @@ -1803,7 +1925,7 @@ bool Main::is_iterating() {  	return iterating > 0;  } -// For performance metrics +// For performance metrics.  static uint64_t physics_process_max = 0;  static uint64_t idle_process_max = 0; @@ -1831,6 +1953,7 @@ bool Main::iteration() {  	double scaled_step = step * time_scale;  	Engine::get_singleton()->_frame_step = step; +	Engine::get_singleton()->_physics_interpolation_fraction = advance.interpolation_fraction;  	uint64_t physics_process_ticks = 0;  	uint64_t idle_process_ticks = 0; @@ -1949,7 +2072,7 @@ bool Main::iteration() {  		return exit;  	if (OS::get_singleton()->is_in_low_processor_usage_mode() || !OS::get_singleton()->can_draw()) -		OS::get_singleton()->delay_usec(OS::get_singleton()->get_low_processor_usage_mode_sleep_usec()); //apply some delay to force idle time (results in about 60 FPS max) +		OS::get_singleton()->delay_usec(OS::get_singleton()->get_low_processor_usage_mode_sleep_usec()); //apply some delay to force idle time  	else {  		uint32_t frame_delay = Engine::get_singleton()->get_frame_delay();  		if (frame_delay) @@ -1992,6 +2115,11 @@ void Main::cleanup() {  	ERR_FAIL_COND(!_start_success); +	if (script_debugger) { +		// Flush any remaining messages +		script_debugger->idle_poll(); +	} +  	ResourceLoader::remove_custom_loaders();  	ResourceSaver::remove_custom_savers(); @@ -2039,6 +2167,10 @@ void Main::cleanup() {  		memdelete(audio_server);  	} +	if (camera_server) { +		memdelete(camera_server); +	} +  	OS::get_singleton()->finalize();  	finalize_physics(); @@ -2069,6 +2201,5 @@ void Main::cleanup() {  	unregister_core_driver_types();  	unregister_core_types(); -	OS::get_singleton()->clear_last_error();  	OS::get_singleton()->finalize_core();  } |