Commit 9d33b76b205194857f17ca35a7581b1f500f0462
1 parent
f1afe02a
linux-user: fix fstatat64()/newfstatat() syscall implementation
There are two different syscall names for the same goal. On systems with sizeof(long) == 64 it calls newfstatat. On systems with sizeof(long) == 32 it calls fstatat64. Signed-off-by: Kirill A. Shutemov <kirill@shutemov.name> Signed-off-by: Riku Voipio <riku.voipio@iki.fi> Signed-off-by: Aurelien Jarno <aurelien@aurel32.net> git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@7050 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
1 changed file
with
25 additions
and
4 deletions
linux-user/syscall.c
@@ -169,6 +169,7 @@ static type name (type1 arg1,type2 arg2,type3 arg3,type4 arg4,type5 arg5, \ | @@ -169,6 +169,7 @@ static type name (type1 arg1,type2 arg2,type3 arg3,type4 arg4,type5 arg5, \ | ||
169 | #define __NR_sys_linkat __NR_linkat | 169 | #define __NR_sys_linkat __NR_linkat |
170 | #define __NR_sys_mkdirat __NR_mkdirat | 170 | #define __NR_sys_mkdirat __NR_mkdirat |
171 | #define __NR_sys_mknodat __NR_mknodat | 171 | #define __NR_sys_mknodat __NR_mknodat |
172 | +#define __NR_sys_newfstatat __NR_newfstatat | ||
172 | #define __NR_sys_openat __NR_openat | 173 | #define __NR_sys_openat __NR_openat |
173 | #define __NR_sys_readlinkat __NR_readlinkat | 174 | #define __NR_sys_readlinkat __NR_readlinkat |
174 | #define __NR_sys_renameat __NR_renameat | 175 | #define __NR_sys_renameat __NR_renameat |
@@ -209,7 +210,8 @@ _syscall4(int,sys_fchmodat,int,dirfd,const char *,pathname, | @@ -209,7 +210,8 @@ _syscall4(int,sys_fchmodat,int,dirfd,const char *,pathname, | ||
209 | _syscall5(int,sys_fchownat,int,dirfd,const char *,pathname, | 210 | _syscall5(int,sys_fchownat,int,dirfd,const char *,pathname, |
210 | uid_t,owner,gid_t,group,int,flags) | 211 | uid_t,owner,gid_t,group,int,flags) |
211 | #endif | 212 | #endif |
212 | -#if defined(TARGET_NR_fstatat64) && defined(__NR_fstatat64) | 213 | +#if (defined(TARGET_NR_fstatat64) || defined(TARGET_NR_newfstatat)) && \ |
214 | + defined(__NR_fstatat64) | ||
213 | _syscall4(int,sys_fstatat64,int,dirfd,const char *,pathname, | 215 | _syscall4(int,sys_fstatat64,int,dirfd,const char *,pathname, |
214 | struct stat *,buf,int,flags) | 216 | struct stat *,buf,int,flags) |
215 | #endif | 217 | #endif |
@@ -240,6 +242,11 @@ _syscall3(int,sys_mkdirat,int,dirfd,const char *,pathname,mode_t,mode) | @@ -240,6 +242,11 @@ _syscall3(int,sys_mkdirat,int,dirfd,const char *,pathname,mode_t,mode) | ||
240 | _syscall4(int,sys_mknodat,int,dirfd,const char *,pathname, | 242 | _syscall4(int,sys_mknodat,int,dirfd,const char *,pathname, |
241 | mode_t,mode,dev_t,dev) | 243 | mode_t,mode,dev_t,dev) |
242 | #endif | 244 | #endif |
245 | +#if (defined(TARGET_NR_newfstatat) || defined(TARGET_NR_fstatat64) ) && \ | ||
246 | + defined(__NR_newfstatat) | ||
247 | +_syscall4(int,sys_newfstatat,int,dirfd,const char *,pathname, | ||
248 | + struct stat *,buf,int,flags) | ||
249 | +#endif | ||
243 | #if defined(TARGET_NR_openat) && defined(__NR_openat) | 250 | #if defined(TARGET_NR_openat) && defined(__NR_openat) |
244 | _syscall4(int,sys_openat,int,dirfd,const char *,pathname,int,flags,mode_t,mode) | 251 | _syscall4(int,sys_openat,int,dirfd,const char *,pathname,int,flags,mode_t,mode) |
245 | #endif | 252 | #endif |
@@ -3280,7 +3287,7 @@ static inline abi_long host_to_target_timespec(abi_ulong target_addr, | @@ -3280,7 +3287,7 @@ static inline abi_long host_to_target_timespec(abi_ulong target_addr, | ||
3280 | return 0; | 3287 | return 0; |
3281 | } | 3288 | } |
3282 | 3289 | ||
3283 | -#ifdef TARGET_NR_stat64 | 3290 | +#if defined(TARGET_NR_stat64) || defined(TARGET_NR_newfstatat) |
3284 | static inline abi_long host_to_target_stat64(void *cpu_env, | 3291 | static inline abi_long host_to_target_stat64(void *cpu_env, |
3285 | abi_ulong target_addr, | 3292 | abi_ulong target_addr, |
3286 | struct stat *host_st) | 3293 | struct stat *host_st) |
@@ -3312,11 +3319,15 @@ static inline abi_long host_to_target_stat64(void *cpu_env, | @@ -3312,11 +3319,15 @@ static inline abi_long host_to_target_stat64(void *cpu_env, | ||
3312 | } else | 3319 | } else |
3313 | #endif | 3320 | #endif |
3314 | { | 3321 | { |
3322 | +#if TARGET_LONG_BITS == 64 | ||
3323 | + struct target_stat *target_st; | ||
3324 | +#else | ||
3315 | struct target_stat64 *target_st; | 3325 | struct target_stat64 *target_st; |
3326 | +#endif | ||
3316 | 3327 | ||
3317 | if (!lock_user_struct(VERIFY_WRITE, target_st, target_addr, 0)) | 3328 | if (!lock_user_struct(VERIFY_WRITE, target_st, target_addr, 0)) |
3318 | return -TARGET_EFAULT; | 3329 | return -TARGET_EFAULT; |
3319 | - memset(target_st, 0, sizeof(struct target_stat64)); | 3330 | + memset(target_st, 0, sizeof(*target_st)); |
3320 | __put_user(host_st->st_dev, &target_st->st_dev); | 3331 | __put_user(host_st->st_dev, &target_st->st_dev); |
3321 | __put_user(host_st->st_ino, &target_st->st_ino); | 3332 | __put_user(host_st->st_ino, &target_st->st_ino); |
3322 | #ifdef TARGET_STAT64_HAS_BROKEN_ST_INO | 3333 | #ifdef TARGET_STAT64_HAS_BROKEN_ST_INO |
@@ -5459,11 +5470,21 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1, | @@ -5459,11 +5470,21 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1, | ||
5459 | ret = host_to_target_stat64(cpu_env, arg2, &st); | 5470 | ret = host_to_target_stat64(cpu_env, arg2, &st); |
5460 | break; | 5471 | break; |
5461 | #endif | 5472 | #endif |
5462 | -#if defined(TARGET_NR_fstatat64) && defined(__NR_fstatat64) | 5473 | +#if (defined(TARGET_NR_fstatat64) || defined(TARGET_NR_newfstatat)) && \ |
5474 | + (defined(__NR_fstatat64) || defined(__NR_newfstatat)) | ||
5475 | +#ifdef TARGET_NR_fstatat64 | ||
5463 | case TARGET_NR_fstatat64: | 5476 | case TARGET_NR_fstatat64: |
5477 | +#endif | ||
5478 | +#ifdef TARGET_NR_newfstatat | ||
5479 | + case TARGET_NR_newfstatat: | ||
5480 | +#endif | ||
5464 | if (!(p = lock_user_string(arg2))) | 5481 | if (!(p = lock_user_string(arg2))) |
5465 | goto efault; | 5482 | goto efault; |
5483 | +#ifdef __NR_fstatat64 | ||
5466 | ret = get_errno(sys_fstatat64(arg1, path(p), &st, arg4)); | 5484 | ret = get_errno(sys_fstatat64(arg1, path(p), &st, arg4)); |
5485 | +#else | ||
5486 | + ret = get_errno(sys_newfstatat(arg1, path(p), &st, arg4)); | ||
5487 | +#endif | ||
5467 | if (!is_error(ret)) | 5488 | if (!is_error(ret)) |
5468 | ret = host_to_target_stat64(cpu_env, arg3, &st); | 5489 | ret = host_to_target_stat64(cpu_env, arg3, &st); |
5469 | break; | 5490 | break; |