#!/bin/sh
#
#       $NetBSD: makerumpif.sh,v 1.10 2016/01/26 23:21:18 pooka Exp $
#
# Copyright (c) 2009, 2015 Antti Kantee.  All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
# 1. Redistributions of source code must retain the above copyright
#    notice, this list of conditions and the following disclaimer.
# 2. Redistributions in binary form must reproduce the above copyright
#    notice, this list of conditions and the following disclaimer in the
#    documentation and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
# IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
# OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
# IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
# NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
# THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#

#
# This reads a rump component interface description and creates:
#  1: rump private prototypes for internal calls
#  2: public prototypes for calls outside of rump
#  3: public interface implementations which run the rump scheduler
#     and call the private interface
#

usage ()
{

       echo "usage: $0 spec"
       exit 1
}

boom ()
{

       echo $*
       exit 1
}

unset TOPDIR
if [ $# -eq 3 ]; then
       if [ $1 = '-R' ]; then
               TOPDIR=$2
       else
               usage
       fi
       shift; shift
fi
[ $# -ne 1 ] && usage

MYDIR=`pwd`
if [ -z "${TOPDIR}" ]; then
       while [ ! -f Makefile.rump  ]; do
               [ `pwd` = '/' ] && boom Could not find rump topdir.
               cd ..
       done
       TOPDIR="`pwd`"
fi
cd ${MYDIR}

sed -e '
:again
       /\\$/{
               N
               s/[     ]*\\\n[         ]*/ /
               b again
       }
' ${1} | awk -F\| -v topdir=${TOPDIR} '
function fileheaders(file, srcstr)
{
       printf("/*\t$NetBSD: makerumpif.sh,v 1.10 2016/01/26 23:21:18 pooka Exp $\t*/\n\n") > file
       printf("/*\n * Automatically generated.  DO NOT EDIT.\n") > file
       genstr = "$NetBSD: makerumpif.sh,v 1.10 2016/01/26 23:21:18 pooka Exp $"
       gsub("\\$", "", genstr)
       printf(" * from: %s\n", srcstr) > file
       printf(" * by:   %s\n", genstr) > file
       printf(" */\n") > file
}

function die(str)
{

       print str
       exit(1)
}

NR == 1 {
       sub(";[^\\$]*\\$", "")
       sub("\\$", "")
       fromstr = $0
       next
}

$1 == "NAME"{myname = $2;next}
$1 == "PUBHDR"{pubhdr = topdir "/" $2;print pubhdr;next}
$1 == "PRIVHDR"{privhdr = topdir "/" $2;print privhdr;next}
$1 == "WRAPPERS"{gencalls = topdir "/" $2;print gencalls;next}

/^;/{next}
/\\$/{sub("\\\n", "");getline nextline;$0 = $0 nextline}
/^[ \t]*$/{next}
{
       if (NF != 3 && NF != 4) {
               die("error: unexpected number of fields\n")
       }
       isweak = 0
       iscompat = 0
       if (NF == 4) {
               if ($4 == "WEAK") {
                       isweak = 1
               } else if (match($4, "COMPAT_")) {
                       iscompat = 1
                       compat = $4
               } else {
                       die("error: unexpected fourth field");
               }
       }
       if (!myname)
               die("name not specified");
       if (!pubhdr)
               die("public header not specified");
       if (!privhdr)
               die("private header not specified");
       if (!gencalls)
               die("wrapper file not specified");

       if (!once) {
               fileheaders(pubhdr, fromstr)
               fileheaders(privhdr, fromstr)
               fileheaders(gencalls, fromstr)
               once = 1

               pubfile = pubhdr
               sub(".*/", "", pubfile)

               privfile = privhdr
               sub(".*/", "", privfile)

               printf("\n") > pubhdr
               printf("\n") > privhdr

               printf("#ifndef _RUMP_PRIF_%s_H_\n", toupper(myname)) > privhdr
               printf("#define _RUMP_PRIF_%s_H_\n", toupper(myname)) > privhdr
               printf("\n") > privhdr

               printf("\n#include <sys/cdefs.h>\n") > gencalls
               printf("#include <sys/systm.h>\n\n") > gencalls
               printf("#include <rump-sys/kern.h>\n", privfile) > gencalls
               printf("#include <rump-sys/%s>\n\n", privfile) > gencalls
               printf("#include <rump/rump.h>\n") > gencalls
               printf("#include <rump/%s>\n\n", pubfile) > gencalls
               printf("void __dead rump_%s_unavailable(void);\n",      \
                   myname) > gencalls
               printf("void __dead\nrump_%s_unavailable(void)\n{\n",   \
                   myname) > gencalls
               printf("\n\tpanic(\"%s interface unavailable\");\n}\n", \
                   myname) > gencalls
       }

       funtype = $1
       sub("[ \t]*$", "", funtype)
       funname = $2
       sub("[ \t]*$", "", funname)
       funargs = $3
       sub("[ \t]*$", "", funargs)

       printf("%s rump_pub_%s(%s);\n", funtype, funname, funargs) > pubhdr
       printf("%s rump_%s(%s);\n", funtype, funname, funargs) > privhdr
       printf("typedef %s (*rump_%s_fn)(%s);\n",       \
           funtype, funname, funargs) > privhdr

       if (funtype == "void")
               voidret = 1
       else
               voidret = 0
       if (funargs == "void")
               voidarg = 1
       else
               voidarg = 0

       printf("\n") > gencalls
       if (iscompat) {
               printf("#ifdef %s\n", compat) > gencalls
       }

       printf("%s\nrump_pub_%s(", funtype, funname) > gencalls
       if (!voidarg) {
               narg = split(funargs, argv, ",")
               for (i = 1; i <= narg; i++) {
                       sub(" *", "", argv[i])
                       if (match(argv[i], "\\*$") != 0)
                               printf("%sarg%d", argv[i], i) > gencalls
                       else
                               printf("%s arg%d", argv[i], i) > gencalls
                       if (i != narg)
                               printf(", ") > gencalls
               }
       } else {
               narg = 0
               printf("void") > gencalls
       }
       printf(")\n{\n") > gencalls

       if (!voidret) {
               printf("\t%s rv;\n", funtype) > gencalls
       }
       printf("\n\trump_schedule();\n\t") > gencalls
       if (!voidret)
               printf("rv = ") > gencalls
       printf("rump_%s(", funname) > gencalls
       for (i = 1; i <= narg; i++) {
               printf("arg%i", i) > gencalls
               if (i < narg)
                       printf(", ") > gencalls
       }
       printf(");\n\trump_unschedule();\n") > gencalls
       if (!voidret)
               printf("\n\treturn rv;\n") > gencalls
       printf("}\n") > gencalls
       if (iscompat) {
               printf("#else\n") > gencalls
               printf("__strong_alias(rump_pub_%s,rump_%s_unavailable);\n", \
                   funname, myname) > gencalls
               printf("#endif /* %s */\n", compat) > gencalls
       }
       if (isweak)
               printf("__weak_alias(rump_%s,rump_%s_unavailable);\n", \
                   funname, myname) > gencalls
}

END { printf("\n#endif /* _RUMP_PRIF_%s_H_ */\n", toupper(myname)) > privhdr }'