#define PETSC_DLL /* Contains the data structure for plotting several line graphs in a window with an axis. This is intended for line graphs that change dynamically by adding more points onto the end of the X axis. */ #include "petsc.h" /*I "petsc.h" I*/ PetscCookie DRAWLG_COOKIE = 0; struct _p_DrawLG { PETSCHEADER(int); PetscErrorCode (*destroy)(PetscDrawLG); PetscErrorCode (*view)(PetscDrawLG,PetscViewer); int len,loc; PetscDraw win; PetscDrawAxis axis; PetscReal xmin,xmax,ymin,ymax,*x,*y; int nopts,dim; PetscTruth use_dots; }; #define CHUNCKSIZE 100 #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGCreate" /*@ PetscDrawLGCreate - Creates a line graph data structure. Collective over PetscDraw Input Parameters: + draw - the window where the graph will be made. - dim - the number of line cures which will be drawn Output Parameters: . outctx - the line graph context Level: intermediate Concepts: line graph^creating .seealso: PetscDrawLGDestroy() @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGCreate(PetscDraw draw,int dim,PetscDrawLG *outctx) { PetscErrorCode ierr; PetscTruth isnull; PetscObject obj = (PetscObject)draw; PetscDrawLG lg; PetscFunctionBegin; PetscValidHeaderSpecific(draw,PETSC_DRAW_COOKIE,1); PetscValidPointer(outctx,2); ierr = PetscTypeCompare(obj,PETSC_DRAW_NULL,&isnull);CHKERRQ(ierr); if (isnull) { ierr = PetscDrawOpenNull(obj->comm,(PetscDraw*)outctx);CHKERRQ(ierr); PetscFunctionReturn(0); } ierr = PetscHeaderCreate(lg,_p_DrawLG,int,DRAWLG_COOKIE,0,"PetscDrawLG",obj->comm,PetscDrawLGDestroy,0);CHKERRQ(ierr); lg->view = 0; lg->destroy = 0; lg->nopts = 0; lg->win = draw; lg->dim = dim; lg->xmin = 1.e20; lg->ymin = 1.e20; lg->xmax = -1.e20; lg->ymax = -1.e20; ierr = PetscMalloc(2*dim*CHUNCKSIZE*sizeof(PetscReal),&lg->x);CHKERRQ(ierr); ierr = PetscLogObjectMemory(lg,2*dim*CHUNCKSIZE*sizeof(PetscReal));CHKERRQ(ierr); lg->y = lg->x + dim*CHUNCKSIZE; lg->len = dim*CHUNCKSIZE; lg->loc = 0; lg->use_dots= PETSC_FALSE; ierr = PetscDrawAxisCreate(draw,&lg->axis);CHKERRQ(ierr); ierr = PetscLogObjectParent(lg,lg->axis);CHKERRQ(ierr); *outctx = lg; PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGSetDimension" /*@ PetscDrawLGSetDimension - Change the number of lines that are to be drawn. Collective over PetscDrawLG Input Parameter: + lg - the line graph context. - dim - the number of curves. Level: intermediate Concepts: line graph^setting number of lines @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGSetDimension(PetscDrawLG lg,int dim) { PetscErrorCode ierr; PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) PetscFunctionReturn(0); PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); if (lg->dim == dim) PetscFunctionReturn(0); ierr = PetscFree(lg->x);CHKERRQ(ierr); lg->dim = dim; ierr = PetscMalloc(2*dim*CHUNCKSIZE*sizeof(PetscReal),&lg->x);CHKERRQ(ierr); ierr = PetscLogObjectMemory(lg,2*dim*CHUNCKSIZE*sizeof(PetscReal));CHKERRQ(ierr); lg->y = lg->x + dim*CHUNCKSIZE; lg->len = dim*CHUNCKSIZE; PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGReset" /*@ PetscDrawLGReset - Clears line graph to allow for reuse with new data. Collective over PetscDrawLG Input Parameter: . lg - the line graph context. Level: intermediate Concepts: line graph^restarting @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGReset(PetscDrawLG lg) { PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) PetscFunctionReturn(0); PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); lg->xmin = 1.e20; lg->ymin = 1.e20; lg->xmax = -1.e20; lg->ymax = -1.e20; lg->loc = 0; lg->nopts = 0; PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGDestroy" /*@ PetscDrawLGDestroy - Frees all space taken up by line graph data structure. Collective over PetscDrawLG Input Parameter: . lg - the line graph context Level: intermediate .seealso: PetscDrawLGCreate() @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGDestroy(PetscDrawLG lg) { PetscErrorCode ierr; PetscFunctionBegin; if (!lg || lg->cookie != PETSC_DRAW_COOKIE) { PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); } if (--lg->refct > 0) PetscFunctionReturn(0); if (lg && lg->cookie == PETSC_DRAW_COOKIE) { ierr = PetscObjectDestroy((PetscObject)lg);CHKERRQ(ierr); PetscFunctionReturn(0); } ierr = PetscDrawAxisDestroy(lg->axis);CHKERRQ(ierr); ierr = PetscFree(lg->x);CHKERRQ(ierr); ierr = PetscHeaderDestroy(lg);CHKERRQ(ierr); PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGAddPoint" /*@ PetscDrawLGAddPoint - Adds another point to each of the line graphs. The new point must have an X coordinate larger than the old points. Not Collective, but ignored by all processors except processor 0 in PetscDrawLG Input Parameters: + lg - the LineGraph data structure - x, y - the points to two vectors containing the new x and y point for each curve. Level: intermediate Concepts: line graph^adding points .seealso: PetscDrawLGAddPoints() @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGAddPoint(PetscDrawLG lg,PetscReal *x,PetscReal *y) { PetscErrorCode ierr; int i; PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) PetscFunctionReturn(0); PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); if (lg->loc+lg->dim >= lg->len) { /* allocate more space */ PetscReal *tmpx,*tmpy; ierr = PetscMalloc((2*lg->len+2*lg->dim*CHUNCKSIZE)*sizeof(PetscReal),&tmpx);CHKERRQ(ierr); ierr = PetscLogObjectMemory(lg,2*lg->dim*CHUNCKSIZE*sizeof(PetscReal));CHKERRQ(ierr); tmpy = tmpx + lg->len + lg->dim*CHUNCKSIZE; ierr = PetscMemcpy(tmpx,lg->x,lg->len*sizeof(PetscReal));CHKERRQ(ierr); ierr = PetscMemcpy(tmpy,lg->y,lg->len*sizeof(PetscReal));CHKERRQ(ierr); ierr = PetscFree(lg->x);CHKERRQ(ierr); lg->x = tmpx; lg->y = tmpy; lg->len += lg->dim*CHUNCKSIZE; } for (i=0; idim; i++) { if (x[i] > lg->xmax) lg->xmax = x[i]; if (x[i] < lg->xmin) lg->xmin = x[i]; if (y[i] > lg->ymax) lg->ymax = y[i]; if (y[i] < lg->ymin) lg->ymin = y[i]; lg->x[lg->loc] = x[i]; lg->y[lg->loc++] = y[i]; } lg->nopts++; PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGIndicateDataPoints" /*@ PetscDrawLGIndicateDataPoints - Causes LG to draw a big dot for each data-point. Not Collective, but ignored by all processors except processor 0 in PetscDrawLG Input Parameters: . lg - the linegraph context Level: intermediate Concepts: line graph^showing points @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGIndicateDataPoints(PetscDrawLG lg) { PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) PetscFunctionReturn(0); lg->use_dots = PETSC_TRUE; PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGAddPoints" /*@C PetscDrawLGAddPoints - Adds several points to each of the line graphs. The new points must have an X coordinate larger than the old points. Not Collective, but ignored by all processors except processor 0 in PetscDrawLG Input Parameters: + lg - the LineGraph data structure . xx,yy - points to two arrays of pointers that point to arrays containing the new x and y points for each curve. - n - number of points being added Level: intermediate Concepts: line graph^adding points .seealso: PetscDrawLGAddPoint() @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGAddPoints(PetscDrawLG lg,int n,PetscReal **xx,PetscReal **yy) { PetscErrorCode ierr; int i,j,k; PetscReal *x,*y; PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) PetscFunctionReturn(0); PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); if (lg->loc+n*lg->dim >= lg->len) { /* allocate more space */ PetscReal *tmpx,*tmpy; int chunk = CHUNCKSIZE; if (n > chunk) chunk = n; ierr = PetscMalloc((2*lg->len+2*lg->dim*chunk)*sizeof(PetscReal),&tmpx);CHKERRQ(ierr); ierr = PetscLogObjectMemory(lg,2*lg->dim*chunk*sizeof(PetscReal));CHKERRQ(ierr); tmpy = tmpx + lg->len + lg->dim*chunk; ierr = PetscMemcpy(tmpx,lg->x,lg->len*sizeof(PetscReal));CHKERRQ(ierr); ierr = PetscMemcpy(tmpy,lg->y,lg->len*sizeof(PetscReal));CHKERRQ(ierr); ierr = PetscFree(lg->x);CHKERRQ(ierr); lg->x = tmpx; lg->y = tmpy; lg->len += lg->dim*chunk; } for (j=0; jdim; j++) { x = xx[j]; y = yy[j]; k = lg->loc + j; for (i=0; i lg->xmax) lg->xmax = x[i]; if (x[i] < lg->xmin) lg->xmin = x[i]; if (y[i] > lg->ymax) lg->ymax = y[i]; if (y[i] < lg->ymin) lg->ymin = y[i]; lg->x[k] = x[i]; lg->y[k] = y[i]; k += lg->dim; } } lg->loc += n*lg->dim; lg->nopts += n; PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGDraw" /*@ PetscDrawLGDraw - Redraws a line graph. Not Collective,but ignored by all processors except processor 0 in PetscDrawLG Input Parameter: . lg - the line graph context Level: intermediate .seealso: PetscDrawSPDraw(), PetscDrawLGSPDraw() @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGDraw(PetscDrawLG lg) { PetscReal xmin=lg->xmin,xmax=lg->xmax,ymin=lg->ymin,ymax=lg->ymax; PetscErrorCode ierr; int i,j,dim = lg->dim,nopts = lg->nopts,rank; PetscDraw draw = lg->win; PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) PetscFunctionReturn(0); PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); ierr = PetscDrawClear(draw);CHKERRQ(ierr); ierr = PetscDrawAxisSetLimits(lg->axis,xmin,xmax,ymin,ymax);CHKERRQ(ierr); ierr = PetscDrawAxisDraw(lg->axis);CHKERRQ(ierr); ierr = MPI_Comm_rank(lg->comm,&rank);CHKERRQ(ierr); if (!rank) { for (i=0; ix[(j-1)*dim+i],lg->y[(j-1)*dim+i],lg->x[j*dim+i],lg->y[j*dim+i],PETSC_DRAW_BLACK+i);CHKERRQ(ierr); if (lg->use_dots) { ierr = PetscDrawString(draw,lg->x[j*dim+i],lg->y[j*dim+i],PETSC_DRAW_RED,"x");CHKERRQ(ierr); } } } } ierr = PetscDrawFlush(lg->win);CHKERRQ(ierr); ierr = PetscDrawPause(lg->win);CHKERRQ(ierr); PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGPrint" /*@ PetscDrawLGPrint - Prints a line graph. Not collective Input Parameter: . lg - the line graph context Level: beginner Contributed by Matthew Knepley .keywords: draw, line, graph @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGPrint(PetscDrawLG lg) { PetscReal xmin=lg->xmin, xmax=lg->xmax, ymin=lg->ymin, ymax=lg->ymax; int i, j, dim = lg->dim, nopts = lg->nopts; PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) PetscFunctionReturn(0); PetscValidHeaderSpecific(lg, DRAWLG_COOKIE,1); if (nopts < 1) PetscFunctionReturn(0); if (xmin > xmax || ymin > ymax) PetscFunctionReturn(0); for(i = 0; i < dim; i++) { PetscPrintf(lg->comm, "Line %d>\n", i); for(j = 0; j < nopts; j++) { PetscPrintf(lg->comm, " X: %G Y: %G\n", lg->x[j*dim+i], lg->y[j*dim+i]); } } PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGSetLimits" /*@ PetscDrawLGSetLimits - Sets the axis limits for a line graph. If more points are added after this call, the limits will be adjusted to include those additional points. Not Collective, but ignored by all processors except processor 0 in PetscDrawLG Input Parameters: + xlg - the line graph context - x_min,x_max,y_min,y_max - the limits Level: intermediate Concepts: line graph^setting axis @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGSetLimits(PetscDrawLG lg,PetscReal x_min,PetscReal x_max,PetscReal y_min,PetscReal y_max) { PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) PetscFunctionReturn(0); PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); (lg)->xmin = x_min; (lg)->xmax = x_max; (lg)->ymin = y_min; (lg)->ymax = y_max; PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGGetAxis" /*@ PetscDrawLGGetAxis - Gets the axis context associated with a line graph. This is useful if one wants to change some axis property, such as labels, color, etc. The axis context should not be destroyed by the application code. Not Collective, if PetscDrawLG is parallel then PetscDrawAxis is parallel Input Parameter: . lg - the line graph context Output Parameter: . axis - the axis context Level: advanced @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGGetAxis(PetscDrawLG lg,PetscDrawAxis *axis) { PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) { *axis = 0; PetscFunctionReturn(0); } PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); PetscValidPointer(axis,2); *axis = lg->axis; PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGGetDraw" /*@ PetscDrawLGGetDraw - Gets the draw context associated with a line graph. Not Collective, if PetscDrawLG is parallel then PetscDraw is parallel Input Parameter: . lg - the line graph context Output Parameter: . draw - the draw context Level: intermediate @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGGetDraw(PetscDrawLG lg,PetscDraw *draw) { PetscFunctionBegin; PetscValidHeader(lg,1); PetscValidPointer(draw,2); if (lg->cookie == PETSC_DRAW_COOKIE) { *draw = (PetscDraw)lg; } else { PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); *draw = lg->win; } PetscFunctionReturn(0); } #undef __FUNCT__ #define __FUNCT__ "PetscDrawLGSPDraw" /*@ PetscDrawLGSPDraw - Redraws a line graph. Not Collective,but ignored by all processors except processor 0 in PetscDrawLG Input Parameter: . lg - the line graph context Level: intermediate .seealso: PetscDrawLGDraw(), PetscDrawSPDraw() Developer Notes: This code cheats and uses the fact that the LG and SP structs are the same @*/ PetscErrorCode PETSC_DLLEXPORT PetscDrawLGSPDraw(PetscDrawLG lg,PetscDrawSP spin) { PetscDrawLG sp = (PetscDrawLG)spin; PetscReal xmin,xmax,ymin,ymax; PetscErrorCode ierr; int i,j,dim,nopts,rank; PetscDraw draw = lg->win; PetscFunctionBegin; if (lg && lg->cookie == PETSC_DRAW_COOKIE) PetscFunctionReturn(0); PetscValidHeaderSpecific(lg,DRAWLG_COOKIE,1); PetscValidHeaderSpecific(sp,DRAWSP_COOKIE,2); xmin = PetscMin(lg->xmin,sp->xmin); ymin = PetscMin(lg->ymin,sp->ymin); xmax = PetscMax(lg->xmax,sp->xmax); ymax = PetscMax(lg->ymax,sp->ymax); ierr = PetscDrawClear(draw);CHKERRQ(ierr); ierr = PetscDrawAxisSetLimits(lg->axis,xmin,xmax,ymin,ymax);CHKERRQ(ierr); ierr = PetscDrawAxisDraw(lg->axis);CHKERRQ(ierr); ierr = MPI_Comm_rank(lg->comm,&rank);CHKERRQ(ierr); if (!rank) { dim = lg->dim; nopts = lg->nopts; for (i=0; ix[(j-1)*dim+i],lg->y[(j-1)*dim+i],lg->x[j*dim+i],lg->y[j*dim+i],PETSC_DRAW_BLACK+i);CHKERRQ(ierr); if (lg->use_dots) { ierr = PetscDrawString(draw,lg->x[j*dim+i],lg->y[j*dim+i],PETSC_DRAW_RED,"x");CHKERRQ(ierr); } } } dim = sp->dim; nopts = sp->nopts; for (i=0; ix[j*dim+i],sp->y[j*dim+i],PETSC_DRAW_RED,"x");CHKERRQ(ierr); } } } ierr = PetscDrawFlush(lg->win);CHKERRQ(ierr); ierr = PetscDrawPause(lg->win);CHKERRQ(ierr); PetscFunctionReturn(0); }