OpenSolaris_b135/cmd/setuname/setuname.c

Compare this file to the similar file:
Show the results in this format:

/*
 * CDDL HEADER START
 *
 * The contents of this file are subject to the terms of the
 * Common Development and Distribution License (the "License").
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
 * or http://www.opensolaris.org/os/licensing.
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing Covered Code, include this CDDL HEADER in each
 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
 * If applicable, add the following below this CDDL HEADER, with the
 * fields enclosed by brackets "[]" replaced with your own identifying
 * information: Portions Copyright [yyyy] [name of copyright owner]
 *
 * CDDL HEADER END
 */
/*	Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T	*/
/*	  All Rights Reserved  	*/


/*
 * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
 * Use is subject to license terms.
 */

#pragma ident	"%Z%%M%	%I%	%E% SMI"       /* SVr4.0 1.3 */

/*
 *  setuname [-t] [-s name] [-n node] 
 */

/*
 *  Header files referenced:
 *	<stdio.h>	Standard I/O 
 *	<unistd.h>	Standard UNIX definitions
 *	<string.h>	String handling 
 *	<fmtmsg.h>	Standard message generation 
 *	<ctype.h>	Character types
 *	<errno.h>	Error handling
 *	<signal.h>	Signal handling 
 *	<sys/types.h>	Data types
 *	<sys/fcntl.h>	File control
 *	<sys/utsname.h>	System Name
 *	<sys/sys3b.h>	sys3b() definitions
 *	<nlist.h>	Definitions for Sun symbol table entries
 */

#include	<stdio.h>
#include	<unistd.h>
#include	<string.h>
#include	<fmtmsg.h>
#include	<ctype.h>
#include	<errno.h>
#include	<signal.h>
#include	<sys/types.h>
#include	<sys/uio.h>
#include	<sys/fcntl.h>
#include	<sys/psw.h>
#include	<sys/utsname.h>

#if u3b || u3b15 || u3b2
#include	<sys/sys3b.h>
#endif

#if sun
#include	<nlist.h>
#include	<kvm.h>
#endif

/*
 * Externals referenced (and not defined in a header)
 *	optind		index to the next arg for getopt()
 *	opterr		FLAG, TRUE tells getopt() to write messages
 *	optarg		Ptr to an option's argument
 *	getopt()	Gets an option from the command line
 *	putenv()	Writes values into the environment
 *	exit()		Exit the process
 *	access()	Check accessibility of a file
 *	malloc()	Allocate a block of main memory
 *	free()		Free allocated space
 *	lseek()		Seek within a file 
 *	open()		Open a file
 *	close()		Close an open file
 */
 
extern	int		optind;		/* argv[] index of next arg */
extern	int		opterr;		/* TRUE if getopt() is to print msgs */
extern	char	       *optarg;		/* Argument to parsed option */
extern	int		getopt();	/* Get an option from the command line */
extern	int	       	putenv();	/* Put a value into the environment */
extern	void		exit();		/* Exit the process */
extern	int		access();	/* Check the accessibility of a file */
extern	void	       *malloc();	/* Get a chunk of main memory */
extern	void		free();		/* Free alloc'd space */
extern	long		lseek();	/* Seek within a file */
extern	int		open();		/* Open a file */
extern	int		close();	/* Close an open a file */

/*
 *  L O C A L   D E F I N I T I O N S
 */

/*
 * Constants 
 */

#ifndef	TRUE
#define	TRUE		(1)
#endif

#ifndef	FALSE
#define	FALSE		(0)
#endif

#ifndef	NULL
#define	NULL		(0)
#endif

#define	OPTSTRING	"tn:s:"

#define	EX_OK		0
#define	EX_ERROR	1

#define	RC_FILENAME	"/etc/rc2.d/S18setuname"
#define RC_DIRNAME	"/etc/rc2.d"


/*
 *  Messages
 */

#define	E_USAGE		"usage: setuname [-t] [-s name] [-n node]"
#define	E_MISSING	"Either -s name or -n node must be specified"
#define	E_UNAME		"Unable to get existing uname values"
#define E_INVNAME	"System-name invalid: %s"
#define E_LONGNAME	"System-name too long: %s"
#define E_INVNODE	"Network node-name invalid: %s"
#define E_LONGNODE	"Network node-name too long: %s"
#define E_NOPERMS	"No permissions, request denied"
#define E_NOSUCHDIR	"Directory doesn't exist: %s"
#define	E_INTERNAL	"Internal error: %d"

