2 Unix SMB/CIFS implementation.
4 POSIX NTVFS backend - oplock handling
6 Copyright (C) Stefan Metzmacher 2008
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program. If not, see <http://www.gnu.org/licenses/>.
23 #include "lib/messaging/messaging.h"
24 #include "lib/messaging/irpc.h"
25 #include "system/time.h"
26 #include "vfs_posix.h"
30 struct pvfs_file_handle
*handle
;
31 struct pvfs_file
*file
;
33 struct timeval break_to_level_II
;
34 struct timeval break_to_none
;
35 struct imessaging_context
*msg_ctx
;
38 static NTSTATUS
pvfs_oplock_release_internal(struct pvfs_file_handle
*h
,
41 struct odb_lock
*olck
;
45 return NT_STATUS_FILE_IS_A_DIRECTORY
;
48 if (!h
->have_opendb_entry
) {
49 return NT_STATUS_FOOBAR
;
53 return NT_STATUS_FOOBAR
;
56 olck
= odb_lock(h
, h
->pvfs
->odb_context
, &h
->odb_locking_key
);
58 DEBUG(0,("Unable to lock opendb for oplock update\n"));
59 return NT_STATUS_FOOBAR
;
62 if (oplock_break
== OPLOCK_BREAK_TO_NONE
) {
63 h
->oplock
->level
= OPLOCK_NONE
;
64 } else if (oplock_break
== OPLOCK_BREAK_TO_LEVEL_II
) {
65 h
->oplock
->level
= OPLOCK_LEVEL_II
;
67 /* fallback to level II in case of a invalid value */
68 DEBUG(1,("unexpected oplock break level[0x%02X]\n", oplock_break
));
69 h
->oplock
->level
= OPLOCK_LEVEL_II
;
71 status
= odb_update_oplock(olck
, h
, h
->oplock
->level
);
72 if (!NT_STATUS_IS_OK(status
)) {
73 DEBUG(0,("Unable to update oplock level for '%s' - %s\n",
74 h
->name
->full_name
, nt_errstr(status
)));
81 /* after a break to none, we no longer have an oplock attached */
82 if (h
->oplock
->level
== OPLOCK_NONE
) {
83 talloc_free(h
->oplock
);
91 receive oplock breaks and forward them to the client
93 static void pvfs_oplock_break(struct pvfs_oplock
*opl
, uint8_t level
)
96 struct pvfs_file
*f
= opl
->file
;
97 struct pvfs_file_handle
*h
= opl
->handle
;
98 struct pvfs_state
*pvfs
= h
->pvfs
;
99 struct timeval cur
= timeval_current();
100 struct timeval
*last
= NULL
;
104 case OPLOCK_BREAK_TO_LEVEL_II
:
105 last
= &opl
->break_to_level_II
;
107 case OPLOCK_BREAK_TO_NONE
:
108 last
= &opl
->break_to_none
;
113 DEBUG(0,("%s: got unexpected level[0x%02X]\n",
114 __FUNCTION__
, level
));
118 if (timeval_is_zero(last
)) {
120 * this is the first break we for this level
125 DEBUG(5,("%s: sending oplock break level %d for '%s' %p\n",
126 __FUNCTION__
, level
, h
->name
->original_name
, h
));
127 status
= ntvfs_send_oplock_break(pvfs
->ntvfs
, f
->ntvfs
, level
);
128 if (!NT_STATUS_IS_OK(status
)) {
129 DEBUG(0,("%s: sending oplock break failed: %s\n",
130 __FUNCTION__
, nt_errstr(status
)));
135 end
= timeval_add(last
, pvfs
->oplock_break_timeout
, 0);
137 if (timeval_compare(&cur
, &end
) < 0) {
139 * If it's not expired just ignore the break
140 * as we already sent the break request to the client
142 DEBUG(0,("%s: do not resend oplock break level %d for '%s' %p\n",
143 __FUNCTION__
, level
, h
->name
->original_name
, h
));
148 * If the client did not send a release within the
149 * oplock break timeout time frame we auto release
152 DEBUG(0,("%s: auto release oplock level %d for '%s' %p\n",
153 __FUNCTION__
, level
, h
->name
->original_name
, h
));
154 status
= pvfs_oplock_release_internal(h
, level
);
155 if (!NT_STATUS_IS_OK(status
)) {
156 DEBUG(0,("%s: failed to auto release the oplock[0x%02X]: %s\n",
157 __FUNCTION__
, level
, nt_errstr(status
)));
161 static void pvfs_oplock_break_dispatch(struct imessaging_context
*msg
,
162 void *private_data
, uint32_t msg_type
,
163 struct server_id src
, DATA_BLOB
*data
)
165 struct pvfs_oplock
*opl
= talloc_get_type(private_data
,
167 struct opendb_oplock_break opb
;
171 /* we need to check that this one is for us. See
172 imessaging_send_ptr() for the other side of this.
174 if (data
->length
== sizeof(struct opendb_oplock_break
)) {
175 struct opendb_oplock_break
*p
;
176 p
= (struct opendb_oplock_break
*)data
->data
;
179 DEBUG(0,("%s: ignore oplock break with length[%u]\n",
180 __location__
, (unsigned)data
->length
));
183 if (opb
.file_handle
!= opl
->handle
) {
188 * maybe we should use ntvfs_setup_async()
190 pvfs_oplock_break(opl
, opb
.level
);
193 static int pvfs_oplock_destructor(struct pvfs_oplock
*opl
)
195 imessaging_deregister(opl
->msg_ctx
, MSG_NTVFS_OPLOCK_BREAK
, opl
);
199 NTSTATUS
pvfs_setup_oplock(struct pvfs_file
*f
, uint32_t oplock_granted
)
202 struct pvfs_oplock
*opl
;
203 uint32_t level
= OPLOCK_NONE
;
205 f
->handle
->oplock
= NULL
;
207 switch (oplock_granted
) {
208 case EXCLUSIVE_OPLOCK_RETURN
:
209 level
= OPLOCK_EXCLUSIVE
;
211 case BATCH_OPLOCK_RETURN
:
212 level
= OPLOCK_BATCH
;
214 case LEVEL_II_OPLOCK_RETURN
:
215 level
= OPLOCK_LEVEL_II
;
219 if (level
== OPLOCK_NONE
) {
223 opl
= talloc_zero(f
->handle
, struct pvfs_oplock
);
224 NT_STATUS_HAVE_NO_MEMORY(opl
);
226 opl
->handle
= f
->handle
;
229 opl
->msg_ctx
= f
->pvfs
->ntvfs
->ctx
->msg_ctx
;
231 status
= imessaging_register(opl
->msg_ctx
,
233 MSG_NTVFS_OPLOCK_BREAK
,
234 pvfs_oplock_break_dispatch
);
235 NT_STATUS_NOT_OK_RETURN(status
);
238 talloc_set_destructor(opl
, pvfs_oplock_destructor
);
240 f
->handle
->oplock
= opl
;
245 NTSTATUS
pvfs_oplock_release(struct ntvfs_module_context
*ntvfs
,
246 struct ntvfs_request
*req
, union smb_lock
*lck
)
248 struct pvfs_state
*pvfs
= talloc_get_type(ntvfs
->private_data
,
251 uint8_t oplock_break
;
254 f
= pvfs_find_fd(pvfs
, req
, lck
->lockx
.in
.file
.ntvfs
);
256 return NT_STATUS_INVALID_HANDLE
;
259 oplock_break
= (lck
->lockx
.in
.mode
>> 8) & 0xFF;
261 status
= pvfs_oplock_release_internal(f
->handle
, oplock_break
);
262 if (!NT_STATUS_IS_OK(status
)) {
263 DEBUG(0,("%s: failed to release the oplock[0x%02X]: %s\n",
264 __FUNCTION__
, oplock_break
, nt_errstr(status
)));
271 NTSTATUS
pvfs_break_level2_oplocks(struct pvfs_file
*f
)
273 struct pvfs_file_handle
*h
= f
->handle
;
274 struct odb_lock
*olck
;
277 if (h
->oplock
&& h
->oplock
->level
!= OPLOCK_LEVEL_II
) {
281 olck
= odb_lock(h
, h
->pvfs
->odb_context
, &h
->odb_locking_key
);
283 DEBUG(0,("Unable to lock opendb for oplock update\n"));
284 return NT_STATUS_FOOBAR
;
287 status
= odb_break_oplocks(olck
);
288 if (!NT_STATUS_IS_OK(status
)) {
289 DEBUG(0,("Unable to break level2 oplocks to none for '%s' - %s\n",
290 h
->name
->full_name
, nt_errstr(status
)));