os/ossrv/ofdbus/dbus-glib/dbus/dbus-glib-tool.c
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
/* -*- mode: C; c-file-style: "gnu" -*- */
sl@0
     2
/* dbus-glib-tool.c Tool used by apps using glib bindings
sl@0
     3
 *
sl@0
     4
 * Copyright (C) 2003, 2004 Red Hat, Inc.
sl@0
     5
 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
sl@0
     6
 * Licensed under the Academic Free License version 2.1
sl@0
     7
 *
sl@0
     8
 * This program is free software; you can redistribute it and/or modify
sl@0
     9
 * it under the terms of the GNU General Public License as published by
sl@0
    10
 * the Free Software Foundation; either version 2 of the License, or
sl@0
    11
 * (at your option) any later version.
sl@0
    12
 *
sl@0
    13
 * This program is distributed in the hope that it will be useful,
sl@0
    14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
sl@0
    15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
sl@0
    16
 * GNU General Public License for more details.
sl@0
    17
 *
sl@0
    18
 * You should have received a copy of the GNU General Public License
sl@0
    19
 * along with this program; if not, write to the Free Software
sl@0
    20
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
sl@0
    21
 *
sl@0
    22
 */
sl@0
    23
 
sl@0
    24
 #ifdef __SYMBIAN32__
sl@0
    25
 #define VERSION "1.0.2"
sl@0
    26
 #endif
sl@0
    27
sl@0
    28
#ifndef __SYMBIAN32__
sl@0
    29
#include <config.h>
sl@0
    30
#else
sl@0
    31
#include "config.h"
sl@0
    32
#endif //__SYMBIAN32__
sl@0
    33
#include "dbus-gidl.h"
sl@0
    34
#include "dbus-gparser.h"
sl@0
    35
#include "dbus-gutils.h"
sl@0
    36
#include "dbus-glib-tool.h"
sl@0
    37
#include "dbus-binding-tool-glib.h"
sl@0
    38
#include <locale.h>
sl@0
    39
#ifndef __SYMBIAN32__
sl@0
    40
#include <libintl.h>
sl@0
    41
#define _(x) dgettext (GETTEXT_PACKAGE, x)
sl@0
    42
#define N_(x) x
sl@0
    43
#else
sl@0
    44
#define _(x)  x
sl@0
    45
#define N_(x) x
sl@0
    46
#endif
sl@0
    47
#include <stdio.h>
sl@0
    48
#include <stdlib.h>
sl@0
    49
#include <errno.h>
sl@0
    50
#include <sys/stat.h>
sl@0
    51
#include <string.h>
sl@0
    52
#include <time.h>
sl@0
    53
sl@0
    54
#ifdef DBUS_BUILD_TESTS
sl@0
    55
static void run_all_tests (const char *test_data_dir);
sl@0
    56
#endif
sl@0
    57
sl@0
    58
typedef enum {
sl@0
    59
  DBUS_BINDING_OUTPUT_NONE,
sl@0
    60
  DBUS_BINDING_OUTPUT_PRETTY,
sl@0
    61
  DBUS_BINDING_OUTPUT_GLIB_SERVER,
sl@0
    62
  DBUS_BINDING_OUTPUT_GLIB_CLIENT
sl@0
    63
} DBusBindingOutputMode;
sl@0
    64
sl@0
    65
static void
sl@0
    66
indent (int depth)
sl@0
    67
{
sl@0
    68
  depth *= 2; /* 2-space indent */
sl@0
    69
  
sl@0
    70
  while (depth > 0)
sl@0
    71
    {
sl@0
    72
      putc (' ', stdout);
sl@0
    73
      --depth;
sl@0
    74
    }
sl@0
    75
}
sl@0
    76
sl@0
    77
static void pretty_print (BaseInfo *base,
sl@0
    78
                          int       depth);
sl@0
    79
sl@0
    80
static void
sl@0
    81
pretty_print_list (GSList *list,
sl@0
    82
                   int     depth)
