Blame lib/getprogname.c

Packit 709fb3
/* Program name management.
Packit 709fb3
   Copyright (C) 2016-2017 Free Software Foundation, Inc.
Packit 709fb3
Packit 709fb3
   This program is free software: you can redistribute it and/or modify
Packit 709fb3
   it under the terms of the GNU General Public License as published by
Packit 709fb3
   the Free Software Foundation; either version 3 of the License, or
Packit 709fb3
   (at your option) any later version.
Packit 709fb3
Packit 709fb3
   This program is distributed in the hope that it will be useful,
Packit 709fb3
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 709fb3
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
Packit 709fb3
   GNU General Public License for more details.
Packit 709fb3
Packit 709fb3
   You should have received a copy of the GNU General Public License
Packit 709fb3
   along with this program.  If not, see <http://www.gnu.org/licenses/>.  */
Packit 709fb3
Packit 709fb3
#include <config.h>
Packit 709fb3
Packit 709fb3
/* Specification.  */
Packit 709fb3
#include "getprogname.h"
Packit 709fb3
Packit 709fb3
#include <errno.h> /* get program_invocation_name declaration */
Packit 709fb3
#include <stdlib.h> /* get __argv declaration */
Packit 709fb3
Packit 709fb3
#ifdef _AIX
Packit 709fb3
# include <unistd.h>
Packit 709fb3
# include <procinfo.h>
Packit 709fb3
# include <string.h>
Packit 709fb3
#endif
Packit 709fb3
Packit 709fb3
#ifdef __MVS__
Packit 709fb3
# ifndef _OPEN_SYS
Packit 709fb3
#  define _OPEN_SYS
Packit 709fb3
# endif
Packit 709fb3
# include <string.h>
Packit 709fb3
# include <sys/ps.h>
Packit 709fb3
#endif
Packit 709fb3
Packit 709fb3
#ifdef __hpux
Packit 709fb3
# include <unistd.h>
Packit 709fb3
# include <sys/param.h>
Packit 709fb3
# include <sys/pstat.h>
Packit 709fb3
# include <string.h>
Packit 709fb3
#endif
Packit 709fb3
Packit 709fb3
#ifdef __sgi
Packit 709fb3
# include <string.h>
Packit 709fb3
# include <unistd.h>
Packit 709fb3
# include <stdio.h>
Packit 709fb3
# include <fcntl.h>
Packit 709fb3
# include <sys/procfs.h>
Packit 709fb3
#endif
Packit 709fb3
Packit 709fb3
#include "dirname.h"
Packit 709fb3
Packit 709fb3
#ifndef HAVE_GETPROGNAME             /* not Mac OS X, FreeBSD, NetBSD, OpenBSD >= 5.4, Cygwin */
Packit 709fb3
char const *
Packit 709fb3
getprogname (void)
Packit 709fb3
{
Packit 709fb3
# if HAVE_DECL_PROGRAM_INVOCATION_SHORT_NAME                /* glibc, BeOS */
Packit 709fb3
  /* https://www.gnu.org/software/libc/manual/html_node/Error-Messages.html */
Packit 709fb3
  return program_invocation_short_name;
Packit 709fb3
# elif HAVE_DECL_PROGRAM_INVOCATION_NAME                    /* glibc, BeOS */
Packit 709fb3
  /* https://www.gnu.org/software/libc/manual/html_node/Error-Messages.html */
Packit 709fb3
  return last_component (program_invocation_name);
Packit 709fb3
# elif HAVE_GETEXECNAME                                     /* Solaris */
Packit 709fb3
  /* http://docs.oracle.com/cd/E19253-01/816-5168/6mbb3hrb1/index.html */
Packit 709fb3
  const char *p = getexecname ();
Packit 709fb3
  if (!p)
Packit 709fb3
    p = "?";
Packit 709fb3
  return last_component (p);
Packit 709fb3
# elif HAVE_DECL___ARGV                                     /* mingw, MSVC */
Packit 709fb3
  /* https://msdn.microsoft.com/en-us/library/dn727674.aspx */
Packit 709fb3
  const char *p = __argv && __argv[0] ? __argv[0] : "?";
Packit 709fb3
  return last_component (p);
Packit 709fb3
# elif HAVE_VAR___PROGNAME                                  /* OpenBSD, QNX */
Packit 709fb3
  /* http://man.openbsd.org/style.9 */
Packit 709fb3
  /* http://www.qnx.de/developers/docs/6.5.0/index.jsp?topic=%2Fcom.qnx.doc.neutrino_lib_ref%2Fp%2F__progname.html */
Packit 709fb3
  /* Be careful to declare this only when we absolutely need it
Packit 709fb3
     (OpenBSD 5.1), rather than when it's available.  Otherwise,
Packit 709fb3
     its mere declaration makes program_invocation_short_name
Packit 709fb3
     malfunction (have zero length) with Fedora 25's glibc.  */
Packit 709fb3
  extern char *__progname;
Packit 709fb3
  const char *p = __progname;
Packit 709fb3
  return p && p[0] ? p : "?";
Packit 709fb3
# elif _AIX                                                 /* AIX */
Packit 709fb3
  /* Idea by Bastien ROUCARIÈS,
Packit 709fb3
     http://lists.gnu.org/archive/html/bug-gnulib/2010-12/msg00095.html
Packit 709fb3
     Reference: http://
Packit 709fb3
   ibm.biz/knowctr#ssw_aix_53/com.ibm.aix.basetechref/doc/basetrf1/getprocs.htm
Packit 709fb3
  */
Packit 709fb3
  static char *p;
Packit 709fb3
  static int first = 1;
Packit 709fb3
  if (first)
Packit 709fb3
    {
Packit 709fb3
      first = 0;
Packit 709fb3
      pid_t pid = getpid ();
Packit 709fb3
      struct procentry64 procs;
Packit 709fb3
      p = (0 < getprocs64 (&procs, sizeof procs, NULL, 0, &pid, 1)
Packit 709fb3
           ? strdup (procs.pi_comm)
Packit 709fb3
           : NULL);
Packit 709fb3
      if (!p)
Packit 709fb3
        p = "?";
Packit 709fb3
    }
Packit 709fb3
  return p;
Packit 709fb3
# elif defined __hpux
Packit 709fb3
  static char *p;
Packit 709fb3
  static int first = 1;
Packit 709fb3
  if (first)
Packit 709fb3
    {
Packit 709fb3
      first = 0;
Packit 709fb3
      pid_t pid = getpid ();
Packit 709fb3
      struct pst_status status;
Packit 709fb3
      p = (0 < pstat_getproc (&status, sizeof status, 0, pid)
Packit 709fb3
           ? strdup (status.pst_ucomm)
Packit 709fb3
           : NULL);
Packit 709fb3
      if (!p)
Packit 709fb3
        p = "?";
Packit 709fb3
    }
Packit 709fb3
  return p;
Packit 709fb3
# elif __MVS__                                              /* z/OS */
Packit 709fb3
  /* https://www.ibm.com/support/knowledgecenter/SSLTBW_2.1.0/com.ibm.zos.v2r1.bpxbd00/rtwgetp.htm */
Packit 709fb3
  static char *p = "?";
Packit 709fb3
  static int first = 1;
Packit 709fb3
  if (first)
Packit 709fb3
    {
Packit 709fb3
      pid_t pid = getpid ();
Packit 709fb3
      int token;
Packit 709fb3
      W_PSPROC buf;
Packit 709fb3
      first = 0;
Packit 709fb3
      memset (&buf, 0, sizeof(buf));
Packit 709fb3
      buf.ps_cmdptr    = (char *) malloc (buf.ps_cmdlen    = PS_CMDBLEN_LONG);
Packit 709fb3
      buf.ps_conttyptr = (char *) malloc (buf.ps_conttylen = PS_CONTTYBLEN);
Packit 709fb3
      buf.ps_pathptr   = (char *) malloc (buf.ps_pathlen   = PS_PATHBLEN);
Packit 709fb3
      if (buf.ps_cmdptr && buf.ps_conttyptr && buf.ps_pathptr)
Packit 709fb3
        {
Packit 709fb3
          for (token = 0; token >= 0;
Packit 709fb3
               token = w_getpsent (token, &buf, sizeof(buf)))
Packit 709fb3
            {
Packit 709fb3
              if (token > 0 && buf.ps_pid == pid)
Packit 709fb3
                {
Packit 709fb3
                  char *s = strdup (last_component (buf.ps_pathptr));
Packit 709fb3
                  if (s)
Packit 709fb3
                    p = s;
Packit 709fb3
                  break;
Packit 709fb3
                }
Packit 709fb3
            }
Packit 709fb3
        }
Packit 709fb3
      free (buf.ps_cmdptr);
Packit 709fb3
      free (buf.ps_conttyptr);
Packit 709fb3
      free (buf.ps_pathptr);
Packit 709fb3
    }
Packit 709fb3
  return p;
Packit 709fb3
# elif defined __sgi                                        /* IRIX */
Packit 709fb3
  char filename[50];
Packit 709fb3
  int fd;
Packit 709fb3
Packit 709fb3
  sprintf (filename, "/proc/pinfo/%d", (int) getpid ());
Packit 709fb3
  fd = open (filename, O_RDONLY);
Packit 709fb3
  if (0 <= fd)
Packit 709fb3
    {
Packit 709fb3
      prpsinfo_t buf;
Packit 709fb3
      int ioctl_ok = 0 <= ioctl (fd, PIOCPSINFO, &buf;;
Packit 709fb3
      close (fd);
Packit 709fb3
      if (ioctl_ok)
Packit 709fb3
        {
Packit 709fb3
          char *name = buf.pr_fname;
Packit 709fb3
          char *namesize = sizeof buf.pr_fname;
Packit 709fb3
          char *namenul = memchr (name, '\0', namesize);
Packit 709fb3
          size_t namelen = namenul ? namenul - name : namesize;
Packit 709fb3
          char *namecopy = malloc (namelen + 1);
Packit 709fb3
          if (namecopy)
Packit 709fb3
            {
Packit 709fb3
              namecopy[namelen] = 0;
Packit 709fb3
              return memcpy (namecopy, name, namelen);
Packit 709fb3
            }
Packit 709fb3
        }
Packit 709fb3
    }
Packit 709fb3
  return NULL;
Packit 709fb3
# else
Packit 709fb3
#  error "getprogname module not ported to this OS"
Packit 709fb3
# endif
Packit 709fb3
}
Packit 709fb3
Packit 709fb3
#endif