Blame elf/sln.c

Packit 6c4009
/* `sln' program to create symbolic links between files.
Packit 6c4009
   Copyright (C) 1998-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
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 <error.h>
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <libintl.h>
Packit 6c4009
#include <locale.h>
Packit 6c4009
#include <sys/types.h>
Packit 6c4009
#include <sys/stat.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <ctype.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <limits.h>
Packit 6c4009
Packit 6c4009
#include "../version.h"
Packit 6c4009
Packit 6c4009
#define PACKAGE _libc_intl_domainname
Packit 6c4009
Packit 6c4009
static int makesymlink (const char *src, const char *dest);
Packit 6c4009
static int makesymlinks (const char *file);
Packit 6c4009
static void usage (void);
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
main (int argc, char **argv)
Packit 6c4009
{
Packit 6c4009
  /* Set locale via LC_ALL.  */
Packit 6c4009
  setlocale (LC_ALL, "");
Packit 6c4009
Packit 6c4009
  /* Set the text message domain.  */
Packit 6c4009
  textdomain (PACKAGE);
Packit 6c4009
Packit 6c4009
  switch (argc)
Packit 6c4009
    {
Packit 6c4009
    case 2:
Packit 6c4009
      if (strcmp (argv[1], "--version") == 0) {
Packit 6c4009
	printf ("sln %s%s\n", PKGVERSION, VERSION);
Packit 6c4009
	return 0;
Packit 6c4009
      } else if (strcmp (argv[1], "--help") == 0) {
Packit 6c4009
	usage ();
Packit 6c4009
	return 0;
Packit 6c4009
      }
Packit 6c4009
      return makesymlinks (argv [1]);
Packit 6c4009
      break;
Packit 6c4009
Packit 6c4009
    case 3:
Packit 6c4009
      return makesymlink (argv [1], argv [2]);
Packit 6c4009
      break;
Packit 6c4009
Packit 6c4009
    default:
Packit 6c4009
      usage ();
Packit 6c4009
      return 1;
Packit 6c4009
      break;
Packit 6c4009
    }
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static void
Packit 6c4009
usage (void)
Packit 6c4009
{
Packit 6c4009
  printf (_("Usage: sln src dest|file\n\n"));
Packit 6c4009
  printf (_("For bug reporting instructions, please see:\n\
Packit 6c4009
%s.\n"), REPORT_BUGS_TO);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static int
Packit 6c4009
makesymlinks (const char *file)
Packit 6c4009
{
Packit 6c4009
  char *buffer = NULL;
Packit 6c4009
  size_t bufferlen = 0;
Packit 6c4009
  int ret;
Packit 6c4009
  int lineno;
Packit 6c4009
  FILE *fp;
Packit 6c4009
Packit 6c4009
  if (strcmp (file, "-") == 0)
Packit 6c4009
    fp = stdin;
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      fp = fopen (file, "r");
Packit 6c4009
      if (fp == NULL)
Packit 6c4009
	{
Packit 6c4009
	  fprintf (stderr, _("%s: file open error: %m\n"), file);
Packit 6c4009
	  return 1;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  ret = 0;
Packit 6c4009
  lineno = 0;
Packit 6c4009
  while (!feof_unlocked (fp))
Packit 6c4009
    {
Packit 6c4009
      ssize_t n = getline (&buffer, &bufferlen, fp);
Packit 6c4009
      char *src;
Packit 6c4009
      char *dest;
Packit 6c4009
      char *cp = buffer;
Packit 6c4009
Packit 6c4009
      if (n < 0)
Packit 6c4009
	break;
Packit 6c4009
      if (buffer[n - 1] == '\n')
Packit 6c4009
	buffer[n - 1] = '\0';
Packit 6c4009
Packit 6c4009
      ++lineno;
Packit 6c4009
      while (isspace (*cp))
Packit 6c4009
	++cp;
Packit 6c4009
      if (*cp == '\0')
Packit 6c4009
	/* Ignore empty lines.  */
Packit 6c4009
	continue;
Packit 6c4009
      src = cp;
Packit 6c4009
Packit 6c4009
      do
Packit 6c4009
	++cp;
Packit 6c4009
      while (*cp != '\0' && ! isspace (*cp));
Packit 6c4009
      if (*cp != '\0')
Packit 6c4009
	*cp++ = '\0';
Packit 6c4009
Packit 6c4009
      while (isspace (*cp))
Packit 6c4009
	++cp;
Packit 6c4009
      if (*cp == '\0')
Packit 6c4009
	{
Packit 6c4009
	  fprintf (stderr, _("No target in line %d\n"), lineno);
Packit 6c4009
	  ret = 1;
Packit 6c4009
	  continue;
Packit 6c4009
	}
Packit 6c4009
      dest = cp;
Packit 6c4009
Packit 6c4009
      do
Packit 6c4009
	++cp;
Packit 6c4009
      while (*cp != '\0' && ! isspace (*cp));
Packit 6c4009
      if (*cp != '\0')
Packit 6c4009
	*cp++ = '\0';
Packit 6c4009
Packit 6c4009
      ret |= makesymlink (src, dest);
Packit 6c4009
    }
Packit 6c4009
  fclose (fp);
Packit 6c4009
Packit 6c4009
  return ret;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
static int
Packit 6c4009
makesymlink (const char *src, const char *dest)
Packit 6c4009
{
Packit 6c4009
  struct stat64 stats;
Packit 6c4009
  const char *error;
Packit 6c4009
Packit 6c4009
  /* Destination must not be a directory. */
Packit 6c4009
  if (lstat64 (dest, &stats) == 0)
Packit 6c4009
    {
Packit 6c4009
      if (S_ISDIR (stats.st_mode))
Packit 6c4009
	{
Packit 6c4009
	  fprintf (stderr, _("%s: destination must not be a directory\n"),
Packit 6c4009
		   dest);
Packit 6c4009
	  return 1;
Packit 6c4009
	}
Packit 6c4009
      else if (unlink (dest) && errno != ENOENT)
Packit 6c4009
	{
Packit 6c4009
	  fprintf (stderr, _("%s: failed to remove the old destination\n"),
Packit 6c4009
		   dest);
Packit 6c4009
	  return 1;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
  else if (errno != ENOENT)
Packit 6c4009
    {
Packit 6c4009
      error = strerror (errno);
Packit 6c4009
      fprintf (stderr, _("%s: invalid destination: %s\n"), dest, error);
Packit 6c4009
      return -1;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (symlink (src, dest) == 0)
Packit 6c4009
    {
Packit 6c4009
      /* Destination must exist by now. */
Packit 6c4009
      if (access (dest, F_OK))
Packit 6c4009
        {
Packit 6c4009
	  error = strerror (errno);
Packit 6c4009
	  unlink (dest);
Packit 6c4009
	  fprintf (stderr, _("Invalid link from \"%s\" to \"%s\": %s\n"),
Packit 6c4009
		   src, dest, error);
Packit 6c4009
	  return 1;
Packit 6c4009
	}
Packit 6c4009
      return 0;
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      error = strerror (errno);
Packit 6c4009
      fprintf (stderr, _("Invalid link from \"%s\" to \"%s\": %s\n"),
Packit 6c4009
	       src, dest, error);
Packit 6c4009
      return 1;
Packit 6c4009
    }
Packit 6c4009
}