sl@0
    83
{
sl@0
    84
  GSList *tmp;
sl@0
    85
  
sl@0
    86
  tmp = list;
sl@0
    87
  while (tmp != NULL)
sl@0
    88
    {
sl@0
    89
      pretty_print (tmp->data, depth);
sl@0
    90
      tmp = tmp->next;
sl@0
    91
    }
sl@0
    92
}
sl@0
    93
sl@0
    94
static void
sl@0
    95
pretty_print (BaseInfo *base,
sl@0
    96
              int       depth)
sl@0
    97
{
sl@0
    98
  InfoType t;
sl@0
    99
  const char *name;
sl@0
   100
sl@0
   101
  t = base_info_get_type (base);
sl@0
   102
  name = base_info_get_name (base);
sl@0
   103
sl@0
   104
  indent (depth);
sl@0
   105
  
sl@0
   106
  switch (t)
sl@0
   107
    {
sl@0
   108
    case INFO_TYPE_NODE:
sl@0
   109
      {
sl@0
   110
        NodeInfo *n = (NodeInfo*) base;
sl@0
   111
        
sl@0
   112
        if (name == NULL)
sl@0
   113
          printf (_("<anonymous node> {\n"));
sl@0
   114
        else
sl@0
   115
          printf (_("node \"%s\" {\n"), name);
sl@0
   116
sl@0
   117
        pretty_print_list (node_info_get_interfaces (n), depth + 1);
sl@0
   118
        pretty_print_list (node_info_get_nodes (n), depth + 1);
sl@0
   119
sl@0
   120
        indent (depth);
sl@0
   121
        printf ("}\n");
sl@0
   122
      }
sl@0
   123
      break;
sl@0
   124
    case INFO_TYPE_INTERFACE:
sl@0
   125
      {
sl@0
   126
        InterfaceInfo *i = (InterfaceInfo*) base;
sl@0
   127
	GSList *annotations, *elt;
sl@0
   128
sl@0
   129
        g_assert (name != NULL);
sl@0
   130
sl@0
   131
        printf (_("interface \"%s\" {\n"), name);
sl@0
   132
sl@0
   133
	annotations = interface_info_get_annotations (i);
sl@0
   134
	for (elt = annotations; elt; elt = elt->next)
sl@0
   135
	  {
sl@0
   136
	    const char *name = elt->data;
sl@0
   137
	    const char *value = interface_info_get_annotation (i, name);
sl@0
   138
sl@0
   139
	    printf (_(" (binding \"%s\": \"%s\") "),
sl@0
   140
		    name, value);
sl@0
   141
	  }
sl@0
   142
	g_slist_free (annotations);
sl@0
   143
sl@0
   144
        pretty_print_list (interface_info_get_methods (i), depth + 1);
sl@0
   145
        pretty_print_list (interface_info_get_signals (i), depth + 1);
sl@0
   146
        pretty_print_list (interface_info_get_properties (i), depth + 1);
sl@0
   147
sl@0
   148
        indent (depth);
sl@0
   149
        printf ("}\n");
sl@0
   150
      }
sl@0
   151
      break;
sl@0
   152
    case INFO_TYPE_METHOD:
sl@0
   153
      {
sl@0
   154
        MethodInfo *m = (MethodInfo*) base;
sl@0
   155
	GSList *annotations, *elt;
sl@0
   156
sl@0
   157
        g_assert (name != NULL);
sl@0
   158
sl@0
   159
	annotations = method_info_get_annotations (m);
sl@0
   160
        printf (_("method \"%s\""), name);
sl@0
   161
	for (elt = annotations; elt; elt = elt->next)
sl@0
   162
	  {
sl@0
   163
	    const char *name = elt->data;
sl@0
   164
	    const char *value = method_info_get_annotation (m, name);
sl@0
   165
sl@0
   166
	    printf (_(" (annotation \"%s\": \"%s\") "),
sl@0
   167
		    name, value);
sl@0
   168
	  }
sl@0
   169
	g_slist_free (annotations);
sl@0
   170
sl@0
   171
        pretty_print_list (method_info_get_args (m), depth + 1);
sl@0
   172
sl@0
   173
        indent (depth);
sl@0
   174
        printf (")\n");
sl@0
   175
      }
sl@0
   176
      break;
sl@0
   177
    case INFO_TYPE_SIGNAL:
sl@0
   178
      {
sl@0
   179
        SignalInfo *s = (SignalInfo*) base;
sl@0
   180
sl@0
   181
        g_assert (name != NULL);
sl@0
   182
sl@0
   183
        printf (_("signal \"%s\" (\n"), name);
sl@0
   184
sl@0
   185
        pretty_print_list (signal_info_get_args (s), depth + 1);
sl@0
   186
sl@0
   187
        indent (depth);
sl@0
   188
        printf (")\n");
sl@0
   189
      }
sl@0
   190
      break;
sl@0
   191
    case INFO_TYPE_PROPERTY:
sl@0
   192
      {
sl@0
   193
        PropertyInfo *a = (PropertyInfo*) base;
sl@0
   194
        const char *pt = property_info_get_type (a);
sl@0
   195
        PropertyAccessFlags acc = property_info_get_access (a);
sl@0
   196
sl@0
   197
        printf ("%s%s %s",
sl@0
   198
                acc & PROPERTY_READ ? "read" : "",
sl@0
   199
                acc & PROPERTY_WRITE ? "write" : "",
sl@0
   200
                pt);
sl@0
   201
        if (name)
sl@0
   202
          printf (" %s\n", name);
sl@0
   203
        else
sl@0
   204
          printf ("\n");
sl@0
   205
      }
sl@0
   206
      break;
sl@0
   207
    case INFO_TYPE_ARG:
sl@0
   208
      {
sl@0
   209
        ArgInfo *a = (ArgInfo*) base;
sl@0
   210
        const char *at = arg_info_get_type (a);
sl@0
   211
        ArgDirection d = arg_info_get_direction (a);
sl@0
   212
sl@0
   213
        printf ("%s %s",
sl@0
   214
                d == ARG_IN ? "in" : "out",
sl@0
   215
                at);
sl@0
   216
        if (name)
sl@0
   217
          printf (" %s\n", name);
sl@0
   218
        else
sl@0
   219
          printf ("\n");
sl@0
   220
      }
sl@0
   221
      break;
sl@0
   222
    }
sl@0
   223
}
sl@0
   224
