SMOLNET PORTAL home about changes
/********************************************************************
 * wilkinson
 * 3.16VMS
 * 1995/09/25  11:11 
 * gopher_root1:[gopher.g2.vms2_13.object]compatible.c,v
 * Exp
 *
 * Paul Lindner, University of Minnesota CIS.
 *
 * Copyright 1991, 1992 by the Regents of the University of Minnesota
 * see the file "Copyright" in the distribution for conditions of use.
 *********************************************************************
 * MODULE: compatible.c
 * Compatibility routines
 *********************************************************************
 * Revision History:
 * compatible.c,v
 * Revision 3.16VMS 1995/09/25 11:11    wilkinson
 * Consolodate VMS/Unix source code for server as well as client
 *  - move syslog() routines here so available to both server and client
 *  - use __VMS instead of VMS as trigger for VMS items
 *
 * Revision 3.16  1995/02/06  22:14:22  lindner
 * Remove const
 *
 * Revision 3.15  1994/12/15  17:30:16  lindner
 * A replacement fgetpwent
 *
 * Revision 3.14  1994/07/21  22:18:45  lindner
 * Add putenv() compat code
 *
 * Revision 3.13  1994/04/25  03:34:34  lindner
 * Put back alpha patch (Fote)
 *
 * Revision 3.12  1994/04/08  20:05:55  lindner
 * gcc -Wall fixes
 *
 * Revision 3.11  1994/04/01  04:44:28  lindner
 * Fix for alpha VMS
 *
 * Revision 3.10  1994/03/30  21:38:39  lindner
 * Remove some VMS code
 *
 * Revision 3.9  1994/03/17  04:38:45  lindner
 * VMS weird directory routines
 *
 * Revision 3.8  1994/03/08  03:24:09  lindner
 * Waitpid for vms
 *
 * Revision 3.7  1993/10/27  18:51:10  lindner
 * Updates for VMS files/records
 *
 * Revision 3.6  1993/09/03  03:26:39  lindner
 * Better VMS tempnam() implementation
 *
 * Revision 3.5  1993/06/22  05:53:17  lindner
 * Added getdtablesize() option
 *
 * Revision 3.4  1993/04/15  21:36:30  lindner
 * Emulation of geteuid calls for HPs
 *
 * Revision 3.3  1993/03/18  22:27:46  lindner
 * better portable tempnam()
 *
 * Revision 3.2  1993/02/19  21:33:27  lindner
 * Gopher1.2b2 release
 *
 * Revision 3.1.1.1  1993/02/11  18:03:05  lindner
 * Gopher+1.2beta release
 *
 * Revision 1.4  1993/01/17  03:46:12  lindner
 * Fixed tempnam for VMS
 *
 * Revision 1.3  1993/01/08  23:13:55  lindner
 * Added more VMS mods from jqj
 *
 *
 *********************************************************************/


/*
 * Some functions that aren't implemented on every machine on the net
 *
 * definitions should be in the form "NO_FNNAME"
 * compatible.h looks at preprocessor symbols and automatically defines
 * many of the NO_FNNAME options
 *
 */

#include <string.h>
#include <stdio.h>
#include "Malloc.h"  /*** For NULL ***/
#include "compatible.h"

/*** For machines that don't have strstr ***/

#if defined(NOSTRSTR)

char *
strstr(host_name, cp)
  char *host_name;
  char *cp;
{
     int i, j;

     for (i = 0; i < strlen(host_name); i++) {
          j = strncmp(host_name+i, cp, strlen(cp));
          if (j == 0)
               return(host_name+i);
     }
     return(NULL);
}
#endif

#if defined(sequent)

#include <varargs.h>
vsprintf(va_alist)
  va_dcl
{
        ;
}

vfprintf(va_alist)
  va_dcl
{
        ;
}


#endif

#if defined(NO_TEMPNAM)
/* A tip of the hat to the developers of elm 2.4pl17, from whence 
   the non-VMS portion of this routine comes.  
*/
/* and a tempnam for temporary files */
static int cnt = 0;

