mirror of
https://github.com/samba-team/samba.git
synced 2025-01-11 05:18:09 +03:00
36938bfdd0
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>
740 lines
21 KiB
C
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);
|
|
}
|