Browse Source

Implement sendfile64
-Erik

Eric Andersen 21 years ago
parent
commit
1a953df3b4

+ 23 - 8
include/sys/sendfile.h

@@ -1,4 +1,5 @@
-/* Copyright (C) 1998, 1999, 2001 Free Software Foundation, Inc.
+/* sendfile -- copy data directly from one file descriptor to another
+   Copyright (C) 1998,99,01,2002 Free Software Foundation, Inc.
    This file is part of the GNU C Library.
 
    The GNU C Library is free software; you can redistribute it and/or
@@ -22,16 +23,30 @@
 #include <features.h>
 #include <sys/types.h>
 
-#ifdef __USE_FILE_OFFSET64
-# error "<sys/sendfile.h> cannot be used with _FILE_OFFSET_BITS=64"
-#endif
-
 __BEGIN_DECLS
 
-/* Send COUNT bytes from file associated with IN_FD starting at OFFSET to
-   descriptor OUT_FD.  */
-extern ssize_t sendfile (int __out_fd, int __in_fd, off_t *offset,
+/* Send up to COUNT bytes from file associated with IN_FD starting at
+   *OFFSET to descriptor OUT_FD.  Set *OFFSET to the IN_FD's file position
+   following the read bytes.  If OFFSET is a null pointer, use the normal
+   file position instead.  Return the number of written bytes, or -1 in
+   case of error.  */
+#ifndef __USE_FILE_OFFSET64
+extern ssize_t sendfile (int __out_fd, int __in_fd, off_t *__offset,
 			 size_t __count) __THROW;
+#else
+# ifdef __REDIRECT
+extern ssize_t __REDIRECT (sendfile,
+			   (int __out_fd, int __in_fd, __off64_t *__offset,
+			    size_t __count) __THROW,
+			   sendfile64);
+# else
+#  define sendfile sendfile64
+# endif
+#endif
+#ifdef __USE_LARGEFILE64
+extern ssize_t sendfile64 (int __out_fd, int __in_fd, __off64_t *__offset,
+			   size_t __count) __THROW;
+#endif
 
 __END_DECLS
 

+ 2 - 1
libc/sysdeps/linux/common/Makefile

@@ -34,7 +34,8 @@ CSRC=	waitpid.c getdnnm.c gethstnm.c getcwd.c \
 	wait3.c setpgrp.c getdtablesize.c create_module.c ptrace.c \
 	cmsg_nxthdr.c longjmp.c open64.c ftruncate64.c \
 	truncate64.c getrlimit64.c setrlimit64.c creat64.c mmap64.c \
-	llseek.c pread_write.c _exit.c setuid.c sync.c getdirname.c
+	llseek.c pread_write.c _exit.c setuid.c sync.c getdirname.c \
+	sendfile64.c
 ifneq ($(strip $(EXCLUDE_BRK)),y)
 CSRC+=sbrk.c
 endif

+ 46 - 0
libc/sysdeps/linux/common/sendfile64.c

@@ -0,0 +1,46 @@
+/*
+ * sendfile64 syscall.  Copes with 64 bit and 32 bit machines
+ * and on 32 bit machines this sends things into the kernel as
+ * two 32-bit arguments (high and low 32 bits of length) that 
+ * are ordered based on endianess.  It turns out endian.h has
+ * just the macro we need to order things, __LONG_LONG_PAIR.
+ *
+ *  Copyright (C) 2002  Erik Andersen <andersen@codepoet.org>
+ *
+ * This file is subject to the terms and conditions of the GNU
+ * Lesser General Public License.  See the file COPYING.LIB in
+ * the main directory of this archive for more details.
+ */
+
+#include <features.h>
+#include <unistd.h>
+#include <errno.h>
+#include <endian.h>
+#include <stdint.h>
+#include <sys/sendfile.h>
+#include <sys/syscall.h>
+
+
+#if __WORDSIZE == 64
+/* For a 64 bit machine, life is simple... */
+_syscall4(ssize_t,sendfile64, int, out_fd, int, in_fd, __off64_t *, offset, size_t, count)
+
+#elif __WORDSIZE == 32
+
+#if defined __UCLIBC_HAS_LFS__
+#if defined __NR_sendfile64
+_syscall4(ssize_t,sendfile64, int, out_fd, int, in_fd, __off64_t *, offset, size_t, count)
+
+#else
+ssize_t sendfile64 (int out_fd, int in_fd, __off64_t *offset, size_t count)
+{
+  __set_errno (ENOSYS);
+  return -1;
+}
+#endif
+
+#endif /* __UCLIBC_HAS_LFS__ */
+
+#else /* __WORDSIZE */
+#error Your machine is not 64 bit or 32 bit, I am dazed and confused.
+#endif