char *tempnam(dir, pfx)
  char *dir;
  char *pfx;
{
	char space[512];
	char *newspace;

#ifdef __VMS
	if (dir == NULL) {
		dir = "sys$scratch:";
	} else if (*dir == '\0') {
		dir = "sys$scratch:";
	}
	
	if (pfx == NULL) {
		pfx = "gopher.$";
	} else if (*pfx == '\0') {
		pfx = "gopher.$";
	}

	sprintf(space, "%s%s%d%d", dir, pfx, getpid(), cnt);
#else
	if (dir == NULL) {
		dir = "/usr/tmp";
	} else if (*dir == '\0') {
		dir = "/usr/tmp";
	}
	
	if (pfx == NULL) {
		pfx = "";
	}

	sprintf(space, "%s/%s%d.%d", dir, pfx, getpid(), cnt);
#endif
	cnt++;
	
	newspace = (char *)malloc(strlen(space) + 1);
	if (newspace != NULL) {
		strcpy(newspace, space);
	}
	return newspace;
}
#endif

#if defined(NO_STRDUP)
char *strdup(str)
  char *str;
{
        int len;
        char *temp;

        if (str == NULL) return(NULL);
        len = strlen(str);

        temp = (char *) malloc(sizeof(char) * len + 1);

        strcpy(temp, str);
        return(temp);
}
#endif

#if defined(NO_TZSET)
void
tzset()
{
     ;
}
#endif

#if defined(NO_STRCASECMP)
/*
 * Copyright (c) 1987 Regents of the University of California.
 * 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.
 * 3. All advertising materials mentioning features or use of this software
 *    must display the following acknowledgement:
 *	This product includes software developed by the University of
 *	California, Berkeley and its contributors.
 * 4. Neither the name of the University nor the names of its contributors
 *    may be used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``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 REGENTS OR CONTRIBUTORS 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.
 *
 * Modified for use on VMS by Earl Fogel, University of Saskatchewan
 * Computing Services, January 1992
 */

typedef unsigned char U_char;

/*
 * This array is designed for mapping upper and lower case letter
 * together for a case independent comparison.  The mappings are
 * based upon ascii character sequences.
 */
static U_char charmap[] = {
	'\000', '\001', '\002', '\003', '\004', '\005', '\006', '\007',
	'\010', '\011', '\012', '\013', '\014', '\015', '\016', '\017',
	'\020', '\021', '\022', '\023', '\024', '\025', '\026', '\027',
	'\030', '\031', '\032', '\033', '\034', '\035', '\036', '\037',
	'\040', '\041', '\042', '\043', '\044', '\045', '\046', '\047',
	'\050', '\051', '\052', '\053', '\054', '\055', '\056', '\057',
	'\060', '\061', '\062', '\063', '\064', '\065', '\066', '\067',
	'\070', '\071', '\072', '\073', '\074', '\075', '\076', '\077',
	'\100', '\141', '\142', '\143', '\144', '\145', '\146', '\147',
	'\150', '\151', '\152', '\153', '\154', '\155', '\156', '\157',
	'\160', '\161', '\162', '\163', '\164', '\165', '\166', '\167',
	'\170', '\171', '\172', '\133', '\134', '\135', '\136', '\137',
	'\140', '\141', '\142', '\143', '\144', '\145', '\146', '\147',
	'\150', '\151', '\152', '\153', '\154', '\155', '\156', '\157',
	'\160', '\161', '\162', '\163', '\164', '\165', '\166', '\167',
	'\170', '\171', '\172', '\173', '\174', '\175', '\176', '\177',
	'\200', '\201', '\202', '\203', '\204', '\205', '\206', '\207',
	'\210', '\211', '\212', '\213', '\214', '\215', '\216', '\217',
	'\220', '\221', '\222', '\223', '\224', '\225', '\226', '\227',
	'\230', '\231', '\232', '\233', '\234', '\235', '\236', '\237',
	'\240', '\241', '\242', '\243', '\244', '\245', '\246', '\247',
	'\250', '\251', '\252', '\253', '\254', '\255', '\256', '\257',
	'\260', '\261', '\262', '\263', '\264', '\265', '\266', '\267',
	'\270', '\271', '\272', '\273', '\274', '\275', '\276', '\277',
	'\300', '\301', '\302', '\303', '\304', '\305', '\306', '\307',
	'\310', '\311', '\312', '\313', '\314', '\315', '\316', '\317',
	'\320', '\321', '\322', '\323', '\324', '\325', '\326', '\327',
	'\330', '\331', '\332', '\333', '\334', '\335', '\336', '\337',
	'\340', '\341', '\342', '\343', '\344', '\345', '\346', '\347',
	'\350', '\351', '\352', '\353', '\354', '\355', '\356', '\357',
	'\360', '\361', '\362', '\363', '\364', '\365', '\366', '\367',
	'\370', '\371', '\372', '\373', '\374', '\375', '\376', '\377',
};

