2007-06-11 11:47:01 +00:00
#!/usr/bin/perl -w
#
2010-04-16 22:09:25 -04:00
# This script parses remote_protocol.x or qemu_protocol.x and produces lots of
# boilerplate code for both ends of the remote connection.
#
# The first non-option argument specifies the prefix to be searched for, and
# output to, the boilerplate code. The second non-option argument is the
# file you want to operate on. For instance, to generate the dispatch table
# for both remote_protocol.x and qemu_protocol.x, you would run the
# following:
#
# remote_generate_stubs.pl -c -t remote ../src/remote/remote_protocol.x
# remote_generate_stubs.pl -t qemu ../src/remote/qemu_protocol.x
2007-06-11 11:47:01 +00:00
#
# By Richard Jones <rjones@redhat.com>
use strict ;
use Getopt::Std ;
# Command line options.
2010-04-16 22:09:25 -04:00
our ( $ opt_p , $ opt_t , $ opt_a , $ opt_r , $ opt_d , $ opt_c ) ;
getopts ( 'ptardc' ) ;
my $ structprefix = $ ARGV [ 0 ] ;
my $ procprefix = uc $ structprefix ;
shift ;
2007-06-11 11:47:01 +00:00
# Convert name_of_call to NameOfCall.
sub name_to_ProcName {
my $ name = shift ;
my @ elems = split /_/ , $ name ;
@ elems = map ucfirst , @ elems ;
join "" , @ elems
}
# Read the input file (usually remote_protocol.x) and form an
# opinion about the name, args and return type of each RPC.
2008-12-04 22:03:24 +00:00
my ( $ name , $ ProcName , $ id , % calls , @ calls ) ;
2010-04-16 22:09:25 -04:00
# only generate a close method if -c was passed
if ( $ opt_c ) {
# REMOTE_PROC_CLOSE has no args or ret.
$ calls { close } = {
name = > "close" ,
ProcName = > "Close" ,
UC_NAME = > "CLOSE" ,
args = > "void" ,
ret = > "void" ,
} ;
}
2008-12-04 22:03:24 +00:00
2007-06-11 11:47:01 +00:00
while ( < > ) {
2010-04-16 22:09:25 -04:00
if ( /^struct ${structprefix}_(.*)_args/ ) {
2007-06-11 11:47:01 +00:00
$ name = $ 1 ;
$ ProcName = name_to_ProcName ( $ name ) ;
2010-04-16 22:09:25 -04:00
die "duplicate definition of ${structprefix}_${name}_args"
2007-06-11 11:47:01 +00:00
if exists $ calls { $ name } ;
$ calls { $ name } = {
name = > $ name ,
ProcName = > $ ProcName ,
UC_NAME = > uc $ name ,
2010-04-16 22:09:25 -04:00
args = > "${structprefix}_${name}_args" ,
2007-06-11 11:47:01 +00:00
ret = > "void" ,
} ;
2010-04-16 22:09:25 -04:00
} elsif ( /^struct ${structprefix}_(.*)_ret/ ) {
2007-06-11 11:47:01 +00:00
$ name = $ 1 ;
$ ProcName = name_to_ProcName ( $ name ) ;
if ( exists $ calls { $ name } ) {
2010-04-16 22:09:25 -04:00
$ calls { $ name } - > { ret } = "${structprefix}_${name}_ret" ;
2007-06-11 11:47:01 +00:00
} else {
$ calls { $ name } = {
name = > $ name ,
ProcName = > $ ProcName ,
UC_NAME = > uc $ name ,
args = > "void" ,
2010-04-16 22:09:25 -04:00
ret = > "${structprefix}_${name}_ret"
2007-06-11 11:47:01 +00:00
}
}
2010-04-16 22:09:25 -04:00
} elsif ( /^struct ${structprefix}_(.*)_msg/ ) {
2009-07-10 12:38:41 +01:00
$ name = $ 1 ;
$ ProcName = name_to_ProcName ( $ name ) ;
$ calls { $ name } = {
name = > $ name ,
ProcName = > $ ProcName ,
UC_NAME = > uc $ name ,
2010-04-16 22:09:25 -04:00
msg = > "${structprefix}_${name}_msg"
2009-07-10 12:38:41 +01:00
}
2010-04-16 22:09:25 -04:00
} elsif ( /^\s*${procprefix}_PROC_(.*?)\s+=\s+(\d+),?$/ ) {
2008-12-04 22:03:24 +00:00
$ name = lc $ 1 ;
$ id = $ 2 ;
$ ProcName = name_to_ProcName ( $ name ) ;
$ calls [ $ id ] = $ calls { $ name } ;
2007-06-11 11:47:01 +00:00
}
}
#----------------------------------------------------------------------
# Output
print << __EOF__ ;
/ * Automatically generated by remote_generate_stubs . pl .
* Do not edit this file . Any changes you make will be lost .
* /
__EOF__
# Debugging.
if ( $ opt_d ) {
my @ keys = sort ( keys % calls ) ;
foreach ( @ keys ) {
print "$_:\n" ;
2008-05-22 15:20:25 +00:00
print " name $calls{$_}->{name} ($calls{$_}->{ProcName})\n" ;
print " $calls{$_}->{args} -> $calls{$_}->{ret}\n" ;
2007-06-11 11:47:01 +00:00
}
}
# Prototypes for dispatch functions ("remote_dispatch_prototypes.h").
2008-12-04 22:03:24 +00:00
elsif ( $ opt_p ) {
2007-06-11 11:47:01 +00:00
my @ keys = sort ( keys % calls ) ;
foreach ( @ keys ) {
2009-07-10 12:38:41 +01:00
# Skip things which are REMOTE_MESSAGE
next if $ calls { $ _ } - > { msg } ;
2010-04-16 22:09:25 -04:00
print "static int ${structprefix}Dispatch$calls{$_}->{ProcName}(\n" ;
2008-12-04 22:16:40 +00:00
print " struct qemud_server *server,\n" ;
print " struct qemud_client *client,\n" ;
print " virConnectPtr conn,\n" ;
2009-09-30 12:29:20 +02:00
print " remote_message_header *hdr,\n" ;
2008-12-04 22:16:40 +00:00
print " remote_error *err,\n" ;
print " $calls{$_}->{args} *args,\n" ;
print " $calls{$_}->{ret} *ret);\n" ;
2007-06-11 11:47:01 +00:00
}
}
2008-12-04 22:03:24 +00:00
# Union of all arg types
# ("remote_dispatch_args.h").
elsif ( $ opt_a ) {
for ( $ id = 0 ; $ id <= $# calls ; $ id + + ) {
if ( defined $ calls [ $ id ] &&
2009-07-10 12:38:41 +01:00
! $ calls [ $ id ] - > { msg } &&
2008-12-04 22:03:24 +00:00
$ calls [ $ id ] - > { args } ne "void" ) {
print " $calls[$id]->{args} val_$calls[$id]->{args};\n" ;
2007-06-11 11:47:01 +00:00
}
}
}
2008-12-04 22:03:24 +00:00
# Union of all arg types
# ("remote_dispatch_ret.h").
elsif ( $ opt_r ) {
for ( $ id = 0 ; $ id <= $# calls ; $ id + + ) {
if ( defined $ calls [ $ id ] &&
2009-07-10 12:38:41 +01:00
! $ calls [ $ id ] - > { msg } &&
2008-12-04 22:03:24 +00:00
$ calls [ $ id ] - > { ret } ne "void" ) {
print " $calls[$id]->{ret} val_$calls[$id]->{ret};\n" ;
2007-06-11 11:47:01 +00:00
}
}
}
2008-12-04 22:03:24 +00:00
# Inside the switch statement, prepare the 'fn', 'args_filter', etc
# ("remote_dispatch_table.h").
elsif ( $ opt_t ) {
for ( $ id = 0 ; $ id <= $# calls ; $ id + + ) {
2009-07-10 12:38:41 +01:00
if ( defined $ calls [ $ id ] && ! $ calls [ $ id ] - > { msg } ) {
2008-12-04 22:03:24 +00:00
print "{ /* $calls[$id]->{ProcName} => $id */\n" ;
2010-04-16 22:09:25 -04:00
print " .fn = (dispatch_fn) ${structprefix}Dispatch$calls[$id]->{ProcName},\n" ;
2008-12-04 22:03:24 +00:00
if ( $ calls [ $ id ] - > { args } ne "void" ) {
print " .args_filter = (xdrproc_t) xdr_$calls[$id]->{args},\n" ;
} else {
print " .args_filter = (xdrproc_t) xdr_void,\n" ;
}
if ( $ calls [ $ id ] - > { ret } ne "void" ) {
print " .ret_filter = (xdrproc_t) xdr_$calls[$id]->{ret},\n" ;
} else {
print " .ret_filter = (xdrproc_t) xdr_void,\n" ;
}
print "},\n" ;
2007-06-11 11:47:01 +00:00
} else {
2009-07-10 12:38:41 +01:00
if ( $ calls [ $ id ] - > { msg } ) {
print "{ /* Async event $calls[$id]->{ProcName} => $id */\n" ;
} else {
print "{ /* (unused) => $id */\n" ;
}
2008-12-04 22:03:24 +00:00
print " .fn = NULL,\n" ;
print " .args_filter = (xdrproc_t) xdr_void,\n" ;
print " .ret_filter = (xdrproc_t) xdr_void,\n" ;
print "},\n" ;
2007-06-11 11:47:01 +00:00
}
}
}