dstar2tgba.cc 12.7 KB
Newer Older
1
// -*- coding: utf-8 -*-
2
3
// Copyright (C) 2013, 2014, 2015 Laboratoire de Recherche et
// Développement de l'Epita (LRDE).
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
//
// This file is part of Spot, a model checking library.
//
// Spot 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.
//
// Spot 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/>.

#include "common_sys.hh"

#include <string>
#include <iostream>
24
#include <memory>
25
26
27
28
29
30
31
32

#include <argp.h>
#include "error.h"

#include "common_setup.hh"
#include "common_finput.hh"
#include "common_cout.hh"
#include "common_post.hh"
33
#include "common_file.hh"
34

35
36
37
38
39
#include "twaalgos/dotty.hh"
#include "twaalgos/lbtt.hh"
#include "twaalgos/hoa.hh"
#include "twaalgos/neverclaim.hh"
#include "twaalgos/stats.hh"
40
#include "twa/bddprint.hh"
41
#include "misc/optionmap.hh"
42
#include "misc/timer.hh"
43
#include "dstarparse/public.hh"
44
#include "twaalgos/sccinfo.hh"
45

46
static const char argp_program_doc[] ="\
47
48
49
50
51
52
Convert Rabin and Streett automata into Büchi automata.\n\n\
This reads the output format of ltl2dstar and will output a \n\
Transition-based Generalized Büchi Automata in GraphViz's format by default.\n\
If multiple files are supplied (one automaton per file), several automata\n\
will be output.";

53
54
55
56
57
58
59
enum {
  OPT_DOT = 1,
  OPT_LBTT,
  OPT_NAME,
  OPT_STATS,
  OPT_TGBA,
};
60
61
62
63
64
65
66
67
68
69
70
71
72

static const argp_option options[] =
  {
    /**************************************************/
    { 0, 0, 0, 0, "Input:", 1 },
    { "file", 'F', "FILENAME", 0,
      "process the automaton in FILENAME", 0 },
    /**************************************************/
    { 0, 0, 0, 0, "Output automaton type:", 2 },
    { "tgba", OPT_TGBA, 0, 0,
      "Transition-based Generalized Büchi Automaton (default)", 0 },
    { "ba", 'B', 0, 0, "Büchi Automaton", 0 },
    { "monitor", 'M', 0, 0, "Monitor (accepts all finite prefixes "
73
      "of the given property)", 0 },
74
75
    /**************************************************/
    { 0, 0, 0, 0, "Output format:", 3 },
76
    { "dot", OPT_DOT, "1|a|b|c|f(FONT)|h|n|N|o|r|R|s|t|v", OPTION_ARG_OPTIONAL,
77
      "GraphViz's format (default).  Add letters for "
78
79
      "(1) force numbered states, "
      "(a) acceptance display, (b) acceptance sets as bullets, "
80
81
      "(c) circular nodes, (f(FONT)) use FONT, (h) horizontal layout, "
      "(v) vertical layout, (n) with name, (N) without name, "
82
83
84
      "(o) ordered transitions, "
      "(r) rainbow colors for acceptance sets, "
      "(R) color acceptance sets by Inf/Fin, (s) with SCCs, "
85
      "(t) force transition-based acceptance.", 0 },
86
    { "hoaf", 'H', "i|s|t|m|l", OPTION_ARG_OPTIONAL,
87
      "Output the automaton in HOA format.  Add letters to select "
88
      "(i) use implicit labels for complete deterministic automata, "
89
90
91
92
      "(s) prefer state-based acceptance when possible [default], "
      "(t) force transition-based acceptance, "
      "(m) mix state and transition-based acceptance, "
      "(l) single-line output", 0 },
93
94
95
    { "lbtt", OPT_LBTT, "t", OPTION_ARG_OPTIONAL,
      "LBTT's format (add =t to force transition-based acceptance even"
      " on Büchi automata)", 0 },
96
97
    { "name", OPT_NAME, "FORMAT", 0,
      "set the name of the output automaton", 0 },
98
99
100
101
    { "output", 'o', "FORMAT", 0,
      "send output to a file named FORMAT instead of standard output.  The"
      " first automaton sent to a file truncates it unless FORMAT starts"
      " with '>>'.", 0 },
102
103
104
    { "spin", 's', "6|c", OPTION_ARG_OPTIONAL, "Spin neverclaim (implies --ba)."
      "  Add letters to select (6) Spin's 6.2.4 style, (c) comments on states",
      0 },
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
    { "utf8", '8', 0, 0, "enable UTF-8 characters in output "
      "(ignored with --lbtt or --spin)", 0 },
    { "stats", OPT_STATS, "FORMAT", 0,
      "output statistics about the automaton", 0 },
    /**************************************************/
    { 0, 0, 0, 0, "The FORMAT string passed to --stats may use "\
      "the following interpreted sequences (capitals for input,"
      " minuscules for output):", 4 },
    { "%F", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "name of the input file", 0 },
    { "%S, %s", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "number of states", 0 },
    { "%E, %e", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "number of edges", 0 },
    { "%T, %t", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "number of transitions", 0 },
    { "%A, %a", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "number of acceptance pairs or sets", 0 },
    { "%C, %c", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "number of SCCs", 0 },
    { "%n", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "number of nondeterministic states in output", 0 },
    { "%d", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "1 if the output is deterministic, 0 otherwise", 0 },
129
130
    { "%p", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "1 if the output is complete, 0 otherwise", 0 },
131
132
133
    { "%r", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "conversion time (including post-processings, but not parsing)"
      " in seconds", 0 },
134
135
136
137
138
139
140
141
142
    { "%%", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "a single %", 0 },
    /**************************************************/
    { 0, 0, 0, 0, "Miscellaneous options:", -1 },
    { "extra-options", 'x', "OPTS", 0,
      "fine-tuning options (see spot-x (7))", 0 },
    { 0, 0, 0, 0, 0, 0 }
  };

