Blame sysdeps/mach/hurd/execve.c

Packit 6c4009
/* Copyright (C) 1991-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 <unistd.h>
Packit 6c4009
#include <hurd.h>
Packit 6c4009
#include <fcntl.h>
Packit 6c4009
#include <stdlib.h>
Packit 6c4009
#include <stdio.h>
Packit 6c4009
Packit 6c4009
/* Replace the current process, executing FILE_NAME with arguments ARGV and
Packit 6c4009
   environment ENVP.  ARGV and ENVP are terminated by NULL pointers.  */
Packit 6c4009
int
Packit 6c4009
__execve (const char *file_name, char *const argv[], char *const envp[])
Packit 6c4009
{
Packit 6c4009
  error_t err;
Packit 6c4009
  char *concat_name = NULL;
Packit 6c4009
  const char *abs_path;
Packit 6c4009
Packit 6c4009
  file_t file = __file_name_lookup (file_name, O_EXEC, 0);
Packit 6c4009
  if (file == MACH_PORT_NULL)
Packit 6c4009
    return -1;
Packit 6c4009
Packit 6c4009
  if (file_name[0] == '/')
Packit 6c4009
    {
Packit 6c4009
      /* Already an absolute path */
Packit 6c4009
      abs_path = file_name;
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      /* Relative path */
Packit 6c4009
      char *cwd = __getcwd (NULL, 0);
Packit 6c4009
      if (cwd == NULL)
Packit 6c4009
	{
Packit 6c4009
	  __mach_port_deallocate (__mach_task_self (), file);
Packit 6c4009
	  return -1;
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      int res = __asprintf (&concat_name, "%s/%s", cwd, file_name);
Packit 6c4009
      free (cwd);
Packit 6c4009
      if (res == -1)
Packit 6c4009
	{
Packit 6c4009
	  __mach_port_deallocate (__mach_task_self (), file);
Packit 6c4009
	  return -1;
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      abs_path = concat_name;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  /* Hopefully this will not return.  */
Packit 6c4009
  err = _hurd_exec_paths (__mach_task_self (), file,
Packit 6c4009
			  file_name, abs_path, argv, envp);
Packit 6c4009
Packit 6c4009
  /* Oh well.  Might as well be tidy.  */
Packit 6c4009
  __mach_port_deallocate (__mach_task_self (), file);
Packit 6c4009
  free (concat_name);
Packit 6c4009
Packit 6c4009
  return __hurd_fail (err);
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
weak_alias (__execve, execve)