| 
									
										
										
										
											2005-02-07 12:35:39 +00:00
										 |  |  | #ifndef QEMU_H
 | 
					
						
							|  |  |  | #define QEMU_H
 | 
					
						
							| 
									
										
										
										
											2003-02-18 22:55:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-12 18:47:46 +01:00
										 |  |  | #include "hostdep.h"
 | 
					
						
							| 
									
										
										
										
											2003-09-30 21:04:53 +00:00
										 |  |  | #include "cpu.h"
 | 
					
						
							| 
									
										
										
										
											2016-03-15 13:18:37 +01:00
										 |  |  | #include "exec/exec-all.h"
 | 
					
						
							| 
									
										
										
										
											2014-03-28 19:42:10 +01:00
										 |  |  | #include "exec/cpu_ldst.h"
 | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-12-24 13:47:52 +00:00
										 |  |  | #undef DEBUG_REMAP
 | 
					
						
							|  |  |  | #ifdef DEBUG_REMAP
 | 
					
						
							|  |  |  | #endif /* DEBUG_REMAP */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:19:49 +01:00
										 |  |  | #include "exec/user/abitypes.h"
 | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:19:49 +01:00
										 |  |  | #include "exec/user/thunk.h"
 | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  | #include "syscall_defs.h"
 | 
					
						
							| 
									
										
										
										
											2016-02-01 19:38:42 +01:00
										 |  |  | #include "target_syscall.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:19:49 +01:00
										 |  |  | #include "exec/gdbstub.h"
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:00 +01:00
										 |  |  | #include "qemu/queue.h"
 | 
					
						
							| 
									
										
										
										
											2003-03-23 01:06:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-07 20:50:51 +00:00
										 |  |  | #define THREAD __thread
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-14 12:49:18 +01:00
										 |  |  | /* This is the size of the host kernel's sigset_t, needed where we make
 | 
					
						
							|  |  |  |  * direct system calls that take a sigset_t pointer and a size. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define SIGSET_T_SIZE (_NSIG / 8)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-02-18 22:55:36 +00:00
										 |  |  | /* This struct is used to hold certain information about the image.
 | 
					
						
							|  |  |  |  * Basically, it replicates in user space what would be certain | 
					
						
							|  |  |  |  * task_struct fields in the kernel | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct image_info { | 
					
						
							| 
									
										
										
										
											2010-07-27 10:25:30 -07:00
										 |  |  |         abi_ulong       load_bias; | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  |         abi_ulong       load_addr; | 
					
						
							|  |  |  |         abi_ulong       start_code; | 
					
						
							|  |  |  |         abi_ulong       end_code; | 
					
						
							|  |  |  |         abi_ulong       start_data; | 
					
						
							|  |  |  |         abi_ulong       end_data; | 
					
						
							|  |  |  |         abi_ulong       start_brk; | 
					
						
							|  |  |  |         abi_ulong       brk; | 
					
						
							|  |  |  |         abi_ulong       start_mmap; | 
					
						
							|  |  |  |         abi_ulong       start_stack; | 
					
						
							| 
									
										
										
										
											2010-06-16 13:03:51 +01:00
										 |  |  |         abi_ulong       stack_limit; | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  |         abi_ulong       entry; | 
					
						
							|  |  |  |         abi_ulong       code_offset; | 
					
						
							|  |  |  |         abi_ulong       data_offset; | 
					
						
							| 
									
										
										
										
											2009-04-07 09:57:11 +03:00
										 |  |  |         abi_ulong       saved_auxv; | 
					
						
							| 
									
										
										
										
											2012-01-28 21:12:14 +02:00
										 |  |  |         abi_ulong       auxv_len; | 
					
						
							| 
									
										
										
										
											2009-04-07 09:57:11 +03:00
										 |  |  |         abi_ulong       arg_start; | 
					
						
							|  |  |  |         abi_ulong       arg_end; | 
					
						
							| 
									
										
										
										
											2012-03-30 18:02:50 +01:00
										 |  |  |         uint32_t        elf_flags; | 
					
						
							| 
									
										
										
										
											2003-02-18 22:55:36 +00:00
										 |  |  | 	int		personality; | 
					
						
							| 
									
										
										
										
											2011-02-07 01:05:50 -05:00
										 |  |  | #ifdef CONFIG_USE_FDPIC
 | 
					
						
							|  |  |  |         abi_ulong       loadmap_addr; | 
					
						
							|  |  |  |         uint16_t        nsegs; | 
					
						
							|  |  |  |         void           *loadsegs; | 
					
						
							|  |  |  |         abi_ulong       pt_dynamic_addr; | 
					
						
							|  |  |  |         struct image_info *other_info; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2003-02-18 22:55:36 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-06-15 20:05:50 +00:00
										 |  |  | #ifdef TARGET_I386
 | 
					
						
							| 
									
										
										
										
											2003-03-29 16:53:14 +00:00
										 |  |  | /* Information about the current linux thread */ | 
					
						
							|  |  |  | struct vm86_saved_state { | 
					
						
							|  |  |  |     uint32_t eax; /* return code */ | 
					
						
							|  |  |  |     uint32_t ebx; | 
					
						
							|  |  |  |     uint32_t ecx; | 
					
						
							|  |  |  |     uint32_t edx; | 
					
						
							|  |  |  |     uint32_t esi; | 
					
						
							|  |  |  |     uint32_t edi; | 
					
						
							|  |  |  |     uint32_t ebp; | 
					
						
							|  |  |  |     uint32_t esp; | 
					
						
							|  |  |  |     uint32_t eflags; | 
					
						
							|  |  |  |     uint32_t eip; | 
					
						
							|  |  |  |     uint16_t cs, ss, ds, es, fs, gs; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2003-06-15 20:05:50 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2003-03-29 16:53:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-03 20:12:17 +01:00
										 |  |  | #if defined(TARGET_ARM) && defined(TARGET_ABI32)
 | 
					
						
							| 
									
										
										
										
											2004-02-16 21:47:43 +00:00
										 |  |  | /* FPU emulator */ | 
					
						
							|  |  |  | #include "nwfpe/fpa11.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-31 16:11:38 +00:00
										 |  |  | #define MAX_SIGQUEUE_SIZE 1024
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct emulated_sigtable { | 
					
						
							|  |  |  |     int pending; /* true if signal is pending */ | 
					
						
							| 
									
										
										
										
											2016-05-27 15:51:52 +01:00
										 |  |  |     target_siginfo_t info; | 
					
						
							| 
									
										
										
										
											2008-05-31 16:11:38 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-29 16:53:14 +00:00
										 |  |  | /* NOTE: we force a big alignment so that the stack stored after is
 | 
					
						
							|  |  |  |    aligned too */ | 
					
						
							|  |  |  | typedef struct TaskState { | 
					
						
							| 
									
										
										
										
											2009-04-07 09:57:11 +03:00
										 |  |  |     pid_t ts_tid;     /* tid (or pid) of this task */ | 
					
						
							| 
									
										
										
										
											2004-02-16 21:47:43 +00:00
										 |  |  | #ifdef TARGET_ARM
 | 
					
						
							| 
									
										
										
										
											2013-09-03 20:12:17 +01:00
										 |  |  | # ifdef TARGET_ABI32
 | 
					
						
							| 
									
										
										
										
											2004-02-16 21:47:43 +00:00
										 |  |  |     /* FPA state */ | 
					
						
							|  |  |  |     FPA11 fpa; | 
					
						
							| 
									
										
										
										
											2013-09-03 20:12:17 +01:00
										 |  |  | # endif
 | 
					
						
							| 
									
										
										
										
											2005-04-23 18:25:41 +00:00
										 |  |  |     int swi_errno; | 
					
						
							| 
									
										
										
										
											2004-02-16 21:47:43 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-04-12 16:27:03 +08:00
										 |  |  | #ifdef TARGET_UNICORE32
 | 
					
						
							|  |  |  |     int swi_errno; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2007-04-06 08:56:50 +00:00
										 |  |  | #if defined(TARGET_I386) && !defined(TARGET_X86_64)
 | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  |     abi_ulong target_v86; | 
					
						
							| 
									
										
										
										
											2003-03-29 16:53:14 +00:00
										 |  |  |     struct vm86_saved_state vm86_saved_regs; | 
					
						
							| 
									
										
										
										
											2003-05-14 21:48:51 +00:00
										 |  |  |     struct target_vm86plus_struct vm86plus; | 
					
						
							| 
									
										
										
										
											2003-05-10 13:14:52 +00:00
										 |  |  |     uint32_t v86flags; | 
					
						
							|  |  |  |     uint32_t v86mask; | 
					
						
							| 
									
										
										
										
											2006-10-22 00:18:54 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-03-07 15:24:59 +00:00
										 |  |  |     abi_ulong child_tidptr; | 
					
						
							| 
									
										
										
										
											2006-10-22 00:18:54 +00:00
										 |  |  | #ifdef TARGET_M68K
 | 
					
						
							|  |  |  |     int sim_syscalls; | 
					
						
							| 
									
										
										
										
											2013-07-16 18:44:55 +01:00
										 |  |  |     abi_ulong tp_value; | 
					
						
							| 
									
										
										
										
											2007-05-26 15:09:38 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-04-12 16:27:03 +08:00
										 |  |  | #if defined(TARGET_ARM) || defined(TARGET_M68K) || defined(TARGET_UNICORE32)
 | 
					
						
							| 
									
										
										
										
											2007-05-26 15:09:38 +00:00
										 |  |  |     /* Extra fields for semihosted binaries.  */ | 
					
						
							| 
									
										
										
										
											2016-07-04 13:06:35 +01:00
										 |  |  |     abi_ulong heap_base; | 
					
						
							|  |  |  |     abi_ulong heap_limit; | 
					
						
							| 
									
										
										
										
											2003-06-15 20:05:50 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-07-04 13:06:35 +01:00
										 |  |  |     abi_ulong stack_base; | 
					
						
							| 
									
										
										
										
											2003-03-29 16:53:14 +00:00
										 |  |  |     int used; /* non zero if used */ | 
					
						
							| 
									
										
										
										
											2006-06-17 18:30:42 +00:00
										 |  |  |     struct image_info *info; | 
					
						
							| 
									
										
										
										
											2009-04-07 09:57:11 +03:00
										 |  |  |     struct linux_binprm *bprm; | 
					
						
							| 
									
										
										
										
											2008-05-31 16:11:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-27 15:51:53 +01:00
										 |  |  |     struct emulated_sigtable sync_signal; | 
					
						
							| 
									
										
										
										
											2008-05-31 16:11:38 +00:00
										 |  |  |     struct emulated_sigtable sigtab[TARGET_NSIG]; | 
					
						
							| 
									
										
										
										
											2016-05-27 15:51:49 +01:00
										 |  |  |     /* This thread's signal mask, as requested by the guest program.
 | 
					
						
							|  |  |  |      * The actual signal mask of this thread may differ: | 
					
						
							|  |  |  |      *  + we don't let SIGSEGV and SIGBUS be blocked while running guest code | 
					
						
							|  |  |  |      *  + sometimes we block all signals to avoid races | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     sigset_t signal_mask; | 
					
						
							|  |  |  |     /* The signal mask imposed by a guest sigsuspend syscall, if we are
 | 
					
						
							|  |  |  |      * currently in the middle of such a syscall | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     sigset_t sigsuspend_mask; | 
					
						
							|  |  |  |     /* Nonzero if we're leaving a sigsuspend and sigsuspend_mask is valid. */ | 
					
						
							|  |  |  |     int in_sigsuspend; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Nonzero if process_pending_signals() needs to do something (either
 | 
					
						
							|  |  |  |      * handle a pending signal or unblock signals). | 
					
						
							|  |  |  |      * This flag is written from a signal handler so should be accessed via | 
					
						
							|  |  |  |      * the atomic_read() and atomic_write() functions. (It is not accessed | 
					
						
							|  |  |  |      * from multiple threads.) | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     int signal_pending; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-03-29 16:53:14 +00:00
										 |  |  | } __attribute__((aligned(16))) TaskState; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-30 20:09:01 +00:00
										 |  |  | extern char *exec_path; | 
					
						
							| 
									
										
										
										
											2008-05-31 16:11:38 +00:00
										 |  |  | void init_task_state(TaskState *ts); | 
					
						
							| 
									
										
										
										
											2009-04-07 09:57:11 +03:00
										 |  |  | void task_settid(TaskState *); | 
					
						
							|  |  |  | void stop_all_tasks(void); | 
					
						
							| 
									
										
										
										
											2006-05-14 11:30:38 +00:00
										 |  |  | extern const char *qemu_uname_release; | 
					
						
							| 
									
										
										
										
											2009-07-17 12:48:08 +01:00
										 |  |  | extern unsigned long mmap_min_addr; | 
					
						
							| 
									
										
										
										
											2003-03-29 16:53:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-11 13:32:59 +00:00
										 |  |  | /* ??? See if we can avoid exposing so much of the loader internals.  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-27 10:25:30 -07:00
										 |  |  | /* Read a good amount of data initially, to hopefully get all the
 | 
					
						
							|  |  |  |    program headers loaded.  */ | 
					
						
							|  |  |  | #define BPRM_BUF_SIZE  1024
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-11 13:32:59 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2007-09-16 21:08:06 +00:00
										 |  |  |  * This structure is used to hold the arguments that are | 
					
						
							| 
									
										
										
										
											2006-06-11 13:32:59 +00:00
										 |  |  |  * used when loading binaries. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct linux_binprm { | 
					
						
							| 
									
										
										
										
											2010-07-27 10:25:30 -07:00
										 |  |  |         char buf[BPRM_BUF_SIZE] __attribute__((aligned)); | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  |         abi_ulong p; | 
					
						
							| 
									
										
										
										
											2006-06-11 13:32:59 +00:00
										 |  |  | 	int fd; | 
					
						
							|  |  |  |         int e_uid, e_gid; | 
					
						
							|  |  |  |         int argc, envc; | 
					
						
							|  |  |  |         char **argv; | 
					
						
							|  |  |  |         char **envp; | 
					
						
							|  |  |  |         char * filename;        /* Name of binary */ | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:32 +01:00
										 |  |  |         int (*core_dump)(int, const CPUArchState *); /* coredump routine */ | 
					
						
							| 
									
										
										
										
											2006-06-11 13:32:59 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void do_init_thread(struct target_pt_regs *regs, struct image_info *infop); | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  | abi_ulong loader_build_argptr(int envc, int argc, abi_ulong sp, | 
					
						
							|  |  |  |                               abi_ulong stringp, int push_ptr); | 
					
						
							| 
									
										
										
										
											2013-08-30 01:46:44 +02:00
										 |  |  | int loader_exec(int fdexec, const char *filename, char **argv, char **envp, | 
					
						
							| 
									
										
										
										
											2009-04-07 09:57:11 +03:00
										 |  |  |              struct target_pt_regs * regs, struct image_info *infop, | 
					
						
							|  |  |  |              struct linux_binprm *); | 
					
						
							| 
									
										
										
										
											2003-02-18 22:55:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-09 09:10:50 +00:00
										 |  |  | int load_elf_binary(struct linux_binprm *bprm, struct image_info *info); | 
					
						
							|  |  |  | int load_flt_binary(struct linux_binprm *bprm, struct image_info *info); | 
					
						
							| 
									
										
										
										
											2006-06-11 13:32:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | abi_long memcpy_to_target(abi_ulong dest, const void *src, | 
					
						
							|  |  |  |                           unsigned long len); | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  | void target_set_brk(abi_ulong new_brk); | 
					
						
							|  |  |  | abi_long do_brk(abi_ulong new_brk); | 
					
						
							| 
									
										
										
										
											2003-02-18 22:55:36 +00:00
										 |  |  | void syscall_init(void); | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  | abi_long do_syscall(void *cpu_env, int num, abi_long arg1, | 
					
						
							|  |  |  |                     abi_long arg2, abi_long arg3, abi_long arg4, | 
					
						
							| 
									
										
										
										
											2011-06-16 17:37:13 +01:00
										 |  |  |                     abi_long arg5, abi_long arg6, abi_long arg7, | 
					
						
							|  |  |  |                     abi_long arg8); | 
					
						
							| 
									
										
										
										
											2010-09-23 21:28:03 +02:00
										 |  |  | void gemu_log(const char *fmt, ...) GCC_FMT_ATTR(1, 2); | 
					
						
							| 
									
										
										
										
											2013-06-09 19:47:04 +02:00
										 |  |  | extern THREAD CPUState *thread_cpu; | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:32 +01:00
										 |  |  | void cpu_loop(CPUArchState *env); | 
					
						
							| 
									
										
										
										
											2016-06-06 19:58:18 +01:00
										 |  |  | const char *target_strerror(int err); | 
					
						
							| 
									
										
										
										
											2008-05-06 15:36:17 +00:00
										 |  |  | int get_osversion(void); | 
					
						
							| 
									
										
										
										
											2013-09-03 20:12:20 +01:00
										 |  |  | void init_qemu_uname_release(void); | 
					
						
							| 
									
										
										
										
											2008-06-07 20:50:51 +00:00
										 |  |  | void fork_start(void); | 
					
						
							|  |  |  | void fork_end(int child); | 
					
						
							| 
									
										
										
										
											2003-04-29 20:39:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-26 16:50:01 +00:00
										 |  |  | /* Creates the initial guest address space in the host memory space using
 | 
					
						
							|  |  |  |  * the given host start address hint and size.  The guest_start parameter | 
					
						
							|  |  |  |  * specifies the start address of the guest space.  guest_base will be the | 
					
						
							|  |  |  |  * difference between the host start address computed by this function and | 
					
						
							|  |  |  |  * guest_start.  If fixed is specified, then the mapped address space must | 
					
						
							|  |  |  |  * start at host_start.  The real start address of the mapped memory space is | 
					
						
							|  |  |  |  * returned or -1 if there was an error. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | unsigned long init_guest_space(unsigned long host_start, | 
					
						
							|  |  |  |                                unsigned long host_size, | 
					
						
							|  |  |  |                                unsigned long guest_start, | 
					
						
							|  |  |  |                                bool fixed); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-17 18:20:00 +01:00
										 |  |  | #include "qemu/log.h"
 | 
					
						
							| 
									
										
										
										
											2003-05-10 13:14:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-12 18:47:46 +01:00
										 |  |  | /* safe_syscall.S */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * safe_syscall: | 
					
						
							|  |  |  |  * @int number: number of system call to make | 
					
						
							|  |  |  |  * ...: arguments to the system call | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Call a system call if guest signal not pending. | 
					
						
							|  |  |  |  * This has the same API as the libc syscall() function, except that it | 
					
						
							|  |  |  |  * may return -1 with errno == TARGET_ERESTARTSYS if a signal was pending. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns: the system call result, or -1 with an error code in errno | 
					
						
							|  |  |  |  * (Errnos are host errnos; we rely on TARGET_ERESTARTSYS not clashing | 
					
						
							|  |  |  |  * with any of the host errno values.) | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* A guide to using safe_syscall() to handle interactions between guest
 | 
					
						
							|  |  |  |  * syscalls and guest signals: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Guest syscalls come in two flavours: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * (1) Non-interruptible syscalls | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * These are guest syscalls that never get interrupted by signals and | 
					
						
							|  |  |  |  * so never return EINTR. They can be implemented straightforwardly in | 
					
						
							|  |  |  |  * QEMU: just make sure that if the implementation code has to make any | 
					
						
							|  |  |  |  * blocking calls that those calls are retried if they return EINTR. | 
					
						
							|  |  |  |  * It's also OK to implement these with safe_syscall, though it will be | 
					
						
							|  |  |  |  * a little less efficient if a signal is delivered at the 'wrong' moment. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-05-27 15:51:49 +01:00
										 |  |  |  * Some non-interruptible syscalls need to be handled using block_signals() | 
					
						
							|  |  |  |  * to block signals for the duration of the syscall. This mainly applies | 
					
						
							|  |  |  |  * to code which needs to modify the data structures used by the | 
					
						
							|  |  |  |  * host_signal_handler() function and the functions it calls, including | 
					
						
							|  |  |  |  * all syscalls which change the thread's signal mask. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-05-12 18:47:46 +01:00
										 |  |  |  * (2) Interruptible syscalls | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * These are guest syscalls that can be interrupted by signals and | 
					
						
							|  |  |  |  * for which we need to either return EINTR or arrange for the guest | 
					
						
							|  |  |  |  * syscall to be restarted. This category includes both syscalls which | 
					
						
							|  |  |  |  * always restart (and in the kernel return -ERESTARTNOINTR), ones | 
					
						
							|  |  |  |  * which only restart if there is no handler (kernel returns -ERESTARTNOHAND | 
					
						
							|  |  |  |  * or -ERESTART_RESTARTBLOCK), and the most common kind which restart | 
					
						
							|  |  |  |  * if the handler was registered with SA_RESTART (kernel returns | 
					
						
							|  |  |  |  * -ERESTARTSYS). System calls which are only interruptible in some | 
					
						
							|  |  |  |  * situations (like 'open') also need to be handled this way. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Here it is important that the host syscall is made | 
					
						
							|  |  |  |  * via this safe_syscall() function, and *not* via the host libc. | 
					
						
							|  |  |  |  * If the host libc is used then the implementation will appear to work | 
					
						
							|  |  |  |  * most of the time, but there will be a race condition where a | 
					
						
							|  |  |  |  * signal could arrive just before we make the host syscall inside libc, | 
					
						
							|  |  |  |  * and then then guest syscall will not correctly be interrupted. | 
					
						
							|  |  |  |  * Instead the implementation of the guest syscall can use the safe_syscall | 
					
						
							|  |  |  |  * function but otherwise just return the result or errno in the usual | 
					
						
							|  |  |  |  * way; the main loop code will take care of restarting the syscall | 
					
						
							|  |  |  |  * if appropriate. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * (If the implementation needs to make multiple host syscalls this is | 
					
						
							|  |  |  |  * OK; any which might really block must be via safe_syscall(); for those | 
					
						
							|  |  |  |  * which are only technically blocking (ie which we know in practice won't | 
					
						
							|  |  |  |  * stay in the host kernel indefinitely) it's OK to use libc if necessary. | 
					
						
							|  |  |  |  * You must be able to cope with backing out correctly if some safe_syscall | 
					
						
							|  |  |  |  * you make in the implementation returns either -TARGET_ERESTARTSYS or | 
					
						
							|  |  |  |  * EINTR though.) | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-05-27 15:51:49 +01:00
										 |  |  |  * block_signals() cannot be used for interruptible syscalls. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-05-12 18:47:46 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * How and why the safe_syscall implementation works: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The basic setup is that we make the host syscall via a known | 
					
						
							|  |  |  |  * section of host native assembly. If a signal occurs, our signal | 
					
						
							|  |  |  |  * handler checks the interrupted host PC against the addresse of that | 
					
						
							|  |  |  |  * known section. If the PC is before or at the address of the syscall | 
					
						
							|  |  |  |  * instruction then we change the PC to point at a "return | 
					
						
							|  |  |  |  * -TARGET_ERESTARTSYS" code path instead, and then exit the signal handler | 
					
						
							|  |  |  |  * (causing the safe_syscall() call to immediately return that value). | 
					
						
							|  |  |  |  * Then in the main.c loop if we see this magic return value we adjust | 
					
						
							|  |  |  |  * the guest PC to wind it back to before the system call, and invoke | 
					
						
							|  |  |  |  * the guest signal handler as usual. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This winding-back will happen in two cases: | 
					
						
							|  |  |  |  * (1) signal came in just before we took the host syscall (a race); | 
					
						
							|  |  |  |  *   in this case we'll take the guest signal and have another go | 
					
						
							|  |  |  |  *   at the syscall afterwards, and this is indistinguishable for the | 
					
						
							|  |  |  |  *   guest from the timing having been different such that the guest | 
					
						
							|  |  |  |  *   signal really did win the race | 
					
						
							|  |  |  |  * (2) signal came in while the host syscall was blocking, and the | 
					
						
							|  |  |  |  *   host kernel decided the syscall should be restarted; | 
					
						
							|  |  |  |  *   in this case we want to restart the guest syscall also, and so | 
					
						
							|  |  |  |  *   rewinding is the right thing. (Note that "restart" semantics mean | 
					
						
							|  |  |  |  *   "first call the signal handler, then reattempt the syscall".) | 
					
						
							|  |  |  |  * The other situation to consider is when a signal came in while the | 
					
						
							|  |  |  |  * host syscall was blocking, and the host kernel decided that the syscall | 
					
						
							|  |  |  |  * should not be restarted; in this case QEMU's host signal handler will | 
					
						
							|  |  |  |  * be invoked with the PC pointing just after the syscall instruction, | 
					
						
							|  |  |  |  * with registers indicating an EINTR return; the special code in the | 
					
						
							|  |  |  |  * handler will not kick in, and we will return EINTR to the guest as | 
					
						
							|  |  |  |  * we should. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Notice that we can leave the host kernel to make the decision for | 
					
						
							|  |  |  |  * us about whether to do a restart of the syscall or not; we do not | 
					
						
							|  |  |  |  * need to check SA_RESTART flags in QEMU or distinguish the various | 
					
						
							|  |  |  |  * kinds of restartability. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #ifdef HAVE_SAFE_SYSCALL
 | 
					
						
							|  |  |  | /* The core part of this function is implemented in assembly */ | 
					
						
							|  |  |  | extern long safe_syscall_base(int *pending, long number, ...); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define safe_syscall(...)                                               \
 | 
					
						
							|  |  |  |     ({                                                                  \ | 
					
						
							|  |  |  |         long ret_;                                                      \ | 
					
						
							|  |  |  |         int *psp_ = &((TaskState *)thread_cpu->opaque)->signal_pending; \ | 
					
						
							|  |  |  |         ret_ = safe_syscall_base(psp_, __VA_ARGS__);                    \ | 
					
						
							|  |  |  |         if (is_error(ret_)) {                                           \ | 
					
						
							|  |  |  |             errno = -ret_;                                              \ | 
					
						
							|  |  |  |             ret_ = -1;                                                  \ | 
					
						
							|  |  |  |         }                                                               \ | 
					
						
							|  |  |  |         ret_;                                                           \ | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Fallback for architectures which don't yet provide a safe-syscall assembly
 | 
					
						
							|  |  |  |  * fragment; note that this is racy! | 
					
						
							|  |  |  |  * This should go away when all host architectures have been updated. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define safe_syscall syscall
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-15 11:34:20 -07:00
										 |  |  | /* syscall.c */ | 
					
						
							|  |  |  | int host_to_target_waitstatus(int status); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-01 00:07:38 +00:00
										 |  |  | /* strace.c */ | 
					
						
							|  |  |  | void print_syscall(int num, | 
					
						
							| 
									
										
										
										
											2007-11-11 17:23:29 +00:00
										 |  |  |                    abi_long arg1, abi_long arg2, abi_long arg3, | 
					
						
							|  |  |  |                    abi_long arg4, abi_long arg5, abi_long arg6); | 
					
						
							|  |  |  | void print_syscall_ret(int num, abi_long arg1); | 
					
						
							| 
									
										
										
										
											2007-11-01 00:07:38 +00:00
										 |  |  | extern int do_strace; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-06-15 20:05:50 +00:00
										 |  |  | /* signal.c */ | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:32 +01:00
										 |  |  | void process_pending_signals(CPUArchState *cpu_env); | 
					
						
							| 
									
										
										
										
											2003-06-15 20:05:50 +00:00
										 |  |  | void signal_init(void); | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:32 +01:00
										 |  |  | int queue_signal(CPUArchState *env, int sig, target_siginfo_t *info); | 
					
						
							| 
									
										
										
										
											2009-10-01 16:12:16 -05:00
										 |  |  | void host_to_target_siginfo(target_siginfo_t *tinfo, const siginfo_t *info); | 
					
						
							|  |  |  | void target_to_host_siginfo(siginfo_t *info, const target_siginfo_t *tinfo); | 
					
						
							| 
									
										
										
										
											2008-05-30 18:05:19 +00:00
										 |  |  | int target_to_host_signal(int sig); | 
					
						
							| 
									
										
										
										
											2009-04-16 15:17:02 +00:00
										 |  |  | int host_to_target_signal(int sig); | 
					
						
							| 
									
										
										
										
											2012-03-14 01:38:32 +01:00
										 |  |  | long do_sigreturn(CPUArchState *env); | 
					
						
							|  |  |  | long do_rt_sigreturn(CPUArchState *env); | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | abi_long do_sigaltstack(abi_ulong uss_addr, abi_ulong uoss_addr, abi_ulong sp); | 
					
						
							| 
									
										
										
										
											2014-03-14 14:36:55 +00:00
										 |  |  | int do_sigprocmask(int how, const sigset_t *set, sigset_t *oldset); | 
					
						
							| 
									
										
										
										
											2016-05-27 15:51:49 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * block_signals: block all signals while handling this guest syscall | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Block all signals, and arrange that the signal mask is returned to | 
					
						
							|  |  |  |  * its correct value for the guest before we resume execution of guest code. | 
					
						
							|  |  |  |  * If this function returns non-zero, then the caller should immediately | 
					
						
							|  |  |  |  * return -TARGET_ERESTARTSYS to the main loop, which will take the pending | 
					
						
							|  |  |  |  * signal and restart execution of the syscall. | 
					
						
							|  |  |  |  * If block_signals() returns zero, then the caller can continue with | 
					
						
							|  |  |  |  * emulation of the system call knowing that no signals can be taken | 
					
						
							|  |  |  |  * (and therefore that no race conditions will result). | 
					
						
							|  |  |  |  * This should only be called once, because if it is called a second time | 
					
						
							|  |  |  |  * it will always return non-zero. (Think of it like a mutex that can't | 
					
						
							|  |  |  |  * be recursively locked.) | 
					
						
							|  |  |  |  * Signals will be unblocked again by process_pending_signals(). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Return value: non-zero if there was a pending signal, zero if not. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int block_signals(void); /* Returns non zero if signal pending */ | 
					
						
							| 
									
										
										
										
											2003-06-15 20:05:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef TARGET_I386
 | 
					
						
							| 
									
										
										
										
											2003-05-10 13:14:52 +00:00
										 |  |  | /* vm86.c */ | 
					
						
							|  |  |  | void save_v86_state(CPUX86State *env); | 
					
						
							| 
									
										
										
										
											2003-05-10 15:10:36 +00:00
										 |  |  | void handle_vm86_trap(CPUX86State *env, int trapno); | 
					
						
							| 
									
										
										
										
											2003-05-10 13:14:52 +00:00
										 |  |  | void handle_vm86_fault(CPUX86State *env); | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  | int do_vm86(CPUX86State *env, long subfunction, abi_ulong v86_addr); | 
					
						
							| 
									
										
										
										
											2007-10-05 17:01:51 +00:00
										 |  |  | #elif defined(TARGET_SPARC64)
 | 
					
						
							|  |  |  | void sparc64_set_context(CPUSPARCState *env); | 
					
						
							|  |  |  | void sparc64_get_context(CPUSPARCState *env); | 
					
						
							| 
									
										
										
										
											2003-06-15 20:05:50 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2003-05-10 13:14:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2003-05-13 00:25:15 +00:00
										 |  |  | /* mmap.c */ | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  | int target_mprotect(abi_ulong start, abi_ulong len, int prot); | 
					
						
							|  |  |  | abi_long target_mmap(abi_ulong start, abi_ulong len, int prot, | 
					
						
							|  |  |  |                      int flags, int fd, abi_ulong offset); | 
					
						
							|  |  |  | int target_munmap(abi_ulong start, abi_ulong len); | 
					
						
							|  |  |  | abi_long target_mremap(abi_ulong old_addr, abi_ulong old_size, | 
					
						
							|  |  |  |                        abi_ulong new_size, unsigned long flags, | 
					
						
							|  |  |  |                        abi_ulong new_addr); | 
					
						
							|  |  |  | int target_msync(abi_ulong start, abi_ulong len, int flags); | 
					
						
							| 
									
										
										
										
											2008-05-31 16:33:53 +00:00
										 |  |  | extern unsigned long last_brk; | 
					
						
							| 
									
										
										
										
											2012-03-08 14:40:33 +00:00
										 |  |  | extern abi_ulong mmap_next_start; | 
					
						
							| 
									
										
										
										
											2009-04-21 17:23:23 +03:00
										 |  |  | abi_ulong mmap_find_vma(abi_ulong, abi_ulong); | 
					
						
							| 
									
										
										
										
											2009-03-07 15:24:59 +00:00
										 |  |  | void cpu_list_lock(void); | 
					
						
							|  |  |  | void cpu_list_unlock(void); | 
					
						
							| 
									
										
										
										
											2008-06-07 20:50:51 +00:00
										 |  |  | void mmap_fork_start(void); | 
					
						
							|  |  |  | void mmap_fork_end(int child); | 
					
						
							| 
									
										
										
										
											2003-05-13 00:25:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-10-05 11:05:14 +00:00
										 |  |  | /* main.c */ | 
					
						
							| 
									
										
										
										
											2010-03-19 14:21:13 -07:00
										 |  |  | extern unsigned long guest_stack_size; | 
					
						
							| 
									
										
										
										
											2008-10-05 11:05:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | /* user access */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define VERIFY_READ 0
 | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | #define VERIFY_WRITE 1 /* implies read access */
 | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-14 10:51:00 +00:00
										 |  |  | static inline int access_ok(int type, abi_ulong addr, abi_ulong size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return page_check_range((target_ulong)addr, size, | 
					
						
							|  |  |  |                             (type == VERIFY_READ) ? PAGE_READ : (PAGE_READ | PAGE_WRITE)) == 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-04 16:39:31 -08:00
										 |  |  | /* NOTE __get_user and __put_user use host pointers and don't check access.
 | 
					
						
							|  |  |  |    These are usually used to access struct data members once the struct has | 
					
						
							|  |  |  |    been locked - usually with lock_user_struct.  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Tricky points:
 | 
					
						
							|  |  |  |    - Use __builtin_choose_expr to avoid type promotion from ?:, | 
					
						
							|  |  |  |    - Invalid sizes result in a compile time error stemming from | 
					
						
							|  |  |  |      the fact that abort has no parameters. | 
					
						
							|  |  |  |    - It's easier to use the endian-specific unaligned load/store | 
					
						
							|  |  |  |      functions than host-endian unaligned load/store plus tswapN.  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define __put_user_e(x, hptr, e)                                        \
 | 
					
						
							|  |  |  |   (__builtin_choose_expr(sizeof(*(hptr)) == 1, stb_p,                   \ | 
					
						
							|  |  |  |    __builtin_choose_expr(sizeof(*(hptr)) == 2, stw_##e##_p,             \ | 
					
						
							|  |  |  |    __builtin_choose_expr(sizeof(*(hptr)) == 4, stl_##e##_p,             \ | 
					
						
							|  |  |  |    __builtin_choose_expr(sizeof(*(hptr)) == 8, stq_##e##_p, abort))))   \ | 
					
						
							| 
									
										
										
										
											2014-04-22 15:40:50 +03:00
										 |  |  |      ((hptr), (x)), (void)0) | 
					
						
							| 
									
										
										
										
											2013-01-04 16:39:31 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define __get_user_e(x, hptr, e)                                        \
 | 
					
						
							| 
									
										
										
										
											2013-01-31 12:50:40 +00:00
										 |  |  |   ((x) = (typeof(*hptr))(                                               \ | 
					
						
							| 
									
										
										
										
											2013-01-04 16:39:31 -08:00
										 |  |  |    __builtin_choose_expr(sizeof(*(hptr)) == 1, ldub_p,                  \ | 
					
						
							|  |  |  |    __builtin_choose_expr(sizeof(*(hptr)) == 2, lduw_##e##_p,            \ | 
					
						
							|  |  |  |    __builtin_choose_expr(sizeof(*(hptr)) == 4, ldl_##e##_p,             \ | 
					
						
							|  |  |  |    __builtin_choose_expr(sizeof(*(hptr)) == 8, ldq_##e##_p, abort))))   \ | 
					
						
							| 
									
										
										
										
											2014-04-22 15:40:50 +03:00
										 |  |  |      (hptr)), (void)0) | 
					
						
							| 
									
										
										
										
											2013-01-04 16:39:31 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef TARGET_WORDS_BIGENDIAN
 | 
					
						
							|  |  |  | # define __put_user(x, hptr)  __put_user_e(x, hptr, be)
 | 
					
						
							|  |  |  | # define __get_user(x, hptr)  __get_user_e(x, hptr, be)
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | # define __put_user(x, hptr)  __put_user_e(x, hptr, le)
 | 
					
						
							|  |  |  | # define __get_user(x, hptr)  __get_user_e(x, hptr, le)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | /* put_user()/get_user() take a guest address and check access */ | 
					
						
							|  |  |  | /* These are usually used to access an atomic data type, such as an int,
 | 
					
						
							|  |  |  |  * that has been passed by address.  These internally perform locking | 
					
						
							|  |  |  |  * and unlocking on the data type. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #define put_user(x, gaddr, target_type)					\
 | 
					
						
							|  |  |  | ({									\ | 
					
						
							|  |  |  |     abi_ulong __gaddr = (gaddr);					\ | 
					
						
							|  |  |  |     target_type *__hptr;						\ | 
					
						
							| 
									
										
										
										
											2014-04-22 15:40:50 +03:00
										 |  |  |     abi_long __ret = 0;							\ | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |     if ((__hptr = lock_user(VERIFY_WRITE, __gaddr, sizeof(target_type), 0))) { \ | 
					
						
							| 
									
										
										
										
											2014-04-22 15:40:50 +03:00
										 |  |  |         __put_user((x), __hptr);				\ | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |         unlock_user(__hptr, __gaddr, sizeof(target_type));		\ | 
					
						
							|  |  |  |     } else								\ | 
					
						
							|  |  |  |         __ret = -TARGET_EFAULT;						\ | 
					
						
							|  |  |  |     __ret;								\ | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | }) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | #define get_user(x, gaddr, target_type)					\
 | 
					
						
							|  |  |  | ({									\ | 
					
						
							|  |  |  |     abi_ulong __gaddr = (gaddr);					\ | 
					
						
							|  |  |  |     target_type *__hptr;						\ | 
					
						
							| 
									
										
										
										
											2014-04-22 15:40:50 +03:00
										 |  |  |     abi_long __ret = 0;							\ | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |     if ((__hptr = lock_user(VERIFY_READ, __gaddr, sizeof(target_type), 1))) { \ | 
					
						
							| 
									
										
										
										
											2014-04-22 15:40:50 +03:00
										 |  |  |         __get_user((x), __hptr);				\ | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |         unlock_user(__hptr, __gaddr, 0);				\ | 
					
						
							| 
									
										
										
										
											2007-11-16 10:46:05 +00:00
										 |  |  |     } else {								\ | 
					
						
							|  |  |  |         /* avoid warning */						\ | 
					
						
							|  |  |  |         (x) = 0;							\ | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |         __ret = -TARGET_EFAULT;						\ | 
					
						
							| 
									
										
										
										
											2007-11-16 10:46:05 +00:00
										 |  |  |     }									\ | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |     __ret;								\ | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | }) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-16 10:46:05 +00:00
										 |  |  | #define put_user_ual(x, gaddr) put_user((x), (gaddr), abi_ulong)
 | 
					
						
							|  |  |  | #define put_user_sal(x, gaddr) put_user((x), (gaddr), abi_long)
 | 
					
						
							|  |  |  | #define put_user_u64(x, gaddr) put_user((x), (gaddr), uint64_t)
 | 
					
						
							|  |  |  | #define put_user_s64(x, gaddr) put_user((x), (gaddr), int64_t)
 | 
					
						
							|  |  |  | #define put_user_u32(x, gaddr) put_user((x), (gaddr), uint32_t)
 | 
					
						
							|  |  |  | #define put_user_s32(x, gaddr) put_user((x), (gaddr), int32_t)
 | 
					
						
							|  |  |  | #define put_user_u16(x, gaddr) put_user((x), (gaddr), uint16_t)
 | 
					
						
							|  |  |  | #define put_user_s16(x, gaddr) put_user((x), (gaddr), int16_t)
 | 
					
						
							|  |  |  | #define put_user_u8(x, gaddr)  put_user((x), (gaddr), uint8_t)
 | 
					
						
							|  |  |  | #define put_user_s8(x, gaddr)  put_user((x), (gaddr), int8_t)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define get_user_ual(x, gaddr) get_user((x), (gaddr), abi_ulong)
 | 
					
						
							|  |  |  | #define get_user_sal(x, gaddr) get_user((x), (gaddr), abi_long)
 | 
					
						
							|  |  |  | #define get_user_u64(x, gaddr) get_user((x), (gaddr), uint64_t)
 | 
					
						
							|  |  |  | #define get_user_s64(x, gaddr) get_user((x), (gaddr), int64_t)
 | 
					
						
							|  |  |  | #define get_user_u32(x, gaddr) get_user((x), (gaddr), uint32_t)
 | 
					
						
							|  |  |  | #define get_user_s32(x, gaddr) get_user((x), (gaddr), int32_t)
 | 
					
						
							|  |  |  | #define get_user_u16(x, gaddr) get_user((x), (gaddr), uint16_t)
 | 
					
						
							|  |  |  | #define get_user_s16(x, gaddr) get_user((x), (gaddr), int16_t)
 | 
					
						
							|  |  |  | #define get_user_u8(x, gaddr)  get_user((x), (gaddr), uint8_t)
 | 
					
						
							|  |  |  | #define get_user_s8(x, gaddr)  get_user((x), (gaddr), int8_t)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | /* copy_from_user() and copy_to_user() are usually used to copy data
 | 
					
						
							|  |  |  |  * buffers between the target and host.  These internally perform | 
					
						
							|  |  |  |  * locking/unlocking of the memory. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | abi_long copy_from_user(void *hptr, abi_ulong gaddr, size_t len); | 
					
						
							|  |  |  | abi_long copy_to_user(abi_ulong gaddr, void *hptr, size_t len); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  | /* Functions for accessing guest memory.  The tget and tput functions
 | 
					
						
							| 
									
										
										
										
											2013-09-12 19:57:15 +02:00
										 |  |  |    read/write single values, byteswapping as necessary.  The lock_user function | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  |    gets a pointer to a contiguous area of guest memory, but does not perform | 
					
						
							| 
									
										
										
										
											2013-09-12 19:57:15 +02:00
										 |  |  |    any byteswapping.  lock_user may return either a pointer to the guest | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  |    memory, or a temporary buffer.  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Lock an area of guest memory into the host.  If copy is true then the
 | 
					
						
							|  |  |  |    host area will have the same contents as the guest.  */ | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | static inline void *lock_user(int type, abi_ulong guest_addr, long len, int copy) | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |     if (!access_ok(type, guest_addr, len)) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  | #ifdef DEBUG_REMAP
 | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |         void *addr; | 
					
						
							|  |  |  |         addr = malloc(len); | 
					
						
							|  |  |  |         if (copy) | 
					
						
							|  |  |  |             memcpy(addr, g2h(guest_addr), len); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             memset(addr, 0, len); | 
					
						
							|  |  |  |         return addr; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     return g2h(guest_addr); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | /* Unlock an area of guest memory.  The first LEN bytes must be
 | 
					
						
							| 
									
										
										
										
											2008-06-03 19:51:57 +00:00
										 |  |  |    flushed back to guest memory. host_ptr = NULL is explicitly | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |    allowed and does nothing. */ | 
					
						
							|  |  |  | static inline void unlock_user(void *host_ptr, abi_ulong guest_addr, | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  |                                long len) | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  | #ifdef DEBUG_REMAP
 | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |     if (!host_ptr) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     if (host_ptr == g2h(guest_addr)) | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  |         return; | 
					
						
							|  |  |  |     if (len > 0) | 
					
						
							| 
									
										
										
										
											2007-12-24 13:47:52 +00:00
										 |  |  |         memcpy(g2h(guest_addr), host_ptr, len); | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |     free(host_ptr); | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | /* Return the length of a string in target memory or -TARGET_EFAULT if
 | 
					
						
							|  |  |  |    access error. */ | 
					
						
							|  |  |  | abi_long target_strlen(abi_ulong gaddr); | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Like lock_user but for null terminated strings.  */ | 
					
						
							| 
									
										
										
										
											2007-10-14 16:27:31 +00:00
										 |  |  | static inline void *lock_user_string(abi_ulong guest_addr) | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  |     abi_long len; | 
					
						
							|  |  |  |     len = target_strlen(guest_addr); | 
					
						
							|  |  |  |     if (len < 0) | 
					
						
							|  |  |  |         return NULL; | 
					
						
							|  |  |  |     return lock_user(VERIFY_READ, guest_addr, (long)(len + 1), 1); | 
					
						
							| 
									
										
										
										
											2004-02-22 13:40:13 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-12 19:57:41 +02:00
										 |  |  | /* Helper macros for locking/unlocking a target struct.  */ | 
					
						
							| 
									
										
										
										
											2007-11-11 14:26:47 +00:00
										 |  |  | #define lock_user_struct(type, host_ptr, guest_addr, copy)	\
 | 
					
						
							|  |  |  |     (host_ptr = lock_user(type, guest_addr, sizeof(*host_ptr), copy)) | 
					
						
							|  |  |  | #define unlock_user_struct(host_ptr, guest_addr, copy)		\
 | 
					
						
							| 
									
										
										
										
											2006-03-25 19:31:22 +00:00
										 |  |  |     unlock_user(host_ptr, guest_addr, (copy) ? sizeof(*host_ptr) : 0) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-06-02 16:16:42 +00:00
										 |  |  | #include <pthread.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-16 18:44:52 +01:00
										 |  |  | /* Include target-specific struct and function definitions;
 | 
					
						
							|  |  |  |  * they may need access to the target-independent structures | 
					
						
							|  |  |  |  * above, so include them last. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include "target_cpu.h"
 | 
					
						
							|  |  |  | #include "target_signal.h"
 | 
					
						
							| 
									
										
										
										
											2013-10-30 14:46:31 +01:00
										 |  |  | #include "target_structs.h"
 | 
					
						
							| 
									
										
										
										
											2013-07-16 18:44:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-02-07 12:35:39 +00:00
										 |  |  | #endif /* QEMU_H */
 |