143
static const struct argp_child children[] =
144
145
146
147
148
149
  {
    { &post_argp, 0, 0, 20 },
    { &misc_argp, 0, 0, -1 },
    { 0, 0, 0, 0 }
  };

Alexandre Duret-Lutz's avatar
Alexandre Duret-Lutz committed
150
enum output_format { Dot, Lbtt, Lbtt_t, Spin, Stats, Hoa };
151
152
153
154
static output_format format = Dot;
static const char* opt_dot = nullptr;
static const char* stats = "";
static const char* hoa_opt = nullptr;
155
static const char* opt_never = nullptr;
156
static const char* opt_name = nullptr;
157
static const char* opt_output = nullptr;
158
static spot::option_map extra_options;
159
160
161
162
163
164
165
166
167
168
169
170
171
172

static int
parse_opt(int key, char* arg, struct argp_state*)
{
  // This switch is alphabetically-ordered.
  switch (key)
    {
    case '8':
      spot::enable_utf8();
      break;
    case 'B':
      type = spot::postprocessor::BA;
      break;
    case 'F':
173
      jobs.emplace_back(arg, true);
174
      break;
175
176
    case 'H':
      format = Hoa;
177
      hoa_opt = arg;
178
      break;
179
180
181
    case 'M':
      type = spot::postprocessor::Monitor;
      break;
182
183
184
    case 'o':
      opt_output = arg;
      break;
185
186
187
188
    case 's':
      format = Spin;
      if (type != spot::postprocessor::Monitor)
	type = spot::postprocessor::BA;
189
190
      if (arg)
	opt_never = arg;
191
192
193
194
195
196
197
198
199
200
      break;
    case 'x':
      {
	const char* opt = extra_options.parse_options(arg);
	if (opt)
	  error(2, 0, "failed to parse --options near '%s'", opt);
      }
      break;
    case OPT_DOT:
      format = Dot;
201
      opt_dot = arg;
202
203
204
205
206
207
208
209
210
211
212
213
214
215
      break;
    case OPT_LBTT:
      if (arg)
	{
	  if (arg[0] == 't' && arg[1] == 0)
	    format = Lbtt_t;
	  else
	    error(2, 0, "unknown argument for --lbtt: '%s'", arg);
	}
      else
	{
	  format = Lbtt;
	}
      break;
216
217
218
    case OPT_NAME:
      opt_name = arg;
      break;
219
220
221
222
223
224
225
226
227
228
229
230
    case OPT_STATS:
      if (!*arg)
	error(2, 0, "empty format string for --stats");
      stats = arg;
      format = Stats;
      break;
    case OPT_TGBA:
      if (format == Spin)
	error(2, 0, "--spin and --tgba are incompatible");
      type = spot::postprocessor::TGBA;
      break;
    case ARGP_KEY_ARG:
231
      jobs.emplace_back(arg, true);
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
      break;

    default:
      return ARGP_ERR_UNKNOWN;
    }
  return 0;
}


namespace
{
  /// \brief prints various statistics about a TGBA
  ///
  /// This object can be configured to display various statistics
  /// about a TGBA.  Some %-sequence of characters are interpreted in
  /// the format string, and replaced by the corresponding statistics.
  class dstar_stat_printer: protected spot::stat_printer
  {
  public:
    dstar_stat_printer(std::ostream& os, const char* format)
      : spot::stat_printer(os, format)
    {
      declare('A', &daut_acc_);
      declare('C', &daut_scc_);
      declare('E', &daut_edges_);
      declare('F', &filename_);
      declare('f', &filename_);	// Override the formula printer.
      declare('S', &daut_states_);
      declare('T', &daut_trans_);
261
      declare('m', &aut_name_);
262
263
    }

264
265
    using spot::formater::set_output;

266
267
268
269
270
    /// \brief print the configured statistics.
    ///
    /// The \a f argument is not needed if the Formula does not need
    /// to be output.
    std::ostream&
271
    print(const spot::const_dstar_aut_ptr& daut,
272
	  const spot::const_twa_graph_ptr& aut,
273
	  const char* filename, double run_time)
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
    {
      filename_ = filename;

      if (has('T'))
	{
	  spot::tgba_sub_statistics s = sub_stats_reachable(daut->aut);
	  daut_states_ = s.states;
	  daut_edges_ = s.transitions;
	  daut_trans_ = s.sub_transitions;
	}
      else if (has('E'))
	{
	  spot::tgba_sub_statistics s = sub_stats_reachable(daut->aut);
	  daut_states_ = s.states;
	  daut_edges_ = s.transitions;
	}
      else if (has('S'))
	{
	  daut_states_ = daut->aut->num_states();
	}

      if (has('A'))
	daut_acc_ = daut->accpair_count;

      if (has('C'))
299
	daut_scc_ = spot::scc_info(daut->aut).scc_count();
300

301
302
303
304
305
306
307
308
309
      if (has('m'))
	{
	  auto n = aut->get_named_prop<std::string>("automaton-name");
	  if (n)
	    aut_name_ = *n;
	  else
	    aut_name_.val().clear();
	}

310
      return this->spot::stat_printer::print(aut, 0, run_time);
311
312
313
314
315
316
317
318
319
    }