sl@0
   225
GQuark
sl@0
   226
dbus_binding_tool_error_quark (void)
sl@0
   227
{
sl@0
   228
  static GQuark quark = 0;
sl@0
   229
  if (!quark)
sl@0
   230
    quark = g_quark_from_static_string ("dbus_binding_tool_error");
sl@0
   231
sl@0
   232
  return quark;
sl@0
   233
}
sl@0
   234
sl@0
   235
static void lose (const char *fmt, ...) G_GNUC_NORETURN G_GNUC_PRINTF (1, 2);
sl@0
   236
static void lose_gerror (const char *prefix, GError *error) G_GNUC_NORETURN;
sl@0
   237
sl@0
   238
static void
sl@0
   239
lose (const char *str, ...)
sl@0
   240
{
sl@0
   241
  va_list args;
sl@0
   242
sl@0
   243
  va_start (args, str);
sl@0
   244
sl@0
   245
  vfprintf (stderr, str, args);
sl@0
   246
  fputc ('\n', stderr);
sl@0
   247
sl@0
   248
  va_end (args);
sl@0
   249
sl@0
   250
  exit (1);
sl@0
   251
}
sl@0
   252
sl@0
   253
static void
sl@0
   254
lose_gerror (const char *prefix, GError *error) 
sl@0
   255
{
sl@0
   256
  lose ("%s: %s", prefix, error->message);
sl@0
   257
}
sl@0
   258
sl@0
   259
static void
sl@0
   260
usage (int ecode)
sl@0
   261
{
sl@0
   262
  fprintf (stderr, "dbus-binding-tool [--version] [--help]\n");
sl@0
   263
  fprintf (stderr, "dbus-binding-tool --mode=[pretty|glib-server|glib-client] [--prefix=SYMBOL_PREFIX] [--ignore-unsupported] [--force] [--output=FILE] [\n");
sl@0
   264
  fprintf (stderr, "dbus-binding-tool --mode=glib-server --prefix=SYMBOL_PREFIX [--ignore-unsupported] [--force] [--output=FILE] [\n");
sl@0
   265
  exit (ecode);
sl@0
   266
}
sl@0
   267
