| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*  os_unix.cpp                                                          */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*                       This file is part of:                           */ | 
					
						
							|  |  |  | /*                           GODOT ENGINE                                */ | 
					
						
							|  |  |  | /*                    http://www.godotengine.org                         */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2016-01-01 11:50:53 -02:00
										 |  |  | /* Copyright (c) 2007-2016 Juan Linietsky, Ariel Manzur.                 */ | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | /*                                                                       */ | 
					
						
							|  |  |  | /* Permission is hereby granted, free of charge, to any person obtaining */ | 
					
						
							|  |  |  | /* a copy of this software and associated documentation files (the       */ | 
					
						
							|  |  |  | /* "Software"), to deal in the Software without restriction, including   */ | 
					
						
							|  |  |  | /* without limitation the rights to use, copy, modify, merge, publish,   */ | 
					
						
							|  |  |  | /* distribute, sublicense, and/or sell copies of the Software, and to    */ | 
					
						
							|  |  |  | /* permit persons to whom the Software is furnished to do so, subject to */ | 
					
						
							|  |  |  | /* the following conditions:                                             */ | 
					
						
							|  |  |  | /*                                                                       */ | 
					
						
							|  |  |  | /* The above copyright notice and this permission notice shall be        */ | 
					
						
							|  |  |  | /* included in all copies or substantial portions of the Software.       */ | 
					
						
							|  |  |  | /*                                                                       */ | 
					
						
							|  |  |  | /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,       */ | 
					
						
							|  |  |  | /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF    */ | 
					
						
							|  |  |  | /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ | 
					
						
							|  |  |  | /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY  */ | 
					
						
							|  |  |  | /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,  */ | 
					
						
							|  |  |  | /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE     */ | 
					
						
							|  |  |  | /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | #include "os_unix.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef UNIX_ENABLED
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "memory_pool_static_malloc.h"
 | 
					
						
							|  |  |  | #include "os/memory_pool_dynamic_static.h"
 | 
					
						
							|  |  |  | #include "thread_posix.h"
 | 
					
						
							|  |  |  | #include "semaphore_posix.h"
 | 
					
						
							|  |  |  | #include "mutex_posix.h"
 | 
					
						
							|  |  |  | #include "core/os/thread_dummy.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //#include "core/io/file_access_buffered_fa.h"
 | 
					
						
							|  |  |  | #include "file_access_unix.h"
 | 
					
						
							|  |  |  | #include "dir_access_unix.h"
 | 
					
						
							|  |  |  | #include "tcp_server_posix.h"
 | 
					
						
							|  |  |  | #include "stream_peer_tcp_posix.h"
 | 
					
						
							| 
									
										
										
										
											2014-11-12 11:23:23 -03:00
										 |  |  | #include "packet_peer_udp_posix.h"
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-20 17:49:48 +10:00
										 |  |  | #ifdef __APPLE__
 | 
					
						
							|  |  |  | #include <mach-o/dyld.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 19:31:37 +10:30
										 |  |  | #ifdef __FreeBSD__
 | 
					
						
							|  |  |  | #include <sys/param.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | #include <stdarg.h>
 | 
					
						
							|  |  |  | #include <sys/time.h>
 | 
					
						
							|  |  |  | #include <sys/wait.h>
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #include <signal.h>
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #include <poll.h>
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <assert.h>
 | 
					
						
							|  |  |  | #include "globals.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-30 11:28:43 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | extern bool _print_error_enabled; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void OS_Unix::print_error(const char* p_function,const char* p_file,int p_line,const char *p_code,const char*p_rationale,ErrorType p_type) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-30 11:28:43 -03:00
										 |  |  | 	if (!_print_error_enabled) | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 20:13:05 +01:00
										 |  |  | 	const char* err_details; | 
					
						
							|  |  |  | 	if (p_rationale && p_rationale[0]) | 
					
						
							|  |  |  | 		err_details=p_rationale; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		err_details=p_code; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	switch(p_type) { | 
					
						
							|  |  |  | 		case ERR_ERROR: | 
					
						
							| 
									
										
										
										
											2015-11-10 08:04:48 +01:00
										 |  |  | 			print("\E[1;31mERROR: %s: \E[0m\E[1m%s\n",p_function,err_details); | 
					
						
							|  |  |  | 			print("\E[0;31m   At: %s:%i.\E[0m\n",p_file,p_line); | 
					
						
							| 
									
										
										
										
											2015-11-05 20:13:05 +01:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case ERR_WARNING: | 
					
						
							| 
									
										
										
										
											2015-11-10 08:04:48 +01:00
										 |  |  | 			print("\E[1;33mWARNING: %s: \E[0m\E[1m%s\n",p_function,err_details); | 
					
						
							| 
									
										
										
										
											2016-01-13 09:42:03 -03:00
										 |  |  | 			print("\E[0;33m   At: %s:%i.\E[0m\n",p_file,p_line); | 
					
						
							| 
									
										
										
										
											2015-11-05 20:13:05 +01:00
										 |  |  | 			break; | 
					
						
							|  |  |  | 		case ERR_SCRIPT: | 
					
						
							| 
									
										
										
										
											2015-11-10 08:04:48 +01:00
										 |  |  | 			print("\E[1;35mSCRIPT ERROR: %s: \E[0m\E[1m%s\n",p_function,err_details); | 
					
						
							| 
									
										
										
										
											2016-01-13 09:42:03 -03:00
										 |  |  | 			print("\E[0;35m   At: %s:%i.\E[0m\n",p_file,p_line); | 
					
						
							| 
									
										
										
										
											2015-11-05 20:13:05 +01:00
										 |  |  | 			break; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OS_Unix::debug_break() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	assert(false); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int OS_Unix::get_audio_driver_count() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | const char * OS_Unix::get_audio_driver_name(int p_driver) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return "dummy"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | int OS_Unix::unix_initialize_audio(int p_audio_driver) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | static MemoryPoolStaticMalloc *mempool_static=NULL; | 
					
						
							|  |  |  | static MemoryPoolDynamicStatic *mempool_dynamic=NULL; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | void OS_Unix::initialize_core() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef NO_PTHREADS
 | 
					
						
							|  |  |  | 	ThreadDummy::make_default(); | 
					
						
							|  |  |  | 	SemaphoreDummy::make_default(); | 
					
						
							|  |  |  | 	MutexDummy::make_default(); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	ThreadPosix::make_default();	 | 
					
						
							|  |  |  | 	SemaphorePosix::make_default(); | 
					
						
							|  |  |  | 	MutexPosix::make_default();	 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	FileAccess::make_default<FileAccessUnix>(FileAccess::ACCESS_RESOURCES); | 
					
						
							|  |  |  | 	FileAccess::make_default<FileAccessUnix>(FileAccess::ACCESS_USERDATA); | 
					
						
							|  |  |  | 	FileAccess::make_default<FileAccessUnix>(FileAccess::ACCESS_FILESYSTEM); | 
					
						
							|  |  |  | 	//FileAccessBufferedFA<FileAccessUnix>::make_default();
 | 
					
						
							|  |  |  | 	DirAccess::make_default<DirAccessUnix>(DirAccess::ACCESS_RESOURCES); | 
					
						
							|  |  |  | 	DirAccess::make_default<DirAccessUnix>(DirAccess::ACCESS_USERDATA); | 
					
						
							|  |  |  | 	DirAccess::make_default<DirAccessUnix>(DirAccess::ACCESS_FILESYSTEM); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef NO_NETWORK
 | 
					
						
							|  |  |  | 	TCPServerPosix::make_default(); | 
					
						
							|  |  |  | 	StreamPeerTCPPosix::make_default(); | 
					
						
							| 
									
										
										
										
											2014-11-12 11:23:23 -03:00
										 |  |  | 	PacketPeerUDPPosix::make_default(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	IP_Unix::make_default(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	mempool_static = new MemoryPoolStaticMalloc; | 
					
						
							|  |  |  | 	mempool_dynamic = memnew( MemoryPoolDynamicStatic ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ticks_start=0; | 
					
						
							|  |  |  | 	ticks_start=get_ticks_usec(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OS_Unix::finalize_core() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (mempool_dynamic) | 
					
						
							|  |  |  | 		memdelete( mempool_dynamic ); | 
					
						
							| 
									
										
										
										
											2016-01-04 11:46:16 +01:00
										 |  |  | 	delete mempool_static; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OS_Unix::vprint(const char* p_format, va_list p_list,bool p_stder) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p_stder) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		vfprintf(stderr,p_format,p_list); | 
					
						
							|  |  |  | 		fflush(stderr); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		vprintf(p_format,p_list); | 
					
						
							|  |  |  | 		fflush(stdout); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void OS_Unix::print(const char *p_format, ... ) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	va_list argp; | 
					
						
							|  |  |  | 	va_start(argp, p_format); | 
					
						
							|  |  |  | 	vprintf(p_format, argp ); | 
					
						
							|  |  |  | 	va_end(argp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | void OS_Unix::alert(const String& p_alert,const String& p_title) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fprintf(stderr,"ERROR: %s\n",p_alert.utf8().get_data()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int has_data(FILE* p_fd, int timeout_usec = 0) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	fd_set readset; | 
					
						
							|  |  |  | 	int fd = fileno(p_fd); | 
					
						
							|  |  |  | 	FD_ZERO(&readset); | 
					
						
							|  |  |  | 	FD_SET(fd, &readset); | 
					
						
							|  |  |  | 	timeval time; | 
					
						
							|  |  |  | 	time.tv_sec = 0; | 
					
						
							|  |  |  | 	time.tv_usec = timeout_usec; | 
					
						
							|  |  |  | 	int res = 0;//select(fd + 1, &readset, NULL, NULL, &time);
 | 
					
						
							|  |  |  | 	return res > 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String OS_Unix::get_stdin_string(bool p_block) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	String ret; | 
					
						
							|  |  |  | 	if (p_block) { | 
					
						
							|  |  |  | 		char buff[1024]; | 
					
						
							|  |  |  | 		ret = stdin_buf + fgets(buff,1024,stdin); | 
					
						
							|  |  |  | 		stdin_buf = ""; | 
					
						
							|  |  |  | 		return ret; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	while (has_data(stdin)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		char ch; | 
					
						
							|  |  |  | 		read(fileno(stdin), &ch, 1); | 
					
						
							|  |  |  | 		if (ch == '\n') { | 
					
						
							|  |  |  | 			ret = stdin_buf; | 
					
						
							|  |  |  | 			stdin_buf = ""; | 
					
						
							|  |  |  | 			return ret; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			char str[2] = { ch, 0 }; | 
					
						
							|  |  |  | 			stdin_buf += str; | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ""; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String OS_Unix::get_name() { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return "Unix"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint64_t OS_Unix::get_unix_time() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return time(NULL); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-10 18:24:55 -03:00
										 |  |  | uint64_t OS_Unix::get_system_time_secs() const { | 
					
						
							| 
									
										
										
										
											2015-08-06 10:29:33 -07:00
										 |  |  | 	struct timeval tv_now; | 
					
						
							| 
									
										
										
										
											2015-08-06 11:08:48 -07:00
										 |  |  | 	gettimeofday(&tv_now, NULL); | 
					
						
							| 
									
										
										
										
											2015-09-09 20:24:38 +00:00
										 |  |  | 	//localtime(&tv_now.tv_usec);
 | 
					
						
							| 
									
										
										
										
											2015-09-10 13:10:23 -03:00
										 |  |  | 	//localtime((const long *)&tv_now.tv_usec);
 | 
					
						
							| 
									
										
										
										
											2016-01-10 18:24:55 -03:00
										 |  |  | 	return uint64_t(tv_now.tv_sec); | 
					
						
							| 
									
										
										
										
											2015-08-06 10:29:33 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-06 03:40:56 +02:00
										 |  |  | OS::Date OS_Unix::get_date(bool utc) const { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	time_t t=time(NULL); | 
					
						
							| 
									
										
										
										
											2015-06-06 03:40:56 +02:00
										 |  |  | 	struct tm *lt; | 
					
						
							|  |  |  | 	if (utc) | 
					
						
							|  |  |  | 		lt=gmtime(&t); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		lt=localtime(&t); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	Date ret; | 
					
						
							|  |  |  | 	ret.year=1900+lt->tm_year; | 
					
						
							| 
									
										
										
										
											2016-03-12 19:13:57 -07:00
										 |  |  | 	// Index starting at 1 to match OS_Unix::get_date
 | 
					
						
							|  |  |  | 	//   and Windows SYSTEMTIME and tm_mon follows the typical structure 
 | 
					
						
							|  |  |  | 	//   of 0-11, noted here: http://www.cplusplus.com/reference/ctime/tm/
 | 
					
						
							| 
									
										
										
										
											2015-08-13 18:56:13 +07:00
										 |  |  | 	ret.month=(Month)(lt->tm_mon + 1); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	ret.day=lt->tm_mday; | 
					
						
							|  |  |  | 	ret.weekday=(Weekday)lt->tm_wday; | 
					
						
							|  |  |  | 	ret.dst=lt->tm_isdst; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-03-15 21:17:10 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-06 03:40:56 +02:00
										 |  |  | OS::Time OS_Unix::get_time(bool utc) const { | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	time_t t=time(NULL); | 
					
						
							| 
									
										
										
										
											2015-06-06 03:40:56 +02:00
										 |  |  | 	struct tm *lt; | 
					
						
							|  |  |  | 	if (utc) | 
					
						
							|  |  |  | 		lt=gmtime(&t); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		lt=localtime(&t); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	Time ret; | 
					
						
							|  |  |  | 	ret.hour=lt->tm_hour; | 
					
						
							|  |  |  | 	ret.min=lt->tm_min; | 
					
						
							|  |  |  | 	ret.sec=lt->tm_sec; | 
					
						
							| 
									
										
										
										
											2015-06-06 05:35:38 +02:00
										 |  |  | 	get_time_zone_info(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-06-06 05:35:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | OS::TimeZoneInfo OS_Unix::get_time_zone_info() const { | 
					
						
							|  |  |  | 	time_t t = time(NULL); | 
					
						
							|  |  |  | 	struct tm *lt = localtime(&t); | 
					
						
							|  |  |  | 	char name[16]; | 
					
						
							|  |  |  | 	strftime(name, 16, "%Z", lt); | 
					
						
							|  |  |  | 	name[15] = 0; | 
					
						
							|  |  |  | 	TimeZoneInfo ret; | 
					
						
							|  |  |  | 	ret.name = name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	char bias_buf[16]; | 
					
						
							|  |  |  | 	strftime(bias_buf, 16, "%z", lt); | 
					
						
							|  |  |  | 	int bias; | 
					
						
							|  |  |  | 	bias_buf[15] = 0; | 
					
						
							|  |  |  | 	sscanf(bias_buf, "%d", &bias); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// convert from ISO 8601 (1 minute=1, 1 hour=100) to minutes
 | 
					
						
							|  |  |  | 	int hour = (int)bias / 100; | 
					
						
							|  |  |  | 	int minutes = bias % 100; | 
					
						
							|  |  |  | 	if (bias < 0) | 
					
						
							|  |  |  | 		ret.bias = hour * 60 - minutes; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		ret.bias = hour * 60 + minutes; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | void OS_Unix::delay_usec(uint32_t p_usec) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	usleep(p_usec); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | uint64_t OS_Unix::get_ticks_usec() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct timeval tv_now; | 
					
						
							|  |  |  | 	gettimeofday(&tv_now,NULL); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	uint64_t longtime = (uint64_t)tv_now.tv_usec + (uint64_t)tv_now.tv_sec*1000000L; | 
					
						
							|  |  |  | 	longtime-=ticks_start; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	return longtime; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Error OS_Unix::execute(const String& p_path, const List<String>& p_arguments,bool p_blocking,ProcessID *r_child_id,String* r_pipe,int *r_exitcode) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p_blocking && r_pipe) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		String argss; | 
					
						
							|  |  |  | 		argss="\""+p_path+"\""; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for(int i=0;i<p_arguments.size();i++) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			argss+=String(" \"")+p_arguments[i]+"\""; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		argss+=" 2>/dev/null"; //silence stderr
 | 
					
						
							|  |  |  | 		FILE* f=popen(argss.utf8().get_data(),"r"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		ERR_FAIL_COND_V(!f,ERR_CANT_OPEN); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		char buf[65535]; | 
					
						
							|  |  |  | 		while(fgets(buf,65535,f)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			(*r_pipe)+=buf; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		int rv = pclose(f); | 
					
						
							|  |  |  | 		if (r_exitcode) | 
					
						
							|  |  |  | 			*r_exitcode=rv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return OK; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	pid_t pid = fork(); | 
					
						
							|  |  |  | 	ERR_FAIL_COND_V(pid<0,ERR_CANT_FORK); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (pid==0) { | 
					
						
							|  |  |  | 		// is child
 | 
					
						
							|  |  |  | 		Vector<CharString> cs; | 
					
						
							|  |  |  | 		cs.push_back(p_path.utf8()); | 
					
						
							|  |  |  | 		for(int i=0;i<p_arguments.size();i++) | 
					
						
							|  |  |  | 			cs.push_back(p_arguments[i].utf8()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		Vector<char*> args; | 
					
						
							|  |  |  | 		for(int i=0;i<cs.size();i++) | 
					
						
							|  |  |  | 			args.push_back((char*)cs[i].get_data());// shitty C cast
 | 
					
						
							|  |  |  | 		args.push_back(0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-27 19:31:37 +10:30
										 |  |  | #ifdef __FreeBSD__
 | 
					
						
							|  |  |  | 		if(p_path.find("/")) { | 
					
						
							|  |  |  | 			// exec name contains path so use it
 | 
					
						
							|  |  |  | 			execv(p_path.utf8().get_data(),&args[0]); | 
					
						
							|  |  |  | 		}else{ | 
					
						
							|  |  |  | 			// use program name and search through PATH to find it
 | 
					
						
							|  |  |  | 			execvp(getprogname(),&args[0]); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		execv(p_path.utf8().get_data(),&args[0]); | 
					
						
							| 
									
										
										
										
											2015-01-27 19:31:37 +10:30
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		// still alive? something failed..
 | 
					
						
							|  |  |  | 		fprintf(stderr,"**ERROR** OS_Unix::execute - Could not create child process while executing: %s\n",p_path.utf8().get_data()); | 
					
						
							|  |  |  | 		abort(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (p_blocking) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		int status; | 
					
						
							|  |  |  | 		pid_t rpid = waitpid(pid,&status,0); | 
					
						
							|  |  |  | 		if (r_exitcode) | 
					
						
							|  |  |  | 			*r_exitcode=WEXITSTATUS(status); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (r_child_id) | 
					
						
							|  |  |  | 			*r_child_id=pid; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return OK; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Error OS_Unix::kill(const ProcessID& p_pid) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int ret = ::kill(p_pid,SIGKILL); | 
					
						
							| 
									
										
										
										
											2014-12-16 22:31:57 -03:00
										 |  |  | 	if (!ret) { | 
					
						
							|  |  |  | 		//avoid zombie process
 | 
					
						
							|  |  |  | 		int st; | 
					
						
							|  |  |  | 		::waitpid(p_pid,&st,0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	return ret?ERR_INVALID_PARAMETER:OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-05 12:39:30 -03:00
										 |  |  | int OS_Unix::get_process_ID() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return getpid(); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | bool OS_Unix::has_environment(const String& p_var) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return getenv(p_var.utf8().get_data())!=NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String OS_Unix::get_locale() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!has_environment("LANG")) | 
					
						
							|  |  |  | 		return "en"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	String locale = get_environment("LANG"); | 
					
						
							|  |  |  | 	int tp = locale.find("."); | 
					
						
							|  |  |  | 	if (tp!=-1) | 
					
						
							|  |  |  | 		locale=locale.substr(0,tp); | 
					
						
							|  |  |  | 	return locale; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Error OS_Unix::set_cwd(const String& p_cwd) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (chdir(p_cwd.utf8().get_data())!=0) | 
					
						
							|  |  |  | 		return ERR_CANT_OPEN; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return OK; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String OS_Unix::get_environment(const String& p_var) const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (getenv(p_var.utf8().get_data())) | 
					
						
							|  |  |  | 		return getenv(p_var.utf8().get_data()); | 
					
						
							|  |  |  | 	return ""; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int OS_Unix::get_processor_count() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return sysconf(_SC_NPROCESSORS_CONF); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String OS_Unix::get_data_dir() const { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-03 03:39:37 +09:00
										 |  |  | 	String an = get_safe_application_name(); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	if (an!="") { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		if (has_environment("HOME")) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-19 18:39:50 -03:00
										 |  |  | 			bool use_godot = Globals::get_singleton()->get("application/use_shared_user_dir"); | 
					
						
							|  |  |  | 			if (use_godot) | 
					
						
							|  |  |  | 				return get_environment("HOME")+"/.godot/app_userdata/"+an; | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 				return get_environment("HOME")+"/."+an; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return Globals::get_singleton()->get_resource_path(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-30 02:35:59 +01:00
										 |  |  | String OS_Unix::get_installed_templates_path() const { | 
					
						
							|  |  |  | 	String p=get_global_settings_path(); | 
					
						
							|  |  |  | 	if (p!="") | 
					
						
							|  |  |  | 		return p+"/templates/"; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		return ""; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | String OS_Unix::get_executable_path() const { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef __linux__
 | 
					
						
							|  |  |  | 	//fix for running from a symlink
 | 
					
						
							|  |  |  | 	char buf[256]; | 
					
						
							|  |  |  | 	memset(buf,0,256); | 
					
						
							|  |  |  | 	readlink("/proc/self/exe", buf, sizeof(buf)); | 
					
						
							| 
									
										
										
										
											2014-02-15 02:01:39 -03:00
										 |  |  | 	String b; | 
					
						
							|  |  |  | 	b.parse_utf8(buf); | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | 	if (b=="") { | 
					
						
							|  |  |  | 		WARN_PRINT("Couldn't get executable path from /proc/self/exe, using argv[0]"); | 
					
						
							|  |  |  | 		return OS::get_executable_path(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return b; | 
					
						
							| 
									
										
										
										
											2015-01-27 19:31:37 +10:30
										 |  |  | #elif defined(__FreeBSD__)
 | 
					
						
							|  |  |  | 	char resolved_path[MAXPATHLEN]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	realpath(OS::get_executable_path().utf8().get_data(), resolved_path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return String(resolved_path); | 
					
						
							| 
									
										
										
										
											2016-04-20 17:49:48 +10:00
										 |  |  | #elif defined(__APPLE__)
 | 
					
						
							|  |  |  | 	char temp_path[1]; | 
					
						
							|  |  |  | 	uint32_t buff_size=1; | 
					
						
							|  |  |  | 	_NSGetExecutablePath(temp_path, &buff_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	char* resolved_path = new char[buff_size + 1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (_NSGetExecutablePath(resolved_path, &buff_size) == 1) | 
					
						
							|  |  |  | 		WARN_PRINT("MAXPATHLEN is too small"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	String path(resolved_path); | 
					
						
							|  |  |  | 	delete[] resolved_path; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return path; | 
					
						
							| 
									
										
										
										
											2014-02-09 22:10:30 -03:00
										 |  |  | #else
 | 
					
						
							|  |  |  | 	ERR_PRINT("Warning, don't know how to obtain executable path on this OS! Please override this function properly."); | 
					
						
							|  |  |  | 	return OS::get_executable_path(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 |