int
strcasecmp(s1, s2)
	const char *s1, *s2;
{
	register const U_char *cm = charmap,
			*us1 = (const U_char *)s1,
			*us2 = (const U_char *)s2;

	while (cm[*us1] == cm[*us2++])
		if (*us1++ == '\0')
			return (0);
	return (cm[*us1] - cm[*--us2]);
}

int
strncasecmp(s1, s2, n)
	const char *s1, *s2;
	register size_t n;
{
	if (n != 0) {
		register const U_char *cm = charmap,
				*us1 = (const U_char *)s1,
				*us2 = (const U_char *)s2;

		do {
			if (cm[*us1] != cm[*us2++])
				return (cm[*us1] - cm[*--us2]);
			if (*us1++ == '\0')
				break;
		} while (--n != 0);
	}
	return (0);
}

#endif

#if defined(NO_GETDTABLESIZE)
int getdtablesize()
{
     struct rlimit rlp;
     
     rlp.rlim_cur = rlp.rlim_max = RLIM_INFINITY;
     
     if (getrlimit( RLIMIT_NOFILE, &rlp ) )
	  return(-1);
     fds = rlp.rlim_cur;
}
#endif

#if defined(__VMS)
/* In all versions of VMS, fopen() and open() are needlessly inefficient.
 * Define jacket routines to do file opens with more sensible parameters
 * than the VAXCRTL default.
 * [Should we really be doing this for EVERY fopen() and open()?]
 */
#ifdef fopen
#undef fopen
#endif
#ifdef open
#undef open
#endif

FILE *fopen_VAR ( name, mode )
    char *name, *mode;
{
    return fopen ( name, mode, "rfm=var","rat=cr","mbc=32" );
}

FILE *fopen_FIX ( name, mode )
    char *name, *mode;
{
    return fopen ( name, mode, "rfm=fix","mrs=512","mbc=32" );
}

#ifdef VMS_SERVER

int vaxc$errno_stv;
#include <rms.h>
#include <errno.h>

FILE *fopen_VMSopt ( name, mode , alq, deq)
    char *name, *mode, *alq, *deq;
{
    struct FAB	fab;
    int		status;
    FILE	*opened;

    errno = vaxc$errno = vaxc$errno_stv = 0;
    if (strcmp(mode,"a")==0)
	opened = fopen (name, mode, "mbc=32", alq, deq);
    else
	opened = fopen ( name, mode, "mbc=32" );
    if (opened==NULL) {
	fab = cc$rms_fab;
	fab.fab$l_fna = name;
	fab.fab$b_fns = strlen(fab.fab$l_fna);
	if (((status = SYS$OPEN(&fab, 0, 0)) &1) != 1) {
	    vaxc$errno = fab.fab$l_sts;
	    vaxc$errno_stv = fab.fab$l_stv;
	}
	else SYS$CLOSE(&fab,0,0);
    }
    return opened;
}
#else
FILE *fopen_VMSopt ( name, mode )
    char *name, *mode;
{
    return fopen ( name, mode, "mbc=32" );
}
#endif

int open_VMSopt ( name, flags, mode )
    char *name;
    int flags;
    unsigned int mode;
{
    return  open ( name, flags, mode, "mbc=32");
}

#endif

#if defined(VMS_SERVER) && !defined(MULTINET)

/* Multinet defines a handy vms_errno_string() that essentially packages
 * strerror with no parameters.  (This insulates other platforms from
 * VMS' oddball two-argument strerror.) We need to supply it here for
 * other IP packages under VMS.
 */

char *vms_errno_string (void) { return(strerror(errno, vaxc$errno)); }

#endif


#if defined(__VMS) || defined(NO_WAITPID_WAIT3)
#  include "Wait.h"

