mirror of
git://sourceware.org/git/glibc.git
synced 2025-01-18 12:16:13 +08:00
ce460d04a5
changed from [USE_TLS && HAVE___THREAD]. * sysdeps/i386/dl-machine.h (elf_machine_type_class, elf_machine_rel): Disable TLS relocs if [RTLD_BOOTSTRAP && !USE___THREAD]. * sysdeps/x86_64/dl-machine.h (elf_machine_type_class, elf_machine_rela): Likewise. * sysdeps/sh/dl-machine.h (elf_machine_type_class, elf_machine_rela): Likewise. * include/link.h (struct link_map): Remove member l_tls_tp_initialized. * elf/rtld.c (_dl_start_final, dl_main): Don't use it. (_dl_start): Conditionalize PT_TLS check on [USE___THREAD]. * sysdeps/i386/dl-tls.h (__TLS_GET_ADDR): Use ___tls_get_addr_internal instead of ___tls_get_addr. (___tls_get_addr_internal): Add attribute_hidden to decl. * sysdeps/generic/ldsodefs.h (struct rtld_global): New variable _dl_error_catch_tsd. * elf/rtld.c (startup_error_tsd): New function. (dl_main): Point _dl_error_catch_tsd at that. * elf/dl-error.c: Don't use libc-tsd.h for DL_ERROR, use new function pointer instead. * elf/dl-tsd.c: New file. * elf/Makefile (routines): Add it. 2002-10-07 Roland McGrath <roland@redhat.com> * elf/dl-misc.c (_dl_debug_vdprintf): Use INTERNAL_SYSCALL macro for writev if it's available. Otherwise if [RTLD_PRIVATE_ERRNO] then take _dl_load_lock around calling __writev. * sysdeps/unix/sysv/linux/i386/sysdep.h (INTERNAL_SYSCALL): New macro. (INLINE_SYSCALL): Use that. * sysdeps/generic/dl-sysdep.h: New file. * sysdeps/mach/hurd/dl-sysdep.h: New file. * sysdeps/generic/ldsodefs.h: Include <dl-sysdep.h>. * include/errno.h [IS_IN_rtld]: Include <dl-sysdep.h> to define ... [RTLD_PRIVATE_ERRNO]: Use a hidden global variable for errno and access it directly. * elf/dl-minimal.c (__errno_location): Removed. * sysdeps/unix/i386/sysdep.S (__syscall_errno) [RTLD_PRIVATE_ERRNO]: Use GOTOFF access for errno. * sysdeps/unix/sysv/linux/i386/sysdep.h [RTLD_PRIVATE_ERRNO] (SYSCALL_ERROR_HANDLER): Likewise. * sysdeps/unix/x86_64/sysdep.S (__syscall_errno) [RTLD_PRIVATE_ERRNO]: Use PC-relative access for errno. * sysdeps/unix/sysv/linux/x86_64/sysdep.h [RTLD_PRIVATE_ERRNO] (SYSCALL_ERROR_HANDLER): Likewise. * include/tls.h: New file. (USE___THREAD): New macro. Define to 1 under [USE_TLS && HAVE___THREAD] and only when compiling libc or libpthread. * sysdeps/unix/sysv/linux/i386/sysdep.h [USE___THREAD]: Conditional changed from [USE_TLS && HAVE___THREAD]. * sysdeps/unix/sysv/linux/x86_64/sysdep.h: Likewise. * sysdeps/unix/i386/sysdep.S: Likewise. * sysdeps/unix/x86_64/sysdep.S: Likewise. * include/errno.h: Likewise. * include/netdb.h: Likewise. * include/resolv.h: Likewise. * sysdeps/generic/errno.c: New file. * csu/Makefile (aux): New variable, list errno. * sysdeps/unix/sysv/linux/i386/sysdep.S (errno, _errno): Remove defns. * sysdeps/unix/sysv/linux/m68k/sysdep.S: Likewise. * sysdeps/unix/sysv/linux/x86_64/sysdep.S: Likewise. * sysdeps/unix/sysv/linux/s390/s390-64/sysdep.S: Likewise. * sysdeps/unix/sysv/linux/s390/s390-32/sysdep.S: Likewise. * sysdeps/unix/sysv/linux/arm/sysdep.S: Likewise. * sysdeps/unix/sysv/linux/cris/sysdep.S: Likewise. * sysdeps/unix/sysv/linux/hppa/sysdep.c: Likewise. * sysdeps/unix/sysv/linux/ia64/sysdep.S: Likewise. * sysdeps/unix/sysv/linux/powerpc/sysdep.c: Likewise. * sysdeps/unix/sysv/linux/sparc/sysdep.S: Likewise. * sysdeps/unix/sysv/linux/sh/sysdep.S: Likewise. * sysdeps/unix/alpha/sysdep.S: Likewise. * sysdeps/generic/start.c: Likewise. * sysdeps/unix/start.c: Likewise. * sysdeps/unix/arm/start.c: Likewise. * sysdeps/unix/bsd/ultrix4/mips/start.S: Likewise. * sysdeps/unix/sparc/start.c: Likewise. * sysdeps/unix/sysv/irix4/start.c: Likewise. * sysdeps/unix/sysv/linux/mips/sysdep.S: File removed. * manual/search.texi (Tree Search Function, Hash Search Function): Mention search.h clearly. 2002-10-05 Roland McGrath <roland@redhat.com> * elf/dl-fxstat64.c: File removed. * elf/dl-xstat64.c: File removed. * elf/Makefile (rtld-routines): Remove them. * sysdeps/unix/sysv/linux/xstat64.c: Remove RTLD_STAT64 conditionals. Instead, use strong_alias instead of versioned_symbol in the !SHLIB_COMPAT case. * sysdeps/unix/sysv/linux/fxstat64.c: Likewise. * sysdeps/unix/sysv/linux/lxstat64.c: Likewise. * include/shlib-compat.h (SHLIB_COMPAT): Require that IS_IN_##lib be defined nonzero. [! NOT_IN_libc] (IS_IN_libc): Define it. * cppflags-iterator.mk (CPPFLAGS-$(cpp-src)): Use -Dx=1 not just -Dx. * elf/Makefile (CPPFLAGS-.os): Likewise. * sunrpc/rpc_main.c (main): Don't declare with noreturn attribute. Return the status instead of calling exit. * Makeconfig (CFLAGS): Prepend -std=gnu99. * Makerules (+make-deps): Use $(CFLAGS) only for .c sources. Remove superfluous rm command, whose @ plus make bugs hid all these commands from the make output. * include/stubs-prologue.h: New file. Give #error under #ifdef _LIBC. * Makefile ($(inst_includedir)/gnu/stubs.h): Depend on it. Use that file's contents instead of literal echo's for the prologue. * include/features.h: Include <gnu/stubs.h> unconditionally. * include/gnu/stubs.h: New file. 2002-09-30 Roland McGrath <roland@redhat.com> * elf/rtld-Rules: New file. * elf/Makefile ($(objpfx)librtld.map, $(objpfx)librtld.mk, $(objpfx)rtld-libc.a): New targets. (generated): Add them. (reloc-link): Remove -o $@ from the variable. ($(objpfx)dl-allobjs.os): Add -o $@ after $(reloc-link). (distribute): Add rtld-Rules. (CPPFLAGS-.os): Define this instead of CFLAGS-.os. * Makerules ($(+sysdir_pfx)sysd-rules): Emit rules for rtld-% targets. (common-mostlyclean, common-clean): Clean up rtld-* files. * sysdeps/unix/make-syscalls.sh: Add rtld-*.os target name to rules.
1466 lines
36 KiB
C
1466 lines
36 KiB
C
/*
|
|
* Sun RPC is a product of Sun Microsystems, Inc. and is provided for
|
|
* unrestricted use provided that this legend is included on all tape
|
|
* media and as a part of the software program in whole or part. Users
|
|
* may copy or modify Sun RPC without charge, but are not authorized
|
|
* to license or distribute it to anyone else except as part of a product or
|
|
* program developed by the user or with the express written consent of
|
|
* Sun Microsystems, Inc.
|
|
*
|
|
* SUN RPC IS PROVIDED AS IS WITH NO WARRANTIES OF ANY KIND INCLUDING THE
|
|
* WARRANTIES OF DESIGN, MERCHANTIBILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE, OR ARISING FROM A COURSE OF DEALING, USAGE OR TRADE PRACTICE.
|
|
*
|
|
* Sun RPC is provided with no support and without any obligation on the
|
|
* part of Sun Microsystems, Inc. to assist in its use, correction,
|
|
* modification or enhancement.
|
|
*
|
|
* SUN MICROSYSTEMS, INC. SHALL HAVE NO LIABILITY WITH RESPECT TO THE
|
|
* INFRINGEMENT OF COPYRIGHTS, TRADE SECRETS OR ANY PATENTS BY SUN RPC
|
|
* OR ANY PART THEREOF.
|
|
*
|
|
* In no event will Sun Microsystems, Inc. be liable for any lost revenue
|
|
* or profits or other special, indirect and consequential damages, even if
|
|
* Sun has been advised of the possibility of such damages.
|
|
*
|
|
* Sun Microsystems, Inc.
|
|
* 2550 Garcia Avenue
|
|
* Mountain View, California 94043
|
|
*/
|
|
|
|
/*
|
|
* From @(#)rpc_main.c 1.30 89/03/30 (C) 1987 SMI;
|
|
*/
|
|
const char main_rcsid[] =
|
|
"$Id$";
|
|
|
|
/*
|
|
* rpc_main.c, Top level of the RPC protocol compiler.
|
|
*/
|
|
|
|
#include <errno.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <libintl.h>
|
|
#include <ctype.h>
|
|
#include <sys/types.h>
|
|
#include <sys/param.h>
|
|
#include <sys/file.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/wait.h>
|
|
#include "rpc_parse.h"
|
|
#include "rpc_util.h"
|
|
#include "rpc_scan.h"
|
|
#include "proto.h"
|
|
|
|
#define EXTEND 1 /* alias for TRUE */
|
|
#define DONT_EXTEND 0 /* alias for FALSE */
|
|
|
|
struct commandline
|
|
{
|
|
int cflag; /* xdr C routines */
|
|
int hflag; /* header file */
|
|
int lflag; /* client side stubs */
|
|
int mflag; /* server side stubs */
|
|
int nflag; /* netid flag */
|
|
int sflag; /* server stubs for the given transport */
|
|
int tflag; /* dispatch Table file */
|
|
int Ssflag; /* produce server sample code */
|
|
int Scflag; /* produce client sample code */
|
|
int makefileflag; /* Generate a template Makefile */
|
|
const char *infile; /* input module name */
|
|
const char *outfile; /* output module name */
|
|
};
|
|
|
|
|
|
static const char *cmdname;
|
|
|
|
#define SVR4_CPP "/usr/ccs/lib/cpp"
|
|
#define SUNOS_CPP "/lib/cpp"
|
|
|
|
static const char *svcclosetime = "120";
|
|
static int cppDefined; /* explicit path for C preprocessor */
|
|
static const char *CPP = SUNOS_CPP;
|
|
static const char CPPFLAGS[] = "-C";
|
|
static char *pathbuf;
|
|
static int cpp_pid;
|
|
static const char *allv[] =
|
|
{
|
|
"rpcgen", "-s", "udp", "-s", "tcp"
|
|
};
|
|
static int allc = sizeof (allv) / sizeof (allv[0]);
|
|
static const char *allnv[] =
|
|
{
|
|
"rpcgen", "-s", "netpath",
|
|
};
|
|
static int allnc = sizeof (allnv) / sizeof (allnv[0]);
|
|
|
|
/*
|
|
* machinations for handling expanding argument list
|
|
*/
|
|
static void addarg (const char *); /* add another argument to the list */
|
|
static void putarg (int, const char *); /* put argument at specified location */
|
|
static void clear_args (void); /* clear argument list */
|
|
static void checkfiles (const char *, const char *);
|
|
/* check if out file already exists */
|
|
|
|
static void clear_args (void);
|
|
static char *extendfile (const char *file, const char *ext);
|
|
static void open_output (const char *infile, const char *outfile);
|
|
static void add_warning (void);
|
|
static void clear_args (void);
|
|
static void find_cpp (void);
|
|
static void open_input (const char *infile, const char *define);
|
|
static int check_nettype (const char *name, const char *list_to_check[]);
|
|
static void c_output (const char *infile, const char *define,
|
|
int extend, const char *outfile);
|
|
static void h_output (const char *infile, const char *define,
|
|
int extend, const char *outfile);
|
|
static void s_output (int argc, const char *argv[], const char *infile,
|
|
const char *define, int extend,
|
|
const char *outfile, int nomain, int netflag);
|
|
static void l_output (const char *infile, const char *define,
|
|
int extend, const char *outfile);
|
|
static void t_output (const char *infile, const char *define,
|
|
int extend, const char *outfile);
|
|
static void svc_output (const char *infile, const char *define,
|
|
int extend, const char *outfile);
|
|
static void clnt_output (const char *infile, const char *define,
|
|
int extend, const char *outfile);
|
|
static void mkfile_output (struct commandline *cmd);
|
|
static int do_registers (int argc, const char *argv[]);
|
|
static void addarg (const char *cp);
|
|
static void putarg (int whereto, const char *cp);
|
|
static void checkfiles (const char *infile, const char *outfile);
|
|
static int parseargs (int argc, const char *argv[], struct commandline *cmd);
|
|
static void usage (void) __attribute__ ((noreturn));
|
|
static void options_usage (void) __attribute__ ((noreturn));
|
|
static void c_initialize (void);
|
|
static char *generate_guard (const char *pathname);
|
|
|
|
|
|
#define ARGLISTLEN 20
|
|
#define FIXEDARGS 2
|
|
|
|
static const char *arglist[ARGLISTLEN];
|
|
static int argcount = FIXEDARGS;
|
|
|
|
|
|
int nonfatalerrors; /* errors */
|
|
int inetdflag /* = 1 */ ; /* Support for inetd *//* is now the default */
|
|
int pmflag; /* Support for port monitors */
|
|
int logflag; /* Use syslog instead of fprintf for errors */
|
|
int tblflag; /* Support for dispatch table file */
|
|
int mtflag; /* Support for MT */
|
|
|
|
#define INLINE 3
|
|
/*length at which to start doing an inline */
|
|
|
|
int inlineflag = INLINE; /* length at which to start doing an inline. 3 = default
|
|
if 0, no xdr_inline code */
|
|
|
|
int indefinitewait; /* If started by port monitors, hang till it wants */
|
|
int exitnow; /* If started by port monitors, exit after the call */
|
|
int timerflag; /* TRUE if !indefinite && !exitnow */
|
|
int newstyle; /* newstyle of passing arguments (by value) */
|
|
#ifdef __GNU_LIBRARY__
|
|
int Cflag = 1; /* ANSI C syntax */
|
|
#else
|
|
int Cflag; /* ANSI C/C++ syntax */
|
|
#endif
|
|
int CCflag; /* C++ files */
|
|
static int allfiles; /* generate all files */
|
|
#ifdef __GNU_LIBRARY__
|
|
int tirpcflag; /* generating code for tirpc, by default */
|
|
#else
|
|
int tirpcflag = 1; /* generating code for tirpc, by default */
|
|
#endif
|
|
xdrfunc *xdrfunc_head; /* xdr function list */
|
|
xdrfunc *xdrfunc_tail; /* xdr function list */
|
|
|
|
int
|
|
main (int argc, const char *argv[])
|
|
{
|
|
struct commandline cmd;
|
|
|
|
(void) memset ((char *) &cmd, 0, sizeof (struct commandline));
|
|
clear_args ();
|
|
if (!parseargs (argc, argv, &cmd))
|
|
usage ();
|
|
|
|
if (cmd.cflag || cmd.hflag || cmd.lflag || cmd.tflag || cmd.sflag ||
|
|
cmd.mflag || cmd.nflag || cmd.Ssflag || cmd.Scflag)
|
|
{
|
|
checkfiles (cmd.infile, cmd.outfile);
|
|
}
|
|
else
|
|
checkfiles (cmd.infile, NULL);
|
|
|
|
if (cmd.cflag)
|
|
c_output (cmd.infile, "-DRPC_XDR", DONT_EXTEND, cmd.outfile);
|
|
else if (cmd.hflag)
|
|
h_output (cmd.infile, "-DRPC_HDR", DONT_EXTEND, cmd.outfile);
|
|
else if (cmd.lflag)
|
|
l_output (cmd.infile, "-DRPC_CLNT", DONT_EXTEND, cmd.outfile);
|
|
else if (cmd.sflag || cmd.mflag || (cmd.nflag))
|
|
s_output (argc, argv, cmd.infile, "-DRPC_SVC", DONT_EXTEND,
|
|
cmd.outfile, cmd.mflag, cmd.nflag);
|
|
else if (cmd.tflag)
|
|
t_output (cmd.infile, "-DRPC_TBL", DONT_EXTEND, cmd.outfile);
|
|
else if (cmd.Ssflag)
|
|
svc_output (cmd.infile, "-DRPC_SERVER", DONT_EXTEND, cmd.outfile);
|
|
else if (cmd.Scflag)
|
|
clnt_output (cmd.infile, "-DRPC_CLIENT", DONT_EXTEND, cmd.outfile);
|
|
else if (cmd.makefileflag)
|
|
mkfile_output (&cmd);
|
|
else
|
|
{
|
|
/* the rescans are required, since cpp may effect input */
|
|
c_output (cmd.infile, "-DRPC_XDR", EXTEND, "_xdr.c");
|
|
reinitialize ();
|
|
h_output (cmd.infile, "-DRPC_HDR", EXTEND, ".h");
|
|
reinitialize ();
|
|
l_output (cmd.infile, "-DRPC_CLNT", EXTEND, "_clnt.c");
|
|
reinitialize ();
|
|
if (inetdflag || !tirpcflag)
|
|
s_output (allc, allv, cmd.infile, "-DRPC_SVC", EXTEND,
|
|
"_svc.c", cmd.mflag, cmd.nflag);
|
|
else
|
|
s_output (allnc, allnv, cmd.infile, "-DRPC_SVC",
|
|
EXTEND, "_svc.c", cmd.mflag, cmd.nflag);
|
|
if (tblflag)
|
|
{
|
|
reinitialize ();
|
|
t_output (cmd.infile, "-DRPC_TBL", EXTEND, "_tbl.i");
|
|
}
|
|
if (allfiles)
|
|
{
|
|
reinitialize ();
|
|
svc_output (cmd.infile, "-DRPC_SERVER", EXTEND, "_server.c");
|
|
reinitialize ();
|
|
clnt_output (cmd.infile, "-DRPC_CLIENT", EXTEND, "_client.c");
|
|
}
|
|
if (allfiles || (cmd.makefileflag == 1))
|
|
{
|
|
reinitialize ();
|
|
mkfile_output (&cmd);
|
|
}
|
|
}
|
|
|
|
return nonfatalerrors;
|
|
}
|
|
|
|
/*
|
|
* add extension to filename
|
|
*/
|
|
static char *
|
|
extendfile (const char *file, const char *ext)
|
|
{
|
|
char *res;
|
|
const char *p;
|
|
|
|
res = alloc (strlen (file) + strlen (ext) + 1);
|
|
if (res == NULL)
|
|
abort ();
|
|
p = strrchr (file, '.');
|
|
if (p == NULL)
|
|
p = file + strlen (file);
|
|
strcpy (res, file);
|
|
strcpy (res + (p - file), ext);
|
|
return res;
|
|
}
|
|
|
|
/*
|
|
* Open output file with given extension
|
|
*/
|
|
static void
|
|
open_output (const char *infile, const char *outfile)
|
|
{
|
|
if (outfile == NULL)
|
|
{
|
|
fout = stdout;
|
|
return;
|
|
}
|
|
|
|
if (infile != NULL && streq (outfile, infile))
|
|
{
|
|
fprintf (stderr, _ ("%s: output would overwrite %s\n"), cmdname,
|
|
infile);
|
|
crash ();
|
|
}
|
|
fout = fopen (outfile, "w");
|
|
if (fout == NULL)
|
|
{
|
|
fprintf (stderr, _ ("%s: unable to open %s: %m\n"), cmdname, outfile);
|
|
crash ();
|
|
}
|
|
record_open (outfile);
|
|
}
|
|
|
|
/* Close the output file and check for write errors. */
|
|
static void
|
|
close_output (const char *outfile)
|
|
{
|
|
if (fclose (fout) == EOF)
|
|
{
|
|
fprintf (stderr, _("%s: while writing output %s: %m"), cmdname,
|
|
outfile ?: "<stdout>");
|
|
crash ();
|
|
}
|
|
}
|
|
|
|
static void
|
|
add_warning (void)
|
|
{
|
|
fprintf (fout, "/*\n");
|
|
fprintf (fout, " * Please do not edit this file.\n");
|
|
fprintf (fout, " * It was generated using rpcgen.\n");
|
|
fprintf (fout, " */\n\n");
|
|
}
|
|
|
|
/* clear list of arguments */
|
|
static void
|
|
clear_args (void)
|
|
{
|
|
int i;
|
|
for (i = FIXEDARGS; i < ARGLISTLEN; ++i)
|
|
arglist[i] = NULL;
|
|
argcount = FIXEDARGS;
|
|
}
|
|
|
|
/* make sure that a CPP exists */
|
|
static void
|
|
find_cpp (void)
|
|
{
|
|
struct stat buf;
|
|
|
|
if (stat (CPP, &buf) < 0)
|
|
{ /* /lib/cpp or explicit cpp does not exist */
|
|
if (cppDefined)
|
|
{
|
|
fprintf (stderr, _ ("cannot find C preprocessor: %s \n"), CPP);
|
|
crash ();
|
|
}
|
|
else
|
|
{ /* try the other one */
|
|
CPP = SVR4_CPP;
|
|
if (stat (CPP, &buf) < 0)
|
|
{ /* can't find any cpp */
|
|
fputs (_ ("cannot find any C preprocessor (cpp)\n"), stdout);
|
|
crash ();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Open input file with given define for C-preprocessor
|
|
*/
|
|
static void
|
|
open_input (const char *infile, const char *define)
|
|
{
|
|
int pd[2];
|
|
|
|
infilename = (infile == NULL) ? "<stdin>" : infile;
|
|
if (pipe (pd) != 0)
|
|
{
|
|
perror ("pipe");
|
|
exit (1);
|
|
}
|
|
cpp_pid = fork ();
|
|
switch (cpp_pid)
|
|
{
|
|
case 0:
|
|
find_cpp ();
|
|
putarg (0, CPP);
|
|
putarg (1, CPPFLAGS);
|
|
addarg (define);
|
|
if (infile)
|
|
addarg (infile);
|
|
addarg ((char *) NULL);
|
|
close (1);
|
|
dup2 (pd[1], 1);
|
|
close (pd[0]);
|
|
execv (arglist[0], (char **) arglist);
|
|
perror ("execv");
|
|
exit (1);
|
|
case -1:
|
|
perror ("fork");
|
|
exit (1);
|
|
}
|
|
close (pd[1]);
|
|
fin = fdopen (pd[0], "r");
|
|
if (fin == NULL)
|
|
{
|
|
fprintf (stderr, "%s: ", cmdname);
|
|
perror (infilename);
|
|
crash ();
|
|
}
|
|
}
|
|
|
|
/* Close the connection to the C-preprocessor and check for successfull
|
|
termination. */
|
|
static void
|
|
close_input (void)
|
|
{
|
|
int status;
|
|
|
|
fclose (fin);
|
|
/* Check the termination status. */
|
|
if (waitpid (cpp_pid, &status, 0) < 0)
|
|
{
|
|
perror ("waitpid");
|
|
crash ();
|
|
}
|
|
if (WIFSIGNALED (status) || WEXITSTATUS (status) != 0)
|
|
{
|
|
if (WIFSIGNALED (status))
|
|
fprintf (stderr, _("%s: C preprocessor failed with signal %d\n"),
|
|
cmdname, WTERMSIG (status));
|
|
else
|
|
fprintf (stderr, _("%s: C preprocessor failed with exit code %d\n"),
|
|
cmdname, WEXITSTATUS (status));
|
|
crash ();
|
|
}
|
|
}
|
|
|
|
/* valid tirpc nettypes */
|
|
static const char *valid_ti_nettypes[] =
|
|
{
|
|
"netpath",
|
|
"visible",
|
|
"circuit_v",
|
|
"datagram_v",
|
|
"circuit_n",
|
|
"datagram_n",
|
|
"udp",
|
|
"tcp",
|
|
"raw",
|
|
NULL
|
|
};
|
|
|
|
/* valid inetd nettypes */
|
|
static const char *valid_i_nettypes[] =
|
|
{
|
|
"udp",
|
|
"tcp",
|
|
NULL
|
|
};
|
|
|
|
static int
|
|
check_nettype (const char *name, const char *list_to_check[])
|
|
{
|
|
int i;
|
|
for (i = 0; list_to_check[i] != NULL; i++)
|
|
{
|
|
if (strcmp (name, list_to_check[i]) == 0)
|
|
{
|
|
return 1;
|
|
}
|
|
}
|
|
fprintf (stderr, _ ("illegal nettype :`%s'\n"), name);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Compile into an XDR routine output file
|
|
*/
|
|
|
|
static void
|
|
c_output (const char *infile, const char *define, int extend,
|
|
const char *outfile)
|
|
{
|
|
definition *def;
|
|
char *include;
|
|
const char *outfilename;
|
|
long tell;
|
|
|
|
c_initialize ();
|
|
open_input (infile, define);
|
|
outfilename = extend ? extendfile (infile, outfile) : outfile;
|
|
open_output (infile, outfilename);
|
|
add_warning ();
|
|
if (infile && (include = extendfile (infile, ".h")))
|
|
{
|
|
fprintf (fout, "#include \"%s\"\n", include);
|
|
free (include);
|
|
/* .h file already contains rpc/rpc.h */
|
|
}
|
|
else
|
|
fprintf (fout, "#include <rpc/rpc.h>\n");
|
|
tell = ftell (fout);
|
|
while ((def = get_definition ()) != NULL)
|
|
emit (def);
|
|
|
|
if (extend && tell == ftell (fout))
|
|
unlink (outfilename);
|
|
close_input ();
|
|
close_output (outfilename);
|
|
}
|
|
|
|
void
|
|
c_initialize (void)
|
|
{
|
|
|
|
/* add all the starting basic types */
|
|
|
|
add_type (1, "int");
|
|
add_type (1, "long");
|
|
add_type (1, "short");
|
|
add_type (1, "bool");
|
|
|
|
add_type (1, "u_int");
|
|
add_type (1, "u_long");
|
|
add_type (1, "u_short");
|
|
|
|
}
|
|
|
|
char rpcgen_table_dcl[] = "struct rpcgen_table {\n\
|
|
char *(*proc)();\n\
|
|
xdrproc_t xdr_arg;\n\
|
|
unsigned len_arg;\n\
|
|
xdrproc_t xdr_res;\n\
|
|
unsigned len_res;\n\
|
|
};\n";
|
|
|
|
|
|
static char *
|
|
generate_guard (const char *pathname)
|
|
{
|
|
const char *filename;
|
|
char *guard, *tmp;
|
|
|
|
filename = strrchr (pathname, '/'); /* find last component */
|
|
filename = ((filename == NULL) ? pathname : filename + 1);
|
|
guard = strdup (filename);
|
|
/* convert to upper case */
|
|
tmp = guard;
|
|
while (*tmp)
|
|
{
|
|
if (islower (*tmp))
|
|
*tmp = toupper (*tmp);
|
|
tmp++;
|
|
}
|
|
|
|
guard = extendfile (guard, "_H_RPCGEN");
|
|
return guard;
|
|
}
|
|
|
|
/*
|
|
* Compile into an XDR header file
|
|
*/
|
|
|
|
|
|
static void
|
|
h_output (const char *infile, const char *define, int extend,
|
|
const char *outfile)
|
|
{
|
|
xdrfunc *xdrfuncp;
|
|
definition *def;
|
|
const char *ifilename;
|
|
const char *outfilename;
|
|
long tell;
|
|
char *guard;
|
|
list *l;
|
|
|
|
open_input (infile, define);
|
|
outfilename = extend ? extendfile (infile, outfile) : outfile;
|
|
open_output (infile, outfilename);
|
|
add_warning ();
|
|
ifilename = (infile == NULL) ? "STDIN" : infile;
|
|
guard = generate_guard (outfilename ? outfilename : ifilename);
|
|
|
|
fprintf (fout, "#ifndef _%s\n#define _%s\n\n", guard,
|
|
guard);
|
|
|
|
fprintf (fout, "#include <rpc/rpc.h>\n\n");
|
|
|
|
if (mtflag)
|
|
{
|
|
fprintf (fout, "#include <pthread.h>\n");
|
|
}
|
|
|
|
/* put the C++ support */
|
|
if (Cflag && !CCflag)
|
|
{
|
|
fprintf (fout, "\n#ifdef __cplusplus\n");
|
|
fprintf (fout, "extern \"C\" {\n");
|
|
fprintf (fout, "#endif\n\n");
|
|
}
|
|
|
|
tell = ftell (fout);
|
|
/* print data definitions */
|
|
while ((def = get_definition ()) != NULL)
|
|
{
|
|
print_datadef (def);
|
|
}
|
|
|
|
/* print function declarations.
|
|
Do this after data definitions because they might be used as
|
|
arguments for functions */
|
|
for (l = defined; l != NULL; l = l->next)
|
|
{
|
|
print_funcdef (l->val);
|
|
}
|
|
/* Now print all xdr func declarations */
|
|
if (xdrfunc_head != NULL)
|
|
{
|
|
fprintf (fout, "\n/* the xdr functions */\n");
|
|
if (CCflag)
|
|
{
|
|
fprintf (fout, "\n#ifdef __cplusplus\n");
|
|
fprintf (fout, "extern \"C\" {\n");
|
|
fprintf (fout, "#endif\n");
|
|
}
|
|
if (!Cflag)
|
|
{
|
|
xdrfuncp = xdrfunc_head;
|
|
while (xdrfuncp != NULL)
|
|
{
|
|
print_xdr_func_def (xdrfuncp->name,
|
|
xdrfuncp->pointerp, 2);
|
|
xdrfuncp = xdrfuncp->next;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int i;
|
|
|
|
for (i = 1; i < 3; ++i)
|
|
{
|
|
if (i == 1)
|
|
fprintf (fout, "\n#if defined(__STDC__) || defined(__cplusplus)\n");
|
|
else
|
|
fprintf (fout, "\n#else /* K&R C */\n");
|
|
|
|
xdrfuncp = xdrfunc_head;
|
|
while (xdrfuncp != NULL)
|
|
{
|
|
print_xdr_func_def (xdrfuncp->name,
|
|
xdrfuncp->pointerp, i);
|
|
xdrfuncp = xdrfuncp->next;
|
|
}
|
|
}
|
|
fprintf (fout, "\n#endif /* K&R C */\n");
|
|
}
|
|
}
|
|
|
|
if (extend && tell == ftell (fout))
|
|
{
|
|
unlink (outfilename);
|
|
}
|
|
else if (tblflag)
|
|
{
|
|
fprintf (fout, rpcgen_table_dcl);
|
|
}
|
|
|
|
if (Cflag)
|
|
{
|
|
fprintf (fout, "\n#ifdef __cplusplus\n");
|
|
fprintf (fout, "}\n");
|
|
fprintf (fout, "#endif\n");
|
|
}
|
|
|
|
fprintf (fout, "\n#endif /* !_%s */\n", guard);
|
|
close_input ();
|
|
close_output (outfilename);
|
|
}
|
|
|
|
/*
|
|
* Compile into an RPC service
|
|
*/
|
|
static void
|
|
s_output (int argc, const char *argv[], const char *infile, const char *define,
|
|
int extend, const char *outfile, int nomain, int netflag)
|
|
{
|
|
char *include;
|
|
definition *def;
|
|
int foundprogram = 0;
|
|
const char *outfilename;
|
|
|
|
open_input (infile, define);
|
|
outfilename = extend ? extendfile (infile, outfile) : outfile;
|
|
open_output (infile, outfilename);
|
|
add_warning ();
|
|
if (infile && (include = extendfile (infile, ".h")))
|
|
{
|
|
fprintf (fout, "#include \"%s\"\n", include);
|
|
free (include);
|
|
}
|
|
else
|
|
fprintf (fout, "#include <rpc/rpc.h>\n");
|
|
|
|
fprintf (fout, "#include <stdio.h>\n");
|
|
fprintf (fout, "#include <stdlib.h>\n");
|
|
if (Cflag)
|
|
{
|
|
fprintf (fout, "#include <rpc/pmap_clnt.h>\n");
|
|
fprintf (fout, "#include <string.h>\n");
|
|
}
|
|
if (strcmp (svcclosetime, "-1") == 0)
|
|
indefinitewait = 1;
|
|
else if (strcmp (svcclosetime, "0") == 0)
|
|
exitnow = 1;
|
|
else if (inetdflag || pmflag)
|
|
{
|
|
fprintf (fout, "#include <signal.h>\n");
|
|
timerflag = 1;
|
|
}
|
|
|
|
if (!tirpcflag && inetdflag)
|
|
#ifdef __GNU_LIBRARY__
|
|
fprintf (fout, "#include <sys/ioctl.h> /* ioctl, TIOCNOTTY */\n");
|
|
#else
|
|
fprintf (fout, "#include <sys/ttycom.h>/* TIOCNOTTY */\n");
|
|
#endif
|
|
if (Cflag && (inetdflag || pmflag))
|
|
{
|
|
#ifdef __GNU_LIBRARY__
|
|
fprintf (fout, "#include <sys/types.h> /* open */\n");
|
|
fprintf (fout, "#include <sys/stat.h> /* open */\n");
|
|
fprintf (fout, "#include <fcntl.h> /* open */\n");
|
|
fprintf (fout, "#include <unistd.h> /* getdtablesize */\n");
|
|
#else
|
|
fprintf (fout, "#ifdef __cplusplus\n");
|
|
fprintf (fout, "#include <sysent.h> /* getdtablesize, open */\n");
|
|
fprintf (fout, "#endif /* __cplusplus */\n");
|
|
if (tirpcflag)
|
|
fprintf (fout, "#include <unistd.h> /* setsid */\n");
|
|
#endif
|
|
}
|
|
#ifdef __GNU_LIBRARY__
|
|
if (tirpcflag && !(Cflag && (inetdflag || pmflag)))
|
|
#else
|
|
if (tirpcflag)
|
|
#endif
|
|
fprintf (fout, "#include <sys/types.h>\n");
|
|
|
|
fprintf (fout, "#include <memory.h>\n");
|
|
#ifndef __GNU_LIBRARY__
|
|
fprintf (fout, "#include <stropts.h>\n");
|
|
#endif
|
|
if (inetdflag || !tirpcflag)
|
|
{
|
|
fprintf (fout, "#include <sys/socket.h>\n");
|
|
fprintf (fout, "#include <netinet/in.h>\n");
|
|
}
|
|
|
|
if ((netflag || pmflag) && tirpcflag && !nomain)
|
|
{
|
|
fprintf (fout, "#include <netconfig.h>\n");
|
|
}
|
|
if ( /*timerflag && */ tirpcflag)
|
|
fprintf (fout, "#include <sys/resource.h> /* rlimit */\n");
|
|
if (logflag || inetdflag || pmflag)
|
|
{
|
|
#ifdef __GNU_LIBRARY__
|
|
fprintf (fout, "#include <syslog.h>\n");
|
|
#else
|
|
fprintf (fout, "#ifdef SYSLOG\n");
|
|
fprintf (fout, "#include <syslog.h>\n");
|
|
fprintf (fout, "#else\n");
|
|
fprintf (fout, "#define LOG_ERR 1\n");
|
|
fprintf (fout, "#define openlog(a, b, c)\n");
|
|
fprintf (fout, "#endif\n");
|
|
#endif
|
|
}
|
|
|
|
/* for ANSI-C */
|
|
if (Cflag)
|
|
fprintf (fout, "\n#ifndef SIG_PF\n#define SIG_PF void(*)(int)\n#endif\n");
|
|
|
|
#ifndef __GNU_LIBRARY__
|
|
fprintf (fout, "\n#ifdef DEBUG\n#define RPC_SVC_FG\n#endif\n");
|
|
#endif
|
|
if (timerflag)
|
|
fprintf (fout, "\n#define _RPCSVC_CLOSEDOWN %s\n", svcclosetime);
|
|
while ((def = get_definition ()) != NULL)
|
|
{
|
|
foundprogram |= (def->def_kind == DEF_PROGRAM);
|
|
}
|
|
if (extend && !foundprogram)
|
|
{
|
|
unlink (outfilename);
|
|
return;
|
|
}
|
|
write_most (infile, netflag, nomain);
|
|
if (!nomain)
|
|
{
|
|
if (!do_registers (argc, argv))
|
|
{
|
|
if (outfilename)
|
|
unlink (outfilename);
|
|
usage ();
|
|
}
|
|
write_rest ();
|
|
}
|
|
close_input ();
|
|
close_output (outfilename);
|
|
}
|
|
|
|
/*
|
|
* generate client side stubs
|
|
*/
|
|
static void
|
|
l_output (const char *infile, const char *define, int extend,
|
|
const char *outfile)
|
|
{
|
|
char *include;
|
|
definition *def;
|
|
int foundprogram = 0;
|
|
const char *outfilename;
|
|
|
|
open_input (infile, define);
|
|
outfilename = extend ? extendfile (infile, outfile) : outfile;
|
|
open_output (infile, outfilename);
|
|
add_warning ();
|
|
if (Cflag)
|
|
fprintf (fout, "#include <memory.h> /* for memset */\n");
|
|
if (infile && (include = extendfile (infile, ".h")))
|
|
{
|
|
fprintf (fout, "#include \"%s\"\n", include);
|
|
free (include);
|
|
}
|
|
else
|
|
fprintf (fout, "#include <rpc/rpc.h>\n");
|
|
while ((def = get_definition ()) != NULL)
|
|
{
|
|
foundprogram |= (def->def_kind == DEF_PROGRAM);
|
|
}
|
|
if (extend && !foundprogram)
|
|
{
|
|
unlink (outfilename);
|
|
return;
|
|
}
|
|
write_stubs ();
|
|
close_input ();
|
|
close_output (outfilename);
|
|
}
|
|
|
|
/*
|
|
* generate the dispatch table
|
|
*/
|
|
static void
|
|
t_output (const char *infile, const char *define, int extend,
|
|
const char *outfile)
|
|
{
|
|
definition *def;
|
|
int foundprogram = 0;
|
|
const char *outfilename;
|
|
|
|
open_input (infile, define);
|
|
outfilename = extend ? extendfile (infile, outfile) : outfile;
|
|
open_output (infile, outfilename);
|
|
add_warning ();
|
|
while ((def = get_definition ()) != NULL)
|
|
{
|
|
foundprogram |= (def->def_kind == DEF_PROGRAM);
|
|
}
|
|
if (extend && !foundprogram)
|
|
{
|
|
unlink (outfilename);
|
|
return;
|
|
}
|
|
write_tables ();
|
|
close_input ();
|
|
close_output (outfilename);
|
|
}
|
|
|
|
/* sample routine for the server template */
|
|
static void
|
|
svc_output (const char *infile, const char *define, int extend,
|
|
const char *outfile)
|
|
{
|
|
definition *def;
|
|
char *include;
|
|
const char *outfilename;
|
|
long tell;
|
|
|
|
open_input (infile, define);
|
|
outfilename = extend ? extendfile (infile, outfile) : outfile;
|
|
checkfiles (infile, outfilename);
|
|
/*check if outfile already exists.
|
|
if so, print an error message and exit */
|
|
open_output (infile, outfilename);
|
|
add_sample_msg ();
|
|
|
|
if (infile && (include = extendfile (infile, ".h")))
|
|
{
|
|
fprintf (fout, "#include \"%s\"\n", include);
|
|
free (include);
|
|
}
|
|
else
|
|
fprintf (fout, "#include <rpc/rpc.h>\n");
|
|
|
|
tell = ftell (fout);
|
|
while ((def = get_definition ()) != NULL)
|
|
{
|
|
write_sample_svc (def);
|
|
}
|
|
if (extend && tell == ftell (fout))
|
|
{
|
|
unlink (outfilename);
|
|
}
|
|
close_input ();
|
|
close_output (outfilename);
|
|
}
|
|
|
|
|
|
/* sample main routine for client */
|
|
static void
|
|
clnt_output (const char *infile, const char *define, int extend,
|
|
const char *outfile)
|
|
{
|
|
definition *def;
|
|
char *include;
|
|
const char *outfilename;
|
|
long tell;
|
|
int has_program = 0;
|
|
|
|
open_input (infile, define);
|
|
outfilename = extend ? extendfile (infile, outfile) : outfile;
|
|
checkfiles (infile, outfilename);
|
|
/*check if outfile already exists.
|
|
if so, print an error message and exit */
|
|
|
|
open_output (infile, outfilename);
|
|
add_sample_msg ();
|
|
if (infile && (include = extendfile (infile, ".h")))
|
|
{
|
|
fprintf (fout, "#include \"%s\"\n", include);
|
|
free (include);
|
|
}
|
|
else
|
|
fprintf (fout, "#include <rpc/rpc.h>\n");
|
|
tell = ftell (fout);
|
|
while ((def = get_definition ()) != NULL)
|
|
{
|
|
has_program += write_sample_clnt (def);
|
|
}
|
|
|
|
if (has_program)
|
|
write_sample_clnt_main ();
|
|
|
|
if (extend && tell == ftell (fout))
|
|
{
|
|
unlink (outfilename);
|
|
}
|
|
close_input ();
|
|
close_output (outfilename);
|
|
}
|
|
|
|
static char *
|
|
file_name (const char *file, const char *ext)
|
|
{
|
|
char *temp;
|
|
temp = extendfile (file, ext);
|
|
|
|
if (access (temp, F_OK) != -1)
|
|
return (temp);
|
|
else
|
|
return ((char *) " ");
|
|
}
|
|
|
|
static void
|
|
mkfile_output (struct commandline *cmd)
|
|
{
|
|
char *mkfilename;
|
|
const char *clientname, *clntname, *xdrname, *hdrname;
|
|
const char *servername, *svcname, *servprogname, *clntprogname;
|
|
|
|
svcname = file_name (cmd->infile, "_svc.c");
|
|
clntname = file_name (cmd->infile, "_clnt.c");
|
|
xdrname = file_name (cmd->infile, "_xdr.c");
|
|
hdrname = file_name (cmd->infile, ".h");
|
|
|
|
if (allfiles)
|
|
{
|
|
servername = extendfile (cmd->infile, "_server.c");
|
|
clientname = extendfile (cmd->infile, "_client.c");
|
|
}
|
|
else
|
|
{
|
|
servername = " ";
|
|
clientname = " ";
|
|
}
|
|
servprogname = extendfile (cmd->infile, "_server");
|
|
clntprogname = extendfile (cmd->infile, "_client");
|
|
|
|
if (allfiles)
|
|
{
|
|
char *cp, *temp;
|
|
|
|
mkfilename = alloc (strlen ("Makefile.") + strlen (cmd->infile) + 1);
|
|
temp = rindex (cmd->infile, '.');
|
|
cp = stpcpy (mkfilename, "Makefile.");
|
|
strncpy (cp, cmd->infile, (temp - cmd->infile));
|
|
}
|
|
else
|
|
mkfilename = (char *) cmd->outfile;
|
|
|
|
checkfiles (NULL, mkfilename);
|
|
open_output (NULL, mkfilename);
|
|
|
|
fprintf (fout, "\n# This is a template Makefile generated by rpcgen\n");
|
|
|
|
f_print (fout, "\n# Parameters\n\n");
|
|
|
|
f_print (fout, "CLIENT = %s\nSERVER = %s\n\n", clntprogname, servprogname);
|
|
f_print (fout, "SOURCES_CLNT.c = \nSOURCES_CLNT.h = \n");
|
|
f_print (fout, "SOURCES_SVC.c = \nSOURCES_SVC.h = \n");
|
|
f_print (fout, "SOURCES.x = %s\n\n", cmd->infile);
|
|
f_print (fout, "TARGETS_SVC.c = %s %s %s \n",
|
|
svcname, servername, xdrname);
|
|
f_print (fout, "TARGETS_CLNT.c = %s %s %s \n",
|
|
clntname, clientname, xdrname);
|
|
f_print (fout, "TARGETS = %s %s %s %s %s %s\n\n",
|
|
hdrname, xdrname, clntname,
|
|
svcname, clientname, servername);
|
|
|
|
f_print (fout, "OBJECTS_CLNT = $(SOURCES_CLNT.c:%%.c=%%.o) \
|
|
$(TARGETS_CLNT.c:%%.c=%%.o)");
|
|
|
|
f_print (fout, "\nOBJECTS_SVC = $(SOURCES_SVC.c:%%.c=%%.o) \
|
|
$(TARGETS_SVC.c:%%.c=%%.o)");
|
|
|
|
f_print (fout, "\n# Compiler flags \n");
|
|
if (mtflag)
|
|
fprintf (fout, "\nCPPFLAGS += -D_REENTRANT\nCFLAGS += -g \nLDLIBS \
|
|
+= -lnsl -lpthread \n ");
|
|
else
|
|
f_print (fout, "\nCFLAGS += -g \nLDLIBS += -lnsl\n");
|
|
f_print (fout, "RPCGENFLAGS = \n");
|
|
|
|
f_print (fout, "\n# Targets \n\n");
|
|
|
|
f_print (fout, "all : $(CLIENT) $(SERVER)\n\n");
|
|
f_print (fout, "$(TARGETS) : $(SOURCES.x) \n");
|
|
f_print (fout, "\trpcgen $(RPCGENFLAGS) $(SOURCES.x)\n\n");
|
|
f_print (fout, "$(OBJECTS_CLNT) : $(SOURCES_CLNT.c) $(SOURCES_CLNT.h) \
|
|
$(TARGETS_CLNT.c) \n\n");
|
|
|
|
f_print (fout, "$(OBJECTS_SVC) : $(SOURCES_SVC.c) $(SOURCES_SVC.h) \
|
|
$(TARGETS_SVC.c) \n\n");
|
|
f_print (fout, "$(CLIENT) : $(OBJECTS_CLNT) \n");
|
|
f_print (fout, "\t$(LINK.c) -o $(CLIENT) $(OBJECTS_CLNT) \
|
|
$(LDLIBS) \n\n");
|
|
f_print (fout, "$(SERVER) : $(OBJECTS_SVC) \n");
|
|
f_print (fout, "\t$(LINK.c) -o $(SERVER) $(OBJECTS_SVC) $(LDLIBS)\n\n ");
|
|
f_print (fout, "clean:\n\t $(RM) core $(TARGETS) $(OBJECTS_CLNT) \
|
|
$(OBJECTS_SVC) $(CLIENT) $(SERVER)\n\n");
|
|
close_output (mkfilename);
|
|
}
|
|
|
|
/*
|
|
* Perform registrations for service output
|
|
* Return 0 if failed; 1 otherwise.
|
|
*/
|
|
static int
|
|
do_registers (int argc, const char *argv[])
|
|
{
|
|
int i;
|
|
|
|
if (inetdflag || !tirpcflag)
|
|
{
|
|
for (i = 1; i < argc; i++)
|
|
{
|
|
if (streq (argv[i], "-s"))
|
|
{
|
|
if (!check_nettype (argv[i + 1], valid_i_nettypes))
|
|
return 0;
|
|
write_inetd_register (argv[i + 1]);
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (i = 1; i < argc; i++)
|
|
if (streq (argv[i], "-s"))
|
|
{
|
|
if (!check_nettype (argv[i + 1], valid_ti_nettypes))
|
|
return 0;
|
|
write_nettype_register (argv[i + 1]);
|
|
i++;
|
|
}
|
|
else if (streq (argv[i], "-n"))
|
|
{
|
|
write_netid_register (argv[i + 1]);
|
|
i++;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Add another argument to the arg list
|
|
*/
|
|
static void
|
|
addarg (const char *cp)
|
|
{
|
|
if (argcount >= ARGLISTLEN)
|
|
{
|
|
fprintf (stderr, _("rpcgen: too many defines\n"));
|
|
crash ();
|
|
/*NOTREACHED */
|
|
}
|
|
arglist[argcount++] = cp;
|
|
}
|
|
|
|
static void
|
|
putarg (int whereto, const char *cp)
|
|
{
|
|
if (whereto >= ARGLISTLEN)
|
|
{
|
|
fprintf (stderr, _("rpcgen: arglist coding error\n"));
|
|
crash ();
|
|
/*NOTREACHED */
|
|
}
|
|
arglist[whereto] = cp;
|
|
}
|
|
|
|
/*
|
|
* if input file is stdin and an output file is specified then complain
|
|
* if the file already exists. Otherwise the file may get overwritten
|
|
* If input file does not exist, exit with an error
|
|
*/
|
|
|
|
static void
|
|
checkfiles (const char *infile, const char *outfile)
|
|
{
|
|
struct stat buf;
|
|
|
|
if (infile) /* infile ! = NULL */
|
|
if (stat (infile, &buf) < 0)
|
|
{
|
|
perror (infile);
|
|
crash ();
|
|
}
|
|
if (outfile)
|
|
{
|
|
if (stat (outfile, &buf) < 0)
|
|
return; /* file does not exist */
|
|
else
|
|
{
|
|
fprintf (stderr,
|
|
/* TRANS: the file will not be removed; this is an
|
|
TRANS: informative message. */
|
|
_("file `%s' already exists and may be overwritten\n"),
|
|
outfile);
|
|
crash ();
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Parse command line arguments
|
|
*/
|
|
static int
|
|
parseargs (int argc, const char *argv[], struct commandline *cmd)
|
|
{
|
|
int i;
|
|
int j;
|
|
int c;
|
|
char flag[(1 << 8 * sizeof (char))];
|
|
int nflags;
|
|
|
|
cmdname = argv[0];
|
|
cmd->infile = cmd->outfile = NULL;
|
|
if (argc < 2)
|
|
{
|
|
return (0);
|
|
}
|
|
allfiles = 0;
|
|
flag['c'] = 0;
|
|
flag['h'] = 0;
|
|
flag['l'] = 0;
|
|
flag['m'] = 0;
|
|
flag['o'] = 0;
|
|
flag['s'] = 0;
|
|
flag['n'] = 0;
|
|
flag['t'] = 0;
|
|
flag['S'] = 0;
|
|
flag['C'] = 0;
|
|
flag['M'] = 0;
|
|
|
|
for (i = 1; i < argc; i++)
|
|
{
|
|
if (argv[i][0] != '-')
|
|
{
|
|
if (cmd->infile)
|
|
{
|
|
fprintf (stderr,
|
|
_("Cannot specify more than one input file!\n"));
|
|
return 0;
|
|
}
|
|
cmd->infile = argv[i];
|
|
}
|
|
else
|
|
{
|
|
for (j = 1; argv[i][j] != 0; j++)
|
|
{
|
|
c = argv[i][j];
|
|
switch (c)
|
|
{
|
|
case 'a':
|
|
allfiles = 1;
|
|
break;
|
|
case 'c':
|
|
case 'h':
|
|
case 'l':
|
|
case 'm':
|
|
case 't':
|
|
if (flag[c])
|
|
return 0;
|
|
flag[c] = 1;
|
|
break;
|
|
case 'S':
|
|
/* sample flag: Ss or Sc.
|
|
Ss means set flag['S'];
|
|
Sc means set flag['C'];
|
|
Sm means set flag['M']; */
|
|
c = argv[i][++j]; /* get next char */
|
|
if (c == 's')
|
|
c = 'S';
|
|
else if (c == 'c')
|
|
c = 'C';
|
|
else if (c == 'm')
|
|
c = 'M';
|
|
else
|
|
return 0;
|
|
|
|
if (flag[c])
|
|
return 0;
|
|
flag[c] = 1;
|
|
break;
|
|
case 'C': /* ANSI C syntax */
|
|
Cflag = 1;
|
|
break;
|
|
|
|
#ifdef __GNU_LIBRARY__
|
|
case 'k': /* K&R C syntax */
|
|
Cflag = 0;
|
|
break;
|
|
|
|
#endif
|
|
case 'b': /* turn TIRPC flag off for
|
|
generating backward compatible
|
|
*/
|
|
tirpcflag = 0;
|
|
break;
|
|
|
|
#ifdef __GNU_LIBRARY__
|
|
case '5': /* turn TIRPC flag on for
|
|
generating SysVr4 compatible
|
|
*/
|
|
tirpcflag = 1;
|
|
break;
|
|
#endif
|
|
case 'I':
|
|
inetdflag = 1;
|
|
break;
|
|
case 'N':
|
|
newstyle = 1;
|
|
break;
|
|
case 'L':
|
|
logflag = 1;
|
|
break;
|
|
case 'K':
|
|
if (++i == argc)
|
|
{
|
|
return (0);
|
|
}
|
|
svcclosetime = argv[i];
|
|
goto nextarg;
|
|
case 'T':
|
|
tblflag = 1;
|
|
break;
|
|
case 'M':
|
|
mtflag = 1;
|
|
break;
|
|
case 'i':
|
|
if (++i == argc)
|
|
{
|
|
return (0);
|
|
}
|
|
inlineflag = atoi (argv[i]);
|
|
goto nextarg;
|
|
case 'n':
|
|
case 'o':
|
|
case 's':
|
|
if (argv[i][j - 1] != '-' ||
|
|
argv[i][j + 1] != 0)
|
|
{
|
|
return (0);
|
|
}
|
|
flag[c] = 1;
|
|
if (++i == argc)
|
|
{
|
|
return (0);
|
|
}
|
|
if (c == 's')
|
|
{
|
|
if (!streq (argv[i], "udp") &&
|
|
!streq (argv[i], "tcp"))
|
|
return 0;
|
|
}
|
|
else if (c == 'o')
|
|
{
|
|
if (cmd->outfile)
|
|
return 0;
|
|
cmd->outfile = argv[i];
|
|
}
|
|
goto nextarg;
|
|
case 'D':
|
|
if (argv[i][j - 1] != '-')
|
|
return 0;
|
|
addarg (argv[i]);
|
|
goto nextarg;
|
|
case 'Y':
|
|
if (++i == argc)
|
|
return 0;
|
|
{
|
|
size_t len = strlen (argv[i]);
|
|
pathbuf = malloc (len + 5);
|
|
if (pathbuf == NULL)
|
|
{
|
|
perror (cmdname);
|
|
crash ();
|
|
}
|
|
stpcpy (stpcpy (pathbuf,
|
|
argv[i]),
|
|
"/cpp");
|
|
CPP = pathbuf;
|
|
cppDefined = 1;
|
|
goto nextarg;
|
|
}
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
nextarg:
|
|
;
|
|
}
|
|
}
|
|
|
|
cmd->cflag = flag['c'];
|
|
cmd->hflag = flag['h'];
|
|
cmd->lflag = flag['l'];
|
|
cmd->mflag = flag['m'];
|
|
cmd->nflag = flag['n'];
|
|
cmd->sflag = flag['s'];
|
|
cmd->tflag = flag['t'];
|
|
cmd->Ssflag = flag['S'];
|
|
cmd->Scflag = flag['C'];
|
|
cmd->makefileflag = flag['M'];
|
|
|
|
#ifndef _RPC_THREAD_SAFE_
|
|
if (mtflag || newstyle)
|
|
{
|
|
/* glibc doesn't support these flags. */
|
|
f_print (stderr,
|
|
_("This implementation doesn't support newstyle or MT-safe code!\n"));
|
|
return (0);
|
|
}
|
|
#endif
|
|
if (tirpcflag)
|
|
{
|
|
pmflag = inetdflag ? 0 : 1; /* pmflag or inetdflag is always TRUE */
|
|
if ((inetdflag && cmd->nflag))
|
|
{ /* netid not allowed with inetdflag */
|
|
fprintf (stderr, _("Cannot use netid flag with inetd flag!\n"));
|
|
return 0;
|
|
}
|
|
}
|
|
else
|
|
{ /* 4.1 mode */
|
|
pmflag = 0; /* set pmflag only in tirpcmode */
|
|
#ifndef __GNU_LIBRARY__
|
|
inetdflag = 1; /* inetdflag is TRUE by default */
|
|
#endif
|
|
if (cmd->nflag)
|
|
{ /* netid needs TIRPC */
|
|
f_print (stderr, _("Cannot use netid flag without TIRPC!\n"));
|
|
return (0);
|
|
}
|
|
}
|
|
|
|
if (newstyle && (tblflag || cmd->tflag))
|
|
{
|
|
f_print (stderr, _("Cannot use table flags with newstyle!\n"));
|
|
return (0);
|
|
}
|
|
|
|
/* check no conflicts with file generation flags */
|
|
nflags = cmd->cflag + cmd->hflag + cmd->lflag + cmd->mflag +
|
|
cmd->sflag + cmd->nflag + cmd->tflag + cmd->Ssflag + cmd->Scflag;
|
|
|
|
if (nflags == 0)
|
|
{
|
|
if (cmd->outfile != NULL || cmd->infile == NULL)
|
|
{
|
|
return (0);
|
|
}
|
|
}
|
|
else if (cmd->infile == NULL &&
|
|
(cmd->Ssflag || cmd->Scflag || cmd->makefileflag))
|
|
{
|
|
fprintf (stderr,
|
|
_("\"infile\" is required for template generation flags.\n"));
|
|
return 0;
|
|
}
|
|
if (nflags > 1)
|
|
{
|
|
fprintf (stderr, _("Cannot have more than one file generation flag!\n"));
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static void
|
|
usage (void)
|
|
{
|
|
fprintf (stderr, _("usage: %s infile\n"), cmdname);
|
|
fprintf (stderr, _("\t%s [-abkCLNTM][-Dname[=value]] [-i size] \
|
|
[-I [-K seconds]] [-Y path] infile\n"), cmdname);
|
|
fprintf (stderr, _("\t%s [-c | -h | -l | -m | -t | -Sc | -Ss | -Sm] \
|
|
[-o outfile] [infile]\n"), cmdname);
|
|
fprintf (stderr, _("\t%s [-s nettype]* [-o outfile] [infile]\n"), cmdname);
|
|
fprintf (stderr, _("\t%s [-n netid]* [-o outfile] [infile]\n"), cmdname);
|
|
options_usage ();
|
|
exit (1);
|
|
}
|
|
|
|
static void
|
|
options_usage (void)
|
|
{
|
|
f_print (stderr, "options:\n");
|
|
f_print (stderr, "-a\t\tgenerate all files, including samples\n");
|
|
f_print (stderr, "-b\t\tbackward compatibility mode (generates code for SunOS 4.1)\n");
|
|
f_print (stderr, "-c\t\tgenerate XDR routines\n");
|
|
f_print (stderr, "-C\t\tANSI C mode\n");
|
|
f_print (stderr, "-Dname[=value]\tdefine a symbol (same as #define)\n");
|
|
f_print (stderr, "-h\t\tgenerate header file\n");
|
|
f_print (stderr, "-i size\t\tsize at which to start generating inline code\n");
|
|
f_print (stderr, "-I\t\tgenerate code for inetd support in server (for SunOS 4.1)\n");
|
|
f_print (stderr, "-K seconds\tserver exits after K seconds of inactivity\n");
|
|
f_print (stderr, "-l\t\tgenerate client side stubs\n");
|
|
f_print (stderr, "-L\t\tserver errors will be printed to syslog\n");
|
|
f_print (stderr, "-m\t\tgenerate server side stubs\n");
|
|
f_print (stderr, "-M\t\tgenerate MT-safe code\n");
|
|
f_print (stderr, "-n netid\tgenerate server code that supports named netid\n");
|
|
f_print (stderr, "-N\t\tsupports multiple arguments and call-by-value\n");
|
|
f_print (stderr, "-o outfile\tname of the output file\n");
|
|
f_print (stderr, "-s nettype\tgenerate server code that supports named nettype\n");
|
|
f_print (stderr, "-Sc\t\tgenerate sample client code that uses remote procedures\n");
|
|
f_print (stderr, "-Ss\t\tgenerate sample server code that defines remote procedures\n");
|
|
f_print (stderr, "-Sm \t\tgenerate makefile template \n");
|
|
f_print (stderr, "-t\t\tgenerate RPC dispatch table\n");
|
|
f_print (stderr, "-T\t\tgenerate code to support RPC dispatch tables\n");
|
|
f_print (stderr, "-Y path\t\tdirectory name to find C preprocessor (cpp)\n");
|
|
|
|
exit (1);
|
|
}
|