mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2024-11-10 11:26:16 +01:00
Don't fork a new process just to fix the permissions of the created temp
2005-12-03 Matthias Clasen <mclasen@redhat.com> * glib/gfileutils.c: Don't fork a new process just to fix the permissions of the created temp file. (#321318, Alexis S. L. Carvalho)
This commit is contained in:
parent
d1379dbdf8
commit
1fe373b279
@ -1,3 +1,9 @@
|
|||||||
|
2005-12-03 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
|
||||||
|
* glib/gfileutils.c: Don't fork a new process just to
|
||||||
|
fix the permissions of the created temp file. (#321318,
|
||||||
|
Alexis S. L. Carvalho)
|
||||||
|
|
||||||
2005-12-02 Matthias Clasen <mclasen@redhat.com>
|
2005-12-02 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
|
||||||
* README.in: Add a note about Solaris threads.
|
* README.in: Add a note about Solaris threads.
|
||||||
|
@ -1,3 +1,9 @@
|
|||||||
|
2005-12-03 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
|
||||||
|
* glib/gfileutils.c: Don't fork a new process just to
|
||||||
|
fix the permissions of the created temp file. (#321318,
|
||||||
|
Alexis S. L. Carvalho)
|
||||||
|
|
||||||
2005-12-02 Matthias Clasen <mclasen@redhat.com>
|
2005-12-02 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
|
||||||
* README.in: Add a note about Solaris threads.
|
* README.in: Add a note about Solaris threads.
|
||||||
|
@ -1,3 +1,9 @@
|
|||||||
|
2005-12-03 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
|
||||||
|
* glib/gfileutils.c: Don't fork a new process just to
|
||||||
|
fix the permissions of the created temp file. (#321318,
|
||||||
|
Alexis S. L. Carvalho)
|
||||||
|
|
||||||
2005-12-02 Matthias Clasen <mclasen@redhat.com>
|
2005-12-02 Matthias Clasen <mclasen@redhat.com>
|
||||||
|
|
||||||
* README.in: Add a note about Solaris threads.
|
* README.in: Add a note about Solaris threads.
|
||||||
|
@ -33,9 +33,6 @@
|
|||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <sys/stat.h>
|
#include <sys/stat.h>
|
||||||
#ifndef G_OS_WIN32
|
|
||||||
#include <sys/wait.h>
|
|
||||||
#endif
|
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
@ -57,6 +54,9 @@
|
|||||||
|
|
||||||
#include "galias.h"
|
#include "galias.h"
|
||||||
|
|
||||||
|
static gint create_temp_file (gchar *tmpl,
|
||||||
|
int permissions);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* g_mkdir_with_parents:
|
* g_mkdir_with_parents:
|
||||||
* @pathname: a pathname in the GLib file name encoding
|
* @pathname: a pathname in the GLib file name encoding
|
||||||
@ -920,114 +920,6 @@ rename_file (const char *old_name,
|
|||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
|
||||||
set_umask_permissions (int fd,
|
|
||||||
GError **err)
|
|
||||||
{
|
|
||||||
#ifdef G_OS_WIN32
|
|
||||||
|
|
||||||
return TRUE;
|
|
||||||
|
|
||||||
#else
|
|
||||||
/* All of this function is just to work around the fact that
|
|
||||||
* there is no way to get the umask without changing it.
|
|
||||||
*
|
|
||||||
* We can't just change-and-reset the umask because that would
|
|
||||||
* lead to a race condition if another thread tried to change
|
|
||||||
* the umask in between the getting and the setting of the umask.
|
|
||||||
* So we have to do the whole thing in a child process.
|
|
||||||
*/
|
|
||||||
|
|
||||||
int save_errno;
|
|
||||||
pid_t pid;
|
|
||||||
|
|
||||||
pid = fork ();
|
|
||||||
|
|
||||||
if (pid == -1)
|
|
||||||
{
|
|
||||||
save_errno = errno;
|
|
||||||
g_set_error (err,
|
|
||||||
G_FILE_ERROR,
|
|
||||||
g_file_error_from_errno (save_errno),
|
|
||||||
_("Could not change file mode: fork() failed: %s"),
|
|
||||||
g_strerror (save_errno));
|
|
||||||
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
else if (pid == 0)
|
|
||||||
{
|
|
||||||
/* child */
|
|
||||||
mode_t mask = umask (0666);
|
|
||||||
|
|
||||||
errno = 0;
|
|
||||||
if (fchmod (fd, 0666 & ~mask) == -1)
|
|
||||||
_exit (errno);
|
|
||||||
else
|
|
||||||
_exit (0);
|
|
||||||
|
|
||||||
return TRUE; /* To quiet gcc */
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* parent */
|
|
||||||
int status;
|
|
||||||
|
|
||||||
errno = 0;
|
|
||||||
if (waitpid (pid, &status, 0) == -1)
|
|
||||||
{
|
|
||||||
save_errno = errno;
|
|
||||||
|
|
||||||
g_set_error (err,
|
|
||||||
G_FILE_ERROR,
|
|
||||||
g_file_error_from_errno (save_errno),
|
|
||||||
_("Could not change file mode: waitpid() failed: %s"),
|
|
||||||
g_strerror (save_errno));
|
|
||||||
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (WIFEXITED (status))
|
|
||||||
{
|
|
||||||
save_errno = WEXITSTATUS (status);
|
|
||||||
|
|
||||||
if (save_errno == 0)
|
|
||||||
{
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
g_set_error (err,
|
|
||||||
G_FILE_ERROR,
|
|
||||||
g_file_error_from_errno (save_errno),
|
|
||||||
_("Could not change file mode: chmod() failed: %s"),
|
|
||||||
g_strerror (save_errno));
|
|
||||||
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (WIFSIGNALED (status))
|
|
||||||
{
|
|
||||||
g_set_error (err,
|
|
||||||
G_FILE_ERROR,
|
|
||||||
G_FILE_ERROR_FAILED,
|
|
||||||
_("Could not change file mode: Child terminated by signal: %s"),
|
|
||||||
g_strsignal (WTERMSIG (status)));
|
|
||||||
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/* This shouldn't happen */
|
|
||||||
g_set_error (err,
|
|
||||||
G_FILE_ERROR,
|
|
||||||
G_FILE_ERROR_FAILED,
|
|
||||||
_("Could not change file mode: Child terminated abnormally"));
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
static gchar *
|
static gchar *
|
||||||
write_to_temp_file (const gchar *contents,
|
write_to_temp_file (const gchar *contents,
|
||||||
gssize length,
|
gssize length,
|
||||||
@ -1046,7 +938,7 @@ write_to_temp_file (const gchar *contents,
|
|||||||
tmp_name = g_strdup_printf ("%s.XXXXXX", template);
|
tmp_name = g_strdup_printf ("%s.XXXXXX", template);
|
||||||
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
fd = g_mkstemp (tmp_name);
|
fd = create_temp_file (tmp_name, 0666);
|
||||||
display_name = g_filename_display_name (tmp_name);
|
display_name = g_filename_display_name (tmp_name);
|
||||||
|
|
||||||
if (fd == -1)
|
if (fd == -1)
|
||||||
@ -1061,14 +953,6 @@ write_to_temp_file (const gchar *contents,
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!set_umask_permissions (fd, err))
|
|
||||||
{
|
|
||||||
close (fd);
|
|
||||||
g_unlink (tmp_name);
|
|
||||||
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
file = fdopen (fd, "wb");
|
file = fdopen (fd, "wb");
|
||||||
if (!file)
|
if (!file)
|
||||||
@ -1267,35 +1151,13 @@ g_file_set_contents (const gchar *filename,
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* mkstemp() implementation is from the GNU C library.
|
* create_temp_file based on the mkstemp implementation from the GNU C library.
|
||||||
* Copyright (C) 1991,92,93,94,95,96,97,98,99 Free Software Foundation, Inc.
|
* Copyright (C) 1991,92,93,94,95,96,97,98,99 Free Software Foundation, Inc.
|
||||||
*/
|
*/
|
||||||
/**
|
static gint
|
||||||
* g_mkstemp:
|
create_temp_file (gchar *tmpl,
|
||||||
* @tmpl: template filename
|
int permissions)
|
||||||
*
|
|
||||||
* Opens a temporary file. See the mkstemp() documentation
|
|
||||||
* on most UNIX-like systems. This is a portability wrapper, which simply calls
|
|
||||||
* mkstemp() on systems that have it, and implements
|
|
||||||
* it in GLib otherwise.
|
|
||||||
*
|
|
||||||
* The parameter is a string that should match the rules for
|
|
||||||
* mkstemp(), i.e. end in "XXXXXX". The X string will
|
|
||||||
* be modified to form the name of a file that didn't exist.
|
|
||||||
* The string should be in the GLib file name encoding. Most importantly,
|
|
||||||
* on Windows it should be in UTF-8.
|
|
||||||
*
|
|
||||||
* Return value: A file handle (as from open()) to the file
|
|
||||||
* opened for reading and writing. The file is opened in binary mode
|
|
||||||
* on platforms where there is a difference. The file handle should be
|
|
||||||
* closed with close(). In case of errors, -1 is returned.
|
|
||||||
*/
|
|
||||||
gint
|
|
||||||
g_mkstemp (gchar *tmpl)
|
|
||||||
{
|
{
|
||||||
#ifdef HAVE_MKSTEMP
|
|
||||||
return mkstemp (tmpl);
|
|
||||||
#else
|
|
||||||
int len;
|
int len;
|
||||||
char *XXXXXX;
|
char *XXXXXX;
|
||||||
int count, fd;
|
int count, fd;
|
||||||
@ -1338,7 +1200,7 @@ g_mkstemp (gchar *tmpl)
|
|||||||
XXXXXX[5] = letters[v % NLETTERS];
|
XXXXXX[5] = letters[v % NLETTERS];
|
||||||
|
|
||||||
/* tmpl is in UTF-8 on Windows, thus use g_open() */
|
/* tmpl is in UTF-8 on Windows, thus use g_open() */
|
||||||
fd = g_open (tmpl, O_RDWR | O_CREAT | O_EXCL | O_BINARY, 0600);
|
fd = g_open (tmpl, O_RDWR | O_CREAT | O_EXCL | O_BINARY, permissions);
|
||||||
|
|
||||||
if (fd >= 0)
|
if (fd >= 0)
|
||||||
return fd;
|
return fd;
|
||||||
@ -1352,6 +1214,35 @@ g_mkstemp (gchar *tmpl)
|
|||||||
/* We got out of the loop because we ran out of combinations to try. */
|
/* We got out of the loop because we ran out of combinations to try. */
|
||||||
errno = EEXIST;
|
errno = EEXIST;
|
||||||
return -1;
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* g_mkstemp:
|
||||||
|
* @tmpl: template filename
|
||||||
|
*
|
||||||
|
* Opens a temporary file. See the mkstemp() documentation
|
||||||
|
* on most UNIX-like systems. This is a portability wrapper, which simply calls
|
||||||
|
* mkstemp() on systems that have it, and implements
|
||||||
|
* it in GLib otherwise.
|
||||||
|
*
|
||||||
|
* The parameter is a string that should match the rules for
|
||||||
|
* mkstemp(), i.e. end in "XXXXXX". The X string will
|
||||||
|
* be modified to form the name of a file that didn't exist.
|
||||||
|
* The string should be in the GLib file name encoding. Most importantly,
|
||||||
|
* on Windows it should be in UTF-8.
|
||||||
|
*
|
||||||
|
* Return value: A file handle (as from open()) to the file
|
||||||
|
* opened for reading and writing. The file is opened in binary mode
|
||||||
|
* on platforms where there is a difference. The file handle should be
|
||||||
|
* closed with close(). In case of errors, -1 is returned.
|
||||||
|
*/
|
||||||
|
gint
|
||||||
|
g_mkstemp (gchar *tmpl)
|
||||||
|
{
|
||||||
|
#ifdef HAVE_MKSTEMP
|
||||||
|
return mkstemp (tmpl);
|
||||||
|
#else
|
||||||
|
return create_temp_file (tmpl, 0600);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user