pid_t waitpid(pid, status, options)
  pid_t pid;
  int   *status;
  int   options;
{
     int zepid;

     while ((zepid = wait(status)) != pid && pid != -1)
	  /** Should check and make sure process exited... **/
	  ;

     return(zepid);
}
#endif     

/*
 * This is an ugly hack for now, since we only use strftime in one
 * place..... so far..
 */

#if defined(NO_STRFTIME)
#  include <time.h>


int strftime(buf, bufsize, fmt, tm)
  char *buf;
  int bufsize;
  char *fmt;
  struct tm *tm;
{
     sprintf(buf, "%4d%02d%02d%02d%02d%02d\n",
	     tm->tm_year, tm->tm_mon+1,tm->tm_mday,
	     tm->tm_hour, tm->tm_mon, tm->tm_sec);
}

#endif
	     

/*
 * This code was stolen from cnews.  Modified to make "newenv" static so
 * that realloc() can be used on subsequent calls to avoid memory leaks.
 *
 * We only need this if Configure said there isn't a putenv() in libc.
 */

#ifndef __VMS
#ifdef NO_PUTENV /*{*/

/* peculiar return values */
#define WORKED 0
#define FAILED 1

int
putenv(var)			/* put var in the environment */
char *var;
{
     register char **envp;
     register int oldenvcnt;
     extern char **environ;
     static char **newenv = NULL;
     
     /* count variables, look for var */
     for (envp = environ; *envp != 0; envp++) {
	  register char *varp = var, *ep = *envp;
	  register int namesame;
	  
	  namesame = 0;
	  for (; *varp == *ep && *varp != '\0'; ++ep, ++varp)
	       if (*varp == '=')
		    namesame = 1;
	  if (*varp == *ep && *ep == '\0')
	       return WORKED;	/* old & new var's are the same */
	  if (namesame) {
	       *envp = var;	/* replace var with new value */
	       return WORKED;
	  }
     }
     oldenvcnt = envp - environ;
     
     /* allocate new environment with room for one more variable */
     if (newenv == NULL)
	  newenv = (char **)malloc((unsigned)((oldenvcnt+1+1)*sizeof(*envp)));
     else
	  newenv = (char **)realloc((char *)newenv, (unsigned)((oldenvcnt+1+1)*sizeof(*envp)));
     if (newenv == NULL)
	  return FAILED;
     
     /* copy old environment pointers, add var, switch environments */
     (void) bcopy((char *)environ, (char *)newenv, oldenvcnt*sizeof(*envp));
     newenv[oldenvcnt] = var;
     newenv[oldenvcnt+1] = NULL;
     environ = newenv;
     return WORKED;
}

#endif /* NO_PUTENV */
#else
/*	OpenVMS has no putenv() function.  So we'll make one ;-)    */


/* peculiar return values */
#define WORKED 0
#define FAILED 1