/*
 * Macros:
 *	stdmsg(r,l,s,t)	    Write a standard message.  
 *				'r' is the recoverability flag
 *				'l' is the label
 *				's' is the severity 
 *				't' is the text.
 *	strend(p)	    Return the address of the end of a string
 *			    (This is supposed to be defined in <sys/inline.h>
 *			    but that file has string-handing def'ns that
 *			    conflict with <string.h>, so we can't use it!
 *			    MR dn89-04701 requests this fix.
 */
 
#define	stdmsg(r,l,s,t)	(void) fmtmsg(MM_PRINT|MM_UTIL|r,l,s,t,MM_NULLACT,MM_NULLTAG)
#define strend(p)       strrchr(p,'\0')

/*
 * Local functions:
 *	setuname	Changes the system name and the network node name 
 */

static int	setuname();		/* This does the "real" work */


/*
 * Local data
 *	lbl		Buffer for the standard message label
 *	txt		Buffer for the standard message text
 */
 
static	char		lbl[MM_MXLABELLN+1];	/* Space for std msg label */
static	char		msg[MM_MXTXTLN+1];	/* Space for std msg text  */

/*
 *  int main(argc, argv)
 *	int	argc
 *	char   *argv;
 */

int 
main(argc, argv)
	int	argc;			/* Argument count  */
	char   *argv[];			/* Argument vector */
{
	/* Automatic data */
	char	       *n_arg;			/* Ptr to arg for -n */
	char	       *s_arg;			/* Ptr to arg for -s */
	int		t_seen;			/* FLAG, -t option seen */
	char	       *cmdname;		/* Ptr to the command's name */
	char	       *p;			/* Temp pointer */
	int		usageerr;		/* FLAG, TRUE if usage error */
	int		exitcode;		/* Value to exit with */
	int		c;			/* Temp character */
	int		ok;			/* Flag, everything okay? */

	/* Build the standard-message label */
	if (p = strrchr(argv[0], '/')) cmdname = p+1;
	else cmdname = argv[0];
	(void) strcat(strcpy(lbl, "UX:"), cmdname);

	/* Make only the text in standard messages appear (SVR4.0 only) */
	(void) putenv("MSGVERB=text");

	
	/* Initializations */
	n_arg = s_arg = (char *) NULL;
	t_seen = FALSE;


	/* 
	 * Parse command 
	 */

	usageerr = FALSE;
	opterr = FALSE;
	while (!usageerr && (c = getopt(argc, argv, OPTSTRING)) != EOF) switch(c) {

	case 'n':			/* -n node */
	    if (n_arg) usageerr = TRUE;
	    else n_arg = optarg;
	    break;

	case 's':			/* -s name */
	    if (s_arg) usageerr = TRUE;
	    else s_arg = optarg;
	    break;

	case 't':			/* -t */
	    if (t_seen) usageerr = TRUE;
	    else t_seen = TRUE;
	    break;
	    
	default:			/* Something that doesn't exist */
	    usageerr = TRUE;
	}   /* switch() */

	/* If there was a usage error, report the error and exit */
	if ((argc >= (optind+1)) || usageerr) {
	    stdmsg(MM_NRECOV, lbl, MM_ERROR, E_USAGE);
	    exit(EX_ERROR);
	}

	/* Either -n <node> or -s <name> has to be specified */
	if (!(n_arg || s_arg)) {
	    stdmsg(MM_NRECOV, lbl, MM_ERROR, E_MISSING);
	    exit(EX_ERROR);
	}


	/* 
	 * Validate arguments:
	 *  - The length of the system name must be less than SYS_NMLN-1
	 *    characters,
	 *  - The length of the network node-name must be less than 
	 *    SYS_NMLN-1 characters,
	 *  - The system name must equal [a-zA-Z0-9-_]+,
	 *  - The network node-name must equal [a-zA-Z0-9-_]+.
	 */

	/* Check the length and the character-set of the system name */
	if (s_arg) {

	    /* Check length of the system name */
	    if (strlen(s_arg) > (size_t)(SYS_NMLN-1)) {
		(void) sprintf(msg, E_LONGNAME, s_arg);
		stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
		exit(EX_ERROR);
	    }

	    /* Check the character-set */
	    ok = TRUE;
	    for (p = s_arg ; ok && *p ; p++) {
		if (!isalnum(*p) && (*p != '-') && (*p != '_')) ok = FALSE;
	    }
	    if (!ok || (p == s_arg)) {
		(void) sprintf(msg, E_INVNAME, s_arg);
		stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
		exit(EX_ERROR);
	    }
	}

	/* Check the length and the character-set of the network node-name */

	if (n_arg) {

	    /* Check length of the network node-name */
	    if (strlen(n_arg) > (size_t)(SYS_NMLN-1)) {
		(void) sprintf(msg, E_LONGNODE, n_arg);
		stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
		exit(EX_ERROR);
	    }

	    /* Check the character-set */
	    ok = TRUE;
	    for (p = n_arg ; ok && *p ; p++) {
		if (!isalnum(*p) && (*p != '-') && (*p != '_')) ok = FALSE;
	    }
	    if (!ok || (p == n_arg)) {
		(void) sprintf(msg, E_INVNODE, n_arg);
		stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
		exit(EX_ERROR);
	    }
	}


	/*
	 * Make sure we have access to needed resources:
	 *   -  Read/write access to kernel memory (/dev/kmem)
	 *   -  If -t is not specified, read/write access to /etc/rc2.d
	 *   -  If -t is not specified, read access to /etc/rc2.d/S18setuname
	 */
	
	if (access("/dev/kmem", R_OK|W_OK) == 0) {
	    if (access(RC_DIRNAME, R_OK|W_OK) == 0) {
		if ((access(RC_FILENAME, R_OK) != 0) && 
		    (access(RC_FILENAME, F_OK) == 0)) {
		    stdmsg(MM_NRECOV, lbl, MM_ERROR, E_NOPERMS);
		    exit(EX_ERROR);
		}
	    } 
	    else {
		if (access(RC_DIRNAME, F_OK) == 0) {
		    stdmsg(MM_NRECOV, lbl, MM_ERROR, E_NOPERMS);
		    exit(EX_ERROR);
		} 
		else {
		    (void) sprintf(msg, E_NOSUCHDIR, RC_DIRNAME);
		    stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
		    exit(EX_ERROR);
		}
	    }
	} 
	else {
	    stdmsg(MM_NRECOV, lbl, MM_ERROR, E_NOPERMS);
	    exit(EX_ERROR);
	}


	/* Attempt the setuname */
	if (setuname(t_seen, s_arg, n_arg) == 0) exitcode = EX_OK;
	else {
	    (void) sprintf(msg, E_INTERNAL, errno);
	    stdmsg(MM_NRECOV, lbl, MM_ERROR, msg);
	    exitcode = EX_ERROR;
	}

	/* Finished */
	return (exitcode);
}  /* main() */

