- Update netlink types to linux v5.0

- fix accept4/getpeername/getsockname/recvfrom/recvmsg/read
 - add/fix ELF_PLATFORM ofr aarch64 and arm
 - fix "may be used uninitialized" warnings
 - Fix breakpoint support in Nios
 - Nicer strace output of chroot() syscall
 -----BEGIN PGP SIGNATURE-----
 
 iQIcBAABAgAGBQJcgO0YAAoJEPMMOL0/L748HU8P/3IknbXS9/WGLdSlKAbW4o3+
 aCq8a/l6RpvD/4JVQIuPjs5v6j6PsWZwlB0//SeTv0so0yWH6Lw+9JkTNx3TqjgF
 +qhSZxvDd2gmsZhhMpX3XDsGJujPYRGhJGgeuJ1uGalwZO5Q8pt6ta4Jx1NRxtzt
 B38gIAO3D8Rqz49omFvXNv8154ojMIQlLqDjfwtwj7e/O7fKRD8ROv4+HqfJCzHG
 jE/tInZeY5NoRCXhUeaoI8NaTMWGFs7IVgryHWanVKaixYMVb26LIYfaw3aK2CZB
 qnf2u+S5G9fE4ba121teaLlEm31V716cFhPV6axH+qesxeLFC5uACaTI7Z/OwLPQ
 IEnDbuZ5cAP49dBD6bEkS7h4V3OQ/yt+C8v+MH5ObDWzrLZuH6iPVgQka/0swYqc
 7c4uhV948uO84UJIOVfB4xhMdMtigFtTDQHpRAzNwRkER6t8zJIU6GRpg+MbGbwt
 8ivPgTknf+CjyDpw1CFUVkr2PAThFNqce+247EpHLtEtjidW5NjRTrdO5EjFs//p
 E8gGaoxb7NbwNheM/rtk99rLqYybObgV9lFdslyW5JaZ6VnhUkxjdDBMTetegNam
 BDyTf49p93ohom64S8pYz5iQPBm7X0vc7nfK7wCk6cHjGhbedjCOyMssz/2xCh9Y
 I5DK8q28jnJukr/6EgqI
 =Dqxx
 -----END PGP SIGNATURE-----

Merge remote-tracking branch 'remotes/vivier2/tags/linux-user-for-4.0-pull-request' into staging

- Update netlink types to linux v5.0
- fix accept4/getpeername/getsockname/recvfrom/recvmsg/read
- add/fix ELF_PLATFORM ofr aarch64 and arm
- fix "may be used uninitialized" warnings
- Fix breakpoint support in Nios
- Nicer strace output of chroot() syscall

# gpg: Signature made Thu 07 Mar 2019 10:06:16 GMT
# gpg:                using RSA key F30C38BD3F2FBE3C
# gpg: Good signature from "Laurent Vivier <lvivier@redhat.com>" [full]
# gpg:                 aka "Laurent Vivier <laurent@vivier.eu>" [full]
# gpg:                 aka "Laurent Vivier (Red Hat) <lvivier@redhat.com>" [full]
# Primary key fingerprint: CD2F 75DD C8E3 A4DC 2E4F  5173 F30C 38BD 3F2F BE3C

* remotes/vivier2/tags/linux-user-for-4.0-pull-request:
  linux-user: add new netlink types
  linux-user: Nicer strace output of chroot() syscall
  linux-user: fix "may be used uninitialized" warnings
  linux-user: don't short-circuit read with zero length
  Fix breakpoint support in Nios II user-mode emulation.
  linux-user: fix emulation of accept4/getpeername/getsockname/recvfrom syscalls
  linux-user: Fix ELF_PLATFORM for aarch64_be-linux-user
  linux-user: Add ELF_PLATFORM for arm
  linux-user: fix recvmsg emulation

Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
Peter Maydell 2019-03-07 18:40:43 +00:00
commit c4e0780ed1
6 changed files with 97 additions and 22 deletions

View File

