]> arthur.barton.de Git - netatalk.git/blob - libatalk/adouble/ad_sendfile.c
on linux put sendfile back
[netatalk.git] / libatalk / adouble / ad_sendfile.c
1 /*
2  * $Id: ad_sendfile.c,v 1.10 2009-10-25 09:47:04 didg Exp $
3  *
4  * Copyright (c) 1999 Adrian Sun (asun@zoology.washington.edu)
5  * All rights reserved. See COPYRIGHT.
6  *
7  * NOTE: the following uses the fact that sendfile() only exists on
8  * machines with SA_RESTART behaviour. this is all very machine specific. 
9  *
10  * sendfile chainsaw from samba.
11  Unix SMB/Netbios implementation.
12  Version 2.2.x / 3.0.x
13  sendfile implementations.
14  Copyright (C) Jeremy Allison 2002.
15  
16  This program is free software; you can redistribute it and/or modify
17  it under the terms of the GNU General Public License as published by
18  the Free Software Foundation; either version 2 of the License, or
19  (at your option) any later version.
20  This program is distributed in the hope that it will be useful,
21  but WITHOUT ANY WARRANTY; without even the implied warranty of
22  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
23  GNU General Public License for more details.
24  
25  You should have received a copy of the GNU General Public License
26  along with this program; if not, write to the Free Software
27  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
28  */
29
30 #ifdef HAVE_CONFIG_H
31 #include "config.h"
32 #endif /* HAVE_CONFIG_H */
33
34 #ifdef WITH_SENDFILE
35
36 #include <atalk/adouble.h>
37
38 #include <stdio.h>
39
40 #include <sys/socket.h>
41 #include <sys/uio.h>
42
43 #include <errno.h>  
44
45 #include <atalk/logger.h>
46 #include "ad_private.h"
47
48 #if defined(LINUX_BROKEN_SENDFILE_API)
49
50 extern int32_t sendfile (int fdout, int fdin, int32_t *offset, u_int32_t count);
51
52 ssize_t sys_sendfile(int tofd, int fromfd, off_t *offset, size_t count)
53 {
54 u_int32_t small_total;
55 int32_t small_offset;
56 int32_t nwritten;
57
58     /*
59      * Fix for broken Linux 2.4 systems with no working sendfile64().
60      * If the offset+count > 2 GB then pretend we don't have the
61      * system call sendfile at all. The upper layer catches this
62      * and uses a normal read. JRA.
63      */
64  
65      if ((sizeof(off_t) >= 8) && (*offset + count > (off_t)0x7FFFFFFF)) {
66          errno = ENOSYS;
67          return -1;
68      }
69      small_offset = (int32_t)*offset;
70      small_total = (u_int32_t)count;
71      nwritten = sendfile(tofd, fromfd, &small_offset, small_total);
72      if (nwritten > = 0)
73          *offset += nwritten;
74      
75     return nwritten;
76 }
77
78 #elif defined(SENDFILE_FLAVOR_LINUX)
79 #include <sys/sendfile.h>
80
81 ssize_t sys_sendfile(int tofd, int fromfd, off_t *offset, size_t count)
82 {
83     return sendfile(tofd, fromfd, offset, count);
84 }
85
86
87 #elif defined(SENDFILE_FLAVOR_BSD )
88 /* FIXME untested */
89 #error sendfile semantic broken
90 #include <sys/sendfile.h>
91 ssize_t sys_sendfile(int tofd, int fromfd, off_t *offset, size_t count)
92 {
93 size_t total=0;
94 int    ret;
95
96     total = count;
97     while (total) {
98         ssize_t nwritten;
99         do {
100            ret = sendfile(fromfd, tofd, offset, count, NULL, &nwritten, 0);
101         while (ret == -1 && errno == EINTR);
102         if (ret == -1)
103             return -1;
104         total -= nwritten;
105         offset += nwritten;
106     }
107     return count;
108 }
109
110 #else
111
112 ssize_t sys_sendfile(int out_fd, int in_fd, off_t *_offset, size_t count)
113 {
114     /* No sendfile syscall. */
115     errno = ENOSYS;
116     return -1;
117 }
118 #endif
119
120 /* ------------------------------- */
121 int ad_readfile_init(const struct adouble *ad, 
122                                        const int eid, off_t *off,
123                                        const int end)
124 {
125   int fd;
126
127   if (end) 
128     *off = ad_size(ad, eid) - *off;
129
130   if (eid == ADEID_DFORK) {
131     fd = ad_data_fileno(ad);
132   } else {
133     *off += ad_getentryoff(ad, eid);
134     fd = ad_reso_fileno(ad);
135   }
136
137   return fd;
138 }
139
140
141 /* ------------------------ */
142 #if 0
143 #ifdef HAVE_SENDFILE_WRITE
144 /* read from a socket and write to an adouble file */
145 ssize_t ad_writefile(struct adouble *ad, const int eid, 
146                      const int sock, off_t off, const int end,
147                      const size_t len)
148 {
149 #ifdef __linux__
150   ssize_t cc;
151   int fd;
152
153   fd = ad_sendfile_init(ad, eid, &off, end);
154   if ((cc = sys_sendfile(fd, sock, &off, len)) < 0)
155     return -1;
156
157   if ((eid != ADEID_DFORK) && (off > ad_getentrylen(ad, eid))) 
158     ad_setentrylen(ad, eid, off);
159
160   return cc;
161 #endif /* __linux__ */
162 }
163 #endif /* HAVE_SENDFILE_WRITE */
164 #endif /* 0 */
165 #endif