⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 parmetis.c

📁 一个用来实现偏微分方程中网格的计算库
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * Copyright 1997, Regents of the University of Minnesota * * parmetis.c * * This file contains top level routines that are used by ParMETIS * * Started 10/14/97 * George * * $Id: parmetis.c 2501 2007-11-20 02:33:29Z benkirk $ * */#include <metis.h>/************************************************************************** This function is the entry point for KMETIS with seed specification* in options[7] **************************************************************************/void METIS_PartGraphKway2(int *nvtxs, idxtype *xadj, idxtype *adjncy, idxtype *vwgt,                          idxtype *adjwgt, int *wgtflag, int *numflag, int *nparts,                          int *options, int *edgecut, idxtype *part){  int i;  float *tpwgts;  tpwgts = fmalloc(*nparts, "KMETIS: tpwgts");  for (i=0; i<*nparts; i++)     tpwgts[i] = 1.0/(1.0*(*nparts));  METIS_WPartGraphKway2(nvtxs, xadj, adjncy, vwgt, adjwgt, wgtflag, numflag, nparts,                        tpwgts, options, edgecut, part);  free(tpwgts);}/************************************************************************** This function is the entry point for KWMETIS with seed specification* in options[7] **************************************************************************/void METIS_WPartGraphKway2(int *nvtxs, idxtype *xadj, idxtype *adjncy, idxtype *vwgt,                           idxtype *adjwgt, int *wgtflag, int *numflag, int *nparts,                           float *tpwgts, int *options, int *edgecut, idxtype *part){  int i, j;  GraphType graph;  CtrlType ctrl;  if (*numflag == 1)    Change2CNumbering(*nvtxs, xadj, adjncy);  SetUpGraph(&graph, OP_KMETIS, *nvtxs, 1, xadj, adjncy, vwgt, adjwgt, *wgtflag);  if (options[0] == 0) {  /* Use the default parameters */    ctrl.CType = KMETIS_CTYPE;    ctrl.IType = KMETIS_ITYPE;    ctrl.RType = KMETIS_RTYPE;    ctrl.dbglvl = KMETIS_DBGLVL;  }  else {    ctrl.CType = options[OPTION_CTYPE];    ctrl.IType = options[OPTION_ITYPE];    ctrl.RType = options[OPTION_RTYPE];    ctrl.dbglvl = options[OPTION_DBGLVL];  }  ctrl.optype = OP_KMETIS;  ctrl.CoarsenTo = 20*(*nparts);  ctrl.maxvwgt = 1.5*((graph.vwgt ? idxsum(*nvtxs, graph.vwgt) : (*nvtxs))/ctrl.CoarsenTo);  InitRandom(options[7]);  AllocateWorkSpace(&ctrl, &graph, *nparts);  IFSET(ctrl.dbglvl, DBG_TIME, InitTimers(&ctrl));  IFSET(ctrl.dbglvl, DBG_TIME, starttimer(ctrl.TotalTmr));  *edgecut = MlevelKWayPartitioning(&ctrl, &graph, *nparts, part, tpwgts, 1.000);  IFSET(ctrl.dbglvl, DBG_TIME, stoptimer(ctrl.TotalTmr));  IFSET(ctrl.dbglvl, DBG_TIME, PrintTimers(&ctrl));  FreeWorkSpace(&ctrl, &graph);  if (*numflag == 1)    Change2FNumbering(*nvtxs, xadj, adjncy, part);}/************************************************************************** This function is the entry point for the node ND code for ParMETIS**************************************************************************/void METIS_NodeNDP(int nvtxs, idxtype *xadj, idxtype *adjncy, int npes,                    int *options, idxtype *perm, idxtype *iperm, idxtype *sizes) {  int i, ii, j, l, wflag, nflag;  GraphType graph;  CtrlType ctrl;  idxtype *cptr, *cind;  if (options[0] == 0) {  /* Use the default parameters */    ctrl.CType   = ONMETIS_CTYPE;    ctrl.IType   = ONMETIS_ITYPE;    ctrl.RType   = ONMETIS_RTYPE;    ctrl.dbglvl  = ONMETIS_DBGLVL;    ctrl.oflags  = ONMETIS_OFLAGS;    ctrl.pfactor = ONMETIS_PFACTOR;    ctrl.nseps   = ONMETIS_NSEPS;  }  else {    ctrl.CType   = options[OPTION_CTYPE];    ctrl.IType   = options[OPTION_ITYPE];    ctrl.RType   = options[OPTION_RTYPE];    ctrl.dbglvl  = options[OPTION_DBGLVL];    ctrl.oflags  = options[OPTION_OFLAGS];    ctrl.pfactor = options[OPTION_PFACTOR];    ctrl.nseps   = options[OPTION_NSEPS];  }  if (ctrl.nseps < 1)    ctrl.nseps = 1;  ctrl.optype = OP_ONMETIS;  ctrl.CoarsenTo = 100;  IFSET(ctrl.dbglvl, DBG_TIME, InitTimers(&ctrl));  IFSET(ctrl.dbglvl, DBG_TIME, starttimer(ctrl.TotalTmr));  InitRandom(-1);  if (ctrl.oflags&OFLAG_COMPRESS) {    /*============================================================    * Compress the graph     ==============================================================*/    cptr = idxmalloc(nvtxs+1, "ONMETIS: cptr");    cind = idxmalloc(nvtxs, "ONMETIS: cind");    CompressGraph(&ctrl, &graph, nvtxs, xadj, adjncy, cptr, cind);    if (graph.nvtxs >= COMPRESSION_FRACTION*(nvtxs)) {      ctrl.oflags--; /* We actually performed no compression */      GKfree(&cptr, &cind, LTERM);    }    else if (2*graph.nvtxs < nvtxs && ctrl.nseps == 1)      ctrl.nseps = 2;  }  else {    SetUpGraph(&graph, OP_ONMETIS, nvtxs, 1, xadj, adjncy, NULL, NULL, 0);  }  /*=============================================================  * Do the nested dissection ordering   --=============================================================*/  ctrl.maxvwgt = 1.5*(idxsum(graph.nvtxs, graph.vwgt)/ctrl.CoarsenTo);  AllocateWorkSpace(&ctrl, &graph, 2);  idxset(2*npes-1, 0, sizes);  MlevelNestedDissectionP(&ctrl, &graph, iperm, graph.nvtxs, npes, 0, sizes);  FreeWorkSpace(&ctrl, &graph);  if (ctrl.oflags&OFLAG_COMPRESS) { /* Uncompress the ordering */    if (graph.nvtxs < COMPRESSION_FRACTION*(nvtxs)) {       /* construct perm from iperm */      for (i=0; i<graph.nvtxs; i++)        perm[iperm[i]] = i;       for (l=ii=0; ii<graph.nvtxs; ii++) {        i = perm[ii];        for (j=cptr[i]; j<cptr[i+1]; j++)          iperm[cind[j]] = l++;      }    }    GKfree(&cptr, &cind, LTERM);  }  for (i=0; i<nvtxs; i++)    perm[iperm[i]] = i;  IFSET(ctrl.dbglvl, DBG_TIME, stoptimer(ctrl.TotalTmr));  IFSET(ctrl.dbglvl, DBG_TIME, PrintTimers(&ctrl));}/************************************************************************** This function takes a graph and produces a bisection of it**************************************************************************/void MlevelNestedDissectionP(CtrlType *ctrl, GraphType *graph, idxtype *order, int lastvtx,                              int npes, int cpos, idxtype *sizes){  int i, j, nvtxs, nbnd, tvwgt, tpwgts2[2];  idxtype *label, *bndind;  GraphType lgraph, rgraph;  float ubfactor;  nvtxs = graph->nvtxs;  if (nvtxs == 0) {    GKfree(&graph->gdata, &graph->rdata, &graph->label, LTERM);    return;  }  /* Determine the weights of the partitions */  tvwgt = idxsum(nvtxs, graph->vwgt);  tpwgts2[0] = tvwgt/2;  tpwgts2[1] = tvwgt-tpwgts2[0];  if (cpos >= npes-1)     ubfactor = ORDER_UNBALANCE_FRACTION;  else     ubfactor = 1.05;  MlevelNodeBisectionMultiple(ctrl, graph, tpwgts2, ubfactor);  IFSET(ctrl->dbglvl, DBG_SEPINFO, printf("Nvtxs: %6d, [%6d %6d %6d]\n", graph->nvtxs, graph->pwgts[0], graph->pwgts[1], graph->pwgts[2]));  if (cpos < npes-1) {    sizes[2*npes-2-cpos] = graph->pwgts[2];    sizes[2*npes-2-(2*cpos+1)] = graph->pwgts[1];    sizes[2*npes-2-(2*cpos+2)] = graph->pwgts[0];  }  /* Order the nodes in the separator */  nbnd = graph->nbnd;  bndind = graph->bndind;  label = graph->label;  for (i=0; i<nbnd; i++)     order[label[bndind[i]]] = --lastvtx;  SplitGraphOrder(ctrl, graph, &lgraph, &rgraph);  /* Free the memory of the top level graph */  GKfree(&graph->gdata, &graph->rdata, &graph->label, LTERM);  if (rgraph.nvtxs > MMDSWITCH || 2*cpos+1 < npes-1)     MlevelNestedDissectionP(ctrl, &rgraph, order, lastvtx, npes, 2*cpos+1, sizes);  else {    MMDOrder(ctrl, &rgraph, order, lastvtx);     GKfree(&rgraph.gdata, &rgraph.rdata, &rgraph.label, LTERM);  }  if (lgraph.nvtxs > MMDSWITCH || 2*cpos+2 < npes-1)     MlevelNestedDissectionP(ctrl, &lgraph, order, lastvtx-rgraph.nvtxs, npes, 2*cpos+2, sizes);  else {    MMDOrder(ctrl, &lgraph, order, lastvtx-rgraph.nvtxs);     GKfree(&lgraph.gdata, &lgraph.rdata, &lgraph.label, LTERM);  }}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -