Minix2.0/src/commands/simple/su.c

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

/* su - become super-user		Author: Patrick van Kleef */

#include <sys/types.h>
#include <pwd.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <minix/minlib.h>

/* True iff the invoker need not give a password. */
#define privileged()	(getgid() == 0)

_PROTOTYPE(int main, (int argc, char **argv));

int main(argc, argv)
int argc;
char *argv[];
{
  register char *name, *password;
  char *shell;
  char arg0[20];
  static char shell1[] = "/bin/sh";
  static char shell2[] = "/usr/bin/sh";
  int nr, login_shell = 0;
  register struct passwd *pwd;
  static char USER[20], LOGNAME[25], HOME[100], SHELL[100];

  if (argc > 1 && strcmp(argv[1], "-") == 0) {
	login_shell = 1;		/* Read .profile */
	argv[1] = argv[0];
	argv++;
	argc--;
  }
  if (argc > 1) {
	name = argv[1];
	argv[1] = argv[0];
	argv++;
  } else {
	name = "root";
  }

  if ((pwd = getpwnam(name)) == 0) {
	std_err("Unknown id: ");
	std_err(name);
	std_err("\n");
	exit(1);
  }
  if (!privileged() && strcmp(pwd->pw_passwd, crypt("", pwd->pw_passwd)) != 0) {
	password = getpass("Password:");
	if (strcmp(pwd->pw_passwd, crypt(password, pwd->pw_passwd))) {
		std_err("Sorry\n");
		exit(2);
	}
  }
  if (login_shell) {
	if ((shell = pwd->pw_shell)[0] == 0) shell = shell1;
  } else {
	if ((shell = getenv("SHELL")) == NULL) shell = shell1;
  }
  if (access(shell, 0) < 0) shell = shell2;
  if ((argv[0] = strrchr(shell, '/')) == NULL) argv[0] = shell; else argv[0]++;

  if (login_shell) {
	arg0[0] = '-';
	strncpy(arg0+1, argv[0], sizeof(arg0)-2);
	arg0[sizeof(arg0)-1] = 0;
	argv[0] = arg0;
	strcpy(USER, "USER=");
	strcpy(USER + 5, name);
	putenv(USER);
	strcpy(LOGNAME, "LOGNAME=");
	strcpy(LOGNAME + 8, name);
	putenv(LOGNAME);
	strcpy(SHELL, "SHELL=");
	strcpy(SHELL + 6, shell);
	putenv(SHELL);
	strcpy(HOME, "HOME=");
	strcpy(HOME + 5, pwd->pw_dir);
	putenv(HOME);
	(void) chdir(pwd->pw_dir);
  }
  setgid(pwd->pw_gid);
  setuid(pwd->pw_uid);
  execv(shell, argv);
  std_err("No shell\n");
  return(3);
}