ltlfilt.cc 18.6 KB
Newer Older
1
// -*- coding: utf-8 -*-
2
// Copyright (C) 2012, 2013, 2014, 2015 Laboratoire de Recherche et
3
// Développement de l'Epita (LRDE).
4 5 6 7 8
//
// 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
9
// the Free Software Foundation; either version 3 of the License, or
10 11 12 13 14 15 16 17
// (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
18
// along with this program.  If not, see <http://www.gnu.org/licenses/>.
19

20
#include "common_sys.hh"
21 22 23 24 25 26

#include <cstdlib>
#include <string>
#include <iostream>
#include <fstream>
#include <argp.h>
27
#include <cstring>
28
#include "error.h"
29

30
#include "common_setup.hh"
31
#include "common_finput.hh"
32
#include "common_output.hh"
33
#include "common_cout.hh"
34
#include "common_conv.hh"
35
#include "common_r.hh"
36

37 38 39
#include "misc/hash.hh"
#include "ltlvisit/simplify.hh"
#include "ltlvisit/length.hh"
40
#include "ltlvisit/relabel.hh"
41
#include "ltlvisit/wmunabbrev.hh"
42
#include "ltlvisit/remove_x.hh"
43
#include "ltlvisit/apcollect.hh"
44
#include "ltlast/unop.hh"
45
#include "ltlast/multop.hh"
46 47 48
#include "tgbaalgos/ltl2tgba_fm.hh"
#include "tgbaalgos/minimize.hh"
#include "tgbaalgos/safety.hh"
49
#include "tgbaalgos/stutter.hh"
50 51

const char argp_program_doc[] ="\
52 53 54
Read a list of formulas and output them back after some optional processing.\v\
Exit status:\n\
  0  if some formulas were output (skipped syntax errors do not count)\n\
55
  1  if no formulas were output (no match)\n\
56
  2  if any error has been reported";
57

58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
enum {
  OPT_AP_N = 1,
  OPT_BOOLEAN,
  OPT_BOOLEAN_TO_ISOP,
  OPT_BSIZE_MAX,
  OPT_BSIZE_MIN,
  OPT_DROP_ERRORS,
  OPT_EQUIVALENT_TO,
  OPT_EVENTUAL,
  OPT_GUARANTEE,
  OPT_IGNORE_ERRORS,
  OPT_IMPLIED_BY,
  OPT_IMPLY,
  OPT_LTL,
  OPT_NEGATE,
  OPT_NNF,
  OPT_OBLIGATION,
  OPT_RELABEL,
  OPT_RELABEL_BOOL,
  OPT_REMOVE_WM,
  OPT_REMOVE_X,
  OPT_SAFETY,
  OPT_SIZE_MAX,
  OPT_SIZE_MIN,
  OPT_SKIP_ERRORS,
  OPT_STUTTER_INSENSITIVE,
  OPT_SYNTACTIC_GUARANTEE,
  OPT_SYNTACTIC_OBLIGATION,
  OPT_SYNTACTIC_PERSISTENCE,
  OPT_SYNTACTIC_RECURRENCE,
  OPT_SYNTACTIC_SAFETY,
  OPT_SYNTACTIC_SI,
  OPT_UNIVERSAL,
};
92 93 94 95 96 97 98 99 100

