src/Pure/General/graph.ML
author wenzelm
Mon, 17 May 1999 21:31:08 +0200
changeset 6659 7a056250899d
parent 6152 bc1e27bcc195
child 8806 a202293db3f6
permissions -rw-r--r--
removed get_nodes; added keys; all_preds / all_succs: topological order; added del_nodes;
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
     1
(*  Title:      Pure/General/graph.ML
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
     2
    ID:         $Id$
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
     3
    Author:     Markus Wenzel, TU Muenchen
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
     4
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
     5
Directed graphs.
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
     6
*)
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
     7
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
     8
signature GRAPH =
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
     9
sig
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    10
  type key
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    11
  type 'a T
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    12
  exception UNDEFINED of key
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    13
  val empty: 'a T
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
    14
  val keys: 'a T -> key list
6142
wenzelm
parents: 6134
diff changeset
    15
  val map_nodes: ('a -> 'b) -> 'a T -> 'b T
wenzelm
parents: 6134
diff changeset
    16
  val get_node: 'a T -> key -> 'a
wenzelm
parents: 6134
diff changeset
    17
  val map_node: key -> ('a -> 'a) -> 'a T -> 'a T
wenzelm
parents: 6134
diff changeset
    18
  val imm_preds: 'a T -> key -> key list
wenzelm
parents: 6134
diff changeset
    19
  val imm_succs: 'a T -> key -> key list
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    20
  val all_preds: 'a T -> key list -> key list
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    21
  val all_succs: 'a T -> key list -> key list
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    22
  val find_paths: 'a T -> key * key -> key list list
6142
wenzelm
parents: 6134
diff changeset
    23
  exception DUPLICATE of key
6152
wenzelm
parents: 6142
diff changeset
    24
  val new_node: key * 'a -> 'a T -> 'a T
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
    25
  val del_nodes: key list -> 'a T -> 'a T
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    26
  val add_edge: key * key -> 'a T -> 'a T
6152
wenzelm
parents: 6142
diff changeset
    27
  val del_edge: key * key -> 'a T -> 'a T
6142
wenzelm
parents: 6134
diff changeset
    28
  exception CYCLES of key list list
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    29
  val add_edge_acyclic: key * key -> 'a T -> 'a T
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    30
end;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    31
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    32
functor GraphFun(Key: KEY): GRAPH =
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    33
struct
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    34
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    35
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    36
(* keys *)
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    37
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    38
type key = Key.key;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    39
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    40
val eq_key = equal EQUAL o Key.ord;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    41
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    42
infix mem_key;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    43
val op mem_key = gen_mem eq_key;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    44
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    45
infix ins_key;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    46
val op ins_key = gen_ins eq_key;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    47
6152
wenzelm
parents: 6142
diff changeset
    48
infix del_key;
wenzelm
parents: 6142
diff changeset
    49
fun xs del_key x = if x mem_key xs then gen_rem eq_key (xs, x) else xs;
wenzelm
parents: 6142
diff changeset
    50
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
    51
infix del_keys;
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
    52
val op del_keys = foldl (op del_key);
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
    53
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    54
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    55
(* tables and sets of keys *)
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    56
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    57
structure Table = TableFun(Key);
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    58
type keys = unit Table.table;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    59
6142
wenzelm
parents: 6134
diff changeset
    60
val empty_keys = Table.empty: keys;
wenzelm
parents: 6134
diff changeset
    61
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    62
infix mem_keys;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    63
fun x mem_keys tab = is_some (Table.lookup (tab: keys, x));
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    64
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    65
infix ins_keys;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    66
fun x ins_keys tab = if x mem_keys tab then tab else Table.update ((x, ()), tab);
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    67
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    68
6142
wenzelm
parents: 6134
diff changeset
    69
(* graphs *)
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    70
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    71
datatype 'a T = Graph of ('a * (key list * key list)) Table.table;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    72
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    73
exception UNDEFINED of key;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    74
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    75
val empty = Graph Table.empty;
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
    76
fun keys (Graph tab) = Table.keys tab;
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    77
6142
wenzelm
parents: 6134
diff changeset
    78
fun get_entry (Graph tab) x =
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    79
  (case Table.lookup (tab, x) of
6142
wenzelm
parents: 6134
diff changeset
    80
    Some entry => entry
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    81
  | None => raise UNDEFINED x);
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    82
6142
wenzelm
parents: 6134
diff changeset
    83
fun map_entry x f (G as Graph tab) = Graph (Table.update ((x, f (get_entry G x)), tab));
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    84
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    85
6142
wenzelm
parents: 6134
diff changeset
    86
(* nodes *)
wenzelm
parents: 6134
diff changeset
    87
wenzelm
parents: 6134
diff changeset
    88
fun map_nodes f (Graph tab) = Graph (Table.map (fn (i, ps) => (f i, ps)) tab);
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    89
6142
wenzelm
parents: 6134
diff changeset
    90
