proto: Added handling for nested packets

This commit is contained in:
Christian Decker 2016-12-12 14:55:15 +01:00 committed by Rusty Russell
parent 1c4d4f8c91
commit db481d881a
4 changed files with 56 additions and 1 deletions

View File

@ -213,6 +213,19 @@ void queue_pkt_revocation(struct peer *peer,
queue_pkt(peer, PKT__PKT_UPDATE_REVOCATION, u); queue_pkt(peer, PKT__PKT_UPDATE_REVOCATION, u);
} }
/* Send a serialized nested packet. */
void queue_pkt_nested(struct peer *peer,
int type,
const u8 *nested_pkt)
{
NestedPkt *pb = tal(peer, NestedPkt);
nested_pkt__init(pb);
pb->type = type;
pb->inner_pkt.len = tal_count(nested_pkt);
pb->inner_pkt.data = tal_dup_arr(pb, u8, nested_pkt, pb->inner_pkt.len, 0);
queue_pkt(peer, PKT__PKT_NESTED, pb);
}
Pkt *pkt_err(struct peer *peer, const char *msg, ...) Pkt *pkt_err(struct peer *peer, const char *msg, ...)
{ {
Error *e = tal(peer, Error); Error *e = tal(peer, Error);

View File

@ -24,6 +24,7 @@ void queue_pkt_revocation(struct peer *peer,
const struct sha256 *next_hash); const struct sha256 *next_hash);
void queue_pkt_close_shutdown(struct peer *peer); void queue_pkt_close_shutdown(struct peer *peer);
void queue_pkt_close_signature(struct peer *peer); void queue_pkt_close_signature(struct peer *peer);
void queue_pkt_nested(struct peer *peer, int type, const u8 *nested_pkt);
Pkt *pkt_err(struct peer *peer, const char *msg, ...); Pkt *pkt_err(struct peer *peer, const char *msg, ...);
Pkt *pkt_init(struct peer *peer, u64 ack); Pkt *pkt_init(struct peer *peer, u64 ack);

View File

@ -1854,6 +1854,39 @@ static bool peer_start_shutdown(struct peer *peer)
return db_commit_transaction(peer) == NULL; return db_commit_transaction(peer) == NULL;
} }
/* Shim to handle the new packet format until we complete the
* switch. Handing the protobuf in anyway to fall back on protobuf
* based error handling. */
static bool nested_pkt_in(struct peer *peer, const u32 type,
const u8 *innerpkt, size_t innerpktlen,
const Pkt *pkt)
{
switch (type) {
case WIRE_CHANNEL_ANNOUNCEMENT:
handle_channel_announcement(
peer, fromwire_channel_announcement(peer, innerpkt, &innerpktlen));
break;
case WIRE_CHANNEL_UPDATE:
handle_channel_update(
peer, fromwire_channel_update(peer, innerpkt, &innerpktlen));
break;
case WIRE_NODE_ANNOUNCEMENT:
handle_node_announcement(
peer, fromwire_node_announcement(peer, innerpkt, &innerpktlen));
break;
default:
/* BOLT01: Unknown even typed packets MUST kill the
connection, unknown odd-typed packets MAY be ignored. */
if (type % 2 == 0){
return peer_received_unexpected_pkt(peer, pkt, __func__);
} else {
log_debug(peer->log, "Ignoring odd typed (%d) unknown packet.", type);
return true;
}
}
return true;
}
/* This is the io loop while we're in normal mode. */ /* This is the io loop while we're in normal mode. */
static bool normal_pkt_in(struct peer *peer, const Pkt *pkt) static bool normal_pkt_in(struct peer *peer, const Pkt *pkt)
{ {
@ -2164,11 +2197,18 @@ static void clear_output_queue(struct peer *peer)
static struct io_plan *pkt_in(struct io_conn *conn, struct peer *peer) static struct io_plan *pkt_in(struct io_conn *conn, struct peer *peer)
{ {
bool keep_going; bool keep_going = true;
/* We ignore packets if they tell us to, or we're closing already */ /* We ignore packets if they tell us to, or we're closing already */
if (peer->fake_close || !state_can_io(peer->state)) if (peer->fake_close || !state_can_io(peer->state))
keep_going = true; keep_going = true;
/* Sidestep the state machine for nested packets */
else if (peer->inpkt->pkt_case == PKT__PKT_NESTED)
keep_going = nested_pkt_in(peer, peer->inpkt->nested->type,
peer->inpkt->nested->inner_pkt.data,
peer->inpkt->nested->inner_pkt.len,
peer->inpkt);
else if (state_is_normal(peer->state)) else if (state_is_normal(peer->state))
keep_going = normal_pkt_in(peer, peer->inpkt); keep_going = normal_pkt_in(peer, peer->inpkt);
else if (state_is_shutdown(peer->state)) else if (state_is_shutdown(peer->state))

View File

@ -14,6 +14,7 @@
#include "netaddr.h" #include "netaddr.h"
#include "protobuf_convert.h" #include "protobuf_convert.h"
#include "state.h" #include "state.h"
#include "wire/gen_wire.h"
#include <ccan/crypto/sha256/sha256.h> #include <ccan/crypto/sha256/sha256.h>
#include <ccan/crypto/shachain/shachain.h> #include <ccan/crypto/shachain/shachain.h>
#include <ccan/list/list.h> #include <ccan/list/list.h>