static const argp_option options[] =
  {
    /**************************************************/
    { 0, 0, 0, 0, "Error handling:", 2 },
    { "skip-errors", OPT_SKIP_ERRORS, 0, 0,
      "output erroneous lines as-is without processing", 0 },
    { "drop-errors", OPT_DROP_ERRORS, 0, 0,
      "discard erroneous lines (default)", 0 },
101 102
    { "ignore-errors", OPT_IGNORE_ERRORS, 0, 0,
      "do not report syntax errors", 0 },
103 104
    /**************************************************/
    { 0, 0, 0, 0, "Transformation options:", 3 },
Alexandre Duret-Lutz's avatar
Alexandre Duret-Lutz committed
105
    { "negate", OPT_NEGATE, 0, 0, "negate each formula", 0 },
106
    { "nnf", OPT_NNF, 0, 0, "rewrite formulas in negative normal form", 0 },
107 108 109
    { "relabel", OPT_RELABEL, "abc|pnn", OPTION_ARG_OPTIONAL,
      "relabel all atomic propositions, alphabetically unless " \
      "specified otherwise", 0 },
110 111 112
    { "relabel-bool", OPT_RELABEL_BOOL, "abc|pnn", OPTION_ARG_OPTIONAL,
      "relabel Boolean subexpressions, alphabetically unless " \
      "specified otherwise", 0 },
113 114
    { "remove-wm", OPT_REMOVE_WM, 0, 0,
      "rewrite operators W and M using U and R", 0 },
115 116 117
    { "boolean-to-isop", OPT_BOOLEAN_TO_ISOP, 0, 0,
      "rewrite Boolean subformulas as irredundant sum of products "
      "(implies at least -r1)", 0 },
118 119 120
    { "remove-x", OPT_REMOVE_X, 0, 0,
      "remove X operators (valid only for stutter-insensitive properties)",
      0 },
121 122
    DECLARE_OPT_R,
    LEVEL_DOC(4),
123 124 125
    /**************************************************/
    { 0, 0, 0, 0,
      "Filtering options (matching is done after transformation):", 5 },
126
    { "ltl", OPT_LTL, 0, 0, "match only LTL formulas (no PSL operator)", 0 },
127 128 129 130 131 132 133 134 135 136 137 138 139
    { "boolean", OPT_BOOLEAN, 0, 0, "match Boolean formulas", 0 },
    { "eventual", OPT_EVENTUAL, 0, 0, "match pure eventualities", 0 },
    { "universal", OPT_UNIVERSAL, 0, 0, "match purely universal formulas", 0 },
    { "syntactic-safety", OPT_SYNTACTIC_SAFETY, 0, 0,
      "match syntactic-safety formulas", 0 },
    { "syntactic-guarantee", OPT_SYNTACTIC_GUARANTEE, 0, 0,
      "match syntactic-guarantee formulas", 0 },
    { "syntactic-obligation", OPT_SYNTACTIC_OBLIGATION, 0, 0,
      "match syntactic-obligation formulas", 0 },
    { "syntactic-recurrence", OPT_SYNTACTIC_RECURRENCE, 0, 0,
      "match syntactic-recurrence formulas", 0 },
    { "syntactic-persistence", OPT_SYNTACTIC_PERSISTENCE, 0, 0,
      "match syntactic-persistence formulas", 0 },
140 141 142
    { "syntactic-stutter-invariant", OPT_SYNTACTIC_SI, 0, 0,
      "match stutter-invariant formulas syntactically (LTL-X or siPSL)", 0 },
    { "nox", 0, 0, OPTION_ALIAS, 0, 0 },
143 144 145 146 147 148 149 150 151 152 153 154 155 156
    { "safety", OPT_SAFETY, 0, 0,
      "match safety formulas (even pathological)", 0 },
    { "guarantee", OPT_GUARANTEE, 0, 0,
      "match guarantee formulas (even pathological)", 0 },
    { "obligation", OPT_OBLIGATION, 0, 0,
      "match obligation formulas (even pathological)", 0 },
    { "size-max", OPT_SIZE_MAX, "INT", 0,
      "match formulas with size <= INT", 0 },
    { "size-min", OPT_SIZE_MIN, "INT", 0,
      "match formulas with size >= INT", 0 },
    { "bsize-max", OPT_BSIZE_MAX, "INT", 0,
      "match formulas with Boolean size <= INT", 0 },
    { "bsize-min", OPT_BSIZE_MIN, "INT", 0,
      "match formulas with Boolean size >= INT", 0 },
157 158 159 160 161 162
    { "implied-by", OPT_IMPLIED_BY, "FORMULA", 0,
      "match formulas implied by FORMULA", 0 },
    { "imply", OPT_IMPLY, "FORMULA", 0,
      "match formulas implying FORMULA", 0 },
    { "equivalent-to", OPT_EQUIVALENT_TO, "FORMULA", 0,
      "match formulas equivalent to FORMULA", 0 },
163 164 165
    { "stutter-insensitive", OPT_STUTTER_INSENSITIVE, 0, 0,
      "match stutter-insensitive LTL formulas", 0 },
    { "stutter-invariant", 0, 0, OPTION_ALIAS, 0, 0 },
166 167
    { "ap", OPT_AP_N, "N", 0,
      "match formulas which use exactly N atomic propositions", 0 },
168 169 170
    { "invert-match", 'v', 0, 0, "select non-matching formulas", 0},
    { "unique", 'u', 0, 0,
      "drop formulas that have already been output (not affected by -v)", 0 },
171
    /**************************************************/
172
    { 0, 0, 0, 0, "Output options:", -20 },
173
    { "count", 'c', 0, 0, "print only a count of matched formulas", 0 },
174
    { "quiet", 'q', 0, 0, "suppress all normal output", 0 },
175
    { "max-count", 'n', "NUM", 0, "output at most NUM formulas", 0 },
176 177 178 179 180 181 182 183
    { 0, 0, 0, 0, "The FORMAT string passed to --format may use "\
      "the following interpreted sequences:", -19 },
    { "%f", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "the formula (in the selected syntax)", 0 },
    { "%F", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "the name of the input file", 0 },
    { "%L", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "the original line number in the input file", 0 },
184 185 186 187 188 189
    { "%<", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "the part of the line before the formula if it "
      "comes from a column extracted from a CSV file", 0 },
    { "%>", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "the part of the line after the formula if it "
      "comes from a column extracted from a CSV file", 0 },
190 191
    { "%%", 0, 0, OPTION_DOC | OPTION_NO_USAGE,
      "a single %", 0 },
192 193 194 195
    { 0, 0, 0, 0, "Miscellaneous options:", -1 },
    { 0, 0, 0, 0, 0, 0 }
  };

