#include "dbwrap_open.h"
#include "lib/util/util_tdb.h"
#include "lib/util/tevent_ntstatus.h"
+#include "server_id_watch.h"
static struct db_context *dbwrap_record_watchers_db(void)
{
struct tevent_req *req;
struct messaging_context *msg;
TDB_DATA w_key;
+ bool blockerdead;
+ struct server_id blocker;
};
static bool dbwrap_record_watch_filter(struct messaging_rec *rec,
void *private_data);
static void dbwrap_record_watch_done(struct tevent_req *subreq);
+static void dbwrap_record_watch_blocker_died(struct tevent_req *subreq);
static int dbwrap_record_watch_state_destructor(
struct dbwrap_record_watch_state *state);
struct tevent_req *dbwrap_record_watch_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct db_record *rec,
- struct messaging_context *msg)
+ struct messaging_context *msg,
+ struct server_id blocker)
{
struct tevent_req *req, *subreq;
struct dbwrap_record_watch_state *state;
state->ev = ev;
state->req = req;
state->msg = msg;
+ state->blocker = blocker;
watchers_db = dbwrap_record_watchers_db();
if (watchers_db == NULL) {
}
tevent_req_set_callback(subreq, dbwrap_record_watch_done, req);
+ if (blocker.pid != 0) {
+ subreq = server_id_watch_send(state, ev, msg, blocker);
+ if (tevent_req_nomem(subreq, req)) {
+ return tevent_req_post(req, ev);
+ }
+ tevent_req_set_callback(
+ subreq, dbwrap_record_watch_blocker_died, req);
+ }
+
status = dbwrap_record_add_watcher(
state->w_key, messaging_server_id(state->msg));
if (tevent_req_nterror(req, status)) {
tevent_req_done(req);
}
+static void dbwrap_record_watch_blocker_died(struct tevent_req *subreq)
+{
+ struct tevent_req *req = tevent_req_callback_data(
+ subreq, struct tevent_req);
+ struct dbwrap_record_watch_state *state = tevent_req_data(
+ req, struct dbwrap_record_watch_state);
+ int ret;
+
+ ret = server_id_watch_recv(subreq, NULL);
+ TALLOC_FREE(subreq);
+ if (ret != 0) {
+ tevent_req_nterror(req, map_nt_error_from_unix(ret));
+ return;
+ }
+ state->blockerdead = true;
+ tevent_req_done(req);
+}
+
NTSTATUS dbwrap_record_watch_recv(struct tevent_req *req,
TALLOC_CTX *mem_ctx,
- struct db_record **prec)
+ struct db_record **prec,
+ bool *blockerdead,
+ struct server_id *blocker)
{
struct dbwrap_record_watch_state *state = tevent_req_data(
req, struct dbwrap_record_watch_state);
if (tevent_req_is_nterror(req, &status)) {
return status;
}
+ if (blockerdead != NULL) {
+ *blockerdead = state->blockerdead;
+ }
+ if (blocker != NULL) {
+ *blocker = state->blocker;
+ }
if (prec == NULL) {
return NT_STATUS_OK;
}
struct tevent_req *dbwrap_record_watch_send(TALLOC_CTX *mem_ctx,
struct tevent_context *ev,
struct db_record *rec,
- struct messaging_context *msg);
+ struct messaging_context *msg,
+ struct server_id blocker);
NTSTATUS dbwrap_record_watch_recv(struct tevent_req *req,
TALLOC_CTX *mem_ctx,
- struct db_record **prec);
+ struct db_record **prec,
+ bool *blockerdead,
+ struct server_id *blocker);
void dbwrap_watchers_traverse_read(
int (*fn)(const uint8_t *db_id, size_t db_id_len, const TDB_DATA key,
return tevent_req_post(req, ev);
}
subreq = dbwrap_record_watch_send(state, state->ev, rec,
- state->ctx->msg);
+ state->ctx->msg,
+ (struct server_id){0});
TALLOC_FREE(rec);
if (tevent_req_nomem(subreq, req)) {
return tevent_req_post(req, ev);
struct db_record *rec;
NTSTATUS status;
- status = dbwrap_record_watch_recv(subreq, talloc_tos(), &rec);
+ status = dbwrap_record_watch_recv(subreq, talloc_tos(), &rec, NULL,
+ NULL);
TALLOC_FREE(subreq);
if (NT_STATUS_EQUAL(status, NT_STATUS_IO_TIMEOUT)) {
return;
}
subreq = dbwrap_record_watch_send(state, state->ev, rec,
- state->ctx->msg);
+ state->ctx->msg,
+ (struct server_id){0});
TALLOC_FREE(rec);
if (tevent_req_nomem(subreq, req)) {
return;
watch_req = dbwrap_record_watch_send(
watch_state, req->sconn->ev_ctx, lck->data->record,
- req->sconn->msg_ctx);
+ req->sconn->msg_ctx, (struct server_id){0});
if (watch_req == NULL) {
exit_server("Could not watch share mode record");
}
NTSTATUS status;
bool ret;
- status = dbwrap_record_watch_recv(req, talloc_tos(), NULL);
+ status = dbwrap_record_watch_recv(req, talloc_tos(), NULL, NULL,
+ NULL);
TALLOC_FREE(req);
if (!NT_STATUS_IS_OK(status)) {
DEBUG(5, ("dbwrap_record_watch_recv returned %s\n",
rename_state,
ev,
lck->data->record,
- fsp->conn->sconn->msg_ctx);
+ fsp->conn->sconn->msg_ctx,
+ (struct server_id){0});
if (subreq == NULL) {
exit_server("Could not watch share mode record for rename\n");
int ret_size = 0;
bool ok;
- status = dbwrap_record_watch_recv(subreq, state->req, NULL);
+ status = dbwrap_record_watch_recv(subreq, state->req, NULL, NULL,
+ NULL);
TALLOC_FREE(subreq);
if (!NT_STATUS_IS_OK(status)) {
DEBUG(5, ("dbwrap_record_watch_recv returned %s\n",
}
subreq = dbwrap_record_watch_send(state, state->ev,
- state->db_rec, conn->msg_ctx);
+ state->db_rec, conn->msg_ctx,
+ (struct server_id){0});
if (tevent_req_nomem(subreq, req)) {
TALLOC_FREE(state->db_rec);
return;
struct smb2srv_session_close_previous_state);
NTSTATUS status;
- status = dbwrap_record_watch_recv(subreq, state, &state->db_rec);
+ status = dbwrap_record_watch_recv(subreq, state, &state->db_rec, NULL,
+ NULL);
TALLOC_FREE(subreq);
if (tevent_req_nterror(req, status)) {
return;
fprintf(stderr, "dbwrap_fetch_locked failed\n");
goto fail;
}
- req = dbwrap_record_watch_send(talloc_tos(), ev, rec, msg);
+ req = dbwrap_record_watch_send(talloc_tos(), ev, rec, msg,
+ (struct server_id){0});
if (req == NULL) {
fprintf(stderr, "dbwrap_record_watch_send failed\n");
goto fail;
goto fail;
}
- status = dbwrap_record_watch_recv(req, talloc_tos(), &rec);
+ status = dbwrap_record_watch_recv(req, talloc_tos(), &rec, NULL,
+ NULL);
if (!NT_STATUS_IS_OK(status)) {
fprintf(stderr, "dbwrap_record_watch_recv failed: %s\n",
nt_errstr(status));