2006-02-12 03:02:21 +08:00
|
|
|
/* VMS functions
|
2014-09-01 01:11:08 +08:00
|
|
|
Copyright (C) 1996-2014 Free Software Foundation, Inc.
|
2006-02-12 03:02:21 +08:00
|
|
|
This file is part of GNU Make.
|
|
|
|
|
|
|
|
GNU Make is free software; you can redistribute it and/or modify it under the
|
|
|
|
terms of the GNU General Public License as published by the Free Software
|
2007-07-05 03:35:15 +08:00
|
|
|
Foundation; either version 3 of the License, or (at your option) any later
|
|
|
|
version.
|
2006-02-12 03:02:21 +08:00
|
|
|
|
|
|
|
GNU Make is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
|
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
2007-07-05 03:35:15 +08:00
|
|
|
this program. If not, see <http://www.gnu.org/licenses/>. */
|
1996-03-20 22:57:41 +08:00
|
|
|
|
2013-01-21 00:01:01 +08:00
|
|
|
#include "makeint.h"
|
1999-11-22 14:15:35 +08:00
|
|
|
#include "debug.h"
|
2005-05-03 21:57:20 +08:00
|
|
|
#include "job.h"
|
1999-11-22 14:15:35 +08:00
|
|
|
|
2014-09-01 01:11:08 +08:00
|
|
|
#include <ctype.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
1996-03-20 22:57:41 +08:00
|
|
|
#ifdef __DECC
|
|
|
|
#include <starlet.h>
|
|
|
|
#endif
|
2014-09-01 01:11:08 +08:00
|
|
|
|
1996-03-20 22:57:41 +08:00
|
|
|
#include <rms.h>
|
|
|
|
#include "vmsdir.h"
|
|
|
|
|
2000-01-22 13:59:28 +08:00
|
|
|
#ifdef HAVE_VMSDIR_H
|
1998-07-31 04:54:47 +08:00
|
|
|
|
|
|
|
DIR *
|
2002-10-15 05:54:04 +08:00
|
|
|
opendir (char *dspec)
|
1996-03-20 22:57:41 +08:00
|
|
|
{
|
2009-09-17 01:07:01 +08:00
|
|
|
struct DIR *dir = xcalloc (sizeof (struct DIR));
|
|
|
|
struct NAM *dnam = xmalloc (sizeof (struct NAM));
|
2000-01-22 13:43:03 +08:00
|
|
|
struct FAB *dfab = &dir->fab;
|
2006-04-10 06:09:24 +08:00
|
|
|
char *searchspec = xmalloc (MAXNAMLEN + 1);
|
1998-07-31 04:54:47 +08:00
|
|
|
|
2000-01-22 13:43:03 +08:00
|
|
|
*dfab = cc$rms_fab;
|
|
|
|
*dnam = cc$rms_nam;
|
1998-07-31 04:54:47 +08:00
|
|
|
sprintf (searchspec, "%s*.*;", dspec);
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
|
|
dfab->fab$l_fna = searchspec;
|
1998-07-31 04:54:47 +08:00
|
|
|
dfab->fab$b_fns = strlen (searchspec);
|
1996-03-20 22:57:41 +08:00
|
|
|
dfab->fab$l_nam = dnam;
|
|
|
|
|
|
|
|
*dnam = cc$rms_nam;
|
|
|
|
dnam->nam$l_esa = searchspec;
|
|
|
|
dnam->nam$b_ess = MAXNAMLEN;
|
|
|
|
|
1998-07-31 04:54:47 +08:00
|
|
|
if (! (sys$parse (dfab) & 1))
|
|
|
|
{
|
2000-01-22 13:43:03 +08:00
|
|
|
free (dir);
|
1998-07-31 04:54:47 +08:00
|
|
|
free (dnam);
|
|
|
|
free (searchspec);
|
|
|
|
return (NULL);
|
|
|
|
}
|
1996-03-20 22:57:41 +08:00
|
|
|
|
2000-01-22 13:43:03 +08:00
|
|
|
return dir;
|
1996-03-20 22:57:41 +08:00
|
|
|
}
|
|
|
|
|
1998-07-31 04:54:47 +08:00
|
|
|
#define uppercasify(str) \
|
|
|
|
do \
|
|
|
|
{ \
|
|
|
|
char *tmp; \
|
|
|
|
for (tmp = (str); *tmp != '\0'; tmp++) \
|
1999-10-15 15:00:58 +08:00
|
|
|
if (islower ((unsigned char)*tmp)) \
|
|
|
|
*tmp = toupper ((unsigned char)*tmp); \
|
1998-07-31 04:54:47 +08:00
|
|
|
} \
|
|
|
|
while (0)
|
|
|
|
|
|
|
|
struct direct *
|
2002-10-15 05:54:04 +08:00
|
|
|
readdir (DIR *dir)
|
1996-03-20 22:57:41 +08:00
|
|
|
{
|
2000-01-22 13:43:03 +08:00
|
|
|
struct FAB *dfab = &dir->fab;
|
|
|
|
struct NAM *dnam = (struct NAM *)(dfab->fab$l_nam);
|
|
|
|
struct direct *dentry = &dir->dir;
|
1996-03-20 22:57:41 +08:00
|
|
|
int i;
|
|
|
|
|
2000-01-22 13:43:03 +08:00
|
|
|
memset (dentry, 0, sizeof *dentry);
|
1996-03-20 22:57:41 +08:00
|
|
|
|
2000-01-22 13:43:03 +08:00
|
|
|
dnam->nam$l_rsa = dir->d_result;
|
|
|
|
dnam->nam$b_rss = MAXNAMLEN;
|
1996-03-20 22:57:41 +08:00
|
|
|
|
2000-02-05 15:50:47 +08:00
|
|
|
DB (DB_VERBOSE, ("."));
|
1996-03-20 22:57:41 +08:00
|
|
|
|
2000-01-22 13:43:03 +08:00
|
|
|
if (!((i = sys$search (dfab)) & 1))
|
1998-07-31 04:54:47 +08:00
|
|
|
{
|
2009-06-09 23:35:38 +08:00
|
|
|
DB (DB_VERBOSE, (_("sys$search() failed with %d\n"), i));
|
1998-07-31 04:54:47 +08:00
|
|
|
return (NULL);
|
|
|
|
}
|
1996-03-20 22:57:41 +08:00
|
|
|
|
|
|
|
dentry->d_off = 0;
|
2000-01-22 13:43:03 +08:00
|
|
|
if (dnam->nam$w_fid == 0)
|
1996-03-20 22:57:41 +08:00
|
|
|
dentry->d_fileno = 1;
|
1998-07-31 04:54:47 +08:00
|
|
|
else
|
2000-01-22 13:43:03 +08:00
|
|
|
dentry->d_fileno = dnam->nam$w_fid[0] + (dnam->nam$w_fid[1] << 16);
|
|
|
|
|
1996-03-20 22:57:41 +08:00
|
|
|
dentry->d_reclen = sizeof (struct direct);
|
2000-01-22 13:43:03 +08:00
|
|
|
dentry->d_namlen = dnam->nam$b_name + dnam->nam$b_type;
|
|
|
|
strncpy (dentry->d_name, dnam->nam$l_name, dentry->d_namlen);
|
1996-03-20 22:57:41 +08:00
|
|
|
dentry->d_name[dentry->d_namlen] = '\0';
|
2006-01-04 22:45:16 +08:00
|
|
|
|
|
|
|
#ifdef HAVE_CASE_INSENSITIVE_FS
|
1998-07-31 04:54:47 +08:00
|
|
|
uppercasify (dentry->d_name);
|
2006-01-04 22:45:16 +08:00
|
|
|
#endif
|
1996-03-20 22:57:41 +08:00
|
|
|
|
1998-07-31 04:54:47 +08:00
|
|
|
return (dentry);
|
1996-03-20 22:57:41 +08:00
|
|
|
}
|
|
|
|
|
2000-01-22 13:43:03 +08:00
|
|
|
int
|
2002-10-15 05:54:04 +08:00
|
|
|
closedir (DIR *dir)
|
1996-03-20 22:57:41 +08:00
|
|
|
{
|
2000-01-22 13:43:03 +08:00
|
|
|
if (dir != NULL)
|
1998-07-31 04:54:47 +08:00
|
|
|
{
|
2000-01-22 13:43:03 +08:00
|
|
|
struct FAB *dfab = &dir->fab;
|
|
|
|
struct NAM *dnam = (struct NAM *)(dfab->fab$l_nam);
|
|
|
|
if (dnam != NULL)
|
|
|
|
free (dnam->nam$l_esa);
|
|
|
|
free (dnam);
|
|
|
|
free (dir);
|
1998-07-31 04:54:47 +08:00
|
|
|
}
|
2000-01-22 13:43:03 +08:00
|
|
|
|
|
|
|
return 0;
|
1996-03-20 22:57:41 +08:00
|
|
|
}
|
1998-07-31 04:54:47 +08:00
|
|
|
#endif /* compiled for OpenVMS prior to V7.x */
|
1996-03-20 22:57:41 +08:00
|
|
|
|
2014-09-01 01:11:08 +08:00
|
|
|
/* Argv0 will be a full vms file specification, like
|
|
|
|
node$dka100:[utils.gnumake]make.exe;47
|
|
|
|
prefix it with "mcr " to make it a vms command, executable for DCL. */
|
|
|
|
const char *
|
|
|
|
vms_command(const char* argv0)
|
1996-03-20 22:57:41 +08:00
|
|
|
{
|
2014-09-01 01:11:08 +08:00
|
|
|
size_t l = strlen(argv0) + 1;
|
|
|
|
char* s = xmalloc(l + 4);
|
|
|
|
memcpy(s, "mcr ", 4);
|
|
|
|
memcpy(s+4, argv0, l);
|
|
|
|
return s;
|
1996-03-20 22:57:41 +08:00
|
|
|
}
|
|
|
|
|
2014-09-01 01:11:08 +08:00
|
|
|
/* Argv0 aka argv[0] will be a full vms file specification, like
|
|
|
|
node$dka100:[utils.gnumake]make.exe;47, set up by the CRTL.
|
|
|
|
The vms progname should be ^^^^, the file name without
|
|
|
|
file type .exe and ;version.
|
|
|
|
Use sys$parse to get the name part of the file specification. That is
|
|
|
|
in the above example, pick up "make" and return a copy of that string.
|
|
|
|
If something goes wrong in sys$parse (unlikely, this is a VMS/CRTL supplied
|
|
|
|
file specification) or if there is an empty name part (not easy to produce,
|
|
|
|
but it is possible) just return "make".
|
|
|
|
Somes notes ...
|
|
|
|
NAM[L]$M_SYNCHK requests a syntax check, only.
|
|
|
|
NAM is for ODS2 names (shorter parts, output usually converted to UPPERCASE).
|
|
|
|
NAML is for ODS2/ODS5 names (longer parts, output unchanged).
|
|
|
|
NAM$M_NO_SHORT_UPCASE may not be available for older versions of VMS.
|
|
|
|
NAML is not available on older versions of VMS (NAML$C_BID not defined).
|
|
|
|
argv[0] on older versions of VMS (no extended parse style and no
|
|
|
|
CRTL feature DECC$ARGV_PARSE_STYLE) is always in lowercase. */
|
|
|
|
const char *
|
|
|
|
vms_progname(const char* argv0)
|
1996-03-20 22:57:41 +08:00
|
|
|
{
|
|
|
|
int status;
|
2014-09-01 01:11:08 +08:00
|
|
|
static struct FAB fab;
|
|
|
|
char *progname;
|
|
|
|
const char *fallback = "make";
|
1996-03-20 22:57:41 +08:00
|
|
|
|
2014-09-01 01:11:08 +08:00
|
|
|
#ifdef NAML$C_BID
|
|
|
|
static char esa[NAML$C_MAXRSS];
|
|
|
|
static struct NAML naml;
|
1998-07-31 04:54:47 +08:00
|
|
|
#else
|
2014-09-01 01:11:08 +08:00
|
|
|
static char esa[NAM$C_MAXRSS];
|
|
|
|
static struct NAM nam;
|
1998-07-31 04:54:47 +08:00
|
|
|
#endif
|
1996-03-20 22:57:41 +08:00
|
|
|
|
2014-09-01 01:11:08 +08:00
|
|
|
fab = cc$rms_fab;
|
|
|
|
fab.fab$l_fna = (char*)argv0;
|
|
|
|
fab.fab$b_fns = strlen(argv0);
|
|
|
|
|
|
|
|
#ifdef NAML$C_BID
|
|
|
|
fab.fab$l_naml = &naml;
|
|
|
|
naml = cc$rms_naml;
|
|
|
|
naml.naml$l_long_expand = esa;
|
|
|
|
naml.naml$l_long_expand_alloc = NAML$C_MAXRSS;
|
|
|
|
naml.naml$b_nop = NAML$M_SYNCHK;
|
|
|
|
naml.naml$l_input_flags = NAML$M_NO_SHORT_OUTPUT;
|
1996-03-20 22:57:41 +08:00
|
|
|
#else
|
2014-09-01 01:11:08 +08:00
|
|
|
fab.fab$l_nam = &nam;
|
|
|
|
nam = cc$rms_nam;
|
|
|
|
nam.nam$l_esa = esa;
|
|
|
|
nam.nam$b_ess = NAM$C_MAXRSS;
|
|
|
|
# ifdef NAM$M_NO_SHORT_UPCASE
|
|
|
|
nam.nam$b_nop = NAM$M_SYNCHK | NAM$M_NO_SHORT_UPCASE;
|
|
|
|
# else
|
|
|
|
nam.nam$b_nop = NAM$M_SYNCHK;
|
|
|
|
# endif
|
1996-03-20 22:57:41 +08:00
|
|
|
#endif
|
|
|
|
|
2014-09-01 01:11:08 +08:00
|
|
|
status = sys$parse(&fab);
|
1996-03-20 22:57:41 +08:00
|
|
|
if (!(status & 1))
|
2014-09-01 01:11:08 +08:00
|
|
|
return fallback;
|
|
|
|
|
|
|
|
#ifdef NAML$C_BID
|
|
|
|
if (naml.naml$l_long_name_size == 0)
|
|
|
|
return fallback;
|
|
|
|
progname = xmalloc(naml.naml$l_long_name_size + 1);
|
|
|
|
memcpy(progname, naml.naml$l_long_name, naml.naml$l_long_name_size);
|
|
|
|
progname[naml.naml$l_long_name_size] = '\0';
|
|
|
|
#else
|
|
|
|
if (nam.nam$b_name == 0)
|
|
|
|
return fallback;
|
|
|
|
progname = xmalloc(nam.nam$b_name + 1);
|
|
|
|
# ifdef NAM$M_NO_SHORT_UPCASE
|
|
|
|
memcpy(progname, nam.nam$l_name, nam.nam$b_name);
|
|
|
|
# else
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < nam.nam$b_name; i++)
|
|
|
|
progname[i] = tolower(nam.nam$l_name[i]);
|
|
|
|
}
|
|
|
|
# endif
|
|
|
|
progname[nam.nam$b_name] = '\0';
|
2009-11-16 10:31:23 +08:00
|
|
|
#endif
|
1996-03-20 22:57:41 +08:00
|
|
|
|
2014-09-01 01:11:08 +08:00
|
|
|
return progname;
|
1996-03-20 22:57:41 +08:00
|
|
|
}
|