196 197
const struct argp_child children[] =
  {
198
    { &finput_argp, 0, 0, 1 },
199
    { &output_argp, 0, 0, -20 },
200
    { &misc_argp, 0, 0, -1 },
201 202 203
    { 0, 0, 0, 0 }
  };

204 205
static bool one_match = false;

206 207
enum error_style_t { drop_errors, skip_errors };
static error_style_t error_style = drop_errors;
208
static bool ignore_errors = false;
209 210
static bool nnf = false;
static bool negate = false;
211
static bool boolean_to_isop = false;
212 213 214 215 216 217 218 219 220 221 222 223
static bool unique = false;
static bool psl = false;
static bool ltl = false;
static bool invert = false;
static bool boolean = false;
static bool universal = false;
static bool eventual = false;
static bool syntactic_safety = false;
static bool syntactic_guarantee = false;
static bool syntactic_obligation = false;
static bool syntactic_recurrence = false;
static bool syntactic_persistence = false;
224
static bool syntactic_si = false;
225 226 227 228 229 230 231
static bool safety = false;
static bool guarantee = false;
static bool obligation = false;
static int size_min = -1;
static int size_max = -1;
static int bsize_min = -1;
static int bsize_max = -1;
232 233
enum relabeling_mode { NoRelabeling = 0, ApRelabeling, BseRelabeling };
static relabeling_mode relabeling = NoRelabeling;
234
static spot::ltl::relabeling_style style = spot::ltl::Abc;
235
static bool remove_wm = false;
236 237
static bool remove_x = false;
static bool stutter_insensitive = false;
238 239
static bool ap = false;
static unsigned ap_n = 0;
240
static int opt_max_count = -1;
241
static long int match_count = 0;
242

243 244 245 246 247 248 249 250
static const spot::ltl::formula* implied_by = 0;
static const spot::ltl::formula* imply = 0;
static const spot::ltl::formula* equivalent_to = 0;

static const spot::ltl::formula*
parse_formula_arg(const std::string& input)
{
  spot::ltl::parse_error_list pel;
251
  const spot::ltl::formula* f = parse_formula(input, pel);
252
  if (spot::ltl::format_parse_errors(std::cerr, input, pel))
253
    error(2, 0, "parse error when parsing an argument");
254 255 256 257 258
  return f;
}



