2023-01-30 18:33:54 -08:00
#!/usr/bin/env python3
2023-03-15 16:03:50 -07:00
# SPDX-License-Identifier: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause)
2023-01-20 09:50:36 -08:00
import argparse
2023-01-25 16:02:35 -08:00
import collections
2023-01-20 09:50:36 -08:00
import os
import yaml
2023-03-07 16:39:22 -08:00
from lib import SpecFamily , SpecAttrSet , SpecAttr , SpecOperation , SpecEnumSet , SpecEnumEntry
2023-01-30 18:33:44 -08:00
2023-01-20 09:50:36 -08:00
def c_upper ( name ) :
return name . upper ( ) . replace ( ' - ' , ' _ ' )
def c_lower ( name ) :
return name . lower ( ) . replace ( ' - ' , ' _ ' )
class BaseNlLib :
def get_family_id ( self ) :
return ' ys->family_id '
def parse_cb_run ( self , cb , data , is_dump = False , indent = 1 ) :
ind = ' \n \t \t ' + ' \t ' * indent + ' '
if is_dump :
return f " mnl_cb_run2(ys->rx_buf, len, 0, 0, { cb } , { data } , { ind } ynl_cb_array, NLMSG_MIN_TYPE) "
else :
return f " mnl_cb_run2(ys->rx_buf, len, ys->seq, ys->portid, { ind } { cb } , { data } , { ind } " + \
" ynl_cb_array, NLMSG_MIN_TYPE) "
2023-01-30 18:33:44 -08:00
class Type ( SpecAttr ) :
def __init__ ( self , family , attr_set , attr , value ) :
super ( ) . __init__ ( family , attr_set , attr , value )
2023-01-20 09:50:36 -08:00
self . attr = attr
2023-01-30 18:33:44 -08:00
self . attr_set = attr_set
2023-01-20 09:50:36 -08:00
self . type = attr [ ' type ' ]
self . checks = attr . get ( ' checks ' , { } )
if ' len ' in attr :
self . len = attr [ ' len ' ]
if ' nested-attributes ' in attr :
self . nested_attrs = attr [ ' nested-attributes ' ]
if self . nested_attrs == family . name :
self . nested_render_name = f " { family . name } "
else :
self . nested_render_name = f " { family . name } _ { c_lower ( self . nested_attrs ) } "
self . c_name = c_lower ( self . name )
if self . c_name in _C_KW :
self . c_name + = ' _ '
2023-01-30 18:33:44 -08:00
# Added by resolve():
self . enum_name = None
delattr ( self , " enum_name " )
2023-01-20 09:50:36 -08:00
2023-01-30 18:33:44 -08:00
def resolve ( self ) :
self . enum_name = f " { self . attr_set . name_prefix } { self . name } "
self . enum_name = c_upper ( self . enum_name )
2023-01-20 09:50:36 -08:00
def is_multi_val ( self ) :
return None
def is_scalar ( self ) :
return self . type in { ' u8 ' , ' u16 ' , ' u32 ' , ' u64 ' , ' s32 ' , ' s64 ' }
def presence_type ( self ) :
return ' bit '
def presence_member ( self , space , type_filter ) :
if self . presence_type ( ) != type_filter :
return
if self . presence_type ( ) == ' bit ' :
pfx = ' __ ' if space == ' user ' else ' '
return f " { pfx } u32 { self . c_name } :1; "
if self . presence_type ( ) == ' len ' :
pfx = ' __ ' if space == ' user ' else ' '
return f " { pfx } u32 { self . c_name } _len; "
def _complex_member_type ( self , ri ) :
return None
def free_needs_iter ( self ) :
return False
def free ( self , ri , var , ref ) :
if self . is_multi_val ( ) or self . presence_type ( ) == ' len ' :
ri . cw . p ( f ' free( { var } -> { ref } { self . c_name } ); ' )
def arg_member ( self , ri ) :
member = self . _complex_member_type ( ri )
if member :
2023-06-06 12:43:00 -07:00
arg = [ member + ' * ' + self . c_name ]
if self . presence_type ( ) == ' count ' :
arg + = [ ' unsigned int n_ ' + self . c_name ]
return arg
2023-01-20 09:50:36 -08:00
raise Exception ( f " Struct member not implemented for class type { self . type } " )
def struct_member ( self , ri ) :
if self . is_multi_val ( ) :
ri . cw . p ( f " unsigned int n_ { self . c_name } ; " )
member = self . _complex_member_type ( ri )
if member :
ptr = ' * ' if self . is_multi_val ( ) else ' '
ri . cw . p ( f " { member } { ptr } { self . c_name } ; " )
return
members = self . arg_member ( ri )
for one in members :
ri . cw . p ( one + ' ; ' )
def _attr_policy ( self , policy ) :
return ' { .type = ' + policy + ' , } '
def attr_policy ( self , cw ) :
policy = c_upper ( ' nla- ' + self . attr [ ' type ' ] )
spec = self . _attr_policy ( policy )
cw . p ( f " \t [ { self . enum_name } ] = { spec } , " )
def _attr_typol ( self ) :
raise Exception ( f " Type policy not implemented for class type { self . type } " )
def attr_typol ( self , cw ) :
typol = self . _attr_typol ( )
cw . p ( f ' [ { self . enum_name } ] = { " { " } .name = " { self . name } " , { typol } { " } " } , ' )
def _attr_put_line ( self , ri , var , line ) :
if self . presence_type ( ) == ' bit ' :
ri . cw . p ( f " if ( { var } ->_present. { self . c_name } ) " )
elif self . presence_type ( ) == ' len ' :
ri . cw . p ( f " if ( { var } ->_present. { self . c_name } _len) " )
ri . cw . p ( f " { line } ; " )
def _attr_put_simple ( self , ri , var , put_type ) :
line = f " mnl_attr_put_ { put_type } (nlh, { self . enum_name } , { var } -> { self . c_name } ) "
self . _attr_put_line ( ri , var , line )
def attr_put ( self , ri , var ) :
raise Exception ( f " Put not implemented for class type { self . type } " )
def _attr_get ( self , ri , var ) :
raise Exception ( f " Attr get not implemented for class type { self . type } " )
def attr_get ( self , ri , var , first ) :
lines , init_lines , local_vars = self . _attr_get ( ri , var )
if type ( lines ) is str :
lines = [ lines ]
if type ( init_lines ) is str :
init_lines = [ init_lines ]
kw = ' if ' if first else ' else if '
ri . cw . block_start ( line = f " { kw } (mnl_attr_get_type(attr) == { self . enum_name } ) " )
if local_vars :
for local in local_vars :
ri . cw . p ( local )
ri . cw . nl ( )
if not self . is_multi_val ( ) :
ri . cw . p ( " if (ynl_attr_validate(yarg, attr)) " )
ri . cw . p ( " return MNL_CB_ERROR; " )
if self . presence_type ( ) == ' bit ' :
ri . cw . p ( f " { var } ->_present. { self . c_name } = 1; " )
if init_lines :
ri . cw . nl ( )
for line in init_lines :
ri . cw . p ( line )
for line in lines :
ri . cw . p ( line )
ri . cw . block_end ( )
2023-06-01 19:35:40 -07:00
return True
2023-01-20 09:50:36 -08:00
def _setter_lines ( self , ri , member , presence ) :
raise Exception ( f " Setter not implemented for class type { self . type } " )
def setter ( self , ri , space , direction , deref = False , ref = None ) :
ref = ( ref if ref else [ ] ) + [ self . c_name ]
var = " req "
member = f " { var } -> { ' . ' . join ( ref ) } "
code = [ ]
presence = ' '
for i in range ( 0 , len ( ref ) ) :
presence = f " { var } -> { ' . ' . join ( ref [ : i ] + [ ' ' ] ) } _present. { ref [ i ] } "
if self . presence_type ( ) == ' bit ' :
code . append ( presence + ' = 1; ' )
code + = self . _setter_lines ( ri , member , presence )
2023-06-06 12:43:00 -07:00
func_name = f " { op_prefix ( ri , direction , deref = deref ) } _set_ { ' _ ' . join ( ref ) } "
free = bool ( [ x for x in code if ' free( ' in x ] )
alloc = bool ( [ x for x in code if ' alloc( ' in x ] )
if free and not alloc :
func_name = ' __ ' + func_name
ri . cw . write_func ( ' static inline void ' , func_name , body = code ,
2023-01-20 09:50:36 -08:00
args = [ f ' { type_name ( ri , direction , deref = deref ) } * { var } ' ] + self . arg_member ( ri ) )
class TypeUnused ( Type ) :
def presence_type ( self ) :
return ' '
2023-06-01 19:35:40 -07:00
def arg_member ( self , ri ) :
return [ ]
def _attr_get ( self , ri , var ) :
return [ ' return MNL_CB_ERROR; ' ] , None , None
2023-01-20 09:50:36 -08:00
def _attr_typol ( self ) :
return ' .type = YNL_PT_REJECT, '
def attr_policy ( self , cw ) :
pass
class TypePad ( Type ) :
def presence_type ( self ) :
return ' '
2023-06-01 19:35:40 -07:00
def arg_member ( self , ri ) :
return [ ]
2023-01-20 09:50:36 -08:00
def _attr_typol ( self ) :
2023-06-01 19:35:40 -07:00
return ' .type = YNL_PT_IGNORE, '
def attr_get ( self , ri , var , first ) :
pass
2023-01-20 09:50:36 -08:00
def attr_policy ( self , cw ) :
pass
class TypeScalar ( Type ) :
2023-01-30 18:33:44 -08:00
def __init__ ( self , family , attr_set , attr , value ) :
super ( ) . __init__ ( family , attr_set , attr , value )
self . byte_order_comment = ' '
if ' byte-order ' in attr :
self . byte_order_comment = f " /* { attr [ ' byte-order ' ] } */ "
# Added by resolve():
self . is_bitfield = None
delattr ( self , " is_bitfield " )
self . type_name = None
delattr ( self , " type_name " )
def resolve ( self ) :
self . resolve_up ( super ( ) )
2023-01-20 09:50:36 -08:00
if ' enum-as-flags ' in self . attr and self . attr [ ' enum-as-flags ' ] :
self . is_bitfield = True
2023-01-30 18:33:44 -08:00
elif ' enum ' in self . attr :
self . is_bitfield = self . family . consts [ self . attr [ ' enum ' ] ] [ ' type ' ] == ' flags '
else :
self . is_bitfield = False
2023-01-20 09:50:36 -08:00
if ' enum ' in self . attr and not self . is_bitfield :
2023-01-30 18:33:44 -08:00
self . type_name = f " enum { self . family . name } _ { c_lower ( self . attr [ ' enum ' ] ) } "
2023-01-20 09:50:36 -08:00
else :
self . type_name = ' __ ' + self . type
def _mnl_type ( self ) :
t = self . type
# mnl does not have a helper for signed types
if t [ 0 ] == ' s ' :
t = ' u ' + t [ 1 : ]
return t
def _attr_policy ( self , policy ) :
if ' flags-mask ' in self . checks or self . is_bitfield :
if self . is_bitfield :
2023-03-24 12:03:56 -07:00
enum = self . family . consts [ self . attr [ ' enum ' ] ]
mask = enum . get_mask ( as_flags = True )
2023-01-20 09:50:36 -08:00
else :
flags = self . family . consts [ self . checks [ ' flags-mask ' ] ]
flag_cnt = len ( flags [ ' entries ' ] )
mask = ( 1 << flag_cnt ) - 1
return f " NLA_POLICY_MASK( { policy } , 0x { mask : x } ) "
elif ' min ' in self . checks :
return f " NLA_POLICY_MIN( { policy } , { self . checks [ ' min ' ] } ) "
elif ' enum ' in self . attr :
enum = self . family . consts [ self . attr [ ' enum ' ] ]
cnt = len ( enum [ ' entries ' ] )
return f " NLA_POLICY_MAX( { policy } , { cnt - 1 } ) "
return super ( ) . _attr_policy ( policy )
def _attr_typol ( self ) :
return f ' .type = YNL_PT_U { self . type [ 1 : ] } , '
def arg_member ( self , ri ) :
return [ f ' { self . type_name } { self . c_name } { self . byte_order_comment } ' ]
def attr_put ( self , ri , var ) :
self . _attr_put_simple ( ri , var , self . _mnl_type ( ) )
def _attr_get ( self , ri , var ) :
return f " { var } -> { self . c_name } = mnl_attr_get_ { self . _mnl_type ( ) } (attr); " , None , None
def _setter_lines ( self , ri , member , presence ) :
return [ f " { member } = { self . c_name } ; " ]
class TypeFlag ( Type ) :
def arg_member ( self , ri ) :
return [ ]
def _attr_typol ( self ) :
return ' .type = YNL_PT_FLAG, '
def attr_put ( self , ri , var ) :
self . _attr_put_line ( ri , var , f " mnl_attr_put(nlh, { self . enum_name } , 0, NULL) " )
def _attr_get ( self , ri , var ) :
return [ ] , None , None
def _setter_lines ( self , ri , member , presence ) :
return [ ]
class TypeString ( Type ) :
def arg_member ( self , ri ) :
return [ f " const char * { self . c_name } " ]
def presence_type ( self ) :
return ' len '
def struct_member ( self , ri ) :
ri . cw . p ( f " char * { self . c_name } ; " )
def _attr_typol ( self ) :
return f ' .type = YNL_PT_NUL_STR, '
def _attr_policy ( self , policy ) :
mem = ' { .type = ' + policy
if ' max-len ' in self . checks :
mem + = ' , .len = ' + str ( self . checks [ ' max-len ' ] )
mem + = ' , } '
return mem
def attr_policy ( self , cw ) :
if self . checks . get ( ' unterminated-ok ' , False ) :
policy = ' NLA_STRING '
else :
policy = ' NLA_NUL_STRING '
spec = self . _attr_policy ( policy )
cw . p ( f " \t [ { self . enum_name } ] = { spec } , " )
def attr_put ( self , ri , var ) :
self . _attr_put_simple ( ri , var , ' strz ' )
def _attr_get ( self , ri , var ) :
len_mem = var + ' ->_present. ' + self . c_name + ' _len '
return [ f " { len_mem } = len; " ,
f " { var } -> { self . c_name } = malloc(len + 1); " ,
f " memcpy( { var } -> { self . c_name } , mnl_attr_get_str(attr), len); " ,
f " { var } -> { self . c_name } [len] = 0; " ] , \
[ ' len = strnlen(mnl_attr_get_str(attr), mnl_attr_get_payload_len(attr)); ' ] , \
[ ' unsigned int len; ' ]
def _setter_lines ( self , ri , member , presence ) :
return [ f " free( { member } ); " ,
f " { presence } _len = strlen( { self . c_name } ); " ,
f " { member } = malloc( { presence } _len + 1); " ,
f ' memcpy( { member } , { self . c_name } , { presence } _len); ' ,
f ' { member } [ { presence } _len] = 0; ' ]
class TypeBinary ( Type ) :
def arg_member ( self , ri ) :
return [ f " const void * { self . c_name } " , ' size_t len ' ]
def presence_type ( self ) :
return ' len '
def struct_member ( self , ri ) :
ri . cw . p ( f " void * { self . c_name } ; " )
def _attr_typol ( self ) :
return f ' .type = YNL_PT_BINARY, '
def _attr_policy ( self , policy ) :
mem = ' { '
if len ( self . checks ) == 1 and ' min-len ' in self . checks :
mem + = ' .len = ' + str ( self . checks [ ' min-len ' ] )
elif len ( self . checks ) == 0 :
mem + = ' .type = NLA_BINARY '
else :
raise Exception ( ' One or more of binary type checks not implemented, yet ' )
mem + = ' , } '
return mem
def attr_put ( self , ri , var ) :
self . _attr_put_line ( ri , var , f " mnl_attr_put(nlh, { self . enum_name } , " +
f " { var } ->_present. { self . c_name } _len, { var } -> { self . c_name } ) " )
def _attr_get ( self , ri , var ) :
len_mem = var + ' ->_present. ' + self . c_name + ' _len '
return [ f " { len_mem } = len; " ,
f " { var } -> { self . c_name } = malloc(len); " ,
f " memcpy( { var } -> { self . c_name } , mnl_attr_get_payload(attr), len); " ] , \
[ ' len = mnl_attr_get_payload_len(attr); ' ] , \
[ ' unsigned int len; ' ]
def _setter_lines ( self , ri , member , presence ) :
return [ f " free( { member } ); " ,
f " { member } = malloc( { presence } _len); " ,
f ' memcpy( { member } , { self . c_name } , { presence } _len); ' ]
class TypeNest ( Type ) :
def _complex_member_type ( self , ri ) :
return f " struct { self . nested_render_name } "
def free ( self , ri , var , ref ) :
ri . cw . p ( f ' { self . nested_render_name } _free(& { var } -> { ref } { self . c_name } ); ' )
def _attr_typol ( self ) :
return f ' .type = YNL_PT_NEST, .nest = & { self . nested_render_name } _nest, '
def _attr_policy ( self , policy ) :
return ' NLA_POLICY_NESTED( ' + self . nested_render_name + ' _nl_policy) '
def attr_put ( self , ri , var ) :
self . _attr_put_line ( ri , var , f " { self . nested_render_name } _put(nlh, " +
f " { self . enum_name } , & { var } -> { self . c_name } ) " )
def _attr_get ( self , ri , var ) :
2023-06-01 19:35:43 -07:00
get_lines = [ f " if ( { self . nested_render_name } _parse(&parg, attr)) " ,
" return MNL_CB_ERROR; " ]
2023-01-20 09:50:36 -08:00
init_lines = [ f " parg.rsp_policy = & { self . nested_render_name } _nest; " ,
f " parg.data = & { var } -> { self . c_name } ; " ]
return get_lines , init_lines , None
def setter ( self , ri , space , direction , deref = False , ref = None ) :
ref = ( ref if ref else [ ] ) + [ self . c_name ]
for _ , attr in ri . family . pure_nested_structs [ self . nested_attrs ] . member_list ( ) :
attr . setter ( ri , self . nested_attrs , direction , deref = deref , ref = ref )
class TypeMultiAttr ( Type ) :
def is_multi_val ( self ) :
return True
def presence_type ( self ) :
return ' count '
2023-06-06 12:43:00 -07:00
def _mnl_type ( self ) :
t = self . type
# mnl does not have a helper for signed types
if t [ 0 ] == ' s ' :
t = ' u ' + t [ 1 : ]
return t
2023-01-20 09:50:36 -08:00
def _complex_member_type ( self , ri ) :
if ' type ' not in self . attr or self . attr [ ' type ' ] == ' nest ' :
return f " struct { self . nested_render_name } "
elif self . attr [ ' type ' ] in scalars :
scalar_pfx = ' __ ' if ri . ku_space == ' user ' else ' '
return scalar_pfx + self . attr [ ' type ' ]
else :
raise Exception ( f " Sub-type { self . attr [ ' type ' ] } not supported yet " )
def free_needs_iter ( self ) :
return ' type ' not in self . attr or self . attr [ ' type ' ] == ' nest '
def free ( self , ri , var , ref ) :
2023-06-06 12:43:00 -07:00
if self . attr [ ' type ' ] in scalars :
ri . cw . p ( f " free( { var } -> { ref } { self . c_name } ); " )
elif ' type ' not in self . attr or self . attr [ ' type ' ] == ' nest ' :
2023-01-20 09:50:36 -08:00
ri . cw . p ( f " for (i = 0; i < { var } -> { ref } n_ { self . c_name } ; i++) " )
ri . cw . p ( f ' { self . nested_render_name } _free(& { var } -> { ref } { self . c_name } [i]); ' )
2023-06-06 12:43:00 -07:00
ri . cw . p ( f " free( { var } -> { ref } { self . c_name } ); " )
else :
raise Exception ( f " Free of MultiAttr sub-type { self . attr [ ' type ' ] } not supported yet " )
2023-01-20 09:50:36 -08:00
def _attr_typol ( self ) :
if ' type ' not in self . attr or self . attr [ ' type ' ] == ' nest ' :
return f ' .type = YNL_PT_NEST, .nest = & { self . nested_render_name } _nest, '
elif self . attr [ ' type ' ] in scalars :
return f " .type = YNL_PT_U { self . attr [ ' type ' ] [ 1 : ] } , "
else :
raise Exception ( f " Sub-type { self . attr [ ' type ' ] } not supported yet " )
def _attr_get ( self , ri , var ) :
2023-06-06 12:43:01 -07:00
return f ' n_ { self . c_name } ++; ' , None , None
2023-01-20 09:50:36 -08:00
2023-06-06 12:43:00 -07:00
def attr_put ( self , ri , var ) :
if self . attr [ ' type ' ] in scalars :
put_type = self . _mnl_type ( )
ri . cw . p ( f " for (unsigned int i = 0; i < { var } ->n_ { self . c_name } ; i++) " )
ri . cw . p ( f " mnl_attr_put_ { put_type } (nlh, { self . enum_name } , { var } -> { self . c_name } [i]); " )
elif ' type ' not in self . attr or self . attr [ ' type ' ] == ' nest ' :
ri . cw . p ( f " for (unsigned int i = 0; i < { var } ->n_ { self . c_name } ; i++) " )
self . _attr_put_line ( ri , var , f " { self . nested_render_name } _put(nlh, " +
f " { self . enum_name } , & { var } -> { self . c_name } [i]) " )
else :
raise Exception ( f " Put of MultiAttr sub-type { self . attr [ ' type ' ] } not supported yet " )
def _setter_lines ( self , ri , member , presence ) :
# For multi-attr we have a count, not presence, hack up the presence
presence = presence [ : - ( len ( ' _present. ' ) + len ( self . c_name ) ) ] + " n_ " + self . c_name
return [ f " free( { member } ); " ,
f " { member } = { self . c_name } ; " ,
f " { presence } = n_ { self . c_name } ; " ]
2023-01-20 09:50:36 -08:00
class TypeArrayNest ( Type ) :
def is_multi_val ( self ) :
return True
def presence_type ( self ) :
return ' count '
def _complex_member_type ( self , ri ) :
if ' sub-type ' not in self . attr or self . attr [ ' sub-type ' ] == ' nest ' :
return f " struct { self . nested_render_name } "
elif self . attr [ ' sub-type ' ] in scalars :
scalar_pfx = ' __ ' if ri . ku_space == ' user ' else ' '
return scalar_pfx + self . attr [ ' sub-type ' ]
else :
raise Exception ( f " Sub-type { self . attr [ ' sub-type ' ] } not supported yet " )
def _attr_typol ( self ) :
return f ' .type = YNL_PT_NEST, .nest = & { self . nested_render_name } _nest, '
def _attr_get ( self , ri , var ) :
local_vars = [ ' const struct nlattr *attr2; ' ]
get_lines = [ f ' attr_ { self . c_name } = attr; ' ,
' mnl_attr_for_each_nested(attr2, attr) ' ,
f ' \t { var } ->n_ { self . c_name } ++; ' ]
return get_lines , None , local_vars
class TypeNestTypeValue ( Type ) :
def _complex_member_type ( self , ri ) :
return f " struct { self . nested_render_name } "
def _attr_typol ( self ) :
return f ' .type = YNL_PT_NEST, .nest = & { self . nested_render_name } _nest, '
def _attr_get ( self , ri , var ) :
prev = ' attr '
tv_args = ' '
get_lines = [ ]
local_vars = [ ]
init_lines = [ f " parg.rsp_policy = & { self . nested_render_name } _nest; " ,
f " parg.data = & { var } -> { self . c_name } ; " ]
if ' type-value ' in self . attr :
tv_names = [ c_lower ( x ) for x in self . attr [ " type-value " ] ]
local_vars + = [ f ' const struct nlattr *attr_ { " , *attr_ " . join ( tv_names ) } ; ' ]
local_vars + = [ f ' __u32 { " , " . join ( tv_names ) } ; ' ]
for level in self . attr [ " type-value " ] :
level = c_lower ( level )
get_lines + = [ f ' attr_ { level } = mnl_attr_get_payload( { prev } ); ' ]
get_lines + = [ f ' { level } = mnl_attr_get_type(attr_ { level } ); ' ]
prev = ' attr_ ' + level
tv_args = f " , { ' , ' . join ( tv_names ) } "
get_lines + = [ f " { self . nested_render_name } _parse(&parg, { prev } { tv_args } ); " ]
return get_lines , init_lines , local_vars
class Struct :
def __init__ ( self , family , space_name , type_list = None , inherited = None ) :
self . family = family
self . space_name = space_name
self . attr_set = family . attr_sets [ space_name ]
# Use list to catch comparisons with empty sets
self . _inherited = inherited if inherited is not None else [ ]
self . inherited = [ ]
self . nested = type_list is None
if family . name == c_lower ( space_name ) :
self . render_name = f " { family . name } "
else :
self . render_name = f " { family . name } _ { c_lower ( space_name ) } "
self . struct_name = ' struct ' + self . render_name
self . ptr_name = self . struct_name + ' * '
self . request = False
self . reply = False
self . attr_list = [ ]
self . attrs = dict ( )
if type_list :
for t in type_list :
self . attr_list . append ( ( t , self . attr_set [ t ] ) , )
else :
for t in self . attr_set :
self . attr_list . append ( ( t , self . attr_set [ t ] ) , )
max_val = 0
self . attr_max_val = None
for name , attr in self . attr_list :
2023-02-23 10:31:39 -08:00
if attr . value > = max_val :
2023-01-20 09:50:36 -08:00
max_val = attr . value
self . attr_max_val = attr
self . attrs [ name ] = attr
def __iter__ ( self ) :
yield from self . attrs
def __getitem__ ( self , key ) :
return self . attrs [ key ]
def member_list ( self ) :
return self . attr_list
def set_inherited ( self , new_inherited ) :
if self . _inherited != new_inherited :
raise Exception ( " Inheriting different members not supported " )
self . inherited = [ c_lower ( x ) for x in sorted ( self . _inherited ) ]
2023-03-07 16:39:22 -08:00
class EnumEntry ( SpecEnumEntry ) :
2023-01-20 09:50:36 -08:00
def __init__ ( self , enum_set , yaml , prev , value_start ) :
2023-03-07 16:39:22 -08:00
super ( ) . __init__ ( enum_set , yaml , prev , value_start )
if prev :
self . value_change = ( self . value != prev . value + 1 )
2023-01-20 09:50:36 -08:00
else :
self . value_change = ( self . value != 0 )
2023-01-25 16:02:33 -08:00
self . value_change = self . value_change or self . enum_set [ ' type ' ] == ' flags '
2023-03-07 16:39:22 -08:00
# Added by resolve:
self . c_name = None
delattr ( self , " c_name " )
2023-01-20 09:50:36 -08:00
2023-03-07 16:39:22 -08:00
def resolve ( self ) :
self . resolve_up ( super ( ) )
2023-01-25 16:02:33 -08:00
2023-03-07 16:39:22 -08:00
self . c_name = c_upper ( self . enum_set . value_pfx + self . name )
2023-01-25 16:02:33 -08:00
2023-01-20 09:50:36 -08:00
2023-03-07 16:39:22 -08:00
class EnumSet ( SpecEnumSet ) :
2023-01-20 09:50:36 -08:00
def __init__ ( self , family , yaml ) :
self . render_name = c_lower ( family . name + ' - ' + yaml [ ' name ' ] )
self . enum_name = ' enum ' + self . render_name
self . value_pfx = yaml . get ( ' name-prefix ' , f " { family . name } - { yaml [ ' name ' ] } - " )
2023-03-07 16:39:22 -08:00
super ( ) . __init__ ( family , yaml )
2023-01-20 09:50:36 -08:00
2023-03-07 16:39:22 -08:00
def new_entry ( self , entry , prev_entry , value_start ) :
return EnumEntry ( self , entry , prev_entry , value_start )
2023-01-20 09:50:36 -08:00
2023-01-30 18:33:44 -08:00
class AttrSet ( SpecAttrSet ) :
2023-01-20 09:50:36 -08:00
def __init__ ( self , family , yaml ) :
2023-01-30 18:33:44 -08:00
super ( ) . __init__ ( family , yaml )
2023-01-20 09:50:36 -08:00
2023-01-30 18:33:44 -08:00
if self . subset_of is None :
2023-01-20 09:50:36 -08:00
if ' name-prefix ' in yaml :
pfx = yaml [ ' name-prefix ' ]
elif self . name == family . name :
pfx = family . name + ' -a- '
else :
pfx = f " { family . name } -a- { self . name } - "
self . name_prefix = c_upper ( pfx )
self . max_name = c_upper ( self . yaml . get ( ' attr-max-name ' , f " { self . name_prefix } max " ) )
else :
self . name_prefix = family . attr_sets [ self . subset_of ] . name_prefix
self . max_name = family . attr_sets [ self . subset_of ] . max_name
2023-01-30 18:33:44 -08:00
# Added by resolve:
self . c_name = None
delattr ( self , " c_name " )
def resolve ( self ) :
2023-01-20 09:50:36 -08:00
self . c_name = c_lower ( self . name )
if self . c_name in _C_KW :
self . c_name + = ' _ '
2023-01-30 18:33:44 -08:00
if self . c_name == self . family . c_name :
2023-01-20 09:50:36 -08:00
self . c_name = ' '
2023-01-30 18:33:44 -08:00
def new_attr ( self , elem , value ) :
if ' multi-attr ' in elem and elem [ ' multi-attr ' ] :
return TypeMultiAttr ( self . family , self , elem , value )
elif elem [ ' type ' ] in scalars :
return TypeScalar ( self . family , self , elem , value )
elif elem [ ' type ' ] == ' unused ' :
return TypeUnused ( self . family , self , elem , value )
elif elem [ ' type ' ] == ' pad ' :
return TypePad ( self . family , self , elem , value )
elif elem [ ' type ' ] == ' flag ' :
return TypeFlag ( self . family , self , elem , value )
elif elem [ ' type ' ] == ' string ' :
return TypeString ( self . family , self , elem , value )
elif elem [ ' type ' ] == ' binary ' :
return TypeBinary ( self . family , self , elem , value )
elif elem [ ' type ' ] == ' nest ' :
return TypeNest ( self . family , self , elem , value )
elif elem [ ' type ' ] == ' array-nest ' :
return TypeArrayNest ( self . family , self , elem , value )
elif elem [ ' type ' ] == ' nest-type-value ' :
return TypeNestTypeValue ( self . family , self , elem , value )
else :
raise Exception ( f " No typed class for type { elem [ ' type ' ] } " )
2023-01-20 09:50:36 -08:00
2023-01-30 18:33:44 -08:00
class Operation ( SpecOperation ) :
def __init__ ( self , family , yaml , req_value , rsp_value ) :
super ( ) . __init__ ( family , yaml , req_value , rsp_value )
2023-01-20 09:50:36 -08:00
2023-01-30 18:33:44 -08:00
if req_value != rsp_value :
raise Exception ( " Directional messages not supported by codegen " )
2023-01-20 09:50:36 -08:00
self . render_name = family . name + ' _ ' + c_lower ( self . name )
self . dual_policy = ( ' do ' in yaml and ' request ' in yaml [ ' do ' ] ) and \
( ' dump ' in yaml and ' request ' in yaml [ ' dump ' ] )
2023-01-30 18:33:44 -08:00
# Added by resolve:
self . enum_name = None
delattr ( self , " enum_name " )
2023-01-20 09:50:36 -08:00
2023-01-30 18:33:44 -08:00
def resolve ( self ) :
self . resolve_up ( super ( ) )
if not self . is_async :
self . enum_name = self . family . op_prefix + c_upper ( self . name )
else :
self . enum_name = self . family . async_op_prefix + c_upper ( self . name )
2023-01-20 09:50:36 -08:00
def add_notification ( self , op ) :
if ' notify ' not in self . yaml :
self . yaml [ ' notify ' ] = dict ( )
self . yaml [ ' notify ' ] [ ' reply ' ] = self . yaml [ ' do ' ] [ ' reply ' ]
self . yaml [ ' notify ' ] [ ' cmds ' ] = [ ]
self . yaml [ ' notify ' ] [ ' cmds ' ] . append ( op )
2023-01-30 18:33:44 -08:00
class Family ( SpecFamily ) :
2023-01-20 09:50:36 -08:00
def __init__ ( self , file_name ) :
2023-01-30 18:33:44 -08:00
# Added by resolve:
self . c_name = None
delattr ( self , " c_name " )
self . op_prefix = None
delattr ( self , " op_prefix " )
self . async_op_prefix = None
delattr ( self , " async_op_prefix " )
self . mcgrps = None
delattr ( self , " mcgrps " )
self . consts = None
delattr ( self , " consts " )
self . hooks = None
delattr ( self , " hooks " )
super ( ) . __init__ ( file_name )
2023-01-20 09:50:36 -08:00
self . fam_key = c_upper ( self . yaml . get ( ' c-family-name ' , self . yaml [ " name " ] + ' _FAMILY_NAME ' ) )
self . ver_key = c_upper ( self . yaml . get ( ' c-version-name ' , self . yaml [ " name " ] + ' _FAMILY_VERSION ' ) )
if ' definitions ' not in self . yaml :
self . yaml [ ' definitions ' ] = [ ]
if ' uapi-header ' in self . yaml :
self . uapi_header = self . yaml [ ' uapi-header ' ]
else :
self . uapi_header = f " linux/ { self . name } .h "
2023-01-30 18:33:44 -08:00
def resolve ( self ) :
self . resolve_up ( super ( ) )
if self . yaml . get ( ' protocol ' , ' genetlink ' ) not in { ' genetlink ' , ' genetlink-c ' , ' genetlink-legacy ' } :
raise Exception ( " Codegen only supported for genetlink " )
self . c_name = c_lower ( self . name )
2023-01-20 09:50:36 -08:00
if ' name-prefix ' in self . yaml [ ' operations ' ] :
self . op_prefix = c_upper ( self . yaml [ ' operations ' ] [ ' name-prefix ' ] )
else :
self . op_prefix = c_upper ( self . yaml [ ' name ' ] + ' -cmd- ' )
if ' async-prefix ' in self . yaml [ ' operations ' ] :
self . async_op_prefix = c_upper ( self . yaml [ ' operations ' ] [ ' async-prefix ' ] )
else :
self . async_op_prefix = self . op_prefix
self . mcgrps = self . yaml . get ( ' mcast-groups ' , { ' list ' : [ ] } )
self . hooks = dict ( )
for when in [ ' pre ' , ' post ' ] :
self . hooks [ when ] = dict ( )
for op_mode in [ ' do ' , ' dump ' ] :
self . hooks [ when ] [ op_mode ] = dict ( )
self . hooks [ when ] [ op_mode ] [ ' set ' ] = set ( )
self . hooks [ when ] [ op_mode ] [ ' list ' ] = [ ]
# dict space-name -> 'request': set(attrs), 'reply': set(attrs)
self . root_sets = dict ( )
# dict space-name -> set('request', 'reply')
self . pure_nested_structs = dict ( )
self . all_notify = dict ( )
self . _mock_up_events ( )
self . _dictify ( )
self . _load_root_sets ( )
self . _load_nested_sets ( )
self . _load_all_notify ( )
self . _load_hooks ( )
self . kernel_policy = self . yaml . get ( ' kernel-policy ' , ' split ' )
if self . kernel_policy == ' global ' :
self . _load_global_policy ( )
2023-03-07 16:39:22 -08:00
def new_enum ( self , elem ) :
return EnumSet ( self , elem )
2023-01-30 18:33:44 -08:00
def new_attr_set ( self , elem ) :
return AttrSet ( self , elem )
2023-01-20 09:50:36 -08:00
2023-01-30 18:33:44 -08:00
def new_operation ( self , elem , req_value , rsp_value ) :
return Operation ( self , elem , req_value , rsp_value )
2023-01-20 09:50:36 -08:00
# Fake a 'do' equivalent of all events, so that we can render their response parsing
def _mock_up_events ( self ) :
for op in self . yaml [ ' operations ' ] [ ' list ' ] :
if ' event ' in op :
op [ ' do ' ] = {
' reply ' : {
' attributes ' : op [ ' event ' ] [ ' attributes ' ]
}
}
def _dictify ( self ) :
ntf = [ ]
2023-01-30 18:33:44 -08:00
for msg in self . msgs . values ( ) :
if ' notify ' in msg :
ntf . append ( msg )
2023-01-20 09:50:36 -08:00
for n in ntf :
self . ops [ n [ ' notify ' ] ] . add_notification ( n )
def _load_root_sets ( self ) :
for op_name , op in self . ops . items ( ) :
if ' attribute-set ' not in op :
continue
req_attrs = set ( )
rsp_attrs = set ( )
for op_mode in [ ' do ' , ' dump ' ] :
if op_mode in op and ' request ' in op [ op_mode ] :
req_attrs . update ( set ( op [ op_mode ] [ ' request ' ] [ ' attributes ' ] ) )
if op_mode in op and ' reply ' in op [ op_mode ] :
rsp_attrs . update ( set ( op [ op_mode ] [ ' reply ' ] [ ' attributes ' ] ) )
if op [ ' attribute-set ' ] not in self . root_sets :
self . root_sets [ op [ ' attribute-set ' ] ] = { ' request ' : req_attrs , ' reply ' : rsp_attrs }
else :
self . root_sets [ op [ ' attribute-set ' ] ] [ ' request ' ] . update ( req_attrs )
self . root_sets [ op [ ' attribute-set ' ] ] [ ' reply ' ] . update ( rsp_attrs )
def _load_nested_sets ( self ) :
for root_set , rs_members in self . root_sets . items ( ) :
for attr , spec in self . attr_sets [ root_set ] . items ( ) :
if ' nested-attributes ' in spec :
inherit = set ( )
nested = spec [ ' nested-attributes ' ]
if nested not in self . root_sets :
2023-06-01 19:35:41 -07:00
if nested not in self . pure_nested_structs :
self . pure_nested_structs [ nested ] = Struct ( self , nested , inherited = inherit )
2023-01-20 09:50:36 -08:00
if attr in rs_members [ ' request ' ] :
self . pure_nested_structs [ nested ] . request = True
if attr in rs_members [ ' reply ' ] :
self . pure_nested_structs [ nested ] . reply = True
if ' type-value ' in spec :
if nested in self . root_sets :
raise Exception ( " Inheriting members to a space used as root not supported " )
inherit . update ( set ( spec [ ' type-value ' ] ) )
elif spec [ ' type ' ] == ' array-nest ' :
inherit . add ( ' idx ' )
self . pure_nested_structs [ nested ] . set_inherited ( inherit )
def _load_all_notify ( self ) :
for op_name , op in self . ops . items ( ) :
if not op :
continue
if ' notify ' in op :
self . all_notify [ op_name ] = op [ ' notify ' ] [ ' cmds ' ]
def _load_global_policy ( self ) :
global_set = set ( )
attr_set_name = None
for op_name , op in self . ops . items ( ) :
if not op :
continue
if ' attribute-set ' not in op :
continue
if attr_set_name is None :
attr_set_name = op [ ' attribute-set ' ]
if attr_set_name != op [ ' attribute-set ' ] :
raise Exception ( ' For a global policy all ops must use the same set ' )
2023-01-30 18:33:41 -08:00
for op_mode in [ ' do ' , ' dump ' ] :
2023-01-20 09:50:36 -08:00
if op_mode in op :
global_set . update ( op [ op_mode ] . get ( ' request ' , [ ] ) )
self . global_policy = [ ]
self . global_policy_set = attr_set_name
for attr in self . attr_sets [ attr_set_name ] :
if attr in global_set :
self . global_policy . append ( attr )
def _load_hooks ( self ) :
for op in self . ops . values ( ) :
for op_mode in [ ' do ' , ' dump ' ] :
if op_mode not in op :
continue
for when in [ ' pre ' , ' post ' ] :
if when not in op [ op_mode ] :
continue
name = op [ op_mode ] [ when ]
if name in self . hooks [ when ] [ op_mode ] [ ' set ' ] :
continue
self . hooks [ when ] [ op_mode ] [ ' set ' ] . add ( name )
self . hooks [ when ] [ op_mode ] [ ' list ' ] . append ( name )
2023-06-01 19:35:48 -07:00
def has_notifications ( self ) :
for op in self . ops . values ( ) :
if ' notify ' in op or ' event ' in op :
return True
return False
2023-01-20 09:50:36 -08:00
class RenderInfo :
def __init__ ( self , cw , family , ku_space , op , op_name , op_mode , attr_set = None ) :
self . family = family
self . nl = cw . nlib
self . ku_space = ku_space
self . op = op
self . op_name = op_name
self . op_mode = op_mode
# 'do' and 'dump' response parsing is identical
2023-06-01 19:35:42 -07:00
self . type_consistent = True
if op_mode != ' do ' and ' dump ' in op and ' do ' in op :
if ( ' reply ' in op [ ' do ' ] ) != ( ' reply ' in op [ " dump " ] ) :
self . type_consistent = False
elif ' reply ' in op [ ' do ' ] and op [ " do " ] [ " reply " ] != op [ " dump " ] [ " reply " ] :
self . type_consistent = False
2023-01-20 09:50:36 -08:00
self . attr_set = attr_set
if not self . attr_set :
self . attr_set = op [ ' attribute-set ' ]
if op :
self . type_name = c_lower ( op_name )
else :
self . type_name = c_lower ( attr_set )
self . cw = cw
self . struct = dict ( )
for op_dir in [ ' request ' , ' reply ' ] :
if op and op_dir in op [ op_mode ] :
self . struct [ op_dir ] = Struct ( family , self . attr_set ,
type_list = op [ op_mode ] [ op_dir ] [ ' attributes ' ] )
if op_mode == ' event ' :
self . struct [ ' reply ' ] = Struct ( family , self . attr_set , type_list = op [ ' event ' ] [ ' attributes ' ] )
class CodeWriter :
def __init__ ( self , nlib , out_file ) :
self . nlib = nlib
self . _nl = False
self . _silent_block = False
self . _ind = 0
self . _out = out_file
@classmethod
def _is_cond ( cls , line ) :
return line . startswith ( ' if ' ) or line . startswith ( ' while ' ) or line . startswith ( ' for ' )
2023-06-05 12:01:05 -07:00
def p ( self , line , add_ind = 0 , eat_nl = False ) :
2023-01-20 09:50:36 -08:00
if self . _nl :
2023-06-05 12:01:05 -07:00
if not eat_nl :
self . _out . write ( ' \n ' )
2023-01-20 09:50:36 -08:00
self . _nl = False
ind = self . _ind
if line [ - 1 ] == ' : ' :
ind - = 1
if self . _silent_block :
ind + = 1
self . _silent_block = line . endswith ( ' ) ' ) and CodeWriter . _is_cond ( line )
if add_ind :
ind + = add_ind
self . _out . write ( ' \t ' * ind + line + ' \n ' )
def nl ( self ) :
self . _nl = True
def block_start ( self , line = ' ' ) :
if line :
line = line + ' '
self . p ( line + ' { ' )
self . _ind + = 1
def block_end ( self , line = ' ' ) :
if line and line [ 0 ] not in { ' ; ' , ' , ' } :
line = ' ' + line
self . _ind - = 1
2023-06-05 12:01:05 -07:00
self . p ( ' } ' + line , eat_nl = True )
2023-01-20 09:50:36 -08:00
def write_doc_line ( self , doc , indent = True ) :
words = doc . split ( )
line = ' * '
for word in words :
if len ( line ) + len ( word ) > = 79 :
self . p ( line )
line = ' * '
if indent :
line + = ' '
line + = ' ' + word
self . p ( line )
def write_func_prot ( self , qual_ret , name , args = None , doc = None , suffix = ' ' ) :
if not args :
args = [ ' void ' ]
if doc :
self . p ( ' /* ' )
self . p ( ' * ' + doc )
self . p ( ' */ ' )
oneline = qual_ret
if qual_ret [ - 1 ] != ' * ' :
oneline + = ' '
oneline + = f " { name } ( { ' , ' . join ( args ) } ) { suffix } "
if len ( oneline ) < 80 :
self . p ( oneline )
return
v = qual_ret
if len ( v ) > 3 :
self . p ( v )
v = ' '
elif qual_ret [ - 1 ] != ' * ' :
v + = ' '
v + = name + ' ( '
ind = ' \t ' * ( len ( v ) / / 8 ) + ' ' * ( len ( v ) % 8 )
delta_ind = len ( v ) - len ( ind )
v + = args [ 0 ]
i = 1
while i < len ( args ) :
next_len = len ( v ) + len ( args [ i ] )
if v [ 0 ] == ' \t ' :
next_len + = delta_ind
if next_len > 76 :
self . p ( v + ' , ' )
v = ind
else :
v + = ' , '
v + = args [ i ]
i + = 1
self . p ( v + ' ) ' + suffix )
def write_func_lvar ( self , local_vars ) :
if not local_vars :
return
if type ( local_vars ) is str :
local_vars = [ local_vars ]
local_vars . sort ( key = len , reverse = True )
for var in local_vars :
self . p ( var )
self . nl ( )
def write_func ( self , qual_ret , name , body , args = None , local_vars = None ) :
self . write_func_prot ( qual_ret = qual_ret , name = name , args = args )
self . write_func_lvar ( local_vars = local_vars )
self . block_start ( )
for line in body :
self . p ( line )
self . block_end ( )
def writes_defines ( self , defines ) :
longest = 0
for define in defines :
if len ( define [ 0 ] ) > longest :
longest = len ( define [ 0 ] )
longest = ( ( longest + 8 ) / / 8 ) * 8
for define in defines :
line = ' #define ' + define [ 0 ]
line + = ' \t ' * ( ( longest - len ( define [ 0 ] ) + 7 ) / / 8 )
if type ( define [ 1 ] ) is int :
line + = str ( define [ 1 ] )
elif type ( define [ 1 ] ) is str :
line + = ' " ' + define [ 1 ] + ' " '
self . p ( line )
def write_struct_init ( self , members ) :
longest = max ( [ len ( x [ 0 ] ) for x in members ] )
longest + = 1 # because we prepend a .
longest = ( ( longest + 8 ) / / 8 ) * 8
for one in members :
line = ' . ' + one [ 0 ]
line + = ' \t ' * ( ( longest - len ( one [ 0 ] ) - 1 + 7 ) / / 8 )
line + = ' = ' + one [ 1 ] + ' , '
self . p ( line )
scalars = { ' u8 ' , ' u16 ' , ' u32 ' , ' u64 ' , ' s32 ' , ' s64 ' }
direction_to_suffix = {
' reply ' : ' _rsp ' ,
' request ' : ' _req ' ,
' ' : ' '
}
op_mode_to_wrapper = {
' do ' : ' ' ,
' dump ' : ' _list ' ,
' notify ' : ' _ntf ' ,
' event ' : ' ' ,
}
_C_KW = {
' do '
}
def rdir ( direction ) :
if direction == ' reply ' :
return ' request '
if direction == ' request ' :
return ' reply '
return direction
def op_prefix ( ri , direction , deref = False ) :
suffix = f " _ { ri . type_name } "
if not ri . op_mode or ri . op_mode == ' do ' :
suffix + = f " { direction_to_suffix [ direction ] } "
else :
if direction == ' request ' :
suffix + = ' _req_dump '
else :
if ri . type_consistent :
if deref :
suffix + = f " { direction_to_suffix [ direction ] } "
else :
suffix + = op_mode_to_wrapper [ ri . op_mode ]
else :
suffix + = ' _rsp '
suffix + = ' _dump ' if deref else ' _list '
return f " { ri . family [ ' name ' ] } { suffix } "
def type_name ( ri , direction , deref = False ) :
return f " struct { op_prefix ( ri , direction , deref = deref ) } "
def print_prototype ( ri , direction , terminate = True , doc = None ) :
suffix = ' ; ' if terminate else ' '
fname = ri . op . render_name
if ri . op_mode == ' dump ' :
fname + = ' _dump '
args = [ ' struct ynl_sock *ys ' ]
if ' request ' in ri . op [ ri . op_mode ] :
args . append ( f " { type_name ( ri , direction ) } * " + f " { direction_to_suffix [ direction ] [ 1 : ] } " )
ret = ' int '
if ' reply ' in ri . op [ ri . op_mode ] :
ret = f " { type_name ( ri , rdir ( direction ) ) } * "
ri . cw . write_func_prot ( ret , fname , args , doc = doc , suffix = suffix )
def print_req_prototype ( ri ) :
print_prototype ( ri , " request " , doc = ri . op [ ' doc ' ] )
def print_dump_prototype ( ri ) :
print_prototype ( ri , " request " )
def put_typol_fwd ( cw , struct ) :
cw . p ( f ' extern struct ynl_policy_nest { struct . render_name } _nest; ' )
def put_typol ( cw , struct ) :
type_max = struct . attr_set . max_name
cw . block_start ( line = f ' struct ynl_policy_attr { struct . render_name } _policy[ { type_max } + 1] = ' )
for _ , arg in struct . member_list ( ) :
arg . attr_typol ( cw )
cw . block_end ( line = ' ; ' )
cw . nl ( )
cw . block_start ( line = f ' struct ynl_policy_nest { struct . render_name } _nest = ' )
cw . p ( f ' .max_attr = { type_max } , ' )
cw . p ( f ' .table = { struct . render_name } _policy, ' )
cw . block_end ( line = ' ; ' )
cw . nl ( )
2023-06-01 19:35:44 -07:00
def put_op_name_fwd ( family , cw ) :
cw . write_func_prot ( ' const char * ' , f ' { family . name } _op_str ' , [ ' int op ' ] , suffix = ' ; ' )
def put_op_name ( family , cw ) :
map_name = f ' { family . name } _op_strmap '
cw . block_start ( line = f " static const char * const { map_name } [] = " )
for op_name , op in family . msgs . items ( ) :
2023-06-07 13:23:54 -07:00
if op . rsp_value :
if op . req_value == op . rsp_value :
cw . p ( f ' [ { op . enum_name } ] = " { op_name } " , ' )
else :
cw . p ( f ' [ { op . rsp_value } ] = " { op_name } " , ' )
2023-06-01 19:35:44 -07:00
cw . block_end ( line = ' ; ' )
cw . nl ( )
cw . write_func_prot ( ' const char * ' , f ' { family . name } _op_str ' , [ ' int op ' ] )
cw . block_start ( )
cw . p ( f ' if (op < 0 || op >= (int)MNL_ARRAY_SIZE( { map_name } )) ' )
cw . p ( ' return NULL; ' )
cw . p ( f ' return { map_name } [op]; ' )
cw . block_end ( )
cw . nl ( )
def put_enum_to_str_fwd ( family , cw , enum ) :
args = [ f ' enum { enum . render_name } value ' ]
if ' enum-name ' in enum and not enum [ ' enum-name ' ] :
args = [ ' int value ' ]
cw . write_func_prot ( ' const char * ' , f ' { enum . render_name } _str ' , args , suffix = ' ; ' )
def put_enum_to_str ( family , cw , enum ) :
map_name = f ' { enum . render_name } _strmap '
cw . block_start ( line = f " static const char * const { map_name } [] = " )
for entry in enum . entries . values ( ) :
cw . p ( f ' [ { entry . value } ] = " { entry . name } " , ' )
cw . block_end ( line = ' ; ' )
cw . nl ( )
args = [ f ' enum { enum . render_name } value ' ]
if ' enum-name ' in enum and not enum [ ' enum-name ' ] :
args = [ ' int value ' ]
cw . write_func_prot ( ' const char * ' , f ' { enum . render_name } _str ' , args )
cw . block_start ( )
if enum . type == ' flags ' :
cw . p ( ' value = ffs(value) - 1; ' )
cw . p ( f ' if (value < 0 || value >= (int)MNL_ARRAY_SIZE( { map_name } )) ' )
cw . p ( ' return NULL; ' )
cw . p ( f ' return { map_name } [value]; ' )
cw . block_end ( )
cw . nl ( )
2023-01-20 09:50:36 -08:00
def put_req_nested ( ri , struct ) :
func_args = [ ' struct nlmsghdr *nlh ' ,
' unsigned int attr_type ' ,
f ' { struct . ptr_name } obj ' ]
ri . cw . write_func_prot ( ' int ' , f ' { struct . render_name } _put ' , func_args )
ri . cw . block_start ( )
ri . cw . write_func_lvar ( ' struct nlattr *nest; ' )
ri . cw . p ( " nest = mnl_attr_nest_start(nlh, attr_type); " )
for _ , arg in struct . member_list ( ) :
arg . attr_put ( ri , " obj " )
ri . cw . p ( " mnl_attr_nest_end(nlh, nest); " )
ri . cw . nl ( )
ri . cw . p ( ' return 0; ' )
ri . cw . block_end ( )
ri . cw . nl ( )
def _multi_parse ( ri , struct , init_lines , local_vars ) :
if struct . nested :
iter_line = " mnl_attr_for_each_nested(attr, nested) "
else :
iter_line = " mnl_attr_for_each(attr, nlh, sizeof(struct genlmsghdr)) "
array_nests = set ( )
multi_attrs = set ( )
needs_parg = False
for arg , aspec in struct . member_list ( ) :
if aspec [ ' type ' ] == ' array-nest ' :
local_vars . append ( f ' const struct nlattr *attr_ { aspec . c_name } ; ' )
array_nests . add ( arg )
if ' multi-attr ' in aspec :
multi_attrs . add ( arg )
needs_parg | = ' nested-attributes ' in aspec
if array_nests or multi_attrs :
local_vars . append ( ' int i; ' )
if needs_parg :
local_vars . append ( ' struct ynl_parse_arg parg; ' )
init_lines . append ( ' parg.ys = yarg->ys; ' )
2023-06-06 12:43:01 -07:00
all_multi = array_nests | multi_attrs
for anest in sorted ( all_multi ) :
local_vars . append ( f " unsigned int n_ { struct [ anest ] . c_name } = 0; " )
2023-01-20 09:50:36 -08:00
ri . cw . block_start ( )
ri . cw . write_func_lvar ( local_vars )
for line in init_lines :
ri . cw . p ( line )
ri . cw . nl ( )
for arg in struct . inherited :
ri . cw . p ( f ' dst-> { arg } = { arg } ; ' )
2023-06-06 12:43:01 -07:00
for anest in sorted ( all_multi ) :
aspec = struct [ anest ]
ri . cw . p ( f " if (dst-> { aspec . c_name } ) " )
ri . cw . p ( f ' return ynl_error_parse(yarg, " attribute already present ( { struct . attr_set . name } . { aspec . name } ) " ); ' )
2023-01-20 09:50:36 -08:00
ri . cw . nl ( )
ri . cw . block_start ( line = iter_line )
first = True
for _ , arg in struct . member_list ( ) :
2023-06-01 19:35:40 -07:00
good = arg . attr_get ( ri , ' dst ' , first = first )
# First may be 'unused' or 'pad', ignore those
first & = not good
2023-01-20 09:50:36 -08:00
ri . cw . block_end ( )
ri . cw . nl ( )
for anest in sorted ( array_nests ) :
aspec = struct [ anest ]
2023-06-06 12:43:01 -07:00
ri . cw . block_start ( line = f " if (n_ { aspec . c_name } ) " )
ri . cw . p ( f " dst-> { aspec . c_name } = calloc( { aspec . c_name } , sizeof(*dst-> { aspec . c_name } )); " )
ri . cw . p ( f " dst->n_ { aspec . c_name } = n_ { aspec . c_name } ; " )
2023-01-20 09:50:36 -08:00
ri . cw . p ( ' i = 0; ' )
ri . cw . p ( f " parg.rsp_policy = & { aspec . nested_render_name } _nest; " )
ri . cw . block_start ( line = f " mnl_attr_for_each_nested(attr, attr_ { aspec . c_name } ) " )
ri . cw . p ( f " parg.data = &dst-> { aspec . c_name } [i]; " )
ri . cw . p ( f " if ( { aspec . nested_render_name } _parse(&parg, attr, mnl_attr_get_type(attr))) " )
ri . cw . p ( ' return MNL_CB_ERROR; ' )
ri . cw . p ( ' i++; ' )
ri . cw . block_end ( )
ri . cw . block_end ( )
ri . cw . nl ( )
for anest in sorted ( multi_attrs ) :
aspec = struct [ anest ]
2023-06-06 12:43:01 -07:00
ri . cw . block_start ( line = f " if (n_ { aspec . c_name } ) " )
ri . cw . p ( f " dst-> { aspec . c_name } = calloc(n_ { aspec . c_name } , sizeof(*dst-> { aspec . c_name } )); " )
ri . cw . p ( f " dst->n_ { aspec . c_name } = n_ { aspec . c_name } ; " )
2023-01-20 09:50:36 -08:00
ri . cw . p ( ' i = 0; ' )
if ' nested-attributes ' in aspec :
ri . cw . p ( f " parg.rsp_policy = & { aspec . nested_render_name } _nest; " )
ri . cw . block_start ( line = iter_line )
ri . cw . block_start ( line = f " if (mnl_attr_get_type(attr) == { aspec . enum_name } ) " )
if ' nested-attributes ' in aspec :
ri . cw . p ( f " parg.data = &dst-> { aspec . c_name } [i]; " )
ri . cw . p ( f " if ( { aspec . nested_render_name } _parse(&parg, attr)) " )
ri . cw . p ( ' return MNL_CB_ERROR; ' )
elif aspec [ ' type ' ] in scalars :
t = aspec [ ' type ' ]
if t [ 0 ] == ' s ' :
t = ' u ' + t [ 1 : ]
ri . cw . p ( f " dst-> { aspec . c_name } [i] = mnl_attr_get_ { t } (attr); " )
else :
raise Exception ( ' Nest parsing type not supported yet ' )
ri . cw . p ( ' i++; ' )
ri . cw . block_end ( )
ri . cw . block_end ( )
ri . cw . block_end ( )
ri . cw . nl ( )
if struct . nested :
ri . cw . p ( ' return 0; ' )
else :
ri . cw . p ( ' return MNL_CB_OK; ' )
ri . cw . block_end ( )
ri . cw . nl ( )
def parse_rsp_nested ( ri , struct ) :
func_args = [ ' struct ynl_parse_arg *yarg ' ,
' const struct nlattr *nested ' ]
for arg in struct . inherited :
func_args . append ( ' __u32 ' + arg )
local_vars = [ ' const struct nlattr *attr; ' ,
f ' { struct . ptr_name } dst = yarg->data; ' ]
init_lines = [ ]
ri . cw . write_func_prot ( ' int ' , f ' { struct . render_name } _parse ' , func_args )
_multi_parse ( ri , struct , init_lines , local_vars )
def parse_rsp_msg ( ri , deref = False ) :
if ' reply ' not in ri . op [ ri . op_mode ] and ri . op_mode != ' event ' :
return
func_args = [ ' const struct nlmsghdr *nlh ' ,
' void *data ' ]
local_vars = [ f ' { type_name ( ri , " reply " , deref = deref ) } *dst; ' ,
' struct ynl_parse_arg *yarg = data; ' ,
' const struct nlattr *attr; ' ]
init_lines = [ ' dst = yarg->data; ' ]
ri . cw . write_func_prot ( ' int ' , f ' { op_prefix ( ri , " reply " , deref = deref ) } _parse ' , func_args )
_multi_parse ( ri , ri . struct [ " reply " ] , init_lines , local_vars )
def print_req ( ri ) :
ret_ok = ' 0 '
ret_err = ' -1 '
direction = " request "
local_vars = [ ' struct nlmsghdr *nlh; ' ,
2023-06-01 19:35:45 -07:00
' int err; ' ]
2023-01-20 09:50:36 -08:00
if ' reply ' in ri . op [ ri . op_mode ] :
ret_ok = ' rsp '
ret_err = ' NULL '
local_vars + = [ f ' { type_name ( ri , rdir ( direction ) ) } *rsp; ' ,
2023-06-01 19:35:45 -07:00
' struct ynl_req_state yrs = { .yarg = { .ys = ys, }, }; ' ]
2023-01-20 09:50:36 -08:00
print_prototype ( ri , direction , terminate = False )
ri . cw . block_start ( )
ri . cw . write_func_lvar ( local_vars )
ri . cw . p ( f " nlh = ynl_gemsg_start_req(ys, { ri . nl . get_family_id ( ) } , { ri . op . enum_name } , 1); " )
ri . cw . p ( f " ys->req_policy = & { ri . struct [ ' request ' ] . render_name } _nest; " )
if ' reply ' in ri . op [ ri . op_mode ] :
2023-06-01 19:35:45 -07:00
ri . cw . p ( f " yrs.yarg.rsp_policy = & { ri . struct [ ' reply ' ] . render_name } _nest; " )
2023-01-20 09:50:36 -08:00
ri . cw . nl ( )
for _ , attr in ri . struct [ " request " ] . member_list ( ) :
attr . attr_put ( ri , " req " )
ri . cw . nl ( )
2023-06-01 19:35:45 -07:00
parse_arg = " NULL "
2023-01-20 09:50:36 -08:00
if ' reply ' in ri . op [ ri . op_mode ] :
ri . cw . p ( ' rsp = calloc(1, sizeof(*rsp)); ' )
2023-06-01 19:35:45 -07:00
ri . cw . p ( ' yrs.yarg.data = rsp; ' )
ri . cw . p ( f " yrs.cb = { op_prefix ( ri , ' reply ' ) } _parse; " )
if ri . op . value is not None :
ri . cw . p ( f ' yrs.rsp_cmd = { ri . op . enum_name } ; ' )
else :
ri . cw . p ( f ' yrs.rsp_cmd = { ri . op . rsp_value } ; ' )
2023-01-20 09:50:36 -08:00
ri . cw . nl ( )
2023-06-01 19:35:45 -07:00
parse_arg = ' &yrs '
ri . cw . p ( f " err = ynl_exec(ys, nlh, { parse_arg } ); " )
ri . cw . p ( ' if (err < 0) ' )
if ' reply ' in ri . op [ ri . op_mode ] :
2023-01-20 09:50:36 -08:00
ri . cw . p ( ' goto err_free; ' )
2023-06-01 19:35:45 -07:00
else :
ri . cw . p ( ' return -1; ' )
2023-01-20 09:50:36 -08:00
ri . cw . nl ( )
2023-06-01 19:35:45 -07:00
2023-01-20 09:50:36 -08:00
ri . cw . p ( f " return { ret_ok } ; " )
ri . cw . nl ( )
if ' reply ' in ri . op [ ri . op_mode ] :
2023-06-01 19:35:45 -07:00
ri . cw . p ( ' err_free: ' )
2023-01-20 09:50:36 -08:00
ri . cw . p ( f " { call_free ( ri , rdir ( direction ) , ' rsp ' ) } " )
2023-06-01 19:35:45 -07:00
ri . cw . p ( f " return { ret_err } ; " )
2023-01-20 09:50:36 -08:00
ri . cw . block_end ( )
def print_dump ( ri ) :
direction = " request "
print_prototype ( ri , direction , terminate = False )
ri . cw . block_start ( )
local_vars = [ ' struct ynl_dump_state yds = {} ; ' ,
' struct nlmsghdr *nlh; ' ,
2023-06-01 19:35:45 -07:00
' int err; ' ]
2023-01-20 09:50:36 -08:00
for var in local_vars :
ri . cw . p ( f ' { var } ' )
ri . cw . nl ( )
ri . cw . p ( ' yds.ys = ys; ' )
ri . cw . p ( f " yds.alloc_sz = sizeof( { type_name ( ri , rdir ( direction ) ) } ); " )
ri . cw . p ( f " yds.cb = { op_prefix ( ri , ' reply ' , deref = True ) } _parse; " )
2023-06-01 19:35:45 -07:00
if ri . op . value is not None :
ri . cw . p ( f ' yds.rsp_cmd = { ri . op . enum_name } ; ' )
else :
ri . cw . p ( f ' yds.rsp_cmd = { ri . op . rsp_value } ; ' )
2023-01-20 09:50:36 -08:00
ri . cw . p ( f " yds.rsp_policy = & { ri . struct [ ' reply ' ] . render_name } _nest; " )
ri . cw . nl ( )
ri . cw . p ( f " nlh = ynl_gemsg_start_dump(ys, { ri . nl . get_family_id ( ) } , { ri . op . enum_name } , 1); " )
if " request " in ri . op [ ri . op_mode ] :
ri . cw . p ( f " ys->req_policy = & { ri . struct [ ' request ' ] . render_name } _nest; " )
ri . cw . nl ( )
for _ , attr in ri . struct [ " request " ] . member_list ( ) :
attr . attr_put ( ri , " req " )
ri . cw . nl ( )
2023-06-01 19:35:45 -07:00
ri . cw . p ( ' err = ynl_exec_dump(ys, nlh, &yds); ' )
2023-01-20 09:50:36 -08:00
ri . cw . p ( ' if (err < 0) ' )
ri . cw . p ( ' goto free_list; ' )
ri . cw . nl ( )
ri . cw . p ( ' return yds.first; ' )
ri . cw . nl ( )
ri . cw . p ( ' free_list: ' )
ri . cw . p ( call_free ( ri , rdir ( direction ) , ' yds.first ' ) )
ri . cw . p ( ' return NULL; ' )
ri . cw . block_end ( )
def call_free ( ri , direction , var ) :
return f " { op_prefix ( ri , direction ) } _free( { var } ); "
def free_arg_name ( direction ) :
if direction :
return direction_to_suffix [ direction ] [ 1 : ]
return ' obj '
2023-06-01 19:35:46 -07:00
def print_alloc_wrapper ( ri , direction ) :
name = op_prefix ( ri , direction )
ri . cw . write_func_prot ( f ' static inline struct { name } * ' , f " { name } _alloc " , [ f " void " ] )
ri . cw . block_start ( )
ri . cw . p ( f ' return calloc(1, sizeof(struct { name } )); ' )
ri . cw . block_end ( )
2023-01-20 09:50:36 -08:00
def print_free_prototype ( ri , direction , suffix = ' ; ' ) :
name = op_prefix ( ri , direction )
arg = free_arg_name ( direction )
ri . cw . write_func_prot ( ' void ' , f " { name } _free " , [ f " struct { name } * { arg } " ] , suffix = suffix )
def _print_type ( ri , direction , struct ) :
suffix = f ' _ { ri . type_name } { direction_to_suffix [ direction ] } '
if ri . op_mode == ' dump ' :
suffix + = ' _dump '
ri . cw . block_start ( line = f " struct { ri . family [ ' name ' ] } { suffix } " )
meta_started = False
for _ , attr in struct . member_list ( ) :
for type_filter in [ ' len ' , ' bit ' ] :
line = attr . presence_member ( ri . ku_space , type_filter )
if line :
if not meta_started :
ri . cw . block_start ( line = f " struct " )
meta_started = True
ri . cw . p ( line )
if meta_started :
ri . cw . block_end ( line = ' _present; ' )
ri . cw . nl ( )
for arg in struct . inherited :
ri . cw . p ( f " __u32 { arg } ; " )
for _ , attr in struct . member_list ( ) :
attr . struct_member ( ri )
ri . cw . block_end ( line = ' ; ' )
ri . cw . nl ( )
def print_type ( ri , direction ) :
_print_type ( ri , direction , ri . struct [ direction ] )
def print_type_full ( ri , struct ) :
_print_type ( ri , " " , struct )
def print_type_helpers ( ri , direction , deref = False ) :
print_free_prototype ( ri , direction )
2023-06-01 19:35:46 -07:00
ri . cw . nl ( )
2023-01-20 09:50:36 -08:00
if ri . ku_space == ' user ' and direction == ' request ' :
for _ , attr in ri . struct [ direction ] . member_list ( ) :
attr . setter ( ri , ri . attr_set , direction , deref = deref )
ri . cw . nl ( )
def print_req_type_helpers ( ri ) :
2023-06-01 19:35:46 -07:00
print_alloc_wrapper ( ri , " request " )
2023-01-20 09:50:36 -08:00
print_type_helpers ( ri , " request " )
def print_rsp_type_helpers ( ri ) :
if ' reply ' not in ri . op [ ri . op_mode ] :
return
print_type_helpers ( ri , " reply " )
def print_parse_prototype ( ri , direction , terminate = True ) :
suffix = " _rsp " if direction == " reply " else " _req "
term = ' ; ' if terminate else ' '
ri . cw . write_func_prot ( ' void ' , f " { ri . op . render_name } { suffix } _parse " ,
[ ' const struct nlattr **tb ' ,
f " struct { ri . op . render_name } { suffix } *req " ] ,
suffix = term )
def print_req_type ( ri ) :
print_type ( ri , " request " )
2023-06-01 19:35:46 -07:00
def print_req_free ( ri ) :
if ' request ' not in ri . op [ ri . op_mode ] :
return
_free_type ( ri , ' request ' , ri . struct [ ' request ' ] )
2023-01-20 09:50:36 -08:00
def print_rsp_type ( ri ) :
if ( ri . op_mode == ' do ' or ri . op_mode == ' dump ' ) and ' reply ' in ri . op [ ri . op_mode ] :
direction = ' reply '
elif ri . op_mode == ' event ' :
direction = ' reply '
else :
return
print_type ( ri , direction )
def print_wrapped_type ( ri ) :
ri . cw . block_start ( line = f " { type_name ( ri , ' reply ' ) } " )
if ri . op_mode == ' dump ' :
ri . cw . p ( f " { type_name ( ri , ' reply ' ) } *next; " )
elif ri . op_mode == ' notify ' or ri . op_mode == ' event ' :
ri . cw . p ( ' __u16 family; ' )
ri . cw . p ( ' __u8 cmd; ' )
2023-06-01 19:35:48 -07:00
ri . cw . p ( ' struct ynl_ntf_base_type *next; ' )
2023-01-20 09:50:36 -08:00
ri . cw . p ( f " void (*free)( { type_name ( ri , ' reply ' ) } *ntf); " )
ri . cw . p ( f " { type_name ( ri , ' reply ' , deref = True ) } obj __attribute__ ((aligned (8))); " )
ri . cw . block_end ( line = ' ; ' )
ri . cw . nl ( )
print_free_prototype ( ri , ' reply ' )
ri . cw . nl ( )
def _free_type_members_iter ( ri , struct ) :
for _ , attr in struct . member_list ( ) :
if attr . free_needs_iter ( ) :
ri . cw . p ( ' unsigned int i; ' )
ri . cw . nl ( )
break
def _free_type_members ( ri , var , struct , ref = ' ' ) :
for _ , attr in struct . member_list ( ) :
attr . free ( ri , var , ref )
def _free_type ( ri , direction , struct ) :
var = free_arg_name ( direction )
print_free_prototype ( ri , direction , suffix = ' ' )
ri . cw . block_start ( )
_free_type_members_iter ( ri , struct )
_free_type_members ( ri , var , struct )
if direction :
ri . cw . p ( f ' free( { var } ); ' )
ri . cw . block_end ( )
ri . cw . nl ( )
def free_rsp_nested ( ri , struct ) :
_free_type ( ri , " " , struct )
def print_rsp_free ( ri ) :
if ' reply ' not in ri . op [ ri . op_mode ] :
return
_free_type ( ri , ' reply ' , ri . struct [ ' reply ' ] )
def print_dump_type_free ( ri ) :
sub_type = type_name ( ri , ' reply ' )
print_free_prototype ( ri , ' reply ' , suffix = ' ' )
ri . cw . block_start ( )
ri . cw . p ( f " { sub_type } *next = rsp; " )
ri . cw . nl ( )
2023-06-01 19:35:45 -07:00
ri . cw . block_start ( line = ' while ((void *)next != YNL_LIST_END) ' )
2023-01-20 09:50:36 -08:00
_free_type_members_iter ( ri , ri . struct [ ' reply ' ] )
ri . cw . p ( ' rsp = next; ' )
ri . cw . p ( ' next = rsp->next; ' )
ri . cw . nl ( )
_free_type_members ( ri , ' rsp ' , ri . struct [ ' reply ' ] , ref = ' obj. ' )
ri . cw . p ( f ' free(rsp); ' )
ri . cw . block_end ( )
ri . cw . block_end ( )
ri . cw . nl ( )
def print_ntf_type_free ( ri ) :
print_free_prototype ( ri , ' reply ' , suffix = ' ' )
ri . cw . block_start ( )
_free_type_members_iter ( ri , ri . struct [ ' reply ' ] )
_free_type_members ( ri , ' rsp ' , ri . struct [ ' reply ' ] , ref = ' obj. ' )
ri . cw . p ( f ' free(rsp); ' )
ri . cw . block_end ( )
ri . cw . nl ( )
def print_ntf_parse_prototype ( family , cw , suffix = ' ; ' ) :
cw . write_func_prot ( ' struct ynl_ntf_base_type * ' , f " { family [ ' name ' ] } _ntf_parse " ,
[ ' struct ynl_sock *ys ' ] , suffix = suffix )
def print_ntf_type_parse ( family , cw , ku_mode ) :
print_ntf_parse_prototype ( family , cw , suffix = ' ' )
cw . block_start ( )
cw . write_func_lvar ( [ ' struct genlmsghdr *genlh; ' ,
' struct nlmsghdr *nlh; ' ,
' struct ynl_parse_arg yarg = { .ys = ys, }; ' ,
' struct ynl_ntf_base_type *rsp; ' ,
' int len, err; ' ,
' mnl_cb_t parse; ' ] )
cw . p ( ' len = mnl_socket_recvfrom(ys->sock, ys->rx_buf, MNL_SOCKET_BUFFER_SIZE); ' )
cw . p ( ' if (len < (ssize_t)(sizeof(*nlh) + sizeof(*genlh))) ' )
cw . p ( ' return NULL; ' )
cw . nl ( )
cw . p ( ' nlh = (struct nlmsghdr *)ys->rx_buf; ' )
cw . p ( ' genlh = mnl_nlmsg_get_payload(nlh); ' )
cw . nl ( )
cw . block_start ( line = ' switch (genlh->cmd) ' )
for ntf_op in sorted ( family . all_notify . keys ( ) ) :
op = family . ops [ ntf_op ]
ri = RenderInfo ( cw , family , ku_mode , op , ntf_op , " notify " )
for ntf in op [ ' notify ' ] [ ' cmds ' ] :
cw . p ( f " case { ntf . enum_name } : " )
cw . p ( f " rsp = calloc(1, sizeof( { type_name ( ri , ' notify ' ) } )); " )
cw . p ( f " parse = { op_prefix ( ri , ' reply ' , deref = True ) } _parse; " )
cw . p ( f " yarg.rsp_policy = & { ri . struct [ ' reply ' ] . render_name } _nest; " )
cw . p ( f " rsp->free = (void *) { op_prefix ( ri , ' notify ' ) } _free; " )
cw . p ( ' break; ' )
for op_name , op in family . ops . items ( ) :
if ' event ' not in op :
continue
ri = RenderInfo ( cw , family , ku_mode , op , op_name , " event " )
cw . p ( f " case { op . enum_name } : " )
cw . p ( f " rsp = calloc(1, sizeof( { type_name ( ri , ' event ' ) } )); " )
cw . p ( f " parse = { op_prefix ( ri , ' reply ' , deref = True ) } _parse; " )
cw . p ( f " yarg.rsp_policy = & { ri . struct [ ' reply ' ] . render_name } _nest; " )
cw . p ( f " rsp->free = (void *) { op_prefix ( ri , ' notify ' ) } _free; " )
cw . p ( ' break; ' )
cw . p ( ' default: ' )
cw . p ( ' ynl_error_unknown_notification(ys, genlh->cmd); ' )
cw . p ( ' return NULL; ' )
cw . block_end ( )
cw . nl ( )
cw . p ( ' yarg.data = rsp->data; ' )
cw . nl ( )
cw . p ( f " err = { cw . nlib . parse_cb_run ( ' parse ' , ' &yarg ' , True ) } ; " )
cw . p ( ' if (err < 0) ' )
cw . p ( ' goto err_free; ' )
cw . nl ( )
cw . p ( ' rsp->family = nlh->nlmsg_type; ' )
cw . p ( ' rsp->cmd = genlh->cmd; ' )
cw . p ( ' return rsp; ' )
cw . nl ( )
cw . p ( ' err_free: ' )
cw . p ( ' free(rsp); ' )
cw . p ( ' return NULL; ' )
cw . block_end ( )
cw . nl ( )
def print_req_policy_fwd ( cw , struct , ri = None , terminate = True ) :
if terminate and ri and kernel_can_gen_family_struct ( struct . family ) :
return
if terminate :
prefix = ' extern '
else :
if kernel_can_gen_family_struct ( struct . family ) and ri :
prefix = ' static '
else :
prefix = ' '
suffix = ' ; ' if terminate else ' = { '
max_attr = struct . attr_max_val
if ri :
name = ri . op . render_name
if ri . op . dual_policy :
name + = ' _ ' + ri . op_mode
else :
name = struct . render_name
cw . p ( f " { prefix } const struct nla_policy { name } _nl_policy[ { max_attr . enum_name } + 1] { suffix } " )
def print_req_policy ( cw , struct , ri = None ) :
print_req_policy_fwd ( cw , struct , ri = ri , terminate = False )
for _ , arg in struct . member_list ( ) :
arg . attr_policy ( cw )
cw . p ( " }; " )
def kernel_can_gen_family_struct ( family ) :
return family . proto == ' genetlink '
def print_kernel_op_table_fwd ( family , cw , terminate ) :
exported = not kernel_can_gen_family_struct ( family )
if not terminate or exported :
cw . p ( f " /* Ops table for { family . name } */ " )
pol_to_struct = { ' global ' : ' genl_small_ops ' ,
' per-op ' : ' genl_ops ' ,
' split ' : ' genl_split_ops ' }
struct_type = pol_to_struct [ family . kernel_policy ]
2023-03-20 21:41:59 -07:00
if not exported :
cnt = " "
elif family . kernel_policy == ' split ' :
2023-01-20 09:50:36 -08:00
cnt = 0
for op in family . ops . values ( ) :
if ' do ' in op :
cnt + = 1
if ' dump ' in op :
cnt + = 1
else :
cnt = len ( family . ops )
qual = ' static const ' if not exported else ' const '
line = f " { qual } struct { struct_type } { family . name } _nl_ops[ { cnt } ] "
if terminate :
cw . p ( f " extern { line } ; " )
else :
cw . block_start ( line = line + ' = ' )
if not terminate :
return
cw . nl ( )
for name in family . hooks [ ' pre ' ] [ ' do ' ] [ ' list ' ] :
cw . write_func_prot ( ' int ' , c_lower ( name ) ,
[ ' const struct genl_split_ops *ops ' ,
' struct sk_buff *skb ' , ' struct genl_info *info ' ] , suffix = ' ; ' )
for name in family . hooks [ ' post ' ] [ ' do ' ] [ ' list ' ] :
cw . write_func_prot ( ' void ' , c_lower ( name ) ,
[ ' const struct genl_split_ops *ops ' ,
' struct sk_buff *skb ' , ' struct genl_info *info ' ] , suffix = ' ; ' )
for name in family . hooks [ ' pre ' ] [ ' dump ' ] [ ' list ' ] :
cw . write_func_prot ( ' int ' , c_lower ( name ) ,
[ ' struct netlink_callback *cb ' ] , suffix = ' ; ' )
for name in family . hooks [ ' post ' ] [ ' dump ' ] [ ' list ' ] :
cw . write_func_prot ( ' int ' , c_lower ( name ) ,
[ ' struct netlink_callback *cb ' ] , suffix = ' ; ' )
cw . nl ( )
for op_name , op in family . ops . items ( ) :
if op . is_async :
continue
if ' do ' in op :
name = c_lower ( f " { family . name } -nl- { op_name } -doit " )
cw . write_func_prot ( ' int ' , name ,
[ ' struct sk_buff *skb ' , ' struct genl_info *info ' ] , suffix = ' ; ' )
if ' dump ' in op :
name = c_lower ( f " { family . name } -nl- { op_name } -dumpit " )
cw . write_func_prot ( ' int ' , name ,
[ ' struct sk_buff *skb ' , ' struct netlink_callback *cb ' ] , suffix = ' ; ' )
cw . nl ( )
def print_kernel_op_table_hdr ( family , cw ) :
print_kernel_op_table_fwd ( family , cw , terminate = True )
def print_kernel_op_table ( family , cw ) :
print_kernel_op_table_fwd ( family , cw , terminate = False )
if family . kernel_policy == ' global ' or family . kernel_policy == ' per-op ' :
for op_name , op in family . ops . items ( ) :
if op . is_async :
continue
cw . block_start ( )
members = [ ( ' cmd ' , op . enum_name ) ]
if ' dont-validate ' in op :
members . append ( ( ' validate ' ,
' | ' . join ( [ c_upper ( ' genl-dont-validate- ' + x )
for x in op [ ' dont-validate ' ] ] ) ) , )
for op_mode in [ ' do ' , ' dump ' ] :
if op_mode in op :
name = c_lower ( f " { family . name } -nl- { op_name } - { op_mode } it " )
members . append ( ( op_mode + ' it ' , name ) )
if family . kernel_policy == ' per-op ' :
struct = Struct ( family , op [ ' attribute-set ' ] ,
type_list = op [ ' do ' ] [ ' request ' ] [ ' attributes ' ] )
name = c_lower ( f " { family . name } - { op_name } -nl-policy " )
members . append ( ( ' policy ' , name ) )
members . append ( ( ' maxattr ' , struct . attr_max_val . enum_name ) )
if ' flags ' in op :
members . append ( ( ' flags ' , ' | ' . join ( [ c_upper ( ' genl- ' + x ) for x in op [ ' flags ' ] ] ) ) )
cw . write_struct_init ( members )
cw . block_end ( line = ' , ' )
elif family . kernel_policy == ' split ' :
cb_names = { ' do ' : { ' pre ' : ' pre_doit ' , ' post ' : ' post_doit ' } ,
' dump ' : { ' pre ' : ' start ' , ' post ' : ' done ' } }
for op_name , op in family . ops . items ( ) :
for op_mode in [ ' do ' , ' dump ' ] :
if op . is_async or op_mode not in op :
continue
cw . block_start ( )
members = [ ( ' cmd ' , op . enum_name ) ]
if ' dont-validate ' in op :
members . append ( ( ' validate ' ,
' | ' . join ( [ c_upper ( ' genl-dont-validate- ' + x )
for x in op [ ' dont-validate ' ] ] ) ) , )
name = c_lower ( f " { family . name } -nl- { op_name } - { op_mode } it " )
if ' pre ' in op [ op_mode ] :
members . append ( ( cb_names [ op_mode ] [ ' pre ' ] , c_lower ( op [ op_mode ] [ ' pre ' ] ) ) )
members . append ( ( op_mode + ' it ' , name ) )
if ' post ' in op [ op_mode ] :
members . append ( ( cb_names [ op_mode ] [ ' post ' ] , c_lower ( op [ op_mode ] [ ' post ' ] ) ) )
if ' request ' in op [ op_mode ] :
struct = Struct ( family , op [ ' attribute-set ' ] ,
type_list = op [ op_mode ] [ ' request ' ] [ ' attributes ' ] )
if op . dual_policy :
name = c_lower ( f " { family . name } - { op_name } - { op_mode } -nl-policy " )
else :
name = c_lower ( f " { family . name } - { op_name } -nl-policy " )
members . append ( ( ' policy ' , name ) )
members . append ( ( ' maxattr ' , struct . attr_max_val . enum_name ) )
flags = ( op [ ' flags ' ] if ' flags ' in op else [ ] ) + [ ' cmd-cap- ' + op_mode ]
members . append ( ( ' flags ' , ' | ' . join ( [ c_upper ( ' genl- ' + x ) for x in flags ] ) ) )
cw . write_struct_init ( members )
cw . block_end ( line = ' , ' )
cw . block_end ( line = ' ; ' )
cw . nl ( )
def print_kernel_mcgrp_hdr ( family , cw ) :
if not family . mcgrps [ ' list ' ] :
return
cw . block_start ( ' enum ' )
for grp in family . mcgrps [ ' list ' ] :
grp_id = c_upper ( f " { family . name } -nlgrp- { grp [ ' name ' ] } , " )
cw . p ( grp_id )
cw . block_end ( ' ; ' )
cw . nl ( )
def print_kernel_mcgrp_src ( family , cw ) :
if not family . mcgrps [ ' list ' ] :
return
cw . block_start ( ' static const struct genl_multicast_group ' + family . name + ' _nl_mcgrps[] = ' )
for grp in family . mcgrps [ ' list ' ] :
name = grp [ ' name ' ]
grp_id = c_upper ( f " { family . name } -nlgrp- { name } " )
cw . p ( ' [ ' + grp_id + ' ] = { " ' + name + ' " , }, ' )
cw . block_end ( ' ; ' )
cw . nl ( )
def print_kernel_family_struct_hdr ( family , cw ) :
if not kernel_can_gen_family_struct ( family ) :
return
cw . p ( f " extern struct genl_family { family . name } _nl_family; " )
cw . nl ( )
def print_kernel_family_struct_src ( family , cw ) :
if not kernel_can_gen_family_struct ( family ) :
return
cw . block_start ( f " struct genl_family { family . name } _nl_family __ro_after_init = " )
cw . p ( ' .name \t \t = ' + family . fam_key + ' , ' )
cw . p ( ' .version \t = ' + family . ver_key + ' , ' )
cw . p ( ' .netnsok \t = true, ' )
cw . p ( ' .parallel_ops \t = true, ' )
cw . p ( ' .module \t \t = THIS_MODULE, ' )
if family . kernel_policy == ' per-op ' :
cw . p ( f ' .ops \t \t = { family . name } _nl_ops, ' )
cw . p ( f ' .n_ops \t \t = ARRAY_SIZE( { family . name } _nl_ops), ' )
elif family . kernel_policy == ' split ' :
cw . p ( f ' .split_ops \t = { family . name } _nl_ops, ' )
cw . p ( f ' .n_split_ops \t = ARRAY_SIZE( { family . name } _nl_ops), ' )
if family . mcgrps [ ' list ' ] :
cw . p ( f ' .mcgrps \t \t = { family . name } _nl_mcgrps, ' )
cw . p ( f ' .n_mcgrps \t = ARRAY_SIZE( { family . name } _nl_mcgrps), ' )
cw . block_end ( ' ; ' )
def uapi_enum_start ( family , cw , obj , ckey = ' ' , enum_name = ' enum-name ' ) :
start_line = ' enum '
if enum_name in obj :
if obj [ enum_name ] :
start_line = ' enum ' + c_lower ( obj [ enum_name ] )
elif ckey and ckey in obj :
start_line = ' enum ' + family . name + ' _ ' + c_lower ( obj [ ckey ] )
cw . block_start ( line = start_line )
def render_uapi ( family , cw ) :
hdr_prot = f " _UAPI_LINUX_ { family . name . upper ( ) } _H "
cw . p ( ' #ifndef ' + hdr_prot )
cw . p ( ' #define ' + hdr_prot )
cw . nl ( )
defines = [ ( family . fam_key , family [ " name " ] ) ,
( family . ver_key , family . get ( ' version ' , 1 ) ) ]
cw . writes_defines ( defines )
cw . nl ( )
defines = [ ]
for const in family [ ' definitions ' ] :
if const [ ' type ' ] != ' const ' :
cw . writes_defines ( defines )
defines = [ ]
cw . nl ( )
2023-01-25 16:02:33 -08:00
# Write kdoc for enum and flags (one day maybe also structs)
if const [ ' type ' ] == ' enum ' or const [ ' type ' ] == ' flags ' :
2023-01-20 09:50:36 -08:00
enum = family . consts [ const [ ' name ' ] ]
if enum . has_doc ( ) :
cw . p ( ' /** ' )
doc = ' '
if ' doc ' in enum :
doc = ' - ' + enum [ ' doc ' ]
cw . write_doc_line ( enum . enum_name + doc )
2023-03-07 16:39:22 -08:00
for entry in enum . entries . values ( ) :
2023-01-20 09:50:36 -08:00
if entry . has_doc ( ) :
doc = ' @ ' + entry . c_name + ' : ' + entry [ ' doc ' ]
cw . write_doc_line ( doc )
cw . p ( ' */ ' )
uapi_enum_start ( family , cw , const , ' name ' )
name_pfx = const . get ( ' name-prefix ' , f " { family . name } - { const [ ' name ' ] } - " )
2023-03-07 16:39:22 -08:00
for entry in enum . entries . values ( ) :
2023-01-20 09:50:36 -08:00
suffix = ' , '
2023-01-25 16:02:33 -08:00
if entry . value_change :
suffix = f " = { entry . user_value ( ) } " + suffix
2023-01-20 09:50:36 -08:00
cw . p ( entry . c_name + suffix )
if const . get ( ' render-max ' , False ) :
cw . nl ( )
2023-03-09 13:25:25 +01:00
if const [ ' type ' ] == ' flags ' :
max_name = c_upper ( name_pfx + ' mask ' )
max_val = f ' = { enum . get_mask ( ) } , '
cw . p ( max_name + max_val )
else :
max_name = c_upper ( name_pfx + ' max ' )
cw . p ( ' __ ' + max_name + ' , ' )
cw . p ( max_name + ' = (__ ' + max_name + ' - 1) ' )
2023-01-20 09:50:36 -08:00
cw . block_end ( line = ' ; ' )
cw . nl ( )
elif const [ ' type ' ] == ' const ' :
defines . append ( [ c_upper ( family . get ( ' c-define-name ' ,
f " { family . name } - { const [ ' name ' ] } " ) ) ,
const [ ' value ' ] ] )
if defines :
cw . writes_defines ( defines )
cw . nl ( )
max_by_define = family . get ( ' max-by-define ' , False )
2023-01-30 18:33:44 -08:00
for _ , attr_set in family . attr_sets . items ( ) :
2023-01-20 09:50:36 -08:00
if attr_set . subset_of :
continue
cnt_name = c_upper ( family . get ( ' attr-cnt-name ' , f " __ { attr_set . name_prefix } MAX " ) )
max_value = f " ( { cnt_name } - 1) "
val = 0
uapi_enum_start ( family , cw , attr_set . yaml , ' enum-name ' )
for _ , attr in attr_set . items ( ) :
suffix = ' , '
2023-01-30 18:33:44 -08:00
if attr . value != val :
suffix = f " = { attr . value } , "
val = attr . value
2023-01-20 09:50:36 -08:00
val + = 1
cw . p ( attr . enum_name + suffix )
cw . nl ( )
cw . p ( cnt_name + ( ' ' if max_by_define else ' , ' ) )
if not max_by_define :
cw . p ( f " { attr_set . max_name } = { max_value } " )
cw . block_end ( line = ' ; ' )
if max_by_define :
cw . p ( f " #define { attr_set . max_name } { max_value } " )
cw . nl ( )
# Commands
separate_ntf = ' async-prefix ' in family [ ' operations ' ]
max_name = c_upper ( family . get ( ' cmd-max-name ' , f " { family . op_prefix } MAX " ) )
cnt_name = c_upper ( family . get ( ' cmd-cnt-name ' , f " __ { family . op_prefix } MAX " ) )
max_value = f " ( { cnt_name } - 1) "
uapi_enum_start ( family , cw , family [ ' operations ' ] , ' enum-name ' )
2023-03-01 10:36:41 -08:00
val = 0
2023-01-30 18:33:44 -08:00
for op in family . msgs . values ( ) :
2023-01-20 09:50:36 -08:00
if separate_ntf and ( ' notify ' in op or ' event ' in op ) :
continue
suffix = ' , '
2023-03-01 10:36:41 -08:00
if op . value != val :
suffix = f " = { op . value } , "
val = op . value
2023-01-20 09:50:36 -08:00
cw . p ( op . enum_name + suffix )
2023-03-01 10:36:41 -08:00
val + = 1
2023-01-20 09:50:36 -08:00
cw . nl ( )
cw . p ( cnt_name + ( ' ' if max_by_define else ' , ' ) )
if not max_by_define :
cw . p ( f " { max_name } = { max_value } " )
cw . block_end ( line = ' ; ' )
if max_by_define :
cw . p ( f " #define { max_name } { max_value } " )
cw . nl ( )
if separate_ntf :
uapi_enum_start ( family , cw , family [ ' operations ' ] , enum_name = ' async-enum ' )
2023-01-30 18:33:44 -08:00
for op in family . msgs . values ( ) :
2023-01-20 09:50:36 -08:00
if separate_ntf and not ( ' notify ' in op or ' event ' in op ) :
continue
suffix = ' , '
if ' value ' in op :
suffix = f " = { op [ ' value ' ] } , "
cw . p ( op . enum_name + suffix )
cw . block_end ( line = ' ; ' )
cw . nl ( )
# Multicast
defines = [ ]
for grp in family . mcgrps [ ' list ' ] :
name = grp [ ' name ' ]
defines . append ( [ c_upper ( grp . get ( ' c-define-name ' , f " { family . name } -mcgrp- { name } " ) ) ,
f ' { name } ' ] )
cw . nl ( )
if defines :
cw . writes_defines ( defines )
cw . nl ( )
cw . p ( f ' #endif /* { hdr_prot } */ ' )
2023-06-01 19:35:48 -07:00
def _render_user_ntf_entry ( ri , op ) :
ri . cw . block_start ( line = f " [ { op . enum_name } ] = " )
ri . cw . p ( f " .alloc_sz \t = sizeof( { type_name ( ri , ' event ' ) } ), " )
ri . cw . p ( f " .cb \t \t = { op_prefix ( ri , ' reply ' , deref = True ) } _parse, " )
ri . cw . p ( f " .policy \t \t = & { ri . struct [ ' reply ' ] . render_name } _nest, " )
ri . cw . p ( f " .free \t \t = (void *) { op_prefix ( ri , ' notify ' ) } _free, " )
ri . cw . block_end ( line = ' , ' )
2023-06-01 19:35:47 -07:00
def render_user_family ( family , cw , prototype ) :
symbol = f ' const struct ynl_family ynl_ { family . c_name } _family '
if prototype :
cw . p ( f ' extern { symbol } ; ' )
2023-06-01 19:35:48 -07:00
return
ntf = family . has_notifications ( )
if ntf :
cw . block_start ( line = f " static const struct ynl_ntf_info { family [ ' name ' ] } _ntf_info[] = " )
for ntf_op in sorted ( family . all_notify . keys ( ) ) :
op = family . ops [ ntf_op ]
ri = RenderInfo ( cw , family , " user " , op , ntf_op , " notify " )
for ntf in op [ ' notify ' ] [ ' cmds ' ] :
_render_user_ntf_entry ( ri , ntf )
for op_name , op in family . ops . items ( ) :
if ' event ' not in op :
continue
ri = RenderInfo ( cw , family , " user " , op , op_name , " event " )
_render_user_ntf_entry ( ri , op )
cw . block_end ( line = " ; " )
cw . nl ( )
cw . block_start ( f ' { symbol } = ' )
cw . p ( f ' .name \t \t = " { family . name } " , ' )
if ntf :
cw . p ( f " .ntf_info \t = { family [ ' name ' ] } _ntf_info, " )
cw . p ( f " .ntf_info_size \t = MNL_ARRAY_SIZE( { family [ ' name ' ] } _ntf_info), " )
cw . block_end ( line = ' ; ' )
2023-06-01 19:35:47 -07:00
2023-01-20 09:50:36 -08:00
def find_kernel_root ( full_path ) :
sub_path = ' '
while True :
sub_path = os . path . join ( os . path . basename ( full_path ) , sub_path )
full_path = os . path . dirname ( full_path )
maintainers = os . path . join ( full_path , " MAINTAINERS " )
if os . path . exists ( maintainers ) :
return full_path , sub_path [ : - 1 ]
def main ( ) :
parser = argparse . ArgumentParser ( description = ' Netlink simple parsing generator ' )
parser . add_argument ( ' --mode ' , dest = ' mode ' , type = str , required = True )
parser . add_argument ( ' --spec ' , dest = ' spec ' , type = str , required = True )
parser . add_argument ( ' --header ' , dest = ' header ' , action = ' store_true ' , default = None )
parser . add_argument ( ' --source ' , dest = ' header ' , action = ' store_false ' )
parser . add_argument ( ' --user-header ' , nargs = ' + ' , default = [ ] )
parser . add_argument ( ' -o ' , dest = ' out_file ' , type = str )
args = parser . parse_args ( )
out_file = open ( args . out_file , ' w+ ' ) if args . out_file else os . sys . stdout
if args . header is None :
parser . error ( " --header or --source is required " )
try :
parsed = Family ( args . spec )
2023-03-15 16:03:51 -07:00
if parsed . license != ' ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause) ' :
print ( ' Spec license: ' , parsed . license )
print ( ' License must be: ((GPL-2.0 WITH Linux-syscall-note) OR BSD-3-Clause) ' )
os . sys . exit ( 1 )
2023-01-20 09:50:36 -08:00
except yaml . YAMLError as exc :
print ( exc )
os . sys . exit ( 1 )
return
cw = CodeWriter ( BaseNlLib ( ) , out_file )
_ , spec_kernel = find_kernel_root ( args . spec )
2023-03-15 16:03:51 -07:00
if args . mode == ' uapi ' or args . header :
cw . p ( f ' /* SPDX-License-Identifier: { parsed . license } */ ' )
2023-01-20 09:50:36 -08:00
else :
2023-03-15 16:03:51 -07:00
cw . p ( f ' // SPDX-License-Identifier: { parsed . license } ' )
2023-01-20 09:50:36 -08:00
cw . p ( " /* Do not edit directly, auto-generated from: */ " )
cw . p ( f " /* \t { spec_kernel } */ " )
cw . p ( f " /* YNL-GEN { args . mode } { ' header ' if args . header else ' source ' } */ " )
cw . nl ( )
if args . mode == ' uapi ' :
render_uapi ( parsed , cw )
return
hdr_prot = f " _LINUX_ { parsed . name . upper ( ) } _GEN_H "
if args . header :
cw . p ( ' #ifndef ' + hdr_prot )
cw . p ( ' #define ' + hdr_prot )
cw . nl ( )
if args . mode == ' kernel ' :
cw . p ( ' #include <net/netlink.h> ' )
cw . p ( ' #include <net/genetlink.h> ' )
cw . nl ( )
if not args . header :
if args . out_file :
cw . p ( f ' #include " { os . path . basename ( args . out_file [ : - 2 ] ) } .h " ' )
cw . nl ( )
2023-05-24 10:09:01 -07:00
headers = [ ' uapi/ ' + parsed . uapi_header ]
else :
2023-06-01 19:35:39 -07:00
cw . p ( ' #include <stdlib.h> ' )
if args . header :
cw . p ( ' #include <string.h> ' )
cw . p ( ' #include <linux/types.h> ' )
else :
cw . p ( f ' #include " { parsed . name } -user.h " ' )
cw . p ( ' #include " ynl.h " ' )
2023-05-24 10:09:01 -07:00
headers = [ parsed . uapi_header ]
2023-01-20 09:50:36 -08:00
for definition in parsed [ ' definitions ' ] :
if ' header ' in definition :
headers . append ( definition [ ' header ' ] )
for one in headers :
cw . p ( f " #include < { one } > " )
cw . nl ( )
if args . mode == " user " :
if not args . header :
cw . p ( " #include <stdlib.h> " )
cw . p ( " #include <stdio.h> " )
cw . p ( " #include <string.h> " )
cw . p ( " #include <libmnl/libmnl.h> " )
cw . p ( " #include <linux/genetlink.h> " )
cw . nl ( )
for one in args . user_header :
cw . p ( f ' #include " { one } " ' )
else :
cw . p ( ' struct ynl_sock; ' )
2023-06-01 19:35:47 -07:00
cw . nl ( )
render_user_family ( parsed , cw , True )
2023-01-20 09:50:36 -08:00
cw . nl ( )
if args . mode == " kernel " :
if args . header :
for _ , struct in sorted ( parsed . pure_nested_structs . items ( ) ) :
if struct . request :
cw . p ( ' /* Common nested types */ ' )
break
for attr_set , struct in sorted ( parsed . pure_nested_structs . items ( ) ) :
if struct . request :
print_req_policy_fwd ( cw , struct )
cw . nl ( )
if parsed . kernel_policy == ' global ' :
cw . p ( f " /* Global operation policy for { parsed . name } */ " )
struct = Struct ( parsed , parsed . global_policy_set , type_list = parsed . global_policy )
print_req_policy_fwd ( cw , struct )
cw . nl ( )
if parsed . kernel_policy in { ' per-op ' , ' split ' } :
for op_name , op in parsed . ops . items ( ) :
if ' do ' in op and ' event ' not in op :
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , " do " )
print_req_policy_fwd ( cw , ri . struct [ ' request ' ] , ri = ri )
cw . nl ( )
print_kernel_op_table_hdr ( parsed , cw )
print_kernel_mcgrp_hdr ( parsed , cw )
print_kernel_family_struct_hdr ( parsed , cw )
else :
for _ , struct in sorted ( parsed . pure_nested_structs . items ( ) ) :
if struct . request :
cw . p ( ' /* Common nested types */ ' )
break
for attr_set , struct in sorted ( parsed . pure_nested_structs . items ( ) ) :
if struct . request :
print_req_policy ( cw , struct )
cw . nl ( )
if parsed . kernel_policy == ' global ' :
cw . p ( f " /* Global operation policy for { parsed . name } */ " )
struct = Struct ( parsed , parsed . global_policy_set , type_list = parsed . global_policy )
print_req_policy ( cw , struct )
cw . nl ( )
for op_name , op in parsed . ops . items ( ) :
if parsed . kernel_policy in { ' per-op ' , ' split ' } :
2023-01-30 18:33:41 -08:00
for op_mode in [ ' do ' , ' dump ' ] :
2023-01-20 09:50:36 -08:00
if op_mode in op and ' request ' in op [ op_mode ] :
cw . p ( f " /* { op . enum_name } - { op_mode } */ " )
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , op_mode )
print_req_policy ( cw , ri . struct [ ' request ' ] , ri = ri )
cw . nl ( )
print_kernel_op_table ( parsed , cw )
print_kernel_mcgrp_src ( parsed , cw )
print_kernel_family_struct_src ( parsed , cw )
if args . mode == " user " :
if args . header :
2023-06-01 19:35:44 -07:00
cw . p ( ' /* Enums */ ' )
put_op_name_fwd ( parsed , cw )
for name , const in parsed . consts . items ( ) :
if isinstance ( const , EnumSet ) :
put_enum_to_str_fwd ( parsed , cw , const )
cw . nl ( )
2023-01-20 09:50:36 -08:00
cw . p ( ' /* Common nested types */ ' )
for attr_set , struct in sorted ( parsed . pure_nested_structs . items ( ) ) :
ri = RenderInfo ( cw , parsed , args . mode , " " , " " , " " , attr_set )
print_type_full ( ri , struct )
for op_name , op in parsed . ops . items ( ) :
cw . p ( f " /* ============== { op . enum_name } ============== */ " )
if ' do ' in op and ' event ' not in op :
cw . p ( f " /* { op . enum_name } - do */ " )
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , " do " )
print_req_type ( ri )
print_req_type_helpers ( ri )
cw . nl ( )
print_rsp_type ( ri )
print_rsp_type_helpers ( ri )
cw . nl ( )
print_req_prototype ( ri )
cw . nl ( )
if ' dump ' in op :
cw . p ( f " /* { op . enum_name } - dump */ " )
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , ' dump ' )
if ' request ' in op [ ' dump ' ] :
print_req_type ( ri )
print_req_type_helpers ( ri )
if not ri . type_consistent :
print_rsp_type ( ri )
print_wrapped_type ( ri )
print_dump_prototype ( ri )
cw . nl ( )
if ' notify ' in op :
cw . p ( f " /* { op . enum_name } - notify */ " )
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , ' notify ' )
if not ri . type_consistent :
2023-06-01 19:35:42 -07:00
raise Exception ( f ' Only notifications with consistent types supported ( { op . name } ) ' )
2023-01-20 09:50:36 -08:00
print_wrapped_type ( ri )
if ' event ' in op :
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , ' event ' )
cw . p ( f " /* { op . enum_name } - event */ " )
print_rsp_type ( ri )
cw . nl ( )
print_wrapped_type ( ri )
2023-06-01 19:35:48 -07:00
if parsed . has_notifications ( ) :
2023-01-20 09:50:36 -08:00
cw . p ( ' /* --------------- Common notification parsing --------------- */ ' )
print_ntf_parse_prototype ( parsed , cw )
cw . nl ( )
else :
2023-06-01 19:35:44 -07:00
cw . p ( ' /* Enums */ ' )
put_op_name ( parsed , cw )
for name , const in parsed . consts . items ( ) :
if isinstance ( const , EnumSet ) :
put_enum_to_str ( parsed , cw , const )
cw . nl ( )
2023-01-20 09:50:36 -08:00
cw . p ( ' /* Policies */ ' )
for name , _ in parsed . attr_sets . items ( ) :
struct = Struct ( parsed , name )
put_typol_fwd ( cw , struct )
cw . nl ( )
for name , _ in parsed . attr_sets . items ( ) :
struct = Struct ( parsed , name )
put_typol ( cw , struct )
cw . p ( ' /* Common nested types */ ' )
for attr_set , struct in sorted ( parsed . pure_nested_structs . items ( ) ) :
ri = RenderInfo ( cw , parsed , args . mode , " " , " " , " " , attr_set )
free_rsp_nested ( ri , struct )
if struct . request :
put_req_nested ( ri , struct )
if struct . reply :
parse_rsp_nested ( ri , struct )
for op_name , op in parsed . ops . items ( ) :
cw . p ( f " /* ============== { op . enum_name } ============== */ " )
if ' do ' in op and ' event ' not in op :
cw . p ( f " /* { op . enum_name } - do */ " )
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , " do " )
2023-06-01 19:35:46 -07:00
print_req_free ( ri )
2023-01-20 09:50:36 -08:00
print_rsp_free ( ri )
parse_rsp_msg ( ri )
print_req ( ri )
cw . nl ( )
if ' dump ' in op :
cw . p ( f " /* { op . enum_name } - dump */ " )
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , " dump " )
if not ri . type_consistent :
parse_rsp_msg ( ri , deref = True )
print_dump_type_free ( ri )
print_dump ( ri )
cw . nl ( )
if ' notify ' in op :
cw . p ( f " /* { op . enum_name } - notify */ " )
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , ' notify ' )
if not ri . type_consistent :
2023-06-01 19:35:42 -07:00
raise Exception ( f ' Only notifications with consistent types supported ( { op . name } ) ' )
2023-01-20 09:50:36 -08:00
print_ntf_type_free ( ri )
if ' event ' in op :
cw . p ( f " /* { op . enum_name } - event */ " )
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , " do " )
parse_rsp_msg ( ri )
ri = RenderInfo ( cw , parsed , args . mode , op , op_name , " event " )
print_ntf_type_free ( ri )
2023-06-01 19:35:48 -07:00
if parsed . has_notifications ( ) :
2023-01-20 09:50:36 -08:00
cw . p ( ' /* --------------- Common notification parsing --------------- */ ' )
print_ntf_type_parse ( parsed , cw , args . mode )
2023-06-01 19:35:47 -07:00
cw . nl ( )
render_user_family ( parsed , cw , False )
2023-01-20 09:50:36 -08:00
if args . header :
cw . p ( f ' #endif /* { hdr_prot } */ ' )
if __name__ == " __main__ " :
main ( )