2019-07-18 07:20:29 +02:00
|
|
|
#ifndef LIGHTNING_WIRE_TLVSTREAM_H
|
|
|
|
#define LIGHTNING_WIRE_TLVSTREAM_H
|
|
|
|
#include "config.h"
|
2020-05-19 19:14:49 +02:00
|
|
|
#include <bitcoin/short_channel_id.h>
|
2019-07-18 07:20:29 +02:00
|
|
|
#include <ccan/short_types/short_types.h>
|
|
|
|
#include <ccan/tal/tal.h>
|
|
|
|
|
|
|
|
struct tlv_record_type {
|
|
|
|
u64 type;
|
|
|
|
/* If this type is present return marshalled value. Otherwise
|
|
|
|
* returns NULL. */
|
|
|
|
u8 *(*towire)(const tal_t *ctx, const void *record);
|
2019-07-21 05:04:59 +02:00
|
|
|
/* Must call fromwire_fail() if it can't parse. */
|
2019-07-18 07:20:29 +02:00
|
|
|
void (*fromwire)(const u8 **cursor, size_t *max, void *record);
|
|
|
|
};
|
|
|
|
|
2019-11-18 15:53:27 +01:00
|
|
|
/* A single TLV field, consisting of the data and its associated metadata. */
|
|
|
|
struct tlv_field {
|
|
|
|
/* If this is a type that is known to c-lightning we have a pointer to
|
|
|
|
* the metadata. */
|
|
|
|
const struct tlv_record_type *meta;
|
|
|
|
|
|
|
|
/* In any case we'll have the numeric type, even if we don't have a
|
|
|
|
* name that we can call it. */
|
|
|
|
u64 numtype;
|
|
|
|
size_t length;
|
|
|
|
u8 *value;
|
|
|
|
};
|
|
|
|
|
2020-04-08 20:51:27 +02:00
|
|
|
/* Given any tlvstream serialize the raw fields (untyped ones). */
|
2021-06-17 18:21:27 +02:00
|
|
|
void towire_tlvstream_raw(u8 **pptr, struct tlv_field *fields);
|
2020-04-08 20:51:27 +02:00
|
|
|
|
2020-12-04 01:00:22 +01:00
|
|
|
/* Given a tlv record with manually-set fields, populate ->fields */
|
|
|
|
#define tlv_make_fields(tlv, type) \
|
|
|
|
tlv_make_fields_(tlvs_##type, TLVS_ARRAY_SIZE_##type, (tlv))
|
|
|
|
|
|
|
|
struct tlv_field *tlv_make_fields_(const struct tlv_record_type *types,
|
|
|
|
size_t num_types,
|
|
|
|
const void *record);
|
|
|
|
|
2022-03-23 03:44:36 +01:00
|
|
|
/**
|
|
|
|
* fromwire_tlv: generic TLV decode engine
|
|
|
|
* @cursor: cursor to update (set to NULL if we fail).
|
|
|
|
* @max: max len to update (always set to 0 if we succeed).
|
|
|
|
* @types / @num_types: table of known tlv types
|
|
|
|
* @record: the tlv to hand to @type-specific decode
|
|
|
|
* @fields: the fields array to populate
|
2022-03-23 03:44:38 +01:00
|
|
|
* @extra_types: tal_arr of unknown types to allow, or NULL, or FROMWIRE_TLV_ANY_TYPE.
|
2022-03-23 03:44:36 +01:00
|
|
|
* @err_off: NULL, or set to offset in tlv stream which failed.
|
|
|
|
* @err_type: NULL, or set to tlv type which failed (or 0 if malformed)
|
|
|
|
*/
|
2020-12-03 10:34:03 +01:00
|
|
|
bool fromwire_tlv(const u8 **cursor, size_t *max,
|
|
|
|
const struct tlv_record_type *types, size_t num_types,
|
2022-03-23 03:44:36 +01:00
|
|
|
void *record, struct tlv_field **fields,
|
|
|
|
const u64 *extra_types, size_t *err_off, u64 *err_type);
|
2020-12-03 10:34:03 +01:00
|
|
|
void towire_tlv(u8 **pptr,
|
|
|
|
const struct tlv_record_type *types, size_t num_types,
|
|
|
|
const void *record);
|
2020-05-19 19:14:49 +02:00
|
|
|
|
2022-03-22 09:48:13 +01:00
|
|
|
/* Get the offset of this field: returns size of msg if not found (or
|
|
|
|
* tlv malformed) */
|
|
|
|
size_t tlv_field_offset(const u8 *tlvstream, size_t tlvlen, u64 fieldtype);
|
|
|
|
|
2022-03-23 03:44:38 +01:00
|
|
|
/* Constant for fromwire_tlv to allow absolutely any unknown type. */
|
|
|
|
extern const u64 *FROMWIRE_TLV_ANY_TYPE;
|
|
|
|
|
2020-05-19 19:14:49 +02:00
|
|
|
/* Generic primitive setters for tlvstreams. */
|
2021-09-07 06:06:06 +02:00
|
|
|
void tlvstream_set_raw(struct tlv_field **stream, u64 type, void *value TAKES, size_t valuelen);
|
2020-05-19 19:14:49 +02:00
|
|
|
void tlvstream_set_short_channel_id(struct tlv_field **stream, u64 type,
|
|
|
|
struct short_channel_id *value);
|
|
|
|
void tlvstream_set_tu64(struct tlv_field **stream, u64 type, u64 value);
|
|
|
|
void tlvstream_set_tu32(struct tlv_field **stream, u64 type, u32 value);
|
|
|
|
|
2019-07-18 07:20:29 +02:00
|
|
|
#endif /* LIGHTNING_WIRE_TLVSTREAM_H */
|