@ -293,20 +293,23 @@ fromwire_tlv_templ = """bool frowire_{name}({ctx}const void *p{args})
printwire_header_templ = """ void printwire_ {name} (const char *fieldname, const u8 *cursor);
printwire_header_templ = """ void printwire_ {name} (const char *fieldname, const u8 *cursor);
"""
"""
printwire_impl_templ = """ void printwire_ {name} (const char *fieldname, const u8 *cursor)
{ {
printwire_toplevel_tmpl = """ \t size_t plen = tal_count(cursor);
\tsize_t plen = tal_count ( cursor ) ;
\tif ( fromwire_u16 ( & cursor , & plen ) != { enum . name } ) { {
\tif ( fromwire_u16 ( & cursor , & plen ) != { enum . name } ) { {
\t \tprintf ( " WRONG TYPE?! \\ n " ) ;
\t \tprintf ( " WRONG TYPE?! \\ n " ) ;
\t \treturn ;
\t \treturn ;
\t } }
\t } } """
{ subcalls }
printwire_impl_templ = """ {is_internal} void printwire_ {name} (const char *fieldname, const u8 * {cursor_ptr} cursor {tlv_args} )
{ {
{ toplevel_msg_setup } { subcalls } { lencheck }
} }
"""
printwire_lencheck = """
\tif ( plen != 0 )
\tif ( plen != 0 )
\t \tprintf ( " EXTRA: %s \\ n " , tal_hexstr ( NULL , cursor , plen ) ) ;
\t \tprintf ( " EXTRA: %s \\ n " , tal_hexstr ( NULL , cursor , plen ) ) ;
} }
"""
"""
@ -707,43 +710,50 @@ class Message(object):
subcalls = str ( subcalls ) ,
subcalls = str ( subcalls ) ,
)
)
def add_truncate_check ( self , subcalls ) :
def add_truncate_check ( self , subcalls , ref ) :
# Report if truncated, otherwise print.
# Report if truncated, otherwise print.
subcalls . append ( ' if (!cursor) { \n '
call = ' if (! {} cursor) {{ \n printf( " **TRUNCATED** \\ n " ); \n return; \n }} ' . format ( ref )
' printf( " **TRUNCATED** \\ n " ); \n '
subcalls . append ( call )
' return; \n '
' } ' )
def print_printwire_array ( self , subcalls , basetype , f , num_elems ) :
def print_printwire_array ( self , subcalls , basetype , f , num_elems , ref ) :
truncate_check_ref = ' ' if ref else ' * '
if f . has_array_helper ( ) :
if f . has_array_helper ( ) :
subcalls . append ( ' printwire_ {} _array(tal_fmt(NULL, " %s . {} " , fieldname), &cursor, & plen, {} ); '
subcalls . append ( ' printwire_ {} _array(tal_fmt(NULL, " %s . {} " , fieldname), {} cursor, {} plen, {} ); '
. format ( basetype , f . name , num_elems ) )
. format ( basetype , f . name , ref , ref , num_elems ) )
else :
else :
subcalls . append ( ' printf( " [ " ); ' )
subcalls . append ( ' printf( " [ " ); ' )
subcalls . append ( ' for (size_t i = 0; i < {} ; i++) {{ '
subcalls . append ( ' for (size_t i = 0; i < {} ; i++) {{ '
. format ( num_elems ) )
. format ( num_elems ) )
subcalls . append ( ' {} v; ' . format ( f . fieldtype . name ) )
subcalls . append ( ' {} v; ' . format ( f . fieldtype . name ) )
if f . fieldtype . is_assignable ( ) :
if f . fieldtype . is_assignable ( ) :
subcalls . append ( ' v = fromwire_ {} (&cursor, plen); '
subcalls . append ( ' v = fromwire_ {} ( {} cursor, {} plen);'
. format ( f . fieldtype . name , basetype ) )
. format ( f . fieldtype . name , basetype , ref , ref ) )
else :
else :
# We don't handle this yet!
# We don't handle this yet!
assert ( basetype not in varlen_structs )
assert ( basetype not in varlen_structs )
subcalls . append ( ' fromwire_ {} (&cursor, & plen, &v); '
subcalls . append ( ' fromwire_ {} ( {} cursor, {} plen, &v);'
. format ( basetype ) )
. format ( basetype , ref , ref ) )
self . add_truncate_check ( subcalls )
self . add_truncate_check ( subcalls , truncate_check_ref )
subcalls . append ( ' printwire_ {} (tal_fmt(NULL, " %s . {} " , fieldname), &v); '
subcalls . append ( ' printwire_ {} (tal_fmt(NULL, " %s . {} " , fieldname), &v); '
. format ( basetype , f . name ) )
. format ( basetype , f . name ) )
subcalls . append ( ' } ' )
subcalls . append ( ' } ' )
subcalls . append ( ' printf( " ] " ); ' )
subcalls . append ( ' printf( " ] " ); ' )
def print_printwire ( self , is_header ) :
def print_printwire ( self , is_header , is_tlv = False ) :
template = printwire_header_templ if is_header else printwire_impl_templ
template = printwire_header_templ if is_header else printwire_impl_templ
fields = [ ' \t {} {} ; \n ' . format ( f . fieldtype . name , f . name ) for f in self . fields if f . is_len_var ]
fields = [ ' \t {} {} ; \n ' . format ( f . fieldtype . name , f . name ) for f in self . fields if f . is_len_var ]
tlv_args = ' ' if not is_tlv else ' , size_t *plen '
ref = ' & ' if not is_tlv else ' '
truncate_check_ref = ' ' if not is_tlv else ' * '
toplevel_msg_setup = ' '
if not is_tlv :
toplevel_msg_setup = printwire_toplevel_tmpl . format ( enum = self . enum )
subcalls = CCode ( )
subcalls = CCode ( )
for f in self . fields :
for f in self . fields :
basetype = f . fieldtype . base ( )
basetype = f . fieldtype . base ( )
@ -752,50 +762,59 @@ class Message(object):
subcalls . append ( ' /* {} */ ' . format ( c ) )
subcalls . append ( ' /* {} */ ' . format ( c ) )
if f . is_len_var :
if f . is_len_var :
subcalls . append ( ' {} {} = fromwire_ {} (&cursor, &plen); '
if f . fieldtype . is_var_int ( ) :
. format ( f . fieldtype . name , f . name , basetype ) )
subcalls . append ( ' {} {} = fromwire_ {} ( {} cursor, {} plen); '
self . add_truncate_check ( subcalls )
. format ( basetype , f . name , ' var_int ' , ref , ref ) )
else :
subcalls . append ( ' {} {} = fromwire_ {} ( {} cursor, {} plen); '
. format ( f . fieldtype . name , f . name , basetype , ref , ref ) )
self . add_truncate_check ( subcalls , truncate_check_ref )
continue
continue
subcalls . append ( ' printf( " {} = " ); ' . format ( f . name ) )
subcalls . append ( ' printf( " {} = " ); ' . format ( f . name ) )
if f . is_padding ( ) :
if f . is_padding ( ) :
subcalls . append ( ' printwire_pad(tal_fmt(NULL, " %s . {} " , fieldname), &cursor, & plen, {} ); '
subcalls . append ( ' printwire_pad(tal_fmt(NULL, " %s . {} " , fieldname), {} cursor, {} plen, {} ); '
. format ( f . name , f . num_elems ) )
. format ( f . name , ref , ref , f . num_elems ) )
self . add_truncate_check ( subcalls )
self . add_truncate_check ( subcalls , truncate_check_ref )
elif f . is_array ( ) :
elif f . is_array ( ) :
self . print_printwire_array ( subcalls , basetype , f , f . num_elems )
self . print_printwire_array ( subcalls , basetype , f , f . num_elems , ref )
self . add_truncate_check ( subcalls )
self . add_truncate_check ( subcalls , truncate_check_ref )
elif f . is_variable_size ( ) :
elif f . is_variable_size ( ) :
self . print_printwire_array ( subcalls , basetype , f , f . lenvar )
self . print_printwire_array ( subcalls , basetype , f , f . lenvar , ref )
self . add_truncate_check ( subcalls )
self . add_truncate_check ( subcalls , truncate_check_ref )
else :
else :
if f . optional :
if f . optional :
subcalls . append ( " if (fromwire_bool(&cursor, & plen)) { " )
subcalls . append ( " if (fromwire_bool( {} cursor, {} plen)) { " . format ( ref , ref ) )
if f . is_assignable ( ) :
if f . is_assignable ( ) :
subcalls . append ( ' {} {} = fromwire_ {} (&cursor, & plen); '
subcalls . append ( ' {} {} = fromwire_ {} ( {} cursor, {} plen);'
. format ( f . fieldtype . name , f . name , basetype ) )
. format ( f . fieldtype . name , f . name , basetype , ref , ref ) )
else :
else :
# Don't handle these yet.
# Don't handle these yet.
assert ( basetype not in varlen_structs )
assert ( basetype not in varlen_structs )
subcalls . append ( ' {} {} ; ' .
subcalls . append ( ' {} {} ; ' .
format ( f . fieldtype . name , f . name ) )
format ( f . fieldtype . name , f . name ) )
subcalls . append ( ' fromwire_ {} (&cursor, & plen, & {} ); '
subcalls . append ( ' fromwire_ {} ( {} cursor, {} plen, &{} ); '
. format ( basetype , f . name ) )
. format ( basetype , ref , ref , f . name ) )
self . add_truncate_check ( subcalls )
self . add_truncate_check ( subcalls , truncate_check_ref )
subcalls . append ( ' printwire_ {} (tal_fmt(NULL, " %s . {} " , fieldname), & {} ); '
subcalls . append ( ' printwire_ {} (tal_fmt(NULL, " %s . {} " , fieldname), & {} ); '
. format ( basetype , f . name , f . name ) )
. format ( basetype , f . name , f . name ) )
if f . optional :
if f . optional :
subcalls . append ( " } else { " )
subcalls . append ( " } else { " )
self . add_truncate_check ( subcalls )
self . add_truncate_check ( subcalls , truncate_check_ref )
subcalls . append ( " } " )
subcalls . append ( " } " )
len_check = ' ' if is_tlv else printwire_lencheck
return template . format (
return template . format (
tlv_args = tlv_args ,
name = self . name ,
name = self . name ,
fields = ' ' . join ( fields ) ,
fields = ' ' . join ( fields ) ,
enum = self . enum ,
toplevel_msg_setup = toplevel_msg_setup ,
subcalls = str ( subcalls )
subcalls = str ( subcalls ) ,
lencheck = len_check ,
cursor_ptr = ( ' ' if not is_tlv else ' * ' ) ,
is_internal = ( ' ' if not is_tlv else ' static ' )
)
)
def print_struct ( self ) :
def print_struct ( self ) :
@ -906,6 +925,28 @@ case_tmpl = """\t\tcase {tlv_msg_enum}:
\t \t \tbreak ;
\t \t \tbreak ;
"""
"""
print_tlv_template = """ static void printwire_ {tlv_name} (const char *fieldname, const u8 *cursor)
{ {
\tu8 msg_type ;
\tu64 msg_size ;
\tsize_t plen = tal_count ( cursor ) ;
\twhile ( cursor ) { {
\t \tmsg_type = fromwire_u8 ( & cursor , & plen ) ;
\t \tmsg_size = fromwire_var_int ( & cursor , & plen ) ;
\t \tif ( ! cursor )
\t \t \tbreak ;
\t \tswitch ( ( enum { tlv_name } _type ) msg_type ) { {
\t \t \t { printcases }
\t \t \tdefault :
\t \t \t \tprintf ( " WARNING:No message matching type %d \\ n " , msg_type ) ;
\t \t } }
\t } }
\tif ( plen != 0 )
\t \tprintf ( " EXTRA: %s \\ n " , tal_hexstr ( NULL , cursor , plen ) ) ;
} }
"""
def build_tlv_fromwires ( tlv_fields ) :
def build_tlv_fromwires ( tlv_fields ) :
fromwires = [ ]
fromwires = [ ]
@ -970,6 +1011,32 @@ def find_message(messages, name):
return None
return None
def print_tlv_printwire ( tlv_name , messages ) :
printcases = ' '
for m in messages :
printcases + = ' case {enum.name} : printf( " {enum.name} (size % " PRIu64 " ): \\ n " , msg_size); printwire_ {name} ( " {name} " , &cursor, &plen); break; ' . format (
enum = m . enum , name = m . name , tlv_name = tlv_name )
return print_tlv_template . format (
tlv_name = tlv_name ,
printcases = printcases )
def print_tlv_printwires ( tlv_fields ) :
decls = [ ]
switches = ' '
for name , messages in tlv_fields . items ( ) :
# Print each of the message parsers
decls + = [ m . print_printwire ( options . header , is_tlv = True ) for m in messages ]
# Print the TLV body parser
decls . append ( print_tlv_printwire ( name , messages ) )
# Print the 'master' print_tlv_messages cases
switches + = tlv_switch_template . format ( tlv_name = name )
decls . append ( print_master_tlv_template . format ( tlv_switches = switches ) )
return decls
def find_message_with_option ( messages , optional_messages , name , option ) :
def find_message_with_option ( messages , optional_messages , name , option ) :
fullname = name + " _ " + option . replace ( ' - ' , ' _ ' )
fullname = name + " _ " + option . replace ( ' - ' , ' _ ' )
@ -1152,6 +1219,12 @@ impl_template = """/* This file was generated by generate-wire.py */
{ func_decls }
{ func_decls }
"""
"""
print_tlv_message_printwire_empty = """ void print {enumname} _tlv_message(const char *tlv_name, const u8 *msg)
{ {
\tprintf ( " ~~ No TLV definition found for %s ~~ \\ n " , tlv_name ) ;
} }
"""
print_header_template = """ /* This file was generated by generate-wire.py */
print_header_template = """ /* This file was generated by generate-wire.py */
/ * Do not modify this file ! Modify the _csv file it was generated from . * /
/ * Do not modify this file ! Modify the _csv file it was generated from . * /
#ifndef LIGHTNING_{idem}
#ifndef LIGHTNING_{idem}
@ -1162,6 +1235,8 @@ print_header_template = """/* This file was generated by generate-wire.py */
void print { enumname } _message ( const u8 * msg ) ;
void print { enumname } _message ( const u8 * msg ) ;
void print { enumname } _tlv_message ( const char * tlv_name , const u8 * msg ) ;
{ func_decls }
{ func_decls }
#endif /* LIGHTNING_{idem} */
#endif /* LIGHTNING_{idem} */
"""
"""
@ -1172,6 +1247,7 @@ print_template = """/* This file was generated by generate-wire.py */
#include <ccan/mem/mem.h>
#include <ccan/mem/mem.h>
#include <ccan/tal/str/str.h>
#include <ccan/tal/str/str.h>
#include <common/utils.h>
#include <common/utils.h>
#include <inttypes.h>
#include <stdio.h>
#include <stdio.h>
void print { enumname } _message ( const u8 * msg )
void print { enumname } _message ( const u8 * msg )
@ -1186,6 +1262,21 @@ void print{enumname}_message(const u8 *msg)
{ func_decls }
{ func_decls }
"""
"""
print_master_tlv_template = """
void print_tlv_message ( const char * tlv_name , const u8 * msg )
{ {
\t { tlv_switches }
\tprintf ( " ERR: Unknown TLV message type: %s \n " , tlv_name ) ;
} }
"""
tlv_switch_template = """
\tif ( strcmp ( tlv_name , " {tlv_name} " ) == 0 ) { {
\t \tprintwire_ { tlv_name } ( " {tlv_name} " , msg ) ;
\t \treturn ;
\t } }
"""
idem = re . sub ( r ' [^A-Z]+ ' , ' _ ' , options . headerfilename . upper ( ) )
idem = re . sub ( r ' [^A-Z]+ ' , ' _ ' , options . headerfilename . upper ( ) )
if options . printwire :
if options . printwire :
if options . header :
if options . header :
@ -1207,7 +1298,12 @@ includes = '\n'.join(includes)
printcases = [ ' case {enum.name} : printf( " {enum.name} : \\ n " ); printwire_ {name} ( " {name} " , msg); return; ' . format ( enum = m . enum , name = m . name ) for m in toplevel_messages ]
printcases = [ ' case {enum.name} : printf( " {enum.name} : \\ n " ); printwire_ {name} ( " {name} " , msg); return; ' . format ( enum = m . enum , name = m . name ) for m in toplevel_messages ]
if options . printwire :
if options . printwire :
decls = [ m . print_printwire ( options . header ) for m in messages + messages_with_option ]
decls = [ m . print_printwire ( options . header ) for m in toplevel_messages + messages_with_option ]
if not options . header :
if len ( tlv_fields ) :
decls + = print_tlv_printwires ( tlv_fields )
else :
decls + = [ print_tlv_message_printwire_empty . format ( enumname = options . enumname ) ]
else :
else :
towire_decls = [ ]
towire_decls = [ ]
fromwire_decls = [ ]
fromwire_decls = [ ]