int
putenv(var)			/* put var in the environment */
char *var;
{       /*  If there is a : imbedded in the variable name, then the text 
	    prefixing that : is the logical name table where we want this 
	    logical, and the text after it is the logical name; otherwise it's
	    to be defined as a symbol; if the table name is null, the 
	    LNM$PROCESS table is assumed.  A null value says to delete the
	    logical or symbol instead of set it.
	*/

#include <ctype.h>
#include <descrip.h>
#include <ssdef.h>
#include <libdef.h>
#include <libclidef.h>
#include <lnmdef.h>
#include <psldef.h>

    char *s;
    $DESCRIPTOR(dsc$var,NULL);
    $DESCRIPTOR(dsc$val,NULL);
    $DESCRIPTOR(dsc$tbl,NULL);
    int	i;
    unsigned char x;
    long global = LIB$K_CLI_GLOBAL_SYM;
    char *value = strchr(var,'=');
    struct  {
	  unsigned short int length;
	  unsigned short int code;
	  char *bufadr;
	  int *retlen;
    }	  itmlst[2] = {  { 0, LNM$_STRING, NULL, NULL},
			    { 0, 0, 0, 0 } };


    if (!value)
	return(FAILED);
    *(value++) = '\0';
    if (s = strpbrk(var,":;")) {	/* before : is LNM Table, after is logical  */
	x = (unsigned char)*s;		/* mode -- :=super, ;=exec  */
	*(s++) = '\0';
	dsc$var.dsc$a_pointer = s;
	dsc$var.dsc$w_length = strlen(s);
	if (strlen(var)) {
	    dsc$tbl.dsc$a_pointer = var;
	    dsc$tbl.dsc$w_length = strlen(var);
	}
	else {
	    dsc$tbl.dsc$a_pointer = NULL;
	    dsc$tbl.dsc$w_length = 0;
	}
    }
    else {			/* no : means var is a symbol		    */
	dsc$var.dsc$a_pointer = var;
	dsc$var.dsc$w_length = strlen(var);
	dsc$tbl.dsc$a_pointer = NULL;
	dsc$tbl.dsc$w_length = 0;	
    }

    for (i=0, s = dsc$var.dsc$a_pointer; i < dsc$var.dsc$w_length; i++, s++)
	*s = _toupper(*s);
    
    if (!strlen(value)) {	/*  Hmmm... null value ought to mean delete */
	if (dsc$tbl.dsc$w_length)
	    i = lib$delete_logical(&dsc$var, &dsc$tbl);
	else 
	    i = lib$delete_symbol(&dsc$var, &global);
	if (i==LIB$_NOSUCHSYM || i==SS$_NOLOGNAM)
	    i = SS$_NORMAL;
    }
    else {			/*  non-null value means define var=value   */
	dsc$val.dsc$a_pointer = value;
	dsc$val.dsc$w_length = strlen(value);
	if (dsc$tbl.dsc$w_length) {
	    switch(x) {
	    case ':':	i = lib$set_logical(&dsc$var, &dsc$val, &dsc$tbl);
			break;
	    case ';':	x = PSL$C_EXEC;
			itmlst[0].bufadr = dsc$val.dsc$a_pointer;
			itmlst[0].length = dsc$val.dsc$w_length;
			i = sys$crelnm(NULL, &dsc$tbl, &dsc$var, &x, &itmlst);
			break;
	    }
	}
	else 
	    i = lib$set_symbol(&dsc$var, &dsc$val, &global);
    }
    vaxc$errno = i;
    return(i==SS$_NORMAL?WORKED:FAILED);
}
#endif

#ifdef NO_FGETPWENT
struct passwd *
fgetpwent(f)
  FILE *f;
{
     static char input[256];
     static struct passwd *p  = NULL;
     char *cp, *cp2;

     if (p == NULL) {
	  p = (struct passwd *) malloc(sizeof(struct passwd));
     }
     
     if (fgets(input, 256, f) == NULL)
	  return(NULL);

     /* Username */
     cp = strchr(input, ':');
     if (cp == NULL) return(NULL);

     *cp = '\0'; cp++;
     p->pw_name = input;
     
     /* Password */
     cp2 = strchr(cp, ':');
     if (cp2 == NULL) return(NULL);
     *cp2 = '\0'; cp2++;
     p->pw_passwd = cp;

     /* UID */
     cp = strchr(cp2, ':');
     if (cp == NULL) return(NULL);
     
     *cp = '\0'; cp++;
     p->pw_uid = atoi(cp2);

     /* GID */
     cp2 = strchr(cp, ':');
     if (cp2 == NULL) return(NULL);
     *cp2 = '\0'; cp2++;
     p->pw_gid = atoi(cp);

     /* GECOS */
     cp = strchr(cp2, ':');
     if (cp == NULL) return(NULL);
     *cp = '\0'; cp++;
     p->pw_gecos = cp2;

     /* Home dir */
     cp2 = strchr(cp, ':');
     if (cp2 == NULL) return(NULL);
     *cp2 = '\0'; cp2++;
     p->pw_dir = cp;

     /* Shell */
     p->pw_shell = cp2;

     return(p);
}
#endif /* NO_FGETPWENT */

#if defined(__VMS) && defined(MULTINET)
#if ( defined(CLIENT_LOGGER) || defined(TELNET_TRACE) || defined(VMS_SERVER))
/*
 * JL Wilkinson 16-Jul-1994 Merged in TGV's syslog() functions to allow
 * the CLIENT_LOGGER and TELNET_TRACE functionality to work with
 * Multinet.  Not tested against other VMS TCP/IP agents, and *NOT
 * SUPPORTED* thru TGV -- please don't anybody ask TGV for help in using
 * this.  Somebody who wants to use it can Email me or the
 * VMSGopher-L@trln.lib.unc.edu group and I'll try to assist them for
 * MULTINET only (JLW@psulias.psu.edu).  Basic documentation is in
 * the file [.doc]syslogd.vms.  An implementation for UCX of SYSLOGD is
 * apparently available, but so far this code has not been adapted to it.
 */


