/* * This file is part of libFirm. * Copyright (C) 2012 Karlsruhe Institute of Technology. */ #include #include #include "adt/array.h" #include "bucket.h" #include "heuristical_co_ld.h" #include "optimal.h" #if KAPS_DUMP #include "html_dumper.h" #endif #include "kaps.h" #include "matrix.h" #include "pbqp_edge.h" #include "pbqp_edge_t.h" #include "pbqp_node.h" #include "pbqp_node_t.h" #include "vector.h" #include "pdeq.h" #include "timing.h" static void back_propagate_RI(pbqp_t *pbqp, pbqp_node_t *node) { (void)pbqp; pbqp_edge_t *edge = node->edges[0]; pbqp_matrix_t *mat = edge->costs; bool is_src = edge->src == node; vector_t *vec = node->costs; if (is_src) { pbqp_node_t *other = edge->tgt; node->solution = pbqp_matrix_get_col_min_index(mat, other->solution, vec); } else { pbqp_node_t *other = edge->src; node->solution = pbqp_matrix_get_row_min_index(mat, other->solution, vec); } #if KAPS_DUMP if (pbqp->dump_file) { fprintf(pbqp->dump_file, "node n%u is set to %u
\n", node->index, node->solution); } #endif } static void back_propagate_RII(pbqp_t *pbqp, pbqp_node_t *node) { pbqp_edge_t *src_edge = node->edges[0]; pbqp_edge_t *tgt_edge = node->edges[1]; bool src_is_src = src_edge->src == node; bool tgt_is_src = tgt_edge->src == node; pbqp_node_t *src_node; pbqp_node_t *tgt_node; if (src_is_src) { src_node = src_edge->tgt; } else { src_node = src_edge->src; } if (tgt_is_src) { tgt_node = tgt_edge->tgt; } else { tgt_node = tgt_edge->src; } /* Swap nodes if necessary. */ if (tgt_node->index < src_node->index) { pbqp_node_t *tmp_node = src_node; src_node = tgt_node; tgt_node = tmp_node; pbqp_edge_t *tmp_edge = src_edge; src_edge = tgt_edge; tgt_edge = tmp_edge; src_is_src = src_edge->src == node; tgt_is_src = tgt_edge->src == node; } pbqp_matrix_t *src_mat = src_edge->costs; vector_t *node_vec = node->costs; unsigned row_index = src_node->solution; vector_t *vec = vector_copy(pbqp, node_vec); if (src_is_src) { vector_add_matrix_col(vec, src_mat, row_index); } else { vector_add_matrix_row(vec, src_mat, row_index); } pbqp_matrix_t *tgt_mat = tgt_edge->costs; unsigned col_index = tgt_node->solution; if (tgt_is_src) { vector_add_matrix_col(vec, tgt_mat, col_index); } else { vector_add_matrix_row(vec, tgt_mat, col_index); } node->solution = vector_get_min_index(vec); #if KAPS_DUMP if (pbqp->dump_file) { fprintf(pbqp->dump_file, "node n%u is set to %u
\n", node->index, node->solution); } #endif obstack_free(&pbqp->obstack, vec); } static void back_propagate_RN(pbqp_t *pbqp, pbqp_node_t *node) { vector_t *vec = vector_copy(pbqp, node->costs); for (unsigned edge_index = 0; edge_index < pbqp_node_get_degree(node); edge_index++) { /* get neighbor node */ pbqp_edge_t *edge = node->edges[edge_index]; pbqp_node_t *neighbor = edge->src == node ? edge->tgt : edge->src; /* node is edge src node */ if (edge->src == node) vector_add_matrix_col(vec, edge->costs, neighbor->solution); /* node is edge tgt node */ else vector_add_matrix_row(vec, edge->costs, neighbor->solution); } assert(vector_get_min(vec) != INF_COSTS); node->solution = vector_get_min_index(vec); #if KAPS_DUMP if (pbqp->dump_file) { fprintf(pbqp->dump_file, "node n%u is set to %u
\n", node->index, node->solution); } #endif obstack_free(&pbqp->obstack, vec); } static void back_propagate_ld(pbqp_t *pbqp) { #if KAPS_DUMP if (pbqp->dump_file) { pbqp_dump_section(pbqp->dump_file, 2, "Back Propagation"); } #endif unsigned node_len = node_bucket_get_length(reduced_bucket); for (unsigned node_index = node_len; node_index-- != 0;) { pbqp_node_t *node = reduced_bucket[node_index]; switch (pbqp_node_get_degree(node)) { case 1: back_propagate_RI(pbqp, node); break; case 2: back_propagate_RII(pbqp, node); break; default: back_propagate_RN(pbqp, node); break; } } } static void merge_into_RN_node(pbqp_t *pbqp, deq_t *rpeo) { pbqp_node_t *node; do { /* get last element from reverse perfect elimination order */ node = deq_pop_pointer_right(pbqp_node_t, rpeo); /* insert node at the beginning of rpeo so the rpeo already exits after * pbqp solving */ deq_push_pointer_left(rpeo, node); } while (node_is_reduced(node)); assert(pbqp_node_get_degree(node) > 2); /* Check whether we can merge a neighbor into the current node. */ apply_RM(pbqp, node); } static void apply_RN_co_without_selection(pbqp_t *pbqp) { (void)pbqp; pbqp_node_t *node = merged_node; merged_node = NULL; if (node_is_reduced(node)) return; #if KAPS_DUMP if (pbqp->dump_file) { char txt[100]; sprintf(txt, "RN-Reduction of Node n%u", node->index); pbqp_dump_section(pbqp->dump_file, 2, txt); pbqp_dump_graph(pbqp); } #endif /* Disconnect neighbor nodes */ for (unsigned edge_index = 0; edge_index < pbqp_node_get_degree(node); edge_index++) { pbqp_edge_t *edge; pbqp_node_t *neighbor; /* get neighbor node */ edge = node->edges[edge_index]; neighbor = edge->src == node ? edge->tgt : edge->src; assert(neighbor != node); if (!is_connected(neighbor, edge)) continue; disconnect_edge(neighbor, edge); reorder_node_after_edge_deletion(neighbor); } /* Remove node from old bucket */ node_bucket_remove(&node_buckets[3], node); /* Add node to back propagation list. */ node_bucket_insert(&reduced_bucket, node); } static void apply_heuristic_reductions_co(pbqp_t *pbqp, deq_t *rpeo) { #if KAPS_TIMING /* create timers */ ir_timer_t *t_edge = ir_timer_new(); ir_timer_t *t_r1 = ir_timer_new(); ir_timer_t *t_r2 = ir_timer_new(); ir_timer_t *t_rn = ir_timer_new(); #endif for (;;) { if (edge_bucket_get_length(edge_bucket) > 0) { #if KAPS_TIMING ir_timer_start(t_edge); #endif apply_edge(pbqp); #if KAPS_TIMING ir_timer_stop(t_edge); #endif } else if (node_bucket_get_length(node_buckets[1]) > 0) { #if KAPS_TIMING ir_timer_start(t_r1); #endif apply_RI(pbqp); #if KAPS_TIMING ir_timer_stop(t_r1); #endif } else if (node_bucket_get_length(node_buckets[2]) > 0) { #if KAPS_TIMING ir_timer_start(t_r2); #endif apply_RII(pbqp); #if KAPS_TIMING ir_timer_stop(t_r2); #endif } else if (merged_node != NULL) { #if KAPS_TIMING ir_timer_start(t_rn); #endif apply_RN_co_without_selection(pbqp); #if KAPS_TIMING ir_timer_stop(t_rn); #endif } else if (node_bucket_get_length(node_buckets[3]) > 0) { #if KAPS_TIMING ir_timer_start(t_rn); #endif merge_into_RN_node(pbqp, rpeo); #if KAPS_TIMING ir_timer_stop(t_rn); #endif } else { #if KAPS_TIMING printf("PBQP RE reductions: %10.3lf msec\n", (double)ir_timer_elapsed_usec(t_edge) / 1000.0); printf("PBQP R1 reductions: %10.3lf msec\n", (double)ir_timer_elapsed_usec(t_r1) / 1000.0); printf("PBQP R2 reductions: %10.3lf msec\n", (double)ir_timer_elapsed_usec(t_r2) / 1000.0); printf("PBQP RN reductions: %10.3lf msec\n", (double)ir_timer_elapsed_usec(t_rn) / 1000.0); #endif return; } } } void solve_pbqp_heuristical_co_ld(pbqp_t *pbqp, deq_t *rpeo) { #ifndef NDEBUG assert(pbqp); assert(rpeo); assert(pbqp->solution == INF_COSTS && "PBQP already solved"); pbqp->solution = 0; #endif /* Reduce nodes degree ... */ initial_simplify_edges(pbqp); /* ... and put node into bucket representing their degree. */ fill_node_buckets(pbqp); #if KAPS_STATISTIC FILE *fh = fopen("solutions.pb", "a"); fprintf(fh, "Solution"); fclose(fh); #endif apply_heuristic_reductions_co(pbqp, rpeo); pbqp->solution = determine_solution(pbqp); #if KAPS_STATISTIC fh = fopen("solutions.pb", "a"); #if KAPS_USE_UNSIGNED fprintf(fh, ": %u RE:%u R0:%u R1:%u R2:%u RM:%u RN/BF:%u\n", pbqp->solution, pbqp->num_edges, pbqp->num_r0, pbqp->num_r1, pbqp->num_r2, pbqp->num_rm, pbqp->num_rn); #else fprintf(fh, ": %lld RE:%u R0:%u R1:%u R2:%u RM:%u RN/BF:%u\n", pbqp->solution, pbqp->num_edges, pbqp->num_r0, pbqp->num_r1, pbqp->num_r2, pbqp->num_rm, pbqp->num_rn); #endif fclose(fh); #endif /* Solve reduced nodes. */ back_propagate_ld(pbqp); free_buckets(); }