Line data Source code
1 : /*
2 : Unix SMB/CIFS implementation.
3 : Watch dbwrap record changes
4 : Copyright (C) Volker Lendecke 2012
5 :
6 : This program is free software; you can redistribute it and/or modify
7 : it under the terms of the GNU General Public License as published by
8 : the Free Software Foundation; either version 3 of the License, or
9 : (at your option) any later version.
10 :
11 : This program is distributed in the hope that it will be useful,
12 : but WITHOUT ANY WARRANTY; without even the implied warranty of
13 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 : GNU General Public License for more details.
15 :
16 : You should have received a copy of the GNU General Public License
17 : along with this program. If not, see <http://www.gnu.org/licenses/>.
18 : */
19 :
20 : #include "includes.h"
21 : #include "system/filesys.h"
22 : #include "lib/util/server_id.h"
23 : #include "dbwrap/dbwrap.h"
24 : #include "dbwrap_watch.h"
25 : #include "dbwrap_open.h"
26 : #include "lib/util/util_tdb.h"
27 : #include "lib/util/tevent_ntstatus.h"
28 : #include "serverid.h"
29 : #include "server_id_watch.h"
30 : #include "lib/dbwrap/dbwrap_private.h"
31 :
32 : struct dbwrap_watcher {
33 : /*
34 : * Process watching this record
35 : */
36 : struct server_id pid;
37 : /*
38 : * Individual instance inside the waiter, incremented each
39 : * time a watcher is created
40 : */
41 : uint64_t instance;
42 : };
43 :
44 : #define DBWRAP_WATCHER_BUF_LENGTH (SERVER_ID_BUF_LENGTH + sizeof(uint64_t))
45 : #define DBWRAP_MAX_WATCHERS (INT32_MAX/DBWRAP_WATCHER_BUF_LENGTH)
46 :
47 : /*
48 : * Watched records contain a header of:
49 : *
50 : * [uint32] num_records
51 : * 0 [DBWRAP_WATCHER_BUF_LENGTH] \
52 : * 1 [DBWRAP_WATCHER_BUF_LENGTH] |
53 : * .. |- Array of watchers
54 : * (num_records-1)[DBWRAP_WATCHER_BUF_LENGTH] /
55 : *
56 : * [Remainder of record....]
57 : *
58 : * If this header is absent then this is a
59 : * fresh record of length zero (no watchers).
60 : */
61 :
62 216498 : static bool dbwrap_watch_rec_parse(
63 : TDB_DATA data,
64 : uint8_t **pwatchers,
65 : size_t *pnum_watchers,
66 : TDB_DATA *pdata)
67 : {
68 : size_t num_watchers;
69 :
70 216498 : if (data.dsize == 0) {
71 : /* Fresh record */
72 18875 : if (pwatchers != NULL) {
73 18875 : *pwatchers = NULL;
74 : }
75 18875 : if (pnum_watchers != NULL) {
76 18875 : *pnum_watchers = 0;
77 : }
78 18875 : if (pdata != NULL) {
79 18875 : *pdata = (TDB_DATA) { .dptr = NULL };
80 : }
81 18875 : return true;
82 : }
83 :
84 197623 : if (data.dsize < sizeof(uint32_t)) {
85 : /* Invalid record */
86 0 : return false;
87 : }
88 :
89 197623 : num_watchers = IVAL(data.dptr, 0);
90 :
91 197623 : data.dptr += sizeof(uint32_t);
92 197623 : data.dsize -= sizeof(uint32_t);
93 :
94 197623 : if (num_watchers > data.dsize/DBWRAP_WATCHER_BUF_LENGTH) {
95 : /* Invalid record */
96 0 : return false;
97 : }
98 :
99 197623 : if (pwatchers != NULL) {
100 115667 : *pwatchers = data.dptr;
101 : }
102 197623 : if (pnum_watchers != NULL) {
103 115667 : *pnum_watchers = num_watchers;
104 : }
105 197623 : if (pdata != NULL) {
106 197623 : size_t watchers_len = num_watchers * DBWRAP_WATCHER_BUF_LENGTH;
107 197623 : *pdata = (TDB_DATA) {
108 197623 : .dptr = data.dptr + watchers_len,
109 197623 : .dsize = data.dsize - watchers_len
110 : };
111 : }
112 :
113 197623 : return true;
114 : }
115 :
116 1430 : static void dbwrap_watcher_get(struct dbwrap_watcher *w,
117 : const uint8_t buf[DBWRAP_WATCHER_BUF_LENGTH])
118 : {
119 1430 : server_id_get(&w->pid, buf);
120 1430 : w->instance = BVAL(buf, SERVER_ID_BUF_LENGTH);
121 1430 : }
122 :
123 461 : static void dbwrap_watcher_put(uint8_t buf[DBWRAP_WATCHER_BUF_LENGTH],
124 : const struct dbwrap_watcher *w)
125 : {
126 461 : server_id_put(buf, w->pid);
127 461 : SBVAL(buf, SERVER_ID_BUF_LENGTH, w->instance);
128 461 : }
129 :
130 0 : static void dbwrap_watch_log_invalid_record(
131 : struct db_context *db, TDB_DATA key, TDB_DATA value)
132 : {
133 0 : DBG_ERR("Found invalid record in %s\n", dbwrap_name(db));
134 0 : dump_data(1, key.dptr, key.dsize);
135 0 : dump_data(1, value.dptr, value.dsize);
136 0 : }
137 :
138 : struct db_watched_ctx {
139 : struct db_context *backend;
140 : struct messaging_context *msg;
141 : };
142 :
143 : struct db_watched_record {
144 : struct db_record *rec;
145 : struct server_id self;
146 : struct {
147 : struct db_record *rec;
148 : TDB_DATA initial_value;
149 : bool initial_valid;
150 : } backend;
151 : bool force_fini_store;
152 : struct dbwrap_watcher added;
153 : bool removed_first;
154 : struct {
155 : /*
156 : * The is the number of watcher records
157 : * parsed from backend.initial_value
158 : */
159 : size_t count;
160 : /*
161 : * This is the pointer to
162 : * the optentially first watcher record
163 : * parsed from backend.initial_value
164 : *
165 : * The pointer actually points to memory
166 : * in backend.initial_value.
167 : *
168 : * Note it might be NULL, if count is 0.
169 : */
170 : uint8_t *first;
171 : /*
172 : * This remembers if we already
173 : * notified the watchers.
174 : *
175 : * As we only need to do that once during:
176 : * do_locked
177 : * or:
178 : * between rec = fetch_locked
179 : * and
180 : * TALLOC_FREE(rec)
181 : */
182 : bool alerted;
183 : } watchers;
184 : struct {
185 : struct dbwrap_watcher watcher;
186 : } wakeup;
187 : };
188 :
189 210990 : static struct db_watched_record *db_record_get_watched_record(struct db_record *rec)
190 : {
191 : /*
192 : * we can't use wrec = talloc_get_type_abort() here!
193 : * because wrec is likely a stack variable in
194 : * dbwrap_watched_do_locked_fn()
195 : *
196 : * In order to have a least some protection
197 : * we verify the cross reference pointers
198 : * between rec and wrec
199 : */
200 210990 : struct db_watched_record *wrec =
201 : (struct db_watched_record *)rec->private_data;
202 210990 : SMB_ASSERT(wrec->rec == rec);
203 210990 : return wrec;
204 : }
205 :
206 : static NTSTATUS dbwrap_watched_record_storev(
207 : struct db_watched_record *wrec,
208 : const TDB_DATA *dbufs, int num_dbufs, int flags);
209 : static NTSTATUS dbwrap_watched_storev(struct db_record *rec,
210 : const TDB_DATA *dbufs, int num_dbufs,
211 : int flags);
212 : static NTSTATUS dbwrap_watched_delete(struct db_record *rec);
213 : static void dbwrap_watched_trigger_wakeup(struct messaging_context *msg_ctx,
214 : struct dbwrap_watcher *watcher);
215 : static int db_watched_record_destructor(struct db_watched_record *wrec);
216 :
217 134542 : static void db_watched_record_init(struct db_context *db,
218 : struct messaging_context *msg_ctx,
219 : struct db_record *rec,
220 : struct db_watched_record *wrec,
221 : struct db_record *backend_rec,
222 : TDB_DATA backend_value)
223 : {
224 : bool ok;
225 :
226 134542 : *rec = (struct db_record) {
227 : .db = db,
228 134542 : .key = dbwrap_record_get_key(backend_rec),
229 : .storev = dbwrap_watched_storev,
230 : .delete_rec = dbwrap_watched_delete,
231 : .private_data = wrec,
232 : };
233 :
234 134542 : *wrec = (struct db_watched_record) {
235 : .rec = rec,
236 134542 : .self = messaging_server_id(msg_ctx),
237 : .backend = {
238 : .rec = backend_rec,
239 : .initial_value = backend_value,
240 : .initial_valid = true,
241 : },
242 : };
243 :
244 134542 : ok = dbwrap_watch_rec_parse(backend_value,
245 : &wrec->watchers.first,
246 : &wrec->watchers.count,
247 : &rec->value);
248 134542 : if (!ok) {
249 0 : dbwrap_watch_log_invalid_record(rec->db, rec->key, backend_value);
250 : /* wipe invalid data */
251 0 : rec->value = (TDB_DATA) { .dptr = NULL, .dsize = 0 };
252 : }
253 134542 : }
254 :
255 37772 : static struct db_record *dbwrap_watched_fetch_locked(
256 : struct db_context *db, TALLOC_CTX *mem_ctx, TDB_DATA key)
257 : {
258 37772 : struct db_watched_ctx *ctx = talloc_get_type_abort(
259 : db->private_data, struct db_watched_ctx);
260 37772 : struct db_record *rec = NULL;
261 37772 : struct db_watched_record *wrec = NULL;
262 37772 : struct db_record *backend_rec = NULL;
263 37772 : TDB_DATA backend_value = { .dptr = NULL, };
264 :
265 37772 : rec = talloc_zero(mem_ctx, struct db_record);
266 37772 : if (rec == NULL) {
267 0 : return NULL;
268 : }
269 37772 : wrec = talloc_zero(rec, struct db_watched_record);
270 37772 : if (wrec == NULL) {
271 0 : TALLOC_FREE(rec);
272 0 : return NULL;
273 : }
274 :
275 37772 : backend_rec = dbwrap_fetch_locked(ctx->backend, wrec, key);
276 37772 : if (backend_rec == NULL) {
277 0 : TALLOC_FREE(rec);
278 0 : return NULL;
279 : }
280 37772 : backend_value = dbwrap_record_get_value(backend_rec);
281 :
282 37772 : db_watched_record_init(db, ctx->msg,
283 : rec, wrec,
284 : backend_rec, backend_value);
285 37772 : rec->value_valid = true;
286 37772 : talloc_set_destructor(wrec, db_watched_record_destructor);
287 :
288 37772 : return rec;
289 : }
290 :
291 : struct db_watched_record_fini_state {
292 : struct db_watched_record *wrec;
293 : TALLOC_CTX *frame;
294 : TDB_DATA dbufs[2];
295 : int num_dbufs;
296 : bool ok;
297 : };
298 :
299 0 : static void db_watched_record_fini_fetcher(TDB_DATA key,
300 : TDB_DATA backend_value,
301 : void *private_data)
302 : {
303 0 : struct db_watched_record_fini_state *state =
304 : (struct db_watched_record_fini_state *)private_data;
305 0 : struct db_watched_record *wrec = state->wrec;
306 0 : struct db_record *rec = wrec->rec;
307 0 : TDB_DATA value = {};
308 : bool ok;
309 : size_t copy_size;
310 :
311 : /*
312 : * We're within dbwrap_parse_record()
313 : * and backend_value directly points into
314 : * the mmap'ed tdb, so we need to copy the
315 : * parts we require.
316 : */
317 :
318 0 : ok = dbwrap_watch_rec_parse(backend_value, NULL, NULL, &value);
319 0 : if (!ok) {
320 0 : struct db_context *db = dbwrap_record_get_db(rec);
321 :
322 0 : dbwrap_watch_log_invalid_record(db, key, backend_value);
323 :
324 : /* wipe invalid data */
325 0 : value = (TDB_DATA) { .dptr = NULL, .dsize = 0 };
326 : }
327 :
328 0 : copy_size = MIN(rec->value.dsize, value.dsize);
329 0 : if (copy_size != 0) {
330 : /*
331 : * First reuse the buffer we already had
332 : * as much as we can.
333 : */
334 0 : memcpy(rec->value.dptr, value.dptr, copy_size);
335 0 : state->dbufs[state->num_dbufs++] = rec->value;
336 0 : value.dsize -= copy_size;
337 0 : value.dptr += copy_size;
338 : }
339 :
340 0 : if (value.dsize != 0) {
341 0 : uint8_t *p = NULL;
342 :
343 : /*
344 : * There's still new data left
345 : * allocate it on callers stackframe
346 : */
347 0 : p = talloc_memdup(state->frame, value.dptr, value.dsize);
348 0 : if (p == NULL) {
349 0 : DBG_WARNING("failed to allocate %zu bytes\n",
350 : value.dsize);
351 0 : return;
352 : }
353 :
354 0 : state->dbufs[state->num_dbufs++] = (TDB_DATA) {
355 0 : .dptr = p, .dsize = value.dsize,
356 : };
357 : }
358 :
359 0 : state->ok = true;
360 : }
361 :
362 134542 : static void db_watched_record_fini(struct db_watched_record *wrec)
363 : {
364 134542 : struct db_watched_record_fini_state state = { .wrec = wrec, };
365 134542 : struct db_context *backend = dbwrap_record_get_db(wrec->backend.rec);
366 134542 : struct db_record *rec = wrec->rec;
367 134542 : TDB_DATA key = dbwrap_record_get_key(wrec->backend.rec);
368 : NTSTATUS status;
369 :
370 134542 : if (!wrec->force_fini_store) {
371 134027 : return;
372 : }
373 :
374 515 : if (wrec->backend.initial_valid) {
375 515 : if (rec->value.dsize != 0) {
376 510 : state.dbufs[state.num_dbufs++] = rec->value;
377 : }
378 : } else {
379 : /*
380 : * We need to fetch the current
381 : * value from the backend again,
382 : * which may need to allocate memory
383 : * on the provided stackframe.
384 : */
385 :
386 0 : state.frame = talloc_stackframe();
387 :
388 0 : status = dbwrap_parse_record(backend, key,
389 : db_watched_record_fini_fetcher, &state);
390 0 : if (!NT_STATUS_IS_OK(status)) {
391 0 : DBG_WARNING("dbwrap_parse_record failed: %s\n",
392 : nt_errstr(status));
393 0 : TALLOC_FREE(state.frame);
394 0 : return;
395 : }
396 0 : if (!state.ok) {
397 0 : TALLOC_FREE(state.frame);
398 0 : return;
399 : }
400 : }
401 :
402 : /*
403 : * We don't want to wake up others just because
404 : * we added ourself as new watcher. But if we
405 : * removed outself from the first position
406 : * we need to alert the next one.
407 : */
408 515 : if (!wrec->removed_first) {
409 461 : dbwrap_watched_watch_skip_alerting(rec);
410 : }
411 :
412 515 : status = dbwrap_watched_record_storev(wrec, state.dbufs, state.num_dbufs, 0);
413 515 : TALLOC_FREE(state.frame);
414 515 : if (!NT_STATUS_IS_OK(status)) {
415 0 : DBG_WARNING("dbwrap_watched_record_storev failed: %s\n",
416 : nt_errstr(status));
417 0 : return;
418 : }
419 :
420 515 : return;
421 : }
422 :
423 37772 : static int db_watched_record_destructor(struct db_watched_record *wrec)
424 : {
425 37772 : struct db_record *rec = wrec->rec;
426 37772 : struct db_watched_ctx *ctx = talloc_get_type_abort(
427 : rec->db->private_data, struct db_watched_ctx);
428 :
429 37772 : db_watched_record_fini(wrec);
430 37772 : TALLOC_FREE(wrec->backend.rec);
431 37772 : dbwrap_watched_trigger_wakeup(ctx->msg, &wrec->wakeup.watcher);
432 37772 : return 0;
433 : }
434 :
435 : struct dbwrap_watched_do_locked_state {
436 : struct db_context *db;
437 : struct messaging_context *msg_ctx;
438 : struct db_watched_record *wrec;
439 : struct db_record *rec;
440 : void (*fn)(struct db_record *rec,
441 : TDB_DATA value,
442 : void *private_data);
443 : void *private_data;
444 : };
445 :
446 96770 : static void dbwrap_watched_do_locked_fn(
447 : struct db_record *backend_rec,
448 : TDB_DATA backend_value,
449 : void *private_data)
450 : {
451 96770 : struct dbwrap_watched_do_locked_state *state =
452 : (struct dbwrap_watched_do_locked_state *)private_data;
453 :
454 96770 : db_watched_record_init(state->db, state->msg_ctx,
455 : state->rec, state->wrec,
456 : backend_rec, backend_value);
457 :
458 96770 : state->fn(state->rec, state->rec->value, state->private_data);
459 :
460 96770 : db_watched_record_fini(state->wrec);
461 96770 : }
462 :
463 96770 : static NTSTATUS dbwrap_watched_do_locked(struct db_context *db, TDB_DATA key,
464 : void (*fn)(struct db_record *rec,
465 : TDB_DATA value,
466 : void *private_data),
467 : void *private_data)
468 : {
469 96770 : struct db_watched_ctx *ctx = talloc_get_type_abort(
470 : db->private_data, struct db_watched_ctx);
471 : struct db_watched_record wrec;
472 : struct db_record rec;
473 183456 : struct dbwrap_watched_do_locked_state state = {
474 96770 : .db = db, .msg_ctx = ctx->msg,
475 : .rec = &rec, .wrec = &wrec,
476 : .fn = fn, .private_data = private_data,
477 : };
478 : NTSTATUS status;
479 :
480 96770 : status = dbwrap_do_locked(
481 : ctx->backend, key, dbwrap_watched_do_locked_fn, &state);
482 96770 : if (!NT_STATUS_IS_OK(status)) {
483 0 : DBG_DEBUG("dbwrap_do_locked returned %s\n", nt_errstr(status));
484 0 : return status;
485 : }
486 :
487 96770 : DBG_DEBUG("dbwrap_watched_do_locked_fn returned\n");
488 :
489 96770 : dbwrap_watched_trigger_wakeup(state.msg_ctx, &wrec.wakeup.watcher);
490 :
491 96770 : return NT_STATUS_OK;
492 : }
493 :
494 133521 : static void dbwrap_watched_record_prepare_wakeup(
495 : struct db_watched_record *wrec)
496 : {
497 : /*
498 : * Wakeup only needs to happen once (if at all)
499 : */
500 133521 : if (wrec->watchers.alerted) {
501 : /* already done */
502 69802 : return;
503 : }
504 63719 : wrec->watchers.alerted = true;
505 :
506 63719 : if (wrec->watchers.count == 0) {
507 62750 : DBG_DEBUG("No watchers\n");
508 62750 : return;
509 : }
510 :
511 1936 : while (wrec->watchers.count != 0) {
512 : struct server_id_buf tmp;
513 : bool exists;
514 :
515 969 : dbwrap_watcher_get(&wrec->wakeup.watcher, wrec->watchers.first);
516 969 : exists = serverid_exists(&wrec->wakeup.watcher.pid);
517 969 : if (!exists) {
518 0 : DBG_DEBUG("Discard non-existing waiter %s:%"PRIu64"\n",
519 : server_id_str_buf(wrec->wakeup.watcher.pid, &tmp),
520 : wrec->wakeup.watcher.instance);
521 0 : wrec->watchers.first += DBWRAP_WATCHER_BUF_LENGTH;
522 0 : wrec->watchers.count -= 1;
523 0 : continue;
524 : }
525 :
526 : /*
527 : * We will only wakeup the first waiter, via
528 : * dbwrap_watched_trigger_wakeup(), but keep
529 : * all (including the first one) in the list that
530 : * will be flushed back to the backend record
531 : * again. Waiters are removing their entries
532 : * via dbwrap_watched_watch_remove_instance()
533 : * when they no longer want to monitor the record.
534 : */
535 969 : DBG_DEBUG("Will alert first waiter %s:%"PRIu64"\n",
536 : server_id_str_buf(wrec->wakeup.watcher.pid, &tmp),
537 : wrec->wakeup.watcher.instance);
538 969 : break;
539 : }
540 : }
541 :
542 134542 : static void dbwrap_watched_trigger_wakeup(struct messaging_context *msg_ctx,
543 : struct dbwrap_watcher *watcher)
544 : {
545 : struct server_id_buf tmp;
546 : uint8_t instance_buf[8];
547 : NTSTATUS status;
548 :
549 134542 : if (watcher->instance == 0) {
550 133573 : DBG_DEBUG("No one to wakeup\n");
551 133573 : return;
552 : }
553 :
554 969 : DBG_DEBUG("Alerting %s:%"PRIu64"\n",
555 : server_id_str_buf(watcher->pid, &tmp),
556 : watcher->instance);
557 :
558 969 : SBVAL(instance_buf, 0, watcher->instance);
559 :
560 969 : status = messaging_send_buf(
561 : msg_ctx,
562 : watcher->pid,
563 : MSG_DBWRAP_MODIFIED,
564 : instance_buf,
565 : sizeof(instance_buf));
566 969 : if (!NT_STATUS_IS_OK(status)) {
567 0 : DBG_WARNING("messaging_send_buf to %s failed: %s - ignoring...\n",
568 : server_id_str_buf(watcher->pid, &tmp),
569 : nt_errstr(status));
570 : }
571 : }
572 :
573 133521 : static NTSTATUS dbwrap_watched_record_storev(
574 : struct db_watched_record *wrec,
575 : const TDB_DATA *dbufs, int num_dbufs, int flags)
576 133521 : {
577 133521 : uint8_t num_watchers_buf[4] = { 0 };
578 : uint8_t add_buf[DBWRAP_WATCHER_BUF_LENGTH];
579 : size_t num_store_watchers;
580 133521 : TDB_DATA my_dbufs[num_dbufs+3];
581 133521 : int num_my_dbufs = 0;
582 : NTSTATUS status;
583 133521 : size_t add_count = 0;
584 :
585 133521 : dbwrap_watched_record_prepare_wakeup(wrec);
586 :
587 133521 : wrec->backend.initial_valid = false;
588 133521 : wrec->force_fini_store = false;
589 :
590 133521 : if (wrec->added.pid.pid != 0) {
591 461 : dbwrap_watcher_put(add_buf, &wrec->added);
592 461 : add_count = 1;
593 : }
594 :
595 133521 : num_store_watchers = wrec->watchers.count + add_count;
596 133521 : if (num_store_watchers == 0 && num_dbufs == 0) {
597 18875 : status = dbwrap_record_delete(wrec->backend.rec);
598 18875 : return status;
599 : }
600 114646 : if (num_store_watchers >= DBWRAP_MAX_WATCHERS) {
601 0 : DBG_WARNING("Can't handle %zu watchers\n",
602 : num_store_watchers);
603 0 : return NT_STATUS_INSUFFICIENT_RESOURCES;
604 : }
605 :
606 114646 : SIVAL(num_watchers_buf, 0, num_store_watchers);
607 :
608 114646 : my_dbufs[num_my_dbufs++] = (TDB_DATA) {
609 : .dptr = num_watchers_buf, .dsize = sizeof(num_watchers_buf),
610 : };
611 114646 : if (wrec->watchers.count != 0) {
612 5808 : my_dbufs[num_my_dbufs++] = (TDB_DATA) {
613 2905 : .dptr = wrec->watchers.first, .dsize = wrec->watchers.count * DBWRAP_WATCHER_BUF_LENGTH,
614 : };
615 : }
616 114646 : if (add_count != 0) {
617 461 : my_dbufs[num_my_dbufs++] = (TDB_DATA) {
618 : .dptr = add_buf,
619 : .dsize = sizeof(add_buf),
620 : };
621 : }
622 114646 : if (num_dbufs != 0) {
623 114591 : memcpy(my_dbufs+num_my_dbufs, dbufs, num_dbufs * sizeof(*dbufs));
624 114591 : num_my_dbufs += num_dbufs;
625 : }
626 :
627 114646 : SMB_ASSERT(num_my_dbufs <= ARRAY_SIZE(my_dbufs));
628 :
629 114646 : status = dbwrap_record_storev(
630 : wrec->backend.rec, my_dbufs, num_my_dbufs, flags);
631 114646 : return status;
632 : }
633 :
634 114081 : static NTSTATUS dbwrap_watched_storev(struct db_record *rec,
635 : const TDB_DATA *dbufs, int num_dbufs,
636 : int flags)
637 : {
638 114081 : struct db_watched_record *wrec = db_record_get_watched_record(rec);
639 :
640 114081 : return dbwrap_watched_record_storev(wrec, dbufs, num_dbufs, flags);
641 : }
642 :
643 18925 : static NTSTATUS dbwrap_watched_delete(struct db_record *rec)
644 : {
645 18925 : struct db_watched_record *wrec = db_record_get_watched_record(rec);
646 :
647 : /*
648 : * dbwrap_watched_record_storev() will figure out
649 : * if the record should be deleted or if there are still
650 : * watchers to be stored.
651 : */
652 18925 : return dbwrap_watched_record_storev(wrec, NULL, 0, 0);
653 : }
654 :
655 : struct dbwrap_watched_traverse_state {
656 : int (*fn)(struct db_record *rec, void *private_data);
657 : void *private_data;
658 : };
659 :
660 144 : static int dbwrap_watched_traverse_fn(struct db_record *rec,
661 : void *private_data)
662 : {
663 144 : struct dbwrap_watched_traverse_state *state = private_data;
664 144 : struct db_record prec = *rec;
665 : bool ok;
666 :
667 144 : ok = dbwrap_watch_rec_parse(rec->value, NULL, NULL, &prec.value);
668 144 : if (!ok) {
669 0 : return 0;
670 : }
671 144 : if (prec.value.dsize == 0) {
672 0 : return 0;
673 : }
674 144 : prec.value_valid = true;
675 :
676 144 : return state->fn(&prec, state->private_data);
677 : }
678 :
679 0 : static int dbwrap_watched_traverse(struct db_context *db,
680 : int (*fn)(struct db_record *rec,
681 : void *private_data),
682 : void *private_data)
683 : {
684 0 : struct db_watched_ctx *ctx = talloc_get_type_abort(
685 : db->private_data, struct db_watched_ctx);
686 0 : struct dbwrap_watched_traverse_state state = {
687 : .fn = fn, .private_data = private_data };
688 : NTSTATUS status;
689 : int ret;
690 :
691 0 : status = dbwrap_traverse(
692 : ctx->backend, dbwrap_watched_traverse_fn, &state, &ret);
693 0 : if (!NT_STATUS_IS_OK(status)) {
694 0 : return -1;
695 : }
696 0 : return ret;
697 : }
698 :
699 128 : static int dbwrap_watched_traverse_read(struct db_context *db,
700 : int (*fn)(struct db_record *rec,
701 : void *private_data),
702 : void *private_data)
703 : {
704 128 : struct db_watched_ctx *ctx = talloc_get_type_abort(
705 : db->private_data, struct db_watched_ctx);
706 128 : struct dbwrap_watched_traverse_state state = {
707 : .fn = fn, .private_data = private_data };
708 : NTSTATUS status;
709 : int ret;
710 :
711 128 : status = dbwrap_traverse_read(
712 : ctx->backend, dbwrap_watched_traverse_fn, &state, &ret);
713 128 : if (!NT_STATUS_IS_OK(status)) {
714 0 : return -1;
715 : }
716 128 : return ret;
717 : }
718 :
719 467 : static int dbwrap_watched_get_seqnum(struct db_context *db)
720 : {
721 467 : struct db_watched_ctx *ctx = talloc_get_type_abort(
722 : db->private_data, struct db_watched_ctx);
723 467 : return dbwrap_get_seqnum(ctx->backend);
724 : }
725 :
726 0 : static int dbwrap_watched_transaction_start(struct db_context *db)
727 : {
728 0 : struct db_watched_ctx *ctx = talloc_get_type_abort(
729 : db->private_data, struct db_watched_ctx);
730 0 : return dbwrap_transaction_start(ctx->backend);
731 : }
732 :
733 0 : static int dbwrap_watched_transaction_commit(struct db_context *db)
734 : {
735 0 : struct db_watched_ctx *ctx = talloc_get_type_abort(
736 : db->private_data, struct db_watched_ctx);
737 0 : return dbwrap_transaction_commit(ctx->backend);
738 : }
739 :
740 0 : static int dbwrap_watched_transaction_cancel(struct db_context *db)
741 : {
742 0 : struct db_watched_ctx *ctx = talloc_get_type_abort(
743 : db->private_data, struct db_watched_ctx);
744 0 : return dbwrap_transaction_cancel(ctx->backend);
745 : }
746 :
747 : struct dbwrap_watched_parse_record_state {
748 : struct db_context *db;
749 : void (*parser)(TDB_DATA key, TDB_DATA data, void *private_data);
750 : void *private_data;
751 : bool ok;
752 : };
753 :
754 81812 : static void dbwrap_watched_parse_record_parser(TDB_DATA key, TDB_DATA data,
755 : void *private_data)
756 : {
757 81812 : struct dbwrap_watched_parse_record_state *state = private_data;
758 : TDB_DATA userdata;
759 :
760 81812 : state->ok = dbwrap_watch_rec_parse(data, NULL, NULL, &userdata);
761 81812 : if (!state->ok) {
762 0 : dbwrap_watch_log_invalid_record(state->db, key, data);
763 0 : return;
764 : }
765 :
766 81812 : state->parser(key, userdata, state->private_data);
767 : }
768 :
769 82749 : static NTSTATUS dbwrap_watched_parse_record(
770 : struct db_context *db, TDB_DATA key,
771 : void (*parser)(TDB_DATA key, TDB_DATA data, void *private_data),
772 : void *private_data)
773 : {
774 82749 : struct db_watched_ctx *ctx = talloc_get_type_abort(
775 : db->private_data, struct db_watched_ctx);
776 82749 : struct dbwrap_watched_parse_record_state state = {
777 : .db = db,
778 : .parser = parser,
779 : .private_data = private_data,
780 : };
781 : NTSTATUS status;
782 :
783 82749 : status = dbwrap_parse_record(
784 : ctx->backend, key, dbwrap_watched_parse_record_parser, &state);
785 82749 : if (!NT_STATUS_IS_OK(status)) {
786 937 : return status;
787 : }
788 81812 : if (!state.ok) {
789 0 : return NT_STATUS_NOT_FOUND;
790 : }
791 81812 : return NT_STATUS_OK;
792 : }
793 :
794 : static void dbwrap_watched_parse_record_done(struct tevent_req *subreq);
795 :
796 0 : static struct tevent_req *dbwrap_watched_parse_record_send(
797 : TALLOC_CTX *mem_ctx,
798 : struct tevent_context *ev,
799 : struct db_context *db,
800 : TDB_DATA key,
801 : void (*parser)(TDB_DATA key, TDB_DATA data, void *private_data),
802 : void *private_data,
803 : enum dbwrap_req_state *req_state)
804 : {
805 0 : struct db_watched_ctx *ctx = talloc_get_type_abort(
806 : db->private_data, struct db_watched_ctx);
807 0 : struct tevent_req *req = NULL;
808 0 : struct tevent_req *subreq = NULL;
809 0 : struct dbwrap_watched_parse_record_state *state = NULL;
810 :
811 0 : req = tevent_req_create(mem_ctx, &state,
812 : struct dbwrap_watched_parse_record_state);
813 0 : if (req == NULL) {
814 0 : *req_state = DBWRAP_REQ_ERROR;
815 0 : return NULL;
816 : }
817 :
818 0 : *state = (struct dbwrap_watched_parse_record_state) {
819 : .parser = parser,
820 : .private_data = private_data,
821 : .ok = true,
822 : };
823 :
824 0 : subreq = dbwrap_parse_record_send(state,
825 : ev,
826 : ctx->backend,
827 : key,
828 : dbwrap_watched_parse_record_parser,
829 : state,
830 : req_state);
831 0 : if (tevent_req_nomem(subreq, req)) {
832 0 : *req_state = DBWRAP_REQ_ERROR;
833 0 : return tevent_req_post(req, ev);
834 : }
835 :
836 0 : tevent_req_set_callback(subreq, dbwrap_watched_parse_record_done, req);
837 0 : return req;
838 : }
839 :
840 0 : static void dbwrap_watched_parse_record_done(struct tevent_req *subreq)
841 : {
842 0 : struct tevent_req *req = tevent_req_callback_data(
843 : subreq, struct tevent_req);
844 0 : struct dbwrap_watched_parse_record_state *state = tevent_req_data(
845 : req, struct dbwrap_watched_parse_record_state);
846 : NTSTATUS status;
847 :
848 0 : status = dbwrap_parse_record_recv(subreq);
849 0 : TALLOC_FREE(subreq);
850 0 : if (tevent_req_nterror(req, status)) {
851 0 : return;
852 : }
853 :
854 0 : if (!state->ok) {
855 0 : tevent_req_nterror(req, NT_STATUS_NOT_FOUND);
856 0 : return;
857 : }
858 :
859 0 : tevent_req_done(req);
860 0 : return;
861 : }
862 :
863 0 : static NTSTATUS dbwrap_watched_parse_record_recv(struct tevent_req *req)
864 : {
865 : NTSTATUS status;
866 :
867 0 : if (tevent_req_is_nterror(req, &status)) {
868 0 : tevent_req_received(req);
869 0 : return status;
870 : }
871 :
872 0 : tevent_req_received(req);
873 0 : return NT_STATUS_OK;
874 : }
875 :
876 0 : static int dbwrap_watched_exists(struct db_context *db, TDB_DATA key)
877 : {
878 0 : struct db_watched_ctx *ctx = talloc_get_type_abort(
879 : db->private_data, struct db_watched_ctx);
880 :
881 0 : return dbwrap_exists(ctx->backend, key);
882 : }
883 :
884 0 : static size_t dbwrap_watched_id(struct db_context *db, uint8_t *id,
885 : size_t idlen)
886 : {
887 0 : struct db_watched_ctx *ctx = talloc_get_type_abort(
888 : db->private_data, struct db_watched_ctx);
889 :
890 0 : return dbwrap_db_id(ctx->backend, id, idlen);
891 : }
892 :
893 226 : struct db_context *db_open_watched(TALLOC_CTX *mem_ctx,
894 : struct db_context **backend,
895 : struct messaging_context *msg)
896 : {
897 : struct db_context *db;
898 : struct db_watched_ctx *ctx;
899 :
900 226 : db = talloc_zero(mem_ctx, struct db_context);
901 226 : if (db == NULL) {
902 0 : return NULL;
903 : }
904 226 : ctx = talloc_zero(db, struct db_watched_ctx);
905 226 : if (ctx == NULL) {
906 0 : TALLOC_FREE(db);
907 0 : return NULL;
908 : }
909 226 : db->private_data = ctx;
910 :
911 226 : ctx->msg = msg;
912 :
913 226 : ctx->backend = talloc_move(ctx, backend);
914 226 : db->lock_order = ctx->backend->lock_order;
915 226 : ctx->backend->lock_order = DBWRAP_LOCK_ORDER_NONE;
916 :
917 226 : db->fetch_locked = dbwrap_watched_fetch_locked;
918 226 : db->do_locked = dbwrap_watched_do_locked;
919 226 : db->traverse = dbwrap_watched_traverse;
920 226 : db->traverse_read = dbwrap_watched_traverse_read;
921 226 : db->get_seqnum = dbwrap_watched_get_seqnum;
922 226 : db->transaction_start = dbwrap_watched_transaction_start;
923 226 : db->transaction_commit = dbwrap_watched_transaction_commit;
924 226 : db->transaction_cancel = dbwrap_watched_transaction_cancel;
925 226 : db->parse_record = dbwrap_watched_parse_record;
926 226 : db->parse_record_send = dbwrap_watched_parse_record_send;
927 226 : db->parse_record_recv = dbwrap_watched_parse_record_recv;
928 226 : db->exists = dbwrap_watched_exists;
929 226 : db->id = dbwrap_watched_id;
930 226 : db->name = dbwrap_name(ctx->backend);
931 :
932 226 : return db;
933 : }
934 :
935 461 : uint64_t dbwrap_watched_watch_add_instance(struct db_record *rec)
936 : {
937 461 : struct db_watched_record *wrec = db_record_get_watched_record(rec);
938 : static uint64_t global_instance = 1;
939 :
940 461 : SMB_ASSERT(wrec->added.instance == 0);
941 :
942 461 : wrec->added = (struct dbwrap_watcher) {
943 461 : .pid = wrec->self,
944 461 : .instance = global_instance++,
945 : };
946 :
947 461 : wrec->force_fini_store = true;
948 :
949 461 : return wrec->added.instance;
950 : }
951 :
952 5957 : void dbwrap_watched_watch_remove_instance(struct db_record *rec, uint64_t instance)
953 : {
954 5957 : struct db_watched_record *wrec = db_record_get_watched_record(rec);
955 5957 : struct dbwrap_watcher clear_watcher = {
956 : .pid = wrec->self,
957 : .instance = instance,
958 : };
959 : size_t i;
960 : struct server_id_buf buf;
961 :
962 5957 : if (instance == 0) {
963 5496 : return;
964 : }
965 :
966 461 : if (wrec->added.instance == instance) {
967 0 : SMB_ASSERT(server_id_equal(&wrec->added.pid, &wrec->self));
968 0 : DBG_DEBUG("Watcher %s:%"PRIu64" reverted from adding\n",
969 : server_id_str_buf(clear_watcher.pid, &buf),
970 : clear_watcher.instance);
971 0 : ZERO_STRUCT(wrec->added);
972 : }
973 :
974 895 : for (i=0; i < wrec->watchers.count; i++) {
975 : struct dbwrap_watcher watcher;
976 461 : size_t off = i*DBWRAP_WATCHER_BUF_LENGTH;
977 : size_t next_off;
978 : size_t full_len;
979 : size_t move_len;
980 :
981 461 : dbwrap_watcher_get(&watcher, wrec->watchers.first + off);
982 :
983 461 : if (clear_watcher.instance != watcher.instance) {
984 0 : continue;
985 : }
986 461 : if (!server_id_equal(&clear_watcher.pid, &watcher.pid)) {
987 0 : continue;
988 : }
989 :
990 461 : wrec->force_fini_store = true;
991 :
992 461 : if (i == 0) {
993 461 : DBG_DEBUG("Watcher %s:%"PRIu64" removed from first position of %zu\n",
994 : server_id_str_buf(clear_watcher.pid, &buf),
995 : clear_watcher.instance,
996 : wrec->watchers.count);
997 461 : wrec->watchers.first += DBWRAP_WATCHER_BUF_LENGTH;
998 461 : wrec->watchers.count -= 1;
999 461 : wrec->removed_first = true;
1000 895 : return;
1001 : }
1002 0 : if (i == (wrec->watchers.count-1)) {
1003 0 : DBG_DEBUG("Watcher %s:%"PRIu64" removed from last position of %zu\n",
1004 : server_id_str_buf(clear_watcher.pid, &buf),
1005 : clear_watcher.instance,
1006 : wrec->watchers.count);
1007 0 : wrec->watchers.count -= 1;
1008 0 : return;
1009 : }
1010 :
1011 0 : DBG_DEBUG("Watcher %s:%"PRIu64" cleared at position %zu from %zu\n",
1012 : server_id_str_buf(clear_watcher.pid, &buf),
1013 : clear_watcher.instance, i+1,
1014 : wrec->watchers.count);
1015 :
1016 0 : next_off = off + DBWRAP_WATCHER_BUF_LENGTH;
1017 0 : full_len = wrec->watchers.count * DBWRAP_WATCHER_BUF_LENGTH;
1018 0 : move_len = full_len - next_off;
1019 0 : memmove(wrec->watchers.first + off,
1020 0 : wrec->watchers.first + next_off,
1021 : move_len);
1022 0 : wrec->watchers.count -= 1;
1023 0 : return;
1024 : }
1025 :
1026 0 : DBG_DEBUG("Watcher %s:%"PRIu64" not found in %zu watchers\n",
1027 : server_id_str_buf(clear_watcher.pid, &buf),
1028 : clear_watcher.instance,
1029 : wrec->watchers.count);
1030 0 : return;
1031 : }
1032 :
1033 71048 : void dbwrap_watched_watch_skip_alerting(struct db_record *rec)
1034 : {
1035 71048 : struct db_watched_record *wrec = db_record_get_watched_record(rec);
1036 :
1037 71048 : wrec->watchers.alerted = true;
1038 71048 : }
1039 :
1040 : struct dbwrap_watched_watch_state {
1041 : struct db_context *db;
1042 : TDB_DATA key;
1043 : struct dbwrap_watcher watcher;
1044 : struct server_id blocker;
1045 : bool blockerdead;
1046 : };
1047 :
1048 : static bool dbwrap_watched_msg_filter(struct messaging_rec *rec,
1049 : void *private_data);
1050 : static void dbwrap_watched_watch_done(struct tevent_req *subreq);
1051 : static void dbwrap_watched_watch_blocker_died(struct tevent_req *subreq);
1052 : static int dbwrap_watched_watch_state_destructor(
1053 : struct dbwrap_watched_watch_state *state);
1054 :
1055 518 : struct tevent_req *dbwrap_watched_watch_send(TALLOC_CTX *mem_ctx,
1056 : struct tevent_context *ev,
1057 : struct db_record *rec,
1058 : uint64_t resumed_instance,
1059 : struct server_id blocker)
1060 : {
1061 518 : struct db_context *db = dbwrap_record_get_db(rec);
1062 518 : struct db_watched_ctx *ctx = talloc_get_type_abort(
1063 : db->private_data, struct db_watched_ctx);
1064 518 : struct db_watched_record *wrec = db_record_get_watched_record(rec);
1065 : struct tevent_req *req, *subreq;
1066 : struct dbwrap_watched_watch_state *state;
1067 : uint64_t instance;
1068 :
1069 518 : req = tevent_req_create(mem_ctx, &state,
1070 : struct dbwrap_watched_watch_state);
1071 518 : if (req == NULL) {
1072 0 : return NULL;
1073 : }
1074 518 : state->db = db;
1075 518 : state->blocker = blocker;
1076 :
1077 518 : if (ctx->msg == NULL) {
1078 0 : tevent_req_nterror(req, NT_STATUS_NOT_SUPPORTED);
1079 0 : return tevent_req_post(req, ev);
1080 : }
1081 :
1082 518 : if (resumed_instance == 0 && wrec->added.instance == 0) {
1083 : /*
1084 : * Adding a new instance
1085 : */
1086 0 : instance = dbwrap_watched_watch_add_instance(rec);
1087 518 : } else if (resumed_instance != 0 && wrec->added.instance == 0) {
1088 : /*
1089 : * Resuming an existing instance that was
1090 : * already present before do_locked started
1091 : */
1092 57 : instance = resumed_instance;
1093 461 : } else if (resumed_instance == wrec->added.instance) {
1094 : /*
1095 : * The caller used dbwrap_watched_watch_add_instance()
1096 : * already during this do_locked() invocation.
1097 : */
1098 461 : instance = resumed_instance;
1099 : } else {
1100 0 : tevent_req_nterror(req, NT_STATUS_REQUEST_NOT_ACCEPTED);
1101 0 : return tevent_req_post(req, ev);
1102 : }
1103 :
1104 1036 : state->watcher = (struct dbwrap_watcher) {
1105 518 : .pid = messaging_server_id(ctx->msg),
1106 : .instance = instance,
1107 : };
1108 :
1109 518 : state->key = tdb_data_talloc_copy(state, rec->key);
1110 518 : if (tevent_req_nomem(state->key.dptr, req)) {
1111 0 : return tevent_req_post(req, ev);
1112 : }
1113 :
1114 518 : subreq = messaging_filtered_read_send(
1115 : state, ev, ctx->msg, dbwrap_watched_msg_filter, state);
1116 518 : if (tevent_req_nomem(subreq, req)) {
1117 0 : return tevent_req_post(req, ev);
1118 : }
1119 518 : tevent_req_set_callback(subreq, dbwrap_watched_watch_done, req);
1120 :
1121 518 : talloc_set_destructor(state, dbwrap_watched_watch_state_destructor);
1122 :
1123 518 : if (blocker.pid != 0) {
1124 514 : subreq = server_id_watch_send(state, ev, blocker);
1125 514 : if (tevent_req_nomem(subreq, req)) {
1126 0 : return tevent_req_post(req, ev);
1127 : }
1128 514 : tevent_req_set_callback(
1129 : subreq, dbwrap_watched_watch_blocker_died, req);
1130 : }
1131 :
1132 518 : return req;
1133 : }
1134 :
1135 0 : static void dbwrap_watched_watch_blocker_died(struct tevent_req *subreq)
1136 : {
1137 0 : struct tevent_req *req = tevent_req_callback_data(
1138 : subreq, struct tevent_req);
1139 0 : struct dbwrap_watched_watch_state *state = tevent_req_data(
1140 : req, struct dbwrap_watched_watch_state);
1141 : int ret;
1142 :
1143 0 : ret = server_id_watch_recv(subreq, NULL);
1144 0 : TALLOC_FREE(subreq);
1145 0 : if (ret != 0) {
1146 0 : tevent_req_nterror(req, map_nt_error_from_unix(ret));
1147 0 : return;
1148 : }
1149 0 : state->blockerdead = true;
1150 0 : tevent_req_done(req);
1151 : }
1152 :
1153 50 : static void dbwrap_watched_watch_state_destructor_fn(
1154 : struct db_record *rec,
1155 : TDB_DATA value,
1156 : void *private_data)
1157 : {
1158 50 : struct dbwrap_watched_watch_state *state = talloc_get_type_abort(
1159 : private_data, struct dbwrap_watched_watch_state);
1160 :
1161 : /*
1162 : * Here we just remove ourself from the in memory
1163 : * watchers array and let db_watched_record_fini()
1164 : * call dbwrap_watched_record_storev() to do the magic
1165 : * of writing back the modified in memory copy.
1166 : */
1167 50 : dbwrap_watched_watch_remove_instance(rec, state->watcher.instance);
1168 50 : return;
1169 : }
1170 :
1171 50 : static int dbwrap_watched_watch_state_destructor(
1172 : struct dbwrap_watched_watch_state *state)
1173 : {
1174 : NTSTATUS status;
1175 :
1176 50 : status = dbwrap_do_locked(
1177 : state->db,
1178 : state->key,
1179 : dbwrap_watched_watch_state_destructor_fn,
1180 : state);
1181 50 : if (!NT_STATUS_IS_OK(status)) {
1182 0 : DBG_WARNING("dbwrap_do_locked failed: %s\n",
1183 : nt_errstr(status));
1184 : }
1185 50 : return 0;
1186 : }
1187 :
1188 468 : static bool dbwrap_watched_msg_filter(struct messaging_rec *rec,
1189 : void *private_data)
1190 : {
1191 468 : struct dbwrap_watched_watch_state *state = talloc_get_type_abort(
1192 : private_data, struct dbwrap_watched_watch_state);
1193 : uint64_t instance;
1194 :
1195 468 : if (rec->msg_type != MSG_DBWRAP_MODIFIED) {
1196 0 : return false;
1197 : }
1198 468 : if (rec->num_fds != 0) {
1199 0 : return false;
1200 : }
1201 :
1202 468 : if (rec->buf.length != sizeof(instance)) {
1203 0 : DBG_DEBUG("Got size %zu, expected %zu\n",
1204 : rec->buf.length,
1205 : sizeof(instance));
1206 0 : return false;
1207 : }
1208 :
1209 468 : instance = BVAL(rec->buf.data, 0);
1210 :
1211 468 : if (instance != state->watcher.instance) {
1212 0 : DBG_DEBUG("Got instance %"PRIu64", expected %"PRIu64"\n",
1213 : instance,
1214 : state->watcher.instance);
1215 0 : return false;
1216 : }
1217 :
1218 468 : return true;
1219 : }
1220 :
1221 468 : static void dbwrap_watched_watch_done(struct tevent_req *subreq)
1222 : {
1223 468 : struct tevent_req *req = tevent_req_callback_data(
1224 : subreq, struct tevent_req);
1225 468 : struct dbwrap_watched_watch_state *state = tevent_req_data(
1226 : req, struct dbwrap_watched_watch_state);
1227 : struct messaging_rec *rec;
1228 : int ret;
1229 :
1230 468 : ret = messaging_filtered_read_recv(subreq, state, &rec);
1231 468 : TALLOC_FREE(subreq);
1232 468 : if (ret != 0) {
1233 0 : tevent_req_nterror(req, map_nt_error_from_unix(ret));
1234 0 : return;
1235 : }
1236 468 : tevent_req_done(req);
1237 : }
1238 :
1239 468 : NTSTATUS dbwrap_watched_watch_recv(struct tevent_req *req,
1240 : uint64_t *pkeep_instance,
1241 : bool *blockerdead,
1242 : struct server_id *blocker)
1243 : {
1244 468 : struct dbwrap_watched_watch_state *state = tevent_req_data(
1245 : req, struct dbwrap_watched_watch_state);
1246 : NTSTATUS status;
1247 :
1248 468 : if (tevent_req_is_nterror(req, &status)) {
1249 0 : tevent_req_received(req);
1250 0 : return status;
1251 : }
1252 468 : if (pkeep_instance != NULL) {
1253 468 : *pkeep_instance = state->watcher.instance;
1254 : /*
1255 : * No need to remove ourselves anymore,
1256 : * the caller will take care of removing itself.
1257 : */
1258 468 : talloc_set_destructor(state, NULL);
1259 : }
1260 468 : if (blockerdead != NULL) {
1261 464 : *blockerdead = state->blockerdead;
1262 : }
1263 468 : if (blocker != NULL) {
1264 464 : *blocker = state->blocker;
1265 : }
1266 468 : tevent_req_received(req);
1267 468 : return NT_STATUS_OK;
1268 : }
1269 :
|