| 
									
										
										
										
											2006-04-10 22:53:26 -07:00
										 |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											2006-03-31 02:30:25 -08:00
										 |  |  | #include <linux/unistd.h>
 | 
					
						
							| 
									
										
										
										
											2006-10-29 22:46:41 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-18 20:09:49 +01:00
										 |  |  | #include <sys/ptrace.h>
 | 
					
						
							| 
									
										
										
										
											2006-10-02 02:18:37 -07:00
										 |  |  | #include <sys/syscall.h>
 | 
					
						
							| 
									
										
										
										
											2006-10-29 22:46:41 -08:00
										 |  |  | #include <unistd.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-08 03:27:32 +01:00
										 |  |  | #include <sysdep/tls.h>
 | 
					
						
							| 
									
										
										
										
											2006-03-31 02:30:25 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-18 20:09:49 +01:00
										 |  |  | #ifndef PTRACE_GET_THREAD_AREA
 | 
					
						
							|  |  |  | #define PTRACE_GET_THREAD_AREA 25
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef PTRACE_SET_THREAD_AREA
 | 
					
						
							|  |  |  | #define PTRACE_SET_THREAD_AREA 26
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-03-31 02:30:25 -08:00
										 |  |  | /* Checks whether host supports TLS, and sets *tls_min according to the value
 | 
					
						
							|  |  |  |  * valid on the host. | 
					
						
							|  |  |  |  * i386 host have it == 6; x86_64 host have it == 12, for i386 emulation. */ | 
					
						
							| 
									
										
										
										
											2011-08-18 20:09:49 +01:00
										 |  |  | void check_host_supports_tls(int *supports_tls, int *tls_min) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-03-31 02:30:25 -08:00
										 |  |  | 	/* Values for x86 and x86_64.*/ | 
					
						
							|  |  |  | 	int val[] = {GDT_ENTRY_TLS_MIN_I386, GDT_ENTRY_TLS_MIN_X86_64}; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (i = 0; i < ARRAY_SIZE(val); i++) { | 
					
						
							|  |  |  | 		user_desc_t info; | 
					
						
							|  |  |  | 		info.entry_number = val[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-02 02:18:37 -07:00
										 |  |  | 		if (syscall(__NR_get_thread_area, &info) == 0) { | 
					
						
							| 
									
										
										
										
											2006-03-31 02:30:25 -08:00
										 |  |  | 			*tls_min = val[i]; | 
					
						
							|  |  |  | 			*supports_tls = 1; | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			if (errno == EINVAL) | 
					
						
							|  |  |  | 				continue; | 
					
						
							|  |  |  | 			else if (errno == ENOSYS) | 
					
						
							|  |  |  | 				*supports_tls = 0; | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	*supports_tls = 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-08-18 20:09:49 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | int os_set_thread_area(user_desc_t *info, int pid) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = ptrace(PTRACE_SET_THREAD_AREA, pid, info->entry_number, | 
					
						
							|  |  |  | 		     (unsigned long) info); | 
					
						
							|  |  |  | 	if (ret < 0) | 
					
						
							|  |  |  | 		ret = -errno; | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int os_get_thread_area(user_desc_t *info, int pid) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ret = ptrace(PTRACE_GET_THREAD_AREA, pid, info->entry_number, | 
					
						
							|  |  |  | 		     (unsigned long) info); | 
					
						
							|  |  |  | 	if (ret < 0) | 
					
						
							|  |  |  | 		ret = -errno; | 
					
						
							|  |  |  | 	return ret; | 
					
						
							|  |  |  | } |