1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-11 05:18:09 +03:00
samba-mirror/ctdb/libctdb/messages.c
Volker Lendecke 8428f48708 Add includes required on FreeBSD
Signed-off-by: Michael Adam <obnox@samba.org>

(This used to be ctdb commit be5c3a57df39d85fbb7ee922924787340e2b0595)
2012-02-13 16:23:17 +01:00

238 lines
6.2 KiB
C

/*
core of libctdb
Copyright (C) Rusty Russell 2010
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 <sys/socket.h>
#include "libctdb_private.h"
#include "messages.h"
#include "io_elem.h"
#include <ctdb.h>
#include <tdb.h>
#include <ctdb_protocol.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
/* Remove type-safety macros. */
#undef ctdb_set_message_handler_send
#undef ctdb_set_message_handler_recv
#undef ctdb_remove_message_handler_send
struct message_handler_info {
struct message_handler_info *next, *prev;
uint64_t srvid;
ctdb_message_fn_t handler;
void *handler_data;
};
void deliver_message(struct ctdb_connection *ctdb, struct ctdb_req_header *hdr)
{
struct message_handler_info *i;
struct ctdb_req_message *msg = (struct ctdb_req_message *)hdr;
TDB_DATA data;
bool found;
data.dptr = msg->data;
data.dsize = msg->datalen;
/* Note: we want to call *every* handler: there may be more than one */
for (i = ctdb->message_handlers; i; i = i->next) {
if (i->srvid == msg->srvid) {
i->handler(ctdb, msg->srvid, data, i->handler_data);
found = true;
}
}
if (!found) {
DEBUG(ctdb, LOG_WARNING,
"ctdb_service: messsage for unregistered srvid %llu",
(unsigned long long)msg->srvid);
}
}
void remove_message_handlers(struct ctdb_connection *ctdb)
{
struct message_handler_info *i;
/* ctdbd should unregister automatically when we close fd, so we don't
need to do that here. */
while ((i = ctdb->message_handlers) != NULL) {
DLIST_REMOVE(ctdb->message_handlers, i);
free(i);
}
}
static void free_info(struct ctdb_connection *ctdb, struct ctdb_request *req);
struct ctdb_request *
ctdb_set_message_handler_send(struct ctdb_connection *ctdb, uint64_t srvid,
ctdb_message_fn_t handler, void *handler_data,
ctdb_callback_t callback, void *private_data)
{
struct message_handler_info *info;
struct ctdb_request *req;
info = malloc(sizeof(*info));
if (!info) {
DEBUG(ctdb, LOG_ERR,
"ctdb_set_message_handler_send: allocating info");
return NULL;
}
req = new_ctdb_control_request(ctdb, CTDB_CONTROL_REGISTER_SRVID,
CTDB_CURRENT_NODE, NULL, 0,
callback, private_data);
if (!req) {
DEBUG(ctdb, LOG_ERR,
"ctdb_set_message_handler_send: allocating request");
free(info);
return NULL;
}
req->extra = info;
req->extra_destructor = free_info;
req->hdr.control->srvid = srvid;
info->srvid = srvid;
info->handler = handler;
info->handler_data = handler_data;
DEBUG(ctdb, LOG_DEBUG,
"ctdb_set_message_handler_send: sending request %u for id %llx",
req->hdr.hdr->reqid, (unsigned long long)srvid);
return req;
}
static void free_info(struct ctdb_connection *ctdb, struct ctdb_request *req)
{
free(req->extra);
}
bool ctdb_set_message_handler_recv(struct ctdb_connection *ctdb,
struct ctdb_request *req)
{
struct message_handler_info *info = req->extra;
struct ctdb_reply_control *reply;
reply = unpack_reply_control(req, CTDB_CONTROL_REGISTER_SRVID);
if (!reply) {
return false;
}
if (reply->status != 0) {
DEBUG(ctdb, LOG_ERR,
"ctdb_set_message_handler_recv: status %i",
reply->status);
return false;
}
/* Put ourselves in list of handlers. */
DLIST_ADD(ctdb->message_handlers, info);
/* Keep safe from destructor */
req->extra = NULL;
return true;
}
struct ctdb_request *
ctdb_remove_message_handler_send(struct ctdb_connection *ctdb, uint64_t srvid,
ctdb_message_fn_t handler, void *hdata,
ctdb_callback_t callback, void *cbdata)
{
struct message_handler_info *i;
struct ctdb_request *req;
for (i = ctdb->message_handlers; i; i = i->next) {
if (i->srvid == srvid
&& i->handler == handler && i->handler_data == hdata) {
break;
}
}
if (!i) {
DEBUG(ctdb, LOG_ALERT,
"ctdb_remove_message_handler_send: no such handler");
errno = ENOENT;
return NULL;
}
req = new_ctdb_control_request(ctdb, CTDB_CONTROL_DEREGISTER_SRVID,
CTDB_CURRENT_NODE, NULL, 0,
callback, cbdata);
if (!req) {
DEBUG(ctdb, LOG_ERR,
"ctdb_remove_message_handler_send: allocating request");
return NULL;
}
req->hdr.control->srvid = srvid;
req->extra = i;
DEBUG(ctdb, LOG_DEBUG,
"ctdb_set_remove_handler_send: sending request %u for id %llu",
req->hdr.hdr->reqid, (unsigned long long)srvid);
return req;
}
bool ctdb_remove_message_handler_recv(struct ctdb_connection *ctdb,
struct ctdb_request *req)
{
struct message_handler_info *handler = req->extra;
struct ctdb_reply_control *reply;
reply = unpack_reply_control(req, CTDB_CONTROL_DEREGISTER_SRVID);
if (!reply) {
return false;
}
if (reply->status != 0) {
DEBUG(ctdb, LOG_ERR,
"ctdb_remove_message_handler_recv: status %i",
reply->status);
return false;
}
/* Remove ourselves from list of handlers. */
DLIST_REMOVE(ctdb->message_handlers, handler);
free(handler);
/* Crash if they call this again! */
req->extra = NULL;
return true;
}
bool ctdb_send_message(struct ctdb_connection *ctdb,
uint32_t pnn, uint64_t srvid,
TDB_DATA data)
{
struct ctdb_request *req;
struct ctdb_req_message *pkt;
/* We just discard it once it's finished: no reply. */
req = new_ctdb_request(
ctdb, offsetof(struct ctdb_req_message, data) + data.dsize,
ctdb_cancel_callback, NULL);
if (!req) {
DEBUG(ctdb, LOG_ERR, "ctdb_set_message: allocating message");
return false;
}
io_elem_init_req_header(req->io,
CTDB_REQ_MESSAGE, pnn, new_reqid(ctdb));
pkt = req->hdr.message;
pkt->srvid = srvid;
pkt->datalen = data.dsize;
memcpy(pkt->data, data.dptr, data.dsize);
DLIST_ADD_END(ctdb->outq, req, struct ctdb_request);
return true;
}