Files
flatpak/common/flatpak-utils-base.c
Simon McVittie 7b1cd20696 Replace flatpak_close_fds_workaround() with g_fdwalk_set_cloexec()
flatpak_close_fds_workaround() wasn't technically async-signal-safe,
because the requirement for sysconf() to be async-signal-safe was
removed in POSIX.1-2008.

It could also leave high fds open in some cases: in practice
sysconf(_SC_OPEN_MAX) returns the soft resource limit, but if our
resource limit has been reduced by an ancestor process, we could
conceivably still have fds open and inherited above that number.

We can fix this by using g_fdwalk_set_cloexec() with GLib >= 2.79.2,
or the backport in libglnx with older GLib. This uses close_range()
if possible, falling back to rummaging in /proc with async-signal-safe
syscalls.

Signed-off-by: Simon McVittie <smcv@collabora.com>
2024-02-15 16:06:41 +00:00

115 lines
3.0 KiB
C

/* vi:set et sw=2 sts=2 cin cino=t0,f0,(0,{s,>2s,n-s,^-s,e-s:
* Copyright © 2019 Red Hat, Inc
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
*
* Authors:
* Alexander Larsson <alexl@redhat.com>
*/
#include "config.h"
#include "flatpak-utils-base-private.h"
#include <stdlib.h>
#include <string.h>
#include <gio/gio.h>
#include "libglnx.h"
char *
flatpak_get_timezone (void)
{
g_autofree gchar *symlink = NULL;
gchar *etc_timezone = NULL;
const gchar *tzdir;
const gchar *default_tzdir = "/usr/share/zoneinfo";
symlink = flatpak_resolve_link ("/etc/localtime", NULL);
if (symlink != NULL)
{
/* Resolve relative path */
g_autofree gchar *canonical = flatpak_canonicalize_filename (symlink);
char *canonical_suffix;
/* Strip the prefix and slashes if possible. */
tzdir = getenv ("TZDIR");
if (tzdir != NULL && g_str_has_prefix (canonical, tzdir))
{
canonical_suffix = canonical + strlen (tzdir);
while (*canonical_suffix == '/')
canonical_suffix++;
return g_strdup (canonical_suffix);
}
if (g_str_has_prefix (canonical, default_tzdir))
{
canonical_suffix = canonical + strlen (default_tzdir);
while (*canonical_suffix == '/')
canonical_suffix++;
return g_strdup (canonical_suffix);
}
}
if (g_file_get_contents ("/etc/timezone", &etc_timezone,
NULL, NULL))
{
g_strchomp (etc_timezone);
return etc_timezone;
}
/* Final fall-back is UTC */
return g_strdup ("UTC");
}
char *
flatpak_readlink (const char *path,
GError **error)
{
return glnx_readlinkat_malloc (-1, path, NULL, error);
}
char *
flatpak_resolve_link (const char *path,
GError **error)
{
g_autofree char *link = flatpak_readlink (path, error);
g_autofree char *dirname = NULL;
if (link == NULL)
return NULL;
if (g_path_is_absolute (link))
return g_steal_pointer (&link);
dirname = g_path_get_dirname (path);
return g_build_filename (dirname, link, NULL);
}
/*
* Syntactically canonicalize a filename, similar to
* g_canonicalize_filename() in newer GLib.
*
* This function does not do I/O.
*/
char *
flatpak_canonicalize_filename (const char *path)
{
g_autoptr(GFile) file = g_file_new_for_path (path);
return g_file_get_path (file);
}