gem-graph-client/libide/core/ide-global.c

359 lines
8.3 KiB
C

/* ide-global.c
*
* Copyright 2014-2019 Christian Hergert <chergert@redhat.com>
*
* This program 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 3 of the License, or
* (at your option) any later version.
*
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
*
* SPDX-License-Identifier: GPL-3.0-or-later
*/
#define G_LOG_DOMAIN "ide-global"
#include "config.h"
#include <gio/gio.h>
#include <glib/gi18n.h>
#include <string.h>
#include <sys/types.h>
#include <sys/user.h>
#include <sys/utsname.h>
#include <unistd.h>
#include "../../gconstructor.h"
#include "ide-debug.h"
#include "ide-global.h"
#include "ide-macros.h"
#include "ide-private.h"
static GThread *main_thread;
static const gchar *application_id = "org.gnome.Builder";
static IdeProcessKind kind = IDE_PROCESS_KIND_HOST;
#if defined (G_HAS_CONSTRUCTORS)
# ifdef G_DEFINE_CONSTRUCTOR_NEEDS_PRAGMA
# pragma G_DEFINE_CONSTRUCTOR_PRAGMA_ARGS(ide_init_ctor)
# endif
G_DEFINE_CONSTRUCTOR(ide_init_ctor)
#else
# error Your platform/compiler is missing constructor support
#endif
static void
ide_init_ctor (void)
{
main_thread = g_thread_self ();
if (g_file_test ("/.flatpak-info", G_FILE_TEST_EXISTS))
kind = IDE_PROCESS_KIND_FLATPAK;
}
/**
* ide_get_main_thread
*
* Gets #GThread of the main thread.
*
* Generally this is used by macros to determine what thread they code is
* currently running within.
*
* Returns: (transfer none): a #GThread
*
* Since: 3.32
*/
GThread *
ide_get_main_thread (void)
{
return main_thread;
}
/**
* ide_get_process_kind:
*
* Gets the kind of process we're running as.
*
* Returns: an #IdeProcessKind
*
* Since: 3.32
*/
IdeProcessKind
ide_get_process_kind (void)
{
return kind;
}
const gchar *
ide_get_application_id (void)
{
return application_id;
}
/**
* ide_set_application_id:
* @app_id: the application id
*
* Sets the application id that will be used.
*
* This must be set at application startup before any GApplication
* has connected to the D-Bus.
*
* The default is "org.gnome.Builder".
*
* Since: 3.32
*/
void
ide_set_application_id (const gchar *app_id)
{
g_return_if_fail (app_id != NULL);
application_id = g_intern_string (app_id);
}
const gchar *
ide_get_program_name (void)
{
return "gnome-builder";
}
gchar *
ide_create_host_triplet (const gchar *arch,
const gchar *kernel,
const gchar *system)
{
if (arch == NULL || kernel == NULL)
return g_strdup (ide_get_system_type ());
else if (system == NULL)
return g_strdup_printf ("%s-%s", arch, kernel);
else
return g_strdup_printf ("%s-%s-%s", arch, kernel, system);
}
const gchar *
ide_get_system_type (void)
{
static gchar *system_type;
g_autofree gchar *os_lower = NULL;
const gchar *machine = NULL;
struct utsname u;
if (system_type != NULL)
return system_type;
if (uname (&u) < 0)
return g_strdup ("unknown");
os_lower = g_utf8_strdown (u.sysname, -1);
/* config.sub doesn't accept amd64-OS */
machine = strcmp (u.machine, "amd64") ? u.machine : "x86_64";
/*
* TODO: Clearly we want to discover "gnu", but that should be just fine
* for a default until we try to actually run on something non-gnu.
* Which seems unlikely at the moment. If you run FreeBSD, you can
* probably fix this for me :-) And while you're at it, make the
* uname() call more portable.
*/
#ifdef __GLIBC__
system_type = g_strdup_printf ("%s-%s-%s", machine, os_lower, "gnu");
#else
system_type = g_strdup_printf ("%s-%s", machine, os_lower);
#endif
return system_type;
}
gchar *
ide_get_system_arch (void)
{
static GHashTable *remap;
const char *machine;
struct utsname u;
if (uname (&u) < 0)
return g_strdup ("unknown");
if (g_once_init_enter (&remap))
{
GHashTable *mapping;
mapping = g_hash_table_new (g_str_hash, g_str_equal);
g_hash_table_insert (mapping, (gchar *)"amd64", (gchar *)"x86_64");
g_hash_table_insert (mapping, (gchar *)"armv7l", (gchar *)"aarch64");
g_hash_table_insert (mapping, (gchar *)"i686", (gchar *)"i386");
g_once_init_leave (&remap, mapping);
}
if (g_hash_table_lookup_extended (remap, u.machine, NULL, (gpointer *)&machine))
return g_strdup (machine);
else
return g_strdup (u.machine);
}
gsize
ide_get_system_page_size (void)
{
return sysconf (_SC_PAGE_SIZE);
}
static gchar *
get_base_path (const gchar *name)
{
g_autoptr(GKeyFile) keyfile = g_key_file_new ();
if (g_key_file_load_from_file (keyfile, "/.flatpak-info", 0, NULL))
return g_key_file_get_string (keyfile, "Instance", name, NULL);
return NULL;
}
/**
* ide_get_relocatable_path:
* @path: a relocatable path
*
* Gets the path to a resource that may be relocatable at runtime.
*
* Returns: (transfer full): a new string containing the path
*
* Since: 3.32
*/
gchar *
ide_get_relocatable_path (const gchar *path)
{
static gchar *base_path;
if G_UNLIKELY (base_path == NULL)
base_path = get_base_path ("app-path");
return g_build_filename (base_path, path, NULL);
}
const gchar *
ide_gettext (const gchar *message)
{
if (message != NULL)
return g_dgettext (GETTEXT_PACKAGE, message);
return NULL;
}
static IdeTraceVTable trace_vtable;
void
_ide_trace_init (IdeTraceVTable *vtable)
{
trace_vtable = *vtable;
if (trace_vtable.load)
trace_vtable.load ();
}
void
_ide_trace_shutdown (void)
{
if (trace_vtable.unload)
trace_vtable.unload ();
memset (&trace_vtable, 0, sizeof trace_vtable);
}
#ifdef IDE_ENABLE_TRACE
void
ide_trace_function (const gchar *strfunc,
gint64 begin_time_usec,
gint64 end_time_usec)
{
/* In case our clock is not reliable */
if (end_time_usec < begin_time_usec)
end_time_usec = begin_time_usec;
if (trace_vtable.function)
trace_vtable.function (strfunc, begin_time_usec, end_time_usec);
}
#endif
void
_ide_trace_log (GLogLevelFlags log_level,
const gchar *domain,
const gchar *message)
{
if (trace_vtable.log)
trace_vtable.log (log_level, domain, message);
}
static gchar **
get_environ_from_stdout (GSubprocess *subprocess)
{
g_autofree gchar *stdout_buf = NULL;
if (g_subprocess_communicate_utf8 (subprocess, NULL, NULL, &stdout_buf, NULL, NULL))
{
g_auto(GStrv) lines = g_strsplit (stdout_buf, "\n", 0);
g_autoptr(GPtrArray) env = g_ptr_array_new_with_free_func (g_free);
for (guint i = 0; lines[i]; i++)
{
const char *line = lines[i];
if (!g_ascii_isalpha (*line) && *line != '_')
continue;
for (const char *iter = line; *iter; iter = g_utf8_next_char (iter))
{
if (*iter == '=')
{
g_ptr_array_add (env, g_strdup (line));
break;
}
if (!g_ascii_isalnum (*iter) && *iter != '_')
break;
}
}
if (env->len > 0)
{
g_ptr_array_add (env, NULL);
return (gchar **)g_ptr_array_free (g_steal_pointer (&env), FALSE);
}
}
return NULL;
}
const gchar * const *
_ide_host_environ (void)
{
static gchar **host_environ;
if (host_environ == NULL)
{
if (ide_is_flatpak ())
{
g_autoptr(GSubprocessLauncher) launcher = NULL;
g_autoptr(GSubprocess) subprocess = NULL;
g_autoptr(GError) error = NULL;
launcher = g_subprocess_launcher_new (G_SUBPROCESS_FLAGS_STDOUT_PIPE);
subprocess = g_subprocess_launcher_spawn (launcher, &error,
"flatpak-spawn", "--host", "printenv", NULL);
if (subprocess != NULL)
host_environ = get_environ_from_stdout (subprocess);
}
if (host_environ == NULL)
host_environ = g_get_environ ();
}
return (const char * const *)host_environ;
}