I'm looking over the iPXE source code to try to understand what happens early on at this stage of boot up, but there is some major nested macro expansion going on that I'm unfamiliar with and having trouble following.
In one of the source files (core/init.c
), there is:
void initialise ( void ) {
struct init_fn *init_fn;
/* Call registered initialisation functions */
for_each_table_entry ( init_fn, INIT_FNS )
init_fn->initialise ();
}
So I'm walking through the expansion of for_each_table_entry
.
Here are some of the relevant macro definitions:
#define for_each_table_entry( pointer, table ) \
for ( pointer = table_start ( table ) ; \
pointer < table_end ( table ) ; \
pointer++ )
#define table_start( table ) __table_entries ( table, 00 )
#define table_end( table ) __table_entries ( table, 99 )
#define __table_entries( table, idx ) ( { \
static __table_type ( table ) __table_entries[0] \
__table_entry ( table, idx ) \
__attribute__ (( unused )); \
__table_entries; } )
#define __table_type( table ) __table_extract_type table
#define __table_extract_type( type, name ) type
#define __table_entry( table, idx ) \
__attribute__ (( __section__ ( __table_section ( table, idx ) ),\
__aligned__ ( __table_alignment ( table ) ) ))
#define __table_section( table, idx ) \
".tbl." __table_name ( table ) "." __table_str ( idx )
#define __table_str( x ) #x
#define __table_alignment( table ) __alignof__ ( __table_type ( table ) )
#define __table_name( table ) __table_extract_name table
#define __table_extract_name( type, name ) name
Phew, I think that may be it. A ridiculous amount of nested expansion in my opinion.
So if I try to manually expand to see what is going on, I get:
# 1
for_each_table_entry ( init_fn, INIT_FNS )
# 2
for ( init_fn = table_start ( INIT_FNS ) ; \
init_fn < table_end ( INIT_FNS ) ; \
init_fn++ )
# 3
for ( init_fn = __table_entries ( INIT_FNS, 00 ) ; \
init_fn < __table_entries ( INIT_FNS, 00 ) ; \
init_fn++ )
# 4
for ( init_fn = ( { \
static __table_type ( INIT_FNS ) __table_entries[0] \
__table_entry ( INIT_FNS, 00 ) \
__attribute__ (( unused )); \
__table_entries; } ) ; \
init_fn < ( { \
static __table_type ( INIT_FNS ) __table_entries[0] \
__table_entry ( INIT_FNS, 99 ) \
__attribute__ (( unused )); \
__table_entries; } ) ; \
init_fn++ )
# 5
for ( init_fn = ( { \
static __table_extract_type INIT_FNS __table_entries[0] \
__attribute__ (( __section__ ( __table_section ( INIT_FNS, 00 ) ),\
__aligned__ ( __table_alignment ( INIT_FNS ) ) )) \
__attribute__ (( unused )); \
__table_entries; } ) ; \
init_fn < ( { \
static __table_extract_type INIT_FNS __table_entries[0]
__attribute__ (( __section__ ( __table_section ( INIT_FNS, 99 ) ),\
__aligned__ ( __table_alignment ( INIT_FNS ) ) ))
__attribute__ (( unused )); \
__table_entries; } ) ; \
init_fn++ )
# 6
for ( init_fn = ( { \
static __table_extract_type INIT_FNS __table_entries[0] \
__attribute__ (( __section__ ( ".tbl." __table_name ( INIT_FNS ) "." __table_str ( 00 ) ),\
__aligned__ ( __alignof__ ( __table_type ( INIT_FNS ) ) ) )) \
__attribute__ (( unused )); \
__table_entries; } ) ; \
init_fn < ( { \
static __table_extract_type INIT_FNS __table_entries[0]
__attribute__ (( __section__ ( ".tbl." __table_name ( INIT_FNS ) "." __table_str ( 99 ),\
__aligned__ ( __alignof__ ( __table_type ( INIT_FNS ) ) ) ))
__attribute__ (( unused )); \
__table_entries; } ) ; \
init_fn++ )
# 7
for ( init_fn = ( { \
static __table_extract_type INIT_FNS __table_entries[0] \
__attribute__ (( __section__ ( ".tbl." __table_extract_name INIT_FNS "." #00 ),\
__aligned__ ( __alignof__ ( __table_extract_type INIT_FNS ) ) )) \
__attribute__ (( unused )); \
__table_entries; } ) ; \
init_fn < ( { \
static __table_extract_type INIT_FNS __table_entries[0]
__attribute__ (( __section__ ( ".tbl." __table_extract_name INIT_FNS "." #99,\
__aligned__ ( __alignof__ ( __table_extract_type INIT_FNS ) ) ))
__attribute__ (( unused )); \
__table_entries; } ) ; \
init_fn++ )
Holy hell batman, I think that's it. I mean, it looks like there are more macros in there, so I may have messed up, but I don't see any object type macro definitions for
__table_extract_type
__table_extract_name
__table_entries
__attribute__
__aligned__
- There is a def. for
__aligned__attribute__
, maybe I messed that up?
- There is a def. for
__section__
__alignof__
So was my expansion correct? And if so, how is this even valid code? Is this low level C/ASM stuff? I know the basics of assembly, but haven't used it in several years so I'm not sure what's going on here.