/* * This file is part of libFirm. * Copyright (C) 2012 University of Karlsruhe. */ /** * @file * @brief Brute force PBQP solver. * @date 02.12.2008 * @author Sebastian Buchwald */ #include "brute_force.h" #include "bucket.h" #include "kaps.h" #include "matrix.h" #include "optimal.h" #include "panic.h" #include "pbqp_edge.h" #include "pbqp_edge_t.h" #include "pbqp_node.h" #include "pbqp_node_t.h" #include "vector.h" #include #include #if KAPS_DUMP #include "html_dumper.h" #endif #if KAPS_STATISTIC static int dump = 0; #endif /* Forward declarations. */ static void apply_Brute_Force(pbqp_t *pbqp); static void apply_brute_force_reductions(pbqp_t *pbqp) { for (;;) { if (edge_bucket_get_length(edge_bucket) > 0) { apply_edge(pbqp); } else if (node_bucket_get_length(node_buckets[1]) > 0) { apply_RI(pbqp); } else if (node_bucket_get_length(node_buckets[2]) > 0) { apply_RII(pbqp); } else if (node_bucket_get_length(node_buckets[3]) > 0) { apply_Brute_Force(pbqp); } else { return; } } } static unsigned get_minimal_alternative(pbqp_t *pbqp, pbqp_node_t *node) { vector_t *node_vec = node->costs; unsigned node_len = node_vec->len; unsigned min_index = 0; num min = INF_COSTS; unsigned bucket_index = node->bucket_index; for (unsigned node_index = 0; node_index < node_len; ++node_index) { pbqp_node_bucket_t bucket_deg3; num value; unsigned bucket_0_length; unsigned bucket_red_length; char *tmp = (char *)obstack_finish(&pbqp->obstack); node_bucket_init(&bucket_deg3); /* Some node buckets and the edge bucket should be empty. */ assert(node_bucket_get_length(node_buckets[1]) == 0); assert(node_bucket_get_length(node_buckets[2]) == 0); assert(edge_bucket_get_length(edge_bucket) == 0); /* char *tmp = obstack_finish(&pbqp->obstack); */ /* Save current PBQP state. */ node_bucket_copy(&bucket_deg3, node_buckets[3]); node_bucket_shrink(&node_buckets[3], 0); node_bucket_deep_copy(pbqp, &node_buckets[3], bucket_deg3); node_bucket_update(pbqp, node_buckets[3]); bucket_0_length = node_bucket_get_length(node_buckets[0]); bucket_red_length = node_bucket_get_length(reduced_bucket); /* Select alternative and solve PBQP recursively. */ select_alternative(node_buckets[3][bucket_index], node_index); apply_brute_force_reductions(pbqp); value = determine_solution(pbqp); if (value < min) { min = value; min_index = node_index; } /* Some node buckets and the edge bucket should still be empty. */ assert(node_bucket_get_length(node_buckets[1]) == 0); assert(node_bucket_get_length(node_buckets[2]) == 0); assert(edge_bucket_get_length(edge_bucket) == 0); /* Clear modified buckets... */ node_bucket_shrink(&node_buckets[3], 0); /* ... and restore old PBQP state. */ node_bucket_shrink(&node_buckets[0], bucket_0_length); node_bucket_shrink(&reduced_bucket, bucket_red_length); node_bucket_copy(&node_buckets[3], bucket_deg3); node_bucket_update(pbqp, node_buckets[3]); /* Free copies. */ /* obstack_free(&pbqp->obstack, tmp); */ node_bucket_free(&bucket_deg3); obstack_free(&pbqp->obstack, tmp); } return min_index; } static void apply_Brute_Force(pbqp_t *pbqp) { /* We want to reduce a node with maximum degree. */ pbqp_node_t *node = get_node_with_max_degree(); assert(pbqp_node_get_degree(node) > 2); #if KAPS_DUMP if (pbqp->dump_file) { char txt[100]; sprintf(txt, "BF-Reduction of Node n%u", node->index); pbqp_dump_section(pbqp->dump_file, 2, txt); pbqp_dump_graph(pbqp); } #endif #if KAPS_STATISTIC dump++; #endif unsigned min_index = get_minimal_alternative(pbqp, node); node = pbqp->nodes[node->index]; #if KAPS_DUMP if (pbqp->dump_file) { fprintf(pbqp->dump_file, "node n%u is set to %u

\n", node->index, min_index); } #endif #if KAPS_STATISTIC dump--; if (dump == 0) { FILE *fh = fopen("solutions.pb", "a"); fprintf(fh, "[%u]", min_index); fclose(fh); pbqp->num_bf++; } #endif /* Now that we found the minimum set all other costs to infinity. */ select_alternative(node, min_index); } static void back_propagate_RI(pbqp_t *pbqp, pbqp_node_t *node) { pbqp_edge_t *edge = node->edges[0]; pbqp_node_t *other; pbqp_matrix_t *mat = edge->costs; vector_t *vec = node->costs; bool is_src = edge->src == node; if (is_src) { other = edge->tgt; /* Update pointer for brute force solver. */ other = pbqp->nodes[other->index]; node->solution = pbqp_matrix_get_col_min_index(mat, other->solution, vec); } else { other = edge->src; /* Update pointer for brute force solver. */ other = pbqp->nodes[other->index]; 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; } /* Update pointer for brute force solver. */ src_node = pbqp->nodes[src_node->index]; tgt_node = pbqp->nodes[tgt_node->index]; pbqp_matrix_t *src_mat = src_edge->costs; pbqp_matrix_t *tgt_mat = tgt_edge->costs; vector_t *node_vec = node->costs; unsigned col_index = tgt_node->solution; 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); } 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_brute_force(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: panic("only nodes with degree one or two should be in this bucket"); } } } void solve_pbqp_brute_force(pbqp_t *pbqp) { #ifndef NDEBUG assert(pbqp); 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_brute_force_reductions(pbqp); 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_bf); #endif fclose(fh); #endif /* Solve reduced nodes. */ back_propagate_brute_force(pbqp); free_buckets(); }