linux-user: Fix parse_elf_properties GNU0_MAGIC check

linux-user: Various improvements to strace
 linux-user: Add openat2 support
 linux-user/flatload: Take mmap_lock in load_flt_binary()
 accel/tcg: Make page_set_flags() documentation public
 tcg/ppc: Use TCG_REG_TMP2 for scratch tcg_out_qemu_st
 tcg/ppc: Use TCG_REG_TMP2 for scratch index in prepare_host_addr
 target/m68k: Always return a temporary from gen_lea_mode
 -----BEGIN PGP SIGNATURE-----
 
 iQFRBAABCgA7FiEEekgeeIaLTbaoWgXAZN846K9+IV8FAmcFeLQdHHJpY2hhcmQu
 aGVuZGVyc29uQGxpbmFyby5vcmcACgkQZN846K9+IV+8Wgf9FJb/71zPAg7idUo8
 k+P5hOmE0PzPKdkoCh5WP0+TF4m/gMc3ABMFO2ABx4EcAkWxUmNP1Po5Lb0/A9Gq
 LuSDRJVk3jSeGebmMsIH0SVzKY3ewQv7ZGB6A2uFdXtkwDUCxgHUQcfQY9yvjFut
 eU11zWkPbmbI4bwz9Y+Tyv9LOXRk/Br5B/uQEzQ9/Go/0OeJP0iTXre/veGufqND
 wK35AoDCVNomjw6h2Fg0t3kCt/mtR8W8D16+4QhsH+iE9IQMmhj4itz8Uq2ikh29
 lQIUT8acHC8fNUQJXw0nPXbNsimp/GyRGJiYCH7g7V67o/fjDUSbYMxk2TMZPuJV
 dGy3KQ==
 =UeX8
 -----END PGP SIGNATURE-----

Merge tag 'pull-tcg-20241008' of https://gitlab.com/rth7680/qemu into staging

linux-user: Fix parse_elf_properties GNU0_MAGIC check
linux-user: Various improvements to strace
linux-user: Add openat2 support
linux-user/flatload: Take mmap_lock in load_flt_binary()
accel/tcg: Make page_set_flags() documentation public
tcg/ppc: Use TCG_REG_TMP2 for scratch tcg_out_qemu_st
tcg/ppc: Use TCG_REG_TMP2 for scratch index in prepare_host_addr
target/m68k: Always return a temporary from gen_lea_mode

# -----BEGIN PGP SIGNATURE-----
#
# iQFRBAABCgA7FiEEekgeeIaLTbaoWgXAZN846K9+IV8FAmcFeLQdHHJpY2hhcmQu
# aGVuZGVyc29uQGxpbmFyby5vcmcACgkQZN846K9+IV+8Wgf9FJb/71zPAg7idUo8
# k+P5hOmE0PzPKdkoCh5WP0+TF4m/gMc3ABMFO2ABx4EcAkWxUmNP1Po5Lb0/A9Gq
# LuSDRJVk3jSeGebmMsIH0SVzKY3ewQv7ZGB6A2uFdXtkwDUCxgHUQcfQY9yvjFut
# eU11zWkPbmbI4bwz9Y+Tyv9LOXRk/Br5B/uQEzQ9/Go/0OeJP0iTXre/veGufqND
# wK35AoDCVNomjw6h2Fg0t3kCt/mtR8W8D16+4QhsH+iE9IQMmhj4itz8Uq2ikh29
# lQIUT8acHC8fNUQJXw0nPXbNsimp/GyRGJiYCH7g7V67o/fjDUSbYMxk2TMZPuJV
# dGy3KQ==
# =UeX8
# -----END PGP SIGNATURE-----
# gpg: Signature made Tue 08 Oct 2024 19:23:48 BST
# gpg:                using RSA key 7A481E78868B4DB6A85A05C064DF38E8AF7E215F
# gpg:                issuer "richard.henderson@linaro.org"
# gpg: Good signature from "Richard Henderson <richard.henderson@linaro.org>" [full]
# Primary key fingerprint: 7A48 1E78 868B 4DB6 A85A  05C0 64DF 38E8 AF7E 215F

