2017-12-15 11:16:54 +01:00
|
|
|
/* Only possible if we're in developer mode. */
|
2021-12-04 12:23:56 +01:00
|
|
|
#include "config.h"
|
2017-12-15 14:26:06 +01:00
|
|
|
#if DEVELOPER
|
2017-12-15 11:18:54 +01:00
|
|
|
#include <backtrace.h>
|
|
|
|
#include <ccan/tal/str/str.h>
|
2018-12-08 01:39:28 +01:00
|
|
|
#include <common/json_command.h>
|
2017-12-15 11:17:54 +01:00
|
|
|
#include <common/memleak.h>
|
2018-12-08 01:39:28 +01:00
|
|
|
#include <common/param.h>
|
2018-11-21 23:41:49 +01:00
|
|
|
#include <common/timeout.h>
|
2020-08-25 04:16:22 +02:00
|
|
|
#include <connectd/connectd_wiregen.h>
|
2018-11-22 03:17:29 +01:00
|
|
|
#include <errno.h>
|
2020-08-25 04:05:45 +02:00
|
|
|
#include <gossipd/gossipd_wiregen.h>
|
2020-08-25 03:55:38 +02:00
|
|
|
#include <hsmd/hsmd_wiregen.h>
|
2017-12-15 11:17:54 +01:00
|
|
|
#include <lightningd/chaintopology.h>
|
2017-12-15 11:16:54 +01:00
|
|
|
#include <lightningd/jsonrpc.h>
|
2017-12-15 11:17:54 +01:00
|
|
|
#include <lightningd/lightningd.h>
|
2021-12-04 12:23:56 +01:00
|
|
|
#include <lightningd/memdump.h>
|
2020-09-16 03:12:12 +02:00
|
|
|
#include <lightningd/opening_common.h>
|
2018-11-22 03:17:29 +01:00
|
|
|
#include <lightningd/peer_control.h>
|
2018-11-21 23:41:49 +01:00
|
|
|
#include <lightningd/subd.h>
|
2018-11-22 03:17:29 +01:00
|
|
|
#include <wire/wire_sync.h>
|
2017-12-15 11:16:54 +01:00
|
|
|
|
2018-10-19 03:17:49 +02:00
|
|
|
static void json_add_ptr(struct json_stream *response, const char *name,
|
2017-12-15 11:16:54 +01:00
|
|
|
const void *ptr)
|
|
|
|
{
|
|
|
|
char ptrstr[STR_MAX_CHARS(void *)];
|
2018-07-31 14:56:04 +02:00
|
|
|
snprintf(ptrstr, sizeof(ptrstr), "%p", ptr);
|
2017-12-15 11:16:54 +01:00
|
|
|
json_add_string(response, name, ptrstr);
|
|
|
|
}
|
|
|
|
|
2019-08-02 17:17:06 +02:00
|
|
|
static size_t add_memdump(struct json_stream *response,
|
2017-12-15 11:16:54 +01:00
|
|
|
const char *name, const tal_t *root,
|
|
|
|
struct command *cmd)
|
|
|
|
{
|
|
|
|
const tal_t *i;
|
2019-08-02 17:17:06 +02:00
|
|
|
size_t cumulative_size = 0;
|
2017-12-15 11:16:54 +01:00
|
|
|
|
|
|
|
json_array_start(response, name);
|
|
|
|
for (i = tal_first(root); i; i = tal_next(i)) {
|
|
|
|
const char *name = tal_name(i);
|
2019-08-02 17:17:06 +02:00
|
|
|
size_t size = tal_bytelen(i);
|
2017-12-15 11:16:54 +01:00
|
|
|
|
|
|
|
/* Don't try to dump this command! */
|
|
|
|
if (i == cmd || i == cmd->jcon)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Don't dump logs, we know they grow. */
|
|
|
|
if (name && streq(name, "struct log_book"))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
json_object_start(response, NULL);
|
|
|
|
json_add_ptr(response, "parent", tal_parent(i));
|
|
|
|
json_add_ptr(response, "value", i);
|
2019-08-02 17:17:06 +02:00
|
|
|
json_add_u64(response, "size", size);
|
2017-12-15 11:16:54 +01:00
|
|
|
if (name)
|
|
|
|
json_add_string(response, "label", name);
|
|
|
|
|
|
|
|
if (tal_first(i))
|
2019-08-02 17:17:06 +02:00
|
|
|
size += add_memdump(response, "children", i, cmd);
|
|
|
|
json_add_u64(response, "cumulative_size", size);
|
2017-12-15 11:16:54 +01:00
|
|
|
json_object_end(response);
|
2019-08-02 17:17:06 +02:00
|
|
|
cumulative_size += size;
|
2017-12-15 11:16:54 +01:00
|
|
|
}
|
|
|
|
json_array_end(response);
|
2019-08-02 17:17:06 +02:00
|
|
|
return cumulative_size;
|
2017-12-15 11:16:54 +01:00
|
|
|
}
|
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *json_memdump(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
2017-12-15 11:16:54 +01:00
|
|
|
{
|
2018-10-19 03:17:49 +02:00
|
|
|
struct json_stream *response;
|
2017-12-15 11:16:54 +01:00
|
|
|
|
2018-09-14 16:51:04 +02:00
|
|
|
if (!param(cmd, buffer, params, NULL))
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_param_failed();
|
2018-09-14 16:51:04 +02:00
|
|
|
|
2018-10-19 03:17:48 +02:00
|
|
|
response = json_stream_success(cmd);
|
2019-06-12 02:38:54 +02:00
|
|
|
add_memdump(response, "memdump", NULL, cmd);
|
2017-12-15 11:16:54 +01:00
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_success(cmd, response);
|
2017-12-15 11:16:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static const struct json_command dev_memdump_command = {
|
|
|
|
"dev-memdump",
|
2019-05-22 16:08:16 +02:00
|
|
|
"developer",
|
2017-12-15 11:16:54 +01:00
|
|
|
json_memdump,
|
2018-01-22 09:55:07 +01:00
|
|
|
"Show memory objects currently in use"
|
2017-12-15 11:16:54 +01:00
|
|
|
};
|
|
|
|
AUTODATA(json_command, &dev_memdump_command);
|
2017-12-15 11:17:54 +01:00
|
|
|
|
2018-02-21 16:06:07 +01:00
|
|
|
static int json_add_syminfo(void *data, uintptr_t pc UNUSED,
|
2017-12-15 11:18:54 +01:00
|
|
|
const char *filename, int lineno,
|
|
|
|
const char *function)
|
|
|
|
{
|
2018-10-19 03:17:49 +02:00
|
|
|
struct json_stream *response = data;
|
2017-12-15 11:18:54 +01:00
|
|
|
char *str;
|
|
|
|
|
2018-03-15 05:30:12 +01:00
|
|
|
/* This can happen in backtraces. */
|
|
|
|
if (!filename || !function)
|
|
|
|
return 0;
|
|
|
|
|
2017-12-15 11:18:54 +01:00
|
|
|
str = tal_fmt(response, "%s:%u (%s)", filename, lineno, function);
|
|
|
|
json_add_string(response, NULL, str);
|
|
|
|
tal_free(str);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-10-19 03:17:49 +02:00
|
|
|
static void json_add_backtrace(struct json_stream *response,
|
2017-12-15 11:18:54 +01:00
|
|
|
const uintptr_t *bt)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!bt)
|
|
|
|
return;
|
|
|
|
|
|
|
|
json_array_start(response, "backtrace");
|
|
|
|
/* First one serves as counter. */
|
|
|
|
for (i = 1; i < bt[0]; i++) {
|
|
|
|
backtrace_pcinfo(backtrace_state,
|
|
|
|
bt[i], json_add_syminfo,
|
|
|
|
NULL, response);
|
|
|
|
}
|
|
|
|
json_array_end(response);
|
|
|
|
}
|
|
|
|
|
2017-12-15 11:17:54 +01:00
|
|
|
static void scan_mem(struct command *cmd,
|
2018-10-19 03:17:49 +02:00
|
|
|
struct json_stream *response,
|
2018-11-21 23:41:49 +01:00
|
|
|
struct lightningd *ld,
|
|
|
|
const struct subd *leaking_subd)
|
2017-12-15 11:17:54 +01:00
|
|
|
{
|
|
|
|
struct htable *memtable;
|
|
|
|
const tal_t *i;
|
2017-12-15 11:18:54 +01:00
|
|
|
const uintptr_t *backtrace;
|
2017-12-15 11:17:54 +01:00
|
|
|
|
2017-12-15 11:20:54 +01:00
|
|
|
/* Enter everything, except this cmd and its jcon */
|
2020-09-23 03:37:04 +02:00
|
|
|
memtable = memleak_find_allocations(cmd, cmd, cmd->jcon);
|
2017-12-15 11:17:54 +01:00
|
|
|
|
|
|
|
/* First delete known false positives. */
|
2018-08-24 07:20:02 +02:00
|
|
|
memleak_remove_htable(memtable, &ld->topology->txwatches.raw);
|
|
|
|
memleak_remove_htable(memtable, &ld->topology->txowatches.raw);
|
|
|
|
memleak_remove_htable(memtable, &ld->htlcs_in.raw);
|
|
|
|
memleak_remove_htable(memtable, &ld->htlcs_out.raw);
|
2019-12-12 16:31:44 +01:00
|
|
|
memleak_remove_htable(memtable, &ld->htlc_sets.raw);
|
2017-12-15 11:17:54 +01:00
|
|
|
|
|
|
|
/* Now delete ld and those which it has pointers to. */
|
2020-09-23 03:37:04 +02:00
|
|
|
memleak_remove_region(memtable, ld, sizeof(*ld));
|
2017-12-15 11:17:54 +01:00
|
|
|
|
|
|
|
json_array_start(response, "leaks");
|
2017-12-15 11:18:54 +01:00
|
|
|
while ((i = memleak_get(memtable, &backtrace)) != NULL) {
|
2017-12-15 11:17:54 +01:00
|
|
|
const tal_t *p;
|
|
|
|
|
|
|
|
json_object_start(response, NULL);
|
|
|
|
json_add_ptr(response, "value", i);
|
|
|
|
if (tal_name(i))
|
|
|
|
json_add_string(response, "label", tal_name(i));
|
|
|
|
|
2017-12-15 11:18:54 +01:00
|
|
|
json_add_backtrace(response, backtrace);
|
2017-12-15 11:17:54 +01:00
|
|
|
json_array_start(response, "parents");
|
|
|
|
for (p = tal_parent(i); p; p = tal_parent(p)) {
|
|
|
|
json_add_string(response, NULL, tal_name(p));
|
|
|
|
p = tal_parent(p);
|
|
|
|
}
|
|
|
|
json_array_end(response);
|
|
|
|
json_object_end(response);
|
|
|
|
}
|
2018-11-21 23:41:49 +01:00
|
|
|
|
|
|
|
if (leaking_subd) {
|
|
|
|
json_object_start(response, NULL);
|
|
|
|
json_add_string(response, "subdaemon", leaking_subd->name);
|
|
|
|
json_object_end(response);
|
|
|
|
}
|
2017-12-15 11:17:54 +01:00
|
|
|
json_array_end(response);
|
|
|
|
}
|
|
|
|
|
2018-11-21 23:41:49 +01:00
|
|
|
struct leak_info {
|
|
|
|
struct command *cmd;
|
|
|
|
struct subd *leaker;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void report_leak_info2(struct leak_info *leak_info)
|
|
|
|
{
|
|
|
|
struct json_stream *response = json_stream_success(leak_info->cmd);
|
|
|
|
|
|
|
|
scan_mem(leak_info->cmd, response, leak_info->cmd->ld, leak_info->leaker);
|
|
|
|
|
2018-12-16 05:53:06 +01:00
|
|
|
was_pending(command_success(leak_info->cmd, response));
|
2018-11-21 23:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void report_leak_info(struct command *cmd, struct subd *leaker)
|
|
|
|
{
|
|
|
|
struct leak_info *leak_info = tal(cmd, struct leak_info);
|
|
|
|
|
|
|
|
leak_info->cmd = cmd;
|
|
|
|
leak_info->leaker = leaker;
|
|
|
|
|
|
|
|
/* Leak detection in a reply handler thinks we're leaking conn. */
|
2019-06-14 05:49:23 +02:00
|
|
|
notleak(new_reltimer(leak_info->cmd->ld->timers, leak_info->cmd,
|
2018-11-21 23:41:49 +01:00
|
|
|
time_from_sec(0),
|
|
|
|
report_leak_info2, leak_info));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void gossip_dev_memleak_done(struct subd *gossipd,
|
|
|
|
const u8 *reply,
|
|
|
|
const int *fds UNUSED,
|
|
|
|
struct command *cmd)
|
|
|
|
{
|
|
|
|
bool found_leak;
|
|
|
|
|
2020-08-25 04:05:45 +02:00
|
|
|
if (!fromwire_gossipd_dev_memleak_reply(reply, &found_leak)) {
|
2018-12-16 05:53:06 +01:00
|
|
|
was_pending(command_fail(cmd, LIGHTNINGD,
|
|
|
|
"Bad gossip_dev_memleak"));
|
2018-11-21 23:41:49 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
report_leak_info(cmd, found_leak ? gossipd : NULL);
|
|
|
|
}
|
|
|
|
|
2018-11-22 03:17:29 +01:00
|
|
|
static void connect_dev_memleak_done(struct subd *connectd,
|
|
|
|
const u8 *reply,
|
|
|
|
const int *fds UNUSED,
|
|
|
|
struct command *cmd)
|
|
|
|
{
|
|
|
|
bool found_leak;
|
|
|
|
|
2020-08-25 04:16:22 +02:00
|
|
|
if (!fromwire_connectd_dev_memleak_reply(reply, &found_leak)) {
|
2018-12-16 05:53:06 +01:00
|
|
|
was_pending(command_fail(cmd, LIGHTNINGD,
|
|
|
|
"Bad connect_dev_memleak"));
|
2018-11-22 03:17:29 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (found_leak) {
|
|
|
|
report_leak_info(cmd, connectd);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-12-17 04:52:05 +01:00
|
|
|
/* No leak? Ask openingd. */
|
|
|
|
opening_dev_memleak(cmd);
|
2018-11-22 03:17:29 +01:00
|
|
|
}
|
|
|
|
|
2018-11-22 03:17:29 +01:00
|
|
|
static void hsm_dev_memleak_done(struct subd *hsmd,
|
|
|
|
const u8 *reply,
|
|
|
|
struct command *cmd)
|
|
|
|
{
|
|
|
|
struct lightningd *ld = cmd->ld;
|
|
|
|
bool found_leak;
|
|
|
|
|
2020-08-25 03:55:38 +02:00
|
|
|
if (!fromwire_hsmd_dev_memleak_reply(reply, &found_leak)) {
|
2018-12-16 05:53:06 +01:00
|
|
|
was_pending(command_fail(cmd, LIGHTNINGD,
|
|
|
|
"Bad hsm_dev_memleak"));
|
2018-11-22 03:17:29 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (found_leak) {
|
|
|
|
report_leak_info(cmd, hsmd);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-12-17 04:52:05 +01:00
|
|
|
/* No leak? Ask gossipd. */
|
2020-08-25 04:05:45 +02:00
|
|
|
subd_req(ld->gossip, ld->gossip, take(towire_gossipd_dev_memleak(NULL)),
|
2018-12-17 04:52:05 +01:00
|
|
|
-1, 0, gossip_dev_memleak_done, cmd);
|
2018-11-22 03:17:29 +01:00
|
|
|
}
|
|
|
|
|
2018-11-22 03:17:29 +01:00
|
|
|
void peer_memleak_done(struct command *cmd, struct subd *leaker)
|
2018-11-22 03:17:29 +01:00
|
|
|
{
|
|
|
|
if (leaker)
|
|
|
|
report_leak_info(cmd, leaker);
|
|
|
|
else {
|
|
|
|
/* No leak there, try hsmd (we talk to hsm sync) */
|
2020-08-25 03:55:38 +02:00
|
|
|
u8 *msg = towire_hsmd_dev_memleak(NULL);
|
2018-11-22 03:17:29 +01:00
|
|
|
if (!wire_sync_write(cmd->ld->hsm_fd, take(msg)))
|
|
|
|
fatal("Could not write to HSM: %s", strerror(errno));
|
|
|
|
|
|
|
|
hsm_dev_memleak_done(cmd->ld->hsm,
|
|
|
|
wire_sync_read(tmpctx, cmd->ld->hsm_fd),
|
|
|
|
cmd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-22 03:17:29 +01:00
|
|
|
void opening_memleak_done(struct command *cmd, struct subd *leaker)
|
|
|
|
{
|
|
|
|
if (leaker)
|
|
|
|
report_leak_info(cmd, leaker);
|
|
|
|
else {
|
|
|
|
/* No leak there, try normal peers. */
|
|
|
|
peer_dev_memleak(cmd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-16 05:52:06 +01:00
|
|
|
static struct command_result *json_memleak(struct command *cmd,
|
|
|
|
const char *buffer,
|
|
|
|
const jsmntok_t *obj UNNEEDED,
|
|
|
|
const jsmntok_t *params)
|
2017-12-15 11:17:54 +01:00
|
|
|
{
|
2018-12-17 04:52:05 +01:00
|
|
|
struct lightningd *ld = cmd->ld;
|
|
|
|
|
2018-09-14 16:51:04 +02:00
|
|
|
if (!param(cmd, buffer, params, NULL))
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_param_failed();
|
2018-09-14 16:51:04 +02:00
|
|
|
|
2018-01-14 22:58:22 +01:00
|
|
|
if (!getenv("LIGHTNINGD_DEV_MEMLEAK")) {
|
2018-12-16 05:52:06 +01:00
|
|
|
return command_fail(cmd, LIGHTNINGD,
|
|
|
|
"Leak detection needs $LIGHTNINGD_DEV_MEMLEAK");
|
2018-01-14 22:58:22 +01:00
|
|
|
}
|
2017-12-17 04:17:12 +01:00
|
|
|
|
2018-12-17 04:52:05 +01:00
|
|
|
/* Start by asking connectd, which is always async. */
|
|
|
|
subd_req(ld->connectd, ld->connectd,
|
2020-08-25 04:16:22 +02:00
|
|
|
take(towire_connectd_dev_memleak(NULL)),
|
2018-12-17 04:52:05 +01:00
|
|
|
-1, 0, connect_dev_memleak_done, cmd);
|
2018-12-16 05:52:06 +01:00
|
|
|
|
2018-12-17 04:52:05 +01:00
|
|
|
return command_still_pending(cmd);
|
2017-12-15 11:17:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static const struct json_command dev_memleak_command = {
|
|
|
|
"dev-memleak",
|
2019-05-22 16:08:16 +02:00
|
|
|
"developer",
|
2017-12-15 11:17:54 +01:00
|
|
|
json_memleak,
|
2018-01-22 09:55:07 +01:00
|
|
|
"Show unreferenced memory objects"
|
2017-12-15 11:17:54 +01:00
|
|
|
};
|
|
|
|
AUTODATA(json_command, &dev_memleak_command);
|
2017-12-15 11:16:54 +01:00
|
|
|
#endif /* DEVELOPER */
|