| /* |
| * llseek.c -- stub calling the llseek system call |
| * |
| * Copyright (C) 1994, 1995, 1996, 1997 Theodore Ts'o. |
| * |
| * %Begin-Header% |
| * This file may be redistributed under the terms of the GNU Library |
| * General Public License, version 2. |
| * %End-Header% |
| */ |
| |
| #define _LARGEFILE_SOURCE |
| #define _LARGEFILE64_SOURCE |
| |
| #if HAVE_SYS_TYPES_H |
| #include <sys/types.h> |
| #endif |
| |
| #if HAVE_ERRNO_H |
| #include <errno.h> |
| #endif |
| #if HAVE_UNISTD_H |
| #include <unistd.h> |
| #endif |
| #ifdef __MSDOS__ |
| #include <io.h> |
| #endif |
| #include "et/com_err.h" |
| #include "ext2fs/ext2_io.h" |
| |
| #ifdef __linux__ |
| |
| #if defined(HAVE_LSEEK64) && defined(HAVE_LSEEK64_PROTOTYPE) |
| |
| #define my_llseek lseek64 |
| |
| #else |
| #if defined(HAVE_LLSEEK) |
| #include <syscall.h> |
| |
| #ifndef HAVE_LLSEEK_PROTOTYPE |
| extern long long llseek (int fd, long long offset, int origin); |
| #endif |
| |
| #define my_llseek llseek |
| |
| #else /* ! HAVE_LLSEEK */ |
| |
| #if SIZEOF_LONG == SIZEOF_LONG_LONG |
| |
| #define my_llseek lseek |
| |
| #else /* SIZEOF_LONG != SIZEOF_LONG_LONG */ |
| |
| #include <linux/unistd.h> |
| |
| #ifndef __NR__llseek |
| #define __NR__llseek 140 |
| #endif |
| |
| #ifndef __i386__ |
| static int _llseek (unsigned int, unsigned long, |
| unsigned long, ext2_loff_t *, unsigned int); |
| |
| static _syscall5(int,_llseek,unsigned int,fd,unsigned long,offset_high, |
| unsigned long, offset_low,ext2_loff_t *,result, |
| unsigned int, origin) |
| #endif |
| |
| static ext2_loff_t my_llseek (int fd, ext2_loff_t offset, int origin) |
| { |
| ext2_loff_t result; |
| int retval; |
| |
| #ifndef __i386__ |
| retval = _llseek(fd, ((unsigned long long) offset) >> 32, |
| #else |
| retval = syscall(__NR__llseek, fd, (unsigned long long) (offset >> 32), |
| #endif |
| ((unsigned long long) offset) & 0xffffffff, |
| &result, origin); |
| return (retval == -1 ? (ext2_loff_t) retval : result); |
| } |
| |
| #endif /* SIZE_LONG == SIZEOF_LONG_LONG */ |
| |
| #endif /* HAVE_LLSEEK */ |
| #endif /* defined(HAVE_LSEEK64) && defined(HAVE_LSEEK64_PROTOTYPE) */ |
| |
| ext2_loff_t ext2fs_llseek (int fd, ext2_loff_t offset, int origin) |
| { |
| #if SIZEOF_OFF_T >= SIZEOF_LONG_LONG |
| return my_llseek (fd, offset, origin); |
| #else |
| ext2_loff_t result; |
| static int do_compat = 0; |
| |
| if (do_compat) |
| goto fallback; |
| |
| result = my_llseek (fd, offset, origin); |
| if (result == -1 && errno == ENOSYS) { |
| /* |
| * Just in case this code runs on top of an old kernel |
| * which does not support the llseek system call |
| */ |
| do_compat++; |
| fallback: |
| if (offset < ((ext2_loff_t) 1 << ((sizeof(off_t)*8) -1))) |
| return lseek(fd, (off_t) offset, origin); |
| errno = EINVAL; |
| return -1; |
| } |
| return result; |
| #endif |
| } |
| |
| #else /* !linux */ |
| |
| #ifndef EINVAL |
| #define EINVAL EXT2_ET_INVALID_ARGUMENT |
| #endif |
| |
| ext2_loff_t ext2fs_llseek (int fd, ext2_loff_t offset, int origin) |
| { |
| #if defined(HAVE_LSEEK64) && defined(HAVE_LSEEK64_PROTOTYPE) |
| return lseek64 (fd, offset, origin); |
| #else |
| if ((sizeof(off_t) < sizeof(ext2_loff_t)) && |
| (offset >= ((ext2_loff_t) 1 << ((sizeof(off_t)*8) -1)))) { |
| errno = EINVAL; |
| return -1; |
| } |
| return lseek (fd, (off_t) offset, origin); |
| #endif |
| } |
| |
| #endif /* linux */ |
| |
| |