mirror of
https://github.com/samba-team/samba.git
synced 2025-01-27 14:04:05 +03:00
4652a8b911
Instead of using messaging to send individual cleanup events, it works this way: o parent smbd stores cleanup events (ie exitted children) in smbd_cleanup.tdb o it sends cleanupd an empty MSG_SMB_NOTIFY_CLEANUP message o cleanupd does a traverse on the smbd_cleanupd.tdb and collects all childs in a list o after the traverse cleanupd walks the list and does the real work It would have been possible to optimize for the common case by passing info about exitted childs with the message (as was done before this patch), adding a new message type for triggering a db traverse that would be used when cleanupd had to be restarted and cleanup events may have been accumulated in cleanup.tdb. But this could be subject to subtle race conditions and could loose events if cleanupd dies randomly. Bug: https://bugzilla.samba.org/show_bug.cgi?id=12022 Signed-off-by: Ralph Boehme <slow@samba.org> Reviewed-by: Volker Lendecke <vl@samba.org>
221 lines
5.8 KiB
C
221 lines
5.8 KiB
C
/*
|
|
* Unix SMB/CIFS implementation.
|
|
*
|
|
* Copyright (C) Volker Lendecke 2015
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "includes.h"
|
|
#include "smbd_cleanupd.h"
|
|
#include "lib/util_procid.h"
|
|
#include "lib/util/tevent_ntstatus.h"
|
|
#include "lib/util/debug.h"
|
|
#include "smbprofile.h"
|
|
#include "serverid.h"
|
|
#include "locking/proto.h"
|
|
#include "cleanupdb.h"
|
|
|
|
struct smbd_cleanupd_state {
|
|
pid_t parent_pid;
|
|
};
|
|
|
|
static void smbd_cleanupd_shutdown(struct messaging_context *msg,
|
|
void *private_data, uint32_t msg_type,
|
|
struct server_id server_id,
|
|
DATA_BLOB *data);
|
|
static void smbd_cleanupd_process_exited(struct messaging_context *msg,
|
|
void *private_data, uint32_t msg_type,
|
|
struct server_id server_id,
|
|
DATA_BLOB *data);
|
|
static void smbd_cleanupd_unlock(struct messaging_context *msg,
|
|
void *private_data, uint32_t msg_type,
|
|
struct server_id server_id,
|
|
DATA_BLOB *data);
|
|
|
|
struct tevent_req *smbd_cleanupd_send(TALLOC_CTX *mem_ctx,
|
|
struct tevent_context *ev,
|
|
struct messaging_context *msg,
|
|
pid_t parent_pid)
|
|
{
|
|
struct tevent_req *req;
|
|
struct smbd_cleanupd_state *state;
|
|
NTSTATUS status;
|
|
|
|
req = tevent_req_create(mem_ctx, &state, struct smbd_cleanupd_state);
|
|
if (req == NULL) {
|
|
return NULL;
|
|
}
|
|
state->parent_pid = parent_pid;
|
|
|
|
status = messaging_register(msg, req, MSG_SHUTDOWN,
|
|
smbd_cleanupd_shutdown);
|
|
if (tevent_req_nterror(req, status)) {
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
|
|
status = messaging_register(msg, req, MSG_SMB_NOTIFY_CLEANUP,
|
|
smbd_cleanupd_process_exited);
|
|
if (tevent_req_nterror(req, status)) {
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
|
|
status = messaging_register(msg, NULL, MSG_SMB_UNLOCK,
|
|
smbd_cleanupd_unlock);
|
|
if (tevent_req_nterror(req, status)) {
|
|
return tevent_req_post(req, ev);
|
|
}
|
|
|
|
return req;
|
|
}
|
|
|
|
static void smbd_cleanupd_shutdown(struct messaging_context *msg,
|
|
void *private_data, uint32_t msg_type,
|
|
struct server_id server_id,
|
|
DATA_BLOB *data)
|
|
{
|
|
struct tevent_req *req = talloc_get_type_abort(
|
|
private_data, struct tevent_req);
|
|
tevent_req_done(req);
|
|
}
|
|
|
|
static void smbd_cleanupd_unlock(struct messaging_context *msg,
|
|
void *private_data, uint32_t msg_type,
|
|
struct server_id server_id,
|
|
DATA_BLOB *data)
|
|
{
|
|
DBG_WARNING("Cleaning up brl and lock database after unclean "
|
|
"shutdown\n");
|
|
|
|
message_send_all(msg, MSG_SMB_UNLOCK, NULL, 0, NULL);
|
|
|
|
brl_revalidate(msg, private_data, msg_type, server_id, data);
|
|
}
|
|
|
|
struct cleanup_child {
|
|
struct cleanup_child *prev, *next;
|
|
pid_t pid;
|
|
bool unclean;
|
|
};
|
|
|
|
struct cleanupdb_traverse_state {
|
|
TALLOC_CTX *mem_ctx;
|
|
bool ok;
|
|
struct cleanup_child *childs;
|
|
};
|
|
|
|
static int cleanupdb_traverse_fn(const pid_t pid,
|
|
const bool unclean,
|
|
void *private_data)
|
|
{
|
|
struct cleanupdb_traverse_state *cleanup_state =
|
|
(struct cleanupdb_traverse_state *)private_data;
|
|
struct cleanup_child *child = NULL;
|
|
|
|
child = talloc_zero(cleanup_state->mem_ctx, struct cleanup_child);
|
|
if (child == NULL) {
|
|
DBG_ERR("talloc_zero failed\n");
|
|
return -1;
|
|
}
|
|
|
|
child->pid = pid;
|
|
child->unclean = unclean;
|
|
DLIST_ADD(cleanup_state->childs, child);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void smbd_cleanupd_process_exited(struct messaging_context *msg,
|
|
void *private_data, uint32_t msg_type,
|
|
struct server_id server_id,
|
|
DATA_BLOB *data)
|
|
{
|
|
struct tevent_req *req = talloc_get_type_abort(
|
|
private_data, struct tevent_req);
|
|
struct smbd_cleanupd_state *state = tevent_req_data(
|
|
req, struct smbd_cleanupd_state);
|
|
int ret;
|
|
struct cleanupdb_traverse_state cleanup_state;
|
|
TALLOC_CTX *frame = talloc_stackframe();
|
|
struct cleanup_child *child = NULL;
|
|
|
|
cleanup_state = (struct cleanupdb_traverse_state) {
|
|
.mem_ctx = frame
|
|
};
|
|
|
|
/*
|
|
* This merely collect childs in a list, whatever we're
|
|
* supposed to cleanup for every child, it has to take place
|
|
* *after* the db traverse in a list loop. This is to minimize
|
|
* locking interaction between the traverse and writers (ie
|
|
* the parent smbd).
|
|
*/
|
|
ret = cleanupdb_traverse_read(cleanupdb_traverse_fn, &cleanup_state);
|
|
if (ret < 0) {
|
|
DBG_ERR("cleanupdb_traverse_read failed\n");
|
|
TALLOC_FREE(frame);
|
|
return;
|
|
}
|
|
|
|
if (ret == 0) {
|
|
DBG_ERR("got 0 cleanup events, expected at least 1\n");
|
|
TALLOC_FREE(frame);
|
|
return;
|
|
}
|
|
|
|
for (child = cleanup_state.childs;
|
|
child != NULL;
|
|
child = child->next)
|
|
{
|
|
struct server_id child_id;
|
|
bool ok;
|
|
|
|
ok = cleanupdb_delete_child(child->pid);
|
|
if (!ok) {
|
|
DBG_ERR("failed to delete pid %d\n", (int)child->pid);
|
|
}
|
|
|
|
/*
|
|
* Get child_id before messaging_cleanup which wipes
|
|
* the unique_id. Not that it really matters here for
|
|
* functionality (the child should have properly
|
|
* cleaned up :-)) though, but it looks nicer.
|
|
*/
|
|
child_id = pid_to_procid(child->pid);
|
|
|
|
smbprofile_cleanup(child->pid, state->parent_pid);
|
|
|
|
ret = messaging_cleanup(msg, child->pid);
|
|
|
|
if ((ret != 0) && (ret != ENOENT)) {
|
|
DBG_DEBUG("messaging_cleanup returned %s\n",
|
|
strerror(ret));
|
|
}
|
|
|
|
if (!serverid_deregister(child_id)) {
|
|
DBG_ERR("Could not remove pid %d from serverid.tdb\n",
|
|
(int)child->pid);
|
|
}
|
|
|
|
DBG_DEBUG("cleaned up pid %d\n", (int)child->pid);
|
|
}
|
|
|
|
TALLOC_FREE(frame);
|
|
}
|
|
|
|
NTSTATUS smbd_cleanupd_recv(struct tevent_req *req)
|
|
{
|
|
return tevent_req_simple_recv_ntstatus(req);
|
|
}
|