2023-09-09 20:41:49 +09:00
|
|
|
//-----------------------------------------------------------------------------------
|
|
|
|
// Implementation of a MPS algorithm via PC-tree.
|
|
|
|
//-----------------------------------------------------------------------------------
|
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
#include <chrono>
|
2023-09-09 20:41:49 +09:00
|
|
|
#include "mps.h"
|
2024-03-03 22:28:42 +09:00
|
|
|
|
2023-09-09 21:29:57 +09:00
|
|
|
#include <ogdf/fileformats/GraphIO.h>
|
2023-09-09 20:41:49 +09:00
|
|
|
|
2024-02-23 11:36:40 +09:00
|
|
|
// #define DEBUG
|
2024-02-20 17:00:51 +09:00
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
#define TIME
|
|
|
|
|
2023-09-09 20:41:49 +09:00
|
|
|
//-----------------------------------------------------------------------------------
|
|
|
|
// Finding MPS
|
|
|
|
//-----------------------------------------------------------------------------------
|
2023-09-11 12:21:02 +09:00
|
|
|
|
|
|
|
// programs to call from main:
|
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
// read input file of gml format
|
|
|
|
ogdf::Graph read_from_gml(string input_file) {
|
|
|
|
ogdf::Graph G;
|
|
|
|
|
|
|
|
// utilize OGDF readGML
|
|
|
|
if (!ogdf::GraphIO::read(G, input_file, ogdf::GraphIO::readGML)) {
|
|
|
|
std::cerr << "Could not read " << input_file << ".gml" << std::endl;
|
|
|
|
}
|
|
|
|
return G;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int find_mps(const ogdf::Graph &G) {
|
2023-09-09 20:41:49 +09:00
|
|
|
maximal_planar_subgraph_finder m;
|
2024-03-03 22:28:42 +09:00
|
|
|
return m.find_mps(G);
|
2023-09-09 20:41:49 +09:00
|
|
|
}
|
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
int compute_removed_edge_size(const ogdf::Graph &G, vector<int> post_order) {
|
2023-09-10 13:48:33 +09:00
|
|
|
maximal_planar_subgraph_finder m;
|
2024-03-03 22:28:42 +09:00
|
|
|
return m.compute_removed_edge_size(G, post_order);
|
2023-09-10 13:48:33 +09:00
|
|
|
}
|
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
vector<int> generate_post_order(const ogdf::Graph &G) {
|
2023-09-11 12:21:02 +09:00
|
|
|
maximal_planar_subgraph_finder m;
|
2024-03-03 22:28:42 +09:00
|
|
|
return m.generate_post_order(G);
|
2023-09-11 12:21:02 +09:00
|
|
|
}
|
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
vector<int> generate_mutated_post_order(const ogdf::Graph &G, vector<int> post_order, int mutate_point) {
|
2023-09-11 12:21:02 +09:00
|
|
|
maximal_planar_subgraph_finder m;
|
2024-03-03 22:28:42 +09:00
|
|
|
return m.generate_mutated_post_order(G, post_order, mutate_point);
|
2023-09-11 12:21:02 +09:00
|
|
|
}
|
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
vector<int> generate_guided_post_order(const ogdf::Graph &G, vector<int> post_order) {
|
2024-02-22 10:52:36 +09:00
|
|
|
maximal_planar_subgraph_finder m;
|
2024-03-03 22:28:42 +09:00
|
|
|
return m.generate_guided_post_order(G, post_order);
|
2024-02-22 10:52:36 +09:00
|
|
|
}
|
|
|
|
|
2023-09-11 12:21:02 +09:00
|
|
|
|
|
|
|
// ---------
|
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
int maximal_planar_subgraph_finder::find_mps(const ogdf::Graph &G) {
|
|
|
|
init_from_graph(G);
|
2023-09-09 20:41:49 +09:00
|
|
|
postOrderTraversal();
|
2024-02-20 17:00:51 +09:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
print_post_order();
|
|
|
|
#endif
|
|
|
|
|
2023-09-09 20:41:49 +09:00
|
|
|
sort_adj_list();
|
|
|
|
determine_edges();
|
|
|
|
back_edge_traversal();
|
2023-09-09 21:29:57 +09:00
|
|
|
return output_removed_edge_size();
|
2023-09-09 20:41:49 +09:00
|
|
|
}
|
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
vector<int> maximal_planar_subgraph_finder::generate_post_order(const ogdf::Graph &G) {
|
|
|
|
init_from_graph(G);
|
2024-02-20 17:00:51 +09:00
|
|
|
postOrderTraversal();
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
2024-02-22 10:52:36 +09:00
|
|
|
std::cout << "standard post order traversal" << std::endl;
|
2024-02-20 17:00:51 +09:00
|
|
|
print_post_order();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return return_post_order();
|
2023-09-11 12:21:02 +09:00
|
|
|
}
|
|
|
|
|
2024-02-20 17:00:51 +09:00
|
|
|
// result of this will be used as input to "compute_removed_edge_size"
|
2024-03-03 22:28:42 +09:00
|
|
|
vector<int> maximal_planar_subgraph_finder::generate_mutated_post_order(const ogdf::Graph &G, vector<int> post_order, int mutate_point) {
|
|
|
|
init_from_graph(G);
|
2024-02-23 11:36:40 +09:00
|
|
|
mutatedPostOrderTraversal(post_order, mutate_point);
|
2024-02-20 17:00:51 +09:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
2024-02-22 10:52:36 +09:00
|
|
|
std::cout << "mutated post order traversal" << std::endl;
|
|
|
|
print_post_order();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return return_post_order();
|
|
|
|
}
|
|
|
|
|
|
|
|
// result of this will be used as input to "compute_removed_edge_size"
|
2024-03-03 22:28:42 +09:00
|
|
|
vector<int> maximal_planar_subgraph_finder::generate_guided_post_order(const ogdf::Graph &G, vector<int> post_order) {
|
|
|
|
init_from_graph(G);
|
2024-02-22 10:52:36 +09:00
|
|
|
guidedPostOrderTraversal(post_order);
|
|
|
|
|
2024-02-23 11:36:40 +09:00
|
|
|
// #ifdef DEBUG
|
|
|
|
// std::cout << "guided post order traversal" << std::endl;
|
|
|
|
// print_post_order();
|
|
|
|
// #endif
|
2024-02-20 17:00:51 +09:00
|
|
|
|
|
|
|
return return_post_order();
|
2023-09-11 12:21:02 +09:00
|
|
|
}
|
|
|
|
|
2023-09-10 13:48:33 +09:00
|
|
|
|
2024-02-22 10:52:36 +09:00
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
int maximal_planar_subgraph_finder::compute_removed_edge_size(const ogdf::Graph &G, vector<int> post_order) {
|
|
|
|
// read_from_gml
|
|
|
|
#ifdef TIME
|
|
|
|
auto start = std::chrono::high_resolution_clock::now();
|
|
|
|
#endif
|
|
|
|
init_from_graph(G);
|
|
|
|
#ifdef TIME
|
|
|
|
auto end = std::chrono::high_resolution_clock::now();
|
|
|
|
std::cout << "init from G: " << std::chrono::duration_cast<std::chrono::microseconds>(end - start).count() << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// guidedPostOrderTraversal
|
|
|
|
#ifdef TIME
|
|
|
|
start = std::chrono::high_resolution_clock::now();
|
|
|
|
#endif
|
2023-09-11 12:21:02 +09:00
|
|
|
guidedPostOrderTraversal(post_order);
|
2024-03-03 22:28:42 +09:00
|
|
|
#ifdef TIME
|
|
|
|
end = std::chrono::high_resolution_clock::now();
|
|
|
|
std::cout << "guidedPostOrderTraversal: " << std::chrono::duration_cast<std::chrono::microseconds>(end - start).count() << std::endl;
|
2024-02-20 17:00:51 +09:00
|
|
|
#endif
|
2023-09-11 12:21:02 +09:00
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
// remaining procedure
|
|
|
|
#ifdef TIME
|
|
|
|
start = std::chrono::high_resolution_clock::now();
|
|
|
|
#endif
|
2023-09-10 13:48:33 +09:00
|
|
|
sort_adj_list();
|
|
|
|
determine_edges();
|
|
|
|
back_edge_traversal();
|
2024-03-03 22:28:42 +09:00
|
|
|
#ifdef TIME
|
|
|
|
end = std::chrono::high_resolution_clock::now();
|
|
|
|
std::cout << "remaining procedures: " << std::chrono::duration_cast<std::chrono::microseconds>(end - start).count() << std::endl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
2023-09-10 13:48:33 +09:00
|
|
|
return output_removed_edge_size();
|
|
|
|
}
|
|
|
|
|
2023-09-09 20:41:49 +09:00
|
|
|
//-----------------------------------------------------------------------------------
|
2024-03-03 22:28:42 +09:00
|
|
|
// Input, output
|
2023-09-09 20:41:49 +09:00
|
|
|
//-----------------------------------------------------------------------------------
|
2023-09-09 21:29:57 +09:00
|
|
|
|
2024-03-03 22:28:42 +09:00
|
|
|
void maximal_planar_subgraph_finder::init_from_graph(const ogdf::Graph &G) {
|
2023-09-09 21:29:57 +09:00
|
|
|
// create nodes
|
|
|
|
for (int i = 0; i < G.numberOfNodes(); ++i) {
|
2023-09-09 20:41:49 +09:00
|
|
|
_node_list.push_back(new node(P_NODE));
|
|
|
|
_node_list[i]->set_id(i);
|
|
|
|
}
|
|
|
|
|
2023-09-09 21:29:57 +09:00
|
|
|
// create edges
|
|
|
|
for (ogdf::edge e : G.edges) {
|
|
|
|
ogdf::node source = e->source();
|
|
|
|
ogdf::node target = e->target();
|
|
|
|
_node_list[source->index()]->add_adj(_node_list[target->index()]);
|
|
|
|
_node_list[target->index()]->add_adj(_node_list[source->index()]);
|
|
|
|
}
|
2023-09-09 20:41:49 +09:00
|
|
|
}
|
|
|
|
|
2023-09-09 21:29:57 +09:00
|
|
|
|
|
|
|
|
2024-02-05 11:02:44 +09:00
|
|
|
// count the number of removed edges
|
2023-09-09 21:29:57 +09:00
|
|
|
int maximal_planar_subgraph_finder::output_removed_edge_size() {
|
|
|
|
int sum = 0;
|
|
|
|
for (int i = 0; i < _back_edge_list.size(); ++i) {
|
|
|
|
if (_is_back_edge_eliminate[i]) ++sum;
|
|
|
|
}
|
|
|
|
return sum;
|
|
|
|
}
|