s3: copy nbt/netlogon helper from s4.
[Samba.git] / source / modules / vfs_commit.c
blob1cef6d02433b1e6d5754fed896c1057a8a7d0a67
1 /*
2 * Copyright (c) James Peach 2006, 2007
3 * Copyright (c) David Losada Carballo 2007
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 3 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, see <http://www.gnu.org/licenses/>.
19 #include "includes.h"
21 /* Commit data module.
23 * The purpose of this module is to flush data to disk at regular intervals,
24 * just like the NFS commit operation. There's two rationales for this. First,
25 * it minimises the data loss in case of a power outage without incurring
26 * the poor performance of synchronous I/O. Second, a steady flush rate
27 * can produce better throughput than suddenly dumping massive amounts of
28 * writes onto a disk.
30 * Tunables:
32 * commit: dthresh Amount of dirty data that can accumulate
33 * before we commit (sync) it.
35 * commit: debug Debug level at which to emit messages.
37 * commit: eof mode String. Tunes how the module tries to guess when
38 * the client has written the last bytes of the file.
39 * Possible values (default = hinted):
41 * (*) = hinted Some clients (i.e. Windows Explorer) declare the
42 * size of the file before transferring it. With this
43 * option, we remember that hint, and commit after
44 * writing in that file position. If the client
45 * doesn't declare the size of file, commiting on EOF
46 * is not triggered.
48 * = growth Commits after a write operation has made the file
49 * size grow. If the client declares a file size, it
50 * refrains to commit until the file has reached it.
51 * Useful for defeating writeback on NFS shares.
55 #define MODULE "commit"
57 static int module_debug;
59 enum eof_mode
61 EOF_NONE = 0x0000,
62 EOF_HINTED = 0x0001,
63 EOF_GROWTH = 0x0002
66 struct commit_info
68 /* For chunk-based commits */
69 SMB_OFF_T dbytes; /* Dirty (uncommitted) bytes */
70 SMB_OFF_T dthresh; /* Dirty data threshold */
71 /* For commits on EOF */
72 enum eof_mode on_eof;
73 SMB_OFF_T eof; /* Expected file size */
76 static int commit_do(
77 struct commit_info * c,
78 int fd)
80 int result;
82 DEBUG(module_debug,
83 ("%s: flushing %lu dirty bytes\n",
84 MODULE, (unsigned long)c->dbytes));
86 #if HAVE_FDATASYNC
87 result = fdatasync(fd);
88 #elif HAVE_FSYNC
89 result = fsync(fd);
90 #else
91 result = 0
92 #endif
93 if (result == 0) {
94 c->dbytes = 0; /* on success, no dirty bytes */
96 return result;
99 static int commit_all(
100 struct vfs_handle_struct * handle,
101 files_struct * fsp)
103 struct commit_info *c;
105 if ((c = VFS_FETCH_FSP_EXTENSION(handle, fsp))) {
106 if (c->dbytes) {
107 DEBUG(module_debug,
108 ("%s: flushing %lu dirty bytes\n",
109 MODULE, (unsigned long)c->dbytes));
111 return commit_do(c, fsp->fh->fd);
114 return 0;
117 static int commit(
118 struct vfs_handle_struct * handle,
119 files_struct * fsp,
120 SMB_OFF_T offset,
121 ssize_t last_write)
123 struct commit_info *c;
125 if ((c = VFS_FETCH_FSP_EXTENSION(handle, fsp)) == NULL) {
126 return 0;
129 c->dbytes += last_write; /* dirty bytes always counted */
131 if (c->dthresh && (c->dbytes > c->dthresh)) {
132 return commit_do(c, fsp->fh->fd);
135 /* Return if we are not in EOF mode or if we have temporarily opted
136 * out of it.
138 if (c->on_eof == EOF_NONE || c->eof < 0) {
139 return 0;
142 /* This write hit or went past our cache the file size. */
143 if ((offset + last_write) >= c->eof) {
144 if (commit_do(c, fsp->fh->fd) == -1) {
145 return -1;
148 /* Hinted mode only commits the first time we hit EOF. */
149 if (c->on_eof == EOF_HINTED) {
150 c->eof = -1;
151 } else if (c->on_eof == EOF_GROWTH) {
152 c->eof = offset + last_write;
156 return 0;
159 static int commit_connect(
160 struct vfs_handle_struct * handle,
161 const char * service,
162 const char * user)
164 module_debug = lp_parm_int(SNUM(handle->conn), MODULE, "debug", 100);
165 return SMB_VFS_NEXT_CONNECT(handle, service, user);
168 static int commit_open(
169 vfs_handle_struct * handle,
170 const char * fname,
171 files_struct * fsp,
172 int flags,
173 mode_t mode)
175 SMB_OFF_T dthresh;
176 const char *eof_mode;
177 struct commit_info *c = NULL;
178 int fd;
180 /* Don't bother with read-only files. */
181 if ((flags & O_ACCMODE) == O_RDONLY) {
182 return SMB_VFS_NEXT_OPEN(handle, fname, fsp, flags, mode);
185 /* Read and check module configuration */
186 dthresh = conv_str_size(lp_parm_const_string(SNUM(handle->conn),
187 MODULE, "dthresh", NULL));
189 eof_mode = lp_parm_const_string(SNUM(handle->conn),
190 MODULE, "eof mode", "none");
192 if (dthresh > 0 || !strequal(eof_mode, "none")) {
193 c = VFS_ADD_FSP_EXTENSION(handle, fsp, struct commit_info);
194 /* Process main tunables */
195 if (c) {
196 c->dthresh = dthresh;
197 c->dbytes = 0;
198 c->on_eof = EOF_NONE;
199 c->eof = 0;
202 /* Process eof_mode tunable */
203 if (c) {
204 if (strequal(eof_mode, "hinted")) {
205 c->on_eof = EOF_HINTED;
206 } else if (strequal(eof_mode, "growth")) {
207 c->on_eof = EOF_GROWTH;
211 fd = SMB_VFS_NEXT_OPEN(handle, fname, fsp, flags, mode);
212 if (fd == -1) {
213 VFS_REMOVE_FSP_EXTENSION(handle, fsp);
214 return fd;
217 /* EOF commit modes require us to know the initial file size. */
218 if (c && (c->on_eof != EOF_NONE)) {
219 SMB_STRUCT_STAT st;
220 if (SMB_VFS_FSTAT(fsp, &st) == -1) {
221 return -1;
223 c->eof = st.st_size;
226 return 0;
229 static ssize_t commit_write(
230 vfs_handle_struct * handle,
231 files_struct * fsp,
232 void * data,
233 size_t count)
235 ssize_t ret;
236 ret = SMB_VFS_NEXT_WRITE(handle, fsp, data, count);
238 if (ret > 0) {
239 if (commit(handle, fsp, fsp->fh->pos, ret) == -1) {
240 return -1;
244 return ret;
247 static ssize_t commit_pwrite(
248 vfs_handle_struct * handle,
249 files_struct * fsp,
250 void * data,
251 size_t count,
252 SMB_OFF_T offset)
254 ssize_t ret;
256 ret = SMB_VFS_NEXT_PWRITE(handle, fsp, data, count, offset);
257 if (ret > 0) {
258 if (commit(handle, fsp, offset, ret) == -1) {
259 return -1;
263 return ret;
266 static int commit_close(
267 vfs_handle_struct * handle,
268 files_struct * fsp)
270 /* Commit errors not checked, close() will find them again */
271 commit_all(handle, fsp);
272 return SMB_VFS_NEXT_CLOSE(handle, fsp);
275 static int commit_ftruncate(
276 vfs_handle_struct * handle,
277 files_struct * fsp,
278 SMB_OFF_T len)
280 int result;
282 result = SMB_VFS_NEXT_FTRUNCATE(handle, fsp, len);
283 if (result == 0) {
284 struct commit_info *c;
285 if ((c = VFS_FETCH_FSP_EXTENSION(handle, fsp))) {
286 commit(handle, fsp, len, 0);
287 c->eof = len;
291 return result;
294 static vfs_op_tuple commit_ops [] =
296 {SMB_VFS_OP(commit_open),
297 SMB_VFS_OP_OPEN, SMB_VFS_LAYER_TRANSPARENT},
298 {SMB_VFS_OP(commit_close),
299 SMB_VFS_OP_CLOSE, SMB_VFS_LAYER_TRANSPARENT},
300 {SMB_VFS_OP(commit_write),
301 SMB_VFS_OP_WRITE, SMB_VFS_LAYER_TRANSPARENT},
302 {SMB_VFS_OP(commit_pwrite),
303 SMB_VFS_OP_PWRITE, SMB_VFS_LAYER_TRANSPARENT},
304 {SMB_VFS_OP(commit_connect),
305 SMB_VFS_OP_CONNECT, SMB_VFS_LAYER_TRANSPARENT},
306 {SMB_VFS_OP(commit_ftruncate),
307 SMB_VFS_OP_FTRUNCATE, SMB_VFS_LAYER_TRANSPARENT},
309 {SMB_VFS_OP(NULL), SMB_VFS_OP_NOOP, SMB_VFS_LAYER_NOOP}
312 NTSTATUS vfs_commit_init(void);
313 NTSTATUS vfs_commit_init(void)
315 return smb_register_vfs(SMB_VFS_INTERFACE_VERSION, MODULE, commit_ops);