//----------------------------------------------------------------------------------- // Header for modules: mps.cpp, mps_test.cpp, node.cpp. //----------------------------------------------------------------------------------- #ifndef _MPS_H #define _MPS_H #include #include #include #include #include #include #include #include #include #include #include using namespace std; class node; class maximal_planar_subgraph_finder; enum label { NOT_VISITED = 0, ARTIFICIAL_EDGE = 1, BOUNDARY_PATH = 2, DELETED = 3 }; enum node_type { P_NODE = 0, C_NODE = 1, REPLICA_NODE = 2, AE_VIRTUAL_ROOT = 3 }; class node { public: //CONSTRUCTOR node(node_type t); //DESTRUCTOR ~node() {} //TYPE, ID, INDEX node_type type(); int post_order_index(); void set_id(int i); void set_post_order_index(int i); void recursively_labeling(); int node_id(); //DFS-TREE void add_adj(node* n); int degree(); node* adj(int i); void set_adj_list(vector vec); void DFS_visit(vector &dfsList, int &index); void guided_DFS_visit(vector &dfsList, const vector &node_list, int &return_index, const unordered_map &node_id_to_pos); void mutated_DFS_visit(vector &dfsList, const vector &node_list, int &index, int &traversal_index, const unordered_map &node_id_to_pos, int mutate_point, mt19937 rng); // custom comparator function to sort nodes according to order in given vector bool sortByOrder(const unordered_map& node_id_to_pos, node* a, node* b); //PARENT-CHILDREN void set_parent(node* n) ; node* parent(); int child_num(); node* child(int i); void add_child(node* n); void clear_children(); void remove_child(int i); void remove_child(node* n); vector* get_children_list(); //BOUNDARY_PATH void set_to_boundary_path(node* n0, node* n1); void set_neighbor(int i, node* n); void set_neighbor(node* u, node* v); node* neighbor(int i); node* get_next(node* prev); //ARTIFICIAL EDGE node* AE(int i); void set_AE(int i, node* j); void add_AE(node* j); void inherit_AE(node* u); void init_AE(node* u); //REPLICA node* original_node(); node* get_c_node(); void set_c_node(node* c); bool is_sentinel(); static bool is_same(node* n1, node* n2); void init_replica(node* u, node* c); //LABELING void set_1st_label(int i); void set_2nd_label(label i); int get_1st_label(); label get_2nd_label(); //C-NODE node* get_a_list_node(); int c_node_size(); node* essential(int i); void clear_essential(); void add_essential(node* u); //MARK void mark(); static void init_mark(); void un_mark(); bool is_marked(); private: //Basic information. node_type _type; pair _label; //Information about neighborhood. node* _neighbor[2]; node* _AE_root[2]; //Information about higher hierarchy. node* _original_node; node* _c_node; //Information about parent-children relation. node* _parent; vector _children; //Information about about p-nodes in DFS-tree vector _adj_list; int _post_order_index; int _node_id; //List of essential nodes in c-node vector _essential_list; //Mark int _mark; static int _ref_mark; }; class maximal_planar_subgraph_finder { public: maximal_planar_subgraph_finder(); ~maximal_planar_subgraph_finder(); // functions that prepare state void init_from_graph(const ogdf::Graph &G); vector generate_post_order(const ogdf::Graph &G); vector generate_mutated_post_order(const ogdf::Graph &G, const vector &post_order, int mutate_point); vector generate_guided_post_order(const ogdf::Graph &G, const vector &post_order); void postOrderTraversal(); void guidedPostOrderTraversal(const vector &post_order); void mutatedPostOrderTraversal(const vector &post_order, int mutate_point); // compute_mps combines functionality to reduce repeating object initialization // the results are returned by modifying mutable reference void compute_mps(const ogdf::Graph &G, int mutate_point, vector &post_order, int &return_edge_size); int find_mps(const ogdf::Graph &G); int compute_removed_edge_size(const ogdf::Graph &G, vector post_order); node* get_new_node(node_type t); void reset_state(); int output_removed_edge_size(); vector return_post_order(); // void set_post_order(vector post_order); void print_post_order(); void sort_adj_list(); void determine_edges(); void back_edge_traversal(); bool back_edge_traversal(node* traverse_node, int index); void make_essential(node* p_node, node* c_node); node* find(node* n); void merge(pair, pair > boundary, node* list_node); void eliminate(node* u); void eliminate_AE(node* u, node* v); pair, pair > trim(node* u); void c_node_extension(node* c_node); void recursively_shaving(node* u); pair shave(node* x); pair parallel_search_sentinel(node* x, node* &c); pair parallel_search_sentinel(node* n0, node* n0_prev, node* n1, node* n1_prev, node* & c); pair count_sentinel_elimination(pair sentinel_1, int num_sentinel); node* construct(node* u); node* construct(node* c, node* p); void parenting_labeling_shaving(node* u, node* node_i) ; private: vector _node_list; //List of nodes input. vector > _edge_list; // Edges in DFS-tree. These edges must be contained in the maximal planar subgraph that we found. vector _post_order_list; //The sorted version (increasing with post-order-index) of _node_list. vector > _back_edge_list; // Edges other than that in DFS-tree. (The first node's index is higher than the second's.) vector _is_back_edge_eliminate; //Record that if the back-edge has been eliminated or not. vector _new_node_list; //Newly added nodes. }; #endif // for MPS_H