sl@0
   268
static void
sl@0
   269
version (void)
sl@0
   270
{
sl@0
   271
  printf ("D-BUS Binding Tool %s\n"
sl@0
   272
          "Copyright (C) 2003-2005 Red Hat, Inc.\n"
sl@0
   273
          "This is free software; see the source for copying conditions.\n"
sl@0
   274
          "There is NO warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n",
sl@0
   275
          VERSION
sl@0
   276
          );
sl@0
   277
  exit (0);
sl@0
   278
}
sl@0
   279
sl@0
   280
int
sl@0
   281
main (int argc, char **argv)
sl@0
   282
{
sl@0
   283
  const char *output_file;
sl@0
   284
  const char *prefix;
sl@0
   285
  char *output_file_tmp;
sl@0
   286
  int i;
sl@0
   287
  GSList *files;
sl@0
   288
  DBusBindingOutputMode outputmode;
sl@0
   289
  gboolean end_of_args;
sl@0
   290
  GSList *tmp;
sl@0
   291
  GIOChannel *channel;
sl@0
   292
  GError *error;
sl@0
   293
  time_t newest_src;
sl@0
   294
  struct stat srcbuf;
sl@0
   295
  struct stat targetbuf;
sl@0
   296
  gboolean force;
sl@0
   297
  gboolean ignore_unsupported;
sl@0
   298
  gboolean has_prefix = FALSE;
sl@0
   299
  
sl@0
   300
  #ifndef TODO
sl@0
   301
  setlocale (LC_ALL, "");
sl@0
   302
  bindtextdomain (GETTEXT_PACKAGE, DBUS_LOCALEDIR);
sl@0
   303
  bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
sl@0
   304
  textdomain (GETTEXT_PACKAGE); 
sl@0
   305
  #endif
sl@0
   306
  g_type_init ();
sl@0
   307
sl@0
   308
  outputmode = DBUS_BINDING_OUTPUT_NONE;
sl@0
   309
  end_of_args = FALSE;
sl@0
   310
  files = NULL;
sl@0
   311
  output_file = NULL;
sl@0
   312
  prefix = "";
sl@0
   313
  ignore_unsupported = FALSE;
sl@0
   314
  force = FALSE;
sl@0
   315
  i = 1;
sl@0
   316
  while (i < argc)
sl@0
   317
    {
sl@0
   318
      const char *arg = argv[i];
sl@0
   319
sl@0
   320
      if (!end_of_args)
sl@0
   321
        {
sl@0
   322
          if (strcmp (arg, "--help") == 0 ||
sl@0
   323
              strcmp (arg, "-h") == 0 ||
sl@0
   324
              strcmp (arg, "-?") == 0)
sl@0
   325
            usage (0);
sl@0
   326
          else if (strcmp (arg, "--version") == 0)
sl@0
   327
            version ();
sl@0
   328
          else if (strcmp (arg, "--force") == 0)
sl@0
   329
            force = TRUE;
sl@0
   330
#ifdef DBUS_BUILD_TESTS
sl@0
   331
          else if (strcmp (arg, "--self-test") == 0)
sl@0
   332
            run_all_tests (NULL);
sl@0
   333
#endif /* DBUS_BUILD_TESTS */
sl@0
   334
          else if (strncmp (arg, "--mode=", 7) == 0)
sl@0
   335
            {
sl@0
   336
	      const char *mode = arg + 7;
sl@0
   337
	      if (!strcmp (mode, "pretty"))
sl@0
   338
		outputmode = DBUS_BINDING_OUTPUT_PRETTY;
sl@0
   339
	      else if (!strcmp (mode, "glib-server"))
sl@0
   340
		outputmode = DBUS_BINDING_OUTPUT_GLIB_SERVER;
sl@0
   341
	      else if (!strcmp (mode, "glib-client"))
sl@0
   342
		outputmode = DBUS_BINDING_OUTPUT_GLIB_CLIENT;
sl@0
   343
	      else
sl@0
   344
		usage (1);
sl@0
   345
	    }
sl@0
   346
          else if (strcmp (arg, "--ignore-unsupported") == 0)
sl@0
   347
            ignore_unsupported = TRUE;
sl@0
   348
	  else if (strncmp (arg, "--output=", 9) == 0)
sl@0
   349
	    {
sl@0
   350
	      output_file = arg + 9;
sl@0
   351
	    }
sl@0
   352
          else if (strncmp (arg, "--prefix=", 9) == 0)
sl@0
   353
            {
sl@0
   354
              has_prefix = TRUE;
sl@0
   355
              prefix = arg + 9;
sl@0
   356
            }
sl@0
   357
          else if (arg[0] == '-' &&
sl@0
   358
                   arg[1] == '-' &&
sl@0
   359
                   arg[2] == '\0')
sl@0
   360
            end_of_args = TRUE;
sl@0
   361
          else if (arg[0] == '-')
sl@0
   362
            {
sl@0
   363
              usage (1);
sl@0
   364
            }
sl@0
   365
          else
sl@0
   366
            {
sl@0
   367
              files = g_slist_prepend (files, (char*) arg);
sl@0
   368
            }
sl@0
   369
        }
sl@0
   370
      else
sl@0
   371
        files = g_slist_prepend (files, (char*) arg);
sl@0
   372
      
sl@0
   373
      ++i;
sl@0
   374
    }
sl@0
   375
sl@0
   376
  if (outputmode == DBUS_BINDING_OUTPUT_GLIB_SERVER && !has_prefix)
sl@0
   377
    usage (1);
sl@0
   378
sl@0
   379
  error = NULL;
sl@0
   380
sl@0
   381
  files = g_slist_reverse (files);
sl@0
   382
sl@0
   383
  if (output_file && !force)
sl@0
   384
    {
sl@0
   385
      newest_src = 0;
sl@0
   386
      for (tmp = files; tmp != NULL; tmp = tmp->next)
sl@0
   387
	{
sl@0
   388
	  const char *filename;
sl@0
   389
sl@0
   390
	  filename = tmp->data;
sl@0
   391
	  if (stat (filename, &srcbuf) < 0)
sl@0
   392
	    lose ("Couldn't stat %s: %s", filename, g_strerror (errno));
sl@0
   393
sl@0
   394
	  if (srcbuf.st_mtime > newest_src)
sl@0
   395
	    newest_src = srcbuf.st_mtime;
sl@0
   396
	}
sl@0
   397
sl@0
   398
      if (stat (output_file, &targetbuf) > 0
sl@0
   399
	  && targetbuf.st_mtime >= newest_src)
sl@0
   400
	exit (0);
sl@0
   401
    }
sl@0
   402
  
sl@0
   403
  if (output_file)
sl@0
   404
    {
sl@0
   405
      output_file_tmp = g_strconcat (output_file, ".tmp", NULL);
sl@0
   406
sl@0
   407
      if (!(channel = g_io_channel_new_file (output_file_tmp, "w", &error)))
sl@0
   408
	lose_gerror (_("Couldn't open temporary file"), error);
sl@0
   409
    }
sl@0
   410
  else
sl@0
   411
    {
sl@0
   412
      channel = g_io_channel_unix_new (fileno (stdout));
sl@0
   413
      output_file_tmp = NULL; /* silence gcc */
sl@0
   414
    }
sl@0
   415
  if (!g_io_channel_set_encoding (channel, NULL, &error))
sl@0
   416
    lose_gerror (_("Couldn't set channel encoding to NULL"), error);
sl@0
   417
sl@0
   418
sl@0
   419
  for (tmp = files; tmp != NULL; tmp = tmp->next)
sl@0
   420
    {
sl@0
   421
      NodeInfo *node;
sl@0
   422
      GError *error;
sl@0
   423
      const char *filename;
sl@0
   424
sl@0
   425
      filename = tmp->data;
sl@0
   426
sl@0
   427
      error = NULL;
sl@0
   428
      node = description_load_from_file (filename,
sl@0
   429
                                         &error);
sl@0
   430
      if (node == NULL)
sl@0
   431
        {
sl@0
   432
	  lose_gerror (_("Unable to load \"%s\""), error);
sl@0
   433
        }
sl@0
   434
      else
sl@0
   435
	{
sl@0
   436
	  switch (outputmode)
sl@0
   437
	    {
sl@0
   438
	    case DBUS_BINDING_OUTPUT_PRETTY:
sl@0
   439
	      pretty_print ((BaseInfo*) node, 0);
sl@0
   440
	      break;
sl@0
   441
	    case DBUS_BINDING_OUTPUT_GLIB_SERVER:
sl@0
   442
	      if (!dbus_binding_tool_output_glib_server ((BaseInfo *) node, channel, prefix, &error))
sl@0
   443
		lose_gerror (_("Compilation failed"), error);
sl@0
   444
	      break;
sl@0
   445
	    case DBUS_BINDING_OUTPUT_GLIB_CLIENT:
sl@0
   446
	      if (!dbus_binding_tool_output_glib_client ((BaseInfo *) node, channel, ignore_unsupported, &error))
sl@0
   447
		lose_gerror (_("Compilation failed"), error);
sl@0
   448
	      break;
sl@0
   449
	    case DBUS_BINDING_OUTPUT_NONE:
sl@0
   450
	      break;
sl@0
   451
	    }
sl@0
   452
	}
sl@0
   453
sl@0
   454
      if (node)
sl@0
   455
        node_info_unref (node);
sl@0
   456
    }
sl@0
   457
sl@0
   458
  if (g_io_channel_shutdown (channel, TRUE, &error) != G_IO_STATUS_NORMAL)
sl@0
   459
    lose_gerror (_("Failed to shutdown IO channel"), error);
sl@0
   460
  g_io_channel_unref (channel);
sl@0
   461
sl@0
   462
  if (output_file)
sl@0
   463
    {
sl@0
   464
      if (rename (output_file_tmp, output_file) < 0)
sl@0
   465
	lose ("Failed to rename %s to %s: %s", output_file_tmp, output_file,
sl@0
   466
	      g_strerror (errno));
sl@0
   467
      g_free (output_file_tmp);
sl@0
   468
    }
sl@0
   469
sl@0
   470
  return 0;
sl@0
   471
}
sl@0
   472