@ -500,13 +500,48 @@ static uint32_t get_elf_hwcap2(void)
#undef GET_FEATURE #undef GET_FEATURE
#undef GET_FEATURE_ID #undef GET_FEATURE_ID
#define ELF_PLATFORM get_elf_platform()
static const char *get_elf_platform(void)
{
CPUARMState *env = thread_cpu->env_ptr;
#ifdef TARGET_WORDS_BIGENDIAN
# define END "b"
#else
# define END "l"
#endif
if (arm_feature(env, ARM_FEATURE_V8)) {
return "v8" END;
} else if (arm_feature(env, ARM_FEATURE_V7)) {
if (arm_feature(env, ARM_FEATURE_M)) {
return "v7m" END;
} else {
return "v7" END;
}
} else if (arm_feature(env, ARM_FEATURE_V6)) {
return "v6" END;
} else if (arm_feature(env, ARM_FEATURE_V5)) {
return "v5" END;
} else {
return "v4" END;
}
#undef END
}
#else #else
/* 64 bit ARM definitions */ /* 64 bit ARM definitions */
#define ELF_START_MMAP 0x80000000 #define ELF_START_MMAP 0x80000000
#define ELF_ARCH EM_AARCH64 #define ELF_ARCH EM_AARCH64
#define ELF_CLASS ELFCLASS64 #define ELF_CLASS ELFCLASS64
#define ELF_PLATFORM "aarch64" #ifdef TARGET_WORDS_BIGENDIAN
# define ELF_PLATFORM "aarch64_be"
#else
# define ELF_PLATFORM "aarch64"
#endif
static inline void init_thread(struct target_pt_regs *regs, static inline void init_thread(struct target_pt_regs *regs,
struct image_info *infop) struct image_info *infop)

View File

