/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License, Version 1.0 only * (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 (c) 1997, by Sun Microsystems, Inc. * All rights reserved. */ #pragma ident "%Z%%M% %I% %E% SMI" /* SVr4.0 1.3 */ /* * setuname [-t] [-s name] [-n node] */ /* * Header files referenced: * Standard I/O * Standard UNIX definitions * String handling * Standard message generation * Character types * Error handling * Signal handling * Data types * File control * System Name * sys3b() definitions * Definitions for Sun symbol table entries */ #include #include #include #include #include #include #include #include #include #include #include #include #if u3b || u3b15 || u3b2 #include #endif #if sun #include #include #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 * but that file has string-handing def'ns that * conflict with , 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 or -s 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 . * - 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; 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); }