* In kernel, off_t is always the same as (kernel's) long
* (see include/uapi/asm-generic/posix_types.h),
* which means that on x32 we need to use tcp->ext_arg[N] to get offset argument.
+ * Use test/x32_lseek.c to test lseek decoding.
*/
#if defined(LINUX_MIPSN32) || defined(X32)
int
}
return RVAL_LUDECIMAL;
}
-
-# if defined(X32)
-int
-sys_lseek32(struct tcb *tcp)
-{
- long offset;
- int whence;
-
- if (entering(tcp)) {
- printfd(tcp, tcp->u_arg[0]);
- offset = tcp->u_arg[1];
- whence = tcp->u_arg[2];
- if (whence == SEEK_SET)
- tprintf(", %lu, ", offset);
- else
- tprintf(", %ld, ", offset);
- printxval(whence_codes, whence, "SEEK_???");
- }
- return RVAL_UDECIMAL;
-}
-# endif
#else
int
sys_lseek(struct tcb *tcp)
*
* hi,lo are "unsigned longs" and combined exactly this way in kernel:
* ((loff_t) hi << 32) | lo
- * Note that for architectures with kernel's long wider than userspace longs
+ * Note that for architectures with kernel's long wider than userspace long
* (such as x32), combining code will use *kernel's*, i.e. *wide* longs
- * for hi and lo. You may need to use tcp->ext_arg[N]!
+ * for hi and lo. We may need to use tcp->ext_arg[N]!
*/
int
sys_llseek(struct tcb *tcp)
#define sys_oldstat printargs
#define sys_oldfstat printargs
#define sys_oldlstat printargs
-#define sys_lseek sys_lseek32
#define sys_lstat64 sys_stat64
#define sys_truncate64 sys_truncate
#define sys_ftruncate64 sys_ftruncate
--- /dev/null
+// Test program which explores whether lseek syscall (not llseek!)
+// on x32 uses 64-bit offset argument.
+// IOW: does _kernel_ truncate it on entry?
+// The answer appears to be "no, full 64-bit offset is used".
+// strace must show it correctly too - tricky if strace itself is x32 one!
+//
+// Build: x86_64-gcc -static -Wall -ox32_lseek x32_lseek.c
+// Run: $ strace ./x32_lseek 2>&1 | grep lseek | grep 1250999896321
+// lseek(0, 1250999896321, SEEK_SET) = 1250999896321
+#define _GNU_SOURCE
+#include <sys/types.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <errno.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <stdio.h>
+#include <sys/syscall.h>
+// Ensure we are compiling to 64 bits
+struct bug { int t[sizeof(long) > 4 ? 1 : -1]; };
+int main(int argc, char **argv)
+{
+ long ofs = 0x12345678901;
+ errno = 0;
+ close(0);
+ if (open("/etc/passwd", O_RDONLY))
+ return 1;
+ long r = syscall(
+ (long) (__NR_lseek | 0x40000000), // make x32 call
+ (long) (0),
+ (long) (ofs),
+ (long) (SEEK_SET)
+ );
+ printf("pos:%ld(0x%lx) errno:%m\n", r, r);
+ if (!errno)
+ printf((r == ofs) ? "64-bit offset used\n" : "Kernel truncated offset\n");
+ return 0;
+}