Blame shadow/lckpwdf.c

Packit 6c4009
/* Handle locking of password file.
Packit 6c4009
   Copyright (C) 1996-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Ulrich Drepper <drepper@cygnus.com>, 1996.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <fcntl.h>
Packit 6c4009
#include <libc-lock.h>
Packit 6c4009
#include <shadow.h>
Packit 6c4009
#include <signal.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
#include <sys/file.h>
Packit 6c4009
#include <sigsetops.h>
Packit 6c4009
Packit 6c4009
#include <kernel-features.h>
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* Name of the lock file.  */
Packit 6c4009
#define PWD_LOCKFILE "/etc/.pwd.lock"
Packit 6c4009
Packit 6c4009
/* How long to wait for getting the lock before returning with an
Packit 6c4009
   error.  */
Packit 6c4009
#define TIMEOUT 15 /* sec */
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* File descriptor for lock file.  */
Packit 6c4009
static int lock_fd = -1;
Packit 6c4009
Packit 6c4009
/* Prevent problems in multithreaded program by using mutex.  */
Packit 6c4009
__libc_lock_define_initialized (static, lock)
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* Prototypes for local functions.  */
Packit 6c4009
static void noop_handler (int __sig);
Packit 6c4009
Packit 6c4009
Packit 6c4009
/* We cannot simply return in error cases.  We have to close the file
Packit 6c4009
   and perhaps restore the signal handler.  */
Packit 6c4009
#define RETURN_CLOSE_FD(code)						      \
Packit 6c4009
  do {									      \
Packit 6c4009
    if ((code) < 0 && lock_fd >= 0)					      \
Packit 6c4009
      {									      \
Packit 6c4009
	__close (lock_fd);						      \
Packit 6c4009
	lock_fd = -1;							      \
Packit 6c4009
      }									      \
Packit 6c4009
    __libc_lock_unlock (lock);						      \
Packit 6c4009
    return (code);							      \
Packit 6c4009
  } while (0)
Packit 6c4009
Packit 6c4009
#define RETURN_RESTORE_HANDLER(code)					      \
Packit 6c4009
  do {									      \
Packit 6c4009
    /* Restore old action handler for alarm.  We don't need to know	      \
Packit 6c4009
       about the current one.  */					      \
Packit 6c4009
    __sigaction (SIGALRM, &saved_act, NULL);				      \
Packit 6c4009
    RETURN_CLOSE_FD (code);						      \
Packit 6c4009
  } while (0)
Packit 6c4009
Packit 6c4009
#define RETURN_CLEAR_ALARM(code)					      \
Packit 6c4009
  do {									      \
Packit 6c4009
    /* Clear alarm.  */							      \
Packit 6c4009
    alarm (0);								      \
Packit 6c4009
    /* Restore old set of handled signals.  We don't need to know	      \
Packit 6c4009
       about the current one.*/						      \
Packit 6c4009
    __sigprocmask (SIG_SETMASK, &saved_set, NULL);			      \
Packit 6c4009
    RETURN_RESTORE_HANDLER (code);					      \
Packit 6c4009
  } while (0)
