/* * Copyright 1994-2011, Regents of the University of Minnesota * * gpmetis.c * * Drivers for the partitioning routines * * Started 8/28/94 * George * * $Id: gpmetis.c 14362 2013-05-21 21:35:23Z karypis $ * */ #include "metisbin.h" /*************************************************************************/ /*! Let the game begin! */ /*************************************************************************/ int main(int argc, char *argv[]) { idx_t i; char *curptr, *newptr; idx_t options[METIS_NOPTIONS]; graph_t *graph; idx_t *part; idx_t objval; params_t *params; int status=0; params = parse_cmdline(argc, argv); gk_startcputimer(params->iotimer); graph = ReadGraph(params); ReadTPwgts(params, graph->ncon); gk_stopcputimer(params->iotimer); /* Check if the graph is contiguous */ if (params->contig && !IsConnected(graph, 0)) { printf("***The input graph is not contiguous.\n" "***The specified -contig option will be ignored.\n"); params->contig = 0; } /* Get ubvec if supplied */ if (params->ubvecstr) { params->ubvec = rmalloc(graph->ncon, "main"); curptr = params->ubvecstr; for (i=0; incon; i++) { params->ubvec[i] = strtoreal(curptr, &newptr); if (curptr == newptr) errexit("Error parsing entry #%"PRIDX" of ubvec [%s] (possibly missing).\n", i, params->ubvecstr); curptr = newptr; } } /* Setup iptype */ if (params->iptype == -1) { if (params->ptype == METIS_PTYPE_RB) { if (graph->ncon == 1) params->iptype = METIS_IPTYPE_GROW; else params->iptype = METIS_IPTYPE_RANDOM; } } GPPrintInfo(params, graph); part = imalloc(graph->nvtxs, "main: part"); METIS_SetDefaultOptions(options); options[METIS_OPTION_OBJTYPE] = params->objtype; options[METIS_OPTION_CTYPE] = params->ctype; options[METIS_OPTION_IPTYPE] = params->iptype; options[METIS_OPTION_RTYPE] = params->rtype; options[METIS_OPTION_NO2HOP] = params->no2hop; options[METIS_OPTION_ONDISK] = params->ondisk; options[METIS_OPTION_DROPEDGES] = params->dropedges; options[METIS_OPTION_MINCONN] = params->minconn; options[METIS_OPTION_CONTIG] = params->contig; options[METIS_OPTION_SEED] = params->seed; options[METIS_OPTION_NIPARTS] = params->niparts; options[METIS_OPTION_NITER] = params->niter; options[METIS_OPTION_NCUTS] = params->ncuts; options[METIS_OPTION_UFACTOR] = params->ufactor; options[METIS_OPTION_DBGLVL] = params->dbglvl; gk_malloc_init(); gk_startcputimer(params->parttimer); switch (params->ptype) { case METIS_PTYPE_RB: status = METIS_PartGraphRecursive(&graph->nvtxs, &graph->ncon, graph->xadj, graph->adjncy, graph->vwgt, graph->vsize, graph->adjwgt, ¶ms->nparts, params->tpwgts, params->ubvec, options, &objval, part); break; case METIS_PTYPE_KWAY: status = METIS_PartGraphKway(&graph->nvtxs, &graph->ncon, graph->xadj, graph->adjncy, graph->vwgt, graph->vsize, graph->adjwgt, ¶ms->nparts, params->tpwgts, params->ubvec, options, &objval, part); break; } gk_stopcputimer(params->parttimer); if (gk_GetCurMemoryUsed() != 0) printf("***It seems that Metis did not free all of its memory! Report this.\n"); params->maxmemory = gk_GetMaxMemoryUsed(); gk_malloc_cleanup(0); if (graph->adjwgt == NULL) graph->adjwgt = ismalloc(graph->nedges, 1, "adjwgt"); if (status != METIS_OK) { printf("\n***Metis returned with an error.\n"); } else { if (!params->nooutput) { /* Write the solution */ gk_startcputimer(params->iotimer); WritePartition(params->filename, part, graph->nvtxs, params->nparts); gk_stopcputimer(params->iotimer); } GPReportResults(params, graph, part, objval); } #ifdef XXX { idx_t *old2new = imalloc(graph->nvtxs, "old2new"); METIS_CacheFriendlyReordering(graph->nvtxs, graph->xadj, graph->adjncy, part, old2new); WritePartition("ciperm", old2new, graph->nvtxs, params->nparts); gk_free((void **)&old2new, LTERM); } #endif FreeGraph(&graph); gk_free((void **)&part, LTERM); gk_free((void **)¶ms->filename, ¶ms->tpwgtsfile, ¶ms->tpwgts, ¶ms->ubvecstr, ¶ms->ubvec, ¶ms, LTERM); } /*************************************************************************/ /*! This function prints run parameters */ /*************************************************************************/ void GPPrintInfo(params_t *params, graph_t *graph) { idx_t i; if (params->ufactor == -1) { if (params->ptype == METIS_PTYPE_KWAY) params->ufactor = KMETIS_DEFAULT_UFACTOR; else if (graph->ncon == 1) params->ufactor = PMETIS_DEFAULT_UFACTOR; else params->ufactor = MCPMETIS_DEFAULT_UFACTOR; } printf("******************************************************************************\n"); printf("%s", METISTITLE); printf(" (HEAD: %s, Built on: %s, %s)\n", SVNINFO, __DATE__, __TIME__); printf(" size of idx_t: %zubits, real_t: %zubits, idx_t *: %zubits\n", 8*sizeof(idx_t), 8*sizeof(real_t), 8*sizeof(idx_t *)); printf("\n"); printf("Graph Information -----------------------------------------------------------\n"); printf(" Name: %s, #Vertices: %"PRIDX", #Edges: %"PRIDX", #Parts: %"PRIDX"\n", params->filename, graph->nvtxs, graph->nedges/2, params->nparts); if (graph->ncon > 1) printf(" Balancing constraints: %"PRIDX"\n", graph->ncon); printf("\n"); printf("Options ---------------------------------------------------------------------\n"); printf(" ptype=%s, objtype=%s, ctype=%s, rtype=%s, iptype=%s\n", ptypenames[params->ptype], objtypenames[params->objtype], ctypenames[params->ctype], rtypenames[params->rtype], iptypenames[params->iptype]); printf(" dbglvl=%"PRIDX", ufactor=%.3f, no2hop=%s, minconn=%s, contig=%s\n", params->dbglvl, I2RUBFACTOR(params->ufactor), (params->no2hop ? "YES" : "NO"), (params->minconn ? "YES" : "NO"), (params->contig ? "YES" : "NO") ); printf(" ondisk=%s, nooutput=%s\n", (params->ondisk ? "YES" : "NO"), (params->nooutput ? "YES" : "NO") ); printf(" seed=%"PRIDX", niparts=%"PRIDX", niter=%"PRIDX", ncuts=%"PRIDX"\n", params->seed, params->niparts, params->niter, params->ncuts); if (params->ubvec) { printf(" ubvec=("); for (i=0; incon; i++) printf("%s%.2e", (i==0?"":" "), (double)params->ubvec[i]); printf(")\n"); } printf("\n"); switch (params->ptype) { case METIS_PTYPE_RB: printf("Recursive Partitioning ------------------------------------------------------\n"); break; case METIS_PTYPE_KWAY: printf("Direct k-way Partitioning ---------------------------------------------------\n"); break; } } /*************************************************************************/ /*! This function does any post-partitioning reporting */ /*************************************************************************/ void GPReportResults(params_t *params, graph_t *graph, idx_t *part, idx_t objval) { gk_startcputimer(params->reporttimer); ComputePartitionInfo(params, graph, part); gk_stopcputimer(params->reporttimer); printf("\nTiming Information ----------------------------------------------------------\n"); printf(" I/O: \t\t %7.3"PRREAL" sec\n", gk_getcputimer(params->iotimer)); printf(" Partitioning: \t\t %7.3"PRREAL" sec (METIS time)\n", gk_getcputimer(params->parttimer)); printf(" Reporting: \t\t %7.3"PRREAL" sec\n", gk_getcputimer(params->reporttimer)); printf("\nMemory Information ----------------------------------------------------------\n"); printf(" Max memory used:\t\t %7.3"PRREAL" MB\n", (real_t)(params->maxmemory/(1024.0*1024.0))); #ifndef MACOS { struct rusage usage; getrusage(RUSAGE_SELF, &usage); printf(" rusage.ru_maxrss:\t\t %7.3"PRREAL" MB\n", (real_t)(usage.ru_maxrss/(1024.0))); } printf(" proc/self/stat/VmPeak:\t %7.3"PRREAL" MB\n", (real_t)gk_GetProcVmPeak()/(1024.0*1024.0)); #endif printf("******************************************************************************\n"); }