circular.c   [plain text]


/*
    This software may only be used by you under license from AT&T Corp.
    ("AT&T").  A copy of AT&T's Source Code Agreement is available at
    AT&T's Internet website having the URL:
    <http://www.research.att.com/sw/tools/graphviz/license/source.html>
    If you received this software without first entering into a license
    with AT&T, you have an infringing copy of this software and cannot use
    it without violating AT&T's intellectual property rights.
*/
#pragma prototyped

#include	<string.h>
#include    "circular.h"
#include    "blocktree.h"
#include    "circpos.h"

#define		MINDIST			1.0

/* initGraphAttrs:
 * Set attributes based on original root graph.
 * This is obtained by taking a node of g, finding its node
 * in the original graph, and finding that node's graph.
 */
static void
initGraphAttrs(Agraph_t* g, circ_state* state)
{
	static Agraph_t*  rootg;
	static attrsym_t* N_artpos;
	static attrsym_t* N_root;
	static double	  min_dist;
	static char*	  rootname;
	Agraph_t*         rg;
	attrsym_t*        G_mindist;
	node_t*           n = agfstnode(g);

	rg = ORIGN(n)->graph;
	if (rg != rootg) {   /* new root graph */
		state->blockCount = 0;
		rootg = rg;
		G_mindist = agfindattr(rootg, "mindist");
		min_dist = late_double(rootg, G_mindist, MINDIST, 0.0);
		N_artpos = agfindattr(rootg->proto->n, "articulation_pos");
		N_root= agfindattr(rootg->proto->n, "root");
		rootname = agget(rootg, "root");
	}
	initBlocklist (&state->bl);
	state->orderCount = 1;
	state->bcstack = mkStack();
	state->min_dist = min_dist;
	state->N_artpos = N_artpos;
	state->N_root = N_root;
	state->rootname = rootname;
}

/* cleanup:
 * We need to cleanup objects created in initGraphAttrs
 * and all blocks. All graph objects are components of the
 * initial derived graph and will be freed when it is closed. 
 */
static void
cleanup(block_t* root, circ_state* sp)
{
	freeBlocktree(root);
	freeStack (sp->bcstack);
}

/* circularLayout:
 * Do circular layout of g.
 * Assume g is strict.
 * g is a "connected" component of the derived graph of the
 * original graph.
 */
void
circularLayout(Agraph_t* g)
{
	block_t*   	    root;
	circ_state      state;

	if (agnnodes(g) == 1) {
		Agnode_t* n = agfstnode(g);
		ND_pos(n)[0] = 0;
		ND_pos(n)[1] = 0;
		return;
	}

	initGraphAttrs(g, &state);

	root = createBlocktree(g, &state);
	circPos(g, root, &state);
	
	cleanup (root, &state);
}

#ifdef DEBUG
void prGraph (Agraph_t* g)
{
    Agnode_t* n;
    Agedge_t* e;

	fprintf (stderr, "%s\n", g->name);
    for (n = agfstnode(g); n; n = agnxtnode(g,n)) {
		fprintf (stderr, "%s (%x)\n", n->name, (unsigned int)n);
    	for (e = agfstout(g,n); e; e = agnxtout(g,e)) {
			fprintf (stderr, "%s -- %s (%x)\n", n->name, e->head->name, (unsigned int)e);
    	}
    }
}

cdata*
cvt(Agnode_t* n)
{
	return DATA(n);
}

void
prData (Agnode_t* n, int pass)
{
	char* pname;
	char* bname;
	char* tname;
	char* name1;
	char* name2;
	int   dist1, dist2;

    if (PARENT(n)) pname = PARENT(n)->name;
	else pname = "<P0>";
    if (BLOCK(n)) bname = BLOCK(n)->sub_graph->name;
	else pname = "<B0>";
	fprintf (stderr, "%s: %x %s %s ", n->name, FLAGS(n), pname, bname);
	switch (pass) {
	case 0:
		fprintf (stderr, "%d %d\n", VAL(n), LOWVAL(n));
		break;
	case 1:
    	if (TPARENT(n)) tname = TPARENT(n)->name;
		else tname = "<ROOT>";
		dist1 = DISTONE(n);
		if (dist1 > 0) name1 = LEAFONE(n)->name;
		else name1 = "<null>";
		dist2 = DISTTWO(n);
		if (dist2 > 0) name2 = LEAFTWO(n)->name;
		else name2 = "<null>";
		fprintf (stderr, "%s %s %d %s %d\n", tname, name1, dist1, name2, dist2);
		break;
	default :
		fprintf (stderr, "%d\n", POSITION(n));
		break;
	}
}
#endif