src/Pure/General/graph.scala
author wenzelm
Thu, 23 Feb 2012 19:35:05 +0100
changeset 46623 bce24d3f29e7
parent 46615 c29bf6741ace
child 46659 b257053a4cbe
permissions -rw-r--r--
tuned -- avoid copy of empty value;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
46611
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
     1
/*  Title:      Pure/General/graph.scala
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
     2
    Module:     PIDE
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
     3
    Author:     Makarius
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
     4
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
     5
Directed graphs.
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
     6
*/
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
     7
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
     8
package isabelle
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
     9
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    10
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    11
import scala.annotation.tailrec
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    12
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    13
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    14
object Graph
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    15
{
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    16
  class Duplicate[Key](x: Key) extends Exception
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    17
  class Undefined[Key](x: Key) extends Exception
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    18
  class Cycles[Key](cycles: List[List[Key]]) extends Exception
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    19
46623
bce24d3f29e7 tuned -- avoid copy of empty value;
wenzelm
parents: 46615
diff changeset
    20
  private val empty_val: Graph[Any, Nothing] = new Graph[Any, Nothing](Map.empty)
bce24d3f29e7 tuned -- avoid copy of empty value;
wenzelm
parents: 46615
diff changeset
    21
  def empty[Key, A]: Graph[Key, A] = empty_val.asInstanceOf[Graph[Key, A]]
46611
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    22
}
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    23
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    24
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    25
class Graph[Key, A] private(rep: Map[Key, (A, (Set[Key], Set[Key]))])
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    26
  extends Iterable[(Key, (A, (Set[Key], Set[Key])))]
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    27
{
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    28
  type Keys = Set[Key]
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    29
  type Entry = (A, (Keys, Keys))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    30
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    31
  def iterator: Iterator[(Key, Entry)] = rep.iterator
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    32
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    33
  def is_empty: Boolean = rep.isEmpty
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    34
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    35
  def keys: Set[Key] = rep.keySet.toSet
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    36
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    37
  def dest: List[(Key, List[Key])] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    38
    (for ((x, (_, (_, succs))) <- iterator) yield (x, succs.toList)).toList
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    39
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    40
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    41
  /* entries */
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    42
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    43
  private def get_entry(x: Key): Entry =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    44
    rep.get(x) match {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    45
      case Some(entry) => entry
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    46
      case None => throw new Graph.Undefined(x)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    47
    }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    48
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    49
  private def map_entry(x: Key, f: Entry => Entry): Graph[Key, A] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    50
    new Graph[Key, A](rep + (x -> f(get_entry(x))))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    51
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    52
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    53
  /* nodes */
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    54
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    55
  def get_node(x: Key): A = get_entry(x)._1
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    56
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    57
  def map_node(x: Key, f: A => A): Graph[Key, A] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    58
    map_entry(x, { case (i, ps) => (f(i), ps) })
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    59
46615
wenzelm
parents: 46614
diff changeset
    60
  def map_nodes[B](f: A => B): Graph[Key, B] =
wenzelm
parents: 46614
diff changeset
    61
    new Graph[Key, B](rep mapValues { case (i, ps) => (f(i), ps) })
wenzelm
parents: 46614
diff changeset
    62
46611
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    63
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    64
  /* reachability */
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    65
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    66
  /*nodes reachable from xs -- topologically sorted for acyclic graphs*/
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    67
  def reachable(next: Key => Keys, xs: List[Key]): (List[List[Key]], Keys) =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    68
  {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    69
    def reach(reached: (List[Key], Keys), x: Key): (List[Key], Keys) =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    70
    {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    71
      val (rs, r_set) = reached
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    72
      if (r_set(x)) reached
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    73
      else {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    74
        val (rs1, r_set1) = ((rs, r_set + x) /: next(x))(reach)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    75
        (x :: rs1, r_set1)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    76
      }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    77
    }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    78
    def reachs(reached: (List[List[Key]], Keys), x: Key): (List[List[Key]], Keys) =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    79
    {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    80
      val (rss, r_set) = reached
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    81
      val (rs, r_set1) = reach((Nil, r_set), x)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    82
      (rs :: rss, r_set1)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    83
    }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    84
    ((List.empty[List[Key]], Set.empty[Key]) /: xs)(reachs)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    85
  }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    86
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    87
  /*immediate*/
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    88
  def imm_preds(x: Key): Keys = get_entry(x)._2._1
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    89
  def imm_succs(x: Key): Keys = get_entry(x)._2._2
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    90
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    91
  /*transitive*/
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    92
  def all_preds(xs: List[Key]): List[Key] = reachable(imm_preds, xs)._1.flatten
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    93
  def all_succs(xs: List[Key]): List[Key] = reachable(imm_succs, xs)._1.flatten
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
    94
46613
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
    95
  /*strongly connected components; see: David King and John Launchbury,
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
    96
    "Structuring Depth First Search Algorithms in Haskell"*/
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
    97
  def strong_conn: List[List[Key]] =
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
    98
    reachable(imm_preds, all_succs(keys.toList))._1.filterNot(_.isEmpty).reverse
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
    99
46611
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   100
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   101
  /* minimal and maximal elements */
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   102
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   103
  def minimals: List[Key] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   104
    (List.empty[Key] /: rep) {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   105
      case (ms, (m, (_, (preds, _)))) => if (preds.isEmpty) m :: ms else ms }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   106
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   107
  def maximals: List[Key] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   108
    (List.empty[Key] /: rep) {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   109
      case (ms, (m, (_, (_, succs)))) => if (succs.isEmpty) m :: ms else ms }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   110
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   111
  def is_minimal(x: Key): Boolean = imm_preds(x).isEmpty
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   112
  def is_maximal(x: Key): Boolean = imm_succs(x).isEmpty
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   113
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   114
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   115
  /* nodes */
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   116
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   117
  def new_node(x: Key, info: A): Graph[Key, A] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   118
  {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   119
    if (rep.isDefinedAt(x)) throw new Graph.Duplicate(x)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   120
    else new Graph[Key, A](rep + (x -> (info, (Set.empty, Set.empty))))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   121
  }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   122
46613
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
   123
  def default_node(x: Key, info: A): Graph[Key, A] =
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
   124
  {
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
   125
    if (rep.isDefinedAt(x)) this
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
   126
    else new_node(x, info)
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
   127
  }
74331911375d further graph operations from ML;
wenzelm
parents: 46611
diff changeset
   128
46611
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   129
  def del_nodes(xs: List[Key]): Graph[Key, A] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   130
  {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   131
    xs.foreach(get_entry)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   132
    new Graph[Key, A](
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   133
      (rep -- xs) mapValues { case (i, (preds, succs)) => (i, (preds -- xs, succs -- xs)) })
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   134
  }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   135
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   136
  private def del_adjacent(fst: Boolean, x: Key)(map: Map[Key, Entry], y: Key): Map[Key, Entry] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   137
    map.get(y) match {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   138
      case None => map
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   139
      case Some((i, (preds, succs))) =>
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   140
        map + (y -> (i, if (fst) (preds - x, succs) else (preds, succs - x)))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   141
    }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   142
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   143
  def del_node(x: Key): Graph[Key, A] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   144
  {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   145
    val (preds, succs) = get_entry(x)._2
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   146
    new Graph[Key, A](
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   147
      (((rep - x) /: preds)(del_adjacent(false, x)) /: succs)(del_adjacent(true, x)))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   148
  }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   149
46614
165886a4fe64 clarified Graph.restrict (formerly Graph.subgraph) based on public graph operations;
wenzelm
parents: 46613
diff changeset
   150
  def restrict(pred: Key => Boolean): Graph[Key, A] =
165886a4fe64 clarified Graph.restrict (formerly Graph.subgraph) based on public graph operations;
wenzelm
parents: 46613
diff changeset
   151
    (this /: iterator){ case (graph, (x, _)) => if (!pred(x)) graph.del_node(x) else graph }
165886a4fe64 clarified Graph.restrict (formerly Graph.subgraph) based on public graph operations;
wenzelm
parents: 46613
diff changeset
   152
46611
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   153
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   154
  /* edges */
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   155
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   156
  def is_edge(x: Key, y: Key): Boolean =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   157
    try { imm_succs(x)(y) }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   158
    catch { case _: Graph.Undefined[_] => false }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   159
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   160
  def add_edge(x: Key, y: Key): Graph[Key, A] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   161
    if (is_edge(x, y)) this
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   162
    else
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   163
      map_entry(y, { case (i, (preds, succs)) => (i, (preds + x, succs)) }).
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   164
      map_entry(x, { case (i, (preds, succs)) => (i, (preds, succs + y)) })
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   165
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   166
  def del_edge(x: Key, y: Key): Graph[Key, A] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   167
    if (is_edge(x, y))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   168
      map_entry(y, { case (i, (preds, succs)) => (i, (preds - x, succs)) }).
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   169
      map_entry(x, { case (i, (preds, succs)) => (i, (preds, succs - y)) })
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   170
    else this
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   171
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   172
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   173
  /* irreducible paths -- Hasse diagram */
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   174
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   175
  def irreducible_preds(x_set: Set[Key], path: List[Key], z: Key): List[Key] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   176
  {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   177
    def red(x: Key)(x1: Key) = is_edge(x, x1) && x1 != z
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   178
    @tailrec def irreds(xs0: List[Key], xs1: List[Key]): List[Key] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   179
      xs0 match {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   180
        case Nil => xs1
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   181
        case x :: xs =>
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   182
          if (!(x_set(x)) || x == z || path.contains(x) ||
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   183
              xs.exists(red(x)) || xs1.exists(red(x)))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   184
            irreds(xs, xs1)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   185
          else irreds(xs, x :: xs1)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   186
      }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   187
    irreds(imm_preds(z).toList, Nil)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   188
  }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   189
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   190
  def irreducible_paths(x: Key, y: Key): List[List[Key]] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   191
  {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   192
    val (_, x_set) = reachable(imm_succs, List(x))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   193
    def paths(path: List[Key])(ps: List[List[Key]], z: Key): List[List[Key]] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   194
      if (x == z) (z :: path) :: ps
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   195
      else (ps /: irreducible_preds(x_set, path, z))(paths(z :: path))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   196
    if ((x == y) && !is_edge(x, x)) List(Nil) else paths(Nil)(Nil, y)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   197
  }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   198
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   199
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   200
  /* maintain acyclic graphs */
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   201
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   202
  def add_edge_acyclic(x: Key, y: Key): Graph[Key, A] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   203
    if (is_edge(x, y)) this
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   204
    else {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   205
      irreducible_paths(y, x) match {
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   206
        case Nil => add_edge(x, y)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   207
        case cycles => throw new Graph.Cycles(cycles.map(x :: _))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   208
      }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   209
    }
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   210
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   211
  def add_deps_cyclic(y: Key, xs: List[Key]): Graph[Key, A] =
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   212
    (this /: xs)(_.add_edge_acyclic(_, y))
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   213
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   214
  def topological_order: List[Key] = all_succs(minimals)
669601fa1a62 directed graphs (in Scala);
wenzelm
parents:
diff changeset
   215
}