  private:
    spot::printable_value<const char*> filename_;
    spot::printable_value<unsigned> daut_states_;
    spot::printable_value<unsigned> daut_edges_;
    spot::printable_value<unsigned> daut_trans_;
    spot::printable_value<unsigned> daut_acc_;
    spot::printable_value<unsigned> daut_scc_;
320
    spot::printable_value<std::string> aut_name_;
321
322
323
324
325
326
327
328
  };


  class dstar_processor: public job_processor
  {
  public:
    spot::postprocessor& post;
    dstar_stat_printer statistics;
329
330
    std::ostringstream name;
    dstar_stat_printer namer;
331
332
333
    std::ostringstream outputname;
    dstar_stat_printer outputnamer;
    std::map<std::string, std::unique_ptr<output_file>> outputfiles;
334
335

    dstar_processor(spot::postprocessor& post)
336
      : post(post), statistics(std::cout, stats),
337
338
	namer(name, opt_name),
	outputnamer(outputname, opt_output)
339
340
341
342
343
344
    {
    }

    int
    process_formula(const spot::ltl::formula*, const char*, int)
    {
345
      SPOT_UNREACHABLE();
346
347
348
349
350
351
352
    }


    int
    process_file(const char* filename)
    {
      spot::dstar_parse_error_list pel;
353
      auto daut = spot::dstar_parse(filename, pel, spot::make_bdd_dict());
354
      if (spot::format_dstar_parse_errors(std::cerr, filename, pel))
355
	return 2;
356
      if (!daut)
357
	error(2, 0, "failed to read automaton from %s", filename);
358

359
360
      spot::stopwatch sw;
      sw.start();
361
362
      auto nba = spot::dstar_to_tgba(daut);
      auto aut = post.run(nba, 0);
363
      const double conversion_time = sw.stop();
364

365
366
367
368
369
370
371
372
      // Name the output automaton.
      if (opt_name)
	{
	  name.str("");
	  namer.print(daut, aut, filename, conversion_time);
	  aut->set_named_prop("automaton-name", new std::string(name.str()));
	}

373
374
375
376
377
378
379
380
381
382
383
384
      std::ostream* out = &std::cout;
      if (opt_output)
	{
	  outputname.str("");
	  outputnamer.print(daut, aut, filename, conversion_time);
	  std::string fname = outputname.str();
	  auto p = outputfiles.emplace(fname, nullptr);
	  if (p.second)
	    p.first->second.reset(new output_file(fname.c_str()));
	  out = &p.first->second->ostream();
	}

385
386
387
      switch (format)
	{
	case Dot:
388
	  spot::dotty_reachable(*out, aut, opt_dot);
389
390
	  break;
	case Lbtt:
391
	  spot::lbtt_reachable(*out, aut, type == spot::postprocessor::BA);
392
393
	  break;
	case Lbtt_t:
394
	  spot::lbtt_reachable(*out, aut, false);
395
	  break;
396
	case Hoa:
397
	  spot::hoa_reachable(*out, aut, hoa_opt) << '\n';
398
	  break;
399
	case Spin:
400
	  spot::never_claim_reachable(*out, aut, opt_never);
401
402
	  break;
	case Stats:
403
	  statistics.set_output(*out);
404
	  statistics.print(daut, aut, filename, conversion_time) << '\n';
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
	  break;
	}
      flush_cout();
      return 0;
    }
  };
}

int
main(int argc, char** argv)
{
  setup(argv);

  const argp ap = { options, parse_opt, "[FILENAMES...]",
		    argp_program_doc, children, 0, 0 };

  if (int err = argp_parse(&ap, argc, argv, ARGP_NO_HELP, 0, 0))
    exit(err);

  if (jobs.empty())
425
    jobs.emplace_back("-", true);
426
427

  spot::postprocessor post(&extra_options);
428
  post.set_pref(pref | comp);
429
430
431
  post.set_type(type);
  post.set_level(level);

432
433
434
435
436
437
438
439
440
441
  try
    {
      dstar_processor processor(post);
      if (processor.run())
	return 2;
    }
  catch (const std::runtime_error& e)
    {
      error(2, 0, "%s", e.what());
    }
442
443
  return 0;
}