freebsd-dev/contrib/groff/libgroff/tmpfile.cc
2000-12-05 18:49:44 +00:00

188 lines
5.1 KiB
C++

// -*- C++ -*-
/* Copyright (C) 1989, 1990, 1991, 1992 Free Software Foundation, Inc.
Written by James Clark (jjc@jclark.com)
This file is part of groff.
groff 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 Foundation; either version 2, or (at your option) any later
version.
groff 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 groff; see the file COPYING. If not, write to the Free Software
Foundation, 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <stdlib.h>
#include "posix.h"
#include "lib.h"
#include "errarg.h"
#include "error.h"
#include "nonposix.h"
extern "C" {
// Solaris 2.5.1 has these functions,
// but its stdlib.h fails to declare them.
char *mktemp(char *);
int mkstemp(char *);
}
// If this is set, create temporary files there
#define GROFF_TMPDIR_ENVVAR "GROFF_TMPDIR"
// otherwise if this is set, create temporary files there
#define TMPDIR_ENVVAR "TMPDIR"
// otherwise if P_tmpdir is defined, create temporary files there
#ifdef P_tmpdir
# define DEFAULT_TMPDIR P_tmpdir
#else
// otherwise create temporary files here.
# define DEFAULT_TMPDIR "/tmp"
#endif
// Use this as the prefix for temporary filenames.
#define TMPFILE_PREFIX "groff"
/*
* Generate a temporary name template with a postfix
* immediately after the TMPFILE_PREFIX.
* It uses the groff preferences for a temporary directory.
* Note that no file name is either created or opened,
* only the *template* is returned.
*/
char *xtmptemplate(char *postfix)
{
const char *dir = getenv(GROFF_TMPDIR_ENVVAR);
int postlen = 0;
if (postfix)
postlen = strlen(postfix);
if (!dir) {
dir = getenv(TMPDIR_ENVVAR);
if (!dir)
dir = DEFAULT_TMPDIR;
}
size_t dir_len = strlen(dir);
const char *dir_end = dir + dir_len - 1;
int needs_slash = strchr(DIR_SEPS, *dir_end) == NULL;
char *templ = new char[strlen(dir) + needs_slash
+ sizeof(TMPFILE_PREFIX) - 1 + 6 + 1 + postlen];
strcpy(templ, dir);
if (needs_slash)
strcat(templ, "/");
strcat(templ, TMPFILE_PREFIX);
if (postlen > 0)
strcat(templ, postfix);
strcat(templ, "XXXXXX");
return( templ );
}
// The trick with unlinking the temporary file while it is still in
// use is not portable, it will fail on MS-DOS and most MS-Windows
// filesystems. So it cannot be used on non-Posix systems.
// Instead, we maintain a list of files to be deleted on exit, and
// register an atexit function that will remove them all in one go.
// This should be portable to all platforms.
static struct xtmpfile_list {
struct xtmpfile_list *next;
char *fname;
} *xtmpfiles_to_delete;
static void remove_tmp_files(void)
{
struct xtmpfile_list *p = xtmpfiles_to_delete;
while (p)
{
if (unlink(p->fname) < 0)
error("cannot unlink `%1': %2", p->fname, strerror(errno));
a_delete p->fname;
struct xtmpfile_list *old = p;
p = p->next;
free(old);
}
}
static void add_tmp_file(const char *name)
{
if (xtmpfiles_to_delete == NULL)
atexit(remove_tmp_files);
char *fname = new char[FILENAME_MAX];
struct xtmpfile_list *p
= (struct xtmpfile_list *)malloc(sizeof(struct xtmpfile_list));
if (p == NULL)
{
error("cannot unlink `%1': %2", name, strerror(errno));
return;
}
p->next = xtmpfiles_to_delete;
p->fname = strcpy(fname, name);
xtmpfiles_to_delete = p;
}
// Open a temporary file and with fatal error on failure.
#ifndef _MSC_VER
FILE *xtmpfile(char **namep, char *postfix, int do_unlink)
{
char *templ = xtmptemplate(postfix);
#ifdef HAVE_MKSTEMP
errno = 0;
int fd = mkstemp(templ);
if (fd < 0)
fatal("cannot create temporary file: %1", strerror(errno));
errno = 0;
FILE *fp = fdopen(fd, FOPEN_RWB); // many callers of xtmpfile use binary I/O
if (!fp)
fatal("fdopen: %1", strerror(errno));
#else /* not HAVE_MKSTEMP */
if (!mktemp(templ) || !templ[0])
fatal("cannot create file name for temporary file");
errno = 0;
FILE *fp = fopen(templ, FOPEN_RWB);
if (!fp)
fatal("cannot open `%1': %2", templ, strerror(errno));
#endif /* not HAVE_MKSTEMP */
if (do_unlink)
add_tmp_file(templ);
if ((namep != 0) && ((*namep) != 0)) {
*namep = templ;
} else {
a_delete templ;
}
return fp;
}
#else
// FIXME: does MSVC have mktemp or mkstemp? If so, it should now
// use the version above, as it no longer removes an open file.
// The version below will NOT work with grohtml, since grohtml
// wants to know the name of the file opened by xtmpfile!!
// If you're not running Unix, the following will do:
FILE *xtmpfile(char **namep, char *postfix, int do_unlink)
{
FILE *fp = tmpfile();
if (!fp)
fatal("couldn't create temporary file");
return fp;
}
#endif /* _MSC_VER */