1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-21 18:04:06 +03:00
samba-mirror/ctdb/protocol/protocol_debug.c
Martin Schwenke 36938bfdd0 ctdb-server: Rename CTDB_BROADCAST_VNNMAP -> CTDB_BROADCAST_ACTIVE
This broadcast is misnamed.  Both places where this type of broadcast
is used expect the broadcast to go to all active nodes.

Make the corresponding change to the semantics in the daemon by
sending to all active nodes.

There is a mismatch between the ideas of VNN map and active nodes.  A
node that is not in the VNN map but is active can still host database
records.  These were the same until the LMASTER capability was
introduced and then the logic was not updated.

The only place where the VNN map is relevant is when finding the
location master of a record in the migration code.

BUG: https://bugzilla.samba.org/show_bug.cgi?id=13499

Signed-off-by: Martin Schwenke <martin@meltin.net>
Reviewed-by: Amitay Isaacs <amitay@gmail.com>
2018-07-02 08:51:22 +02:00

740 lines
21 KiB
C

/*
CTDB protocol marshalling
Copyright (C) Amitay Isaacs 2016
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 "replace.h"
#include "system/network.h"
#include "system/locale.h"
#include <talloc.h>
#include <tdb.h>
#include <protocol/protocol.h>
#include <protocol/protocol_api.h>
/*
* Utility functions
*/
struct uint32_map {
uint32_t key;
#define MAP_END 0xffffffff
const char *name;
};
static void uint32_map_print(struct uint32_map *map, uint32_t key, FILE *fp)
{
int i = 0;
while (map[i].key != MAP_END) {
if (key == map[i].key) {
fprintf(fp, "%s", map[i].name);
return;
}
i = i+1;
}
fprintf(fp, "UNKNOWN(%u)", key);
}
static void tdb_data_print(TDB_DATA d, FILE *fp)
{
unsigned char *p = (unsigned char *)d.dptr;
int len = d.dsize;
while (len--) {
if (isprint(*p) && !strchr("\"\\", *p)) {
fputc(*p, fp);
} else {
fprintf(fp, "\\%02X", *p);
}
p++;
}
}
/*
* Data types
*/
static void ctdb_operation_print(uint32_t operation, FILE *fp)
{
struct uint32_map map[] = {
{ CTDB_REQ_CALL, "REQ_CALL" },
{ CTDB_REPLY_CALL, "REPLY_CALL" },
{ CTDB_REQ_DMASTER, "REQ_DMASTER" },
{ CTDB_REPLY_DMASTER, "REPLY_DMASTER" },
{ CTDB_REPLY_ERROR, "REPLY_ERROR" },
{ CTDB_REQ_MESSAGE, "REQ_MESSAGE" },
{ CTDB_REQ_CONTROL, "REQ_CONTROL", },
{ CTDB_REPLY_CONTROL, "REPLY_CONTROL" },
{ CTDB_REQ_KEEPALIVE, "REQ_KEEPALIVE" },
{ MAP_END, "" },
};
uint32_map_print(map, operation, fp);
}
static void ctdb_callid_print(uint32_t callid, FILE *fp)
{
struct uint32_map map[] = {
{ CTDB_NULL_FUNC, "NULL" },
{ CTDB_FETCH_FUNC, "FETCH" },
{ CTDB_FETCH_WITH_HEADER_FUNC, "FETCH_WITH_HEADER" },
{ MAP_END, "" },
};
uint32_map_print(map, callid, fp);
}
static void ctdb_opcode_print(uint32_t opcode, FILE *fp)
{
struct uint32_map map[] = {
{ CTDB_CONTROL_PROCESS_EXISTS, "PROCESS_EXISTS" },
{ CTDB_CONTROL_STATISTICS, "STATISTICS" },
{ CTDB_CONTROL_PING, "PING" },
{ CTDB_CONTROL_GETDBPATH, "GETDBPATH" },
{ CTDB_CONTROL_GETVNNMAP, "GETVNNMAP" },
{ CTDB_CONTROL_SETVNNMAP, "SETVNNMAP" },
{ CTDB_CONTROL_GET_DEBUG, "GET_DEBUG" },
{ CTDB_CONTROL_SET_DEBUG, "SET_DEBUG" },
{ CTDB_CONTROL_GET_DBMAP, "GET_DBMAP" },
{ CTDB_CONTROL_GET_NODEMAPv4, "GET_NODEMAPv4" },
{ CTDB_CONTROL_SET_DMASTER, "SET_DMASTER" },
{ CTDB_CONTROL_PULL_DB, "PULL_DB" },
{ CTDB_CONTROL_PUSH_DB, "PUSH_DB" },
{ CTDB_CONTROL_GET_RECMODE, "GET_RECMODE" },
{ CTDB_CONTROL_SET_RECMODE, "SET_RECMODE" },
{ CTDB_CONTROL_STATISTICS_RESET, "STATISTICS_RESET" },
{ CTDB_CONTROL_DB_ATTACH, "DB_ATTACH" },
{ CTDB_CONTROL_SET_CALL, "SET_CALL" },
{ CTDB_CONTROL_TRAVERSE_START, "TRAVERSE_START" },
{ CTDB_CONTROL_TRAVERSE_ALL, "TRAVERSE_ALL" },
{ CTDB_CONTROL_TRAVERSE_DATA, "TRAVERSE_DATA" },
{ CTDB_CONTROL_REGISTER_SRVID, "REGISTER_SRVID" },
{ CTDB_CONTROL_DEREGISTER_SRVID, "DEREGISTER_SRVID" },
{ CTDB_CONTROL_GET_DBNAME, "GET_DBNAME" },
{ CTDB_CONTROL_ENABLE_SEQNUM, "ENABLE_SEQNUM" },
{ CTDB_CONTROL_UPDATE_SEQNUM, "UPDATE_SEQNUM" },
{ CTDB_CONTROL_DUMP_MEMORY, "DUMP_MEMORY" },
{ CTDB_CONTROL_GET_PID, "GET_PID" },
{ CTDB_CONTROL_GET_RECMASTER, "GET_RECMASTER" },
{ CTDB_CONTROL_SET_RECMASTER, "SET_RECMASTER" },
{ CTDB_CONTROL_FREEZE, "FREEZE" },
{ CTDB_CONTROL_THAW, "THAW" },
{ CTDB_CONTROL_GET_PNN, "GET_PNN" },
{ CTDB_CONTROL_SHUTDOWN, "SHUTDOWN" },
{ CTDB_CONTROL_GET_MONMODE, "GET_MONMODE" },
{ CTDB_CONTROL_TAKEOVER_IPv4, "TAKEOVER_IPv4" },
{ CTDB_CONTROL_RELEASE_IPv4, "RELEASE_IPv4" },
{ CTDB_CONTROL_TCP_CLIENT, "TCP_CLIENT" },
{ CTDB_CONTROL_TCP_ADD, "TCP_ADD" },
{ CTDB_CONTROL_TCP_REMOVE, "TCP_REMOVE" },
{ CTDB_CONTROL_STARTUP, "STARTUP" },
{ CTDB_CONTROL_SET_TUNABLE, "SET_TUNABLE" },
{ CTDB_CONTROL_GET_TUNABLE, "GET_TUNABLE" },
{ CTDB_CONTROL_LIST_TUNABLES, "LIST_TUNABLES" },
{ CTDB_CONTROL_GET_PUBLIC_IPSv4, "GET_PUBLIC_IPSv4" },
{ CTDB_CONTROL_MODIFY_FLAGS, "MODIFY_FLAGS" },
{ CTDB_CONTROL_GET_ALL_TUNABLES, "GET_ALL_TUNABLES" },
{ CTDB_CONTROL_KILL_TCP, "KILL_TCP" },
{ CTDB_CONTROL_GET_TCP_TICKLE_LIST, "GET_TCP_TICKLE_LIST" },
{ CTDB_CONTROL_SET_TCP_TICKLE_LIST, "SET_TCP_TICKLE_LIST" },
{ CTDB_CONTROL_REGISTER_SERVER_ID, "REGISTER_SERVER_ID" },
{ CTDB_CONTROL_UNREGISTER_SERVER_ID, "UNREGISTER_SERVER_ID" },
{ CTDB_CONTROL_CHECK_SERVER_ID, "CHECK_SERVER_ID" },
{ CTDB_CONTROL_GET_SERVER_ID_LIST, "GET_SERVER_ID_LIST" },
{ CTDB_CONTROL_DB_ATTACH_PERSISTENT, "DB_ATTACH_PERSISTENT" },
{ CTDB_CONTROL_PERSISTENT_STORE, "PERSISTENT_STORE" },
{ CTDB_CONTROL_UPDATE_RECORD, "UPDATE_RECORD" },
{ CTDB_CONTROL_SEND_GRATUITOUS_ARP, "SEND_GRATUITOUS_ARP" },
{ CTDB_CONTROL_TRANSACTION_START, "TRANSACTION_START" },
{ CTDB_CONTROL_TRANSACTION_COMMIT, "TRANSACTION_COMMIT" },
{ CTDB_CONTROL_WIPE_DATABASE, "WIPE_DATABASE" },
{ CTDB_CONTROL_UPTIME, "UPTIME" },
{ CTDB_CONTROL_START_RECOVERY, "START_RECOVERY" },
{ CTDB_CONTROL_END_RECOVERY, "END_RECOVERY" },
{ CTDB_CONTROL_RELOAD_NODES_FILE, "RELOAD_NODES_FILE" },
{ CTDB_CONTROL_TRY_DELETE_RECORDS, "TRY_DELETE_RECORDS" },
{ CTDB_CONTROL_ENABLE_MONITOR, "ENABLE_MONITOR" },
{ CTDB_CONTROL_DISABLE_MONITOR, "DISABLE_MONITOR" },
{ CTDB_CONTROL_ADD_PUBLIC_IP, "ADD_PUBLIC_IP" },
{ CTDB_CONTROL_DEL_PUBLIC_IP, "DEL_PUBLIC_IP" },
{ CTDB_CONTROL_RUN_EVENTSCRIPTS, "RUN_EVENTSCRIPTS" },
{ CTDB_CONTROL_GET_CAPABILITIES, "GET_CAPABILITIES" },
{ CTDB_CONTROL_START_PERSISTENT_UPDATE, "START_PERSISTENT_UPDATE" },
{ CTDB_CONTROL_CANCEL_PERSISTENT_UPDATE, "CANCEL_PERSISTENT_UPDATE" },
{ CTDB_CONTROL_TRANS2_COMMIT, "TRANS2_COMMIT" },
{ CTDB_CONTROL_TRANS2_FINISHED, "TRANS2_FINISHED" },
{ CTDB_CONTROL_TRANS2_ERROR, "TRANS2_ERROR" },
{ CTDB_CONTROL_TRANS2_COMMIT_RETRY, "TRANS2_COMMIT_RETRY" },
{ CTDB_CONTROL_RECD_PING, "RECD_PING" },
{ CTDB_CONTROL_RELEASE_IP, "RELEASE_IP" },
{ CTDB_CONTROL_TAKEOVER_IP, "TAKEOVER_IP" },
{ CTDB_CONTROL_GET_PUBLIC_IPS, "GET_PUBLIC_IPS" },
{ CTDB_CONTROL_GET_NODEMAP, "GET_NODEMAP" },
{ CTDB_CONTROL_GET_EVENT_SCRIPT_STATUS, "GET_EVENT_SCRIPT_STATUS" },
{ CTDB_CONTROL_TRAVERSE_KILL, "TRAVERSE_KILL" },
{ CTDB_CONTROL_RECD_RECLOCK_LATENCY, "RECD_RECLOCK_LATENCY" },
{ CTDB_CONTROL_GET_RECLOCK_FILE, "GET_RECLOCK_FILE" },
{ CTDB_CONTROL_STOP_NODE, "STOP_NODE" },
{ CTDB_CONTROL_CONTINUE_NODE, "CONTINUE_NODE" },
{ CTDB_CONTROL_SET_NATGWSTATE, "SET_NATGWSTATE" },
{ CTDB_CONTROL_SET_LMASTERROLE, "SET_LMASTERROLE" },
{ CTDB_CONTROL_SET_RECMASTERROLE, "SET_RECMASTERROLE" },
{ CTDB_CONTROL_ENABLE_SCRIPT, "ENABLE_SCRIPT" },
{ CTDB_CONTROL_DISABLE_SCRIPT, "DISABLE_SCRIPT" },
{ CTDB_CONTROL_SET_BAN_STATE, "SET_BAN_STATE" },
{ CTDB_CONTROL_GET_BAN_STATE, "GET_BAN_STATE" },
{ CTDB_CONTROL_SET_DB_PRIORITY, "SET_DB_PRIORITY" },
{ CTDB_CONTROL_GET_DB_PRIORITY, "GET_DB_PRIORITY" },
{ CTDB_CONTROL_TRANSACTION_CANCEL, "TRANSACTION_CANCEL" },
{ CTDB_CONTROL_REGISTER_NOTIFY, "REGISTER_NOTIFY" },
{ CTDB_CONTROL_DEREGISTER_NOTIFY, "DEREGISTER_NOTIFY" },
{ CTDB_CONTROL_TRANS2_ACTIVE, "TRANS2_ACTIVE" },
{ CTDB_CONTROL_GET_LOG, "GET_LOG" },
{ CTDB_CONTROL_CLEAR_LOG, "CLEAR_LOG" },
{ CTDB_CONTROL_TRANS3_COMMIT, "TRANS3_COMMIT" },
{ CTDB_CONTROL_GET_DB_SEQNUM, "GET_DB_SEQNUM" },
{ CTDB_CONTROL_DB_SET_HEALTHY, "DB_SET_HEALTHY" },
{ CTDB_CONTROL_DB_GET_HEALTH, "DB_GET_HEALTH" },
{ CTDB_CONTROL_GET_PUBLIC_IP_INFO, "GET_PUBLIC_IP_INFO" },
{ CTDB_CONTROL_GET_IFACES, "GET_IFACES" },
{ CTDB_CONTROL_SET_IFACE_LINK_STATE, "SET_IFACE_LINK_STATE" },
{ CTDB_CONTROL_TCP_ADD_DELAYED_UPDATE, "TCP_ADD_DELAYED_UPDATE" },
{ CTDB_CONTROL_GET_STAT_HISTORY, "GET_STAT_HISTORY" },
{ CTDB_CONTROL_SCHEDULE_FOR_DELETION, "SCHEDULE_FOR_DELETION" },
{ CTDB_CONTROL_SET_DB_READONLY, "SET_DB_READONLY" },
{ CTDB_CONTROL_CHECK_SRVIDS, "CHECK_SRVIDS" },
{ CTDB_CONTROL_TRAVERSE_START_EXT, "TRAVERSE_START_EXT" },
{ CTDB_CONTROL_GET_DB_STATISTICS, "GET_DB_STATISTICS" },
{ CTDB_CONTROL_SET_DB_STICKY, "SET_DB_STICKY" },
{ CTDB_CONTROL_RELOAD_PUBLIC_IPS, "RELOAD_PUBLIC_IPS" },
{ CTDB_CONTROL_TRAVERSE_ALL_EXT, "TRAVERSE_ALL_EXT" },
{ CTDB_CONTROL_RECEIVE_RECORDS, "RECEIVE_RECORDS" },
{ CTDB_CONTROL_IPREALLOCATED, "IPREALLOCATED" },
{ CTDB_CONTROL_GET_RUNSTATE, "GET_RUNSTATE" },
{ CTDB_CONTROL_DB_DETACH, "DB_DETACH" },
{ CTDB_CONTROL_GET_NODES_FILE, "GET_NODES_FILE" },
{ CTDB_CONTROL_DB_FREEZE, "DB_FREEZE" },
{ CTDB_CONTROL_DB_THAW, "DB_THAW" },
{ CTDB_CONTROL_DB_TRANSACTION_START, "DB_TRANSACTION_START" },
{ CTDB_CONTROL_DB_TRANSACTION_COMMIT, "DB_TRANSACTION_COMMIT" },
{ CTDB_CONTROL_DB_TRANSACTION_CANCEL, "DB_TRANSACTION_CANCEL" },
{ CTDB_CONTROL_DB_PULL, "DB_PULL" },
{ CTDB_CONTROL_DB_PUSH_START, "DB_PUSH_START" },
{ CTDB_CONTROL_DB_PUSH_CONFIRM, "DB_PUSH_CONFIRM" },
{ CTDB_CONTROL_DB_OPEN_FLAGS, "DB_OPEN_FLAGS" },
{ CTDB_CONTROL_DB_ATTACH_REPLICATED, "DB_ATTACH_REPLICATED" },
{ CTDB_CONTROL_CHECK_PID_SRVID, "CHECK_PID_SRVID" },
{ CTDB_CONTROL_TUNNEL_REGISTER, "TUNNEL_REGISTER" },
{ CTDB_CONTROL_TUNNEL_DEREGISTER, "TUNNEL_DEREGISTER" },
{ MAP_END, "" },
};
uint32_map_print(map, opcode, fp);
}
static void ctdb_control_flags_print(uint32_t flags, FILE *fp)
{
if (flags & CTDB_CTRL_FLAG_NOREPLY) {
fprintf(fp, "NOREPLY ");
}
if (flags & CTDB_CTRL_FLAG_OPCODE_SPECIFIC) {
fprintf(fp, "OPCODE_SPECIFIC ");
}
}
static void ctdb_pnn_print(uint32_t pnn, FILE *fp)
{
if (pnn == CTDB_CURRENT_NODE) {
fprintf(fp, "CURRENT");
} else if (pnn == CTDB_BROADCAST_ALL) {
fprintf(fp, "ALL");
} else if (pnn == CTDB_BROADCAST_ACTIVE) {
fprintf(fp, "ACTIVE");
} else if (pnn == CTDB_BROADCAST_CONNECTED) {
fprintf(fp, "CONNECTED");
} else if (pnn == CTDB_MULTICAST) {
fprintf(fp, "MULTICAST");
} else if (pnn == CTDB_UNKNOWN_PNN) {
fprintf(fp, "UNKNOWN");
} else {
fprintf(fp, "%u", pnn);
}
}
static void ctdb_srvid_print(uint64_t srvid, FILE *fp)
{
uint64_t prefix = 0xFFFF000000000000LL;
if (srvid == CTDB_SRVID_ALL) {
fprintf(fp, "ALL");
} else if ((srvid & prefix) == CTDB_SRVID_RECOVERY) {
srvid = srvid & ~CTDB_SRVID_RECOVERY;
fprintf(fp, "RECOVERY-%"PRIx64"", srvid);
} else if (srvid == CTDB_SRVID_BANNING) {
fprintf(fp, "BANNING");
} else if (srvid == CTDB_SRVID_ELECTION) {
fprintf(fp, "ELECTION");
} else if (srvid == CTDB_SRVID_RECONFIGURE) {
fprintf(fp, "RECONFIGURE");
} else if (srvid == CTDB_SRVID_RELEASE_IP) {
fprintf(fp, "RELEASE_IP");
} else if (srvid == CTDB_SRVID_TAKE_IP) {
fprintf(fp, "TAKE_IP");
} else if (srvid == CTDB_SRVID_SET_NODE_FLAGS) {
fprintf(fp, "SET_NODE_FLAGS");
} else if (srvid == CTDB_SRVID_RECD_UPDATE_IP) {
fprintf(fp, "RECD_UPDATE_IP");
} else if (srvid == CTDB_SRVID_VACUUM_FETCH) {
fprintf(fp, "VACUUM_FETCH");
} else if (srvid == CTDB_SRVID_DETACH_DATABASE) {
fprintf(fp, "DETACH_DATABASE");
} else if (srvid == CTDB_SRVID_MEM_DUMP) {
fprintf(fp, "MEM_DUMP");
} else if (srvid == CTDB_SRVID_GETLOG) {
fprintf(fp, "GETLOG");
} else if (srvid == CTDB_SRVID_CLEARLOG) {
fprintf(fp, "CLEARLOG");
} else if (srvid == CTDB_SRVID_PUSH_NODE_FLAGS) {
fprintf(fp, "PUSH_NODE_FLAGS");
} else if (srvid == CTDB_SRVID_RELOAD_NODES) {
fprintf(fp, "RELOAD_NODES");
} else if (srvid == CTDB_SRVID_TAKEOVER_RUN) {
fprintf(fp, "TAKEOVER_RUN");
} else if (srvid == CTDB_SRVID_REBALANCE_NODE) {
fprintf(fp, "REBALANCE_NODE");
} else if (srvid == CTDB_SRVID_DISABLE_TAKEOVER_RUNS) {
fprintf(fp, "DISABLE_TAKEOVER_RUNS");
} else if (srvid == CTDB_SRVID_DISABLE_RECOVERIES) {
fprintf(fp, "DISABLE_RECOVERIES");
} else if (srvid == CTDB_SRVID_DISABLE_IP_CHECK) {
fprintf(fp, "DISABLE_IP_CHECK");
} else if ((srvid & prefix) == CTDB_SRVID_SAMBA_RANGE) {
if (srvid == CTDB_SRVID_SAMBA_NOTIFY) {
fprintf(fp, "SAMBA_NOTIFY");
} else {
srvid &= ~CTDB_SRVID_SAMBA_RANGE;
fprintf(fp, "samba-0x%"PRIx64"", srvid);
}
} else if ((srvid & prefix) == CTDB_SRVID_NFSD_RANGE) {
srvid &= ~CTDB_SRVID_NFSD_RANGE;
fprintf(fp, "nfsd-0x%"PRIx64"", srvid);
} else if ((srvid & prefix) == CTDB_SRVID_ISCSID_RANGE) {
srvid &= ~CTDB_SRVID_ISCSID_RANGE;
fprintf(fp, "iscsi-0x%"PRIx64"", srvid);
} else if ((srvid & prefix) == CTDB_SRVID_TOOL_RANGE) {
srvid &= ~CTDB_SRVID_TOOL_RANGE;
fprintf(fp, "tool-0x%"PRIx64"", srvid);
} else if ((srvid & prefix) == CTDB_SRVID_TEST_RANGE) {
srvid &= ~CTDB_SRVID_TEST_RANGE;
fprintf(fp, "test-0x%"PRIx64"", srvid);
} else if ((srvid & prefix) == CTDB_SRVID_PID_RANGE) {
if (srvid < UINT16_MAX) {
fprintf(fp, "pid-%"PRIu64, srvid);
} else {
fprintf(fp, "pid-0x%"PRIx64, srvid);
}
} else {
fprintf(fp, "0x%"PRIx64, srvid);
}
}
static void ctdb_tunnel_id_print(uint64_t tunnel_id, FILE *fp)
{
if ((tunnel_id & CTDB_TUNNEL_TEST) == CTDB_TUNNEL_TEST) {
fprintf(fp, "TEST-%"PRIx64, tunnel_id);
} else {
fprintf(fp, "0x%"PRIx64, tunnel_id);
}
}
static void ctdb_tunnel_flags_print(uint32_t flags, FILE *fp)
{
if (flags & CTDB_TUNNEL_FLAG_REQUEST) {
fprintf(fp, "REQUEST ");
}
if (flags & CTDB_TUNNEL_FLAG_REPLY) {
fprintf(fp, "REPLY ");
}
if (flags & CTDB_TUNNEL_FLAG_NOREPLY) {
fprintf(fp, "NOREPLY ");
}
}
/*
* Print routines
*/
static void ctdb_req_header_print(struct ctdb_req_header *h, FILE *fp)
{
fprintf(fp, "Header\n");
fprintf(fp, " length:%u magic:0x%"PRIx32" version:%u generation:0x%"PRIx32"\n",
h->length, h->ctdb_magic, h->ctdb_version, h->generation);
fprintf(fp, " ");
ctdb_operation_print(h->operation, fp);
fprintf(fp, " dst:");
ctdb_pnn_print(h->destnode, fp);
fprintf(fp, " src:");
ctdb_pnn_print(h->srcnode, fp);
fprintf(fp, " reqid:0x%"PRIx32"\n", h->reqid);
}
static void ctdb_req_call_print(struct ctdb_req_call *c, FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " db:0x%"PRIx32" ", c->db_id);
ctdb_callid_print(c->callid, fp);
fprintf(fp, "\n");
fprintf(fp, " key:");
tdb_data_print(c->key, fp);
fprintf(fp, "\n");
}
static void ctdb_reply_call_print(struct ctdb_reply_call *c, FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " status:%d\n", c->status);
if (c->status == 0) {
fprintf(fp, " data:");
tdb_data_print(c->data, fp);
fprintf(fp, "\n");
}
}
static void ctdb_req_dmaster_print(struct ctdb_req_dmaster *c, FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " db:0x%"PRIx32" rsn:0x%"PRIx64" dmaster:%u\n",
c->db_id, c->rsn, c->dmaster);
fprintf(fp, " key:");
tdb_data_print(c->key, fp);
fprintf(fp, "\n");
fprintf(fp, " data:");
tdb_data_print(c->data, fp);
fprintf(fp, "\n");
}
static void ctdb_reply_dmaster_print(struct ctdb_reply_dmaster *c, FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " db:0x%"PRIx32" rsn:0x%"PRIx64"\n", c->db_id, c->rsn);
fprintf(fp, " key:");
tdb_data_print(c->key, fp);
fprintf(fp, "\n");
fprintf(fp, " data:");
tdb_data_print(c->data, fp);
fprintf(fp, "\n");
}
static void ctdb_reply_error_print(struct ctdb_reply_error *c, FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " status:%d\n", c->status);
if (c->status != 0) {
fprintf(fp, " msg:");
tdb_data_print(c->msg, fp);
fprintf(fp, "\n");
}
}
static void ctdb_req_message_data_print(struct ctdb_req_message_data *c,
FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " srvid:");
ctdb_srvid_print(c->srvid, fp);
fprintf(fp, "\n");
fprintf(fp, " data:");
tdb_data_print(c->data, fp);
fprintf(fp, "\n");
}
static void ctdb_req_control_print(struct ctdb_req_control *c, FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " ");
ctdb_opcode_print(c->opcode, fp);
fprintf(fp, " srvid:");
ctdb_srvid_print(c->srvid, fp);
fprintf(fp, " client_id:0x%"PRIx32" ", c->client_id);
ctdb_control_flags_print(c->flags, fp);
fprintf(fp, "\n");
}
static void ctdb_reply_control_print(struct ctdb_reply_control *c, FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " status:%d ", c->status);
if (c->errmsg != NULL) {
fprintf(fp, "errmsg: %s", c->errmsg);
}
fprintf(fp, "\n");
}
static void ctdb_req_keepalive_print(struct ctdb_req_keepalive *c, FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " version:0x%"PRIx32, c->version);
fprintf(fp, " uptime:%"PRIu32, c->uptime);
fprintf(fp, "\n");
}
static void ctdb_req_tunnel_print(struct ctdb_req_tunnel *c, FILE *fp)
{
fprintf(fp, "Data\n");
fprintf(fp, " tunnel_id:");
ctdb_tunnel_id_print(c->tunnel_id, fp);
ctdb_tunnel_flags_print(c->flags, fp);
tdb_data_print(c->data, fp);
fprintf(fp, "\n");
}
/*
* Parse routines
*/
static void ctdb_req_call_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_req_call c;
int ret;
ret = ctdb_req_call_pull(buf, buflen, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REQ_CALL\n");
return;
}
ctdb_req_call_print(&c, fp);
}
static void ctdb_reply_call_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_reply_call c;
int ret;
ret = ctdb_reply_call_pull(buf, buflen, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REPLY_CALL\n");
return;
}
ctdb_reply_call_print(&c, fp);
}
static void ctdb_req_dmaster_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_req_dmaster c;
int ret;
ret = ctdb_req_dmaster_pull(buf, buflen, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REQ_DMASTER\n");
return;
}
ctdb_req_dmaster_print(&c, fp);
}
static void ctdb_reply_dmaster_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_reply_dmaster c;
int ret;
ret = ctdb_reply_dmaster_pull(buf, buflen, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REPLY_DMASTER\n");
return;
}
ctdb_reply_dmaster_print(&c, fp);
}
static void ctdb_reply_error_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_reply_error c;
int ret;
ret = ctdb_reply_error_pull(buf, buflen, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REPLY_ERROR\n");
return;
}
ctdb_reply_error_print(&c, fp);
}
static void ctdb_req_message_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_req_message_data c;
int ret;
ret = ctdb_req_message_data_pull(buf, buflen, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REQ_MESSAGE\n");
return;
}
ctdb_req_message_data_print(&c, fp);
}
static void ctdb_req_control_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_req_control c;
int ret;
ret = ctdb_req_control_pull(buf, buflen, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REQ_CONTROL\n");
return;
}
ctdb_req_control_print(&c, fp);
}
static void ctdb_reply_control_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_reply_control c;
int ret;
ret = ctdb_reply_control_pull(buf, buflen, -1, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REPLY_CONTROL\n");
return;
}
ctdb_reply_control_print(&c, fp);
}
static void ctdb_req_keepalive_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_req_keepalive c;
int ret;
ret = ctdb_req_keepalive_pull(buf, buflen, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REQ_KEEPALIVE\n");
return;
}
ctdb_req_keepalive_print(&c, fp);
}
static void ctdb_req_tunnel_parse(uint8_t *buf, size_t buflen, FILE *fp,
TALLOC_CTX *mem_ctx)
{
struct ctdb_req_tunnel c;
int ret;
ret = ctdb_req_tunnel_pull(buf, buflen, NULL, mem_ctx, &c);
if (ret != 0) {
fprintf(fp, "Failed to parse CTDB_REQ_TUNNEL\n");
return;
}
ctdb_req_tunnel_print(&c, fp);
}
/*
* Packet print
*/
void ctdb_packet_print(uint8_t *buf, size_t buflen, FILE *fp)
{
TALLOC_CTX *mem_ctx = talloc_new(NULL);
struct ctdb_req_header h;
size_t np;
int ret;
fprintf(fp, "Buffer len:%zu\n", buflen);
ret = ctdb_req_header_pull(buf, buflen, &h, &np);
if (ret != 0) {
fprintf(fp, "Failed to parse ctdb packet header\n");
return;
}
ctdb_req_header_print(&h, fp);
if (h.length > buflen) {
fprintf(fp, "Packet length mismatch\n");
}
ret = ctdb_req_header_verify(&h, 0);
if (ret != 0) {
fprintf(fp, "Invalid ctdb packet header\n");
return;
}
switch (h.operation) {
case CTDB_REQ_CALL:
ctdb_req_call_parse(buf, buflen, fp, mem_ctx);
break;
case CTDB_REPLY_CALL:
ctdb_reply_call_parse(buf, buflen, fp, mem_ctx);
break;
case CTDB_REQ_DMASTER:
ctdb_req_dmaster_parse(buf, buflen, fp, mem_ctx);
break;
case CTDB_REPLY_DMASTER:
ctdb_reply_dmaster_parse(buf, buflen, fp, mem_ctx);
break;
case CTDB_REPLY_ERROR:
ctdb_reply_error_parse(buf, buflen, fp, mem_ctx);
break;
case CTDB_REQ_MESSAGE:
ctdb_req_message_parse(buf, buflen, fp, mem_ctx);
break;
case CTDB_REQ_CONTROL:
ctdb_req_control_parse(buf, buflen, fp, mem_ctx);
break;
case CTDB_REPLY_CONTROL:
ctdb_reply_control_parse(buf, buflen, fp, mem_ctx);
break;
case CTDB_REQ_KEEPALIVE:
ctdb_req_keepalive_parse(buf, buflen, fp, mem_ctx);
break;
case CTDB_REQ_TUNNEL:
ctdb_req_tunnel_parse(buf, buflen, fp, mem_ctx);
break;
default:
fprintf(fp, "Invalid ctdb operation\n");
break;
}
talloc_free(mem_ctx);
}