sl@0
   473
sl@0
   474
#ifdef DBUS_BUILD_TESTS
sl@0
   475
static void
sl@0
   476
test_die (const char *failure)
sl@0
   477
{
sl@0
   478
  lose ("Unit test failed: %s", failure);
sl@0
   479
}
sl@0
   480
sl@0
   481
/**
sl@0
   482
 * @ingroup DBusGTool
sl@0
   483
 * Unit test for GLib utility tool
sl@0
   484
 * Returns: #TRUE on success.
sl@0
   485
 */
sl@0
   486
static gboolean
sl@0
   487
_dbus_gtool_test (const char *test_data_dir)
sl@0
   488
{
sl@0
   489
sl@0
   490
  return TRUE;
sl@0
   491
}
sl@0
   492
sl@0
   493
static void
sl@0
   494
run_all_tests (const char *test_data_dir)
sl@0
   495
{
sl@0
   496
  if (test_data_dir == NULL)
sl@0
   497
    test_data_dir = g_getenv ("DBUS_TEST_DATA");
sl@0
   498
sl@0
   499
  if (test_data_dir != NULL)
sl@0
   500
    printf ("Test data in %s\n", test_data_dir);
sl@0
   501
  else
sl@0
   502
    printf ("No test data!\n");
sl@0
   503
sl@0
   504
  printf ("%s: running binding tests\n", "dbus-binding-tool");
sl@0
   505
  if (!_dbus_gtool_test (test_data_dir))
sl@0
   506
    test_die ("gtool");
sl@0
   507
sl@0
   508
  printf ("%s: completed successfully\n", "dbus-binding-tool");
sl@0
   509
}
sl@0
   510
sl@0
   511
#endif /* DBUS_BUILD_TESTS */