2020-08-11 08:24:23 +02:00
|
|
|
#ifndef LIGHTNING_COMMON_GOSSMAP_H
|
|
|
|
#define LIGHTNING_COMMON_GOSSMAP_H
|
|
|
|
#include "config.h"
|
|
|
|
#include <bitcoin/short_channel_id.h>
|
2021-02-02 01:35:23 +01:00
|
|
|
#include <ccan/take/take.h>
|
2020-08-11 08:24:23 +02:00
|
|
|
#include <ccan/typesafe_cb/typesafe_cb.h>
|
|
|
|
#include <common/amount.h>
|
2021-02-01 03:59:18 +01:00
|
|
|
#include <common/fp16.h>
|
2020-08-11 08:24:23 +02:00
|
|
|
|
|
|
|
struct node_id;
|
2024-07-17 11:05:11 +02:00
|
|
|
struct sciddir_or_pubkey;
|
2020-08-11 08:24:23 +02:00
|
|
|
|
|
|
|
struct gossmap_node {
|
|
|
|
/* Offset in memory map for node_announce, or 0. */
|
2024-10-08 06:57:54 +02:00
|
|
|
u64 nann_off;
|
2020-08-11 08:24:23 +02:00
|
|
|
u32 num_chans;
|
|
|
|
u32 *chan_idxs;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct gossmap_chan {
|
2024-10-08 06:57:54 +02:00
|
|
|
u64 cann_off;
|
|
|
|
/* FIXME: Technically redundant */
|
|
|
|
u64 plus_scid_off;
|
2021-05-22 07:00:22 +02:00
|
|
|
/* Offsets of cupdates (0 if missing). Logically inside half_chan,
|
|
|
|
* but that would add padding. */
|
2024-10-08 06:57:54 +02:00
|
|
|
u64 cupdate_off[2];
|
2020-08-11 08:24:23 +02:00
|
|
|
/* two nodes we connect (lesser idx first) */
|
|
|
|
struct half_chan {
|
|
|
|
/* Top bit indicates it's enabled */
|
|
|
|
u32 enabled: 1;
|
|
|
|
u32 nodeidx : 31;
|
|
|
|
fp16_t htlc_min, htlc_max;
|
|
|
|
|
|
|
|
/* millisatoshi. */
|
|
|
|
u64 base_fee : 24;
|
|
|
|
/* millionths */
|
|
|
|
u64 proportional_fee : 20;
|
|
|
|
/* Delay for HTLC in blocks. */
|
|
|
|
u64 delay : 20;
|
|
|
|
} half[2];
|
|
|
|
};
|
|
|
|
|
2021-08-20 06:12:27 +02:00
|
|
|
/* If num_channel_updates_rejected is not NULL, indicates how many channels we
|
|
|
|
* marked inactive because their values were too high to be represented. */
|
|
|
|
struct gossmap *gossmap_load(const tal_t *ctx, const char *filename,
|
|
|
|
size_t *num_channel_updates_rejected);
|
2020-08-11 08:24:23 +02:00
|
|
|
|
2024-01-31 05:20:33 +01:00
|
|
|
/* Version which uses existing fd */
|
2024-01-31 05:21:33 +01:00
|
|
|
#define gossmap_load_fd(ctx, fd, cupdate_fail, unknown_record, cbarg) \
|
|
|
|
gossmap_load_fd_((ctx), (fd), \
|
|
|
|
typesafe_cb_preargs(void, void *, (cupdate_fail), (cbarg), \
|
|
|
|
struct gossmap *, \
|
|
|
|
const struct short_channel_id_dir *, \
|
|
|
|
u16 cltv_expiry_delta, \
|
|
|
|
u32 fee_base_msat, \
|
|
|
|
u32 fee_proportional_millionths), \
|
|
|
|
typesafe_cb_preargs(bool, void *, (unknown_record), (cbarg), \
|
|
|
|
struct gossmap *, \
|
|
|
|
int type, \
|
2024-10-08 06:57:54 +02:00
|
|
|
u64 off, \
|
2024-01-31 05:21:33 +01:00
|
|
|
size_t msglen), \
|
|
|
|
(cbarg))
|
|
|
|
|
|
|
|
struct gossmap *gossmap_load_fd_(const tal_t *ctx, int fd,
|
|
|
|
void (*cupdate_fail)(struct gossmap *map,
|
|
|
|
const struct short_channel_id_dir *scidd,
|
|
|
|
u16 cltv_expiry_delta,
|
|
|
|
u32 fee_base_msat,
|
|
|
|
u32 fee_proportional_millionths,
|
|
|
|
void *cb_arg),
|
|
|
|
bool (*unknown_record)(struct gossmap *map,
|
|
|
|
int type,
|
2024-10-08 06:57:54 +02:00
|
|
|
u64 off,
|
2024-01-31 05:21:33 +01:00
|
|
|
size_t msglen,
|
|
|
|
void *cb_arg),
|
|
|
|
void *cb_arg);
|
|
|
|
|
2024-01-31 05:20:33 +01:00
|
|
|
|
2020-08-11 08:24:23 +02:00
|
|
|
/* Call this before using to ensure it's up-to-date. Returns true if something
|
|
|
|
* was updated. Note: this can scramble node and chan indexes! */
|
2021-08-20 06:12:27 +02:00
|
|
|
bool gossmap_refresh(struct gossmap *map, size_t *num_channel_updates_rejected);
|
2020-08-11 08:24:23 +02:00
|
|
|
|
2024-01-31 05:21:33 +01:00
|
|
|
/* Call this if you have set unknown_cb, and thus this can fail! */
|
|
|
|
bool gossmap_refresh_mayfail(struct gossmap *map, bool *updated);
|
|
|
|
|
2021-02-02 06:05:05 +01:00
|
|
|
/* Local modifications. */
|
|
|
|
struct gossmap_localmods *gossmap_localmods_new(const tal_t *ctx);
|
|
|
|
|
|
|
|
/* Create a local-only channel; if this conflicts with a real channel when added,
|
|
|
|
* that will be used instead.
|
|
|
|
* Returns false (and does nothing) if scid was already in localmods.
|
|
|
|
*/
|
|
|
|
bool gossmap_local_addchan(struct gossmap_localmods *localmods,
|
2021-02-02 01:35:23 +01:00
|
|
|
const struct node_id *n1,
|
|
|
|
const struct node_id *n2,
|
2024-03-20 02:59:51 +01:00
|
|
|
struct short_channel_id scid,
|
2024-10-04 01:36:53 +02:00
|
|
|
struct amount_msat capacity,
|
2021-02-02 01:35:23 +01:00
|
|
|
const u8 *features)
|
2024-03-20 02:59:51 +01:00
|
|
|
NON_NULL_ARGS(1,2,3);
|
2021-02-02 01:35:23 +01:00
|
|
|
|
2021-02-02 06:05:05 +01:00
|
|
|
/* Create a local-only channel_update: can apply to lcoal-only or
|
|
|
|
* normal channels. Returns false if amounts don't fit in our
|
2024-10-04 01:34:53 +02:00
|
|
|
* internal representation (implies channel unusable anyway). Any
|
|
|
|
* NULL arguments mean "leave as is". */
|
2021-02-02 06:05:05 +01:00
|
|
|
bool gossmap_local_updatechan(struct gossmap_localmods *localmods,
|
2024-10-04 01:34:53 +02:00
|
|
|
const struct short_channel_id_dir *scidd,
|
|
|
|
const bool *enabled,
|
|
|
|
const struct amount_msat *htlc_min,
|
|
|
|
const struct amount_msat *htlc_max,
|
|
|
|
const struct amount_msat *base_fee,
|
|
|
|
const u32 *proportional_fee,
|
|
|
|
const u16 *delay);
|
|
|
|
|
|
|
|
/* Convenience version which sets everything (older API) */
|
|
|
|
bool gossmap_local_setchan(struct gossmap_localmods *localmods,
|
|
|
|
struct short_channel_id scid,
|
|
|
|
struct amount_msat htlc_min,
|
|
|
|
struct amount_msat htlc_max,
|
|
|
|
struct amount_msat base_fee,
|
|
|
|
u32 proportional_fee,
|
|
|
|
u16 delay,
|
|
|
|
bool enabled,
|
|
|
|
int dir);
|
2021-02-02 01:35:23 +01:00
|
|
|
|
2021-02-02 06:05:05 +01:00
|
|
|
/* Apply localmods to this map */
|
|
|
|
void gossmap_apply_localmods(struct gossmap *map,
|
|
|
|
struct gossmap_localmods *localmods);
|
|
|
|
|
|
|
|
/* Remove localmods from this map */
|
|
|
|
void gossmap_remove_localmods(struct gossmap *map,
|
|
|
|
const struct gossmap_localmods *localmods);
|
2021-02-02 01:35:23 +01:00
|
|
|
|
2024-01-31 05:18:33 +01:00
|
|
|
/* Is this channel a localmod? */
|
|
|
|
bool gossmap_chan_is_localmod(const struct gossmap *map,
|
|
|
|
const struct gossmap_chan *c);
|
|
|
|
|
2024-02-11 11:30:41 +01:00
|
|
|
/* Is this channel dying? */
|
|
|
|
bool gossmap_chan_is_dying(const struct gossmap *map,
|
|
|
|
const struct gossmap_chan *c);
|
|
|
|
|
2020-08-11 08:24:23 +02:00
|
|
|
/* Each channel has a unique (low) index. */
|
|
|
|
u32 gossmap_node_idx(const struct gossmap *map, const struct gossmap_node *node);
|
|
|
|
u32 gossmap_chan_idx(const struct gossmap *map, const struct gossmap_chan *chan);
|
|
|
|
|
2022-09-18 02:22:50 +02:00
|
|
|
struct gossmap_node *gossmap_node_byidx(const struct gossmap *map, u32 idx);
|
|
|
|
struct gossmap_chan *gossmap_chan_byidx(const struct gossmap *map, u32 idx);
|
|
|
|
|
2020-08-11 08:24:23 +02:00
|
|
|
/* Every node_idx/chan_idx will be < these.
|
|
|
|
* These values can change across calls to gossmap_check. */
|
|
|
|
u32 gossmap_max_node_idx(const struct gossmap *map);
|
|
|
|
u32 gossmap_max_chan_idx(const struct gossmap *map);
|
|
|
|
|
|
|
|
/* Find node with this node_id */
|
|
|
|
struct gossmap_node *gossmap_find_node(const struct gossmap *map,
|
|
|
|
const struct node_id *id);
|
|
|
|
/* Find chan with this short_channel_id */
|
|
|
|
struct gossmap_chan *gossmap_find_chan(const struct gossmap *map,
|
|
|
|
const struct short_channel_id *scid);
|
|
|
|
|
|
|
|
/* Get the short_channel_id of this chan */
|
|
|
|
struct short_channel_id gossmap_chan_scid(const struct gossmap *map,
|
|
|
|
const struct gossmap_chan *c);
|
|
|
|
|
|
|
|
/* Given a struct node, get the node_id */
|
|
|
|
void gossmap_node_get_id(const struct gossmap *map,
|
|
|
|
const struct gossmap_node *node,
|
|
|
|
struct node_id *id);
|
|
|
|
|
|
|
|
/* Do we have any values for this halfchannel ? */
|
|
|
|
static inline bool gossmap_chan_set(const struct gossmap_chan *chan, int dir)
|
|
|
|
{
|
2021-05-22 07:00:22 +02:00
|
|
|
return chan->cupdate_off[dir] != 0;
|
2020-08-11 08:24:23 +02:00
|
|
|
}
|
|
|
|
|
2024-10-04 01:36:53 +02:00
|
|
|
/* Return capacity (in msat). */
|
|
|
|
struct amount_msat gossmap_chan_get_capacity(const struct gossmap *map,
|
|
|
|
const struct gossmap_chan *c);
|
2021-05-22 07:00:22 +02:00
|
|
|
|
2024-01-31 05:18:33 +01:00
|
|
|
/* Get the announcement msg which created this chan (NULL for localmods) */
|
2020-08-11 08:24:23 +02:00
|
|
|
u8 *gossmap_chan_get_announce(const tal_t *ctx,
|
|
|
|
const struct gossmap *map,
|
|
|
|
const struct gossmap_chan *c);
|
|
|
|
|
2024-01-31 05:20:33 +01:00
|
|
|
/* Do we have a node_announcement for this onde ? */
|
|
|
|
static inline bool gossmap_node_announced(const struct gossmap_node *node)
|
|
|
|
{
|
|
|
|
return node->nann_off != 0;
|
|
|
|
}
|
|
|
|
|
2020-08-11 08:24:23 +02:00
|
|
|
/* Get the announcement msg (if any) for this node. */
|
|
|
|
u8 *gossmap_node_get_announce(const tal_t *ctx,
|
|
|
|
const struct gossmap *map,
|
|
|
|
const struct gossmap_node *n);
|
|
|
|
|
2022-11-09 03:31:59 +01:00
|
|
|
/* Return the channel feature bit (odd or even), or -1 if neither. */
|
2020-10-27 01:16:02 +01:00
|
|
|
int gossmap_chan_get_feature(const struct gossmap *map,
|
2020-10-20 05:58:06 +02:00
|
|
|
const struct gossmap_chan *c,
|
|
|
|
int fbit);
|
|
|
|
|
2022-11-09 03:31:59 +01:00
|
|
|
/* Return the channel feature bitmap */
|
2021-05-22 07:00:22 +02:00
|
|
|
u8 *gossmap_chan_get_features(const tal_t *ctx,
|
|
|
|
const struct gossmap *map,
|
|
|
|
const struct gossmap_chan *c);
|
|
|
|
|
2022-11-09 03:31:59 +01:00
|
|
|
/* Return the node feature bit (odd or even), or -1 if neither (or no announcement) */
|
2020-10-27 01:16:02 +01:00
|
|
|
int gossmap_node_get_feature(const struct gossmap *map,
|
2020-10-20 05:58:06 +02:00
|
|
|
const struct gossmap_node *n,
|
|
|
|
int fbit);
|
|
|
|
|
2022-11-09 03:31:59 +01:00
|
|
|
/* Return the node feature bitmap: NULL if no announcement. */
|
|
|
|
u8 *gossmap_node_get_features(const tal_t *ctx,
|
|
|
|
const struct gossmap *map,
|
|
|
|
const struct gossmap_node *n);
|
|
|
|
|
2024-01-31 05:20:33 +01:00
|
|
|
/* Return the channel_update (or NULL if !gossmap_chan_set) */
|
|
|
|
u8 *gossmap_chan_get_update(const tal_t *ctx,
|
|
|
|
const struct gossmap *map,
|
|
|
|
const struct gossmap_chan *chan,
|
|
|
|
int dir);
|
|
|
|
|
2024-07-17 11:05:11 +02:00
|
|
|
/* Return true if we can map this sciddir_or_pubkey to a pubkey. */
|
|
|
|
bool gossmap_scidd_pubkey(struct gossmap *gossmap,
|
|
|
|
struct sciddir_or_pubkey *sciddpk);
|
|
|
|
|
2024-10-11 13:00:38 +02:00
|
|
|
/* Returns details from channel_update (must be gossmap_chan_set)! */
|
2021-05-22 07:00:22 +02:00
|
|
|
void gossmap_chan_get_update_details(const struct gossmap *map,
|
|
|
|
const struct gossmap_chan *chan,
|
|
|
|
int dir,
|
|
|
|
u32 *timestamp,
|
|
|
|
u8 *message_flags,
|
|
|
|
u8 *channel_flags,
|
2024-10-11 13:00:38 +02:00
|
|
|
u16 *cltv_expiry_delta,
|
2021-05-22 07:00:22 +02:00
|
|
|
u32 *fee_base_msat,
|
|
|
|
u32 *fee_proportional_millionths,
|
|
|
|
struct amount_msat *htlc_minimum_msat,
|
|
|
|
struct amount_msat *htlc_maximum_msat);
|
|
|
|
|
2020-08-11 08:24:23 +02:00
|
|
|
/* Given a struct node, get the nth channel, and tell us if we're half[0/1].
|
|
|
|
* n must be less than node->num_chans */
|
|
|
|
struct gossmap_chan *gossmap_nth_chan(const struct gossmap *map,
|
|
|
|
const struct gossmap_node *node,
|
|
|
|
u32 n,
|
|
|
|
int *which_half);
|
|
|
|
|
|
|
|
/* Given a struct chan, get the nth node, where n is 0 or 1. */
|
|
|
|
struct gossmap_node *gossmap_nth_node(const struct gossmap *map,
|
|
|
|
const struct gossmap_chan *chan,
|
|
|
|
int n);
|
|
|
|
|
|
|
|
/* Can this channel send this amount? */
|
2024-01-31 05:18:33 +01:00
|
|
|
bool gossmap_chan_has_capacity(const struct gossmap_chan *chan,
|
|
|
|
int direction,
|
|
|
|
struct amount_msat amount);
|
2020-08-11 08:24:23 +02:00
|
|
|
|
2024-08-07 03:49:53 +02:00
|
|
|
/* Convenience routines to get htlc min/max as amount_msat */
|
|
|
|
static inline struct amount_msat
|
|
|
|
gossmap_chan_htlc_max(const struct gossmap_chan *chan, const int dir)
|
|
|
|
{
|
|
|
|
return amount_msat(fp16_to_u64(chan->half[dir].htlc_max));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline struct amount_msat
|
|
|
|
gossmap_chan_htlc_min(const struct gossmap_chan *chan, const int dir)
|
|
|
|
{
|
|
|
|
return amount_msat(fp16_to_u64(chan->half[dir].htlc_min));
|
|
|
|
}
|
|
|
|
|
2020-08-11 08:24:23 +02:00
|
|
|
/* Remove a channel from the map (warning! realloc can move gossmap_chan
|
|
|
|
* and gossmap_node ptrs!) */
|
|
|
|
void gossmap_remove_chan(struct gossmap *map, struct gossmap_chan *chan);
|
|
|
|
|
|
|
|
/* Remove node (by removing all its channels) */
|
|
|
|
void gossmap_remove_node(struct gossmap *map, struct gossmap_node *node);
|
|
|
|
|
|
|
|
/* Unsorted iterate through (do not add/remove channels or nodes!) */
|
|
|
|
size_t gossmap_num_nodes(const struct gossmap *map);
|
|
|
|
|
|
|
|
struct gossmap_node *gossmap_first_node(const struct gossmap *map);
|
|
|
|
struct gossmap_node *gossmap_next_node(const struct gossmap *map,
|
|
|
|
const struct gossmap_node *prev);
|
|
|
|
|
|
|
|
/* Unsorted iterate through (do not add/remove channels or nodes!) */
|
|
|
|
size_t gossmap_num_chans(const struct gossmap *map);
|
|
|
|
|
|
|
|
struct gossmap_chan *gossmap_first_chan(const struct gossmap *map);
|
|
|
|
struct gossmap_chan *gossmap_next_chan(const struct gossmap *map,
|
|
|
|
struct gossmap_chan *prev);
|
2024-05-23 03:25:59 +02:00
|
|
|
|
2024-07-09 15:12:29 +02:00
|
|
|
/* For iterating the gossmap: returns iterator at start. */
|
|
|
|
struct gossmap_iter *gossmap_iter_new(const tal_t *ctx,
|
|
|
|
const struct gossmap *map);
|
|
|
|
/* Copy an existing iterator (same offset) */
|
|
|
|
struct gossmap_iter *gossmap_iter_dup(const tal_t *ctx,
|
|
|
|
const struct gossmap_iter *iter);
|
|
|
|
|
|
|
|
/* Get next message, and optional timestamp */
|
|
|
|
const void *gossmap_stream_next(const tal_t *ctx,
|
|
|
|
const struct gossmap *map,
|
|
|
|
struct gossmap_iter *iter,
|
|
|
|
u32 *timestamp);
|
|
|
|
/* For fast-forwarding to the given timestamp */
|
|
|
|
void gossmap_iter_fast_forward(const struct gossmap *map,
|
|
|
|
struct gossmap_iter *iter,
|
|
|
|
u64 timestamp);
|
|
|
|
/* Moves iterator to the end. */
|
|
|
|
void gossmap_iter_end(const struct gossmap *map, struct gossmap_iter *iter);
|
|
|
|
|
2024-05-23 03:25:59 +02:00
|
|
|
/* For debugging: returns length read, and total known length of file */
|
2024-10-08 06:57:54 +02:00
|
|
|
u64 gossmap_lengths(const struct gossmap *map, u64 *total);
|
2024-07-09 15:12:29 +02:00
|
|
|
|
|
|
|
/* Debugging: connectd wants to enumerate fds */
|
|
|
|
int gossmap_fd(const struct gossmap *map);
|
2020-08-11 08:24:23 +02:00
|
|
|
#endif /* LIGHTNING_COMMON_GOSSMAP_H */
|