259
static int
260
parse_opt(int key, char* arg, struct argp_state*)
261 262 263 264
{
  // This switch is alphabetically-ordered.
  switch (key)
    {
265 266 267
    case 'c':
      output_format = count_output;
      break;
268 269 270
    case 'n':
      opt_max_count = to_pos_int(arg);
      break;
271
    case 'q':
272
      output_format = quiet_output;
273
      break;
274 275
    case OPT_R:
      parse_r(arg);
276 277 278 279 280 281 282 283 284
      break;
    case 'u':
      unique = true;
      break;
    case 'v':
      invert = true;
      break;
    case ARGP_KEY_ARG:
      // FIXME: use stat() to distinguish filename from string?
285
      jobs.emplace_back(arg, true);
286 287 288 289
      break;
    case OPT_BOOLEAN:
      boolean = true;
      break;
290 291 292
    case OPT_BOOLEAN_TO_ISOP:
      boolean_to_isop = true;
      break;
293 294 295 296 297 298 299 300 301
    case OPT_BSIZE_MIN:
      bsize_min = to_int(arg);
      break;
    case OPT_BSIZE_MAX:
      bsize_max = to_int(arg);
      break;
    case OPT_DROP_ERRORS:
      error_style = drop_errors;
      break;
302 303 304 305 306 307 308
    case OPT_EQUIVALENT_TO:
      {
	if (equivalent_to)
	  error(2, 0, "only one --equivalent-to option can be given");
	equivalent_to = parse_formula_arg(arg);
	break;
      }
309 310 311
    case OPT_EVENTUAL:
      eventual = true;
      break;
312 313 314
    case OPT_GUARANTEE:
      guarantee = obligation = true;
      break;
315 316 317
    case OPT_IGNORE_ERRORS:
      ignore_errors = true;
      break;
318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333
    case OPT_IMPLIED_BY:
      {
	const spot::ltl::formula* i = parse_formula_arg(arg);
	// a→c∧b→c ≡ (a∨b)→c
	implied_by =
	  spot::ltl::multop::instance(spot::ltl::multop::Or, implied_by, i);
	break;
      }
    case OPT_IMPLY:
      {
	// a→b∧a→c ≡ a→(b∧c)
	const spot::ltl::formula* i = parse_formula_arg(arg);
	imply =
	  spot::ltl::multop::instance(spot::ltl::multop::And, imply, i);
	break;
      }
334 335 336
    case OPT_LTL:
      ltl = true;
      break;
Alexandre Duret-Lutz's avatar
Alexandre Duret-Lutz committed
337 338 339
    case OPT_NEGATE:
      negate = true;
      break;
340 341 342 343 344 345
    case OPT_NNF:
      nnf = true;
      break;
    case OPT_OBLIGATION:
      obligation = true;
      break;
346
    case OPT_RELABEL:
347 348
    case OPT_RELABEL_BOOL:
      relabeling = (key == OPT_RELABEL_BOOL ? BseRelabeling : ApRelabeling);
349 350 351 352 353
      if (!arg || !strncasecmp(arg, "abc", 6))
	style = spot::ltl::Abc;
      else if (!strncasecmp(arg, "pnn", 4))
	style = spot::ltl::Pnn;
      else
354 355 356
	error(2, 0, "invalid argument for --relabel%s: '%s'",
	      (key == OPT_RELABEL_BOOL ? "-bool" : ""),
	      arg);
357
      break;
358 359 360
    case OPT_REMOVE_WM:
      remove_wm = true;
      break;
361 362 363
    case OPT_REMOVE_X:
      remove_x = true;
      break;
364 365 366 367 368 369 370 371 372 373 374 375
    case OPT_SAFETY:
      safety = obligation = true;
      break;
    case OPT_SIZE_MIN:
      size_min = to_int(arg);
      break;
    case OPT_SIZE_MAX:
      size_max = to_int(arg);
      break;
    case OPT_SKIP_ERRORS:
      error_style = skip_errors;
      break;
376 377 378
    case OPT_STUTTER_INSENSITIVE:
      stutter_insensitive = true;
      break;
379 380 381 382
    case OPT_AP_N:
      ap = true;
      ap_n = to_int(arg);
      break;
383 384 385 386 387 388 389 390 391 392 393 394 395 396 397
    case OPT_SYNTACTIC_SAFETY:
      syntactic_safety = true;
      break;
    case OPT_SYNTACTIC_GUARANTEE:
      syntactic_guarantee = true;
      break;
    case OPT_SYNTACTIC_OBLIGATION:
      syntactic_obligation = true;
      break;
    case OPT_SYNTACTIC_RECURRENCE:
      syntactic_recurrence = true;
      break;
    case OPT_SYNTACTIC_PERSISTENCE:
      syntactic_persistence = true;
      break;
398 399 400
    case OPT_SYNTACTIC_SI:
      syntactic_si = true;
      break;
401 402 403
    case OPT_UNIVERSAL:
      universal = true;
      break;
404 405 406 407 408 409
    default:
      return ARGP_ERR_UNKNOWN;
    }
  return 0;
}