* tag 'pull-tcg-20241008' of https://gitlab.com/rth7680/qemu:
  accel/tcg: Make page_set_flags() documentation public
  target/m68k: Always return a temporary from gen_lea_mode
  tcg/ppc: Use TCG_REG_TMP2 for scratch index in prepare_host_addr
  tcg/ppc: Use TCG_REG_TMP2 for scratch tcg_out_qemu_st
  linux-user: Add strace for recvfrom()
  linux-user: Add strace for sendto()
  linux-user: Factor print_buf_len() out
  linux-user: Display sockaddr buffer as pointer
  linux-user: Correct print_sockaddr() format
  linux-user: Trace wait4()'s and waitpid()'s wstatus
  linux-user: add strace support for openat2
  linux-user: add openat2 support in linux-user
  linux-user: Fix parse_elf_properties GNU0_MAGIC check
  linux-user/flatload: Take mmap_lock in load_flt_binary()

Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
Peter Maydell 2024-10-09 15:06:42 +01:00
commit 54cdddc6dc
12 changed files with 321 additions and 36 deletions

View File

@ -485,11 +485,6 @@ static bool pageflags_set_clear(target_ulong start, target_ulong last,
return inval_tb;
}
/*
* Modify the flags of a page and invalidate the code if necessary.
* The flag PAGE_WRITE_ORG is positioned automatically depending
* on PAGE_WRITE. The mmap_lock should already be held.
*/
void page_set_flags(target_ulong start, target_ulong last, int flags)
{
bool reset = false;

View File

@ -166,7 +166,20 @@ typedef int (*walk_memory_regions_fn)(void *, target_ulong,
int walk_memory_regions(void *, walk_memory_regions_fn);
int page_get_flags(target_ulong address);
/**
* page_set_flags:
* @start: first byte of range
* @last: last byte of range
* @flags: flags to set
* Context: holding mmap lock
*
* Modify the flags of a page and invalidate the code if necessary.
* The flag PAGE_WRITE_ORG is positioned automatically depending
* on PAGE_WRITE. The mmap_lock should already be held.
*/
void page_set_flags(target_ulong start, target_ulong last, int flags);
void page_reset_target_data(target_ulong start, target_ulong last);
/**

View File

@ -3121,11 +3121,11 @@ static bool parse_elf_properties(const ImageSource *src,
}
/*
* The contents of a valid PT_GNU_PROPERTY is a sequence
* of uint32_t -- swap them all now.
* The contents of a valid PT_GNU_PROPERTY is a sequence of uint32_t.
* Swap most of them now, beyond the header and namesz.
*/
#ifdef BSWAP_NEEDED
for (int i = 0; i < n / 4; i++) {
for (int i = 4; i < n / 4; i++) {
bswap32s(note.data + i);
}
#endif
@ -3135,15 +3135,15 @@ static bool parse_elf_properties(const ImageSource *src,
* immediately follows nhdr and is thus at the 4th word. Further, all
* of the inputs to the kernel's round_up are multiples of 4.
*/
if (note.nhdr.n_type != NT_GNU_PROPERTY_TYPE_0 ||
note.nhdr.n_namesz != NOTE_NAME_SZ ||
if (tswap32(note.nhdr.n_type) != NT_GNU_PROPERTY_TYPE_0 ||
tswap32(note.nhdr.n_namesz) != NOTE_NAME_SZ ||
note.data[3] != GNU0_MAGIC) {
error_setg(errp, "Invalid note in PT_GNU_PROPERTY");
return false;
}
off = sizeof(note.nhdr) + NOTE_NAME_SZ;
datasz = note.nhdr.n_descsz + off;
datasz = tswap32(note.nhdr.n_descsz) + off;
if (datasz > n) {
error_setg(errp, "Invalid note size in PT_GNU_PROPERTY");
return false;

View File

@ -487,7 +487,10 @@ int load_flt_binary(struct linux_binprm *bprm, struct image_info *info)
stack_len += (bprm->envc + 1) * 4; /* the envp array */
mmap_lock();
res = load_flat_file(bprm, libinfo, 0, &stack_len);
mmap_unlock();
if (is_error(res)) {
return res;
}

View File

@ -313,6 +313,15 @@ static inline bool access_ok(CPUState *cpu, int type,
int copy_from_user(void *hptr, abi_ulong gaddr, ssize_t len);
int copy_to_user(abi_ulong gaddr, void *hptr, ssize_t len);
/*
* copy_struct_from_user() copies a target struct to a host struct, in
* a way that guarantees backwards-compatibility for struct syscall
* arguments.
*
* Similar to kernels uaccess.h:copy_struct_from_user()
*/
int copy_struct_from_user(void *dst, size_t ksize, abi_ptr src, size_t usize);
/* Functions for accessing guest memory. The tget and tput functions
read/write single values, byteswapping as necessary. The lock_user function
gets a pointer to a contiguous area of guest memory, but does not perform

View File

@ -13,6 +13,9 @@
#include <linux/if_packet.h>
#include <linux/in6.h>
#include <linux/netlink.h>
#ifdef HAVE_OPENAT2_H
#include <linux/openat2.h>
#endif
#include <sched.h>
#include "qemu.h"
#include "user-internals.h"
@ -373,7 +376,7 @@ print_sockaddr(abi_ulong addr, abi_long addrlen, int last)
un->sun_path[i]; i++) {
qemu_log("%c", un->sun_path[i]);
}
qemu_log("\"}");
qemu_log("\"},");
break;
}
case AF_INET: {
@ -383,7 +386,7 @@ print_sockaddr(abi_ulong addr, abi_long addrlen, int last)
ntohs(in->sin_port));
qemu_log("sin_addr=inet_addr(\"%d.%d.%d.%d\")",
c[0], c[1], c[2], c[3]);
qemu_log("}");
qemu_log("},");
break;
}
case AF_PACKET: {
@ -414,12 +417,12 @@ print_sockaddr(abi_ulong addr, abi_long addrlen, int last)
}
qemu_log(",sll_addr=%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x",
c[0], c[1], c[2], c[3], c[4], c[5], c[6], c[7]);
qemu_log("}");
qemu_log("},");
break;
}
case AF_NETLINK: {
struct target_sockaddr_nl *nl = (struct target_sockaddr_nl *)sa;
qemu_log("{nl_family=AF_NETLINK,nl_pid=%u,nl_groups=%u}",
qemu_log("{nl_family=AF_NETLINK,nl_pid=%u,nl_groups=%u},",
tswap32(nl->nl_pid), tswap32(nl->nl_groups));
break;
}
@ -429,14 +432,14 @@ print_sockaddr(abi_ulong addr, abi_long addrlen, int last)
qemu_log("%02x, ", sa->sa_data[i]);
}
qemu_log("%02x}", sa->sa_data[i]);
qemu_log("}");
qemu_log("},");
break;
}
unlock_user(sa, addr, 0);
} else {
print_raw_param("0x"TARGET_ABI_FMT_lx, addr, 0);
print_pointer(addr, 0);
}
qemu_log(", "TARGET_ABI_FMT_ld"%s", addrlen, get_comma(last));
qemu_log(TARGET_ABI_FMT_ld"%s", addrlen, get_comma(last));
}
static void
@ -1063,6 +1066,18 @@ UNUSED static const struct flags open_flags[] = {
FLAG_END,
};
UNUSED static const struct flags openat2_resolve_flags[] = {
#ifdef HAVE_OPENAT2_H
FLAG_GENERIC(RESOLVE_NO_XDEV),
FLAG_GENERIC(RESOLVE_NO_MAGICLINKS),
FLAG_GENERIC(RESOLVE_NO_SYMLINKS),
FLAG_GENERIC(RESOLVE_BENEATH),
FLAG_GENERIC(RESOLVE_IN_ROOT),
FLAG_GENERIC(RESOLVE_CACHED),
#endif
FLAG_END,
};
UNUSED static const struct flags mount_flags[] = {
#ifdef MS_BIND
FLAG_GENERIC(MS_BIND),
@ -1655,6 +1670,13 @@ print_buf(abi_long addr, abi_long len, int last)
}
}
static void
print_buf_len(abi_long addr, abi_long len, int last)
{
print_buf(addr, len, 0);
print_raw_param(TARGET_ABI_FMT_ld, len, last);
}
/*
* Prints out raw parameter using given format. Caller needs
* to do byte swapping if needed.
@ -2742,8 +2764,7 @@ static void do_print_sendrecv(const char *name, abi_long arg1)
qemu_log("%s(", name);
print_sockfd(sockfd, 0);
print_buf(msg, len, 0);
print_raw_param(TARGET_ABI_FMT_ld, len, 0);
print_buf_len(msg, len, 0);
print_flags(msg_flags, flags, 1);
qemu_log(")");
}
@ -2761,8 +2782,7 @@ static void do_print_msgaddr(const char *name, abi_long arg1)
qemu_log("%s(", name);
print_sockfd(sockfd, 0);
print_buf(msg, len, 0);
print_raw_param(TARGET_ABI_FMT_ld, len, 0);
print_buf_len(msg, len, 0);
print_flags(msg_flags, flags, 0);
print_sockaddr(addr, addrlen, 0);
qemu_log(")");
@ -3122,6 +3142,38 @@ print_bind(CPUArchState *cpu_env, const struct syscallname *name,
}
#endif
#ifdef TARGET_NR_recvfrom
static void
print_recvfrom(CPUArchState *cpu_env, const struct syscallname *name,
abi_long arg0, abi_long arg1, abi_long arg2,
abi_long arg3, abi_long arg4, abi_long arg5)
{
print_syscall_prologue(name);
print_sockfd(arg0, 0);
print_pointer(arg1, 0); /* output */
print_raw_param(TARGET_ABI_FMT_ld, arg2, 0);
print_flags(msg_flags, arg3, 0);
print_pointer(arg4, 0); /* output */
print_pointer(arg5, 1); /* in/out */
print_syscall_epilogue(name);
}
#endif
#ifdef TARGET_NR_sendto
static void
print_sendto(CPUArchState *cpu_env, const struct syscallname *name,
abi_long arg0, abi_long arg1, abi_long arg2,
abi_long arg3, abi_long arg4, abi_long arg5)
{
print_syscall_prologue(name);
print_sockfd(arg0, 0);
print_buf_len(arg1, arg2, 0);
print_flags(msg_flags, arg3, 0);
print_sockaddr(arg4, arg5, 1);
print_syscall_epilogue(name);
}
#endif
#if defined(TARGET_NR_stat) || defined(TARGET_NR_stat64) || \
defined(TARGET_NR_lstat) || defined(TARGET_NR_lstat64)
static void
@ -3483,6 +3535,38 @@ print_openat(CPUArchState *cpu_env, const struct syscallname *name,
}
#endif
#ifdef TARGET_NR_openat2
static void
print_openat2(CPUArchState *cpu_env, const struct syscallname *name,
abi_long arg0, abi_long arg1, abi_long arg2,
abi_long arg3, abi_long arg4, abi_long arg5)
{
struct open_how_ver0 how;
print_syscall_prologue(name);
print_at_dirfd(arg0, 0);
print_string(arg1, 0);
if ((abi_ulong)arg3 >= sizeof(struct target_open_how_ver0) &&
copy_struct_from_user(&how, sizeof(how), arg2, arg3) == 0) {
how.flags = tswap64(how.flags);
how.mode = tswap64(how.mode);
how.resolve = tswap64(how.resolve);
qemu_log("{");
print_open_flags(how.flags, 0);
if (how.flags & TARGET_O_CREAT) {
print_file_mode(how.mode, 0);
}
print_flags(openat2_resolve_flags, how.resolve, 1);
qemu_log("},");
} else {
print_pointer(arg2, 0);
}
print_raw_param(TARGET_ABI_FMT_lu, arg3, 1);
print_syscall_epilogue(name);
}
#endif
#ifdef TARGET_NR_pidfd_send_signal
static void
print_pidfd_send_signal(CPUArchState *cpu_env, const struct syscallname *name,
@ -4168,6 +4252,63 @@ print_ioctl(CPUArchState *cpu_env, const struct syscallname *name,
}
#endif
#if defined(TARGET_NR_wait4) || defined(TARGET_NR_waitpid)
static void print_wstatus(int wstatus)
{
if (WIFSIGNALED(wstatus)) {
qemu_log("{WIFSIGNALED(s) && WTERMSIG(s) == ");
print_signal(WTERMSIG(wstatus), 1);
if (WCOREDUMP(wstatus)) {
qemu_log(" && WCOREDUMP(s)");
}
qemu_log("}");
} else if (WIFEXITED(wstatus)) {
qemu_log("{WIFEXITED(s) && WEXITSTATUS(s) == %d}",
WEXITSTATUS(wstatus));
} else {
print_number(wstatus, 1);
}
}
static void print_ret_wstatus(abi_long ret, abi_long wstatus_addr)
{
int wstatus;
if (!print_syscall_err(ret)
&& wstatus_addr
&& get_user_s32(wstatus, wstatus_addr)) {
qemu_log(TARGET_ABI_FMT_ld " (wstatus=", ret);
print_wstatus(wstatus);
qemu_log(")");
}
qemu_log("\n");
}
#endif
#ifdef TARGET_NR_wait4
static void
print_syscall_ret_wait4(CPUArchState *cpu_env,
const struct syscallname *name,
abi_long ret, abi_long arg0, abi_long arg1,
abi_long arg2, abi_long arg3, abi_long arg4,
abi_long arg5)
{
print_ret_wstatus(ret, arg1);
}
#endif
#ifdef TARGET_NR_waitpid
static void
print_syscall_ret_waitpid(CPUArchState *cpu_env,
const struct syscallname *name,
abi_long ret, abi_long arg0, abi_long arg1,
abi_long arg2, abi_long arg3, abi_long arg4,
abi_long arg5)
{
print_ret_wstatus(ret, arg1);
}
#endif
/*
* An array of all of the syscalls we know about
*/

View File

@ -715,6 +715,9 @@
#ifdef TARGET_NR_openat
{ TARGET_NR_openat, "openat" , NULL, print_openat, NULL },
#endif
#ifdef TARGET_NR_openat2
{ TARGET_NR_openat2, "openat2" , NULL, print_openat2, NULL },
#endif
#ifdef TARGET_NR_osf_adjtime
{ TARGET_NR_osf_adjtime, "osf_adjtime" , NULL, NULL, NULL },
#endif
@ -1135,7 +1138,7 @@
{ TARGET_NR_recv, "recv" , "%s(%d,%p,%u,%d)", NULL, NULL },
#endif
#ifdef TARGET_NR_recvfrom
{ TARGET_NR_recvfrom, "recvfrom" , NULL, NULL, NULL },
{ TARGET_NR_recvfrom, "recvfrom" , NULL, print_recvfrom, NULL },
#endif
#ifdef TARGET_NR_recvmmsg
{ TARGET_NR_recvmmsg, "recvmmsg" , NULL, NULL, NULL },
@ -1285,7 +1288,7 @@
{ TARGET_NR_sendmsg, "sendmsg" , NULL, NULL, NULL },
#endif
#ifdef TARGET_NR_sendto
{ TARGET_NR_sendto, "sendto" , NULL, NULL, NULL },
{ TARGET_NR_sendto, "sendto" , NULL, print_sendto, NULL },
#endif
#ifdef TARGET_NR_setdomainname
{ TARGET_NR_setdomainname, "setdomainname" , NULL, NULL, NULL },
@ -1659,13 +1662,15 @@
{ TARGET_NR_vserver, "vserver" , NULL, NULL, NULL },
#endif
#ifdef TARGET_NR_wait4
{ TARGET_NR_wait4, "wait4" , "%s(%d,%p,%d,%p)", NULL, NULL },
{ TARGET_NR_wait4, "wait4" , "%s(%d,%p,%d,%p)", NULL,
print_syscall_ret_wait4 },
#endif
#ifdef TARGET_NR_waitid
{ TARGET_NR_waitid, "waitid" , "%s(%#x,%d,%p,%#x)", NULL, NULL },
#endif
#ifdef TARGET_NR_waitpid
{ TARGET_NR_waitpid, "waitpid" , "%s(%d,%p,%#x)", NULL, NULL },
{ TARGET_NR_waitpid, "waitpid", "%s(%d,%p,%#x)", NULL,
print_syscall_ret_waitpid },
#endif
#ifdef TARGET_NR_write
{ TARGET_NR_write, "write" , "%s(%d,%#x,%d)", NULL, NULL },

View File

@ -602,6 +602,33 @@ static int check_zeroed_user(abi_long addr, size_t ksize, size_t usize)
return 1;
}
/*
* Copies a target struct to a host struct, in a way that guarantees
* backwards-compatibility for struct syscall arguments.
*
* Similar to kernels uaccess.h:copy_struct_from_user()
*/
int copy_struct_from_user(void *dst, size_t ksize, abi_ptr src, size_t usize)
{
size_t size = MIN(ksize, usize);
size_t rest = MAX(ksize, usize) - size;
/* Deal with trailing bytes. */
if (usize < ksize) {
memset(dst + size, 0, rest);
} else if (usize > ksize) {
int ret = check_zeroed_user(src, ksize, usize);
if (ret <= 0) {
return ret ?: -TARGET_E2BIG;
}
}
/* Copy the interoperable parts of the struct. */
if (copy_from_user(dst, src, size)) {
return -TARGET_EFAULT;
}
return 0;
}
#define safe_syscall0(type, name) \
static type safe_##name(void) \
{ \
@ -653,6 +680,10 @@ safe_syscall3(ssize_t, read, int, fd, void *, buff, size_t, count)
safe_syscall3(ssize_t, write, int, fd, const void *, buff, size_t, count)
safe_syscall4(int, openat, int, dirfd, const char *, pathname, \
int, flags, mode_t, mode)
safe_syscall4(int, openat2, int, dirfd, const char *, pathname, \
const struct open_how_ver0 *, how, size_t, size)
#if defined(TARGET_NR_wait4) || defined(TARGET_NR_waitpid)
safe_syscall4(pid_t, wait4, pid_t, pid, int *, status, int, options, \
struct rusage *, rusage)
@ -8332,8 +8363,9 @@ static int open_net_route(CPUArchState *cpu_env, int fd)
}
#endif
int do_guest_openat(CPUArchState *cpu_env, int dirfd, const char *fname,
int flags, mode_t mode, bool safe)
static int maybe_do_fake_open(CPUArchState *cpu_env, int dirfd,
const char *fname, int flags, mode_t mode,
int openat2_resolve, bool safe)
{
g_autofree char *proc_name = NULL;
const char *pathname;
@ -8370,6 +8402,12 @@ int do_guest_openat(CPUArchState *cpu_env, int dirfd, const char *fname,
}
if (is_proc_myself(pathname, "exe")) {
/* Honor openat2 resolve flags */
if ((openat2_resolve & RESOLVE_NO_MAGICLINKS) ||
(openat2_resolve & RESOLVE_NO_SYMLINKS)) {
errno = ELOOP;
return -1;
}
if (safe) {
return safe_openat(dirfd, exec_path, flags, mode);
} else {
@ -8416,6 +8454,17 @@ int do_guest_openat(CPUArchState *cpu_env, int dirfd, const char *fname,
return fd;
}
return -2;
}
int do_guest_openat(CPUArchState *cpu_env, int dirfd, const char *pathname,
int flags, mode_t mode, bool safe)
{
int fd = maybe_do_fake_open(cpu_env, dirfd, pathname, flags, mode, 0, safe);
if (fd > -2) {
return fd;
}
if (safe) {
return safe_openat(dirfd, path(pathname), flags, mode);
} else {
@ -8423,6 +8472,49 @@ int do_guest_openat(CPUArchState *cpu_env, int dirfd, const char *fname,
}
}
static int do_openat2(CPUArchState *cpu_env, abi_long dirfd,
abi_ptr guest_pathname, abi_ptr guest_open_how,
abi_ulong guest_size)
{
struct open_how_ver0 how = {0};
char *pathname;
int ret;
if (guest_size < sizeof(struct target_open_how_ver0)) {
return -TARGET_EINVAL;
}
ret = copy_struct_from_user(&how, sizeof(how), guest_open_how, guest_size);
if (ret) {
if (ret == -TARGET_E2BIG) {
qemu_log_mask(LOG_UNIMP,
"Unimplemented openat2 open_how size: "
TARGET_ABI_FMT_lu "\n", guest_size);
}
return ret;
}
pathname = lock_user_string(guest_pathname);
if (!pathname) {
return -TARGET_EFAULT;
}
how.flags = target_to_host_bitmask(tswap64(how.flags), fcntl_flags_tbl);
how.mode = tswap64(how.mode);
how.resolve = tswap64(how.resolve);
int fd = maybe_do_fake_open(cpu_env, dirfd, pathname, how.flags, how.mode,
how.resolve, true);
if (fd > -2) {
ret = get_errno(fd);
} else {
ret = get_errno(safe_openat2(dirfd, pathname, &how,
sizeof(struct open_how_ver0)));
}
fd_trans_unregister(ret);
unlock_user(pathname, guest_pathname, 0);
return ret;
}
ssize_t do_guest_readlink(const char *pathname, char *buf, size_t bufsiz)
{
ssize_t ret;
@ -9195,6 +9287,9 @@ static abi_long do_syscall1(CPUArchState *cpu_env, int num, abi_long arg1,
fd_trans_unregister(ret);
unlock_user(p, arg2, 0);
return ret;
case TARGET_NR_openat2:
ret = do_openat2(cpu_env, arg1, arg2, arg3, arg4);
return ret;
#if defined(TARGET_NR_name_to_handle_at) && defined(CONFIG_OPEN_BY_HANDLE)
case TARGET_NR_name_to_handle_at:
ret = do_name_to_handle_at(arg1, arg2, arg3, arg4, arg5);

View File

@ -2748,4 +2748,22 @@ struct target_sched_param {
abi_int sched_priority;
};
/* from kernel's include/uapi/linux/openat2.h */
struct open_how_ver0 {
__u64 flags;
__u64 mode;
__u64 resolve;
};
struct target_open_how_ver0 {
abi_ullong flags;
abi_ullong mode;
abi_ullong resolve;
};
#ifndef RESOLVE_NO_MAGICLINKS
#define RESOLVE_NO_MAGICLINKS 0x02
#endif
#ifndef RESOLVE_NO_SYMLINKS
#define RESOLVE_NO_SYMLINKS 0x04
#endif
#endif

View File

@ -2481,6 +2481,7 @@ config_host_data.set('CONFIG_LINUX_MAGIC_H', cc.has_header('linux/magic.h'))
config_host_data.set('CONFIG_VALGRIND_H', cc.has_header('valgrind/valgrind.h'))
config_host_data.set('HAVE_BTRFS_H', cc.has_header('linux/btrfs.h'))
config_host_data.set('HAVE_DRM_H', cc.has_header('libdrm/drm.h'))
config_host_data.set('HAVE_OPENAT2_H', cc.has_header('linux/openat2.h'))
config_host_data.set('HAVE_PTY_H', cc.has_header('pty.h'))
config_host_data.set('HAVE_SYS_DISK_H', cc.has_header('sys/disk.h'))
config_host_data.set('HAVE_SYS_IOCCOM_H', cc.has_header('sys/ioccom.h'))

View File

@ -720,7 +720,9 @@ static TCGv gen_lea_mode(CPUM68KState *env, DisasContext *s,
}
/* fallthru */
case 2: /* Indirect register */
return get_areg(s, reg0);
tmp = tcg_temp_new();
tcg_gen_mov_i32(tmp, get_areg(s, reg0));
return tmp;
case 4: /* Indirect predecrememnt. */
if (opsize == OS_UNSIZED) {
return NULL_QREG;
@ -747,20 +749,23 @@ static TCGv gen_lea_mode(CPUM68KState *env, DisasContext *s,
switch (reg0) {
case 0: /* Absolute short. */
offset = (int16_t)read_im16(env, s);
return tcg_constant_i32(offset);
break;
case 1: /* Absolute long. */
offset = read_im32(env, s);
return tcg_constant_i32(offset);
break;
case 2: /* pc displacement */
offset = s->pc;
offset += (int16_t)read_im16(env, s);
return tcg_constant_i32(offset);
break;
case 3: /* pc index+displacement. */
return gen_lea_indexed(env, s, NULL_QREG);
case 4: /* Immediate. */
default:
return NULL_QREG;
}
tmp = tcg_temp_new();
tcg_gen_movi_i32(tmp, offset);
return tmp;
}
/* Should never happen. */
return NULL_QREG;

View File

@ -2617,8 +2617,8 @@ static TCGLabelQemuLdst *prepare_host_addr(TCGContext *s, HostAddress *h,
if (TCG_TARGET_REG_BITS == 64 && addr_type == TCG_TYPE_I32) {
/* Zero-extend the guest address for use in the host address. */
tcg_out_ext32u(s, TCG_REG_R0, addrlo);
h->index = TCG_REG_R0;
tcg_out_ext32u(s, TCG_REG_TMP2, addrlo);
h->index = TCG_REG_TMP2;
} else {
h->index = addrlo;
}
@ -2704,9 +2704,9 @@ static void tcg_out_qemu_st(TCGContext *s, TCGReg datalo, TCGReg datahi,
uint32_t insn = qemu_stx_opc[opc & (MO_BSWAP | MO_SIZE)];
if (!have_isa_2_06 && insn == STDBRX) {
tcg_out32(s, STWBRX | SAB(datalo, h.base, h.index));
tcg_out32(s, ADDI | TAI(TCG_REG_TMP1, h.index, 4));
tcg_out32(s, ADDI | TAI(TCG_REG_TMP2, h.index, 4));
tcg_out_shri64(s, TCG_REG_R0, datalo, 32);
tcg_out32(s, STWBRX | SAB(TCG_REG_R0, h.base, TCG_REG_TMP1));
tcg_out32(s, STWBRX | SAB(TCG_REG_R0, h.base, TCG_REG_TMP2));
} else {
tcg_out32(s, insn | SAB(datalo, h.base, h.index));
}