#include #include #include #include /* BOLT#07: * * The `short_channel_id` is the unique description of the funding * transaction. It is constructed as follows: * 1. the most significant 3 bytes: indicating the block height * 2. the next 3 bytes: indicating the transaction index within the block * 3. the least significant 2 bytes: indicating the output index that pays to the channel. * * The standard human readable format for `short_channel_id` is created * by printing the above components, in the order: block height, * transaction index, and output index. Each component is printed as a * decimal number, and separated from each other by the small letter * `x`. For example, a `short_channel_id` might be written as * `539268x845x1`, indicating a channel on the output 1 of the * transaction at index 845 of the block at height 539268. */ bool mk_short_channel_id(struct short_channel_id *scid, u64 blocknum, u64 txnum, u64 outnum) { if ((blocknum & 0xFFFFFF) != blocknum || (txnum & 0xFFFFFF) != txnum || (outnum & 0xFFFF) != outnum) return false; scid->u64 = (((u64)blocknum & 0xFFFFFF) << 40 | ((u64)txnum & 0xFFFFFF) << 16 | (outnum & 0xFFFF)); return true; } bool short_channel_id_from_str(const char *str, size_t strlen, struct short_channel_id *dst) { u32 blocknum, txnum; u16 outnum; int matches; char buf[strlen + 1]; memcpy(buf, str, strlen); buf[strlen] = 0; #ifdef COMPAT_V062 /* Pre-adelaide format vs. post-adelaide format */ if (strchr(buf, ':')) matches = sscanf(buf, "%u:%u:%hu", &blocknum, &txnum, &outnum); else matches = sscanf(buf, "%ux%ux%hu", &blocknum, &txnum, &outnum); #else matches = sscanf(buf, "%ux%ux%hu", &blocknum, &txnum, &outnum); #endif return matches == 3 && mk_short_channel_id(dst, blocknum, txnum, outnum); } char *short_channel_id_to_str(const tal_t *ctx, const struct short_channel_id *scid) { return tal_fmt(ctx, "%dx%dx%d", short_channel_id_blocknum(scid), short_channel_id_txnum(scid), short_channel_id_outnum(scid)); } bool short_channel_id_dir_from_str(const char *str, size_t strlen, struct short_channel_id_dir *scidd) { const char *slash = memchr(str, '/', strlen); if (!slash || slash + 2 != str + strlen) return false; if (!short_channel_id_from_str(str, slash - str, &scidd->scid)) return false; if (slash[1] == '0') scidd->dir = 0; else if (slash[1] == '1') scidd->dir = 1; else return false; return true; } char *short_channel_id_dir_to_str(const tal_t *ctx, const struct short_channel_id_dir *scidd) { char *str, *scidstr = short_channel_id_to_str(NULL, &scidd->scid); str = tal_fmt(ctx, "%s/%u", scidstr, scidd->dir); tal_free(scidstr); return str; }