From 2bc18cd725632e1dadc3c143a111cdbe3d357d49 Mon Sep 17 00:00:00 2001 From: Koen Hillewaert <koen.hillewaert@cenaero.be> Date: Mon, 6 Sep 2010 17:00:32 +0000 Subject: [PATCH] allow for weighted partitioning with Metis --- Common/DefaultOptions.h | 13 ++++++ Common/Options.cpp | 42 ++++++++++++++++++ Common/Options.h | 6 +++ Fltk/partitionDialog.cpp | 85 ++++++++++++++++++++++++++++++++++++- Mesh/meshPartition.cpp | 27 +++++++----- Mesh/meshPartitionObjects.h | 35 +++++++++++++++ Mesh/meshPartitionOptions.h | 16 +++++++ 7 files changed, 211 insertions(+), 13 deletions(-) diff --git a/Common/DefaultOptions.h b/Common/DefaultOptions.h index f188b5deb4..05e96a1aed 100644 --- a/Common/DefaultOptions.h +++ b/Common/DefaultOptions.h @@ -1138,6 +1138,19 @@ StringXNumber MeshOptions_Number[] = { "Version of the MSH file format to use" }, { F|O, "MshFilePartitioned" , opt_mesh_msh_file_partitioned , 0. , "Split MSH file by mesh partition" }, + + { F|O, "PartitionHexWeight" , opt_mesh_partition_hex_weight , 1 , + "Weight of hexahedral element for METIS load balancing" }, + { F|O, "PartitionPrismWeight" , opt_mesh_partition_pri_weight , 1 , + "Weight of prismatic element (wedge) for METIS load balancing" }, + { F|O, "PartitionPyramidWeight" , opt_mesh_partition_pyr_weight , 1 , + "Weight of pyramidal element for METIS load balancing" }, + { F|O, "PartitionQuadWeight" , opt_mesh_partition_qua_weight , 1 , + "Weight of quadrangle for METIS load balancing" }, + { F|O, "PartitionTetWeight" , opt_mesh_partition_tet_weight , 1 , + "Weight of tetrahedral element for METIS load balancing" }, + { F|O, "PartitionTriWeight" , opt_mesh_partition_tri_weight , 1 , + "Weight of triangle for METIS load balancing" ,}, { F|O, "PartitionByExtrusion" , opt_mesh_partition_by_extrusion, 0. , "Special partitioner that annotates all all extruded elements to the same node as the source element" }, diff --git a/Common/Options.cpp b/Common/Options.cpp index 399fafa5f3..5667a5eac0 100644 --- a/Common/Options.cpp +++ b/Common/Options.cpp @@ -5507,6 +5507,48 @@ double opt_mesh_msh_file_partitioned(OPT_ARGS_NUM) return CTX::instance()->mesh.mshFilePartitioned; } +double opt_mesh_partition_hex_weight(OPT_ARGS_NUM) +{ + if (action & GMSH_SET) + CTX::instance()->partitionOptions.hexWeight = (int) val; + return CTX::instance()->partitionOptions.hexWeight; +} + +double opt_mesh_partition_pri_weight(OPT_ARGS_NUM) +{ + if (action & GMSH_SET) + CTX::instance()->partitionOptions.priWeight = (int) val; + return CTX::instance()->partitionOptions.priWeight; +} + +double opt_mesh_partition_pyr_weight(OPT_ARGS_NUM) +{ + if (action & GMSH_SET) + CTX::instance()->partitionOptions.pyrWeight = (int) val; + return CTX::instance()->partitionOptions.pyrWeight; +} + +double opt_mesh_partition_qua_weight(OPT_ARGS_NUM) +{ + if (action & GMSH_SET) + CTX::instance()->partitionOptions.quaWeight = (int) val; + return CTX::instance()->partitionOptions.quaWeight; +} + +double opt_mesh_partition_tet_weight(OPT_ARGS_NUM) +{ + if (action & GMSH_SET) + CTX::instance()->partitionOptions.tetWeight = (int) val; + return CTX::instance()->partitionOptions.tetWeight; +} + +double opt_mesh_partition_tri_weight(OPT_ARGS_NUM) +{ + if (action & GMSH_SET) + CTX::instance()->partitionOptions.triWeight = (int) val; + return CTX::instance()->partitionOptions.triWeight; +} + double opt_mesh_binary(OPT_ARGS_NUM) { if(action & GMSH_SET) diff --git a/Common/Options.h b/Common/Options.h index 46e5dd19e2..799568de88 100644 --- a/Common/Options.h +++ b/Common/Options.h @@ -513,6 +513,12 @@ double opt_mesh_light_two_side(OPT_ARGS_NUM); double opt_mesh_file_format(OPT_ARGS_NUM); double opt_mesh_msh_file_version(OPT_ARGS_NUM); double opt_mesh_msh_file_partitioned(OPT_ARGS_NUM); +double opt_mesh_partition_hex_weight(OPT_ARGS_NUM); +double opt_mesh_partition_pri_weight(OPT_ARGS_NUM); +double opt_mesh_partition_pyr_weight(OPT_ARGS_NUM); +double opt_mesh_partition_qua_weight(OPT_ARGS_NUM); +double opt_mesh_partition_tet_weight(OPT_ARGS_NUM); +double opt_mesh_partition_tri_weight(OPT_ARGS_NUM); double opt_mesh_binary(OPT_ARGS_NUM); double opt_mesh_bdf_field_format(OPT_ARGS_NUM); double opt_mesh_nb_smoothing(OPT_ARGS_NUM); diff --git a/Fltk/partitionDialog.cpp b/Fltk/partitionDialog.cpp index 015d7a8014..ffd16814ca 100644 --- a/Fltk/partitionDialog.cpp +++ b/Fltk/partitionDialog.cpp @@ -69,6 +69,14 @@ struct PartitionDialog // Group 4 Fl_Choice *choiceEdgeMatch; Fl_Choice *choiceRefineAlg; + // weights + Fl_Value_Input *inputTriWeight; + Fl_Value_Input *inputQuaWeight; + Fl_Value_Input *inputTetWeight; + Fl_Value_Input *inputPriWeight; + Fl_Value_Input *inputPyrWeight; + Fl_Value_Input *inputHexWeight; + void write_all_options() { // Group 0 @@ -115,6 +123,14 @@ struct PartitionDialog // Group 4 CTX::instance()->partitionOptions.edge_matching = choiceEdgeMatch->value() + 1; CTX::instance()->partitionOptions.refine_algorithm = choiceRefineAlg->value() + 1; + + CTX::instance()->partitionOptions.triWeight = inputTriWeight->value(); + CTX::instance()->partitionOptions.quaWeight = inputQuaWeight->value(); + CTX::instance()->partitionOptions.tetWeight = inputTetWeight->value(); + CTX::instance()->partitionOptions.priWeight = inputPriWeight->value(); + CTX::instance()->partitionOptions.pyrWeight = inputPyrWeight->value(); + CTX::instance()->partitionOptions.hexWeight = inputHexWeight->value(); + } void read_all_options() { @@ -154,7 +170,14 @@ struct PartitionDialog // Group 4 choiceEdgeMatch->value(CTX::instance()->partitionOptions.edge_matching - 1); choiceRefineAlg->value(CTX::instance()->partitionOptions.refine_algorithm - 1); - + + inputTriWeight->value(CTX::instance()->partitionOptions.triWeight); + inputQuaWeight->value(CTX::instance()->partitionOptions.quaWeight); + inputTetWeight->value(CTX::instance()->partitionOptions.tetWeight); + inputPriWeight->value(CTX::instance()->partitionOptions.priWeight); + inputPyrWeight->value(CTX::instance()->partitionOptions.pyrWeight); + inputHexWeight->value(CTX::instance()->partitionOptions.hexWeight); + // Call all callbacks to ensure consistent options partition_opt_chaco_globalalg_cb(choiceChacoAlg, this); partition_opt_architecture_cb(choiceArchitecture, this); @@ -707,7 +730,7 @@ void partition_dialog() // Metis advanced option group [4] { - const int GH = 2 + WB + (BH + WB) + 2; + const int GH = 2 + WB + 3*(BH + WB) + 2; Fl_Group *g = new Fl_Group(0, y, w, GH); // Box (line) { @@ -732,6 +755,64 @@ void partition_dialog() o->align(FL_ALIGN_RIGHT); } y += BH + WB + 1; // +1 for multiline label + // element weights - line 1 + { + Fl_Value_Input *const o = new Fl_Value_Input + (WB, y, 2*BB/3, BH, "Triangle"); + dlg.inputTriWeight = o; + o->minimum(1); + o->maximum(std::numeric_limits<int>::max()); + o->step(1); + o->align(FL_ALIGN_RIGHT); + } + { + Fl_Value_Input *const o = new Fl_Value_Input + (2*WB + (w/3-WB), y,2*BB/3, BH, "Tetrahedron"); + dlg.inputTetWeight = o; + o->minimum(1); + o->maximum(std::numeric_limits<int>::max()); + o->step(1); + o->align(FL_ALIGN_RIGHT); + } + { + Fl_Value_Input *const o = new Fl_Value_Input + (3*WB + 2*(w/3-WB), y,2*BB/3, BH, "Prism"); + dlg.inputPriWeight = o; + o->minimum(1); + o->maximum(std::numeric_limits<int>::max()); + o->step(1); + o->align(FL_ALIGN_RIGHT); + } + y += 2 + WB + BH + 1; + // element weights - line 2 + { + Fl_Value_Input *const o = new Fl_Value_Input + (WB, y, 2*BB/3, BH, "Quadrangle"); + dlg.inputQuaWeight = o; + o->minimum(1); + o->maximum(std::numeric_limits<int>::max()); + o->step(1); + o->align(FL_ALIGN_RIGHT); + } + { + Fl_Value_Input *const o = new Fl_Value_Input + (2*WB + (w/3-WB), y,2*BB/3, BH, "Hexahedron"); + dlg.inputHexWeight = o; + o->minimum(1); + o->maximum(std::numeric_limits<int>::max()); + o->step(1); + o->align(FL_ALIGN_RIGHT); + } + { + Fl_Value_Input *const o = new Fl_Value_Input + (3*WB + 2*(w/3-WB), y,2*BB/3, BH, "Pyramid"); + dlg.inputPyrWeight = o; + o->minimum(1); + o->maximum(std::numeric_limits<int>::max()); + o->step(1); + o->align(FL_ALIGN_RIGHT); + } + y += BH + WB + 1; // +1 for multiline label g->end(); g->hide(); } diff --git a/Mesh/meshPartition.cpp b/Mesh/meshPartition.cpp index a9c4899741..b336104087 100644 --- a/Mesh/meshPartition.cpp +++ b/Mesh/meshPartition.cpp @@ -451,11 +451,13 @@ int PartitionGraph(Graph &graph, meshPartitionOptions &options) metisOptions[2] = 1; metisOptions[3] = options.refine_algorithm; metisOptions[4] = 0; - METIS_PartGraphKway - (&n, &graph.xadj[graph.section[iSec]], - &graph.adjncy[graph.section[iSec]], NULL, NULL, &wgtflag, &numflag, - &options.num_partitions, metisOptions, &edgeCut, - &graph.partition[graph.section[iSec]]); + if (options.num_partitions > 1) { // partgraphkway aborts with floating point error if np = 1 + METIS_PartGraphKway + (&n, &graph.xadj[graph.section[iSec]], + &graph.adjncy[graph.section[iSec]], NULL, NULL, &wgtflag, &numflag, + &options.num_partitions, metisOptions, &edgeCut, + &graph.partition[graph.section[iSec]]); + } break; case 3: // Nodal weight printf("METIS with weights\n"); @@ -465,12 +467,15 @@ int PartitionGraph(Graph &graph, meshPartitionOptions &options) metisOptions[3] = 1; metisOptions[4] = 0; wgtflag = 2; - graph.fillWeights(options.nodalWeights); - METIS_PartGraphKway - (&n, &graph.xadj[graph.section[iSec]], - &graph.adjncy[graph.section[iSec]], &graph.vwgts[graph.section[iSec]], NULL, &wgtflag, &numflag, - &options.num_partitions, metisOptions, &edgeCut, - &graph.partition[graph.section[iSec]]); + graph.fillDefaultWeights(); + // graph.fillWeights(options.nodalWeights); + if (options.num_partitions > 1) { // partgraphkway aborts with floating point error if np = 1 + METIS_PartGraphKway + (&n, &graph.xadj[graph.section[iSec]], + &graph.adjncy[graph.section[iSec]], &graph.vwgts[graph.section[iSec]], NULL, &wgtflag, &numflag, + &options.num_partitions, metisOptions, &edgeCut, + &graph.partition[graph.section[iSec]]); + } break; } } diff --git a/Mesh/meshPartitionObjects.h b/Mesh/meshPartitionObjects.h index b721ac1e27..134cb58658 100644 --- a/Mesh/meshPartitionObjects.h +++ b/Mesh/meshPartitionObjects.h @@ -10,6 +10,7 @@ #include <vector> #include "MElement.h" #include "GmshMessage.h" +#include "Context.h" /******************************************************************************* @@ -156,6 +157,40 @@ class Graph } } + // Add weights per element, as defined in options + void fillDefaultWeights() + { + std::vector<MElement*>::iterator eIt = element.begin(); + vwgts.resize(element.size()); + std::vector<int>::iterator wIt = vwgts.begin(); + for ( ; eIt != element.end() ; eIt++ , wIt++) { + + switch ((*eIt)->getType()) { + case TYPE_TRI: + *wIt = CTX::instance()->partitionOptions.triWeight; + break; + case TYPE_QUA: + *wIt = CTX::instance()->partitionOptions.quaWeight; + break; + case TYPE_TET: + *wIt = CTX::instance()->partitionOptions.tetWeight; + break; + case TYPE_PYR: + *wIt = CTX::instance()->partitionOptions.pyrWeight; + break; + case TYPE_PRI: + *wIt = CTX::instance()->partitionOptions.priWeight; + break; + case TYPE_HEX: + *wIt = CTX::instance()->partitionOptions.hexWeight; + break; + default: + *wIt = 1; + break; + } + } + } + void markSection() { section.push_back(numGrVert); } // Returns the next index for a graph vertex int getNextIndex() { return cIndex++; } diff --git a/Mesh/meshPartitionOptions.h b/Mesh/meshPartitionOptions.h index 2867ab086c..56aa905f54 100644 --- a/Mesh/meshPartitionOptions.h +++ b/Mesh/meshPartitionOptions.h @@ -72,6 +72,15 @@ struct meshPartitionOptions int partitionByExtrusion; // if true, all extruded elements belong // to the same partition as the source element + // element weights for load-balancing (currently used in METIS algorithm 3) + + int triWeight; + int quaWeight; + int tetWeight; + int priWeight; + int pyrWeight; + int hexWeight; + //--NODAL WEIGHT std::vector<int> nodalWeights; @@ -112,6 +121,13 @@ struct meshPartitionOptions createPartitionBoundaries = true; createGhostCells = true; partitionByExtrusion =false; + triWeight = 1; + quaWeight = 1; + tetWeight = 1; + priWeight = 1; + pyrWeight = 1; + hexWeight = 1; + } }; -- GitLab