Packit 6c4009
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
__lckpwdf (void)
Packit 6c4009
{
Packit 6c4009
  sigset_t saved_set;			/* Saved set of caught signals.  */
Packit 6c4009
  struct sigaction saved_act;		/* Saved signal action.  */
Packit 6c4009
  sigset_t new_set;			/* New set of caught signals.  */
Packit 6c4009
  struct sigaction new_act;		/* New signal action.  */
Packit 6c4009
  struct flock fl;			/* Information struct for locking.  */
Packit 6c4009
  int result;
Packit 6c4009
Packit 6c4009
  if (lock_fd != -1)
Packit 6c4009
    /* Still locked by own process.  */
Packit 6c4009
    return -1;
Packit 6c4009
Packit 6c4009
  /* Prevent problems caused by multiple threads.  */
Packit 6c4009
  __libc_lock_lock (lock);
Packit 6c4009
Packit 6c4009
  int oflags = O_WRONLY | O_CREAT | O_CLOEXEC;
Packit 6c4009
  lock_fd = __open (PWD_LOCKFILE, oflags, 0600);
Packit 6c4009
  if (lock_fd == -1)
Packit 6c4009
    /* Cannot create lock file.  */
Packit 6c4009
    RETURN_CLOSE_FD (-1);
Packit 6c4009
Packit 6c4009
  /* Now we have to get exclusive write access.  Since multiple
Packit 6c4009
     process could try this we won't stop when it first fails.
Packit 6c4009
     Instead we set a timeout for the system call.  Once the timer
Packit 6c4009
     expires it is likely that there are some problems which cannot be
Packit 6c4009
     resolved by waiting.
Packit 6c4009
Packit 6c4009
     It is important that we don't change the signal state.  We must
Packit 6c4009
     restore the old signal behaviour.  */
Packit 6c4009
  memset (&new_act, '\0', sizeof (struct sigaction));
Packit 6c4009
  new_act.sa_handler = noop_handler;
Packit 6c4009
  __sigfillset (&new_act.sa_mask);
Packit 6c4009
  new_act.sa_flags = 0ul;
Packit 6c4009
Packit 6c4009
  /* Install new action handler for alarm and save old.  */
Packit 6c4009
  if (__sigaction (SIGALRM, &new_act, &saved_act) < 0)
Packit 6c4009
    /* Cannot install signal handler.  */
Packit 6c4009
    RETURN_CLOSE_FD (-1);
Packit 6c4009
Packit 6c4009
  /* Now make sure the alarm signal is not blocked.  */
Packit 6c4009
  __sigemptyset (&new_set);
Packit 6c4009
  __sigaddset (&new_set, SIGALRM);
Packit 6c4009
  if (__sigprocmask (SIG_UNBLOCK, &new_set, &saved_set) < 0)
Packit 6c4009
    RETURN_RESTORE_HANDLER (-1);
Packit 6c4009
Packit 6c4009
  /* Start timer.  If we cannot get the lock in the specified time we
Packit 6c4009
     get a signal.  */
Packit 6c4009
  alarm (TIMEOUT);
Packit 6c4009
Packit 6c4009
  /* Try to get the lock.  */
Packit 6c4009
  memset (&fl, '\0', sizeof (struct flock));
Packit 6c4009
  fl.l_type = F_WRLCK;
Packit 6c4009
  fl.l_whence = SEEK_SET;
Packit 6c4009
  result = __fcntl (lock_fd, F_SETLKW, &fl);
Packit 6c4009
Packit 6c4009
  RETURN_CLEAR_ALARM (result);
Packit 6c4009
}
Packit 6c4009
weak_alias (__lckpwdf, lckpwdf)
Packit 6c4009
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
__ulckpwdf (void)
Packit 6c4009
{
Packit 6c4009
  int result;
Packit 6c4009
Packit 6c4009
  if (lock_fd == -1)
Packit 6c4009
    /* There is no lock set.  */
Packit 6c4009
    result = -1;
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      /* Prevent problems caused by multiple threads.  */
Packit 6c4009
      __libc_lock_lock (lock);
Packit 6c4009
Packit 6c4009
      result = __close (lock_fd);
Packit 6c4009
Packit 6c4009
      /* Mark descriptor as unused.  */
Packit 6c4009
      lock_fd = -1;
Packit 6c4009
Packit 6c4009
      /* Clear mutex.  */
Packit 6c4009
      __libc_lock_unlock (lock);
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return result;
Packit 6c4009
}
Packit 6c4009
weak_alias (__ulckpwdf, ulckpwdf)
Packit 6c4009
Packit 6c4009
Packit 6c4009
static void
Packit 6c4009
noop_handler (int sig)
Packit 6c4009
{
Packit 6c4009
  /* We simply return which makes the `fcntl' call return with an error.  */
Packit 6c4009
}