#include "syslog.h"

/*
 *	Copyright (C) 1992	TGV, Incorporated
 */

/*
 * Copyright (c) 1983 Regents of the University of California.
 * All rights reserved.  The Berkeley software License Agreement
 * specifies the terms and conditions for redistribution.
 */

#ifndef lint
static char sccsid[] = "@(#)syslog.c    5.3 (Berkeley) 9/17/85";
#endif /* not lint */

/*
 * SYSLOG -- print message on log file
 *
 * This routine looks a lot like printf, except that it
 * outputs to the log file instead of the standard output.
 * Also:
 *	adds a timestamp,
 *	prints the module name in front of the message,
 *	has some other formatting types (or will sometime),
 *	adds a newline on the end of the message.
 *
 * The output of this routine is intended to be read by /etc/syslogd.
 *
 * Author: Eric Allman
 * Modified to use UNIX domain IPC by Ralph Campbell
 */

#include <stdio.h> /* JDB */
#include <lnmdef.h>  /* JLW */
#include <ssdef.h>   /* JLW */

#include <types.h>
#include <socket.h>
#include <file.h>
#include <signal.h>

#ifndef _sys_syslog_h
#include "syslog.h"	/*  was <syslog.h> -- JLW */
#endif
#include <netdb.h>
#ifdef __ALPHA
#include <varargs.h>
#endif /* __ALPHA */

#define MAXLINE 1024			/* max message size */

#define PRIMASK(p)	(1 << ((p) & LOG_PRIMASK))
#define PRIFAC(p)	(((p) & LOG_FACMASK) >> 3)
#define IMPORTANT	LOG_ERR

static char	logname[] = "/dev/log";
static char	ctty[] = "/dev/console";
static int	LogFile = -1;		/* fd for log */
static int	LogStat = 0;		/* status bits, set by openlog() */
static char	*LogTag = "syslog";	/* string to tag the entry with */
static int	LogMask = 0xff;		/* mask of priorities to be logged */
static int	LogFacility = LOG_USER; /* default facility code */

static struct sockaddr SyslogAddr;	/* AF_UNIX address of local logger */

#ifdef ORIG
extern int errno;
extern	int sys_nerr;
extern	char *sys_errlist[];
#endif

char	*vms_errno_string();

#ifdef __ALPHA
syslog(va_alist)
va_dcl
#else  /* __ALPHA */
syslog(pri, fmt, p0, p1, p2, p3, p4, p5, p6, p7, p8, p9)
	int pri;
	char *fmt;
