Blame login/login.c

Packit Service 82fcde
/* Copyright (C) 1996-2018 Free Software Foundation, Inc.
Packit Service 82fcde
   This file is part of the GNU C Library.
Packit Service 82fcde
   Contributed by Ulrich Drepper <drepper@cygnus.com>, 1996.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is free software; you can redistribute it and/or
Packit Service 82fcde
   modify it under the terms of the GNU Lesser General Public
Packit Service 82fcde
   License as published by the Free Software Foundation; either
Packit Service 82fcde
   version 2.1 of the License, or (at your option) any later version.
Packit Service 82fcde
Packit Service 82fcde
   The GNU C Library is distributed in the hope that it will be useful,
Packit Service 82fcde
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 82fcde
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 82fcde
   Lesser General Public License for more details.
Packit Service 82fcde
Packit Service 82fcde
   You should have received a copy of the GNU Lesser General Public
Packit Service 82fcde
   License along with the GNU C Library; if not, see
Packit Service 82fcde
   <http://www.gnu.org/licenses/>.  */
Packit Service 82fcde
Packit Service 82fcde
#include <assert.h>
Packit Service 82fcde
#include <errno.h>
Packit Service 82fcde
#include <limits.h>
Packit Service 82fcde
#include <string.h>
Packit Service 82fcde
#include <unistd.h>
Packit Service 82fcde
#include <stdlib.h>
Packit Service 82fcde
#include <utmp.h>
Packit Service 82fcde
Packit Service 82fcde
Packit Service 82fcde
/* Return the result of ttyname in the buffer pointed to by TTY, which should
Packit Service 82fcde
   be of length BUF_LEN.  If it is too long to fit in this buffer, a
Packit Service 82fcde
   sufficiently long buffer is allocated using malloc, and returned in TTY.
Packit Service 82fcde
   0 is returned upon success, -1 otherwise.  */
Packit Service 82fcde
static int
Packit Service 82fcde
tty_name (int fd, char **tty, size_t buf_len)
Packit Service 82fcde
{
Packit Service 82fcde
  int rv;			/* Return value.  0 = success.  */
Packit Service 82fcde
  char *buf = *tty;		/* Buffer for ttyname, initially the user's. */
Packit Service 82fcde
Packit Service 82fcde
  for (;;)
Packit Service 82fcde
    {
Packit Service 82fcde
      char *new_buf;
Packit Service 82fcde
Packit Service 82fcde
      if (buf_len)
Packit Service 82fcde
	{
Packit Service 82fcde
	  rv = ttyname_r (fd, buf, buf_len);
Packit Service 82fcde
Packit Service 82fcde
	  if (rv != 0 || memchr (buf, '\0', buf_len))
Packit Service 82fcde
	    /* We either got an error, or we succeeded and the
Packit Service 82fcde
	       returned name fit in the buffer.  */
Packit Service 82fcde
	    break;
Packit Service 82fcde
Packit Service 82fcde
	  /* Try again with a longer buffer.  */
Packit Service 82fcde
	  buf_len += buf_len;	/* Double it */
Packit Service 82fcde
	}
Packit Service 82fcde
      else
Packit Service 82fcde
	/* No initial buffer; start out by mallocing one.  */
Packit Service 82fcde
	buf_len = 128;		/* First time guess.  */
Packit Service 82fcde
Packit Service 82fcde
      if (buf != *tty)
Packit Service 82fcde
	/* We've already malloced another buffer at least once.  */
Packit Service 82fcde
	new_buf = realloc (buf, buf_len);
Packit Service 82fcde
      else
Packit Service 82fcde
	new_buf = malloc (buf_len);
Packit Service 82fcde
      if (! new_buf)
Packit Service 82fcde
	{
Packit Service 82fcde
	  rv = -1;
Packit Service 82fcde
	  __set_errno (ENOMEM);
Packit Service 82fcde
	  break;
Packit Service 82fcde
	}
Packit Service 82fcde
      buf = new_buf;
Packit Service 82fcde
    }
Packit Service 82fcde
Packit Service 82fcde
  if (rv == 0)
Packit Service 82fcde
    *tty = buf;		/* Return buffer to the user.  */
Packit Service 82fcde
  else if (buf != *tty)
Packit Service 82fcde
    free (buf);		/* Free what we malloced when returning an error.  */
Packit Service 82fcde
Packit Service 82fcde
  return rv;
Packit Service 82fcde
}
Packit Service 82fcde