410 411 412
typedef
std::unordered_set<const spot::ltl::formula*,
		   const spot::ptr_hash<const spot::ltl::formula>> fset_t;
413 414 415

namespace
{
416
  class ltl_processor: public job_processor
417 418 419 420 421
  {
  public:
    spot::ltl::ltl_simplifier& simpl;
    fset_t unique_set;

422 423 424 425 426
    ~ltl_processor()
    {
      fset_t::iterator i = unique_set.begin();
      while (i != unique_set.end())
	(*i++)->destroy();
427 428 429 430 431 432 433

      if (equivalent_to)
	equivalent_to->destroy();
      if (implied_by)
	implied_by->destroy();
      if (imply)
	imply->destroy();
434 435
    }

436 437 438 439 440 441
    ltl_processor(spot::ltl::ltl_simplifier& simpl)
    : simpl(simpl)
    {
    }

    int
442
    process_string(const std::string& input,
443 444 445
		    const char* filename = 0, int linenum = 0)
    {
      spot::ltl::parse_error_list pel;
446
      const spot::ltl::formula* f = parse_formula(input, pel);
447 448 449

      if (!f || pel.size() > 0)
	  {
450
	    if (!ignore_errors)
451 452
	      {
		if (filename)
453
		  error_at_line(0, 0, filename, linenum, "parse error:");
454 455 456 457 458 459 460
		spot::ltl::format_parse_errors(std::cerr, input, pel);
	      }

	    if (f)
	      f->destroy();

	    if (error_style == skip_errors)
461 462 463
	      std::cout << input << std::endl;
	    else
	      assert(error_style == drop_errors);
464
	    check_cout();
465
	    return !ignore_errors;
466
	  }
467 468 469 470 471 472 473 474 475
      try
	{
	  return process_formula(f, filename, linenum);
	}
      catch (const std::runtime_error& e)
	{
	  error_at_line(2, 0, filename, linenum, "%s", e.what());
	  SPOT_UNREACHABLE();
	}
476
    }
477

478 479 480 481
    int
    process_formula(const spot::ltl::formula* f,
		    const char* filename = 0, int linenum = 0)
    {
482
      if (opt_max_count >= 0 && match_count >= opt_max_count)
483 484 485 486 487 488
	{
	  abort_run = true;
	  f->destroy();
	  return 0;
	}

489
      if (negate)
490
	f = spot::ltl::unop::instance(spot::ltl::unop::Not, f);
491

492 493 494 495 496 497 498 499 500 501 502 503 504 505 506
      if (remove_x)
	{
	  // If simplification are enabled, we do them before and after.
	  if (simplification_level)
	    {
	      const spot::ltl::formula* res = simpl.simplify(f);
	      f->destroy();
	      f = res;
	    }

	  const spot::ltl::formula* res = spot::ltl::remove_x(f);
	  f->destroy();
	  f = res;
	}

507
      if (simplification_level || boolean_to_isop)
508 509 510 511 512 513 514 515 516 517 518 519 520
	{
	  const spot::ltl::formula* res = simpl.simplify(f);
	  f->destroy();
	  f = res;
	}

      if (nnf)
	{
	  const spot::ltl::formula* res = simpl.negative_normal_form(f);
	  f->destroy();
	  f = res;
	}

521
      switch (relabeling)
522
	{
523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538
	case ApRelabeling:
	  {
	    const spot::ltl::formula* res = spot::ltl::relabel(f, style);
	    f->destroy();
	    f = res;
	    break;
	  }
	case BseRelabeling:
	  {
	    const spot::ltl::formula* res = spot::ltl::relabel_bse(f, style);
	    f->destroy();
	    f = res;
	    break;
	  }
	case NoRelabeling:
	  break;
539 540 541 542 543
	}

      if (remove_wm)
	{
	  const spot::ltl::formula* res = spot::ltl::unabbreviate_wm(f);
544 545 546 547
	  f->destroy();
	  f = res;
	}

548 549 550 551 552 553 554 555 556 557 558 559
      bool matched = true;

      matched &= !ltl || f->is_ltl_formula();
      matched &= !psl || f->is_psl_formula();
      matched &= !boolean || f->is_boolean();
      matched &= !universal || f->is_universal();
      matched &= !eventual || f->is_eventual();
      matched &= !syntactic_safety || f->is_syntactic_safety();
      matched &= !syntactic_guarantee || f->is_syntactic_guarantee();
      matched &= !syntactic_obligation || f->is_syntactic_obligation();
      matched &= !syntactic_recurrence || f->is_syntactic_recurrence();
      matched &= !syntactic_persistence || f->is_syntactic_persistence();
560
      matched &= !syntactic_si || f->is_syntactic_stutter_invariant();
561
      matched &= !ap || atomic_prop_collect(f)->size() == ap_n;
562 563 564 565 566 567 568 569 570 571 572 573 574 575 576

      if (matched && (size_min > 0 || size_max >= 0))
	{
	  int l = spot::ltl::length(f);
	  matched &= (size_min <= 0) || (l >= size_min);
	  matched &= (size_max < 0) || (l <= size_max);
	}

      if (matched && (bsize_min > 0 || bsize_max >= 0))
	{
	  int l = spot::ltl::length_boolone(f);
	  matched &= (bsize_min <= 0) || (l >= bsize_min);
	  matched &= (bsize_max < 0) || (l <= bsize_max);
	}

577 578 579
      matched &= !implied_by || simpl.implication(implied_by, f);
      matched &= !imply || simpl.implication(f, imply);
      matched &= !equivalent_to || simpl.are_equivalent(f, equivalent_to);
580
      matched &= !stutter_insensitive || spot::is_stutter_invariant(f);
581

582 583 584 585 586 587
      // Match obligations and subclasses using WDBA minimization.
      // Because this is costly, we compute it later, so that we don't
      // have to compute it on formulas that have been discarded for
      // other reasons.
      if (matched && obligation)
	{
588 589
	  auto aut = ltl_to_tgba_fm(f, simpl.get_dict());
	  auto min = minimize_obligation(aut, f);
590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614
	  assert(min);
	  if (aut == min)
	    {
	      // Not an obligation
	      matched = false;
	    }
	  else
	    {
	      matched &= !guarantee || is_guarantee_automaton(min);
	      matched &= !safety || is_safety_mwdba(min);
	    }
	}

      matched ^= invert;

      if (unique)
	{
	  if (unique_set.insert(f).second)
	    f->clone();
	  else
	    matched = false;
	}

      if (matched)
	{
615
	  one_match = true;
616
	  output_formula_checked(f, filename, linenum, prefix, suffix);
617
	  ++match_count;
618 619 620 621 622 623 624 625 626 627
	}
      f->destroy();
      return 0;
    }
  };
}

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

630
  const argp ap = { options, parse_opt, "[FILENAME[/COL]...]",
631
		    argp_program_doc, children, 0, 0 };
632

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

  if (jobs.empty())
637
    jobs.emplace_back("-", 1);
638

639 640
  if (boolean_to_isop && simplification_level == 0)
    simplification_level = 1;
641
  spot::ltl::ltl_simplifier_options opt(simplification_level);
642 643
  opt.boolean_to_isop = boolean_to_isop;
  spot::ltl::ltl_simplifier simpl(opt);
644 645 646 647 648 649 650 651 652 653 654 655 656 657
  try
    {
      ltl_processor processor(simpl);
      if (processor.run())
	return 2;
    }
  catch (const std::runtime_error& e)
    {
      error(2, 0, "%s", e.what());
    }

  if (output_format == count_output)
    std::cout << match_count << std::endl;

658
  return one_match ? 0 : 1;
659
}