#endif /* __ALPHA */
{
	char buf[MAXLINE + 1], outline[MAXLINE + 1];
	register char *b, *f, *o;
	register int c;
	long now;
	int pid;
#ifdef ORIG
	int pid, olderrno = errno;
#endif
#ifdef __ALPHA
	int pri, numargs;
	char *fmt;
	__int64 p0,p1,p2,p3,p4,p5,p6,p7,p8,p9;
	va_list ap;

	va_start(ap);
	va_count(numargs);

	pri = va_arg(ap, int);
	fmt = va_arg(ap, char *);

	if (numargs > 2) p0 = va_arg(ap, __int64);
	if (numargs > 3) p1 = va_arg(ap, __int64);
	if (numargs > 4) p2 = va_arg(ap, __int64);
	if (numargs > 5) p3 = va_arg(ap, __int64);
	if (numargs > 6) p4 = va_arg(ap, __int64);
	if (numargs > 7) p5 = va_arg(ap, __int64);
	if (numargs > 8) p6 = va_arg(ap, __int64);
	if (numargs > 9) p7 = va_arg(ap, __int64);
	if (numargs > 10) p8 = va_arg(ap, __int64);
	if (numargs > 11) p9 = va_arg(ap, __int64);

#endif /* __ALPHA */


	/* see if we should just throw out this message */
	if (pri <= 0 || PRIFAC(pri) >= LOG_NFACILITIES || (PRIMASK(pri) & LogMask) == 0)
		return;
	if (LogFile < 0)
		openlog(LogTag, LogStat & ~LOG_ODELAY, 0);

	/* set default facility if none specified */
	if ((pri & LOG_FACMASK) == 0)
		pri |= LogFacility;

	/* build the message */
	o = outline;
	sprintf(o, "<%d>", pri);
	o += strlen(o);
	time(&now);
	sprintf(o, "%.15s ", ctime(&now) + 4);
	o += strlen(o);
	if (LogTag && strcmp(LogTag, "dprintf")) {
		strcpy(o, LogTag);
		o += strlen(o);
	}
	if (LogStat & LOG_PID) {
		sprintf(o, "[Pid 0x%x]", getpid());
		o += strlen(o);
	}
	if (LogTag  &&	strcmp(LogTag, "dprintf")) {
		strcpy(o, ": ");
		o += 2;
	}

	b = buf;
	f = fmt;
	while ((c = *f++) != '\0' && c != '\n' && b < &buf[MAXLINE]) {
		if (c != '%') {
			*b++ = c;
			continue;
		}
		if ((c = *f++) != 'm') {
			*b++ = '%';
			*b++ = c;
			continue;
		}
		sprintf(b, "%s", vms_errno_string() );
		b += strlen(b);
	}
	*b++ = '\n';
	*b = '\0';
	sprintf(o, buf, p0, p1, p2, p3, p4, p5, p6, p7, p8, p9);
	c = strlen(outline);
	if (c > MAXLINE)
		c = MAXLINE;

	/* output the message to the local logger */
#ifndef VMS$TrnLNM
	if (getenv("MULTINET_SYSLOG_DESTINATION") != NULL)
#else
	if (VMS$TrnLNM("MULTINET_SYSLOG_DESTINATION",0)
#endif
	    {
	    if (sendto(LogFile, outline, c, 0, &SyslogAddr, sizeof SyslogAddr) >= 0) {
		    return 1;
	    }
	} 
#ifdef Send_Oper
	if (!(LogStat & LOG_CONS) && (pri & LOG_PRIMASK) <= LOG_ERR)
		return -1;

	 /* Output the message to the console, add the LogTag so it looks
	  * like syslog(), even if syslog is disabled and we're using OPCOM.
	  * make sure there is a newline.
	  */

	o = outline;
	if (LogTag && strcmp(LogTag, "dprintf"))  {
		strcpy(o, LogTag);
		o += strlen(o);
		*o++ = ':';
		*o++ = ' ';
	}
	sprintf(o, fmt, p0, p1, p2, p3, p4, p5, p6, p7, p8, p9);
	c = strlen(outline);
	if (c > MAXLINE)
		c = MAXLINE-1;
	if ( outline[c-1] != '\n' ) {
		outline[c]  = '\n';
		outline[c+1]  = '\0';
	}
	Send_Oper(outline);
#endif
	return 1;
}

/*
 * OPENLOG -- open system log
 */

openlog(ident, logstat, logfac)
	char *ident;
	int logstat, logfac;
{
	if (ident != NULL)
		LogTag = ident;
	LogStat = logstat;
	if (logfac != 0)
		LogFacility = logfac & LOG_FACMASK;
	if (LogFile >= 0)
		return;
	SyslogAddr.sa_family = AF_UNIX;
	strncpy(SyslogAddr.sa_data, logname, sizeof SyslogAddr.sa_data);
	if (!(LogStat & LOG_ODELAY)) {
		LogFile = socket(AF_UNIX, SOCK_DGRAM, 0);
#ifdef ORIG
		fcntl(LogFile, F_SETFD, 1);
#endif
	}
}

/*
 * CLOSELOG -- close the system log
 */
closelog()
{

	(void) close(LogFile);
	LogFile = -1;
}

/*
 * SETLOGMASK -- set the log mask level
 */
setlogmask(pmask)
	int pmask;
{
	int omask;

	omask = LogMask;
	if (pmask != 0)
		LogMask = pmask;
	return (omask);
}

#endif
#endif
.
Response: text/plain
Original URLgopher://bitreich.org/0/gopher2007/2007-gopher-mirror/gop...
Content-Typetext/plain; charset=utf-8