fun get_node G = #1 o get_entry G;
wenzelm
parents: 6134
diff changeset
    91
fun map_node x f = map_entry x (fn (i, ps) => (f i, ps));
wenzelm
parents: 6134
diff changeset
    92
wenzelm
parents: 6134
diff changeset
    93
wenzelm
parents: 6134
diff changeset
    94
(* reachability *)
wenzelm
parents: 6134
diff changeset
    95
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
    96
(*nodes reachable from xs -- topologically sorted for acyclic graphs*)
6142
wenzelm
parents: 6134
diff changeset
    97
fun reachable next xs =
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
    98
  let
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
    99
    fun reach ((rs, R), x) =
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   100
      if x mem_keys R then (rs, R)
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   101
      else apfst (cons x) (reachs ((rs, x ins_keys R), next x))
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   102
    and reachs R_xs = foldl reach R_xs;
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   103
  in reachs (([], empty_keys), xs) end;
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   104
6142
wenzelm
parents: 6134
diff changeset
   105
(*immediate*)
wenzelm
parents: 6134
diff changeset
   106
fun imm_preds G = #1 o #2 o get_entry G;
wenzelm
parents: 6134
diff changeset
   107
fun imm_succs G = #2 o #2 o get_entry G;
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   108
6142
wenzelm
parents: 6134
diff changeset
   109
(*transitive*)
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   110
fun all_preds G = #1 o reachable (imm_preds G);
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   111
fun all_succs G = #1 o reachable (imm_succs G);
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   112
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   113
6142
wenzelm
parents: 6134
diff changeset
   114
(* paths *)
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   115
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   116
fun find_paths G (x, y) =
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   117
  let
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   118
    val (_, X) = reachable (imm_succs G) [x];
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   119
    fun paths ps p =
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   120
      if eq_key (p, x) then [p :: ps]
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   121
      else flat (map (paths (p :: ps))
6142
wenzelm
parents: 6134
diff changeset
   122
        (filter (fn pp => pp mem_keys X andalso not (pp mem_key ps)) (imm_preds G p)));
wenzelm
parents: 6134
diff changeset
   123
  in get_entry G y; if y mem_keys X then (paths [] y) else [] end;
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   124
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   125
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   126
(* build graphs *)
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   127
6142
wenzelm
parents: 6134
diff changeset
   128
exception DUPLICATE of key;
wenzelm
parents: 6134
diff changeset
   129
6152
wenzelm
parents: 6142
diff changeset
   130
fun new_node (x, info) (Graph tab) =
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   131
  Graph (Table.update_new ((x, (info, ([], []))), tab)
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   132
    handle Table.DUP key => raise DUPLICATE key);
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   133
6659
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   134
fun del_nodes xs (Graph tab) =
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   135
  let
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   136
    fun del (x, (i, (preds, succs))) =
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   137
      if x mem_key xs then None
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   138
      else Some (x, (i, (preds del_keys xs, succs del_keys xs)));
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   139
  in Graph (Table.make (mapfilter del (Table.dest tab))) end;
7a056250899d removed get_nodes;
wenzelm
parents: 6152
diff changeset
   140
6152
wenzelm
parents: 6142
diff changeset
   141
wenzelm
parents: 6142
diff changeset
   142
fun add_edge (x, y) =
wenzelm
parents: 6142
diff changeset
   143
  map_entry x (fn (i, (preds, succs)) => (i, (preds, y ins_key succs))) o
wenzelm
parents: 6142
diff changeset
   144
   map_entry y (fn (i, (preds, succs)) => (i, (x ins_key preds, succs)));
wenzelm
parents: 6142
diff changeset
   145
wenzelm
parents: 6142
diff changeset
   146
fun del_edge (x, y) =
wenzelm
parents: 6142
diff changeset
   147
  map_entry x (fn (i, (preds, succs)) => (i, (preds, succs del_key y))) o
wenzelm
parents: 6142
diff changeset
   148
   map_entry y (fn (i, (preds, succs)) => (i, (preds del_key x, succs)));
wenzelm
parents: 6142
diff changeset
   149
6142
wenzelm
parents: 6134
diff changeset
   150
wenzelm
parents: 6134
diff changeset
   151
exception CYCLES of key list list;
6134
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   152
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   153
fun add_edge_acyclic (x, y) G =
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   154
  (case find_paths G (y, x) of
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   155
    [] => add_edge (x, y) G
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   156
  | cycles => raise CYCLES (map (cons x) cycles));
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   157
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   158
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   159
end;
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   160
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   161
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   162
(*graphs indexed by strings*)
ec6092b0599d added General/graph.ML: generic direct graphs;
wenzelm
parents:
diff changeset
   163
structure Graph = GraphFun(type key = string val ord = string_ord);