@ -75,6 +75,8 @@ enum {
QEMU_IFLA_BR_MCAST_STATS_ENABLED, QEMU_IFLA_BR_MCAST_STATS_ENABLED,
QEMU_IFLA_BR_MCAST_IGMP_VERSION, QEMU_IFLA_BR_MCAST_IGMP_VERSION,
QEMU_IFLA_BR_MCAST_MLD_VERSION, QEMU_IFLA_BR_MCAST_MLD_VERSION,
QEMU_IFLA_BR_VLAN_STATS_PER_PORT,
QEMU_IFLA_BR_MULTI_BOOLOPT,
QEMU___IFLA_BR_MAX, QEMU___IFLA_BR_MAX,
}; };
@ -438,6 +440,7 @@ static abi_long host_to_target_data_bridge_nlattr(struct nlattr *nlattr,
case QEMU_IFLA_BR_MCAST_STATS_ENABLED: case QEMU_IFLA_BR_MCAST_STATS_ENABLED:
case QEMU_IFLA_BR_MCAST_IGMP_VERSION: case QEMU_IFLA_BR_MCAST_IGMP_VERSION:
case QEMU_IFLA_BR_MCAST_MLD_VERSION: case QEMU_IFLA_BR_MCAST_MLD_VERSION:
case QEMU_IFLA_BR_VLAN_STATS_PER_PORT:
break; break;
/* uint16_t */ /* uint16_t */
case QEMU_IFLA_BR_PRIORITY: case QEMU_IFLA_BR_PRIORITY:
@ -543,6 +546,12 @@ static abi_long host_to_target_slave_data_bridge_nlattr(struct nlattr *nlattr,
case QEMU_IFLA_BRPORT_ROOT_ID: case QEMU_IFLA_BRPORT_ROOT_ID:
case QEMU_IFLA_BRPORT_BRIDGE_ID: case QEMU_IFLA_BRPORT_BRIDGE_ID:
break; break;
/* br_boolopt_multi { uint32_t, uint32_t } */
case QEMU_IFLA_BR_MULTI_BOOLOPT:
u32 = NLA_DATA(nlattr);
u32[0] = tswap32(u32[0]); /* optval */
u32[1] = tswap32(u32[1]); /* optmask */
break;
default: default:
gemu_log("Unknown QEMU_IFLA_BRPORT type %d\n", nlattr->nla_type); gemu_log("Unknown QEMU_IFLA_BRPORT type %d\n", nlattr->nla_type);
break; break;

View File

@ -73,6 +73,12 @@ void cpu_loop(CPUNios2State *env)
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info); queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
break; break;
} }
case EXCP_DEBUG:
info.si_signo = TARGET_SIGTRAP;
info.si_errno = 0;
info.si_code = TARGET_TRAP_BRKPT;
queue_signal(env, info.si_signo, QEMU_SI_FAULT, &info);
break;
case 0xaa: case 0xaa:
switch (env->regs[R_PC]) { switch (env->regs[R_PC]) {
/*case 0x1000:*/ /* TODO:__kuser_helper_version */ /*case 0x1000:*/ /* TODO:__kuser_helper_version */

View File

@ -1235,6 +1235,18 @@ print_chdir(const struct syscallname *name,
} }
#endif #endif
#ifdef TARGET_NR_chroot
static void
print_chroot(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_string(arg0, 1);
print_syscall_epilogue(name);
}
#endif
#ifdef TARGET_NR_chmod #ifdef TARGET_NR_chmod
static void static void
print_chmod(const struct syscallname *name, print_chmod(const struct syscallname *name,

View File

@ -77,7 +77,7 @@
{ TARGET_NR_chown32, "chown32" , NULL, NULL, NULL }, { TARGET_NR_chown32, "chown32" , NULL, NULL, NULL },
#endif #endif
#ifdef TARGET_NR_chroot #ifdef TARGET_NR_chroot
{ TARGET_NR_chroot, "chroot" , NULL, NULL, NULL }, { TARGET_NR_chroot, "chroot" , NULL, print_chroot, NULL },
#endif #endif
#ifdef TARGET_NR_clock_adjtime #ifdef TARGET_NR_clock_adjtime
{ TARGET_NR_clock_adjtime, "clock_adjtime" , NULL, print_clock_adjtime, NULL }, { TARGET_NR_clock_adjtime, "clock_adjtime" , NULL, print_clock_adjtime, NULL },

View File

@ -2759,6 +2759,7 @@ static abi_long do_sendrecvmsg_locked(int fd, struct target_msghdr *msgp,
} }
if (!is_error(ret)) { if (!is_error(ret)) {
msgp->msg_namelen = tswap32(msg.msg_namelen); msgp->msg_namelen = tswap32(msg.msg_namelen);
msgp->msg_flags = tswap32(msg.msg_flags);
if (msg.msg_name != NULL && msg.msg_name != (void *)-1) { if (msg.msg_name != NULL && msg.msg_name != (void *)-1) {
ret = host_to_target_sockaddr(tswapal(msgp->msg_name), ret = host_to_target_sockaddr(tswapal(msgp->msg_name),
msg.msg_name, msg.msg_namelen); msg.msg_name, msg.msg_namelen);
@ -2846,7 +2847,7 @@ static abi_long do_sendrecvmmsg(int fd, abi_ulong target_msgvec,
static abi_long do_accept4(int fd, abi_ulong target_addr, static abi_long do_accept4(int fd, abi_ulong target_addr,
abi_ulong target_addrlen_addr, int flags) abi_ulong target_addrlen_addr, int flags)
{ {
socklen_t addrlen; socklen_t addrlen, ret_addrlen;
void *addr; void *addr;
abi_long ret; abi_long ret;
int host_flags; int host_flags;
@ -2870,11 +2871,13 @@ static abi_long do_accept4(int fd, abi_ulong target_addr,
addr = alloca(addrlen); addr = alloca(addrlen);
ret = get_errno(safe_accept4(fd, addr, &addrlen, host_flags)); ret_addrlen = addrlen;
ret = get_errno(safe_accept4(fd, addr, &ret_addrlen, host_flags));
if (!is_error(ret)) { if (!is_error(ret)) {
host_to_target_sockaddr(target_addr, addr, addrlen); host_to_target_sockaddr(target_addr, addr, MIN(addrlen, ret_addrlen));
if (put_user_u32(addrlen, target_addrlen_addr)) if (put_user_u32(ret_addrlen, target_addrlen_addr)) {
ret = -TARGET_EFAULT; ret = -TARGET_EFAULT;
}
} }
return ret; return ret;
} }
@ -2883,7 +2886,7 @@ static abi_long do_accept4(int fd, abi_ulong target_addr,
static abi_long do_getpeername(int fd, abi_ulong target_addr, static abi_long do_getpeername(int fd, abi_ulong target_addr,
abi_ulong target_addrlen_addr) abi_ulong target_addrlen_addr)
{ {
socklen_t addrlen; socklen_t addrlen, ret_addrlen;
void *addr; void *addr;
abi_long ret; abi_long ret;
@ -2899,11 +2902,13 @@ static abi_long do_getpeername(int fd, abi_ulong target_addr,
addr = alloca(addrlen); addr = alloca(addrlen);
ret = get_errno(getpeername(fd, addr, &addrlen)); ret_addrlen = addrlen;
ret = get_errno(getpeername(fd, addr, &ret_addrlen));
if (!is_error(ret)) { if (!is_error(ret)) {
host_to_target_sockaddr(target_addr, addr, addrlen); host_to_target_sockaddr(target_addr, addr, MIN(addrlen, ret_addrlen));
if (put_user_u32(addrlen, target_addrlen_addr)) if (put_user_u32(ret_addrlen, target_addrlen_addr)) {
ret = -TARGET_EFAULT; ret = -TARGET_EFAULT;
}
} }
return ret; return ret;
} }
@ -2912,7 +2917,7 @@ static abi_long do_getpeername(int fd, abi_ulong target_addr,
static abi_long do_getsockname(int fd, abi_ulong target_addr, static abi_long do_getsockname(int fd, abi_ulong target_addr,
abi_ulong target_addrlen_addr) abi_ulong target_addrlen_addr)
{ {
socklen_t addrlen; socklen_t addrlen, ret_addrlen;
void *addr; void *addr;
abi_long ret; abi_long ret;
@ -2928,11 +2933,13 @@ static abi_long do_getsockname(int fd, abi_ulong target_addr,
addr = alloca(addrlen); addr = alloca(addrlen);
ret = get_errno(getsockname(fd, addr, &addrlen)); ret_addrlen = addrlen;
ret = get_errno(getsockname(fd, addr, &ret_addrlen));
if (!is_error(ret)) { if (!is_error(ret)) {
host_to_target_sockaddr(target_addr, addr, addrlen); host_to_target_sockaddr(target_addr, addr, MIN(addrlen, ret_addrlen));
if (put_user_u32(addrlen, target_addrlen_addr)) if (put_user_u32(ret_addrlen, target_addrlen_addr)) {
ret = -TARGET_EFAULT; ret = -TARGET_EFAULT;
}
} }
return ret; return ret;
} }
@ -3004,7 +3011,7 @@ static abi_long do_recvfrom(int fd, abi_ulong msg, size_t len, int flags,
abi_ulong target_addr, abi_ulong target_addr,
abi_ulong target_addrlen) abi_ulong target_addrlen)
{ {
socklen_t addrlen; socklen_t addrlen, ret_addrlen;
void *addr; void *addr;
void *host_msg; void *host_msg;
abi_long ret; abi_long ret;
@ -3022,10 +3029,12 @@ static abi_long do_recvfrom(int fd, abi_ulong msg, size_t len, int flags,
goto fail; goto fail;
} }
addr = alloca(addrlen); addr = alloca(addrlen);
ret_addrlen = addrlen;
ret = get_errno(safe_recvfrom(fd, host_msg, len, flags, ret = get_errno(safe_recvfrom(fd, host_msg, len, flags,
addr, &addrlen)); addr, &ret_addrlen));
} else { } else {
addr = NULL; /* To keep compiler quiet. */ addr = NULL; /* To keep compiler quiet. */
addrlen = 0; /* To keep compiler quiet. */
ret = get_errno(safe_recvfrom(fd, host_msg, len, flags, NULL, 0)); ret = get_errno(safe_recvfrom(fd, host_msg, len, flags, NULL, 0));
} }
if (!is_error(ret)) { if (!is_error(ret)) {
@ -3038,8 +3047,9 @@ static abi_long do_recvfrom(int fd, abi_ulong msg, size_t len, int flags,
} }
} }
if (target_addr) { if (target_addr) {
host_to_target_sockaddr(target_addr, addr, addrlen); host_to_target_sockaddr(target_addr, addr,
if (put_user_u32(addrlen, target_addrlen)) { MIN(addrlen, ret_addrlen));
if (put_user_u32(ret_addrlen, target_addrlen)) {
ret = -TARGET_EFAULT; ret = -TARGET_EFAULT;
goto fail; goto fail;
} }
@ -4723,8 +4733,8 @@ static abi_long do_ioctl_rt(const IOCTLEntry *ie, uint8_t *buf_temp,
const int *dst_offsets, *src_offsets; const int *dst_offsets, *src_offsets;
int target_size; int target_size;
void *argptr; void *argptr;
abi_ulong *target_rt_dev_ptr; abi_ulong *target_rt_dev_ptr = NULL;
unsigned long *host_rt_dev_ptr; unsigned long *host_rt_dev_ptr = NULL;
abi_long ret; abi_long ret;
int i; int i;
@ -4770,6 +4780,9 @@ static abi_long do_ioctl_rt(const IOCTLEntry *ie, uint8_t *buf_temp,
unlock_user(argptr, arg, 0); unlock_user(argptr, arg, 0);
ret = get_errno(safe_ioctl(fd, ie->host_cmd, buf_temp)); ret = get_errno(safe_ioctl(fd, ie->host_cmd, buf_temp));
assert(host_rt_dev_ptr != NULL);
assert(target_rt_dev_ptr != NULL);
if (*host_rt_dev_ptr != 0) { if (*host_rt_dev_ptr != 0) {
unlock_user((void *)*host_rt_dev_ptr, unlock_user((void *)*host_rt_dev_ptr,
*target_rt_dev_ptr, 0); *target_rt_dev_ptr, 0);
@ -6999,8 +7012,8 @@ static abi_long do_syscall1(void *cpu_env, int num, abi_long arg1,
_exit(arg1); _exit(arg1);
return 0; /* avoid warning */ return 0; /* avoid warning */
case TARGET_NR_read: case TARGET_NR_read:
if (arg3 == 0) { if (arg2 == 0 && arg3 == 0) {
return 0; return get_errno(safe_read(arg1, 0, 0));
} else { } else {
if (!(p = lock_user(VERIFY_WRITE, arg2, arg3, 0))) if (!(p = lock_user(VERIFY_WRITE, arg2, arg3, 0)))
return -TARGET_EFAULT; return -TARGET_EFAULT;