
It makes the initializations more explicit. G_VALUE_INIT is available since glib 2.30, and NM now require glib >= 2.32
684 lines
17 KiB
C
684 lines
17 KiB
C
/* -*- Mode: C; tab-width: 4; indent-tabs-mode: t; c-basic-offset: 4 -*- */
|
|
/* NetworkManager -- Network link manager
|
|
*
|
|
* 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 2 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, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Copyright (C) 2008 - 2012 Red Hat, Inc.
|
|
*/
|
|
|
|
#include <syslog.h>
|
|
#include <stdio.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <sys/types.h>
|
|
#include <signal.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/wait.h>
|
|
#include <errno.h>
|
|
#include <arpa/inet.h>
|
|
|
|
#include <glib.h>
|
|
#include <dbus/dbus.h>
|
|
#include <dbus/dbus-glib-lowlevel.h>
|
|
#include <dbus/dbus-glib.h>
|
|
|
|
|
|
#include "nm-dispatcher-action.h"
|
|
#include "nm-dispatcher-utils.h"
|
|
#include "nm-glib-compat.h"
|
|
|
|
#define NMD_SCRIPT_DIR NMCONFDIR "/dispatcher.d"
|
|
|
|
static GMainLoop *loop = NULL;
|
|
static gboolean debug = FALSE;
|
|
|
|
typedef struct {
|
|
GObject parent;
|
|
|
|
/* Private data */
|
|
guint quit_id;
|
|
gboolean persist;
|
|
} Handler;
|
|
|
|
typedef struct {
|
|
GObjectClass parent;
|
|
} HandlerClass;
|
|
|
|
GType handler_get_type (void);
|
|
|
|
#define HANDLER_TYPE (handler_get_type ())
|
|
#define HANDLER(object) (G_TYPE_CHECK_INSTANCE_CAST ((object), HANDLER_TYPE, Handler))
|
|
#define HANDLER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), HANDLER_TYPE, HandlerClass))
|
|
|
|
G_DEFINE_TYPE(Handler, handler, G_TYPE_OBJECT)
|
|
|
|
static void
|
|
impl_dispatch (Handler *h,
|
|
const char *action,
|
|
GHashTable *connection_hash,
|
|
GHashTable *connection_props,
|
|
GHashTable *device_props,
|
|
GHashTable *device_ip4_props,
|
|
GHashTable *device_ip6_props,
|
|
GHashTable *device_dhcp4_props,
|
|
GHashTable *device_dhcp6_props,
|
|
const char *vpn_ip_iface,
|
|
GHashTable *vpn_ip4_props,
|
|
GHashTable *vpn_ip6_props,
|
|
DBusGMethodInvocation *context);
|
|
|
|
#include "nm-dispatcher-glue.h"
|
|
|
|
|
|
static void
|
|
handler_init (Handler *h)
|
|
{
|
|
}
|
|
|
|
static void
|
|
handler_class_init (HandlerClass *h_class)
|
|
{
|
|
}
|
|
|
|
typedef struct Request Request;
|
|
|
|
static void dispatch_one_script (Request *request);
|
|
|
|
typedef struct {
|
|
Request *request;
|
|
|
|
char *script;
|
|
GPid pid;
|
|
DispatchResult result;
|
|
char *error;
|
|
} ScriptInfo;
|
|
|
|
struct Request {
|
|
Handler *handler;
|
|
|
|
DBusGMethodInvocation *context;
|
|
char *action;
|
|
char *iface;
|
|
char **envp;
|
|
GPtrArray *scripts; /* list of ScriptInfo */
|
|
guint idx;
|
|
|
|
guint script_watch_id;
|
|
guint script_timeout_id;
|
|
};
|
|
|
|
static void
|
|
script_info_free (ScriptInfo *info)
|
|
{
|
|
g_free (info->script);
|
|
g_free (info->error);
|
|
g_free (info);
|
|
}
|
|
|
|
static void
|
|
request_free (Request *request)
|
|
{
|
|
g_free (request->action);
|
|
g_free (request->iface);
|
|
g_strfreev (request->envp);
|
|
if (request->scripts)
|
|
g_ptr_array_foreach (request->scripts, (GFunc) script_info_free, NULL);
|
|
g_ptr_array_free (request->scripts, TRUE);
|
|
}
|
|
|
|
static gboolean
|
|
quit_timeout_cb (gpointer user_data)
|
|
{
|
|
g_main_loop_quit (loop);
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
quit_timeout_reschedule (Handler *h)
|
|
{
|
|
if (h->quit_id)
|
|
g_source_remove (h->quit_id);
|
|
if (!h->persist)
|
|
h->quit_id = g_timeout_add_seconds (10, quit_timeout_cb, NULL);
|
|
}
|
|
|
|
static gboolean
|
|
next_script (gpointer user_data)
|
|
{
|
|
Request *request = user_data;
|
|
GPtrArray *results;
|
|
GValueArray *item;
|
|
guint i;
|
|
|
|
quit_timeout_reschedule (request->handler);
|
|
|
|
request->idx++;
|
|
if (request->idx < request->scripts->len) {
|
|
dispatch_one_script (request);
|
|
return FALSE;
|
|
}
|
|
|
|
/* All done */
|
|
results = g_ptr_array_sized_new (request->scripts->len);
|
|
for (i = 0; i < request->scripts->len; i++) {
|
|
ScriptInfo *script = g_ptr_array_index (request->scripts, i);
|
|
GValue elt = G_VALUE_INIT;
|
|
|
|
item = g_value_array_new (3);
|
|
|
|
/* Script path */
|
|
g_value_init (&elt, G_TYPE_STRING);
|
|
g_value_set_string (&elt, script->script);
|
|
g_value_array_append (item, &elt);
|
|
g_value_unset (&elt);
|
|
|
|
/* Result */
|
|
g_value_init (&elt, G_TYPE_UINT);
|
|
g_value_set_uint (&elt, script->result);
|
|
g_value_array_append (item, &elt);
|
|
g_value_unset (&elt);
|
|
|
|
/* Error */
|
|
g_value_init (&elt, G_TYPE_STRING);
|
|
g_value_set_string (&elt, script->error ? script->error : "");
|
|
g_value_array_append (item, &elt);
|
|
g_value_unset (&elt);
|
|
|
|
g_ptr_array_add (results, item);
|
|
}
|
|
|
|
dbus_g_method_return (request->context, results);
|
|
|
|
request_free (request);
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
script_watch_cb (GPid pid, gint status, gpointer user_data)
|
|
{
|
|
ScriptInfo *script = user_data;
|
|
guint err;
|
|
|
|
g_assert (pid == script->pid);
|
|
|
|
script->request->script_watch_id = 0;
|
|
g_source_remove (script->request->script_timeout_id);
|
|
script->request->script_timeout_id = 0;
|
|
|
|
if (WIFEXITED (status)) {
|
|
err = WEXITSTATUS (status);
|
|
if (err == 0)
|
|
script->result = DISPATCH_RESULT_SUCCESS;
|
|
else {
|
|
script->error = g_strdup_printf ("Script '%s' exited with error status %d.",
|
|
script->script, err);
|
|
}
|
|
} else if (WIFSTOPPED (status)) {
|
|
script->error = g_strdup_printf ("Script '%s' stopped unexpectedly with signal %d.",
|
|
script->script, WSTOPSIG (status));
|
|
} else if (WIFSIGNALED (status)) {
|
|
script->error = g_strdup_printf ("Script '%s' died with signal %d",
|
|
script->script, WTERMSIG (status));
|
|
} else {
|
|
script->error = g_strdup_printf ("Script '%s' died from an unknown cause",
|
|
script->script);
|
|
}
|
|
|
|
if (script->result != DISPATCH_RESULT_SUCCESS) {
|
|
script->result = DISPATCH_RESULT_FAILED;
|
|
g_warning ("%s", script->error);
|
|
}
|
|
|
|
g_spawn_close_pid (script->pid);
|
|
next_script (script->request);
|
|
}
|
|
|
|
static gboolean
|
|
script_timeout_cb (gpointer user_data)
|
|
{
|
|
ScriptInfo *script = user_data;
|
|
|
|
g_source_remove (script->request->script_watch_id);
|
|
script->request->script_watch_id = 0;
|
|
script->request->script_timeout_id = 0;
|
|
|
|
g_warning ("Script '%s' took too long; killing it.", script->script);
|
|
|
|
if (kill (script->pid, 0) == 0)
|
|
kill (script->pid, SIGKILL);
|
|
waitpid (script->pid, NULL, 0);
|
|
|
|
script->error = g_strdup_printf ("Script '%s' timed out.", script->script);
|
|
script->result = DISPATCH_RESULT_TIMEOUT;
|
|
|
|
g_spawn_close_pid (script->pid);
|
|
g_idle_add (next_script, script->request);
|
|
return FALSE;
|
|
}
|
|
|
|
static inline gboolean
|
|
check_permissions (struct stat *s, GError **error)
|
|
{
|
|
g_return_val_if_fail (s != NULL, FALSE);
|
|
g_return_val_if_fail (error != NULL, FALSE);
|
|
g_return_val_if_fail (*error == NULL, FALSE);
|
|
|
|
/* Only accept regular files */
|
|
if (!S_ISREG (s->st_mode)) {
|
|
g_set_error (error, 0, 0, "not a regular file.");
|
|
return FALSE;
|
|
}
|
|
|
|
/* Only accept files owned by root */
|
|
if (s->st_uid != 0) {
|
|
g_set_error (error, 0, 0, "not owned by root.");
|
|
return FALSE;
|
|
}
|
|
|
|
/* Only accept files not writable by group or other, and not SUID */
|
|
if (s->st_mode & (S_IWGRP | S_IWOTH | S_ISUID)) {
|
|
g_set_error (error, 0, 0, "writable by group or other, or set-UID.");
|
|
return FALSE;
|
|
}
|
|
|
|
/* Only accept files executable by the owner */
|
|
if (!(s->st_mode & S_IXUSR)) {
|
|
g_set_error (error, 0, 0, "not executable by owner.");
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
check_filename (const char *file_name)
|
|
{
|
|
char *bad_suffixes[] = { "~", ".rpmsave", ".rpmorig", ".rpmnew", NULL };
|
|
char *tmp;
|
|
guint i;
|
|
|
|
/* File must not be a backup file, package management file, or start with '.' */
|
|
|
|
if (file_name[0] == '.')
|
|
return FALSE;
|
|
for (i = 0; bad_suffixes[i]; i++) {
|
|
if (g_str_has_suffix (file_name, bad_suffixes[i]))
|
|
return FALSE;
|
|
}
|
|
tmp = g_strrstr (file_name, ".dpkg-");
|
|
if (tmp && (tmp == strrchr (file_name, '.')))
|
|
return FALSE;
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
child_setup (gpointer user_data G_GNUC_UNUSED)
|
|
{
|
|
/* We are in the child process at this point */
|
|
/* Give child a different process group to ensure signal separation. */
|
|
pid_t pid = getpid ();
|
|
setpgid (pid, pid);
|
|
}
|
|
|
|
static void
|
|
dispatch_one_script (Request *request)
|
|
{
|
|
GError *error = NULL;
|
|
gchar *argv[4];
|
|
ScriptInfo *script = g_ptr_array_index (request->scripts, request->idx);
|
|
|
|
argv[0] = script->script;
|
|
argv[1] = request->iface ? request->iface : "none";
|
|
argv[2] = request->action;
|
|
argv[3] = NULL;
|
|
|
|
if (debug)
|
|
g_message ("Script: %s %s %s", script->script, request->iface ? request->iface : "(none)", request->action);
|
|
|
|
if (g_spawn_async ("/", argv, request->envp, G_SPAWN_DO_NOT_REAP_CHILD, child_setup, request, &script->pid, &error)) {
|
|
request->script_watch_id = g_child_watch_add (script->pid, (GChildWatchFunc) script_watch_cb, script);
|
|
request->script_timeout_id = g_timeout_add_seconds (3, script_timeout_cb, script);
|
|
} else {
|
|
g_warning ("Failed to execute script '%s': (%d) %s",
|
|
script->script, error->code, error->message);
|
|
script->result = DISPATCH_RESULT_EXEC_FAILED;
|
|
script->error = g_strdup (error->message);
|
|
g_clear_error (&error);
|
|
|
|
/* Try the next script */
|
|
g_idle_add (next_script, request);
|
|
}
|
|
}
|
|
|
|
static GSList *
|
|
find_scripts (void)
|
|
{
|
|
GDir *dir;
|
|
const char *filename;
|
|
GSList *sorted = NULL;
|
|
GError *error = NULL;
|
|
|
|
if (!(dir = g_dir_open (NMD_SCRIPT_DIR, 0, &error))) {
|
|
g_warning ("Failed to open dispatcher directory '%s': (%d) %s",
|
|
NMD_SCRIPT_DIR, error->code, error->message);
|
|
g_error_free (error);
|
|
return NULL;
|
|
}
|
|
|
|
while ((filename = g_dir_read_name (dir))) {
|
|
char *path;
|
|
struct stat st;
|
|
int err;
|
|
|
|
if (!check_filename (filename))
|
|
continue;
|
|
|
|
path = g_build_filename (NMD_SCRIPT_DIR, filename, NULL);
|
|
|
|
err = stat (path, &st);
|
|
if (err)
|
|
g_warning ("Failed to stat '%s': %d", path, err);
|
|
else if (!check_permissions (&st, &error)) {
|
|
g_warning ("Cannot execute '%s': %s", path, error->message);
|
|
g_clear_error (&error);
|
|
} else {
|
|
/* success */
|
|
sorted = g_slist_insert_sorted (sorted, path, (GCompareFunc) g_strcmp0);
|
|
}
|
|
}
|
|
g_dir_close (dir);
|
|
|
|
return sorted;
|
|
}
|
|
|
|
static void
|
|
impl_dispatch (Handler *h,
|
|
const char *str_action,
|
|
GHashTable *connection_hash,
|
|
GHashTable *connection_props,
|
|
GHashTable *device_props,
|
|
GHashTable *device_ip4_props,
|
|
GHashTable *device_ip6_props,
|
|
GHashTable *device_dhcp4_props,
|
|
GHashTable *device_dhcp6_props,
|
|
const char *vpn_ip_iface,
|
|
GHashTable *vpn_ip4_props,
|
|
GHashTable *vpn_ip6_props,
|
|
DBusGMethodInvocation *context)
|
|
{
|
|
GSList *sorted_scripts = NULL;
|
|
GSList *iter;
|
|
Request *request;
|
|
char **p;
|
|
char *iface = NULL;
|
|
|
|
sorted_scripts = find_scripts ();
|
|
|
|
if (!sorted_scripts) {
|
|
dbus_g_method_return (context, g_ptr_array_new ());
|
|
return;
|
|
}
|
|
|
|
quit_timeout_reschedule (h);
|
|
|
|
request = g_malloc0 (sizeof (*request));
|
|
request->handler = h;
|
|
request->context = context;
|
|
request->action = g_strdup (str_action);
|
|
|
|
request->envp = nm_dispatcher_utils_construct_envp (str_action,
|
|
connection_hash,
|
|
connection_props,
|
|
device_props,
|
|
device_ip4_props,
|
|
device_ip6_props,
|
|
device_dhcp4_props,
|
|
device_dhcp6_props,
|
|
vpn_ip_iface,
|
|
vpn_ip4_props,
|
|
vpn_ip6_props,
|
|
&iface);
|
|
|
|
if (debug) {
|
|
g_message ("------------ Action ID %p '%s' Interface %s Environment ------------",
|
|
context, str_action, iface ? iface : "(none)");
|
|
for (p = request->envp; *p; p++)
|
|
g_message (" %s", *p);
|
|
g_message ("\n");
|
|
}
|
|
|
|
request->iface = g_strdup (iface);
|
|
|
|
request->scripts = g_ptr_array_sized_new (5);
|
|
for (iter = sorted_scripts; iter; iter = g_slist_next (iter)) {
|
|
ScriptInfo *s = g_malloc0 (sizeof (*s));
|
|
s->request = request;
|
|
s->script = iter->data;
|
|
g_ptr_array_add (request->scripts, s);
|
|
}
|
|
g_slist_free (sorted_scripts);
|
|
|
|
/* start dispatching scripts */
|
|
dispatch_one_script (request);
|
|
}
|
|
|
|
static void
|
|
destroy_cb (DBusGProxy *proxy, gpointer user_data)
|
|
{
|
|
g_warning ("Disconnected from the system bus, exiting.");
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
static DBusGConnection *
|
|
dbus_init (void)
|
|
{
|
|
GError *error = NULL;
|
|
DBusGConnection *bus;
|
|
DBusConnection *connection;
|
|
DBusGProxy *proxy;
|
|
int result;
|
|
|
|
dbus_connection_set_change_sigpipe (TRUE);
|
|
|
|
bus = dbus_g_bus_get (DBUS_BUS_SYSTEM, &error);
|
|
if (!bus) {
|
|
g_warning ("Could not get the system bus. Make sure "
|
|
"the message bus daemon is running! Message: %s",
|
|
error->message);
|
|
g_error_free (error);
|
|
return NULL;
|
|
}
|
|
|
|
/* Clean up nicely if we get kicked off the bus */
|
|
connection = dbus_g_connection_get_connection (bus);
|
|
dbus_connection_set_exit_on_disconnect (connection, FALSE);
|
|
|
|
proxy = dbus_g_proxy_new_for_name (bus,
|
|
"org.freedesktop.DBus",
|
|
"/org/freedesktop/DBus",
|
|
"org.freedesktop.DBus");
|
|
if (!proxy) {
|
|
g_warning ("Could not create the DBus proxy!");
|
|
goto error;
|
|
}
|
|
|
|
g_signal_connect (proxy, "destroy", G_CALLBACK (destroy_cb), NULL);
|
|
|
|
if (!dbus_g_proxy_call (proxy, "RequestName", &error,
|
|
G_TYPE_STRING, NM_DISPATCHER_DBUS_SERVICE,
|
|
G_TYPE_UINT, DBUS_NAME_FLAG_DO_NOT_QUEUE,
|
|
G_TYPE_INVALID,
|
|
G_TYPE_UINT, &result,
|
|
G_TYPE_INVALID)) {
|
|
g_warning ("Could not acquire the " NM_DISPATCHER_DBUS_SERVICE " service.\n"
|
|
" Message: '%s'", error->message);
|
|
g_error_free (error);
|
|
goto error;
|
|
}
|
|
|
|
if (result != DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER) {
|
|
g_warning ("Could not acquire the " NM_DISPATCHER_DBUS_SERVICE " service "
|
|
"as it is already taken. Result: %d",
|
|
result);
|
|
goto error;
|
|
}
|
|
|
|
return bus;
|
|
|
|
error:
|
|
if (proxy)
|
|
g_object_unref (proxy);
|
|
dbus_g_connection_unref (bus);
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
log_handler (const gchar *log_domain,
|
|
GLogLevelFlags log_level,
|
|
const gchar *message,
|
|
gpointer ignored)
|
|
{
|
|
int syslog_priority;
|
|
|
|
switch (log_level) {
|
|
case G_LOG_LEVEL_ERROR:
|
|
syslog_priority = LOG_CRIT;
|
|
break;
|
|
case G_LOG_LEVEL_CRITICAL:
|
|
syslog_priority = LOG_ERR;
|
|
break;
|
|
case G_LOG_LEVEL_WARNING:
|
|
syslog_priority = LOG_WARNING;
|
|
break;
|
|
case G_LOG_LEVEL_MESSAGE:
|
|
syslog_priority = LOG_NOTICE;
|
|
break;
|
|
case G_LOG_LEVEL_DEBUG:
|
|
syslog_priority = LOG_DEBUG;
|
|
break;
|
|
case G_LOG_LEVEL_INFO:
|
|
default:
|
|
syslog_priority = LOG_INFO;
|
|
break;
|
|
}
|
|
|
|
syslog (syslog_priority, "%s", message);
|
|
}
|
|
|
|
|
|
static void
|
|
logging_setup (void)
|
|
{
|
|
openlog (G_LOG_DOMAIN, LOG_CONS, LOG_DAEMON);
|
|
g_log_set_handler (G_LOG_DOMAIN,
|
|
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL | G_LOG_FLAG_RECURSION,
|
|
log_handler,
|
|
NULL);
|
|
}
|
|
|
|
static void
|
|
logging_shutdown (void)
|
|
{
|
|
closelog ();
|
|
}
|
|
|
|
static void
|
|
signal_handler (int signo)
|
|
{
|
|
if (signo == SIGINT || signo == SIGTERM) {
|
|
g_message ("Caught signal %d, shutting down...", signo);
|
|
g_main_loop_quit (loop);
|
|
}
|
|
}
|
|
|
|
static void
|
|
setup_signals (void)
|
|
{
|
|
struct sigaction action;
|
|
sigset_t mask;
|
|
|
|
sigemptyset (&mask);
|
|
action.sa_handler = signal_handler;
|
|
action.sa_mask = mask;
|
|
action.sa_flags = 0;
|
|
sigaction (SIGTERM, &action, NULL);
|
|
sigaction (SIGINT, &action, NULL);
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
GOptionContext *opt_ctx;
|
|
GError *error = NULL;
|
|
gboolean persist = FALSE;
|
|
DBusGConnection *bus;
|
|
Handler *handler;
|
|
|
|
GOptionEntry entries[] = {
|
|
{ "debug", 0, 0, G_OPTION_ARG_NONE, &debug, "Output to console rather than syslog", NULL },
|
|
{ "persist", 0, 0, G_OPTION_ARG_NONE, &persist, "Don't quit after a short timeout", NULL },
|
|
{ NULL }
|
|
};
|
|
|
|
opt_ctx = g_option_context_new (NULL);
|
|
g_option_context_set_summary (opt_ctx, "Executes scripts upon actions by NetworkManager.");
|
|
g_option_context_add_main_entries (opt_ctx, entries, NULL);
|
|
|
|
if (!g_option_context_parse (opt_ctx, &argc, &argv, &error)) {
|
|
g_warning ("%s\n", error->message);
|
|
g_error_free (error);
|
|
return 1;
|
|
}
|
|
|
|
g_option_context_free (opt_ctx);
|
|
|
|
g_type_init ();
|
|
setup_signals ();
|
|
|
|
if (!debug)
|
|
logging_setup ();
|
|
|
|
loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
bus = dbus_init ();
|
|
if (!bus)
|
|
return 1;
|
|
|
|
handler = g_object_new (HANDLER_TYPE, NULL);
|
|
if (!handler)
|
|
return 1;
|
|
handler->persist = persist;
|
|
|
|
dbus_g_object_type_install_info (HANDLER_TYPE, &dbus_glib_nm_dispatcher_object_info);
|
|
dbus_g_connection_register_g_object (bus,
|
|
NM_DISPATCHER_DBUS_PATH,
|
|
G_OBJECT (handler));
|
|
|
|
if (!persist)
|
|
handler->quit_id = g_timeout_add_seconds (10, quit_timeout_cb, NULL);
|
|
|
|
g_main_loop_run (loop);
|
|
|
|
g_object_unref (handler);
|
|
dbus_g_connection_unref (bus);
|
|
|
|
if (!debug)
|
|
logging_shutdown ();
|
|
|
|
return 0;
|
|
}
|
|
|