Packit Service 82fcde
void
Packit Service 82fcde
login (const struct utmp *ut)
Packit Service 82fcde
{
Packit Service 82fcde
#ifdef PATH_MAX
Packit Service 82fcde
  char _tty[PATH_MAX + UT_LINESIZE];
Packit Service 82fcde
#else
Packit Service 82fcde
  char _tty[512 + UT_LINESIZE];
Packit Service 82fcde
#endif
Packit Service 82fcde
  char *tty = _tty;
Packit Service 82fcde
  int found_tty;
Packit Service 82fcde
  const char *ttyp;
Packit Service 82fcde
  struct utmp copy = *ut;
Packit Service 82fcde
Packit Service 82fcde
  /* Fill in those fields we supply.  */
Packit Service 1c5418
#if _HAVE_UT_TYPE - 0
Packit Service 82fcde
  copy.ut_type = USER_PROCESS;
Packit Service 1c5418
#endif
Packit Service 1c5418
#if _HAVE_UT_PID - 0
Packit Service 82fcde
  copy.ut_pid = getpid ();
Packit Service 1c5418
#endif
Packit Service 82fcde
Packit Service 82fcde
  /* Seek tty.  */
Packit Service 82fcde
  found_tty = tty_name (STDIN_FILENO, &tty, sizeof (_tty));
Packit Service 82fcde
  if (found_tty < 0)
Packit Service 82fcde
    found_tty = tty_name (STDOUT_FILENO, &tty, sizeof (_tty));
Packit Service 82fcde
  if (found_tty < 0)
Packit Service 82fcde
    found_tty = tty_name (STDERR_FILENO, &tty, sizeof (_tty));
Packit Service 82fcde
Packit Service 82fcde
  if (found_tty >= 0)
Packit Service 82fcde
    {
Packit Service 82fcde
      /* We only want to insert the name of the tty without path.
Packit Service 82fcde
	 But take care of name like /dev/pts/3.  */
Packit Service 82fcde
      if (strncmp (tty, "/dev/", 5) == 0)
Packit Service 82fcde
	ttyp = tty + 5;		/* Skip the "/dev/".  */
Packit Service 82fcde
      else
Packit Service 82fcde
	ttyp = basename (tty);
Packit Service 82fcde
Packit Service 82fcde
      /* Position to record for this tty.  */
Packit Service 82fcde
      strncpy (copy.ut_line, ttyp, UT_LINESIZE);
Packit Service 82fcde
Packit Service 82fcde
      /* Tell that we want to use the UTMP file.  */
Packit Service 82fcde
      if (utmpname (_PATH_UTMP) == 0)
Packit Service 82fcde
	{
Packit Service 82fcde
	  /* Open UTMP file.  */
Packit Service 82fcde
	  setutent ();
Packit Service 82fcde
Packit Service 82fcde
	  /* Write the entry.  */
Packit Service 82fcde
	  pututline (©);
Packit Service 82fcde
Packit Service 82fcde
	  /* Close UTMP file.  */
Packit Service 82fcde
	  endutent ();
Packit Service 82fcde
	}
Packit Service 82fcde
Packit Service 82fcde
      if (tty != _tty)
Packit Service 82fcde
	free (tty);		/* Free buffer malloced by tty_name.  */
Packit Service 82fcde
    }
Packit Service 82fcde
  else
Packit Service 82fcde
    /* We provide a default value so that the output does not contain
Packit Service 82fcde
       an random bytes in this field.  */
Packit Service 82fcde
    strncpy (copy.ut_line, "???", UT_LINESIZE);
Packit Service 82fcde
Packit Service 82fcde
  /* Update the WTMP file.  Here we have to add a new entry.  */
Packit Service 82fcde
  updwtmp (_PATH_WTMP, ©);
Packit Service 82fcde
}