/*
 * int setuname(temp, name, node)
 *	int	temp
 *	char   *name
 *	char   *node
 *
 *	Set any or all of the following machine parameters, either
 *	temporarily or permanently, depending on <temp>.
 *	    - System name
 *	    - Network Node-name
 */

static int 
setuname(temp, sysname, nodename)
	int	temp;		/* Set in kernel only flag */
	char   *sysname;	/* System name */
	char   *nodename;	/* Network node-name */
{
	/* Automatic Data */
	struct utsname	utsname;	/* Space for the kernel's utsname information */
#if u3b || u3b15 || u3b2
	struct s3bsym  *symbtbl;	/* The kernel's symbol table */
#endif
#if sun
	struct nlist nl[] = {
		{"utsname", 0, 0, 0, 0, 0},
		{NULL}
	};
	kvm_t *kd;
#endif
	uintptr_t	utsname_addr;	/* Addr of "utsname" in the kernel */
	char	       *sysnm = (char *)NULL;		/* System name to set (from file or arg) */
	char	       *nodenm = (char *)NULL;		/* Network node-name to set (from file or arg) */
	FILE	       *fd;		/* Std I/O File Descriptor for /etc/rc2.d/S18setuname */
	char	       *p;		/* Temp pointer */
	void	      (*oldsighup)();	/* Function to call for SIGHUP */
	void	      (*oldsigint)();	/* Function to call for SIGINT */
	int		rtncd;		/* Value to return to the caller */
	unsigned long	symbtblsz;	/* The size of the kernel's symbol table, in bytes */
	int		memfd;		/* File descriptor:  open kernel memory */
	int		i;		/* Temp counter */	


	/* Nothing's gone wrong yet (but we've only just begun!) */
	rtncd = 0;


	/*
	 * Get the virtual address of the symbol "utsname" in the kernel
	 * so we can get set the system name and/or the network node-name
	 * directly in the kernel's memory space.
	 */

#if u3b || u3b15 || u3b2
	if ((sys3b(S3BSYM, (struct s3bsym *) &symbtblsz, sizeof(symbtblsz)) == 0) &&
	    (symbtbl = (struct s3bsym *) malloc(symbtblsz))) {

	    (void) sys3b(S3BSYM, symbtbl, symbtblsz);
	    p = (char *) symbtbl;
	    for (i = symbtbl->count; i-- && (strcmp(p, "utsname") != 0) ; p = S3BNXTSYM(p)) ;
	    if (i >= 0) utsname_addr = S3BSVAL(p);
	    else rtncd = -1;
	    free((void *) symbtbl);

	} else rtncd = -1;

#elif sun
        /* Check out namelist and memory files. */
	if ((kd = kvm_open(NULL, NULL, NULL, O_RDWR, NULL)) == NULL)
		rtncd = -1;
	else if (kvm_nlist(kd, nl) != 0)
		rtncd = -1;
	else if (nl[0].n_value == 0)
		rtncd = -1;
	else
		utsname_addr = (uintptr_t)nl[0].n_value;
#else
	if (nlist("/unix", nl) != 0)
		rtncd = -1;
#endif
	if (rtncd != 0) return(rtncd);

	/* 
	 * Open the kernel's memory, get the existing "utsname" structure,
	 * change the system name and/or the network node-name in that struct,
	 * write it back out to kernel memory, then close kernel memory.
	 */
#ifdef sun
	if (kvm_kread(kd, utsname_addr, &utsname, sizeof (utsname)) ==
	    sizeof (utsname)) {
		if (sysname)
			(void) strncpy(utsname.sysname, sysname,
			    sizeof (utsname.sysname));
		if (nodename)
			(void) strncpy(utsname.nodename, nodename,
			    sizeof (utsname.nodename));
		(void) kvm_kwrite(kd, utsname_addr, &utsname, sizeof (utsname));
		kvm_close(kd);
	} else
		return (-1);
#else /* sun */
	if ((memfd = open("/dev/kmem", O_RDWR, 0)) > 0) {
	    if ((lseek(memfd, (long) utsname_addr, SEEK_SET) != -1) && 
		(read(memfd, &utsname, sizeof(utsname)) == sizeof(utsname))) {
		if (sysname) (void) strncpy(utsname.sysname, sysname, sizeof(utsname.sysname));
		if (nodename) (void) strncpy(utsname.nodename, nodename, sizeof(utsname.nodename));
		(void) lseek(memfd, (long) utsname_addr, SEEK_SET);
		(void) write(memfd, &utsname, sizeof(utsname));
		(void) close(memfd);
	    } else rtncd = -1;
	} else rtncd = -1;
	if (rtncd != 0) return(rtncd);
#endif /* sun */


	/*
	 * If the "temp" flag is FALSE, we need to permanently set the
	 * system name in the file  /etc/rc2.d/S18setuname
	 */

	if (!temp) {
	    /* 
	     * If a name was specified by the caller, use that, otherwise, use
	     * whatever was in the "rc" file.
	     */

		if (sysname) sysnm = sysname;
		if (nodename) nodenm = nodename;


	    /* 
	     * Write the file /etc/rc2.d/S18setuname so that the system name is
	     * set on boots and state changes.  
	     *
	     * DISABLED SIGNALS: SIGHUP, SIGINT
	     */

	    /* Give us a reasonable chance to complete without interruptions */
		oldsighup = signal(SIGHUP, SIG_IGN);
		oldsigint = signal(SIGINT, SIG_IGN);

	    /* Write the new setuname "rc" file */
		if (sysname != NULL) {
			if ((fd = fopen(RC_FILENAME, "w")) != (FILE *) NULL) {
				(void) fprintf(fd, "# %s\n", sysnm);
				(void) fprintf(fd, "#\n");
				(void) fprintf(fd, "# This script, generated by the setuname command,\n");
				(void) fprintf(fd, "# sets the system's system-name\n");
				(void) fprintf(fd, "#\n");
			if (sysnm && *sysnm) 
				(void) fprintf(fd, "setuname -t -s %s\n", sysnm);
			(void) fclose(fd);
			} else return(rtncd = -1);
		}

		if(nodename != NULL) {
			char curname[SYS_NMLN];
			int curlen;
			FILE *file;

			if ((file = fopen("/etc/nodename", "r")) != NULL) {
				curlen = fread(curname, sizeof(char), SYS_NMLN, file);
				for (i = 0; i < curlen; i++) {
					if (curname[i] == '\n') {
						curname[i] = '\0';
						break;
					}
				}
				if (i == curlen) {
					curname[curlen] = '\0';
				}
				(void)fclose(file);
			} else {
				curname[0] = '\0';
			}
			if (strcmp(curname, nodenm) != 0) {
				if ((file = fopen("/etc/nodename", "w")) == NULL) {
					(void) fprintf(stderr, "setuname: error in writing name\n");
					exit(1);
				} 
				if (fprintf(file, "%s\n", nodenm) < 0) {
					(void) fprintf(stderr, "setuname: error in writing name\n");
					exit(1);
				}
				(void)fclose(file);
			}		
		}
	    /* Restore signal handling */
		(void) signal(SIGHUP, oldsighup);
		(void) signal(SIGINT, oldsigint);
	}	/* if (!temp) */

	/* Fini */
	return(rtncd);
}